diff --git a/.gitignore b/.gitignore index 8a9487398e..d41b175b62 100644 --- a/.gitignore +++ b/.gitignore @@ -30,3 +30,4 @@ vignettes/*.md inst/doc tests/testthat/_snaps/**/*.new.md tests/testthat/_snaps/**/*.new.svg +teal_app.lock diff --git a/DESCRIPTION b/DESCRIPTION index a5f0e6692d..d86b6728d6 100644 --- a/DESCRIPTION +++ b/DESCRIPTION @@ -45,7 +45,6 @@ Imports: jsonlite, lifecycle (>= 0.2.0), logger (>= 0.2.0), - magrittr (>= 1.5), methods, promises (>= 1.3.0), renv (>= 1.0.7), @@ -98,13 +97,16 @@ Collate: 'init.R' 'landing_popup_module.R' 'module_bookmark_manager.R' + 'module_data_summary.R' + 'module_filter_data.R' 'module_filter_manager.R' + 'module_init_data.R' 'module_nested_tabs.R' 'module_snapshot_manager.R' - 'module_tabs_with_filters.R' 'module_teal.R' + 'module_teal_data.R' 'module_teal_with_splash.R' - 'module_wunder_bar.R' + 'module_transform_data.R' 'reporter_previewer_module.R' 'show_rcode_modal.R' 'tdata.R' @@ -112,6 +114,7 @@ Collate: 'teal_data_module.R' 'teal_data_module-eval_code.R' 'teal_data_module-within.R' + 'teal_data_utils.R' 'teal_lockfile.R' 'teal_reporter.R' 'teal_slices-store.R' diff --git a/NAMESPACE b/NAMESPACE index fcabede46f..edaaa6a136 100644 --- a/NAMESPACE +++ b/NAMESPACE @@ -3,19 +3,17 @@ S3method(c,teal_slices) S3method(format,teal_module) S3method(format,teal_modules) -S3method(get_metadata,default) -S3method(get_metadata,tdata) S3method(join_keys,tdata) S3method(print,teal_module) S3method(print,teal_modules) -S3method(srv_nested_tabs,default) -S3method(srv_nested_tabs,teal_module) -S3method(srv_nested_tabs,teal_modules) -S3method(ui_nested_tabs,default) -S3method(ui_nested_tabs,teal_module) -S3method(ui_nested_tabs,teal_modules) +S3method(srv_teal_module,default) +S3method(srv_teal_module,teal_module) +S3method(srv_teal_module,teal_modules) +S3method(ui_teal_module,default) +S3method(ui_teal_module,shiny.tag) +S3method(ui_teal_module,teal_module) +S3method(ui_teal_module,teal_modules) S3method(within,teal_data_module) -export("%>%") export(TealReportCard) export(as.teal_slices) export(as_tdata) @@ -31,10 +29,13 @@ export(new_tdata) export(report_card_template) export(reporter_previewer_module) export(show_rcode_modal) +export(srv_teal) export(srv_teal_with_splash) export(tdata2env) export(teal_data_module) export(teal_slices) +export(teal_transform_module) +export(ui_teal) export(ui_teal_with_splash) export(validate_has_data) export(validate_has_elements) @@ -47,7 +48,9 @@ export(validate_one_row_per_id) import(shiny) import(teal.data) import(teal.slice) -importFrom(magrittr,"%>%") +importFrom(methods,new) importFrom(methods,setMethod) +importFrom(shiny,reactiveVal) +importFrom(shiny,reactiveValues) importFrom(stats,setNames) importMethodsFrom(teal.code,eval_code) diff --git a/NEWS.md b/NEWS.md index dea3272f2e..ef9d329c8b 100644 --- a/NEWS.md +++ b/NEWS.md @@ -1,14 +1,24 @@ # teal 0.15.2.9050 -### Enhancement -* Provided progress bar for modules loading and data filtering during teal app startup. - ### New features -* Possibility to download lockfile to restore app session for reproducibility. +* Possible to call `ui_teal` and `srv_teal` directly in any application by delivering `data` argument as a `reactive` returning `teal_data` object. #669 +* Introduced `teal_transform_module` to provide a way to interactively modify data delivered to `teal_module`'s `server`. #1228 +* Introduced a new argument `once = FALSE` in `teal_data_module` to possibly reload data during a run time. +* Possibility to download lockfile to restore app session for reproducibility. #479 -### Miscellaneous -* Filter mapping display is no longer coupled to the snapshot manager. + +### Breaking changes + +* The `landing_popup_module()` needs to be passed as the `landing_popup` argument of `init` instead of being passed as a module of the `modules` argument of `init`. +* `teal` no longer re-export `%>%`. Please load `library(magrittr)` instead or use `|>` from `base`. + +### Enhancement + +* Enhanced a system of data validation and a display of error messages. +* Easier way of to call `javascript` events by setting `$(document).ready(function() { ... })`. #1114 +* Provided progress bar for modules loading and data filtering during teal app startup. +* Filter mapping display has a separate icon in the tab. # teal 0.15.2 diff --git a/R/TealAppDriver.R b/R/TealAppDriver.R index d320488000..70eb0da6e9 100644 --- a/R/TealAppDriver.R +++ b/R/TealAppDriver.R @@ -23,7 +23,7 @@ TealAppDriver <- R6::R6Class( # nolint: object_name. #' @description #' Initialize a `TealAppDriver` object for testing a `teal` application. #' - #' @param data,modules,filter,title,header,footer arguments passed to `init` + #' @param data,modules,filter,title,header,footer,landing_popup arguments passed to `init` #' @param timeout (`numeric`) Default number of milliseconds for any timeout or #' timeout_ parameter in the `TealAppDriver` class. #' Defaults to 20s. @@ -45,6 +45,7 @@ TealAppDriver <- R6::R6Class( # nolint: object_name. title = build_app_title(), header = tags$p(), footer = tags$p(), + landing_popup = NULL, timeout = rlang::missing_arg(), load_timeout = rlang::missing_arg(), ...) { @@ -57,7 +58,8 @@ TealAppDriver <- R6::R6Class( # nolint: object_name. filter = filter, title = title, header = header, - footer = footer + footer = footer, + landing_popup = landing_popup, ) # Default timeout is hardcoded to 4s in shinytest2:::resolve_timeout @@ -166,13 +168,11 @@ TealAppDriver <- R6::R6Class( # nolint: object_name. navigate_teal_tab = function(tabs) { checkmate::check_character(tabs, min.len = 1) for (tab in tabs) { - root <- "root" self$set_input( - sprintf("teal-main_ui-%s-active_tab", root), + "teal-teal_modules-active_tab", get_unique_labels(tab), wait_ = FALSE ) - root <- sprintf("%s-%s", private$modules$label, get_unique_labels(tab)) } self$wait_for_idle() private$set_active_ns() @@ -229,6 +229,26 @@ TealAppDriver <- R6::R6Class( # nolint: object_name. private$ns$filter_panel }, #' @description + #' Get the active shiny name space for interacting with the data-summary panel. + #' + #' @return (`string`) The active shiny name space of the data-summary component. + active_data_summary_ns = function() { + if (identical(private$ns$data_summary, character(0))) { + private$set_active_ns() + } + private$ns$data_summary + }, + #' @description + #' Get the active shiny name space bound with a custom `element` name. + #' + #' @param element `character(1)` custom element name. + #' + #' @return (`string`) The active shiny name space of the component bound with the input `element`. + active_data_summary_element = function(element) { + checkmate::assert_string(element) + sprintf("#%s-%s", self$active_data_summary_ns(), element) + }, + #' @description #' Get the input from the module in the `teal` app. #' This function will only access inputs from the name space of the current active teal module. #' @@ -262,9 +282,10 @@ TealAppDriver <- R6::R6Class( # nolint: object_name. get_active_module_table_output = function(table_id, which = 1) { checkmate::check_number(which, lower = 1) checkmate::check_string(table_id) - table <- self$active_module_element(table_id) %>% - self$get_html_rvest() %>% - rvest::html_table(fill = TRUE) + table <- rvest::html_table( + self$get_html_rvest(self$active_module_element(table_id)), + fill = TRUE + ) if (length(table) == 0) { data.frame() } else { @@ -310,12 +331,12 @@ TealAppDriver <- R6::R6Class( # nolint: object_name. #' Get the active datasets that can be accessed via the filter panel of the current active teal module. get_active_filter_vars = function() { displayed_datasets_index <- self$is_visible( - sprintf("#%s-active-filter_active_vars_contents > span", self$active_filters_ns()) + sprintf("#%s-filters-filter_active_vars_contents > span", self$active_filters_ns()) ) available_datasets <- self$get_text( sprintf( - "#%s-active-filter_active_vars_contents .filter_panel_dataname", + "#%s-filters-filter_active_vars_contents .filter_panel_dataname", self$active_filters_ns() ) ) @@ -323,6 +344,24 @@ TealAppDriver <- R6::R6Class( # nolint: object_name. available_datasets[displayed_datasets_index] }, #' @description + #' Get the active data summary table + #' @return `data.frame` + get_active_data_summary_table = function() { + summary_table <- rvest::html_table( + self$get_html_rvest(self$active_data_summary_element("table")), + fill = TRUE + )[[1]] + + col_names <- unlist(summary_table[1, ], use.names = FALSE) + summary_table <- summary_table[-1, ] + colnames(summary_table) <- col_names + if (nrow(summary_table) > 0) { + summary_table + } else { + NULL + } + }, + #' @description #' Test if `DOM` elements are visible on the page with a JavaScript call. #' @param selector (`character(1)`) `CSS` selector to check visibility. #' A `CSS` id will return only one element if the UI is well formed. @@ -371,14 +410,17 @@ TealAppDriver <- R6::R6Class( # nolint: object_name. active_filters <- lapply( datasets, function(x) { - var_names <- self$get_text( - sprintf( - "#%s-active-%s-filters .filter-card-varname", - self$active_filters_ns(), - x + var_names <- gsub( + pattern = "\\s", + replacement = "", + self$get_text( + sprintf( + "#%s-filters-%s .filter-card-varname", + self$active_filters_ns(), + x + ) ) - ) %>% - gsub(pattern = "\\s", replacement = "") + ) structure( lapply(var_names, private$get_active_filter_selection, dataset_name = x), names = var_names @@ -402,10 +444,19 @@ TealAppDriver <- R6::R6Class( # nolint: object_name. add_filter_var = function(dataset_name, var_name, ...) { checkmate::check_string(dataset_name) checkmate::check_string(var_name) + private$set_active_ns() + self$click( + selector = sprintf( + "#%s-filters-%s-add_filter_icon", + private$ns$filter_panel, + dataset_name + ) + ) self$set_input( sprintf( - "%s-add-%s-filter-var_to_add", - self$active_filters_ns(), + "%s-filters-%s-%s-filter-var_to_add", + private$ns$filter_panel, + dataset_name, dataset_name ), var_name, @@ -468,7 +519,7 @@ TealAppDriver <- R6::R6Class( # nolint: object_name. checkmate::check_string(input) input_id_prefix <- sprintf( - "%s-active-%s-filter-%s_%s-inputs", + "%s-filters-%s-filter-%s_%s-inputs", self$active_filters_ns(), dataset_name, dataset_name, @@ -492,7 +543,7 @@ TealAppDriver <- R6::R6Class( # nolint: object_name. # Generate correct namespace slices_input_id <- sprintf( - "%s-active-%s-filter-%s_%s-inputs-%s", + "%s-filters-%s-filter-%s_%s-inputs-%s", self$active_filters_ns(), dataset_name, dataset_name, @@ -542,9 +593,10 @@ TealAppDriver <- R6::R6Class( # nolint: object_name. #' #' @return The `character` vector. get_attr = function(selector, attribute) { - self$get_html_rvest("html") %>% - rvest::html_nodes(selector) %>% - rvest::html_attr(attribute) + rvest::html_attr( + rvest::html_nodes(self$get_html_rvest("html"), selector), + attribute + ) }, #' @description #' Wrapper around `get_html` that passes the output directly to `rvest::read_html`. @@ -602,14 +654,13 @@ TealAppDriver <- R6::R6Class( # nolint: object_name. all_inputs <- self$get_values()$input active_tab_inputs <- all_inputs[grepl("-active_tab$", names(all_inputs))] - tab_ns <- lapply(names(active_tab_inputs), function(name) { + tab_ns <- unlist(lapply(names(active_tab_inputs), function(name) { gsub( pattern = "-active_tab$", replacement = sprintf("-%s", active_tab_inputs[[name]]), name ) - }) %>% - unlist() + })) active_ns <- tab_ns[1] if (length(tab_ns) > 1) { for (i in 2:length(tab_ns)) { @@ -621,11 +672,16 @@ TealAppDriver <- R6::R6Class( # nolint: object_name. } private$ns$module <- sprintf("%s-%s", active_ns, "module") - component <- "filter_panel" - if (!is.null(self$get_html(sprintf("#teal-main_ui-%s", component)))) { - private$ns[[component]] <- sprintf("teal-main_ui-%s", component) - } else { - private$ns[[component]] <- sprintf("%s-module_%s", active_ns, component) + components <- c("filter_panel", "data_summary") + for (component in components) { + if ( + !is.null(self$get_html(sprintf("#%s-%s-panel", active_ns, component))) || + !is.null(self$get_html(sprintf("#%s-%s-table", active_ns, component))) + ) { + private$ns[[component]] <- sprintf("%s-%s", active_ns, component) + } else { + private$ns[[component]] <- sprintf("%s-module_%s", active_ns, component) + } } }, # @description @@ -639,7 +695,7 @@ TealAppDriver <- R6::R6Class( # nolint: object_name. checkmate::check_string(dataset_name) checkmate::check_string(var_name) input_id_prefix <- sprintf( - "%s-active-%s-filter-%s_%s-inputs", + "%s-filters-%s-filter-%s_%s-inputs", self$active_filters_ns(), dataset_name, dataset_name, diff --git a/R/dummy_functions.R b/R/dummy_functions.R index 4624b783d9..d4071c6fbc 100644 --- a/R/dummy_functions.R +++ b/R/dummy_functions.R @@ -13,22 +13,37 @@ #' shinyApp(app$ui, app$server) #' } #' @export -example_module <- function(label = "example teal module", datanames = "all") { +example_module <- function(label = "example teal module", datanames = "all", transformers = list()) { checkmate::assert_string(label) ans <- module( label, server = function(id, data) { - checkmate::assert_class(data(), "teal_data") + checkmate::assert_class(isolate(data()), "teal_data") moduleServer(id, function(input, output, session) { - updateSelectInput( - inputId = "dataname", - choices = isolate(teal.data::datanames(data())), - selected = restoreInput(session$ns("dataname"), NULL) - ) + datanames_rv <- reactive({ + teal.data::datanames(req(data())) + }) + + observeEvent(datanames_rv(), { + selected <- input$dataname + if (identical(selected, "")) { + selected <- restoreInput(session$ns("dataname"), NULL) + } else if (isFALSE(selected %in% datanames_rv())) { + selected <- datanames_rv()[1] + } + updateSelectInput( + session = session, + inputId = "dataname", + choices = datanames_rv(), + selected = selected + ) + }) + output$text <- renderPrint({ req(input$dataname) data()[[input$dataname]] }) + teal.widgets::verbatim_popup_srv( id = "rcode", verbatim_content = reactive(teal.code::get_code(data())), @@ -46,7 +61,8 @@ example_module <- function(label = "example teal module", datanames = "all") { ) ) }, - datanames = datanames + datanames = datanames, + transformers = transformers ) attr(ans, "teal_bookmarkable") <- TRUE ans diff --git a/R/get_rcode_utils.R b/R/get_rcode_utils.R index b112306629..42076a4cfe 100644 --- a/R/get_rcode_utils.R +++ b/R/get_rcode_utils.R @@ -5,19 +5,17 @@ #' @return Character vector of `library()` calls. #' @keywords internal get_rcode_libraries <- function() { - vapply( + libraries <- vapply( utils::sessionInfo()$otherPkgs, function(x) { paste0("library(", x$Package, ")") }, character(1) - ) %>% - # put it into reverse order to correctly simulate executed code - rev() %>% - paste0(sep = "\n") %>% - paste0(collapse = "") + ) + paste0(paste0(rev(libraries), sep = "\n"), collapse = "") } + #' @noRd #' @keywords internal get_rcode_str_install <- function() { @@ -28,47 +26,3 @@ get_rcode_str_install <- function() { "# Add any code to install/load your NEST environment here\n" } } - -#' Get datasets code -#' -#' Retrieve complete code to create, verify, and filter a dataset. -#' -#' @param datanames (`character`) names of datasets to extract code from -#' @param datasets (`FilteredData`) object -#' @param hashes named (`list`) of hashes per dataset -#' -#' @return Character string concatenated from the following elements: -#' - data pre-processing code (from `data` argument in `init`) -#' - hash check of loaded objects -#' - filter code (if any) -#' -#' @keywords internal -get_datasets_code <- function(datanames, datasets, hashes) { - # preprocessing code - str_prepro <- attr(datasets, "preprocessing_code") - if (length(str_prepro) == 0) { - str_prepro <- "message('Preprocessing is empty')" - } else { - str_prepro <- paste(str_prepro, collapse = "\n") - } - - # hash checks - str_hash <- vapply(datanames, function(dataname) { - sprintf( - "stopifnot(%s == %s)", - deparse1(bquote(rlang::hash(.(as.name(dataname))))), - deparse1(hashes[[dataname]]) - ) - }, character(1)) - str_hash <- paste(str_hash, collapse = "\n") - - # filter expressions - str_filter <- teal.slice::get_filter_expr(datasets, datanames) - if (str_filter == "") { - str_filter <- character(0) - } - - # concatenate all code - str_code <- paste(c(str_prepro, str_hash, str_filter), collapse = "\n\n") - sprintf("%s\n", str_code) -} diff --git a/R/include_css_js.R b/R/include_css_js.R index 04cb27dd55..f5a2aebfec 100644 --- a/R/include_css_js.R +++ b/R/include_css_js.R @@ -78,6 +78,6 @@ include_teal_css_js <- function() { include_css_files(), # init.js is executed from the server include_js_files(except = "init.js"), - shinyjs::hidden(icon("gear")), # add hidden icon to load font-awesome css for icons + shinyjs::hidden(icon("fas fa-gear")), # add hidden icon to load font-awesome css for icons ) } diff --git a/R/init.R b/R/init.R index 5aca99e100..092c263dd5 100644 --- a/R/init.R +++ b/R/init.R @@ -34,6 +34,8 @@ #' @param id (`character`) optional #' string specifying the `shiny` module id in cases it is used as a `shiny` module #' rather than a standalone `shiny` app. This is a legacy feature. +#' @param landing_popup (`teal_module`) optional +#' A `landing_popup_module` to show up as soon as the teal app is initialized. #' #' @return Named list containing server and UI functions. #' @@ -99,8 +101,9 @@ init <- function(data, title = build_app_title(), header = tags$p(), footer = tags$p(), - id = character(0)) { - logger::log_trace("init initializing teal app with: data ('{ class(data) }').") + id = character(0), + landing_popup = NULL) { + logger::log_debug("init initializing teal app with: data ('{ class(data) }').") # argument checking (independent) ## `data` @@ -115,6 +118,7 @@ init <- function(data, ) } checkmate::assert_multi_class(data, c("teal_data", "teal_data_module")) + checkmate::assert_class(landing_popup, "teal_module_landing", null.ok = TRUE) ## `modules` checkmate::assert( @@ -159,10 +163,18 @@ init <- function(data, # argument transformations ## `modules` - landing module landing <- extract_module(modules, "teal_module_landing") - landing_module <- NULL if (length(landing) == 1L) { - landing_module <- landing[[1L]] + landing_popup <- landing[[1L]] modules <- drop_module(modules, "teal_module_landing") + # TODO: verify the version before release. + lifecycle::deprecate_soft( + when = "0.16", + what = "landing_popup_module()", + details = paste( + "Pass `landing_popup_module` to the `landing_popup` argument of the `init` ", + "instead of wrapping it into `modules()` and passing to the `modules` argument" + ) + ) } else if (length(landing) > 1L) { stop("Only one `landing_popup_module` can be used.") } @@ -203,17 +215,20 @@ init <- function(data, ## `data` - `modules` if (inherits(data, "teal_data")) { - if (length(teal_data_datanames(data)) == 0) { + if (length(.teal_data_datanames(data)) == 0) { stop("The environment of `data` is empty.") } - # in case of teal_data_module this check is postponed to the srv_teal_with_splash - is_modules_ok <- check_modules_datanames(modules, teal_data_datanames(data)) + + if (!length(teal.data::datanames(data))) { + warning("`data` object has no datanames. Default datanames are set using `teal_data`'s environment.") + } + + is_modules_ok <- check_modules_datanames(modules, .teal_data_datanames(data)) if (!isTRUE(is_modules_ok)) { - logger::log_error(is_modules_ok) - checkmate::assert(is_modules_ok, .var.name = "modules") + lapply(is_modules_ok$string, logger::log_warn) } - is_filter_ok <- check_filter_datanames(filter, teal_data_datanames(data)) + is_filter_ok <- check_filter_datanames(filter, .teal_data_datanames(data)) if (!isTRUE(is_filter_ok)) { warning(is_filter_ok) # we allow app to continue if applied filters are outside @@ -221,21 +236,32 @@ init <- function(data, } } - # Note regarding case `id = character(0)`: - # rather than creating a submodule of this module, we directly modify - # the UI and server with `id = character(0)` and calling the server function directly + reporter <- teal.reporter::Reporter$new()$set_id(attr(filter, "app_id")) + if (is_arg_used(modules, "reporter") && length(extract_module(modules, "teal_module_previewer")) == 0) { + modules <- append_module( + modules, + reporter_previewer_module(server_args = list(previewer_buttons = c("download", "reset"))) + ) + } + + ns <- NS(id) # Note: UI must be a function to support bookmarking. res <- list( - ui = function(request) ui_teal_with_splash(id = id, data = data, title = title, header = header, footer = footer), + ui = function(request) { + ui_teal( + id = ns("teal"), data = if (inherits(data, "teal_data_module")) data, + modules = modules, title = title, header = header, footer = footer + ) + }, server = function(input, output, session) { - if (!is.null(landing_module)) { - do.call(landing_module$server, c(list(id = "landing_module_shiny_id"), landing_module$server_args)) + if (!is.null(landing_popup)) { + do.call(landing_popup$server, c(list(id = "landing_module_shiny_id"), landing_popup$server_args)) } - srv_teal_with_splash(id = id, data = data, modules = modules, filter = deep_copy_filter(filter)) + srv_teal(id = ns("teal"), data = data, modules = modules, filter = deep_copy_filter(filter)) } ) - logger::log_trace("init teal app has been initialized.") + logger::log_debug("init teal app has been initialized.") res } diff --git a/R/landing_popup_module.R b/R/landing_popup_module.R index 752cb0576c..2876a9f1ef 100644 --- a/R/landing_popup_module.R +++ b/R/landing_popup_module.R @@ -17,11 +17,11 @@ #' app1 <- init( #' data = teal_data(iris = iris), #' modules = modules( -#' landing_popup_module( -#' content = "A place for the welcome message or a disclaimer statement.", -#' buttons = modalButton("Proceed") -#' ), #' example_module() +#' ), +#' landing_popup = landing_popup_module( +#' content = "A place for the welcome message or a disclaimer statement.", +#' buttons = modalButton("Proceed") #' ) #' ) #' if (interactive()) { @@ -31,21 +31,21 @@ #' app2 <- init( #' data = teal_data(iris = iris), #' modules = modules( -#' landing_popup_module( -#' title = "Welcome", -#' content = tags$b( -#' "A place for the welcome message or a disclaimer statement.", -#' style = "color: red;" -#' ), -#' buttons = tagList( -#' modalButton("Proceed"), -#' actionButton("read", "Read more", -#' onclick = "window.open('http://google.com', '_blank')" -#' ), -#' actionButton("close", "Reject", onclick = "window.close()") -#' ) -#' ), #' example_module() +#' ), +#' landing_popup = landing_popup_module( +#' title = "Welcome", +#' content = tags$b( +#' "A place for the welcome message or a disclaimer statement.", +#' style = "color: red;" +#' ), +#' buttons = tagList( +#' modalButton("Proceed"), +#' actionButton("read", "Read more", +#' onclick = "window.open('http://google.com', '_blank')" +#' ), +#' actionButton("close", "Reject", onclick = "window.close()") +#' ) #' ) #' ) #' diff --git a/R/module_bookmark_manager.R b/R/module_bookmark_manager.R index 594392ba00..6f1af5b630 100644 --- a/R/module_bookmark_manager.R +++ b/R/module_bookmark_manager.R @@ -9,7 +9,7 @@ #' This module introduces bookmarks into `teal` apps: the `shiny` bookmarking mechanism becomes enabled #' and server-side bookmarks can be created. #' -#' The bookmark manager presents a button with the bookmark icon and is placed in the [`wunder_bar`]. +#' The bookmark manager presents a button with the bookmark icon and is placed in the tab-bar. #' When clicked, the button creates a bookmark and opens a modal which displays the bookmark URL. #' #' `teal` does not guarantee that all modules (`teal_module` objects) are bookmarkable. @@ -29,68 +29,63 @@ #' - set `options(shiny.bookmarkStore = "server")` before running the app #' #' -#' @inheritParams module_wunder_bar +#' @inheritParams init #' #' @return Invisible `NULL`. #' #' @aliases bookmark bookmark_manager bookmark_manager_module #' #' @name module_bookmark_manager +#' @rdname module_bookmark_manager +#' #' @keywords internal #' -bookmark_manager_ui <- function(id) { +NULL + +#' @rdname module_bookmark_manager +ui_bookmark_panel <- function(id, modules) { ns <- NS(id) - uiOutput(ns("bookmark_button"), inline = TRUE) + + bookmark_option <- get_bookmarking_option() + is_unbookmarkable <- need_bookmarking(modules) + shinyOptions(bookmarkStore = bookmark_option) + + # Render bookmark warnings count + if (!all(is_unbookmarkable) && identical(bookmark_option, "server")) { + tags$button( + id = ns("do_bookmark"), + class = "btn action-button wunder_bar_button bookmark_manager_button", + title = "Add bookmark", + tags$span( + suppressMessages(icon("fas fa-bookmark")), + if (any(is_unbookmarkable)) { + tags$span( + sum(is_unbookmarkable), + class = "badge-warning badge-count text-white bg-danger" + ) + } + ) + ) + } } #' @rdname module_bookmark_manager -#' @keywords internal -#' -bookmark_manager_srv <- function(id, modules) { +srv_bookmark_panel <- function(id, modules) { checkmate::assert_character(id) checkmate::assert_class(modules, "teal_modules") moduleServer(id, function(input, output, session) { - logger::log_trace("bookmark_manager_srv initializing") + logger::log_debug("bookmark_manager_srv initializing") ns <- session$ns - bookmark_option <- getShinyOption("bookmarkStore") - if (is.null(bookmark_option) && identical(getOption("shiny.bookmarkStore"), "server")) { - bookmark_option <- getOption("shiny.bookmarkStore") - # option alone doesn't activate bookmarking - we need to set shinyOptions - shinyOptions(bookmarkStore = bookmark_option) - } - - is_unbookmarkable <- unlist(rapply2( - modules_bookmarkable(modules), - Negate(isTRUE) - )) - - # Render bookmark warnings count - output$bookmark_button <- renderUI({ - if (!all(is_unbookmarkable) && identical(bookmark_option, "server")) { - tags$button( - id = ns("do_bookmark"), - class = "btn action-button wunder_bar_button bookmark_manager_button", - title = "Add bookmark", - tags$span( - suppressMessages(icon("solid fa-bookmark")), - if (any(is_unbookmarkable)) { - tags$span( - sum(is_unbookmarkable), - class = "badge-warning badge-count text-white bg-danger" - ) - } - ) - ) - } - }) + bookmark_option <- get_bookmarking_option() + is_unbookmarkable <- need_bookmarking(modules) # Set up bookmarking callbacks ---- # Register bookmark exclusions: do_bookmark button to avoid re-bookmarking setBookmarkExclude(c("do_bookmark")) # This bookmark can only be used on the app session. - app_session <- .subset2(shiny::getDefaultReactiveDomain(), "parent") + app_session <- .subset2(session, "parent") app_session$onBookmarked(function(url) { - logger::log_trace("bookmark_manager_srv@onBookmarked: bookmark button clicked, registering bookmark") + logger::log_debug("bookmark_manager_srv@onBookmarked: bookmark button clicked, registering bookmark") modal_content <- if (bookmark_option != "server") { msg <- sprintf( "Bookmarking has been set to \"%s\".\n%s\n%s", @@ -145,7 +140,7 @@ bookmark_manager_srv <- function(id, modules) { # manually trigger bookmarking because of the problems reported on windows with bookmarkButton in teal observeEvent(input$do_bookmark, { - logger::log_trace("bookmark_manager_srv@1 do_bookmark module clicked.") + logger::log_debug("bookmark_manager_srv@1 do_bookmark module clicked.") session$doBookmark() }) @@ -153,6 +148,25 @@ bookmark_manager_srv <- function(id, modules) { }) } + +#' @rdname module_bookmark_manager +get_bookmarking_option <- function() { + bookmark_option <- getShinyOption("bookmarkStore") + if (is.null(bookmark_option) && identical(getOption("shiny.bookmarkStore"), "server")) { + bookmark_option <- getOption("shiny.bookmarkStore") + } + bookmark_option +} + +#' @rdname module_bookmark_manager +need_bookmarking <- function(modules) { + unlist(rapply2( + modules_bookmarkable(modules), + Negate(isTRUE) + )) +} + + # utilities ---- #' Restore value from bookmark. diff --git a/R/module_data_summary.R b/R/module_data_summary.R new file mode 100644 index 0000000000..ca3c4a7962 --- /dev/null +++ b/R/module_data_summary.R @@ -0,0 +1,271 @@ +#' Data summary +#' @description +#' Module and its utils to display the number of rows and subjects in the filtered and unfiltered data. +#' +#' @details Handling different data classes: +#' `get_object_filter_overview()` is a pseudo S3 method which has variants for: +#' - `array` (`data.frame`, `DataFrame`, `array`, `Matrix` and `SummarizedExperiment`): Method variant +#' can be applied to any two-dimensional objects on which [ncol()] can be used. +#' - `MultiAssayExperiment`: for which summary contains counts for `colData` and all `experiments`. +#' +#' @param id (`character(1)`) +#' `shiny` module instance id. +#' @param teal_data (`reactive` returning `teal_data`) +#' +#' +#' @name module_data_summary +#' @rdname module_data_summary +#' @keywords internal +#' @return `NULL`. +NULL + +#' @rdname module_data_summary +ui_data_summary <- function(id) { + ns <- NS(id) + content_id <- ns("filters_overview_contents") + tags$div( + id = id, + class = "well", + tags$div( + class = "row", + tags$div( + class = "col-sm-9", + tags$label("Active Filter Summary", class = "text-primary mb-4") + ), + tags$div( + class = "col-sm-3", + tags$i( + class = "remove pull-right fa fa-angle-down", + style = "cursor: pointer;", + title = "fold/expand data summary panel", + onclick = sprintf("togglePanelItems(this, '%s', 'fa-angle-right', 'fa-angle-down');", content_id) + ) + ) + ), + tags$div( + id = content_id, + tags$div( + class = "teal_active_summary_filter_panel", + tableOutput(ns("table")) + ) + ) + ) +} + +#' @rdname module_data_summary +srv_data_summary <- function(id, teal_data) { + checkmate::check_class(teal_data, "reactive") + moduleServer( + id = id, + function(input, output, session) { + logger::log_debug("srv_data_summary initializing") + + summary_table <- reactive({ + req(inherits(teal_data(), "teal_data")) + + if (length(teal.data::datanames(teal_data())) == 0) { + return(NULL) + } + + filter_overview <- get_filter_overview(teal_data) + names(filter_overview)[[1]] <- "Data Name" + + filter_overview$Obs <- ifelse( + !is.na(filter_overview$obs), + sprintf("%s/%s", filter_overview$obs_filtered, filter_overview$obs), + ifelse(!is.na(filter_overview$obs_filtered), sprintf("%s", filter_overview$obs_filtered), "") + ) + + filter_overview$Subjects <- ifelse( + !is.na(filter_overview$subjects), + sprintf("%s/%s", filter_overview$subjects_filtered, filter_overview$subjects), + "" + ) + + filter_overview <- filter_overview[, colnames(filter_overview) %in% c("Data Name", "Obs", "Subjects")] + Filter(function(col) !all(col == ""), filter_overview) + }) + + output$table <- renderUI({ + summary_table_out <- try(summary_table(), silent = TRUE) + if (inherits(summary_table_out, "try-error")) { + # Ignore silent shiny error + if (!inherits(attr(summary_table_out, "condition"), "shiny.silent.error")) { + stop("Error occurred during data processing. See details in the main panel.") + } + } else if (is.null(summary_table_out)) { + "no datasets to show" + } else { + body_html <- apply( + summary_table_out, + 1, + function(x) { + tags$tr( + tagList( + tags$td( + if (all(x[-1] == "")) { + icon( + name = "fas fa-exclamation-triangle", + title = "Unsupported dataset", + `data-container` = "body", + `data-toggle` = "popover", + `data-content` = "object not supported by the data_summary module" + ) + }, + x[1] + ), + lapply(x[-1], tags$td) + ) + ) + } + ) + + header_labels <- names(summary_table()) + header_html <- tags$tr(tagList(lapply(header_labels, tags$td))) + + table_html <- tags$table( + class = "table custom-table", + tags$thead(header_html), + tags$tbody(body_html) + ) + table_html + } + }) + + summary_table # testing purpose + } + ) +} + +#' @rdname module_data_summary +get_filter_overview <- function(teal_data) { + datanames <- teal.data::datanames(teal_data()) + joinkeys <- teal.data::join_keys(teal_data()) + filtered_data_objs <- sapply( + datanames, function(name) teal.code::get_env(teal_data())[[name]], + simplify = FALSE + ) + unfiltered_data_objs <- sapply( + datanames, function(name) teal.code::get_env(teal_data())[[paste0(name, "._raw_")]], + simplify = FALSE + ) + + rows <- lapply( + datanames, + function(dataname) { + parent <- teal.data::parent(joinkeys, dataname) + + # todo: what should we display for a parent dataset? + # - Obs and Subjects + # - Obs only + # - Subjects only + # todo (for later): summary table should be displayed in a way that child datasets + # are indented under their parent dataset to form a tree structure + subject_keys <- if (length(parent) > 0) { + names(joinkeys[dataname, parent]) + } else { + joinkeys[dataname, dataname] + } + get_object_filter_overview( + filtered_data = filtered_data_objs[[dataname]], + unfiltered_data = unfiltered_data_objs[[dataname]], + dataname = dataname, + subject_keys = subject_keys + ) + } + ) + + unssuported_idx <- vapply(rows, function(x) all(is.na(x[-1])), logical(1)) # this is mainly for vectors + do.call(rbind, c(rows[!unssuported_idx], rows[unssuported_idx])) +} + +#' @rdname module_data_summary +#' @param filtered_data (`list`) of filtered objects +#' @param unfiltered_data (`list`) of unfiltered objects +#' @param dataname (`character(1)`) +get_object_filter_overview <- function(filtered_data, unfiltered_data, dataname, subject_keys) { + if (inherits(filtered_data, c("data.frame", "DataFrame", "array", "Matrix", "SummarizedExperiment"))) { + get_object_filter_overview_array(filtered_data, unfiltered_data, dataname, subject_keys) + } else if (inherits(filtered_data, "MultiAssayExperiment")) { + get_object_filter_overview_MultiAssayExperiment(filtered_data, unfiltered_data, dataname) + } else { + data.frame( + dataname = dataname, + obs = NA, + obs_filtered = NA, + subjects = NA, + subjects_filtered = NA + ) + } +} + +#' @rdname module_data_summary +get_object_filter_overview_array <- function(filtered_data, # nolint: object_length. + unfiltered_data, + dataname, + subject_keys) { + if (length(subject_keys) == 0) { + data.frame( + dataname = dataname, + obs = ifelse(!is.null(nrow(unfiltered_data)), nrow(unfiltered_data), NA), + obs_filtered = nrow(filtered_data), + subjects = NA, + subjects_filtered = NA + ) + } else { + data.frame( + dataname = dataname, + obs = ifelse(!is.null(nrow(unfiltered_data)), nrow(unfiltered_data), NA), + obs_filtered = nrow(filtered_data), + subjects = nrow(unique(unfiltered_data[subject_keys])), + subjects_filtered = nrow(unique(filtered_data[subject_keys])) + ) + } +} + +#' @rdname module_data_summary +get_object_filter_overview_MultiAssayExperiment <- function(filtered_data, # nolint: object_length, object_name. + unfiltered_data, + dataname) { + experiment_names <- names(unfiltered_data) + mae_info <- data.frame( + dataname = dataname, + obs = NA, + obs_filtered = NA, + subjects = nrow(unfiltered_data@colData), + subjects_filtered = nrow(filtered_data@colData) + ) + + experiment_obs_info <- do.call("rbind", lapply( + experiment_names, + function(experiment_name) { + transform( + get_object_filter_overview( + filtered_data[[experiment_name]], + unfiltered_data[[experiment_name]], + dataname = experiment_name, + subject_keys = join_keys() # empty join keys + ), + dataname = paste0(" - ", experiment_name) + ) + } + )) + + get_experiment_keys <- function(mae, experiment) { + sample_subset <- mae@sampleMap[mae@sampleMap$colname %in% colnames(experiment), ] + length(unique(sample_subset$primary)) + } + + experiment_subjects_info <- do.call("rbind", lapply( + experiment_names, + function(experiment_name) { + data.frame( + subjects = get_experiment_keys(filtered_data, unfiltered_data[[experiment_name]]), + subjects_filtered = get_experiment_keys(filtered_data, filtered_data[[experiment_name]]) + ) + } + )) + + experiment_info <- cbind(experiment_obs_info[, c("dataname", "obs", "obs_filtered")], experiment_subjects_info) + rbind(mae_info, experiment_info) +} diff --git a/R/module_filter_data.R b/R/module_filter_data.R new file mode 100644 index 0000000000..6c2593d3c7 --- /dev/null +++ b/R/module_filter_data.R @@ -0,0 +1,93 @@ +#' Filter panel module in teal +#' +#' Creates filter panel module from `teal_data` object and returns `teal_data`. It is build in a way +#' that filter panel changes and anything what happens before (e.g. [`module_init_data`]) is triggering +#' further reactive events only if something has changed and if the module is visible. Thanks to +#' this special implementation all modules' data are recalculated only for those modules which are +#' currently displayed. +#' +#' @return A `eventReactive` containing `teal_data` containing filtered objects and filter code. +#' `eventReactive` triggers only if all conditions are met: +#' - tab is selected (`is_active`) +#' - when filters are changed (`get_filter_expr` is different than previous) +#' +#' @inheritParams module_teal_module +#' @param active_datanames (`reactive` returning `character`) this module's data names +#' @name module_filter_data +#' @keywords internal +NULL + +#' @rdname module_filter_data +ui_filter_data <- function(id) { + ns <- shiny::NS(id) + uiOutput(ns("panel")) +} + +#' @rdname module_filter_data +srv_filter_data <- function(id, datasets, active_datanames, data_rv, is_active) { + checkmate::assert_class(datasets, "reactive") + moduleServer(id, function(input, output, session) { + output$panel <- renderUI({ + req(inherits(datasets(), "FilteredData")) + isolate({ + # render will be triggered only when FilteredData object changes (not when filters change) + # technically it means that teal_data_module needs to be refreshed + logger::log_debug("srv_filter_panel rendering filter panel.") + if (length(active_datanames())) { + datasets()$srv_active("filters", active_datanames = active_datanames) + # todo: make sure to bump the `teal.slice` version. Please use the branch `669_insertUI@main` in `teal.slice`. + datasets()$ui_active(session$ns("filters"), active_datanames = active_datanames) + } + }) + }) + + trigger_data <- .observe_active_filter_changed(datasets, is_active, active_datanames, data_rv) + + eventReactive(trigger_data(), { + .make_filtered_teal_data(modules, data = data_rv(), datasets = datasets(), datanames = active_datanames()) + }) + }) +} + +#' @rdname module_filter_data +.make_filtered_teal_data <- function(modules, data, datasets = NULL, datanames) { + data <- eval_code(data, sprintf("%1$s._raw_ <- %1$s", datanames)) + filtered_code <- teal.slice::get_filter_expr(datasets = datasets, datanames = datanames) + filtered_teal_data <- .append_evaluated_code(data, filtered_code) + filtered_datasets <- sapply(datanames, function(x) datasets$get_data(x, filtered = TRUE), simplify = FALSE) + filtered_teal_data <- .append_modified_data(filtered_teal_data, filtered_datasets) + filtered_teal_data +} + +#' @rdname module_filter_data +.observe_active_filter_changed <- function(datasets, is_active, active_datanames, data_rv) { + previous_signature <- reactiveVal(NULL) + filter_changed <- reactive({ + req(inherits(datasets(), "FilteredData")) + new_signature <- c( + teal.data::get_code(data_rv()), + teal.slice::get_filter_expr(datasets = datasets(), datanames = active_datanames()) + ) + if (!identical(previous_signature(), new_signature)) { + previous_signature(new_signature) + TRUE + } else { + FALSE + } + }) + + trigger_data <- reactiveVal(NULL) + observe({ + if (isTRUE(is_active() && filter_changed())) { + isolate({ + if (is.null(trigger_data())) { + trigger_data(0) + } else { + trigger_data(trigger_data() + 1) + } + }) + } + }) + + trigger_data +} diff --git a/R/module_filter_manager.R b/R/module_filter_manager.R index 611a1bcc38..69a7039d2a 100644 --- a/R/module_filter_manager.R +++ b/R/module_filter_manager.R @@ -1,41 +1,94 @@ #' Manage multiple `FilteredData` objects #' +#' @description #' Oversee filter states across the entire application. #' -#' This module observes changes in the filters of each `FilteredData` object -#' and keeps track of all filters used. A mapping of filters to modules -#' is kept in the `mapping_matrix` object (which is actually a `data.frame`) -#' that tracks which filters (rows) are active in which modules (columns). +#' @section Slices global: +#' The key role in maintaining the module-specific filter states is played by the `.slicesGlobal` +#' object. It is a reference class that holds the following fields: +#' - `all_slices` (`reactiveVal`) - reactive value containing all filters registered in an app. +#' - `module_slices_api` (`reactiveValues`) - reactive field containing references to each modules' +#' `FilteredData` object methods. At this moment it is used only in `srv_filter_manager` to display +#' the filter states in a table combining informations from `all_slices` and from +#' `FilteredData$get_available_teal_slices()`. +#' +#' During a session only new filters are added to `all_slices` unless [`module_snapshot_manager`] is +#' used to restore previous state. Filters from `all_slices` can be activated or deactivated in a +#' module which is linked (both ways) by `attr(, "mapping")` so that: +#' - If module's filter is added or removed in its `FilteredData` object, this information is passed +#' to `SlicesGlobal` which updates `attr(, "mapping")` accordingly. +#' - When mapping changes in a `SlicesGlobal`, filters are set or removed from module's +#' `FilteredData`. +#' +#' @section Filter manager: +#' Filter-manager is split into two parts: +#' 1. `ui/srv_filter_manager_panel` - Called once for the whole app. This module observes changes in +#' the filters in `slices_global` and displays them in a table utilizing information from `mapping`: +#' - (`TRUE`) - filter is active in the module +#' - (`FALSE`) - filter is inactive in the module +#' - (`NA`) - filter is not available in the module +#' 2. `ui/srv_module_filter_manager` - Called once for each `teal_module`. Handling filter states +#' for of single module and keeping module `FilteredData` consistent with `slices_global`, so that +#' local filters are always reflected in the `slices_global` and its mapping and vice versa. +#' #' #' @param id (`character(1)`) #' `shiny` module instance id. -#' @param datasets (named `list`) -#' A list, possibly nested, of `FilteredData` objects. -#' Each `FilteredData` will be served to one module in the `teal` application. -#' The structure of the list must reflect the nesting of modules in tabs -#' and the names of the list must match the labels of their respective modules. -#' @inheritParams init #' -#' @return -#' A `list` containing: +#' @param slices_global (`reactiveVal`) +#' containing `teal_slices`. +#' +#' @param module_fd (`FilteredData`) +#' Object containing the data to be filtered in a single `teal` module. #' -#' objects used by other manager modules -#' - `datasets_flat`: named list of `FilteredData` objects, -#' - `mapping_matrix`: `reactive` containing a `data.frame`, -#' - `slices_global`: `reactiveVal` containing a `teal_slices` object, +#' @return +#' Module returns a `slices_global` (`reactiveVal`) containing a `teal_slices` object with mapping. #' -#' objects used for testing -#' - modules_out: `list` of `reactive`s, each holding a `teal_slices`, as returned by `filter_manager_module_srv`. +#' @encoding UTF-8 #' #' @name module_filter_manager -#' @aliases filter_manager filter_manager_module +#' @rdname module_filter_manager #' +NULL + +#' @rdname module_filter_manager +ui_filter_manager_panel <- function(id) { + ns <- NS(id) + tags$button( + id = ns("show_filter_manager"), + class = "btn action-button wunder_bar_button", + title = "View filter mapping", + suppressMessages(icon("fas fa-grip")) + ) +} #' @rdname module_filter_manager #' @keywords internal -#' -filter_manager_ui <- function(id) { +srv_filter_manager_panel <- function(id, slices_global) { + checkmate::assert_string(id) + checkmate::assert_class(slices_global, ".slicesGlobal") + moduleServer(id, function(input, output, session) { + setBookmarkExclude(c("show_filter_manager")) + observeEvent(input$show_filter_manager, { + logger::log_debug("srv_filter_manager_panel@1 show_filter_manager button has been clicked.") + showModal( + modalDialog( + ui_filter_manager(session$ns("filter_manager")), + class = "filter_manager_modal", + size = "l", + footer = NULL, + easyClose = TRUE + ) + ) + }) + srv_filter_manager("filter_manager", slices_global = slices_global) + }) +} + +#' @rdname module_filter_manager +ui_filter_manager <- function(id) { ns <- NS(id) + actionButton(ns("filter_manager"), NULL, icon = icon("fas fa-filter")) tags$div( class = "filter_manager_content", tableOutput(ns("slices_table")) @@ -43,45 +96,63 @@ filter_manager_ui <- function(id) { } #' @rdname module_filter_manager -#' @keywords internal -#' -filter_manager_srv <- function(id, datasets, filter) { +srv_filter_manager <- function(id, slices_global) { + checkmate::assert_string(id) + checkmate::assert_class(slices_global, ".slicesGlobal") + moduleServer(id, function(input, output, session) { - logger::log_trace("filter_manager_srv initializing for: { paste(names(datasets), collapse = ', ')}.") + logger::log_debug("filter_manager_srv initializing.") - is_module_specific <- isTRUE(attr(filter, "module_specific")) + # Bookmark slices global with mapping. + session$onBookmark(function(state) { + logger::log_debug("filter_manager_srv@onBookmark: storing filter state") + state$values$filter_state_on_bookmark <- as.list( + slices_global$all_slices(), + recursive = TRUE + ) + }) - # Create a global list of slices. - # Contains all available teal_slice objects available to all modules. - # Passed whole to instances of FilteredData used for individual modules. - # Down there a subset that pertains to the data sets used in that module is applied and displayed. - slices_global <- reactiveVal(filter) + bookmarked_slices <- restoreValue(session$ns("filter_state_on_bookmark"), NULL) + if (!is.null(bookmarked_slices)) { + logger::log_debug("filter_manager_srv: restoring filter state from bookmark.") + slices_global$slices_set(bookmarked_slices) + } - datasets_flat <- - if (!is_module_specific) { - flatten_datasets(unlist(datasets)[[1]]) - } else { - flatten_datasets(datasets) - } + mapping_table <- reactive({ + # We want this to be reactive on slices_global$all_slices() only as get_available_teal_slices() + # is dependent on slices_global$all_slices(). + module_labels <- setdiff( + names(attr(slices_global$all_slices(), "mapping")), + "Report previewer" + ) + isolate({ + mm <- as.data.frame( + sapply( + module_labels, + simplify = FALSE, + function(module_label) { + available_slices <- slices_global$module_slices_api[[module_label]]$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") + active_ids <- global_ids %in% module_ids + setNames(nm = global_ids, ifelse(global_ids %in% allowed_ids, active_ids, NA)) + } + ), + check.names = FALSE + ) + colnames(mm)[colnames(mm) == "global_filters"] <- "Global filters" - # Create mapping of filters to modules in matrix form (presented as data.frame). - # Modules get NAs for filters that cannot be set for them. - mapping_matrix <- reactive({ - state_ids_global <- vapply(slices_global(), `[[`, character(1L), "id") - mapping_smooth <- lapply(datasets_flat, function(x) { - state_ids_local <- vapply(x$get_filter_state(), `[[`, character(1L), "id") - state_ids_allowed <- vapply(x$get_available_teal_slices()(), `[[`, character(1L), "id") - states_active <- state_ids_global %in% state_ids_local - ifelse(state_ids_global %in% state_ids_allowed, states_active, NA) + mm }) - - as.data.frame(mapping_smooth, row.names = state_ids_global, check.names = FALSE) }) output$slices_table <- renderTable( expr = { + logger::log_debug("filter_manager_srv@1 rendering slices_table.") + mm <- mapping_table() + # Display logical values as UTF characters. - mm <- mapping_matrix() mm[] <- lapply(mm, ifelse, yes = intToUtf8(9989), no = intToUtf8(10060)) mm[] <- lapply(mm, function(x) ifelse(is.na(x), intToUtf8(128306), x)) @@ -90,113 +161,231 @@ filter_manager_srv <- function(id, datasets, filter) { mm <- data.frame(`Filter manager` = "No filters specified.", check.names = FALSE) rownames(mm) <- "" } - - # Report Previewer will not be displayed. - mm[names(mm) != "Report previewer"] + mm }, - align = paste(c("l", rep("c", sum(names(datasets_flat) != "Report previewer"))), collapse = ""), rownames = TRUE ) - # Create list of module calls. - modules_out <- lapply(names(datasets_flat), function(module_name) { - filter_manager_module_srv( - id = module_name, - module_fd = datasets_flat[[module_name]], - slices_global = slices_global - ) - }) - - list( - slices_global = slices_global, - mapping_matrix = mapping_matrix, - datasets_flat = datasets_flat, - modules_out = modules_out # returned for testing purpose - ) + mapping_table # for testing purpose }) } -#' Module specific filter manager -#' -#' Tracks filter states in a single module. -#' -#' This module tracks the state of a single `FilteredData` object and global `teal_slices` -#' and updates both objects as necessary. Filter states added in different modules -#' Filter states added any individual module are added to global `teal_slices` -#' and from there become available in other modules -#' by setting `private$available_teal_slices` in each `FilteredData`. -#' -#' @param id (`character(1)`) -#' `shiny` module id. -#' @param module_fd (`FilteredData`) -#' Object containing the data to be filtered in a single `teal` module. -#' @param slices_global (`reactiveVal`) -#' stores `teal_slices` with all available filters; allows the following actions: -#' - to disable/enable a specific filter in a module -#' - to restore saved filter settings -#' - to save current filter panel settings -#' @return A `reactive` expression containing a `teal_slices` with the slices active in this module. -#' @keywords internal -#' -filter_manager_module_srv <- function(id, module_fd, slices_global) { +#' @rdname module_filter_manager +srv_module_filter_manager <- function(id, module_fd, slices_global) { + checkmate::assert_string(id) + checkmate::assert_class(module_fd, "reactive") + checkmate::assert_class(slices_global, ".slicesGlobal") + moduleServer(id, function(input, output, session) { - # Only operate on slices that refer to data sets present in this module. - module_fd$set_available_teal_slices(reactive(slices_global())) - - # Track filter state of this module. - slices_module <- reactive(module_fd$get_filter_state()) - - # Reactive values for comparing states. - previous_slices <- reactiveVal(isolate(slices_module())) - slices_added <- reactiveVal(NULL) - - # Observe changes in module filter state and trigger appropriate actions. - observeEvent(slices_module(), ignoreNULL = FALSE, { - logger::log_trace("filter_manager_srv@1 detecting states deltas in module: { id }.") - added <- setdiff_teal_slices(slices_module(), slices_global()) - if (length(added)) slices_added(added) - previous_slices(slices_module()) + logger::log_debug("srv_module_filter_manager initializing for module: { id }.") + # Track filter global and local states. + slices_global_module <- reactive({ + slices_global$slices_get(module_label = id) }) + slices_module <- reactive(req(module_fd())$get_filter_state()) - observeEvent(slices_added(), ignoreNULL = TRUE, { - logger::log_trace("filter_manager_srv@2 added filter in module: { id }.") - # In case the new state has the same id as an existing state, add a suffix to it. - global_ids <- vapply(slices_global(), `[[`, character(1L), "id") - lapply( - slices_added(), - function(slice) { - if (slice$id %in% global_ids) { - slice$id <- utils::tail(make.unique(c(global_ids, slice$id), sep = "_"), 1) - } - } + module_fd_previous <- reactiveVal(NULL) + + # Set (reactively) available filters for the module. + obs1 <- observeEvent(module_fd(), priority = 1, { + logger::log_debug("srv_module_filter_manager@1 setting initial slices for module: { id }.") + # Filters relevant for the module in module-specific app. + slices <- slices_global_module() + + # Clean up previous filter states and refresh cache of previous module_fd with current + if (!is.null(module_fd_previous())) module_fd_previous()$finalize() + module_fd_previous(module_fd()) + + # Setting filter states from slices_global: + # 1. when app initializes slices_global set to initial filters (specified by app developer) + # 2. when data reinitializes slices_global reflects latest filter states + + module_fd()$set_filter_state(slices) + + # irrelevant filters are discarded in FilteredData$set_available_teal_slices + # it means we don't need to subset slices_global$all_slices() from filters refering to irrelevant datasets + module_fd()$set_available_teal_slices(slices_global$all_slices) + + # this needed in filter_manager_srv + slices_global$module_slices_api_set( + id, + list( + get_available_teal_slices = module_fd()$get_available_teal_slices(), + set_filter_state = module_fd()$set_filter_state, # for testing purpose + get_filter_state = module_fd()$get_filter_state # for testing purpose + ) ) - slices_global_new <- c(slices_global(), slices_added()) - slices_global(slices_global_new) - slices_added(NULL) + }) + + # Update global state and mapping matrix when module filters change. + obs2 <- observeEvent(slices_module(), priority = 0, { + this_slices <- slices_module() + slices_global$slices_append(this_slices) # append new slices to the all_slices list + mapping_elem <- setNames(nm = id, list(vapply(this_slices, `[[`, character(1L), "id"))) + slices_global$slices_active(mapping_elem) + }) + + obs3 <- observeEvent(slices_global_module(), { + global_vs_module <- setdiff_teal_slices(slices_global_module(), slices_module()) + module_vs_global <- setdiff_teal_slices(slices_module(), slices_global_module()) + if (length(global_vs_module) || length(module_vs_global)) { + # Comment: (Nota Bene) Normally new filters for a module are added through module-filter-panel, and slices + # global are updated automatically so slices_module -> slices_global_module are equal. + # this if is valid only when a change is made on the global level so the change needs to be propagated down + # to the module (for example through snapshot manager). If it happens both slices are different + logger::log_debug("srv_module_filter_manager@3 (N.B.) global state has changed for a module:{ id }.") + module_fd()$clear_filter_states() + module_fd()$set_filter_state(slices_global_module()) + } }) slices_module # returned for testing purpose }) } +#' @importFrom shiny reactiveVal reactiveValues +methods::setOldClass("reactiveVal") +methods::setOldClass("reactivevalues") + +#' @importFrom methods new +#' @rdname module_filter_manager +.slicesGlobal <- methods::setRefClass(".slicesGlobal", # nolint: object_name. + fields = list( + all_slices = "reactiveVal", + module_slices_api = "reactivevalues" + ), + methods = list( + initialize = function(slices = teal_slices(), module_labels) { + shiny::isolate({ + checkmate::assert_class(slices, "teal_slices") + # needed on init to not mix "global_filters" with module-specific-slots + if (isTRUE(attr(slices, "module_specific"))) { + old_mapping <- attr(slices, "mapping") + new_mapping <- sapply(module_labels, simplify = FALSE, function(module_label) { + unique(unlist(old_mapping[c(module_label, "global_filters")])) + }) + attr(slices, "mapping") <- new_mapping + } + .self$all_slices <<- shiny::reactiveVal(slices) + .self$module_slices_api <<- shiny::reactiveValues() + .self$slices_append(slices) + .self$slices_active(attr(slices, "mapping")) + invisible(.self) + }) + }, + is_module_specific = function() { + isTRUE(attr(.self$all_slices(), "module_specific")) + }, + module_slices_api_set = function(module_label, functions_list) { + shiny::isolate({ + if (!.self$is_module_specific()) { + module_label <- "global_filters" + } + if (!identical(.self$module_slices_api[[module_label]], functions_list)) { + .self$module_slices_api[[module_label]] <- functions_list + } + invisible(.self) + }) + }, + slices_deactivate_all = function(module_label) { + shiny::isolate({ + new_slices <- .self$all_slices() + old_mapping <- attr(new_slices, "mapping") + + new_mapping <- if (.self$is_module_specific()) { + new_module_mapping <- setNames(nm = module_label, list(character(0))) + modifyList(old_mapping, new_module_mapping) + } else if (missing(module_label)) { + lapply( + attr(.self$all_slices(), "mapping"), + function(x) character(0) + ) + } else { + old_mapping[[module_label]] <- character(0) + old_mapping + } + if (!identical(new_mapping, old_mapping)) { + logger::log_debug(".slicesGlobal@slices_deactivate_all: deactivating all slices.") + attr(new_slices, "mapping") <- new_mapping + .self$all_slices(new_slices) + } + invisible(.self) + }) + }, + slices_active = function(mapping_elem) { + shiny::isolate({ + if (.self$is_module_specific()) { + new_mapping <- modifyList(attr(.self$all_slices(), "mapping"), mapping_elem) + } else { + new_mapping <- setNames(nm = "global_filters", list(unique(unlist(mapping_elem)))) + } -# utilities ---- + if (!identical(new_mapping, attr(.self$all_slices(), "mapping"))) { + mapping_modules <- toString(names(new_mapping)) + logger::log_debug(".slicesGlobal@slices_active: changing mapping for module(s): { mapping_modules }.") + new_slices <- .self$all_slices() + attr(new_slices, "mapping") <- new_mapping + .self$all_slices(new_slices) + } -#' Flatten potentially nested list of FilteredData objects while maintaining useful names. -#' Simply using `unlist` would result in concatenated names. -#' A single `FilteredData` will result in a list named "Global Filters" -#' because that name used in the mapping matrix display. -#' @param x `FilteredData` or a `list` thereof -#' @param name (`character(1)`) string used to name `x` in the resulting list -#' @return Unnested named list of `FilteredData` objects. -#' @keywords internal -#' @noRd -#' -flatten_datasets <- function(x, name = "Global Filters") { - if (inherits(x, "FilteredData")) { - setNames(list(x), name) - } else { - unlist(lapply(names(x), function(name) flatten_datasets(x[[name]], name))) - } -} + invisible(.self) + }) + }, + # - only new filters are appended to the $all_slices + # - mapping is not updated here + slices_append = function(slices, activate = FALSE) { + shiny::isolate({ + if (!is.teal_slices(slices)) { + slices <- as.teal_slices(slices) + } + + # to make sure that we don't unnecessary trigger $all_slices + new_slices <- setdiff_teal_slices(slices, .self$all_slices()) + old_mapping <- attr(.self$all_slices(), "mapping") + if (length(new_slices)) { + new_ids <- vapply(new_slices, `[[`, character(1L), "id") + logger::log_debug(".slicesGlobal@slices_append: appending new slice(s): { new_ids }.") + slices_ids <- vapply(.self$all_slices(), `[[`, character(1L), "id") + lapply(new_slices, function(slice) { + # In case the new state has the same id as an existing one, add a suffix + if (slice$id %in% slices_ids) { + slice$id <- utils::tail(make.unique(c(slices_ids, slice$id), sep = "_"), 1) + } + }) + + new_slices_all <- c(.self$all_slices(), new_slices) + attr(new_slices_all, "mapping") <- old_mapping + .self$all_slices(new_slices_all) + } + + invisible(.self) + }) + }, + slices_get = function(module_label) { + if (missing(module_label)) { + .self$all_slices() + } else { + module_ids <- unlist(attr(.self$all_slices(), "mapping")[c(module_label, "global_filters")]) + Filter( + function(slice) slice$id %in% module_ids, + .self$all_slices() + ) + } + }, + slices_set = function(slices) { + shiny::isolate({ + if (!is.teal_slices(slices)) { + slices <- as.teal_slices(slices) + } + .self$all_slices(slices) + invisible(.self) + }) + }, + show = function() { + shiny::isolate(print(.self$all_slices())) + invisible(.self) + } + ) +) +# todo: prevent any teal_slices attribute except mapping diff --git a/R/module_init_data.R b/R/module_init_data.R new file mode 100644 index 0000000000..bb4ec3e9f9 --- /dev/null +++ b/R/module_init_data.R @@ -0,0 +1,191 @@ +#' Data Module for teal +#' +#' This module manages the `data` argument for `srv_teal`. The `teal` framework uses [teal_data()], +#' which can be provided in various ways: +#' 1. Directly as a [teal.data::teal_data()] object. This will automatically convert it into a `reactive` `teal_data`. +#' 2. As a `reactive` object that returns a [teal.data::teal_data()] object. +#' +#' @details +#' ## Reactive `teal_data`: +#' +#' The data in the application can be reactively updated, prompting [srv_teal()] to rebuild the +#' content accordingly. There are two methods for creating interactive `teal_data`: +#' 1. Using a `reactive` object provided from outside the `teal` application. In this scenario, +#' reactivity is controlled by an external module, and `srv_teal` responds to changes. +#' 2. Using [teal_data_module()], which is embedded within the `teal` application, allowing data to +#' be resubmitted by the user as needed. +#' +#' Since the server of [teal_data_module()] must return a `reactive` `teal_data` object, both +#' methods (1 and 2) produce the same reactive behavior within a `teal` application. The distinction +#' lies in data control: the first method involves external control, while the second method +#' involves control from a custom module within the app. +#' +#' For more details, see [`module_teal_data`]. +#' +#' @inheritParams init +#' +#' @param data (`teal_data`, `teal_data_module`, or `reactive` returning `teal_data`) +#' The `ui` component of this module does not require `data` if `teal_data_module` is not provided. +#' The `data` argument in the `ui` is included solely for the `$ui` function of the +#' `teal_data_module`. Otherwise, it can be disregarded, ensuring that `ui_teal` does not depend on +#' the reactive data of the enclosing application. +#' +#' @return A `reactive` object that returns: +#' - `teal_data` when the object is validated +#' - `shiny.silent.error` when not validated. +#' +#' @rdname module_init_data +#' @name module_init_data +#' @keywords internal +NULL + +#' @rdname module_init_data +ui_init_data <- function(id, data) { + ns <- shiny::NS(id) + shiny::div( + id = ns("content"), + style = "display: inline-block;", + if (inherits(data, "teal_data_module")) { + ui_teal_data(ns("teal_data_module"), data_module = data) + } else { + NULL + } + ) +} + +#' @rdname module_init_data +srv_init_data <- function(id, data, modules, filter = teal_slices()) { + checkmate::assert_character(id, max.len = 1, any.missing = FALSE) + checkmate::assert_multi_class(data, c("teal_data", "teal_data_module", "reactive", "reactiveVal")) + checkmate::assert_class(modules, "teal_modules") + checkmate::assert_class(filter, "teal_slices") + + moduleServer(id, function(input, output, session) { + logger::log_debug("srv_data initializing.") + + if (getOption("teal.show_js_log", default = FALSE)) { + shinyjs::showLog() + } + + # data_rv contains teal_data object + # either passed to teal::init or returned from teal_data_module + data_validated <- if (inherits(data, "teal_data_module")) { + srv_teal_data( + "teal_data_module", + data = reactive(req(FALSE)), # to .fallback_on_failure to shiny.silent.error + data_module = data, + modules = modules, + validate_shiny_silent_error = FALSE + ) + } else if (inherits(data, "teal_data")) { + reactiveVal(data) + } else if (inherits(data, c("reactive", "reactiveVal"))) { + .fallback_on_failure(this = data, that = reactive(req(FALSE)), label = "Reactive data") + } + + if (inherits(data, "teal_data_module")) { + shinyjs::disable(selector = sprintf(".teal-body:has('#%s') .nav li a", session$ns("content"))) + } + + observeEvent(data_validated(), { + showNotification("Data loaded successfully.", duration = 5) + shinyjs::enable(selector = sprintf(".teal-body:has('#%s') .nav li a", session$ns("content"))) + if (isTRUE(attr(data, "once"))) { + # Hiding the data module tab. + shinyjs::hide( + selector = sprintf( + ".teal-body:has('#%s') a[data-value='teal_data_module']", + session$ns("content") + ) + ) + # Clicking the second tab, which is the first module. + shinyjs::runjs( + sprintf( + "document.querySelector('.teal-body:has(#%s) .nav li:nth-child(2) a').click();", + session$ns("content") + ) + ) + } + + is_filter_ok <- check_filter_datanames(filter, .teal_data_datanames(data_validated())) + if (!isTRUE(is_filter_ok)) { + showNotification( + "Some filters were not applied because of incompatibility with data. Contact app developer.", + type = "warning", + duration = 10 + ) + warning(is_filter_ok) + } + }) + + observeEvent(data_validated(), once = TRUE, { + # Excluding the ids from teal_data_module using full namespace and global shiny app session. + app_session <- .subset2(shiny::getDefaultReactiveDomain(), "parent") + setBookmarkExclude( + session$ns( + grep( + pattern = "teal_data_module-", + x = names(reactiveValuesToList(input)), + value = TRUE + ) + ), + session = app_session + ) + }) + + # Adds signature protection to the datanames in the data + reactive(.add_signature_to_data(data_validated())) + }) +} + +#' Adds signature protection to the `datanames` in the data +#' @param data (`teal_data`) +#' @return `teal_data` with additional code that has signature of the `datanames` +#' @keywords internal +.add_signature_to_data <- function(data) { + hashes <- .get_hashes_code(data) + + tdata <- do.call( + teal.data::teal_data, + c( + list(code = trimws(c(teal.code::get_code(data), hashes), which = "right")), + list(join_keys = teal.data::join_keys(data)), + sapply( + ls(teal.code::get_env(data)), + teal.code::get_var, + object = data, + simplify = FALSE + ) + ) + ) + + tdata@verified <- data@verified + teal.data::datanames(tdata) <- teal.data::datanames(data) + tdata +} + +#' Get code that tests the integrity of the reproducible data +#' +#' @param data (`teal_data`) object holding the data +#' @param datanames (`character`) names of `datasets` +#' +#' @return A character vector with the code lines. +#' @keywords internal +#' +.get_hashes_code <- function(data, datanames = .teal_data_datanames(data)) { + # todo: this should be based on data_rv object not on datasets + vapply( + datanames, + function(dataname, datasets) { + hash <- rlang::hash(data[[dataname]]) + sprintf( + "stopifnot(%s == %s) # @linksto %s", + deparse1(bquote(rlang::hash(.(as.name(dataname))))), + deparse1(hash), + dataname + ) + }, + character(1L), + USE.NAMES = TRUE + ) +} diff --git a/R/module_nested_tabs.R b/R/module_nested_tabs.R index 4e320a954c..8350611f51 100644 --- a/R/module_nested_tabs.R +++ b/R/module_nested_tabs.R @@ -1,58 +1,49 @@ -#' Create a UI of nested tabs of `teal_modules` +#' Calls all `modules` #' -#' @section `ui_nested_tabs`: -#' Each `teal_modules` is translated to a `tabsetPanel` and each -#' of its children is another tab-module called recursively. The UI of a -#' `teal_module` is obtained by calling its UI function. +#' On the UI side each `teal_modules` is translated to a `tabsetPanel` and each `teal_module` is a +#' `tabPanel`. Both, UI and server are called recursively so that each tab is a separate module and +#' reflect nested structure of `modules` argument. #' -#' The `datasets` argument is required to resolve the `teal` arguments in an -#' isolated context (with respect to reactivity). +#' @name module_teal_module #' -#' @section `srv_nested_tabs`: -#' This module recursively calls all elements of `modules` and returns currently active one. -#' - `teal_module` returns self as a active module. -#' - `teal_modules` also returns module active within self which is determined by the `input$active_tab`. +#' @inheritParams module_teal #' -#' @name module_nested_tabs +#' @param data_rv (`reactive` returning `teal_data`) #' -#' @inheritParams module_tabs_with_filters +#' @param slices_global (`reactiveVal` returning `modules_teal_slices`) +#' see [`module_filter_manager`] #' #' @param depth (`integer(1)`) #' number which helps to determine depth of the modules nesting. -#' @param is_module_specific (`logical(1)`) -#' flag determining if the filter panel is global or module-specific. -#' When set to `TRUE`, a filter panel is called inside of each module tab. -#' @param progress (`Progress`) object from `shiny` #' -#' @return -#' Depending on the class of `modules`, `ui_nested_tabs` returns: -#' - `teal_module`: instantiated UI of the module. -#' - `teal_modules`: `tabsetPanel` with each tab corresponding to recursively -#' calling this function on it. +#' @param datasets (`reactive` returning `FilteredData` or `NULL`) +#' When `datasets` is passed from the parent module (`srv_teal`) then `dataset` is a singleton +#' which implies in filter-panel to be "global". When `NULL` then filter-panel is "module-specific". #' -#' `srv_nested_tabs` returns a reactive which returns the active module that corresponds to the selected tab. +#' @return +#' output of currently active module. +#' - `srv_teal_module.teal_module` returns `reactiveVal` containing output of the called module. +#' - `srv_teal_module.teal_modules` returns output of module selected by `input$active_tab`. #' #' @keywords internal NULL -#' @rdname module_nested_tabs -ui_nested_tabs <- function(id, modules, datasets, depth = 0L, is_module_specific = FALSE, progress = NULL) { - checkmate::assert_multi_class(modules, c("teal_modules", "teal_module")) +#' @rdname module_teal_module +ui_teal_module <- function(id, modules, depth = 0L) { + checkmate::assert_multi_class(modules, c("teal_modules", "teal_module", "shiny.tag")) checkmate::assert_count(depth) - checkmate::assert_r6(progress, "Progress", null.ok = TRUE) - UseMethod("ui_nested_tabs", modules) + UseMethod("ui_teal_module", modules) } -#' @rdname module_nested_tabs +#' @rdname module_teal_module #' @export -ui_nested_tabs.default <- function(id, modules, datasets, depth = 0L, is_module_specific = FALSE, progress = NULL) { +ui_teal_module.default <- function(id, modules, depth = 0L) { stop("Modules class not supported: ", paste(class(modules), collapse = " ")) } -#' @rdname module_nested_tabs +#' @rdname module_teal_module #' @export -ui_nested_tabs.teal_modules <- function(id, modules, datasets, depth = 0L, is_module_specific = FALSE, progress = NULL) { # nolint: line_length. - checkmate::assert_list(datasets, types = c("list", "FilteredData")) +ui_teal_module.teal_modules <- function(id, modules, depth = 0L) { ns <- NS(id) do.call( tabsetPanel, @@ -66,16 +57,16 @@ ui_nested_tabs.teal_modules <- function(id, modules, datasets, depth = 0L, is_mo names(modules$children), function(module_id) { module_label <- modules$children[[module_id]]$label + if (is.null(module_label)) { + module_label <- icon("fas fa-database") + } tabPanel( title = module_label, value = module_id, # when clicked this tab value changes input$ - ui_nested_tabs( + ui_teal_module( id = ns(module_id), modules = modules$children[[module_id]], - datasets = datasets[[module_label]], - depth = depth + 1L, - is_module_specific = is_module_specific, - progress = progress + depth = depth + 1L ) ) } @@ -84,235 +75,237 @@ ui_nested_tabs.teal_modules <- function(id, modules, datasets, depth = 0L, is_mo ) } -#' @rdname module_nested_tabs +#' @rdname module_teal_module #' @export -ui_nested_tabs.teal_module <- function(id, modules, datasets, depth = 0L, is_module_specific = FALSE, progress = NULL) { - checkmate::assert_class(datasets, classes = "FilteredData") - ns <- NS(id) - - if (!is.null(progress)) { - progress$inc( - amount = 1, - detail = sprintf("%s%%", round(progress$getValue() / progress$getMax(), 2L) * 100) - ) - } +ui_teal_module.shiny.tag <- function(id, modules, depth = 0L) { + modules +} +#' @rdname module_teal_module +#' @export +ui_teal_module.teal_module <- function(id, modules, depth = 0L) { + ns <- NS(id) args <- c(list(id = ns("module")), modules$ui_args) - teal_ui <- tags$div( + ui_teal <- div( + div( + class = "teal_validated", + ui_validate_reactive_teal_data(ns("validate_datanames")) + ), + do.call(modules$ui, args) + ) + + div( id = id, class = "teal_module", uiOutput(ns("data_reactive"), inline = TRUE), tagList( if (depth >= 2L) tags$div(style = "mt-6"), - do.call(modules$ui, args) + if (!is.null(modules$datanames)) { + fluidRow( + column(width = 9, ui_teal, class = "teal_primary_col"), + column( + width = 3, + ui_data_summary(ns("data_summary")), + ui_filter_data(ns("filter_panel")), + if (length(modules$transformers) > 0 && !isTRUE(attr(modules$transformers, "custom_ui"))) { + ui_transform_data(ns("data_transform"), transforms = modules$transformers, class = "well") + }, + class = "teal_secondary_col" + ) + ) + } else { + ui_teal + } ) ) - - if (!is.null(modules$datanames) && is_module_specific) { - fluidRow( - column(width = 9, teal_ui, class = "teal_primary_col"), - column( - width = 3, - datasets$ui_filter_panel(ns("module_filter_panel")), - class = "teal_secondary_col" - ) - ) - } else { - teal_ui - } } -#' @rdname module_nested_tabs -srv_nested_tabs <- function(id, datasets, modules, is_module_specific = FALSE, - reporter = teal.reporter::Reporter$new()) { +#' @rdname module_teal_module +srv_teal_module <- function(id, + data_rv, + modules, + datasets = NULL, + slices_global, + reporter = teal.reporter::Reporter$new(), + is_active = reactive(TRUE)) { + checkmate::assert_string(id) + checkmate::assert_class(data_rv, "reactive") checkmate::assert_multi_class(modules, c("teal_modules", "teal_module")) + checkmate::assert_class(datasets, "reactive", null.ok = TRUE) + checkmate::assert_class(slices_global, ".slicesGlobal") checkmate::assert_class(reporter, "Reporter") - UseMethod("srv_nested_tabs", modules) + UseMethod("srv_teal_module", modules) } -#' @rdname module_nested_tabs +#' @rdname module_teal_module #' @export -srv_nested_tabs.default <- function(id, datasets, modules, is_module_specific = FALSE, - reporter = teal.reporter::Reporter$new()) { +srv_teal_module.default <- function(id, + data_rv, + modules, + datasets = NULL, + slices_global, + reporter = teal.reporter::Reporter$new(), + is_active = reactive(TRUE)) { stop("Modules class not supported: ", paste(class(modules), collapse = " ")) } -#' @rdname module_nested_tabs +#' @rdname module_teal_module #' @export -srv_nested_tabs.teal_modules <- function(id, datasets, modules, is_module_specific = FALSE, - reporter = teal.reporter::Reporter$new()) { - checkmate::assert_list(datasets, types = c("list", "FilteredData")) - +srv_teal_module.teal_modules <- function(id, + data_rv, + modules, + datasets = NULL, + slices_global, + reporter = teal.reporter::Reporter$new(), + is_active = reactive(TRUE)) { moduleServer(id = id, module = function(input, output, session) { - logger::log_trace("srv_nested_tabs.teal_modules initializing the module { deparse1(modules$label) }.") + logger::log_debug("srv_teal_module.teal_modules initializing the module { deparse1(modules$label) }.") - labels <- vapply(modules$children, `[[`, character(1), "label") - modules_reactive <- sapply( + modules_output <- sapply( names(modules$children), function(module_id) { - srv_nested_tabs( + srv_teal_module( id = module_id, - datasets = datasets[[labels[module_id]]], + data_rv = data_rv, modules = modules$children[[module_id]], - is_module_specific = is_module_specific, - reporter = reporter + datasets = datasets, + slices_global = slices_global, + reporter = reporter, + is_active = reactive(is_active() && input$active_tab == module_id) ) }, simplify = FALSE ) - # when not ready input$active_tab would return NULL - this would fail next reactive - input_validated <- eventReactive(input$active_tab, input$active_tab, ignoreNULL = TRUE) - get_active_module <- reactive({ - if (length(modules$children) == 1L) { - # single tab is active by default - modules_reactive[[1]]() - } else { - # switch to active tab - modules_reactive[[input_validated()]]() - } - }) - - get_active_module + modules_output }) } -#' @rdname module_nested_tabs +#' @rdname module_teal_module #' @export -srv_nested_tabs.teal_module <- function(id, datasets, modules, is_module_specific = TRUE, - reporter = teal.reporter::Reporter$new()) { - checkmate::assert_class(datasets, "FilteredData") - logger::log_trace("srv_nested_tabs.teal_module initializing the module: { deparse1(modules$label) }.") - +srv_teal_module.teal_module <- function(id, + data_rv, + modules, + datasets = NULL, + slices_global, + reporter = teal.reporter::Reporter$new(), + 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) { - if (!is.null(modules$datanames) && is_module_specific) { - datasets$srv_filter_panel("module_filter_panel") - } + active_datanames <- reactive(.resolve_module_datanames(data = data_rv(), modules = modules)) + if (is.null(datasets)) { + datasets <- eventReactive(data_rv(), { + if (!inherits(data_rv(), "teal_data")) { + stop("data_rv must be teal_data object.") + } + logger::log_debug("srv_teal_module@1 initializing module-specific FilteredData") - # Create two triggers to limit reactivity between filter-panel and modules. - # We want to recalculate only visible modules - # - trigger the data when the tab is selected - # - trigger module to be called when the tab is selected for the first time - trigger_data <- reactiveVal(1L) - trigger_module <- reactiveVal(NULL) - output$data_reactive <- renderUI({ - lapply(datasets$datanames(), function(x) { - datasets$get_data(x, filtered = TRUE) + teal_data_to_filtered_data(data_rv(), datanames = active_datanames()) }) - isolate(trigger_data(trigger_data() + 1)) - isolate(trigger_module(TRUE)) - - NULL - }) - - # collect arguments to run teal_module - args <- c(list(id = "module"), modules$server_args) - if (is_arg_used(modules$server, "reporter")) { - args <- c(args, list(reporter = reporter)) } - if (is_arg_used(modules$server, "datasets")) { - args <- c(args, datasets = datasets) - } + # 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 + ) - if (is_arg_used(modules$server, "data")) { - data <- eventReactive(trigger_data(), .datasets_to_data(modules, datasets)) - args <- c(args, data = list(data)) - } + transformed_teal_data <- srv_transform_data( + "data_transform", + data = filtered_teal_data, + transforms = modules$transformers, + modules = modules + ) - if (is_arg_used(modules$server, "filter_panel_api")) { - filter_panel_api <- teal.slice::FilterPanelAPI$new(datasets) - args <- c(args, filter_panel_api = filter_panel_api) - } + module_teal_data <- reactive({ + 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) + }) - # This function calls a module server function. - call_module <- function() { - if (is_arg_used(modules$server, "id")) { - do.call(modules$server, args) - } else { - do.call(callModule, c(args, list(module = modules$server))) - } - } + module_teal_data_validated <- srv_validate_reactive_teal_data( + "validate_datanames", + data = module_teal_data, + modules = modules + ) + + summary_table <- srv_data_summary("data_summary", module_teal_data) # Call modules. - if (isTRUE(session$restoreContext$active)) { - # When restoring bookmark, all modules must be initialized on app start. - # Delayed module initiation (below) precludes restoring state b/c inputs do not exist when restoring occurs. - call_module() - } else if (inherits(modules, "teal_module_previewer")) { - # 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). - call_module() - } else { - # When app starts normally, modules are initialized only when corresponding tabs are clicked. - # Observing trigger_module() induces the module only when output$data_reactive is triggered (see above). - observeEvent( + 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 = trigger_module(), - handlerExpr = call_module() + eventExpr = module_teal_data_validated(), + handlerExpr = { + module_out(.call_teal_module(modules, datasets, module_teal_data_validated, 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)) } - reactive(modules) + # 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 }) } -#' Convert `FilteredData` to reactive list of datasets of the `teal_data` type. -#' -#' Converts `FilteredData` object to `teal_data` object containing datasets needed for a specific module. -#' Please note that if a module needs a dataset which has a parent, then the parent will also be returned. -#' A hash per `dataset` is calculated internally and returned in the code. -#' -#' @param module (`teal_module`) module where needed filters are taken from -#' @param datasets (`FilteredData`) object where needed data are taken from -#' -#' @return A `teal_data` object. -#' -#' @keywords internal -.datasets_to_data <- function(module, datasets) { - checkmate::assert_class(module, "teal_module") - checkmate::assert_class(datasets, "FilteredData") - - datanames <- if (is.null(module$datanames) || identical(module$datanames, "all")) { - datasets$datanames() - } else { - include_parent_datanames( - module$datanames, - datasets$get_join_keys() - ) +# This function calls a module server function. +.call_teal_module <- function(modules, datasets, filtered_teal_data, reporter) { + # collect arguments to run teal_module + args <- c(list(id = "module"), modules$server_args) + if (is_arg_used(modules$server, "reporter")) { + args <- c(args, list(reporter = reporter)) } - # list of reactive filtered data - data <- sapply(datanames, function(x) datasets$get_data(x, filtered = TRUE), simplify = FALSE) - - hashes <- calculate_hashes(datanames, datasets) - - code <- c( - get_rcode_str_install(), - get_rcode_libraries(), - get_datasets_code(datanames, datasets, hashes) - ) + if (is_arg_used(modules$server, "datasets")) { + args <- c(args, datasets = datasets()) + warning("datasets argument is not reactive and therefore it won't be updated when data is refreshed.") + } + if (is_arg_used(modules$server, "data")) { + args <- c(args, data = list(filtered_teal_data)) + } - data <- do.call( - teal.data::teal_data, - args = c(data, code = list(code), join_keys = list(datasets$get_join_keys()[datanames])) - ) + if (is_arg_used(modules$server, "filter_panel_api")) { + args <- c(args, filter_panel_api = teal.slice::FilterPanelAPI$new(datasets())) + } - data@verified <- attr(datasets, "verification_status") - data + if (is_arg_used(modules$server, "id")) { + do.call(modules$server, args) + } else { + do.call(callModule, c(args, list(module = modules$server))) + } } -#' Get the hash of a dataset -#' -#' @param datanames (`character`) names of datasets -#' @param datasets (`FilteredData`) object holding the data -#' -#' @return A list of hashes per dataset. -#' @keywords internal -#' -calculate_hashes <- function(datanames, datasets) { - sapply(datanames, function(x) rlang::hash(datasets$get_data(x, filtered = FALSE)), simplify = FALSE) +.resolve_module_datanames <- function(data, modules) { + stopifnot("data_rv must be teal_data object." = inherits(data, "teal_data")) + if (is.null(modules$datanames) || identical(modules$datanames, "all")) { + .teal_data_datanames(data) + } else { + intersect( + include_parent_datanames(modules$datanames, teal.data::join_keys(data)), + .teal_data_ls(data) + ) + } } diff --git a/R/module_snapshot_manager.R b/R/module_snapshot_manager.R index bbdfdea2f5..676ffbd9ff 100644 --- a/R/module_snapshot_manager.R +++ b/R/module_snapshot_manager.R @@ -7,7 +7,7 @@ #' as well as to save it to file in order to share it with an app developer or other users, #' who in turn can upload it to their own session. #' -#' The snapshot manager is accessed with the camera icon in the [`wunder_bar`]. +#' The snapshot manager is accessed with the camera icon in the tabset bar. #' At the beginning of a session it presents three icons: a camera, an upload, and an circular arrow. #' Clicking the camera captures a snapshot, clicking the upload adds a snapshot from a file #' and applies the filter states therein, and clicking the arrow resets initial application state. @@ -74,33 +74,59 @@ #' @param id (`character(1)`) `shiny` module instance id. #' @param slices_global (`reactiveVal`) that contains a `teal_slices` object #' containing all `teal_slice`s existing in the app, both active and inactive. -#' @param mapping_matrix (`reactive`) that contains a `data.frame` representation -#' of the mapping of filter state ids (rows) to modules labels (columns); -#' all columns are `logical` vectors. -#' @param datasets non-nested (named `list`) of `FilteredData` objects. #' #' @return `list` containing the snapshot history, where each element is an unlisted `teal_slices` object. #' #' @name module_snapshot_manager -#' @aliases snapshot snapshot_manager snapshot_manager_module +#' @rdname module_snapshot_manager #' #' @author Aleksander Chlebowski -#' +#' @keywords internal +NULL +#' @rdname module_snapshot_manager +ui_snapshot_manager_panel <- function(id) { + ns <- NS(id) + tags$button( + id = ns("show_snapshot_manager"), + class = "btn action-button wunder_bar_button", + title = "View filter mapping", + suppressMessages(icon("fas fa-camera")) + ) +} #' @rdname module_snapshot_manager -#' @keywords internal -#' -snapshot_manager_ui <- function(id) { +srv_snapshot_manager_panel <- function(id, slices_global) { + moduleServer(id, function(input, output, session) { + logger::log_debug("srv_snapshot_manager_panel initializing") + setBookmarkExclude(c("show_snapshot_manager")) + observeEvent(input$show_snapshot_manager, { + logger::log_debug("srv_snapshot_manager_panel@1 show_snapshot_manager button has been clicked.") + showModal( + modalDialog( + ui_snapshot_manager(session$ns("module")), + class = "snapshot_manager_modal", + size = "m", + footer = NULL, + easyClose = TRUE + ) + ) + }) + srv_snapshot_manager("module", slices_global = slices_global) + }) +} + +#' @rdname module_snapshot_manager +ui_snapshot_manager <- function(id) { ns <- NS(id) tags$div( class = "manager_content", tags$div( class = "manager_table_row", tags$span(tags$b("Snapshot manager")), - actionLink(ns("snapshot_add"), label = NULL, icon = icon("camera"), title = "add snapshot"), - actionLink(ns("snapshot_load"), label = NULL, icon = icon("upload"), title = "upload snapshot"), - actionLink(ns("snapshot_reset"), label = NULL, icon = icon("undo"), title = "reset initial state"), + actionLink(ns("snapshot_add"), label = NULL, icon = icon("fas fa-camera"), title = "add snapshot"), + actionLink(ns("snapshot_load"), label = NULL, icon = icon("fas fa-upload"), title = "upload snapshot"), + actionLink(ns("snapshot_reset"), label = NULL, icon = icon("fas fa-undo"), title = "reset initial state"), NULL ), uiOutput(ns("snapshot_list")) @@ -108,18 +134,11 @@ snapshot_manager_ui <- function(id) { } #' @rdname module_snapshot_manager -#' @keywords internal -#' -snapshot_manager_srv <- function(id, slices_global, mapping_matrix, datasets) { +srv_snapshot_manager <- function(id, slices_global) { checkmate::assert_character(id) - checkmate::assert_true(is.reactive(slices_global)) - checkmate::assert_class(isolate(slices_global()), "teal_slices") - checkmate::assert_true(is.reactive(mapping_matrix)) - checkmate::assert_data_frame(isolate(mapping_matrix()), null.ok = TRUE) - checkmate::assert_list(datasets, types = "FilteredData", any.missing = FALSE, names = "named") moduleServer(id, function(input, output, session) { - logger::log_trace("snapshot_manager_srv initializing") + logger::log_debug("srv_snapshot_manager initializing") # Set up bookmarking callbacks ---- # Register bookmark exclusions (all buttons and text fields). @@ -128,41 +147,33 @@ snapshot_manager_srv <- function(id, slices_global, mapping_matrix, datasets) { "snapshot_name_accept", "snaphot_file_accept", "snapshot_name", "snapshot_file" )) - # Add current filter state to bookmark. - # This is done on the app session because the value is restored in `module_teal` - # and we don't want to have to use this module's name space there. - app_session <- .subset2(shiny::getDefaultReactiveDomain(), "parent") - app_session$onBookmark(function(state) { - logger::log_trace("snapshot_manager_srv@onBookmark: storing filter state") - snapshot <- as.list(slices_global(), recursive = TRUE) - attr(snapshot, "mapping") <- matrix_to_mapping(mapping_matrix()) - state$values$filter_state_on_bookmark <- snapshot - }) # Add snapshot history to bookmark. session$onBookmark(function(state) { - logger::log_trace("snapshot_manager_srv@onBookmark: storing snapshot and bookmark history") + logger::log_debug("srv_snapshot_manager@onBookmark: storing snapshot and bookmark history") state$values$snapshot_history <- snapshot_history() # isolate this? }) ns <- session$ns # Track global filter states ---- - filter <- isolate(slices_global()) snapshot_history <- reactiveVal({ # Restore directly from bookmarked state, if applicable. - restoreValue(ns("snapshot_history"), list("Initial application state" = as.list(filter, recursive = TRUE))) + restoreValue( + ns("snapshot_history"), + list("Initial application state" = shiny::isolate(as.list(slices_global$all_slices(), recursive = TRUE))) + ) }) # Snapshot current application state ---- # Name snaphsot. observeEvent(input$snapshot_add, { - logger::log_trace("snapshot_manager_srv: snapshot_add button clicked") + logger::log_debug("srv_snapshot_manager: snapshot_add button clicked") showModal( modalDialog( textInput(ns("snapshot_name"), "Name the snapshot", width = "100%", placeholder = "Meaningful, unique name"), footer = tagList( - actionButton(ns("snapshot_name_accept"), "Accept", icon = icon("thumbs-up")), - modalButton(label = "Cancel", icon = icon("thumbs-down")) + actionButton(ns("snapshot_name_accept"), "Accept", icon = icon("far fa-thumbs-up")), + modalButton(label = "Cancel", icon = icon("far fa-thumbs-down")) ), size = "s" ) @@ -170,39 +181,38 @@ snapshot_manager_srv <- function(id, slices_global, mapping_matrix, datasets) { }) # Store snaphsot. observeEvent(input$snapshot_name_accept, { - logger::log_trace("snapshot_manager_srv: snapshot_name_accept button clicked") + logger::log_debug("srv_snapshot_manager: snapshot_name_accept button clicked") snapshot_name <- trimws(input$snapshot_name) if (identical(snapshot_name, "")) { - logger::log_trace("snapshot_manager_srv: snapshot name rejected") + logger::log_debug("srv_snapshot_manager: snapshot name rejected") showNotification( "Please name the snapshot.", type = "message" ) updateTextInput(inputId = "snapshot_name", value = "", placeholder = "Meaningful, unique name") } else if (is.element(make.names(snapshot_name), make.names(names(snapshot_history())))) { - logger::log_trace("snapshot_manager_srv: snapshot name rejected") + logger::log_debug("srv_snapshot_manager: snapshot name rejected") showNotification( "This name is in conflict with other snapshot names. Please choose a different one.", type = "message" ) updateTextInput(inputId = "snapshot_name", value = "", placeholder = "Meaningful, unique name") } else { - logger::log_trace("snapshot_manager_srv: snapshot name accepted, adding snapshot") - snapshot <- as.list(slices_global(), recursive = TRUE) - attr(snapshot, "mapping") <- matrix_to_mapping(mapping_matrix()) + logger::log_debug("srv_snapshot_manager: snapshot name accepted, adding snapshot") + snapshot <- as.list(slices_global$all_slices(), recursive = TRUE) snapshot_update <- c(snapshot_history(), list(snapshot)) names(snapshot_update)[length(snapshot_update)] <- snapshot_name snapshot_history(snapshot_update) removeModal() # Reopen filter manager modal by clicking button in the main application. - shinyjs::click(id = "teal-main_ui-wunder_bar-show_snapshot_manager", asis = TRUE) + shinyjs::click(id = "teal-wunder_bar-show_snapshot_manager", asis = TRUE) } }) # Upload a snapshot file ---- # Select file. observeEvent(input$snapshot_load, { - logger::log_trace("snapshot_manager_srv: snapshot_load button clicked") + logger::log_debug("srv_snapshot_manager: snapshot_load button clicked") showModal( modalDialog( fileInput(ns("snapshot_file"), "Choose snapshot file", accept = ".json", width = "100%"), @@ -213,22 +223,22 @@ snapshot_manager_srv <- function(id, slices_global, mapping_matrix, datasets) { placeholder = "Meaningful, unique name" ), footer = tagList( - actionButton(ns("snaphot_file_accept"), "Accept", icon = icon("thumbs-up")), - modalButton(label = "Cancel", icon = icon("thumbs-down")) + actionButton(ns("snaphot_file_accept"), "Accept", icon = icon("far fa-thumbs-up")), + modalButton(label = "Cancel", icon = icon("far fa-thumbs-down")) ) ) ) }) # Store new snapshot to list and restore filter states. observeEvent(input$snaphot_file_accept, { - logger::log_trace("snapshot_manager_srv: snapshot_file_accept button clicked") + logger::log_debug("srv_snapshot_manager: snapshot_file_accept button clicked") snapshot_name <- trimws(input$snapshot_name) if (identical(snapshot_name, "")) { - logger::log_trace("snapshot_manager_srv: no snapshot name provided, naming after file") + logger::log_debug("srv_snapshot_manager: no snapshot name provided, naming after file") snapshot_name <- tools::file_path_sans_ext(input$snapshot_file$name) } if (is.element(make.names(snapshot_name), make.names(names(snapshot_history())))) { - logger::log_trace("snapshot_manager_srv: snapshot name rejected") + logger::log_debug("srv_snapshot_manager: snapshot name rejected") showNotification( "This name is in conflict with other snapshot names. Please choose a different one.", type = "message" @@ -236,40 +246,30 @@ snapshot_manager_srv <- function(id, slices_global, mapping_matrix, datasets) { updateTextInput(inputId = "snapshot_name", value = "", placeholder = "Meaningful, unique name") } else { # Restore snapshot and verify app compatibility. - logger::log_trace("snapshot_manager_srv: snapshot name accepted, loading snapshot") + logger::log_debug("srv_snapshot_manager: snapshot name accepted, loading snapshot") snapshot_state <- try(slices_restore(input$snapshot_file$datapath)) if (!inherits(snapshot_state, "modules_teal_slices")) { - logger::log_trace("snapshot_manager_srv: snapshot file corrupt") + logger::log_debug("srv_snapshot_manager: snapshot file corrupt") showNotification( "File appears to be corrupt.", type = "error" ) - } else if (!identical(attr(snapshot_state, "app_id"), attr(slices_global(), "app_id"))) { - logger::log_trace("snapshot_manager_srv: snapshot not compatible with app") + } else if (!identical(attr(snapshot_state, "app_id"), attr(slices_global$all_slices(), "app_id"))) { + logger::log_debug("srv_snapshot_manager: snapshot not compatible with app") showNotification( "This snapshot file is not compatible with the app and cannot be loaded.", type = "warning" ) } else { # Add to snapshot history. - logger::log_trace("snapshot_manager_srv: snapshot loaded, adding to history") - snapshot <- as.list(snapshot_state, recursive = TRUE) + logger::log_debug("srv_snapshot_manager: snapshot loaded, adding to history") + snapshot <- as.list(slices_global$all_slices(), recursive = TRUE) snapshot_update <- c(snapshot_history(), list(snapshot)) names(snapshot_update)[length(snapshot_update)] <- snapshot_name snapshot_history(snapshot_update) ### Begin simplified restore procedure. ### - logger::log_trace("snapshot_manager_srv: restoring snapshot") - mapping_unfolded <- unfold_mapping(attr(snapshot_state, "mapping"), names(datasets)) - mapply( - function(filtered_data, filter_ids) { - filtered_data$clear_filter_states(force = TRUE) - slices <- Filter(function(x) x$id %in% filter_ids, snapshot_state) - filtered_data$set_filter_state(slices) - }, - filtered_data = datasets, - filter_ids = mapping_unfolded - ) - slices_global(snapshot_state) + logger::log_debug("srv_snapshot_manager: restoring snapshot") + slices_global$slices_set(snapshot_state) removeModal() ### End simplified restore procedure. ### } @@ -279,22 +279,13 @@ snapshot_manager_srv <- function(id, slices_global, mapping_matrix, datasets) { # Restore initial state ---- observeEvent(input$snapshot_reset, { - logger::log_trace("snapshot_manager_srv: snapshot_reset button clicked, restoring snapshot") + logger::log_debug("srv_snapshot_manager: snapshot_reset button clicked, restoring snapshot") s <- "Initial application state" ### Begin restore procedure. ### snapshot <- snapshot_history()[[s]] + # todo: as.teal_slices looses module-mapping if is not global snapshot_state <- as.teal_slices(snapshot) - mapping_unfolded <- unfold_mapping(attr(snapshot_state, "mapping"), names(datasets)) - mapply( - function(filtered_data, filter_ids) { - filtered_data$clear_filter_states(force = TRUE) - slices <- Filter(function(x) x$id %in% filter_ids, snapshot_state) - filtered_data$set_filter_state(slices) - }, - filtered_data = datasets, - filter_ids = mapping_unfolded - ) - slices_global(snapshot_state) + slices_global$slices_set(snapshot_state) removeModal() ### End restore procedure. ### }) @@ -308,7 +299,7 @@ snapshot_manager_srv <- function(id, slices_global, mapping_matrix, datasets) { divs <- reactiveValues() observeEvent(snapshot_history(), { - logger::log_trace("snapshot_manager_srv: snapshot history modified, updating snapshot list") + logger::log_debug("srv_snapshot_manager: snapshot history modified, updating snapshot list") lapply(names(snapshot_history())[-1L], function(s) { id_pickme <- sprintf("pickme_%s", make.names(s)) id_saveme <- sprintf("saveme_%s", make.names(s)) @@ -320,17 +311,8 @@ snapshot_manager_srv <- function(id, slices_global, mapping_matrix, datasets) { ### Begin restore procedure. ### snapshot <- snapshot_history()[[s]] snapshot_state <- as.teal_slices(snapshot) - mapping_unfolded <- unfold_mapping(attr(snapshot_state, "mapping"), names(datasets)) - mapply( - function(filtered_data, filter_ids) { - filtered_data$clear_filter_states(force = TRUE) - slices <- Filter(function(x) x$id %in% filter_ids, snapshot_state) - filtered_data$set_filter_state(slices) - }, - filtered_data = datasets, - filter_ids = mapping_unfolded - ) - slices_global(snapshot_state) + + slices_global$slices_set(snapshot_state) removeModal() ### End restore procedure. ### }) @@ -354,8 +336,8 @@ snapshot_manager_srv <- function(id, slices_global, mapping_matrix, datasets) { divs[[id_rowme]] <- tags$div( class = "manager_table_row", tags$span(tags$h5(s)), - actionLink(inputId = ns(id_pickme), label = icon("circle-check"), title = "select"), - downloadLink(outputId = ns(id_saveme), label = icon("save"), title = "save to file") + actionLink(inputId = ns(id_pickme), label = icon("far fa-circle-check"), title = "select"), + downloadLink(outputId = ns(id_saveme), label = icon("far fa-save"), title = "save to file") ) } }) @@ -377,43 +359,3 @@ snapshot_manager_srv <- function(id, slices_global, mapping_matrix, datasets) { snapshot_history }) } - -### utility functions ---- - -#' Explicitly enumerate global filters. -#' -#' Transform module mapping such that global filters are explicitly specified for every module. -#' -#' @param mapping (named `list`) as stored in mapping parameter of `teal_slices` -#' @param module_names (`character`) vector containing names of all modules in the app -#' @return A `named_list` with one element per module, each element containing all filters applied to that module. -#' -#' @keywords internal -#' -unfold_mapping <- function(mapping, module_names) { - module_names <- structure(module_names, names = module_names) - lapply(module_names, function(x) c(mapping[[x]], mapping[["global_filters"]])) -} - -#' Convert mapping matrix to filter mapping specification. -#' -#' Transform a mapping matrix, i.e. a data frame that maps each filter state to each module, -#' to a list specification like the one used in the `mapping` attribute of `teal_slices`. -#' Global filters are gathered in one list element. -#' If a module has no active filters but the global ones, it will not be mentioned in the output. -#' -#' @param mapping_matrix (`data.frame`) of logical vectors where -#' columns represent modules and row represent `teal_slice`s -#' @return Named `list` like that in the `mapping` attribute of a `teal_slices` object. -#' -#' @keywords internal -#' -matrix_to_mapping <- function(mapping_matrix) { - mapping_matrix[] <- lapply(mapping_matrix, function(x) x | is.na(x)) - global <- vapply(as.data.frame(t(mapping_matrix)), all, logical(1L)) - global_filters <- names(global[global]) - local_filters <- mapping_matrix[!rownames(mapping_matrix) %in% global_filters, ] - - mapping <- c(lapply(local_filters, function(x) rownames(local_filters)[x]), list(global_filters = global_filters)) - Filter(function(x) length(x) != 0L, mapping) -} diff --git a/R/module_tabs_with_filters.R b/R/module_tabs_with_filters.R deleted file mode 100644 index 2807253afb..0000000000 --- a/R/module_tabs_with_filters.R +++ /dev/null @@ -1,138 +0,0 @@ -#' Add right filter panel into each of the top-level `teal_modules` UIs -#' -#' The [ui_nested_tabs] function returns a nested tabbed UI corresponding -#' to the nested modules. -#' This function adds the right filter panel to each main tab. -#' -#' The right filter panel's filter choices affect the `datasets` object. Therefore, -#' all modules using the same `datasets` share the same filters. -#' -#' This works with nested modules of depth greater than 2, though the filter -#' panel is inserted at the right of the modules at depth 1 and not at the leaves. -#' -#' @name module_tabs_with_filters -#' -#' @inheritParams module_teal -#' -#' @param datasets (named `list` of `FilteredData`) -#' object to store filter state and filtered datasets, shared across modules. For more -#' details see [`teal.slice::FilteredData`]. Structure of the list must be the same as structure -#' of the `modules` argument and list names must correspond to the labels in `modules`. -#' When filter is not module-specific then list contains the same object in all elements. -#' @param reporter (`Reporter`) object from `teal.reporter` -#' @param progress (`Progress`) object from `shiny` -#' -#' @return -#' A `shiny.tag.list` containing the main menu, placeholders for filters and placeholders for the `teal` modules. -#' -#' @keywords internal -#' -NULL - -#' @rdname module_tabs_with_filters -ui_tabs_with_filters <- function(id, modules, datasets, filter = teal_slices(), progress = NULL) { - checkmate::assert_class(modules, "teal_modules") - checkmate::assert_list(datasets, types = c("list", "FilteredData")) - checkmate::assert_class(filter, "teal_slices") - checkmate::assert_r6(progress, "Progress", null.ok = TRUE) - - ns <- NS(id) - is_module_specific <- isTRUE(attr(filter, "module_specific")) - - teal_ui <- ui_nested_tabs(ns("root"), modules = modules, datasets, is_module_specific = is_module_specific, progress = progress) # nolint: line_length. - filter_panel_btns <- tags$li( - class = "flex-grow", - tags$button( - class = "btn action-button filter_hamburger", # see sidebar.css for style filter_hamburger - href = "javascript:void(0)", - onclick = "toggleFilterPanel();", # see sidebar.js - title = "Toggle filter panel", - icon("fas fa-bars") - ), - wunder_bar_ui(ns("wunder_bar")) - ) - teal_ui$children[[1]] <- tagAppendChild(teal_ui$children[[1]], filter_panel_btns) - - if (!is.null(progress)) { - progress$set(message = "Preparing main UI", detail = "") - } - - if (!is_module_specific) { - # need to rearrange html so that filter panel is within tabset - tabset_bar <- teal_ui$children[[1]] - teal_modules <- teal_ui$children[[2]] - filter_ui <- unlist(datasets)[[1]]$ui_filter_panel(ns("filter_panel")) - list( - tabset_bar, - tags$hr(class = "my-2"), - fluidRow( - column(width = 9, teal_modules, class = "teal_primary_col"), - column(width = 3, filter_ui, class = "teal_secondary_col") - ) - ) - } else { - teal_ui - } -} - -#' @rdname module_tabs_with_filters -srv_tabs_with_filters <- function(id, - datasets, - modules, - reporter = teal.reporter::Reporter$new(), - filter = teal_slices()) { - checkmate::assert_class(modules, "teal_modules") - checkmate::assert_list(datasets, types = c("list", "FilteredData")) - checkmate::assert_class(reporter, "Reporter") - checkmate::assert_class(filter, "teal_slices") - - moduleServer(id, function(input, output, session) { - logger::log_trace("srv_tabs_with_filters initializing the module.") - - is_module_specific <- isTRUE(attr(filter, "module_specific")) - wunder_bar_out <- wunder_bar_srv("wunder_bar", datasets, filter, modules) - - active_module <- srv_nested_tabs( - id = "root", - datasets = datasets, - modules = modules, - reporter = reporter, - is_module_specific = is_module_specific - ) - - if (!is_module_specific) { - active_datanames <- reactive({ - if (identical(active_module()$datanames, "all")) { - singleton$datanames() - } else { - include_parent_datanames( - active_module()$datanames, - singleton$get_join_keys() - ) - } - }) - singleton <- unlist(datasets)[[1]] - singleton$srv_filter_panel("filter_panel", active_datanames = active_datanames) - - observeEvent( - eventExpr = active_datanames(), - handlerExpr = { - script <- if (length(active_datanames()) == 0 || is.null(active_datanames())) { - # hide the filter panel and disable the burger button - "handleNoActiveDatasets();" - } else { - # show the filter panel and enable the burger button - "handleActiveDatasetsPresent();" - } - shinyjs::runjs(script) - }, - ignoreNULL = FALSE - ) - } - - showNotification("Data loaded - App fully started up") - logger::log_trace("srv_tabs_with_filters initialized the module") - - active_module - }) -} diff --git a/R/module_teal.R b/R/module_teal.R index 43e0ccc37e..4f1ead538b 100644 --- a/R/module_teal.R +++ b/R/module_teal.R @@ -1,55 +1,62 @@ -# This module is the main teal module that puts everything together. - -#' `teal` main app module -#' -#' This is the main `teal` app that puts everything together. +#' `teal` main module #' -#' It displays the splash UI which is used to fetch the data, possibly -#' prompting for a password input to fetch the data. Once the data is ready, -#' the splash screen is replaced by the actual `teal` UI that is tabsetted and -#' has a filter panel with `datanames` that are relevant for the current tab. -#' Nested tabs are possible, but we limit it to two nesting levels for reasons -#' of clarity of the UI. +#' @description +#' `r lifecycle::badge("stable")` +#' Module to create a `teal` app. This module can be called directly instead of [init()] and +#' included in your custom application. Please note that [init()] adds `reporter_previewer_module` +#' automatically, which is not a case when calling `ui/srv_teal` directly. #' -#' The splash screen functionality can also be used -#' for non-delayed data which takes time to load into memory, avoiding -#' `shiny` session timeouts. +#' @details #' -#' Server evaluates the `teal_data_rv` (delayed data mechanism) and creates the -#' `datasets` object that is shared across modules. -#' Once it is ready and non-`NULL`, the splash screen is replaced by the -#' main `teal` UI that depends on the data. -#' The currently active tab is tracked and the right filter panel -#' updates the displayed datasets to filter for according to the active `datanames` -#' of the tab. +#' Module is responsible for creating the main `shiny` app layout and initializing all the necessary +#' components. This module establishes reactive connection between the input `data` and every other +#' component in the app. Reactive change of the `data` passed as an argument, reloads the app and +#' possibly keeps all input settings the same so the user can continue where one left off. #' -#' @name module_teal +#' ## data flow in `teal` application #' -#' @inheritParams module_teal_with_splash +#' This module supports multiple data inputs but eventually, they are all converted to `reactive` +#' returning `teal_data` in this module. On this `reactive teal_data` object several actions are +#' performed: +#' - data loading in [`module_init_data`] +#' - data filtering in [`module_filter_data`] +#' - data transformation in [`module_transform_data`] #' -#' @param splash_ui (`shiny.tag`) UI to display initially, -#' can be a splash screen or a `shiny` module UI. For the latter, see -#' [init()] about how to call the corresponding server function. +#' @rdname module_teal +#' @name module_teal #' -#' @param teal_data_rv (`reactive`) -#' returns the `teal_data`, only evaluated once, `NULL` value is ignored +#' @inheritParams module_init_data +#' @inheritParams init #' #' @return -#' Returns a `reactive` expression which returns the currently active module. -#' -#' @keywords internal -#' +#' `NULL` invisibly NULL #' @rdname module_teal +#' @export ui_teal <- function(id, - splash_ui = tags$h2("Starting the Teal App"), + modules, + data = NULL, title = build_app_title(), header = tags$p(), footer = tags$p()) { checkmate::assert_character(id, max.len = 1, any.missing = FALSE) - - checkmate::assert_multi_class(splash_ui, c("shiny.tag", "shiny.tag.list", "html")) + checkmate::assert_multi_class(data, "teal_data_module", null.ok = TRUE) + checkmate::assert( + .var.name = "title", + checkmate::check_string(title), + checkmate::check_multi_class(title, c("shiny.tag", "shiny.tag.list", "html")) + ) + checkmate::assert( + .var.name = "header", + checkmate::check_string(header), + checkmate::check_multi_class(header, c("shiny.tag", "shiny.tag.list", "html")) + ) + checkmate::assert( + .var.name = "footer", + checkmate::check_string(footer), + checkmate::check_multi_class(footer, c("shiny.tag", "shiny.tag.list", "html")) + ) if (is.character(title)) { title <- build_app_title(title) @@ -57,55 +64,75 @@ ui_teal <- function(id, validate_app_title_tag(title) } - checkmate::assert( - .var.name = "header", - checkmate::check_string(header), - checkmate::check_multi_class(header, c("shiny.tag", "shiny.tag.list", "html")) - ) if (checkmate::test_string(header)) { header <- tags$p(header) } - checkmate::assert( - .var.name = "footer", - checkmate::check_string(footer), - checkmate::check_multi_class(footer, c("shiny.tag", "shiny.tag.list", "html")) - ) if (checkmate::test_string(footer)) { footer <- tags$p(footer) } ns <- NS(id) - # Once the data is loaded, we will remove this element and add the real teal UI instead - splash_ui <- tags$div( - # id so we can remove the splash screen once ready, which is the first child of this container - id = ns("main_ui_container"), - # we put it into a div, so it can easily be removed as a whole, also when it is a tagList (and not - # just the first item of the tagList) - tags$div(splash_ui) - ) - # show busy icon when `shiny` session is busy computing stuff # based on https://stackoverflow.com/questions/17325521/r-shiny-display-loading-message-while-function-is-running/22475216#22475216 # nolint: line_length. shiny_busy_message_panel <- conditionalPanel( condition = "(($('html').hasClass('shiny-busy')) && (document.getElementById('shiny-notification-panel') == null))", # nolint: line_length. tags$div( - icon("arrows-rotate", "spin fa-spin"), + icon("arrows-rotate", class = "fa-spin", prefer_type = "solid"), "Computing ...", # CSS defined in `custom.css` class = "shinybusymessage" ) ) + bookmark_panel_ui <- ui_bookmark_panel(ns("bookmark_manager"), modules) + data_elem <- ui_init_data(ns("data"), data = data) + if (!is.null(data)) { + modules$children <- c(list(teal_data_module = data_elem), modules$children) + } + tabs_elem <- ui_teal_module(id = ns("teal_modules"), modules = modules) + fluidPage( + id = id, title = title, theme = get_teal_bs_theme(), include_teal_css_js(), tags$header(header), tags$hr(class = "my-2"), shiny_busy_message_panel, - splash_ui, + tags$div( + id = ns("tabpanel_wrapper"), + class = "teal-body", + tabs_elem + ), + tags$div( + id = ns("options_buttons"), + style = "margin-left: auto;", + bookmark_panel_ui, + tags$button( + class = "btn action-button filter_hamburger", # see sidebar.css for style filter_hamburger + href = "javascript:void(0)", + onclick = sprintf("toggleFilterPanel('%s');", ns("tabpanel_wrapper")), + title = "Toggle filter panel", + icon("fas fa-bars") + ), + ui_snapshot_manager_panel(ns("snapshot_manager_panel")), + ui_filter_manager_panel(ns("filter_manager_panel")) + ), + tags$script( + HTML( + sprintf( + " + $(document).ready(function() { + $('#%s').appendTo('#%s'); + }); + ", + ns("options_buttons"), + ns("teal_modules-active_tab") + ) + ) + ), tags$hr(), tags$footer( tags$div( @@ -119,12 +146,16 @@ ui_teal <- function(id, ) } - #' @rdname module_teal -srv_teal <- function(id, modules, teal_data_rv, filter = teal_slices()) { - stopifnot(is.reactive(teal_data_rv)) +#' @export +srv_teal <- function(id, data, modules, filter = teal_slices()) { + checkmate::assert_character(id, max.len = 1, any.missing = FALSE) + checkmate::assert_multi_class(data, c("teal_data", "teal_data_module", "reactive", "reactiveVal")) + checkmate::assert_class(modules, "teal_modules") + checkmate::assert_class(filter, "teal_slices") + moduleServer(id, function(input, output, session) { - logger::log_trace("srv_teal initializing the module.") + logger::log_debug("srv_teal initializing.") output$identifier <- renderText( paste0("Pid:", Sys.getpid(), " Token:", substr(session$token, 25, 32)) @@ -150,81 +181,38 @@ srv_teal <- function(id, modules, teal_data_rv, filter = teal_slices()) { once = TRUE, handlerExpr = { session$userData$timezone <- input$timezone - logger::log_trace("srv_teal@1 Timezone set to client's timezone: { input$timezone }.") + logger::log_debug("srv_teal@1 Timezone set to client's timezone: { input$timezone }.") } ) - reporter <- teal.reporter::Reporter$new()$set_id(attr(filter, "app_id")) - if (is_arg_used(modules, "reporter") && length(extract_module(modules, "teal_module_previewer")) == 0) { - modules <- append_module( - modules, - reporter_previewer_module(server_args = list(previewer_buttons = c("download", "reset"))) - ) + data_rv <- srv_init_data("data", data = data, modules = modules, filter = filter) + datasets_rv <- if (!isTRUE(attr(filter, "module_specific"))) { + eventReactive(data_rv(), { + if (!inherits(data_rv(), "teal_data")) { + stop("data_rv must be teal_data object.") + } + logger::log_debug("srv_teal@1 initializing FilteredData") + teal_data_to_filtered_data(data_rv()) + }) } + module_labels <- unlist(module_labels(modules), use.names = FALSE) + slices_global <- methods::new(".slicesGlobal", filter, module_labels) + modules_output <- srv_teal_module( + id = "teal_modules", + data_rv = data_rv, + datasets = datasets_rv, + modules = modules, + slices_global = slices_global + ) + mapping_table <- srv_filter_manager_panel("filter_manager_panel", slices_global = slices_global) + snapshots <- srv_snapshot_manager_panel("snapshot_manager_panel", slices_global = slices_global) + srv_bookmark_panel("bookmark_manager", modules) - datasets_reactive <- eventReactive(teal_data_rv(), { - progress_data <- Progress$new( - max = length(unlist(module_labels(modules))) - ) - on.exit(progress_data$close()) - progress_data$set(message = "Preparing data filtering", detail = "0%") - # Restore filter from bookmarked state, if applicable. - filter_restored <- restoreValue("filter_state_on_bookmark", filter) - if (!is.teal_slices(filter_restored)) { - filter_restored <- as.teal_slices(filter_restored) - } - # Create list of `FilteredData` objects that reflects structure of `modules`. - modules_datasets(teal_data_rv(), modules, filter_restored, teal_data_to_filtered_data(teal_data_rv()), progress_data) # nolint: line_length. - }) - - - # Replace splash / welcome screen once data is loaded ---- - # ignoreNULL to not trigger at the beginning when data is NULL - # just handle it once because data obtained through delayed loading should - # usually not change afterwards - # if restored from bookmarked state, `filter` is ignored - - observeEvent(datasets_reactive(), once = TRUE, { - logger::log_trace("srv_teal@5 setting main ui after data was pulled") - datasets <- datasets_reactive() - - progress_modules <- Progress$new( - max = length(unlist(module_labels(modules))) - ) - on.exit(progress_modules$close()) - progress_modules$set(value = 0, message = "Preparing modules", detail = "0%") - - # main_ui_container contains splash screen first and we remove it and replace it by the real UI - removeUI(sprintf("#%s > div:nth-child(1)", session$ns("main_ui_container"))) - insertUI( - selector = paste0("#", session$ns("main_ui_container")), - where = "beforeEnd", - # we put it into a div, so it can easily be removed as a whole, also when it is a tagList (and not - # just the first item of the tagList) - ui = tags$div(ui_tabs_with_filters( - session$ns("main_ui"), - modules = modules, - datasets = datasets, - filter = filter, - progress = progress_modules - )), - # needed so that the UI inputs are available and can be immediately updated, otherwise, updating may not - # have any effect as they are ignored when not present - immediate = TRUE - ) - - progress_modules$set(message = "Finalizing") - - # must make sure that this is only executed once as modules assume their observers are only - # registered once (calling server functions twice would trigger observers twice each time) - srv_tabs_with_filters( - id = "main_ui", - datasets = datasets, - modules = modules, - reporter = reporter, - filter = filter - ) - }) + if (inherits(data, "teal_data_module")) { + setBookmarkExclude(c("teal_modules-active_tab")) + } }) + + invisible(NULL) } diff --git a/R/module_teal_data.R b/R/module_teal_data.R new file mode 100644 index 0000000000..f2bbaf4145 --- /dev/null +++ b/R/module_teal_data.R @@ -0,0 +1,208 @@ +#' Execute and validate `teal_data_module` +#' +#' This is a low level module to handle `teal_data_module` execution and validation. +#' [teal_transform_module()] inherits from [teal_data_module()] so it is handled by this module too. +#' [srv_teal()] accepts various `data` objects and eventually they are all transformed to `reactive` +#' [teal_data()] which is a standard data class in whole `teal` framework. +#' +#' @section data validation: +#' +#' Executed [teal_data_module()] is validated and output is validated for consistency. +#' Output `data` is invalid if: +#' 1. [teal_data_module()] is invalid if server doesn't return `reactive`. **Immediately crashes an app!** +#' 2. `reactive` throws a `shiny.error` - happens when module creating [teal_data()] fails. +#' 3. `reactive` returns `qenv.error` - happens when [teal_data()] evaluates a failing code. +#' 4. `reactive` object doesn't return [teal_data()]. +#' 5. [teal_data()] object lacks any `datanames` specified in the `modules` argument. +#' +#' `teal` (observers in `srv_teal`) always waits to render an app until `reactive` `teal_data` is +#' returned. If error 2-4 occurs, relevant error message is displayed to app user and after issue is +#' resolved app will continue to run. `teal` guarantees that errors in a data don't crash an app +#' (except error 1). This is possible thanks to `.fallback_on_failure` which returns input-data +#' when output-data fails +#' +#' +#' @param id (`character(1)`) Module id +#' @param data (`reactive teal_data`) +#' @param data_module (`teal_data_module`) +#' @param modules (`teal_modules` or `teal_module`) For `datanames` validation purpose +#' @param validate_shiny_silent_error (`logical`) If `TRUE`, then `shiny.silent.error` is validated and +#' error message is displayed. +#' Default is `FALSE` to handle empty reactive cycle on `init`. +#' +#' @return `reactive` `teal_data` +#' +#' @rdname module_teal_data +#' @name module_teal_data +#' @keywords internal +NULL + +#' @rdname module_teal_data +ui_teal_data <- function(id, data_module) { + checkmate::assert_string(id) + checkmate::assert_class(data_module, "teal_data_module") + ns <- NS(id) + shiny::tagList( + data_module$ui(id = ns("data")), + ui_validate_reactive_teal_data(ns("validate")) + ) +} + +#' @rdname module_teal_data +srv_teal_data <- function(id, + data, + data_module, + modules = NULL, + validate_shiny_silent_error = TRUE) { + checkmate::assert_string(id) + checkmate::assert_class(data, "reactive") + checkmate::assert_class(data_module, "teal_data_module") + checkmate::assert_multi_class(modules, c("teal_modules", "teal_module"), null.ok = TRUE) + + moduleServer(id, function(input, output, session) { + logger::log_debug("srv_teal_data initializing.") + + data_out <- if (is_arg_used(data_module$server, "data")) { + data_module$server(id = "data", data = data) + } else { + data_module$server(id = "data") + } + + data_validated <- srv_validate_reactive_teal_data( + id = "validate", + data = data_out, + modules = modules, + validate_shiny_silent_error = validate_shiny_silent_error + ) + + .fallback_on_failure( + this = data_validated, + that = data, + label = sprintf("Data element '%s' for module '%s'", id, modules$label) + ) + }) +} + +#' @rdname module_teal_data +ui_validate_reactive_teal_data <- function(id) { + tagList( + uiOutput(NS(id, "shiny_errors")), + uiOutput(NS(id, "shiny_warnings")) + ) +} + +#' @rdname module_teal_data +srv_validate_reactive_teal_data <- function(id, # nolint: object_length + data, + modules = NULL, + validate_shiny_silent_error = FALSE) { + moduleServer(id, function(input, output, session) { + if (!is.reactive(data)) { + stop("The `teal_data_module` passed to `data` must return a reactive expression.", call. = FALSE) + } + + data_out_rv <- reactive(tryCatch(data(), error = function(e) e)) + + data_validated <- reactive({ + # custom module can return error + data_out <- data_out_rv() + + # there is an empty reactive cycle on init! + if (inherits(data_out, "shiny.silent.error") && identical(data_out$message, "")) { + if (!validate_shiny_silent_error) { + return(NULL) + } else { + validate( + need( + FALSE, + paste( + strip_style(data_out$message), + "Check your inputs or contact app developer if error persists.", + sep = ifelse(identical(data_out$message, ""), "", "\n") + ) + ) + ) + } + } + + # to handle errors and qenv.error(s) + if (inherits(data_out, c("qenv.error", "error"))) { + validate( + need( + FALSE, + paste( + "Error when executing `teal_data_module` passed to `data`:\n ", + strip_style(paste(data_out$message, collapse = "\n")), + "\n Check your inputs or contact app developer if error persists." + ) + ) + ) + } + + validate( + need( + inherits(data_out, "teal_data"), + paste( + "Error: `teal_data_module` passed to `data` failed to return `teal_data` object, returned", + strip_style(toString(sQuote(class(data_out)))), + "instead.", + "\n Check your inputs or contact app developer if error persists." + ) + ) + ) + + data_out + }) + + output$shiny_errors <- renderUI({ + data_validated() + NULL + }) + + output$shiny_warnings <- renderUI({ + if (inherits(data_out_rv(), "teal_data")) { + is_modules_ok <- check_modules_datanames(modules = modules, datanames = .teal_data_ls(data_validated())) + if (!isTRUE(is_modules_ok)) { + tags$div( + is_modules_ok$html( + # Show modules prefix on message only in teal_data_module tab + grepl(sprintf("data-teal_data_module-%s", id), session$ns(NULL), fixed = TRUE) + ), + class = "teal-output-warning" + ) + } + } + }) + + data_validated + }) +} + +#' Fallback on failure +#' +#' Function returns the previous reactive if the current reactive is invalid (throws error or returns NULL). +#' Application: In `teal` we try to prevent the error from being thrown and instead we replace failing +#' transform module data output with data input from the previous module (or from previous `teal` reactive +#' tree elements). +#' +#' @param this (`reactive`) Current reactive. +#' @param that (`reactive`) Previous reactive. +#' @param label (`character`) Label for identifying problematic `teal_data_module` transform in logging. +#' @return `reactive` `teal_data` +#' @keywords internal +.fallback_on_failure <- function(this, that, label) { + checkmate::assert_class(this, "reactive") + checkmate::assert_class(that, "reactive") + checkmate::assert_string(label) + + reactive({ + res <- try(this(), silent = TRUE) + if (inherits(res, "teal_data")) { + logger::log_debug("{ label } evaluated successfully.") + res + } else { + logger::log_debug("{ label } failed, falling back to previous data.") + that() + } + }) +} diff --git a/R/module_teal_with_splash.R b/R/module_teal_with_splash.R index e91be8ba29..c7404e7920 100644 --- a/R/module_teal_with_splash.R +++ b/R/module_teal_with_splash.R @@ -1,58 +1,14 @@ -# This file adds a splash screen for delayed data loading on top of teal - -#' Add splash screen to `teal` application -#' -#' @description `r lifecycle::badge("stable")` -#' -#' Displays custom splash screen during initial delayed data loading. -#' -#' @details -#' This module pauses app initialization pending delayed data loading. -#' This is necessary because the filter panel and modules depend on the data to initialize. +#' UI and server modules of `teal` #' -#' `teal_with_splash` follows the `shiny` module convention. -#' [`init()`] is a wrapper around this that assumes that `teal` is -#' the top-level module and cannot be embedded. +#' @description `r lifecycle::badge("deprecated")` +#' Please use [`module_teal`] instead. #' -#' Note: It is no longer recommended to embed `teal` in `shiny` apps as a module. -#' but rather use `init` to create a standalone application. -#' -#' @seealso [init()] -#' -#' @param id (`character(1)`) -#' module id -#' @inheritParams init -#' @param modules (`teal_modules`) object containing the output modules which -#' will be displayed in the `teal` application. See [modules()] and [module()] for -#' more details. -#' @inheritParams shiny::moduleServer -#' -#' @section Reproducibility: -#' Reproducibility is supported by multiple features. `teal` includes a `utils::sessioInfo()` output to allow to compare -#' packages used in the session. It also allows to create `renv` lockfile to support project setup reproducibility. -#' For more information about lockfile creation visit [`teal_lockfile()`]. +#' @inheritParams ui_teal +#' @inheritParams srv_teal #' #' @return #' Returns a `reactive` expression containing a `teal_data` object when data is loaded or `NULL` when it is not. #' @name module_teal_with_splash -#' @examples -#' teal_modules <- modules(example_module()) -#' # Shiny app with modular integration of teal -#' ui <- fluidPage( -#' ui_teal_with_splash(id = "app1", data = teal_data()) -#' ) -#' -#' server <- function(input, output, session) { -#' srv_teal_with_splash( -#' id = "app1", -#' data = teal_data(iris = iris), -#' modules = teal_modules -#' ) -#' } -#' -#' if (interactive()) { -#' shinyApp(ui, server) -#' } #' NULL @@ -63,150 +19,21 @@ ui_teal_with_splash <- function(id, title = build_app_title(), header = tags$p(), footer = tags$p()) { - checkmate::assert_character(id, max.len = 1, any.missing = FALSE) - checkmate::assert_multi_class(data, c("teal_data", "teal_data_module")) - checkmate::assert( - .var.name = "title", - checkmate::check_string(title), - checkmate::check_multi_class(title, c("shiny.tag", "shiny.tag.list", "html")) - ) - checkmate::assert( - .var.name = "header", - checkmate::check_string(header), - checkmate::check_multi_class(header, c("shiny.tag", "shiny.tag.list", "html")) - ) - checkmate::assert( - .var.name = "footer", - checkmate::check_string(footer), - checkmate::check_multi_class(footer, c("shiny.tag", "shiny.tag.list", "html")) - ) - - ns <- NS(id) - - # Startup splash screen for delayed loading - # We use delayed loading in all cases, even when the data does not need to be fetched. - # This has the benefit that when filtering the data takes a lot of time initially, the - # Shiny app does not time out. - splash_ui <- if (inherits(data, "teal_data_module")) { - data$ui(ns("teal_data_module")) - } else if (inherits(data, "teal_data")) { - tags$div() - } - ui_teal( - id = ns("teal"), - splash_ui = tags$div(splash_ui, uiOutput(ns("error"))), - title = title, - header = header, - footer = footer + lifecycle::deprecate_soft( + when = "0.16", + what = "ui_teal_with_splash()", + details = "Deprecated, please use `ui_teal` instead" ) + ui_teal(id = id, data = data, title = title, header = header, footer = footer) } #' @export #' @rdname module_teal_with_splash srv_teal_with_splash <- function(id, data, modules, filter = teal_slices()) { - checkmate::assert_character(id, max.len = 1, any.missing = FALSE) - checkmate::assert_multi_class(data, c("teal_data", "teal_data_module")) - checkmate::assert_class(modules, "teal_modules") - checkmate::assert_class(filter, "teal_slices") - - moduleServer(id, function(input, output, session) { - logger::log_trace("srv_teal_with_splash initializing module with data.") - - if (getOption("teal.show_js_log", default = FALSE)) { - shinyjs::showLog() - } - - # teal_data_rv contains teal_data object - # either passed to teal::init or returned from teal_data_module - teal_data_rv <- if (inherits(data, "teal_data_module")) { - data <- data$server(id = "teal_data_module") - if (!is.reactive(data)) { - stop("The `teal_data_module` passed to `data` must return a reactive expression.", call. = FALSE) - } - data - } else if (inherits(data, "teal_data")) { - reactiveVal(data) - } - - teal_data_rv_validate <- reactive({ - # custom module can return error - data <- tryCatch(teal_data_rv(), error = function(e) e) - - # there is an empty reactive cycle on init! - if (inherits(data, "shiny.silent.error") && identical(data$message, "")) { - return(NULL) - } - - # to handle qenv.error - if (inherits(data, "qenv.error")) { - validate( - need( - FALSE, - paste( - "Error when executing `teal_data_module` passed to `data`:\n ", - paste(data$message, collapse = "\n"), - "\n Check your inputs or contact app developer if error persists." - ) - ) - ) - } - - # to handle module non-qenv errors - if (inherits(data, "error")) { - validate( - need( - FALSE, - paste( - "Error when executing `teal_data_module` passed to `data`:\n ", - paste(data$message, collpase = "\n"), - "\n Check your inputs or contact app developer if error persists." - ) - ) - ) - } - - validate( - need( - inherits(data, "teal_data"), - paste( - "Error: `teal_data_module` passed to `data` failed to return `teal_data` object, returned", - toString(sQuote(class(data))), - "instead.", - "\n Check your inputs or contact app developer if error persists." - ) - ) - ) - - if (!length(teal.data::datanames(data))) { - warning("`data` object has no datanames. Default datanames are set using `teal_data`'s environment.") - } - - is_modules_ok <- check_modules_datanames(modules, teal_data_datanames(data)) - if (!isTRUE(is_modules_ok)) { - validate(need(isTRUE(is_modules_ok), sprintf("%s. Contact app developer.", is_modules_ok))) - } - - is_filter_ok <- check_filter_datanames(filter, teal_data_datanames(data)) - if (!isTRUE(is_filter_ok)) { - showNotification( - "Some filters were not applied because of incompatibility with data. Contact app developer.", - type = "warning", - duration = 10 - ) - warning(is_filter_ok) - } - - teal_data_rv() - }) - - output$error <- renderUI({ - teal_data_rv_validate() - NULL - }) - - res <- srv_teal(id = "teal", modules = modules, teal_data_rv = teal_data_rv_validate, filter = filter) - logger::log_trace("srv_teal_with_splash initialized module with data.") - - res - }) + lifecycle::deprecate_soft( + when = "0.16", + what = "srv_teal_with_splash()", + details = "Deprecated, please use `srv_teal` instead" + ) + srv_teal(id = id, data = data, modules = modules, filter = filter) } diff --git a/R/module_transform_data.R b/R/module_transform_data.R new file mode 100644 index 0000000000..6159d2b900 --- /dev/null +++ b/R/module_transform_data.R @@ -0,0 +1,84 @@ +#' Module to transform `reactive` `teal_data` +#' +#' Module calls multiple [`module_teal_data`] in sequence so that `reactive teal_data` output +#' from one module is handed over to the following module's input. +#' +#' @inheritParams module_teal_data +#' @inheritParams teal_modules +#' @return `reactive` `teal_data` +#' +#' +#' @name module_transform_data +#' @keywords internal +NULL + +#' @rdname module_transform_data +ui_transform_data <- function(id, transforms, class = "well") { + checkmate::assert_string(id) + checkmate::assert_list(transforms, "teal_transform_module", null.ok = TRUE) + ns <- NS(id) + labels <- lapply(transforms, function(x) attr(x, "label")) + ids <- get_unique_labels(labels) + names(transforms) <- ids + + lapply( + names(transforms), + function(name) { + data_mod <- transforms[[name]] + wrapper_id <- ns(sprintf("wrapper_%s", name)) + div( # todo: accordion? + # class .teal_validated changes the color of the boarder on error in ui_validate_reactive_teal_data + # For details see tealValidate.js file. + class = c(class, "teal_validated"), + title = attr(data_mod, "label"), + tags$span( + class = "text-primary mb-4", + icon("fas fa-square-pen"), + attr(data_mod, "label") + ), + tags$i( + class = "remove pull-right fa fa-angle-down", + style = "cursor: pointer;", + title = "fold/expand transform panel", + onclick = sprintf("togglePanelItems(this, '%s', 'fa-angle-right', 'fa-angle-down');", wrapper_id) + ), + div( + id = wrapper_id, + ui_teal_data(id = ns(name), data_module = transforms[[name]]) + ) + ) + } + ) +} + +#' @rdname module_transform_data +srv_transform_data <- function(id, data, transforms, modules) { + checkmate::assert_string(id) + checkmate::assert_class(data, "reactive") + checkmate::assert_list(transforms, "teal_transform_module", null.ok = TRUE) + checkmate::assert_class(modules, "teal_module") + + if (length(transforms) == 0L) { + return(data) + } + + labels <- lapply(transforms, function(x) attr(x, "label")) + ids <- get_unique_labels(labels) + names(transforms) <- ids + + moduleServer(id, function(input, output, session) { + logger::log_debug("srv_teal_data_modules initializing.") + Reduce( + function(previous_result, name) { + srv_teal_data( + id = name, + data = previous_result, + data_module = transforms[[name]], + modules = modules + ) + }, + x = names(transforms), + init = data + ) + }) +} diff --git a/R/module_wunder_bar.R b/R/module_wunder_bar.R deleted file mode 100644 index 28b245f7bb..0000000000 --- a/R/module_wunder_bar.R +++ /dev/null @@ -1,93 +0,0 @@ -#' Manager bar module -#' -#' Bar of buttons that open modal dialogs. -#' -#' Creates a bar of buttons that open modal dialogs where manager modules reside. -#' Currently contains three modules: -#' - [`module_filter_manager`] -#' - [`module_snapshot_manager`] -#' - [`module_bookmark_manager`] -#' -#' The bar is placed in the `teal` app UI, next to the filter panel hamburger. -#' -#' @name module_wunder_bar -#' @aliases wunder_bar wunder_bar_module -#' -#' @inheritParams module_tabs_with_filters -#' -#' @return Nothing is returned. - -#' @rdname module_wunder_bar -#' @keywords internal -wunder_bar_ui <- function(id) { - ns <- NS(id) - rev( - tagList( - title = "", - tags$button( - id = ns("show_filter_manager"), - class = "btn action-button wunder_bar_button", - title = "View filter mapping", - suppressMessages(icon("solid fa-grip")) - ), - tags$button( - id = ns("show_snapshot_manager"), - class = "btn action-button wunder_bar_button", - title = "Manage filter state snapshots", - icon("camera") - ), - bookmark_manager_ui(ns("bookmark_manager")) - ) - ) -} - -#' @rdname module_wunder_bar -#' @keywords internal -wunder_bar_srv <- function(id, datasets, filter, modules) { - moduleServer(id, function(input, output, session) { - logger::log_trace("wunder_bar_srv initializing") - - setBookmarkExclude(c("show_filter_manager", "show_bookmark_manager", "show_bookmark_manager")) - - ns <- session$ns - - observeEvent(input$show_filter_manager, { - logger::log_trace("wunder_bar_srv@1 show_filter_manager button has been clicked.") - showModal( - modalDialog( - filter_manager_ui(ns("filter_manager")), - class = "filter_manager_modal", - size = "l", - footer = NULL, - easyClose = TRUE - ) - ) - }) - - observeEvent(input$show_snapshot_manager, { - logger::log_trace("wunder_bar_srv@1 show_snapshot_manager button has been clicked.") - showModal( - modalDialog( - snapshot_manager_ui(ns("snapshot_manager")), - class = "snapshot_manager_modal", - size = "m", - footer = NULL, - easyClose = TRUE - ) - ) - }) - - filter_manager_results <- filter_manager_srv( - id = "filter_manager", - datasets = datasets, - filter = filter - ) - snapshot_history <- snapshot_manager_srv( - id = "snapshot_manager", - slices_global = filter_manager_results$slices_global, - mapping_matrix = filter_manager_results$mapping_matrix, - datasets = filter_manager_results$datasets_flat - ) - bookmark_manager_srv(id = "bookmark_manager", modules = modules) - }) -} diff --git a/R/modules.R b/R/modules.R index 4ec76a2ca7..941e8baa6f 100644 --- a/R/modules.R +++ b/R/modules.R @@ -2,7 +2,6 @@ #' #' @description #' `r lifecycle::badge("stable")` -#' #' Create a nested tab structure to embed modules in a `teal` application. #' #' @details @@ -49,6 +48,12 @@ #' @param ui_args (named `list`) with additional arguments passed on to the UI function. #' @param x (`teal_module` or `teal_modules`) Object to format/print. #' @param indent (`integer(1)`) Indention level; each nested element is indented one level more. +#' @param transformers (`list` of `teal_data_module`) that will be applied to transform the data. +#' Each transform module UI will appear in the `teal` application, unless the `custom_ui` attribute is set on the list. +#' If so, the module developer is responsible to display the UI in the module itself. +#' +#' When the transformation does not have sufficient input data, the resulting data will fallback +#' to the last successful transform or, in case there are none, to the filtered data. #' @param ... #' - For `modules()`: (`teal_module` or `teal_modules`) Objects to wrap into a tab. #' - For `format()` and `print()`: Arguments passed to other methods. @@ -118,21 +123,17 @@ #' if (interactive()) { #' shinyApp(app$ui, app$server) #' } - #' @rdname teal_modules #' @export #' module <- function(label = "module", - server = function(id, ...) { - moduleServer(id, function(input, output, session) {}) - }, - ui = function(id, ...) { - tags$p(paste0("This module has no UI (id: ", id, " )")) - }, + server = function(id, ...) moduleServer(id, function(input, output, session) NULL), + ui = function(id, ...) tags$p(paste0("This module has no UI (id: ", id, " )")), filters, datanames = "all", server_args = NULL, - ui_args = NULL) { + ui_args = NULL, + transformers = list()) { # argument checking (independent) ## `label` checkmate::assert_string(label) @@ -239,11 +240,18 @@ module <- function(label = "module", ) } + ## `transformers` + checkmate::assert_list(transformers, types = "teal_data_module") + structure( list( label = label, - server = server, ui = ui, datanames = unique(datanames), - server_args = server_args, ui_args = ui_args + server = server, + ui = ui, + datanames = unique(datanames), + server_args = server_args, + ui_args = ui_args, + transformers = transformers ), class = "teal_module" ) diff --git a/R/show_rcode_modal.R b/R/show_rcode_modal.R index 83bd248a98..0f7fe88228 100644 --- a/R/show_rcode_modal.R +++ b/R/show_rcode_modal.R @@ -15,7 +15,7 @@ #' @export show_rcode_modal <- function(title = NULL, rcode, session = getDefaultReactiveDomain()) { lifecycle::deprecate_soft( - when = "0.15.3", + when = "0.16", what = "show_rcode_modal()", details = "This function will be removed in the next release." ) diff --git a/R/tdata.R b/R/tdata.R index 698cc90885..bb020c99f7 100644 --- a/R/tdata.R +++ b/R/tdata.R @@ -1,200 +1,62 @@ #' Create a `tdata` object #' -#' @description `r lifecycle::badge("deprecated")` +#' @description `r lifecycle::badge("superseded")` #' -#' Create a new object called `tdata` which contains `data`, a `reactive` list of `data.frames` -#' (or `MultiAssayExperiment`), with attributes: -#' - `code` (`reactive`) containing code used to generate the data -#' - join_keys (`join_keys`) containing the relationships between the data -#' - metadata (named `list`) containing any metadata associated with the data frames +#' Recent changes in `teal` cause modules to fail because modules expect a `tdata` object +#' to be passed to the `data` argument but instead they receive a `teal_data` object, +#' which is additionally wrapped in a reactive expression in the server functions. +#' In order to easily adapt such modules without a proper refactor, +#' use this function to downgrade the `data` argument. #' #' @name tdata -#' @param data (named `list`) A list of `data.frame` or `MultiAssayExperiment` objects, -#' which optionally can be `reactive`. -#' Inside this object all of these items will be made `reactive`. -#' @param code (`character` or `reactive` which evaluates to a `character`) containing -#' the code used to generate the data. This should be `reactive` if the code is changing -#' during a reactive context (e.g. if filtering changes the code). Inside this -#' object `code` will be made reactive -#' @param join_keys (`teal.data::join_keys`) object containing relationships between the -#' datasets. -#' @param metadata (named `list`) each element contains a list of metadata about the named `data.frame` -#' Each element of these list should be atomic and length one. -#' @return A `tdata` object. -#' -#' @seealso `as_tdata` -#' -#' @examples -#' -#' data <- new_tdata( -#' data = list(iris = iris, mtcars = reactive(mtcars), dd = data.frame(x = 1:10)), -#' code = "iris <- iris -#' mtcars <- mtcars -#' dd <- data.frame(x = 1:10)", -#' metadata = list(dd = list(author = "NEST"), iris = list(version = 1)) -#' ) -#' -#' # Extract a data.frame -#' isolate(data[["iris"]]()) -#' -#' # Get code -#' isolate(get_code_tdata(data)) -#' -#' # Get metadata -#' get_metadata(data, "iris") -#' -#' @export -new_tdata <- function(data, code = "", join_keys = NULL, metadata = NULL) { - lifecycle::deprecate_soft( - when = "0.15.0", - what = "tdata()", - details = paste( - "tdata is deprecated and will be removed in the next release. Use `teal_data` instead.\n", - "Please follow migration instructions https://github.com/insightsengineering/teal/discussions/987." - ) - ) - checkmate::assert_list( - data, - any.missing = FALSE, names = "unique", - types = c("data.frame", "reactive", "MultiAssayExperiment") - ) - checkmate::assert_class(join_keys, "join_keys", null.ok = TRUE) - checkmate::assert_multi_class(code, c("character", "reactive")) - - checkmate::assert_list(metadata, names = "unique", null.ok = TRUE) - checkmate::assert_subset(names(metadata), names(data)) +#' @param ... ignored +#' @return nothing +NULL - if (is.reactive(code)) { - isolate(checkmate::assert_class(code(), "character", .var.name = "code")) - } - - # create reactive data.frames - for (x in names(data)) { - if (!is.reactive(data[[x]])) { - data[[x]] <- do.call(reactive, list(as.name(x)), envir = list2env(data[x])) - } - } - - # set attributes - attr(data, "code") <- if (is.reactive(code)) code else reactive(code) - attr(data, "join_keys") <- join_keys - attr(data, "metadata") <- metadata - - # set class - class(data) <- c("tdata", class(data)) - data -} - -#' Function to convert a `tdata` object to an `environment` -#' -#' Any `reactive` expressions inside `tdata` are evaluated first. -#' @param data (`tdata`) object -#' @return An `environment`. -#' @examples -#' -#' data <- new_tdata( -#' data = list(iris = iris, mtcars = reactive(mtcars)), -#' code = "iris <- iris -#' mtcars = mtcars" -#' ) -#' -#' my_env <- isolate(tdata2env(data)) -#' +#' @rdname tdata #' @export -tdata2env <- function(data) { - checkmate::assert_class(data, "tdata") - list2env(lapply(data, function(x) if (is.reactive(x)) x() else x)) +new_tdata <- function(...) { + .deprecate_tdata_msg() } - -#' Wrapper for `get_code.tdata` -#' -#' This wrapper is to be used by downstream packages to extract the code of a `tdata` object. -#' -#' @param data (`tdata`) object -#' -#' @return (`character`) code used in the `tdata` object. +#' @rdname tdata #' @export -get_code_tdata <- function(data) { - checkmate::assert_class(data, "tdata") - attr(data, "code")() +tdata2env <- function(...) { + .deprecate_tdata_msg() } -#' Extract `join_keys` from `tdata` -#' @param data (`tdata`) object -#' @param ... Additional arguments (not used) +#' @rdname tdata #' @export -join_keys.tdata <- function(data, ...) { - attr(data, "join_keys") +get_code_tdata <- function(...) { + .deprecate_tdata_msg() } -#' Function to get metadata from a `tdata` object -#' @param data (`tdata` - object) to extract the data from -#' @param dataname (`character(1)`) the dataset name whose metadata is requested -#' @return Either list of metadata or NULL if no metadata. +#' @rdname tdata #' @export -get_metadata <- function(data, dataname) { - checkmate::assert_string(dataname) - UseMethod("get_metadata", data) +join_keys.tdata <- function(...) { + .deprecate_tdata_msg() } -#' @rdname get_metadata +#' @rdname tdata #' @export -get_metadata.tdata <- function(data, dataname) { - metadata <- attr(data, "metadata") - if (is.null(metadata)) { - return(NULL) - } - metadata[[dataname]] +get_metadata <- function(...) { + .deprecate_tdata_msg() } -#' @rdname get_metadata +#' @rdname tdata #' @export -get_metadata.default <- function(data, dataname) { - stop("get_metadata function not implemented for this object") +as_tdata <- function(...) { + .deprecate_tdata_msg() } -#' Downgrade `teal_data` objects in modules for compatibility -#' -#' Convert `teal_data` to `tdata` in `teal` modules. -#' -#' Recent changes in `teal` cause modules to fail because modules expect a `tdata` object -#' to be passed to the `data` argument but instead they receive a `teal_data` object, -#' which is additionally wrapped in a reactive expression in the server functions. -#' In order to easily adapt such modules without a proper refactor, -#' use this function to downgrade the `data` argument. -#' -#' @param x data object, either `tdata` or `teal_data`, the latter possibly in a reactive expression -#' -#' @return Object of class `tdata`. -#' -#' @examples -#' td <- teal_data() -#' td <- within(td, iris <- iris) %>% within(mtcars <- mtcars) -#' td -#' as_tdata(td) -#' as_tdata(reactive(td)) -#' -#' @export -#' @rdname tdata_deprecation -#' -as_tdata <- function(x) { - if (inherits(x, "tdata")) { - return(x) - } - if (is.reactive(x)) { - checkmate::assert_class(isolate(x()), "teal_data") - datanames <- isolate(teal_data_datanames(x())) - datasets <- sapply(datanames, function(dataname) reactive(x()[[dataname]]), simplify = FALSE) - code <- reactive(teal.code::get_code(x())) - join_keys <- isolate(teal.data::join_keys(x())) - } else if (inherits(x, "teal_data")) { - datanames <- teal_data_datanames(x) - datasets <- sapply(datanames, function(dataname) reactive(x[[dataname]]), simplify = FALSE) - code <- reactive(teal.code::get_code(x)) - join_keys <- isolate(teal.data::join_keys(x)) - } - - new_tdata(data = datasets, code = code, join_keys = join_keys) +.deprecate_tdata_msg <- function() { + lifecycle::deprecate_stop( + when = "0.16", + what = "tdata()", + details = paste( + "tdata has been removed in favour of `teal_data`.\n", + "Please follow migration instructions https://github.com/insightsengineering/teal/discussions/987." + ) + ) } diff --git a/R/teal.R b/R/teal.R index 78de791513..b3b7163a50 100644 --- a/R/teal.R +++ b/R/teal.R @@ -12,6 +12,4 @@ #' @import shiny teal.data teal.slice #' @importFrom stats setNames #' @export -#' @importFrom magrittr %>% -magrittr::`%>%` NULL diff --git a/R/teal_data_module.R b/R/teal_data_module.R index 70258cda07..eddf1fa277 100644 --- a/R/teal_data_module.R +++ b/R/teal_data_module.R @@ -6,8 +6,8 @@ #' Create a `teal_data_module` object and evaluate code on it with history tracking. #' #' @details -#' `teal_data_module` creates a `shiny` module to supply or modify data in a `teal` application. -#' The module allows for running data pre-processing code (creation _and_ some modification) after the app starts. +#' `teal_data_module` creates a `shiny` module to interactively supply or modify data in a `teal` application. +#' The module allows for running any code (creation _and_ some modification) after the app starts or reloads. #' The body of the server function will be run in the app rather than in the global environment. #' This means it will be run every time the app starts, so use sparingly. #' @@ -21,9 +21,16 @@ #' @param server (`function(id)`) #' `shiny` module server function; must only take `id` argument; #' must return reactive expression containing `teal_data` object +#' @param label (`character(1)`) Label of the module. +#' @param once (`logical(1)`) +#' If `TRUE`, the data module will be shown only once and will disappear after successful data loading. +#' App user will no longer be able to interact with this module anymore. +#' If `FALSE`, the data module can be reused multiple times. +#' App user will be able to interact and change the data output from the module multiple times. #' #' @return -#' `teal_data_module` returns an object of class `teal_data_module`. +#' `teal_data_module` returns a list of class `teal_data_module` containing two elements, `ui` and +#' `server` provided via arguments. #' #' @examples #' tdm <- teal_data_module( @@ -53,11 +60,72 @@ #' @seealso [`teal.data::teal_data-class`], [teal.code::qenv()] #' #' @export -teal_data_module <- function(ui, server) { +teal_data_module <- function(ui, server, label = "data module", once = TRUE) { checkmate::assert_function(ui, args = "id", nargs = 1) checkmate::assert_function(server, args = "id", nargs = 1) structure( list(ui = ui, server = server), - class = "teal_data_module" + label = label, + class = "teal_data_module", + once = once + ) +} + +#' Data module for `teal` transformers. +#' +#' @description +#' `r lifecycle::badge("experimental")` +#' +#' Create a `teal_data_module` object for custom transformation of data for pre-processing +#' before passing the data into the module. +#' +#' @details +#' `teal_transform_module` creates a [`teal_data_module`] object to transform data in a `teal` +#' application. This transformation happens after the data has passed through the filtering activity +#' in teal. The transformed data is then sent to the server of the [teal_module()]. +#' +#' See vignette `vignette("data-transform-as-shiny-module", package = "teal")` for more details. +#' +#' +#' @inheritParams teal_data_module +#' @param server (`function(id, data)`) +#' `shiny` module server function; that takes `id` and `data` argument, +#' where the `id` is the module id and `data` is the reactive `teal_data` input. +#' The server function must return reactive expression containing `teal_data` object. +#' @examples +#' my_transformers <- list( +#' teal_transform_module( +#' label = "Custom transform for iris", +#' ui = function(id) { +#' ns <- NS(id) +#' tags$div( +#' numericInput(ns("n_rows"), "Subset n rows", value = 6, min = 1, max = 150, step = 1) +#' ) +#' }, +#' server = function(id, data) { +#' moduleServer(id, function(input, output, session) { +#' reactive({ +#' within(data(), +#' { +#' iris <- head(iris, num_rows) +#' }, +#' num_rows = input$n_rows +#' ) +#' }) +#' }) +#' } +#' ) +#' ) +#' +#' @name teal_transform_module +#' +#' @export +teal_transform_module <- function(ui, server, label = "transform module") { + checkmate::assert_function(ui, args = "id", nargs = 1) + checkmate::assert_function(server, args = c("id", "data"), nargs = 2) + structure( + list(ui = ui, server = server), + label = label, + class = c("teal_transform_module", "teal_data_module") ) } diff --git a/R/teal_data_utils.R b/R/teal_data_utils.R new file mode 100644 index 0000000000..81003d8c5f --- /dev/null +++ b/R/teal_data_utils.R @@ -0,0 +1,86 @@ +#' `teal_data` utils +#' +#' In `teal` we need to recreate the `teal_data` object due to two operations:integer.max +#' - we need to append filter-data code and objects which have been evaluated in `FilteredData` and +#' we want to avoid double-evaluation. +#' - we need to subset `teal_data` to `datanames` used by the module, to shorten obtainable R-code +#' +#' Due to above recreation of `teal_data` object can't be done simply by using public +#' `teal.code` and `teal.data` methods. +#' +#' @param data (`teal_data`) +#' @param code (`character`) code to append to `data@code` +#' @param objects (`list`) objects to append to `data@env` +#' @param datanames (`character`) names of the datasets +#' @return modified `teal_data` +#' @keywords internal +#' @name teal_data_utilities +NULL + +#' @rdname teal_data_utilities +.append_evaluated_code <- function(data, code) { + checkmate::assert_class(data, "teal_data") + data@code <- c(data@code, code) + data@id <- c(data@id, max(data@id) + 1L + seq_along(code)) + data@messages <- c(data@messages, rep("", length(code))) + data@warnings <- c(data@warnings, rep("", length(code))) + methods::validObject(data) + data +} + +#' @rdname teal_data_utilities +.append_modified_data <- function(data, objects) { + checkmate::assert_class(data, "teal_data") + checkmate::assert_class(objects, "list") + new_env <- list2env(objects, parent = .GlobalEnv) + rlang::env_coalesce(new_env, data@env) + data@env <- new_env + data +} + +#' @rdname teal_data_utilities +.subset_teal_data <- function(data, datanames) { + checkmate::assert_class(data, "teal_data") + checkmate::assert_class(datanames, "character") + datanames_corrected <- intersect(datanames, ls(data@env)) + dataname_corrected_with_raw <- intersect(c(datanames, sprintf("%s._raw_", datanames)), ls(data@env)) + + if (!length(datanames)) { + return(teal_data()) + } + + new_data <- do.call( + teal.data::teal_data, + args = c( + mget(x = dataname_corrected_with_raw, envir = data@env), + list( + code = gsub( + "warning('Code was not verified for reproducibility.')\n", + "", + teal.data::get_code(data, datanames = dataname_corrected_with_raw), + fixed = TRUE + ), + join_keys = teal.data::join_keys(data)[datanames_corrected] + ) + ) + ) + new_data@verified <- data@verified + teal.data::datanames(new_data) <- datanames_corrected + new_data +} + +#' @rdname teal_data_utilities +.teal_data_datanames <- function(data) { + checkmate::assert_class(data, "teal_data") + datanames <- teal.data::datanames(data) + if (length(datanames)) { + datanames + } else { + .teal_data_ls(data) + } +} + +#' @rdname teal_data_utilities +.teal_data_ls <- function(data) { + grep("._raw_", ls(teal.code::get_env(data), all.names = TRUE), value = TRUE, invert = TRUE) +} diff --git a/R/teal_lockfile.R b/R/teal_lockfile.R index 0c632a5298..5d253f8e9c 100644 --- a/R/teal_lockfile.R +++ b/R/teal_lockfile.R @@ -48,7 +48,7 @@ teal_lockfile <- function() { lockfile_task <- ExtendedTask$new(create_renv_lockfile) lockfile_task$invoke(close = inherits(old_plan, "sequential"), lockfile_path) - logger::log_trace("lockfile creation invoked.") + logger::log_debug("lockfile creation invoked.") } } @@ -69,9 +69,9 @@ create_renv_lockfile <- function(close = FALSE, lockfile_path = NULL) { ) ) if (any(grepl("Lockfile written", renv_logs))) { - logger::log_trace("lockfile created successfully.") + logger::log_debug("lockfile created successfully.") } else { - logger::log_trace("lockfile created with issues.") + logger::log_debug("lockfile created with issues.") } lockfile_path @@ -94,7 +94,7 @@ teal_lockfile_downloadhandler <- function() { teal_lockfile <- "teal_app.lock" iter <- 1 while (!file.exists(teal_lockfile) && iter <= 100) { - logger::log_trace("lockfile not created yet, retrying...") + logger::log_debug("lockfile not created yet, retrying...") Sys.sleep(0.25) iter <- iter + 1 # max wait time is 25 seconds } diff --git a/R/teal_slices.R b/R/teal_slices.R index f27f339842..33e3801e00 100644 --- a/R/teal_slices.R +++ b/R/teal_slices.R @@ -84,8 +84,13 @@ teal_slices <- function(..., all_slice_id <- vapply(slices, `[[`, character(1L), "id") if (missing(mapping)) { - mapping <- list(global_filters = all_slice_id) + mapping <- if (length(all_slice_id)) { + list(global_filters = all_slice_id) + } else { + list() + } } + if (!module_specific) { mapping[setdiff(names(mapping), "global_filters")] <- NULL } diff --git a/R/utils.R b/R/utils.R index aefb9bceec..410ad9d1ef 100644 --- a/R/utils.R +++ b/R/utils.R @@ -59,18 +59,15 @@ include_parent_datanames <- function(dataname, join_keys) { #' @param datanames (`character`) vector of data set names to include; must be subset of `datanames(x)` #' @return A `FilteredData` object. #' @keywords internal -teal_data_to_filtered_data <- function(x, datanames = teal_data_datanames(x)) { +teal_data_to_filtered_data <- function(x, datanames = .teal_data_datanames(x)) { checkmate::assert_class(x, "teal_data") checkmate::assert_character(datanames, min.chars = 1L, any.missing = FALSE) - ans <- teal.slice::init_filtered_data( + # Otherwise, FilteredData will be created in the modules' scope later + teal.slice::init_filtered_data( x = sapply(datanames, function(dn) x[[dn]], simplify = FALSE), join_keys = teal.data::join_keys(x) ) - # Piggy-back pre-processing code for datasets of interest so that filtering code can be appended later. - attr(ans, "preprocessing_code") <- teal.data::get_code(x, datanames = datanames, check_names = FALSE) - attr(ans, "verification_status") <- x@verified - ans } #' Template function for `TealReportCard` creation and customization @@ -116,28 +113,59 @@ report_card_template <- function(title, label, description = NULL, with_filter, #' @return A `character(1)` containing error message or `TRUE` if validation passes. #' @keywords internal check_modules_datanames <- function(modules, datanames) { - checkmate::assert_class(modules, "teal_modules") + checkmate::assert_multi_class(modules, c("teal_modules", "teal_module")) checkmate::assert_character(datanames) recursive_check_datanames <- function(modules, datanames) { # check teal_modules against datanames if (inherits(modules, "teal_modules")) { - sapply(modules$children, function(module) recursive_check_datanames(module, datanames = datanames)) + result <- lapply(modules$children, function(module) recursive_check_datanames(module, datanames = datanames)) + result <- result[vapply(result, Negate(is.null), logical(1L))] + list( + string = do.call(c, as.list(unname(sapply(result, function(x) x$string)))), + html = function(with_module_name = TRUE) { + tagList( + lapply( + result, + function(x) x$html(with_module_name = with_module_name) + ) + ) + } + ) } else { extra_datanames <- setdiff(modules$datanames, c("all", datanames)) if (length(extra_datanames)) { - sprintf( - "- Module '%s' uses datanames not available in 'data': (%s) not in (%s)", - modules$label, - toString(dQuote(extra_datanames, q = FALSE)), - toString(dQuote(datanames, q = FALSE)) + list( + string = build_datanames_error_message( + modules$label, + datanames, + extra_datanames, + tags = list( + span = function(..., .noWS = NULL) { # nolint: object_name + trimws(paste(..., sep = ifelse(is.null(.noWS), " ", ""), collapse = " ")) + }, + code = function(x) toString(dQuote(x, q = FALSE)) + ), + tagList = function(...) trimws(paste(...)) + ), + # Build HTML representation of the error message with
 formatting
+          html = function(with_module_name = TRUE) {
+            tagList(
+              build_datanames_error_message(
+                if (with_module_name) modules$label,
+                datanames,
+                extra_datanames
+              ),
+              tags$br(.noWS = "before")
+            )
+          }
         )
       }
     }
   }
-  check_datanames <- unlist(recursive_check_datanames(modules, datanames))
+  check_datanames <- recursive_check_datanames(modules, datanames)
   if (length(check_datanames)) {
-    paste(check_datanames, collapse = "\n")
+    check_datanames
   } else {
     TRUE
   }
@@ -180,132 +208,6 @@ check_filter_datanames <- function(filters, datanames) {
   }
 }
 
-
-#' Create filterable data for modules
-#'
-#' Converts input data to a `FilteredData` object(s) to allow filtering before passing data to individual modules.
-#'
-#' @param data (`teal_data`)
-#' @param modules (`teal_modules`) object
-#' @param filters (`teal_slices`) object
-#' @param filtered_data_singleton A result of `teal_data_to_filtered_data` applied to `data`.
-#' @param progress (`Progress`) object from `shiny`, optional.
-#'  The progress bar will be filled during the (possibly recursive) call.
-#' @return Returns list of same shape as `modules`, containing `FilteredData` at every leaf.
-#' If module specific, each leaf contains different instance, otherwise every leaf contains `filtered_data_singleton`.
-#' @keywords internal
-modules_datasets <- function(data,
-                             modules,
-                             filters,
-                             filtered_data_singleton = teal_data_to_filtered_data(data),
-                             progress = NULL) {
-  checkmate::assert_class(data, "teal_data")
-  checkmate::assert_multi_class(modules, c("teal_modules", "teal_module"))
-  checkmate::assert_class(filters, "modules_teal_slices")
-  checkmate::assert_r6(filtered_data_singleton, "FilteredData")
-  checkmate::assert_r6(progress, "Progress", null.ok = TRUE)
-
-  if (!isTRUE(attr(filters, "module_specific"))) {
-    if (!is.null(progress)) {
-      progress$inc(
-        amount = progress$getMax(),
-        detail = "100%"
-      )
-    }
-
-    # subset global filters
-    slices <- shiny::isolate({
-      Filter(function(x) x$id %in% attr(filters, "mapping")$global_filters, filters)
-    })
-    filtered_data_singleton$set_filter_state(slices)
-
-    return(modules_structure(modules, filtered_data_singleton))
-  }
-
-  if (inherits(modules, "teal_module")) {
-    if (!is.null(progress)) {
-      progress$inc(
-        amount = 1,
-        detail = sprintf("%s%%", round(progress$getValue() / progress$getMax(), 2L) * 100)
-      )
-    }
-
-    # 1. get datanames
-    datanames <-
-      if (is.null(modules$datanames) || identical(modules$datanames, "all")) {
-        include_parent_datanames(
-          teal_data_datanames(data),
-          teal.data::join_keys(data)
-        )
-      } else {
-        include_parent_datanames(
-          modules$datanames,
-          teal.data::join_keys(data)
-        )
-      }
-    # 2. subset filters (global + dedicated)
-    slices <- shiny::isolate({
-      Filter(x = filters, f = function(x) {
-        x$dataname %in% datanames &&
-          (x$id %in% attr(filters, "mapping")$global_filters ||
-            x$id %in% unique(unlist(attr(filters, "mapping")[modules$label]))) # nolint: indentation_linter.
-      })
-    })
-    # 2a. subset include/exclude varnames
-    slices$include_varnames <- attr(slices, "include_varnames")[names(attr(slices, "include_varnames")) %in% datanames]
-    slices$exclude_varnames <- attr(slices, "exclude_varnames")[names(attr(slices, "exclude_varnames")) %in% datanames]
-
-    # 3. instantiate FilteredData
-    filtered_data <- teal_data_to_filtered_data(data, datanames)
-    # 4. set state
-    filtered_data$set_filter_state(slices)
-    # 5. return
-    return(filtered_data)
-  } else if (inherits(modules, "teal_modules")) {
-    ans <- lapply(
-      modules$children,
-      modules_datasets,
-      data = data,
-      filters = filters,
-      filtered_data_singleton = filtered_data_singleton,
-      progress = progress
-    )
-    names(ans) <- vapply(modules$children, `[[`, character(1), "label")
-
-    return(ans)
-  }
-
-  stop("something is not right")
-}
-
-# Returns nested list of same shape as `modules` with `value` at every leaf.
-modules_structure <- function(modules, value = TRUE) {
-  if (inherits(modules, "teal_module")) {
-    return(value)
-  } else {
-    stats::setNames(
-      lapply(modules$children, modules_structure, value),
-      vapply(modules$children, `[[`, character(1), "label")
-    )
-  }
-}
-
-#' Wrapper on `teal.data::datanames`
-#'
-#' Special function used in internals of `teal` to return names of datasets even if `datanames`
-#' has not been set.
-#' @param data (`teal_data`)
-#' @return `character`
-#' @keywords internal
-teal_data_datanames <- function(data) {
-  checkmate::assert_class(data, "teal_data")
-  if (length(teal.data::datanames(data))) {
-    teal.data::datanames(data)
-  } else {
-    ls(teal.code::get_env(data), all.names = TRUE)
-  }
-}
-
 #' Function for validating the title parameter of `teal::init`
 #'
 #' Checks if the input of the title from `teal::init` will create a valid title and favicon tag.
@@ -369,7 +271,7 @@ create_app_id <- function(data, modules) {
   checkmate::assert_class(modules, "teal_modules")
 
   data <- if (inherits(data, "teal_data")) {
-    as.list(data@env)
+    as.list(teal.code::get_env(data))
   } else if (inherits(data, "teal_data_module")) {
     deparse1(body(data$server))
   }
@@ -403,3 +305,73 @@ defunction <- function(x) {
 get_unique_labels <- function(labels) {
   make.unique(gsub("[^[:alnum:]]", "_", tolower(labels)), sep = "_")
 }
+
+#' Remove ANSI escape sequences from a string
+#' @noRd
+strip_style <- function(string) {
+  checkmate::assert_string(string)
+
+  gsub(
+    "(?:(?:\\x{001b}\\[)|\\x{009b})(?:(?:[0-9]{1,3})?(?:(?:;[0-9]{0,3})*)?[A-M|f-m])|\\x{001b}[A-M]",
+    "",
+    string,
+    perl = TRUE,
+    useBytes = TRUE
+  )
+}
+
+#' Convert character list to human readable html with commas and "and"
+#' @noRd
+paste_datanames_character <- function(x,
+                                      tags = list(span = shiny::tags$span, code = shiny::tags$code),
+                                      tagList = shiny::tagList) { # nolint: object_name.
+  checkmate::assert_character(x)
+  do.call(
+    tagList,
+    lapply(seq_along(x), function(.ix) {
+      tagList(
+        tags$code(x[.ix]),
+        if (.ix != length(x)) {
+          tags$span(ifelse(.ix == length(x) - 1, " and ", ", "))
+        }
+      )
+    })
+  )
+}
+
+#' Build datanames error string for error message
+#'
+#' tags and tagList are overwritten in arguments allowing to create strings for
+#' logging purposes
+#' @noRd
+build_datanames_error_message <- function(label = NULL,
+                                          datanames,
+                                          extra_datanames,
+                                          tags = list(span = shiny::tags$span, code = shiny::tags$code),
+                                          tagList = shiny::tagList) { # nolint: object_name.
+  tags$span(
+    tags$span(ifelse(length(extra_datanames) > 1, "Datasets", "Dataset")),
+    paste_datanames_character(extra_datanames, tags, tagList),
+    tags$span(
+      paste0(
+        ifelse(length(extra_datanames) > 1, "are missing", "is missing"),
+        ifelse(is.null(label), ".", sprintf(" for tab '%s'.", label))
+      )
+    ),
+    if (length(datanames) >= 1) {
+      tagList(
+        tags$span(ifelse(length(datanames) > 1, "Datasets", "Dataset")),
+        tags$span("available in data:"),
+        tagList(
+          tags$span(
+            paste_datanames_character(datanames, tags, tagList),
+            tags$span(".", .noWS = "outside"),
+            .noWS = c("outside")
+          )
+        )
+      )
+    } else {
+      tags$span("No datasets are available in data.")
+    }
+  )
+}
diff --git a/_pkgdown.yml b/_pkgdown.yml
index 3fa9e7c5a3..14c6b437fe 100644
--- a/_pkgdown.yml
+++ b/_pkgdown.yml
@@ -6,7 +6,8 @@ template:
 
 navbar:
   structure:
-    left: [get-started, reference, articles, blueprint, tutorials, news, reports]
+    left:
+      [get-started, reference, articles, blueprint, tutorials, news, reports]
     right: [search, github]
   components:
     get-started:
@@ -47,49 +48,50 @@ navbar:
 
     github:
       icon: fa-github
+      aria-label: View on Github
       href: https://github.com/insightsengineering/teal
 
 articles:
-- title: Get started
-  navbar: Get started
-  contents:
-    - getting-started-with-teal
-- title: Using `teal`
-  navbar: Using `teal`
-  contents:
-    - filter-panel
-    - teal-options
-    - bootstrap-themes-in-teal
-- title: Data in `teal` apps
-  navbar: Data in `teal` apps
-  contents:
-    - including-data-in-teal-applications
-    - data-as-shiny-module
-- title: Extending `teal`
-  navbar: Extending `teal`
-  contents:
-    - creating-custom-modules
-    - adding-support-for-reporting
-- title: 📃 Technical blueprint
-  desc: >
-    The purpose of the blueprint is to aid new developer’s comprehension of the
-    fundamental principles of the `teal` framework. We will explore crucial `teal`
-    concepts such as data flow, actors, and filter panel, among others.
-  contents:
-    - blueprint/index
-    - blueprint/intro
-    - blueprint/actors
-    - blueprint/dataflow
-    - blueprint/product_map
-- title: ""
-  desc: >
-    Features.
-  contents:
-  - blueprint/input_data
-  - blueprint/in_app_data
-  - blueprint/filter_panel
-  - blueprint/module_encapsulation
-
+  - title: Get started
+    navbar: Get started
+    contents:
+      - getting-started-with-teal
+  - title: Using `teal`
+    navbar: Using `teal`
+    contents:
+      - filter-panel
+      - teal-options
+      - bootstrap-themes-in-teal
+  - title: Data in `teal` apps
+    navbar: Data in `teal` apps
+    contents:
+      - including-data-in-teal-applications
+      - data-as-shiny-module
+      - data-transform-as-shiny-module
+  - title: Extending `teal`
+    navbar: Extending `teal`
+    contents:
+      - creating-custom-modules
+      - adding-support-for-reporting
+  - title: 📃 Technical blueprint
+    desc: >
+      The purpose of the blueprint is to aid new developer’s comprehension of the
+      fundamental principles of the `teal` framework. We will explore crucial `teal`
+      concepts such as data flow, actors, and filter panel, among others.
+    contents:
+      - blueprint/index
+      - blueprint/intro
+      - blueprint/actors
+      - blueprint/dataflow
+      - blueprint/product_map
+  - title: ""
+    desc: >
+      Features.
+    contents:
+      - blueprint/input_data
+      - blueprint/in_app_data
+      - blueprint/filter_panel
+      - blueprint/module_encapsulation
 
 reference:
   - title: Core `teal` functions
@@ -97,10 +99,11 @@ reference:
     contents:
       - init
       - teal_data_module
+      - teal_transform_module
+      - module_teal_with_splash
+      - module_teal
       - module
       - modules
-      - srv_teal_with_splash
-      - ui_teal_with_splash
       - teal_slices
   - title: Helper Functions
     desc: Helper functions for `teal`
diff --git a/inst/WORDLIST b/inst/WORDLIST
index 831c269182..8c079861d8 100644
--- a/inst/WORDLIST
+++ b/inst/WORDLIST
@@ -21,9 +21,11 @@ lockfile
 omics
 pre
 programmatically
+reactively
 repo
 reproducibility
 summarization
+tabset
 tabsetted
 themer
 theming
diff --git a/inst/css/custom.css b/inst/css/custom.css
index d0ee28c782..c6ef6ab5bd 100644
--- a/inst/css/custom.css
+++ b/inst/css/custom.css
@@ -77,3 +77,20 @@ body:not(.modal-open) {
   padding-right: 0 !important;
   overflow: auto !important;
 }
+
+/* teal_data_module modal styling */
+body > div:has(~ #shiny-modal-wrapper .fade .blur_background) {
+  transition: filter 0.3s;
+}
+
+body > div:has(~ #shiny-modal-wrapper .blur_background) {
+  filter: blur(5px);
+}
+
+#shiny-modal.fade.in:has(.hide_background) {
+  transition: background-color 0.3s;
+}
+
+#shiny-modal:has(.hide_background) {
+  background-color: white;
+}
diff --git a/inst/css/sidebar.css b/inst/css/sidebar.css
index 60a9380f7d..d0e21ff7ca 100644
--- a/inst/css/sidebar.css
+++ b/inst/css/sidebar.css
@@ -5,10 +5,13 @@
 .wunder_bar_button {
   font-size: 16px;
   padding: 8px !important;
-  float: right !important;
   background-color: transparent !important;
 }
 
+.filter_hamburger {
+  float: right !important;
+}
+
 .badge-count {
   padding-left: 1em;
   padding-right: 1em;
diff --git a/inst/css/validation.css b/inst/css/validation.css
new file mode 100644
index 0000000000..965b982eae
--- /dev/null
+++ b/inst/css/validation.css
@@ -0,0 +1,50 @@
+/* adding boarder to the validated input */
+.teal_validated:has(.shiny-output-error) {
+  border: 1px solid red;
+  border-radius: 4px;
+}
+
+.teal_validated:has(.teal-output-warning) {
+  border: 1px solid orange;
+  border-radius: 4px;
+}
+
+
+.teal_validated .teal-output-warning {
+  color: #888;
+}
+
+.teal_validated .shiny-output-error,
+.teal_validated .teal-output-warning {
+  margin-top: 0.5em;
+}
+
+.teal_validated .teal-output-warning::before {
+  content: "\26A0\FE0F";
+}
+
+.teal_validated .shiny-output-error::before {
+  content: "\1F6A8";
+}
+
+.teal_primary_col .shiny-output-error::before {
+  content: "\1F6A8";
+}
+
+.teal_primary_col .teal-output-warning::before {
+  content: "\26A0\FE0F";
+}
+
+.teal_primary_col .teal_validated:has(.shiny-output-error),
+.teal_primary_col .teal_validated:has(.teal-output-warning) {
+  margin: 1em 0 1em 0;
+  padding: .5em 0 .5em .5em;
+}
+
+.teal_primary_col > .teal_validated:has(.teal-output-warning),
+.teal_primary_col > .teal_validated:has(.shiny-output-error) {
+  width: 100%;
+  background-color: rgba(223, 70, 97, 0.1);
+  border: 1px solid red;
+  padding: 1em;
+}
diff --git a/inst/js/init.js b/inst/js/init.js
index 01aba99b0b..2bf5a1f179 100644
--- a/inst/js/init.js
+++ b/inst/js/init.js
@@ -3,3 +3,6 @@
 
 // this code alows the show R code "copy to clipbaord" button to work
 var clipboard = new ClipboardJS('.btn[data-clipboard-target]');
+
+
+
diff --git a/inst/js/sidebar.js b/inst/js/sidebar.js
index 52e8625bfd..20095c2468 100644
--- a/inst/js/sidebar.js
+++ b/inst/js/sidebar.js
@@ -3,44 +3,32 @@
 resize is placed at end of functions
 b/c in embedded apps it will throw errors that cause the function to exit early
 */
-var filter_open = true;
-const hideSidebar = () => {
-  $(".teal_secondary_col").css("display", "none");
-  $(".teal_primary_col")
+var filter_open = {};
+const hideSidebar = (tabpanel_wrapper) => {
+  $(`#${tabpanel_wrapper} .teal_secondary_col`).fadeOut(1);
+  $(`#${tabpanel_wrapper} .teal_primary_col`)
     .removeClass("col-sm-9")
     .addClass("col-sm-12");
-  $(".teal_primary_col").trigger("resize");
 };
-const showSidebar = () => {
-  $(".teal_primary_col")
+const showSidebar = (tabpanel_wrapper) => {
+  $(`#${tabpanel_wrapper} .teal_primary_col`)
     .removeClass("col-sm-12")
     .addClass("col-sm-9");
-  setTimeout(
-    () => {
-      $(".teal_secondary_col").css("display", "block");
-    },
-    600);
-  $(".teal_primary_col").trigger("resize");
+  $(`#${tabpanel_wrapper} .teal_secondary_col`).fadeIn(650);
+  $(`#${tabpanel_wrapper} .teal_secondary_col`).trigger("shown");
 };
-const toggleFilterPanel = () => {
-  if (filter_open && !$(".teal_secondary_col").is(':visible')) {
-    showSidebar();
+const toggleFilterPanel = (tabpanel_wrapper) => {
+  if (filter_open[tabpanel_wrapper] === undefined) {
+    filter_open[tabpanel_wrapper] = true;
+  }
+  if (
+    filter_open[tabpanel_wrapper] &&
+    !$(`#${tabpanel_wrapper} .teal_secondary_col`).is(":visible")
+  ) {
+    showSidebar(tabpanel_wrapper);
     return;
   }
-  filter_open = !filter_open;
-  if (filter_open) showSidebar();
-  else hideSidebar();
-};
-
-// Function to hide filter panel and disable the burger button
-const handleNoActiveDatasets = () => {
-  $(".filter_hamburger").addClass("hidden");
-  $(".filter_manager_button").addClass("hidden");
-  hideSidebar();
+  filter_open[tabpanel_wrapper] = !filter_open[tabpanel_wrapper];
+  if (filter_open[tabpanel_wrapper]) showSidebar(tabpanel_wrapper);
+  else hideSidebar(tabpanel_wrapper);
 };
-// Function to show filter panel and enable the burger button
-const handleActiveDatasetsPresent = () => {
-  $(".filter_hamburger").removeClass("hidden");
-  $(".filter_manager_button").removeClass("hidden");
-  if (filter_open) showSidebar();
-}
diff --git a/inst/js/togglePanelItems.js b/inst/js/togglePanelItems.js
new file mode 100644
index 0000000000..f3ebbca038
--- /dev/null
+++ b/inst/js/togglePanelItems.js
@@ -0,0 +1,56 @@
+// When invoked it adds the setClass and removes the removeClass from the element.
+function setAndRemoveClass(element, setClass, removeClass) {
+  if (typeof element === "string") {
+    element = document.querySelector(element);
+  }
+  element.classList.add(setClass);
+  element.classList.remove(removeClass);
+}
+
+// When invoked it toggles the class of the element.
+function toggleClass(element, class1, class2) {
+  if (typeof element === "string") {
+    element = document.querySelector(element);
+  }
+  if (element.classList.contains(class1)) {
+    setAndRemoveClass(element, class2, class1);
+  } else {
+    setAndRemoveClass(element, class1, class2);
+  }
+}
+
+// When invoked it shows the targetSelector element.
+function showPanelItem(targeSelector, duration = 400, easing = "slideInTop") {
+  $(`#${targeSelector}`).show(duration, easing);
+  $(`#${targeSelector}`).trigger("shown");
+}
+
+// When invoked it hides the targetSelector element.
+function hidePanelItem(targeSelector, duration = 400, easing = "slideOutLeft") {
+  $(`#${targeSelector}`).hide(duration, easing);
+}
+
+// When invoked it hides/shows targetSelectors elements
+// and changes class of element from class1 <-> class2
+function togglePanelItems(
+  element,
+  targetSelectors,
+  class1,
+  class2,
+  duration = 400,
+  easing = "swing"
+) {
+  if (!Array.isArray(targetSelectors)) {
+    targetSelectors = [targetSelectors];
+  }
+
+  targetSelectors.forEach((targetSelector) => {
+    if ($(`#${targetSelector}`).is(":visible")) {
+      hidePanelItem(targetSelector, duration, easing);
+    } else {
+      showPanelItem(targetSelector, duration, easing);
+    }
+  });
+
+  toggleClass(element, class1, class2);
+}
diff --git a/man/TealAppDriver.Rd b/man/TealAppDriver.Rd
index b6ce5491be..cfdd8110df 100644
--- a/man/TealAppDriver.Rd
+++ b/man/TealAppDriver.Rd
@@ -31,12 +31,15 @@ driving a teal application for performing interactions for \code{shinytest2} tes
 \item \href{#method-TealAppDriver-active_module_element}{\code{TealAppDriver$active_module_element()}}
 \item \href{#method-TealAppDriver-active_module_element_text}{\code{TealAppDriver$active_module_element_text()}}
 \item \href{#method-TealAppDriver-active_filters_ns}{\code{TealAppDriver$active_filters_ns()}}
+\item \href{#method-TealAppDriver-active_data_summary_ns}{\code{TealAppDriver$active_data_summary_ns()}}
+\item \href{#method-TealAppDriver-active_data_summary_element}{\code{TealAppDriver$active_data_summary_element()}}
 \item \href{#method-TealAppDriver-get_active_module_input}{\code{TealAppDriver$get_active_module_input()}}
 \item \href{#method-TealAppDriver-get_active_module_output}{\code{TealAppDriver$get_active_module_output()}}
 \item \href{#method-TealAppDriver-get_active_module_table_output}{\code{TealAppDriver$get_active_module_table_output()}}
 \item \href{#method-TealAppDriver-get_active_module_plot_output}{\code{TealAppDriver$get_active_module_plot_output()}}
 \item \href{#method-TealAppDriver-set_active_module_input}{\code{TealAppDriver$set_active_module_input()}}
 \item \href{#method-TealAppDriver-get_active_filter_vars}{\code{TealAppDriver$get_active_filter_vars()}}
+\item \href{#method-TealAppDriver-get_active_data_summary_table}{\code{TealAppDriver$get_active_data_summary_table()}}
 \item \href{#method-TealAppDriver-is_visible}{\code{TealAppDriver$is_visible()}}
 \item \href{#method-TealAppDriver-get_active_data_filters}{\code{TealAppDriver$get_active_data_filters()}}
 \item \href{#method-TealAppDriver-add_filter_var}{\code{TealAppDriver$add_filter_var()}}
@@ -98,6 +101,7 @@ Initialize a \code{TealAppDriver} object for testing a \code{teal} application.
   title = build_app_title(),
   header = tags$p(),
   footer = tags$p(),
+  landing_popup = NULL,
   timeout = rlang::missing_arg(),
   load_timeout = rlang::missing_arg(),
   ...
@@ -107,7 +111,7 @@ Initialize a \code{TealAppDriver} object for testing a \code{teal} application.
 \subsection{Arguments}{
 \if{html}{\out{
}} \describe{ -\item{\code{data, modules, filter, title, header, footer}}{arguments passed to \code{init}} +\item{\code{data, modules, filter, title, header, footer, landing_popup}}{arguments passed to \code{init}} \item{\code{timeout}}{(\code{numeric}) Default number of milliseconds for any timeout or timeout_ parameter in the \code{TealAppDriver} class. @@ -309,6 +313,39 @@ Get the active shiny name space for interacting with the filter panel. } } \if{html}{\out{
}} +\if{html}{\out{}} +\if{latex}{\out{\hypertarget{method-TealAppDriver-active_data_summary_ns}{}}} +\subsection{Method \code{active_data_summary_ns()}}{ +Get the active shiny name space for interacting with the data-summary panel. +\subsection{Usage}{ +\if{html}{\out{
}}\preformatted{TealAppDriver$active_data_summary_ns()}\if{html}{\out{
}} +} + +\subsection{Returns}{ +(\code{string}) The active shiny name space of the data-summary component. +} +} +\if{html}{\out{
}} +\if{html}{\out{}} +\if{latex}{\out{\hypertarget{method-TealAppDriver-active_data_summary_element}{}}} +\subsection{Method \code{active_data_summary_element()}}{ +Get the active shiny name space bound with a custom \code{element} name. +\subsection{Usage}{ +\if{html}{\out{
}}\preformatted{TealAppDriver$active_data_summary_element(element)}\if{html}{\out{
}} +} + +\subsection{Arguments}{ +\if{html}{\out{
}} +\describe{ +\item{\code{element}}{\code{character(1)} custom element name.} +} +\if{html}{\out{
}} +} +\subsection{Returns}{ +(\code{string}) The active shiny name space of the component bound with the input \code{element}. +} +} +\if{html}{\out{
}} \if{html}{\out{}} \if{latex}{\out{\hypertarget{method-TealAppDriver-get_active_module_input}{}}} \subsection{Method \code{get_active_module_input()}}{ @@ -429,6 +466,19 @@ Get the active datasets that can be accessed via the filter panel of the current \if{html}{\out{
}}\preformatted{TealAppDriver$get_active_filter_vars()}\if{html}{\out{
}} } +} +\if{html}{\out{
}} +\if{html}{\out{}} +\if{latex}{\out{\hypertarget{method-TealAppDriver-get_active_data_summary_table}{}}} +\subsection{Method \code{get_active_data_summary_table()}}{ +Get the active data summary table +\subsection{Usage}{ +\if{html}{\out{
}}\preformatted{TealAppDriver$get_active_data_summary_table()}\if{html}{\out{
}} +} + +\subsection{Returns}{ +\code{data.frame} +} } \if{html}{\out{
}} \if{html}{\out{}} diff --git a/man/calculate_hashes.Rd b/man/calculate_hashes.Rd deleted file mode 100644 index 12215b5319..0000000000 --- a/man/calculate_hashes.Rd +++ /dev/null @@ -1,20 +0,0 @@ -% Generated by roxygen2: do not edit by hand -% Please edit documentation in R/module_nested_tabs.R -\name{calculate_hashes} -\alias{calculate_hashes} -\title{Get the hash of a dataset} -\usage{ -calculate_hashes(datanames, datasets) -} -\arguments{ -\item{datanames}{(\code{character}) names of datasets} - -\item{datasets}{(\code{FilteredData}) object holding the data} -} -\value{ -A list of hashes per dataset. -} -\description{ -Get the hash of a dataset -} -\keyword{internal} diff --git a/man/dot-add_signature_to_data.Rd b/man/dot-add_signature_to_data.Rd new file mode 100644 index 0000000000..b7e242c32e --- /dev/null +++ b/man/dot-add_signature_to_data.Rd @@ -0,0 +1,18 @@ +% Generated by roxygen2: do not edit by hand +% Please edit documentation in R/module_init_data.R +\name{.add_signature_to_data} +\alias{.add_signature_to_data} +\title{Adds signature protection to the \code{datanames} in the data} +\usage{ +.add_signature_to_data(data) +} +\arguments{ +\item{data}{(\code{teal_data})} +} +\value{ +\code{teal_data} with additional code that has signature of the \code{datanames} +} +\description{ +Adds signature protection to the \code{datanames} in the data +} +\keyword{internal} diff --git a/man/dot-datasets_to_data.Rd b/man/dot-datasets_to_data.Rd deleted file mode 100644 index 36077d1795..0000000000 --- a/man/dot-datasets_to_data.Rd +++ /dev/null @@ -1,22 +0,0 @@ -% Generated by roxygen2: do not edit by hand -% Please edit documentation in R/module_nested_tabs.R -\name{.datasets_to_data} -\alias{.datasets_to_data} -\title{Convert \code{FilteredData} to reactive list of datasets of the \code{teal_data} type.} -\usage{ -.datasets_to_data(module, datasets) -} -\arguments{ -\item{module}{(\code{teal_module}) module where needed filters are taken from} - -\item{datasets}{(\code{FilteredData}) object where needed data are taken from} -} -\value{ -A \code{teal_data} object. -} -\description{ -Converts \code{FilteredData} object to \code{teal_data} object containing datasets needed for a specific module. -Please note that if a module needs a dataset which has a parent, then the parent will also be returned. -A hash per \code{dataset} is calculated internally and returned in the code. -} -\keyword{internal} diff --git a/man/dot-fallback_on_failure.Rd b/man/dot-fallback_on_failure.Rd new file mode 100644 index 0000000000..5d8f168e2d --- /dev/null +++ b/man/dot-fallback_on_failure.Rd @@ -0,0 +1,25 @@ +% Generated by roxygen2: do not edit by hand +% Please edit documentation in R/module_teal_data.R +\name{.fallback_on_failure} +\alias{.fallback_on_failure} +\title{Fallback on failure} +\usage{ +.fallback_on_failure(this, that, label) +} +\arguments{ +\item{this}{(\code{reactive}) Current reactive.} + +\item{that}{(\code{reactive}) Previous reactive.} + +\item{label}{(\code{character}) Label for identifying problematic \code{teal_data_module} transform in logging.} +} +\value{ +\code{reactive} \code{teal_data} +} +\description{ +Function returns the previous reactive if the current reactive is invalid (throws error or returns NULL). +Application: In \code{teal} we try to prevent the error from being thrown and instead we replace failing +transform module data output with data input from the previous module (or from previous \code{teal} reactive +tree elements). +} +\keyword{internal} diff --git a/man/dot-get_hashes_code.Rd b/man/dot-get_hashes_code.Rd new file mode 100644 index 0000000000..6283d2dca9 --- /dev/null +++ b/man/dot-get_hashes_code.Rd @@ -0,0 +1,20 @@ +% Generated by roxygen2: do not edit by hand +% Please edit documentation in R/module_init_data.R +\name{.get_hashes_code} +\alias{.get_hashes_code} +\title{Get code that tests the integrity of the reproducible data} +\usage{ +.get_hashes_code(data, datanames = .teal_data_datanames(data)) +} +\arguments{ +\item{data}{(\code{teal_data}) object holding the data} + +\item{datanames}{(\code{character}) names of \code{datasets}} +} +\value{ +A character vector with the code lines. +} +\description{ +Get code that tests the integrity of the reproducible data +} +\keyword{internal} diff --git a/man/example_module.Rd b/man/example_module.Rd index 6aa96b0325..8227e44e45 100644 --- a/man/example_module.Rd +++ b/man/example_module.Rd @@ -4,7 +4,11 @@ \alias{example_module} \title{An example \code{teal} module} \usage{ -example_module(label = "example teal module", datanames = "all") +example_module( + label = "example teal module", + datanames = "all", + transformers = list() +) } \arguments{ \item{label}{(\code{character(1)}) Label shown in the navigation item for the module or module group. @@ -15,6 +19,13 @@ filter panel will automatically update the shown filters to include only filters in the listed datasets. \code{NULL} will hide the filter panel, and the keyword \code{"all"} will show filters of all datasets. \code{datanames} also determines a subset of datasets which are appended to the \code{data} argument in server function.} + +\item{transformers}{(\code{list} of \code{teal_data_module}) that will be applied to transform the data. +Each transform module UI will appear in the \code{teal} application, unless the \code{custom_ui} attribute is set on the list. +If so, the module developer is responsible to display the UI in the module itself. + +When the transformation does not have sufficient input data, the resulting data will fallback +to the last successful transform or, in case there are none, to the filtered data.} } \value{ A \code{teal} module which can be included in the \code{modules} argument to \code{\link[=init]{init()}}. diff --git a/man/figures/filter_state_reactivity.jpg b/man/figures/filter_state_reactivity.jpg deleted file mode 100644 index cd646939cc..0000000000 Binary files a/man/figures/filter_state_reactivity.jpg and /dev/null differ diff --git a/man/figures/module_nested_tabs.jpg b/man/figures/module_nested_tabs.jpg deleted file mode 100644 index d86ba39127..0000000000 Binary files a/man/figures/module_nested_tabs.jpg and /dev/null differ diff --git a/man/figures/notification.jpg b/man/figures/notification.jpg deleted file mode 100644 index a80c024f69..0000000000 Binary files a/man/figures/notification.jpg and /dev/null differ diff --git a/man/filter_manager_module_srv.Rd b/man/filter_manager_module_srv.Rd deleted file mode 100644 index e00216a5d1..0000000000 --- a/man/filter_manager_module_srv.Rd +++ /dev/null @@ -1,37 +0,0 @@ -% Generated by roxygen2: do not edit by hand -% Please edit documentation in R/module_filter_manager.R -\name{filter_manager_module_srv} -\alias{filter_manager_module_srv} -\title{Module specific filter manager} -\usage{ -filter_manager_module_srv(id, module_fd, slices_global) -} -\arguments{ -\item{id}{(\code{character(1)}) -\code{shiny} module id.} - -\item{module_fd}{(\code{FilteredData}) -Object containing the data to be filtered in a single \code{teal} module.} - -\item{slices_global}{(\code{reactiveVal}) -stores \code{teal_slices} with all available filters; allows the following actions: -\itemize{ -\item to disable/enable a specific filter in a module -\item to restore saved filter settings -\item to save current filter panel settings -}} -} -\value{ -A \code{reactive} expression containing a \code{teal_slices} with the slices active in this module. -} -\description{ -Tracks filter states in a single module. -} -\details{ -This module tracks the state of a single \code{FilteredData} object and global \code{teal_slices} -and updates both objects as necessary. Filter states added in different modules -Filter states added any individual module are added to global \code{teal_slices} -and from there become available in other modules -by setting \code{private$available_teal_slices} in each \code{FilteredData}. -} -\keyword{internal} diff --git a/man/get_code_tdata.Rd b/man/get_code_tdata.Rd deleted file mode 100644 index cc9f336dd3..0000000000 --- a/man/get_code_tdata.Rd +++ /dev/null @@ -1,17 +0,0 @@ -% Generated by roxygen2: do not edit by hand -% Please edit documentation in R/tdata.R -\name{get_code_tdata} -\alias{get_code_tdata} -\title{Wrapper for \code{get_code.tdata}} -\usage{ -get_code_tdata(data) -} -\arguments{ -\item{data}{(\code{tdata}) object} -} -\value{ -(\code{character}) code used in the \code{tdata} object. -} -\description{ -This wrapper is to be used by downstream packages to extract the code of a \code{tdata} object. -} diff --git a/man/get_datasets_code.Rd b/man/get_datasets_code.Rd deleted file mode 100644 index d8108ec921..0000000000 --- a/man/get_datasets_code.Rd +++ /dev/null @@ -1,27 +0,0 @@ -% Generated by roxygen2: do not edit by hand -% Please edit documentation in R/get_rcode_utils.R -\name{get_datasets_code} -\alias{get_datasets_code} -\title{Get datasets code} -\usage{ -get_datasets_code(datanames, datasets, hashes) -} -\arguments{ -\item{datanames}{(\code{character}) names of datasets to extract code from} - -\item{datasets}{(\code{FilteredData}) object} - -\item{hashes}{named (\code{list}) of hashes per dataset} -} -\value{ -Character string concatenated from the following elements: -\itemize{ -\item data pre-processing code (from \code{data} argument in \code{init}) -\item hash check of loaded objects -\item filter code (if any) -} -} -\description{ -Retrieve complete code to create, verify, and filter a dataset. -} -\keyword{internal} diff --git a/man/get_metadata.Rd b/man/get_metadata.Rd deleted file mode 100644 index 0f45078ccd..0000000000 --- a/man/get_metadata.Rd +++ /dev/null @@ -1,25 +0,0 @@ -% Generated by roxygen2: do not edit by hand -% Please edit documentation in R/tdata.R -\name{get_metadata} -\alias{get_metadata} -\alias{get_metadata.tdata} -\alias{get_metadata.default} -\title{Function to get metadata from a \code{tdata} object} -\usage{ -get_metadata(data, dataname) - -\method{get_metadata}{tdata}(data, dataname) - -\method{get_metadata}{default}(data, dataname) -} -\arguments{ -\item{data}{(\code{tdata} - object) to extract the data from} - -\item{dataname}{(\code{character(1)}) the dataset name whose metadata is requested} -} -\value{ -Either list of metadata or NULL if no metadata. -} -\description{ -Function to get metadata from a \code{tdata} object -} diff --git a/man/init.Rd b/man/init.Rd index 2f0e34d2f5..974dd9801a 100644 --- a/man/init.Rd +++ b/man/init.Rd @@ -11,7 +11,8 @@ init( title = build_app_title(), header = tags$p(), footer = tags$p(), - id = character(0) + id = character(0), + landing_popup = NULL ) } \arguments{ @@ -41,6 +42,9 @@ The footer of the app.} \item{id}{(\code{character}) optional string specifying the \code{shiny} module id in cases it is used as a \code{shiny} module rather than a standalone \code{shiny} app. This is a legacy feature.} + +\item{landing_popup}{(\code{teal_module}) optional +A \code{landing_popup_module} to show up as soon as the teal app is initialized.} } \value{ Named list containing server and UI functions. diff --git a/man/join_keys.tdata.Rd b/man/join_keys.tdata.Rd deleted file mode 100644 index ef18d05686..0000000000 --- a/man/join_keys.tdata.Rd +++ /dev/null @@ -1,16 +0,0 @@ -% Generated by roxygen2: do not edit by hand -% Please edit documentation in R/tdata.R -\name{join_keys.tdata} -\alias{join_keys.tdata} -\title{Extract \code{join_keys} from \code{tdata}} -\usage{ -\method{join_keys}{tdata}(data, ...) -} -\arguments{ -\item{data}{(\code{tdata}) object} - -\item{...}{Additional arguments (not used)} -} -\description{ -Extract \code{join_keys} from \code{tdata} -} diff --git a/man/landing_popup_module.Rd b/man/landing_popup_module.Rd index 2979af6555..af7c79c726 100644 --- a/man/landing_popup_module.Rd +++ b/man/landing_popup_module.Rd @@ -34,11 +34,11 @@ The dialog blocks access to the application and must be closed with a button bef app1 <- init( data = teal_data(iris = iris), modules = modules( - landing_popup_module( - content = "A place for the welcome message or a disclaimer statement.", - buttons = modalButton("Proceed") - ), example_module() + ), + landing_popup = landing_popup_module( + content = "A place for the welcome message or a disclaimer statement.", + buttons = modalButton("Proceed") ) ) if (interactive()) { @@ -48,21 +48,21 @@ if (interactive()) { app2 <- init( data = teal_data(iris = iris), modules = modules( - landing_popup_module( - title = "Welcome", - content = tags$b( - "A place for the welcome message or a disclaimer statement.", - style = "color: red;" - ), - buttons = tagList( - modalButton("Proceed"), - actionButton("read", "Read more", - onclick = "window.open('http://google.com', '_blank')" - ), - actionButton("close", "Reject", onclick = "window.close()") - ) - ), example_module() + ), + landing_popup = landing_popup_module( + title = "Welcome", + content = tags$b( + "A place for the welcome message or a disclaimer statement.", + style = "color: red;" + ), + buttons = tagList( + modalButton("Proceed"), + actionButton("read", "Read more", + onclick = "window.open('http://google.com', '_blank')" + ), + actionButton("close", "Reject", onclick = "window.close()") + ) ) ) diff --git a/man/matrix_to_mapping.Rd b/man/matrix_to_mapping.Rd deleted file mode 100644 index 4683866db0..0000000000 --- a/man/matrix_to_mapping.Rd +++ /dev/null @@ -1,22 +0,0 @@ -% Generated by roxygen2: do not edit by hand -% Please edit documentation in R/module_snapshot_manager.R -\name{matrix_to_mapping} -\alias{matrix_to_mapping} -\title{Convert mapping matrix to filter mapping specification.} -\usage{ -matrix_to_mapping(mapping_matrix) -} -\arguments{ -\item{mapping_matrix}{(\code{data.frame}) of logical vectors where -columns represent modules and row represent \code{teal_slice}s} -} -\value{ -Named \code{list} like that in the \code{mapping} attribute of a \code{teal_slices} object. -} -\description{ -Transform a mapping matrix, i.e. a data frame that maps each filter state to each module, -to a list specification like the one used in the \code{mapping} attribute of \code{teal_slices}. -Global filters are gathered in one list element. -If a module has no active filters but the global ones, it will not be mentioned in the output. -} -\keyword{internal} diff --git a/man/module_bookmark_manager.Rd b/man/module_bookmark_manager.Rd index ca7a892cb5..1a6a1e1d8b 100644 --- a/man/module_bookmark_manager.Rd +++ b/man/module_bookmark_manager.Rd @@ -2,22 +2,31 @@ % Please edit documentation in R/module_bookmark_manager.R \name{module_bookmark_manager} \alias{module_bookmark_manager} -\alias{bookmark_manager_ui} \alias{bookmark} \alias{bookmark_manager} \alias{bookmark_manager_module} -\alias{bookmark_manager_srv} +\alias{ui_bookmark_panel} +\alias{srv_bookmark_panel} +\alias{get_bookmarking_option} +\alias{need_bookmarking} \title{App state management.} \usage{ -bookmark_manager_ui(id) +ui_bookmark_panel(id, modules) -bookmark_manager_srv(id, modules) +srv_bookmark_panel(id, modules) + +get_bookmarking_option() + +need_bookmarking(modules) } \arguments{ -\item{id}{(\code{character(1)}) -module id} +\item{id}{(\code{character}) optional +string specifying the \code{shiny} module id in cases it is used as a \code{shiny} module +rather than a standalone \code{shiny} app. This is a legacy feature.} -\item{modules}{(\code{teal_modules}) object containing the output modules which +\item{modules}{(\code{list} or \code{teal_modules} or \code{teal_module}) +nested list of \code{teal_modules} or \code{teal_module} objects or a single +\code{teal_modules} or \code{teal_module} object. These are the specific output modules which will be displayed in the \code{teal} application. See \code{\link[=modules]{modules()}} and \code{\link[=module]{module()}} for more details.} } @@ -33,7 +42,7 @@ Capture and restore the global (app) input state. This module introduces bookmarks into \code{teal} apps: the \code{shiny} bookmarking mechanism becomes enabled and server-side bookmarks can be created. -The bookmark manager presents a button with the bookmark icon and is placed in the \code{\link{wunder_bar}}. +The bookmark manager presents a button with the bookmark icon and is placed in the tab-bar. When clicked, the button creates a bookmark and opens a modal which displays the bookmark URL. \code{teal} does not guarantee that all modules (\code{teal_module} objects) are bookmarkable. diff --git a/man/module_data_summary.Rd b/man/module_data_summary.Rd new file mode 100644 index 0000000000..2bc009a17a --- /dev/null +++ b/man/module_data_summary.Rd @@ -0,0 +1,66 @@ +% Generated by roxygen2: do not edit by hand +% Please edit documentation in R/module_data_summary.R +\name{module_data_summary} +\alias{module_data_summary} +\alias{ui_data_summary} +\alias{srv_data_summary} +\alias{get_filter_overview} +\alias{get_object_filter_overview} +\alias{get_object_filter_overview_array} +\alias{get_object_filter_overview_MultiAssayExperiment} +\title{Data summary} +\usage{ +ui_data_summary(id) + +srv_data_summary(id, teal_data) + +get_filter_overview(teal_data) + +get_object_filter_overview( + filtered_data, + unfiltered_data, + dataname, + subject_keys +) + +get_object_filter_overview_array( + filtered_data, + unfiltered_data, + dataname, + subject_keys +) + +get_object_filter_overview_MultiAssayExperiment( + filtered_data, + unfiltered_data, + dataname +) +} +\arguments{ +\item{id}{(\code{character(1)}) +\code{shiny} module instance id.} + +\item{teal_data}{(\code{reactive} returning \code{teal_data})} + +\item{filtered_data}{(\code{list}) of filtered objects} + +\item{unfiltered_data}{(\code{list}) of unfiltered objects} + +\item{dataname}{(\code{character(1)})} +} +\value{ +\code{NULL}. +} +\description{ +Module and its utils to display the number of rows and subjects in the filtered and unfiltered data. +} +\details{ +Handling different data classes: +\code{get_object_filter_overview()} is a pseudo S3 method which has variants for: +\itemize{ +\item \code{array} (\code{data.frame}, \code{DataFrame}, \code{array}, \code{Matrix} and \code{SummarizedExperiment}): Method variant +can be applied to any two-dimensional objects on which \code{\link[=ncol]{ncol()}} can be used. +\item \code{MultiAssayExperiment}: for which summary contains counts for \code{colData} and all \code{experiments}. +} +} +\keyword{internal} diff --git a/man/module_filter_data.Rd b/man/module_filter_data.Rd new file mode 100644 index 0000000000..a7c6915e9c --- /dev/null +++ b/man/module_filter_data.Rd @@ -0,0 +1,53 @@ +% Generated by roxygen2: do not edit by hand +% Please edit documentation in R/module_filter_data.R +\name{module_filter_data} +\alias{module_filter_data} +\alias{ui_filter_data} +\alias{srv_filter_data} +\alias{.make_filtered_teal_data} +\alias{.observe_active_filter_changed} +\title{Filter panel module in teal} +\usage{ +ui_filter_data(id) + +srv_filter_data(id, datasets, active_datanames, data_rv, is_active) + +.make_filtered_teal_data(modules, data, datasets = NULL, datanames) + +.observe_active_filter_changed(datasets, is_active, active_datanames, data_rv) +} +\arguments{ +\item{id}{(\code{character}) optional +string specifying the \code{shiny} module id in cases it is used as a \code{shiny} module +rather than a standalone \code{shiny} app. This is a legacy feature.} + +\item{datasets}{(\code{reactive} returning \code{FilteredData} or \code{NULL}) +When \code{datasets} is passed from the parent module (\code{srv_teal}) then \code{dataset} is a singleton +which implies in filter-panel to be "global". When \code{NULL} then filter-panel is "module-specific".} + +\item{active_datanames}{(\code{reactive} returning \code{character}) this module's data names} + +\item{data_rv}{(\code{reactive} returning \code{teal_data})} + +\item{modules}{(\code{list} or \code{teal_modules} or \code{teal_module}) +nested list of \code{teal_modules} or \code{teal_module} objects or a single +\code{teal_modules} or \code{teal_module} object. These are the specific output modules which +will be displayed in the \code{teal} application. See \code{\link[=modules]{modules()}} and \code{\link[=module]{module()}} for +more details.} +} +\value{ +A \code{eventReactive} containing \code{teal_data} containing filtered objects and filter code. +\code{eventReactive} triggers only if all conditions are met: +\itemize{ +\item tab is selected (\code{is_active}) +\item when filters are changed (\code{get_filter_expr} is different than previous) +} +} +\description{ +Creates filter panel module from \code{teal_data} object and returns \code{teal_data}. It is build in a way +that filter panel changes and anything what happens before (e.g. \code{\link{module_init_data}}) is triggering +further reactive events only if something has changed and if the module is visible. Thanks to +this special implementation all modules' data are recalculated only for those modules which are +currently displayed. +} +\keyword{internal} diff --git a/man/module_filter_manager.Rd b/man/module_filter_manager.Rd index 1d0e5af2ef..fc06e5be63 100644 --- a/man/module_filter_manager.Rd +++ b/man/module_filter_manager.Rd @@ -1,52 +1,85 @@ % Generated by roxygen2: do not edit by hand % Please edit documentation in R/module_filter_manager.R +\docType{class} +\encoding{UTF-8} \name{module_filter_manager} \alias{module_filter_manager} -\alias{filter_manager_ui} -\alias{filter_manager} -\alias{filter_manager_module} -\alias{filter_manager_srv} +\alias{ui_filter_manager_panel} +\alias{srv_filter_manager_panel} +\alias{ui_filter_manager} +\alias{srv_filter_manager} +\alias{srv_module_filter_manager} +\alias{.slicesGlobal-class} +\alias{.slicesGlobal} \title{Manage multiple \code{FilteredData} objects} \usage{ -filter_manager_ui(id) +ui_filter_manager_panel(id) -filter_manager_srv(id, datasets, filter) +srv_filter_manager_panel(id, slices_global) + +ui_filter_manager(id) + +srv_filter_manager(id, slices_global) + +srv_module_filter_manager(id, module_fd, slices_global) } \arguments{ \item{id}{(\code{character(1)}) \code{shiny} module instance id.} -\item{datasets}{(named \code{list}) -A list, possibly nested, of \code{FilteredData} objects. -Each \code{FilteredData} will be served to one module in the \code{teal} application. -The structure of the list must reflect the nesting of modules in tabs -and the names of the list must match the labels of their respective modules.} +\item{slices_global}{(\code{reactiveVal}) +containing \code{teal_slices}.} -\item{filter}{(\code{teal_slices}) -Specifies the initial filter using \code{\link[=teal_slices]{teal_slices()}}.} +\item{module_fd}{(\code{FilteredData}) +Object containing the data to be filtered in a single \code{teal} module.} } \value{ -A \code{list} containing: +Module returns a \code{slices_global} (\code{reactiveVal}) containing a \code{teal_slices} object with mapping. +} +\description{ +Oversee filter states across the entire application. +} + +\section{Slices global}{ -objects used by other manager modules +The key role in maintaining the module-specific filter states is played by the \code{.slicesGlobal} +object. It is a reference class that holds the following fields: \itemize{ -\item \code{datasets_flat}: named list of \code{FilteredData} objects, -\item \code{mapping_matrix}: \code{reactive} containing a \code{data.frame}, -\item \code{slices_global}: \code{reactiveVal} containing a \code{teal_slices} object, +\item \code{all_slices} (\code{reactiveVal}) - reactive value containing all filters registered in an app. +\item \code{module_slices_api} (\code{reactiveValues}) - reactive field containing references to each modules' +\code{FilteredData} object methods. At this moment it is used only in \code{srv_filter_manager} to display +the filter states in a table combining informations from \code{all_slices} and from +\code{FilteredData$get_available_teal_slices()}. } -objects used for testing +During a session only new filters are added to \code{all_slices} unless \code{\link{module_snapshot_manager}} is +used to restore previous state. Filters from \code{all_slices} can be activated or deactivated in a +module which is linked (both ways) by \code{attr(, "mapping")} so that: \itemize{ -\item modules_out: \code{list} of \code{reactive}s, each holding a \code{teal_slices}, as returned by \code{filter_manager_module_srv}. +\item If module's filter is added or removed in its \code{FilteredData} object, this information is passed +to \code{SlicesGlobal} which updates \code{attr(, "mapping")} accordingly. +\item When mapping changes in a \code{SlicesGlobal}, filters are set or removed from module's +\code{FilteredData}. } } -\description{ -Oversee filter states across the entire application. + +\section{Filter manager}{ + +Filter-manager is split into two parts: +\enumerate{ +\item \code{ui/srv_filter_manager_panel} - Called once for the whole app. This module observes changes in +the filters in \code{slices_global} and displays them in a table utilizing information from \code{mapping}: } -\details{ -This module observes changes in the filters of each \code{FilteredData} object -and keeps track of all filters used. A mapping of filters to modules -is kept in the \code{mapping_matrix} object (which is actually a \code{data.frame}) -that tracks which filters (rows) are active in which modules (columns). +\itemize{ +\item (\code{TRUE}) - filter is active in the module +\item (\code{FALSE}) - filter is inactive in the module +\item (\code{NA}) - filter is not available in the module +} +\enumerate{ +\item \code{ui/srv_module_filter_manager} - Called once for each \code{teal_module}. Handling filter states +for of single module and keeping module \code{FilteredData} consistent with \code{slices_global}, so that +local filters are always reflected in the \code{slices_global} and its mapping and vice versa. } +} + \keyword{internal} diff --git a/man/module_init_data.Rd b/man/module_init_data.Rd new file mode 100644 index 0000000000..8370aefcd0 --- /dev/null +++ b/man/module_init_data.Rd @@ -0,0 +1,68 @@ +% Generated by roxygen2: do not edit by hand +% Please edit documentation in R/module_init_data.R +\name{module_init_data} +\alias{module_init_data} +\alias{ui_init_data} +\alias{srv_init_data} +\title{Data Module for teal} +\usage{ +ui_init_data(id, data) + +srv_init_data(id, data, modules, filter = teal_slices()) +} +\arguments{ +\item{id}{(\code{character}) optional +string specifying the \code{shiny} module id in cases it is used as a \code{shiny} module +rather than a standalone \code{shiny} app. This is a legacy feature.} + +\item{data}{(\code{teal_data}, \code{teal_data_module}, or \code{reactive} returning \code{teal_data}) +The \code{ui} component of this module does not require \code{data} if \code{teal_data_module} is not provided. +The \code{data} argument in the \code{ui} is included solely for the \verb{$ui} function of the +\code{teal_data_module}. Otherwise, it can be disregarded, ensuring that \code{ui_teal} does not depend on +the reactive data of the enclosing application.} + +\item{modules}{(\code{list} or \code{teal_modules} or \code{teal_module}) +nested list of \code{teal_modules} or \code{teal_module} objects or a single +\code{teal_modules} or \code{teal_module} object. These are the specific output modules which +will be displayed in the \code{teal} application. See \code{\link[=modules]{modules()}} and \code{\link[=module]{module()}} for +more details.} + +\item{filter}{(\code{teal_slices}) +Specifies the initial filter using \code{\link[=teal_slices]{teal_slices()}}.} +} +\value{ +A \code{reactive} object that returns: +\itemize{ +\item \code{teal_data} when the object is validated +\item \code{shiny.silent.error} when not validated. +} +} +\description{ +This module manages the \code{data} argument for \code{srv_teal}. The \code{teal} framework uses \code{\link[=teal_data]{teal_data()}}, +which can be provided in various ways: +\enumerate{ +\item Directly as a \code{\link[teal.data:teal_data]{teal.data::teal_data()}} object. This will automatically convert it into a \code{reactive} \code{teal_data}. +\item As a \code{reactive} object that returns a \code{\link[teal.data:teal_data]{teal.data::teal_data()}} object. +} +} +\details{ +\subsection{Reactive \code{teal_data}:}{ + +The data in the application can be reactively updated, prompting \code{\link[=srv_teal]{srv_teal()}} to rebuild the +content accordingly. There are two methods for creating interactive \code{teal_data}: +\enumerate{ +\item Using a \code{reactive} object provided from outside the \code{teal} application. In this scenario, +reactivity is controlled by an external module, and \code{srv_teal} responds to changes. +\item Using \code{\link[=teal_data_module]{teal_data_module()}}, which is embedded within the \code{teal} application, allowing data to +be resubmitted by the user as needed. +} + +Since the server of \code{\link[=teal_data_module]{teal_data_module()}} must return a \code{reactive} \code{teal_data} object, both +methods (1 and 2) produce the same reactive behavior within a \code{teal} application. The distinction +lies in data control: the first method involves external control, while the second method +involves control from a custom module within the app. + +For more details, see \code{\link{module_teal_data}}. +} +} +\keyword{internal} diff --git a/man/module_nested_tabs.Rd b/man/module_nested_tabs.Rd deleted file mode 100644 index 238713117a..0000000000 --- a/man/module_nested_tabs.Rd +++ /dev/null @@ -1,140 +0,0 @@ -% Generated by roxygen2: do not edit by hand -% Please edit documentation in R/module_nested_tabs.R -\name{module_nested_tabs} -\alias{module_nested_tabs} -\alias{ui_nested_tabs} -\alias{ui_nested_tabs.default} -\alias{ui_nested_tabs.teal_modules} -\alias{ui_nested_tabs.teal_module} -\alias{srv_nested_tabs} -\alias{srv_nested_tabs.default} -\alias{srv_nested_tabs.teal_modules} -\alias{srv_nested_tabs.teal_module} -\title{Create a UI of nested tabs of \code{teal_modules}} -\usage{ -ui_nested_tabs( - id, - modules, - datasets, - depth = 0L, - is_module_specific = FALSE, - progress = NULL -) - -\method{ui_nested_tabs}{default}( - id, - modules, - datasets, - depth = 0L, - is_module_specific = FALSE, - progress = NULL -) - -\method{ui_nested_tabs}{teal_modules}( - id, - modules, - datasets, - depth = 0L, - is_module_specific = FALSE, - progress = NULL -) - -\method{ui_nested_tabs}{teal_module}( - id, - modules, - datasets, - depth = 0L, - is_module_specific = FALSE, - progress = NULL -) - -srv_nested_tabs( - id, - datasets, - modules, - is_module_specific = FALSE, - reporter = teal.reporter::Reporter$new() -) - -\method{srv_nested_tabs}{default}( - id, - datasets, - modules, - is_module_specific = FALSE, - reporter = teal.reporter::Reporter$new() -) - -\method{srv_nested_tabs}{teal_modules}( - id, - datasets, - modules, - is_module_specific = FALSE, - reporter = teal.reporter::Reporter$new() -) - -\method{srv_nested_tabs}{teal_module}( - id, - datasets, - modules, - is_module_specific = TRUE, - reporter = teal.reporter::Reporter$new() -) -} -\arguments{ -\item{id}{(\code{character(1)}) -module id} - -\item{modules}{(\code{teal_modules}) object containing the output modules which -will be displayed in the \code{teal} application. See \code{\link[=modules]{modules()}} and \code{\link[=module]{module()}} for -more details.} - -\item{datasets}{(named \code{list} of \code{FilteredData}) -object to store filter state and filtered datasets, shared across modules. For more -details see \code{\link[teal.slice:FilteredData]{teal.slice::FilteredData}}. Structure of the list must be the same as structure -of the \code{modules} argument and list names must correspond to the labels in \code{modules}. -When filter is not module-specific then list contains the same object in all elements.} - -\item{depth}{(\code{integer(1)}) -number which helps to determine depth of the modules nesting.} - -\item{is_module_specific}{(\code{logical(1)}) -flag determining if the filter panel is global or module-specific. -When set to \code{TRUE}, a filter panel is called inside of each module tab.} - -\item{progress}{(\code{Progress}) object from \code{shiny}} - -\item{reporter}{(\code{Reporter}) object from \code{teal.reporter}} -} -\value{ -Depending on the class of \code{modules}, \code{ui_nested_tabs} returns: -\itemize{ -\item \code{teal_module}: instantiated UI of the module. -\item \code{teal_modules}: \code{tabsetPanel} with each tab corresponding to recursively -calling this function on it. -} - -\code{srv_nested_tabs} returns a reactive which returns the active module that corresponds to the selected tab. -} -\description{ -Create a UI of nested tabs of \code{teal_modules} -} -\section{\code{ui_nested_tabs}}{ - -Each \code{teal_modules} is translated to a \code{tabsetPanel} and each -of its children is another tab-module called recursively. The UI of a -\code{teal_module} is obtained by calling its UI function. - -The \code{datasets} argument is required to resolve the \code{teal} arguments in an -isolated context (with respect to reactivity). -} - -\section{\code{srv_nested_tabs}}{ - -This module recursively calls all elements of \code{modules} and returns currently active one. -\itemize{ -\item \code{teal_module} returns self as a active module. -\item \code{teal_modules} also returns module active within self which is determined by the \code{input$active_tab}. -} -} - -\keyword{internal} diff --git a/man/module_snapshot_manager.Rd b/man/module_snapshot_manager.Rd index 80aa9fb44b..723fcd3ccb 100644 --- a/man/module_snapshot_manager.Rd +++ b/man/module_snapshot_manager.Rd @@ -2,28 +2,25 @@ % Please edit documentation in R/module_snapshot_manager.R \name{module_snapshot_manager} \alias{module_snapshot_manager} -\alias{snapshot_manager_ui} -\alias{snapshot} -\alias{snapshot_manager} -\alias{snapshot_manager_module} -\alias{snapshot_manager_srv} +\alias{ui_snapshot_manager_panel} +\alias{srv_snapshot_manager_panel} +\alias{ui_snapshot_manager} +\alias{srv_snapshot_manager} \title{Filter state snapshot management} \usage{ -snapshot_manager_ui(id) +ui_snapshot_manager_panel(id) -snapshot_manager_srv(id, slices_global, mapping_matrix, datasets) +srv_snapshot_manager_panel(id, slices_global) + +ui_snapshot_manager(id) + +srv_snapshot_manager(id, slices_global) } \arguments{ \item{id}{(\code{character(1)}) \code{shiny} module instance id.} \item{slices_global}{(\code{reactiveVal}) that contains a \code{teal_slices} object containing all \code{teal_slice}s existing in the app, both active and inactive.} - -\item{mapping_matrix}{(\code{reactive}) that contains a \code{data.frame} representation -of the mapping of filter state ids (rows) to modules labels (columns); -all columns are \code{logical} vectors.} - -\item{datasets}{non-nested (named \code{list}) of \code{FilteredData} objects.} } \value{ \code{list} containing the snapshot history, where each element is an unlisted \code{teal_slices} object. @@ -37,7 +34,7 @@ Snapshots allow the user to save the current filter state of the application for as well as to save it to file in order to share it with an app developer or other users, who in turn can upload it to their own session. -The snapshot manager is accessed with the camera icon in the \code{\link{wunder_bar}}. +The snapshot manager is accessed with the camera icon in the tabset bar. At the beginning of a session it presents three icons: a camera, an upload, and an circular arrow. Clicking the camera captures a snapshot, clicking the upload adds a snapshot from a file and applies the filter states therein, and clicking the arrow resets initial application state. diff --git a/man/module_tabs_with_filters.Rd b/man/module_tabs_with_filters.Rd deleted file mode 100644 index d91f72132f..0000000000 --- a/man/module_tabs_with_filters.Rd +++ /dev/null @@ -1,61 +0,0 @@ -% Generated by roxygen2: do not edit by hand -% Please edit documentation in R/module_tabs_with_filters.R -\name{module_tabs_with_filters} -\alias{module_tabs_with_filters} -\alias{ui_tabs_with_filters} -\alias{srv_tabs_with_filters} -\title{Add right filter panel into each of the top-level \code{teal_modules} UIs} -\usage{ -ui_tabs_with_filters( - id, - modules, - datasets, - filter = teal_slices(), - progress = NULL -) - -srv_tabs_with_filters( - id, - datasets, - modules, - reporter = teal.reporter::Reporter$new(), - filter = teal_slices() -) -} -\arguments{ -\item{id}{(\code{character(1)}) -module id} - -\item{modules}{(\code{teal_modules}) object containing the output modules which -will be displayed in the \code{teal} application. See \code{\link[=modules]{modules()}} and \code{\link[=module]{module()}} for -more details.} - -\item{datasets}{(named \code{list} of \code{FilteredData}) -object to store filter state and filtered datasets, shared across modules. For more -details see \code{\link[teal.slice:FilteredData]{teal.slice::FilteredData}}. Structure of the list must be the same as structure -of the \code{modules} argument and list names must correspond to the labels in \code{modules}. -When filter is not module-specific then list contains the same object in all elements.} - -\item{filter}{(\code{teal_slices}) -Specifies the initial filter using \code{\link[=teal_slices]{teal_slices()}}.} - -\item{progress}{(\code{Progress}) object from \code{shiny}} - -\item{reporter}{(\code{Reporter}) object from \code{teal.reporter}} -} -\value{ -A \code{shiny.tag.list} containing the main menu, placeholders for filters and placeholders for the \code{teal} modules. -} -\description{ -The \link{ui_nested_tabs} function returns a nested tabbed UI corresponding -to the nested modules. -This function adds the right filter panel to each main tab. -} -\details{ -The right filter panel's filter choices affect the \code{datasets} object. Therefore, -all modules using the same \code{datasets} share the same filters. - -This works with nested modules of depth greater than 2, though the filter -panel is inserted at the right of the modules at depth 1 and not at the leaves. -} -\keyword{internal} diff --git a/man/module_teal.Rd b/man/module_teal.Rd index 81167b194a..bbe43e80b0 100644 --- a/man/module_teal.Rd +++ b/man/module_teal.Rd @@ -4,25 +4,35 @@ \alias{module_teal} \alias{ui_teal} \alias{srv_teal} -\title{\code{teal} main app module} +\title{\code{teal} main module} \usage{ ui_teal( id, - splash_ui = tags$h2("Starting the Teal App"), + modules, + data = NULL, title = build_app_title(), header = tags$p(), footer = tags$p() ) -srv_teal(id, modules, teal_data_rv, filter = teal_slices()) +srv_teal(id, data, modules, filter = teal_slices()) } \arguments{ -\item{id}{(\code{character(1)}) -module id} +\item{id}{(\code{character}) optional +string specifying the \code{shiny} module id in cases it is used as a \code{shiny} module +rather than a standalone \code{shiny} app. This is a legacy feature.} -\item{splash_ui}{(\code{shiny.tag}) UI to display initially, -can be a splash screen or a \code{shiny} module UI. For the latter, see -\code{\link[=init]{init()}} about how to call the corresponding server function.} +\item{modules}{(\code{list} or \code{teal_modules} or \code{teal_module}) +nested list of \code{teal_modules} or \code{teal_module} objects or a single +\code{teal_modules} or \code{teal_module} object. These are the specific output modules which +will be displayed in the \code{teal} application. See \code{\link[=modules]{modules()}} and \code{\link[=module]{module()}} for +more details.} + +\item{data}{(\code{teal_data}, \code{teal_data_module}, or \code{reactive} returning \code{teal_data}) +The \code{ui} component of this module does not require \code{data} if \code{teal_data_module} is not provided. +The \code{data} argument in the \code{ui} is included solely for the \verb{$ui} function of the +\code{teal_data_module}. Otherwise, it can be disregarded, ensuring that \code{ui_teal} does not depend on +the reactive data of the enclosing application.} \item{title}{(\code{shiny.tag} or \code{character(1)}) The browser window title. Defaults to a title "teal app" with the icon of NEST. @@ -35,40 +45,32 @@ The header of the app.} \item{footer}{(\code{shiny.tag} or \code{character(1)}) The footer of the app.} -\item{modules}{(\code{teal_modules}) object containing the output modules which -will be displayed in the \code{teal} application. See \code{\link[=modules]{modules()}} and \code{\link[=module]{module()}} for -more details.} - -\item{teal_data_rv}{(\code{reactive}) -returns the \code{teal_data}, only evaluated once, \code{NULL} value is ignored} - \item{filter}{(\code{teal_slices}) Specifies the initial filter using \code{\link[=teal_slices]{teal_slices()}}.} } \value{ -Returns a \code{reactive} expression which returns the currently active module. +\code{NULL} invisibly } \description{ -This is the main \code{teal} app that puts everything together. +\ifelse{html}{\href{https://lifecycle.r-lib.org/articles/stages.html#stable}{\figure{lifecycle-stable.svg}{options: alt='[Stable]'}}}{\strong{[Stable]}} +Module to create a \code{teal} app. This module can be called directly instead of \code{\link[=init]{init()}} and +included in your custom application. Please note that \code{\link[=init]{init()}} adds \code{reporter_previewer_module} +automatically, which is not a case when calling \code{ui/srv_teal} directly. } \details{ -It displays the splash UI which is used to fetch the data, possibly -prompting for a password input to fetch the data. Once the data is ready, -the splash screen is replaced by the actual \code{teal} UI that is tabsetted and -has a filter panel with \code{datanames} that are relevant for the current tab. -Nested tabs are possible, but we limit it to two nesting levels for reasons -of clarity of the UI. - -The splash screen functionality can also be used -for non-delayed data which takes time to load into memory, avoiding -\code{shiny} session timeouts. +Module is responsible for creating the main \code{shiny} app layout and initializing all the necessary +components. This module establishes reactive connection between the input \code{data} and every other +component in the app. Reactive change of the \code{data} passed as an argument, reloads the app and +possibly keeps all input settings the same so the user can continue where one left off. +\subsection{data flow in \code{teal} application}{ -Server evaluates the \code{teal_data_rv} (delayed data mechanism) and creates the -\code{datasets} object that is shared across modules. -Once it is ready and non-\code{NULL}, the splash screen is replaced by the -main \code{teal} UI that depends on the data. -The currently active tab is tracked and the right filter panel -updates the displayed datasets to filter for according to the active \code{datanames} -of the tab. +This module supports multiple data inputs but eventually, they are all converted to \code{reactive} +returning \code{teal_data} in this module. On this \verb{reactive teal_data} object several actions are +performed: +\itemize{ +\item data loading in \code{\link{module_init_data}} +\item data filtering in \code{\link{module_filter_data}} +\item data transformation in \code{\link{module_transform_data}} +} +} } -\keyword{internal} diff --git a/man/module_teal_data.Rd b/man/module_teal_data.Rd new file mode 100644 index 0000000000..abcbe048ff --- /dev/null +++ b/man/module_teal_data.Rd @@ -0,0 +1,72 @@ +% Generated by roxygen2: do not edit by hand +% Please edit documentation in R/module_teal_data.R +\name{module_teal_data} +\alias{module_teal_data} +\alias{ui_teal_data} +\alias{srv_teal_data} +\alias{ui_validate_reactive_teal_data} +\alias{srv_validate_reactive_teal_data} +\title{Execute and validate \code{teal_data_module}} +\usage{ +ui_teal_data(id, data_module) + +srv_teal_data( + id, + data, + data_module, + modules = NULL, + validate_shiny_silent_error = TRUE +) + +ui_validate_reactive_teal_data(id) + +srv_validate_reactive_teal_data( + id, + data, + modules = NULL, + validate_shiny_silent_error = FALSE +) +} +\arguments{ +\item{id}{(\code{character(1)}) Module id} + +\item{data_module}{(\code{teal_data_module})} + +\item{data}{(\verb{reactive teal_data})} + +\item{modules}{(\code{teal_modules} or \code{teal_module}) For \code{datanames} validation purpose} + +\item{validate_shiny_silent_error}{(\code{logical}) If \code{TRUE}, then \code{shiny.silent.error} is validated and +error message is displayed. +Default is \code{FALSE} to handle empty reactive cycle on \code{init}.} +} +\value{ +\code{reactive} \code{teal_data} +} +\description{ +This is a low level module to handle \code{teal_data_module} execution and validation. +\code{\link[=teal_transform_module]{teal_transform_module()}} inherits from \code{\link[=teal_data_module]{teal_data_module()}} so it is handled by this module too. +\code{\link[=srv_teal]{srv_teal()}} accepts various \code{data} objects and eventually they are all transformed to \code{reactive} +\code{\link[=teal_data]{teal_data()}} which is a standard data class in whole \code{teal} framework. +} +\section{data validation}{ + + +Executed \code{\link[=teal_data_module]{teal_data_module()}} is validated and output is validated for consistency. +Output \code{data} is invalid if: +\enumerate{ +\item \code{\link[=teal_data_module]{teal_data_module()}} is invalid if server doesn't return \code{reactive}. \strong{Immediately crashes an app!} +\item \code{reactive} throws a \code{shiny.error} - happens when module creating \code{\link[=teal_data]{teal_data()}} fails. +\item \code{reactive} returns \code{qenv.error} - happens when \code{\link[=teal_data]{teal_data()}} evaluates a failing code. +\item \code{reactive} object doesn't return \code{\link[=teal_data]{teal_data()}}. +\item \code{\link[=teal_data]{teal_data()}} object lacks any \code{datanames} specified in the \code{modules} argument. +} + +\code{teal} (observers in \code{srv_teal}) always waits to render an app until \code{reactive} \code{teal_data} is +returned. If error 2-4 occurs, relevant error message is displayed to app user and after issue is +resolved app will continue to run. \code{teal} guarantees that errors in a data don't crash an app +(except error 1). This is possible thanks to \code{.fallback_on_failure} which returns input-data +when output-data fails +} + +\keyword{internal} diff --git a/man/module_teal_module.Rd b/man/module_teal_module.Rd new file mode 100644 index 0000000000..01ad3f947d --- /dev/null +++ b/man/module_teal_module.Rd @@ -0,0 +1,101 @@ +% Generated by roxygen2: do not edit by hand +% Please edit documentation in R/module_nested_tabs.R +\name{module_teal_module} +\alias{module_teal_module} +\alias{ui_teal_module} +\alias{ui_teal_module.default} +\alias{ui_teal_module.teal_modules} +\alias{ui_teal_module.shiny.tag} +\alias{ui_teal_module.teal_module} +\alias{srv_teal_module} +\alias{srv_teal_module.default} +\alias{srv_teal_module.teal_modules} +\alias{srv_teal_module.teal_module} +\title{Calls all \code{modules}} +\usage{ +ui_teal_module(id, modules, depth = 0L) + +\method{ui_teal_module}{default}(id, modules, depth = 0L) + +\method{ui_teal_module}{teal_modules}(id, modules, depth = 0L) + +\method{ui_teal_module}{shiny.tag}(id, modules, depth = 0L) + +\method{ui_teal_module}{teal_module}(id, modules, depth = 0L) + +srv_teal_module( + id, + data_rv, + modules, + datasets = NULL, + slices_global, + reporter = teal.reporter::Reporter$new(), + is_active = reactive(TRUE) +) + +\method{srv_teal_module}{default}( + id, + data_rv, + modules, + datasets = NULL, + slices_global, + reporter = teal.reporter::Reporter$new(), + is_active = reactive(TRUE) +) + +\method{srv_teal_module}{teal_modules}( + id, + data_rv, + modules, + datasets = NULL, + slices_global, + reporter = teal.reporter::Reporter$new(), + is_active = reactive(TRUE) +) + +\method{srv_teal_module}{teal_module}( + id, + data_rv, + modules, + datasets = NULL, + slices_global, + reporter = teal.reporter::Reporter$new(), + is_active = reactive(TRUE) +) +} +\arguments{ +\item{id}{(\code{character}) optional +string specifying the \code{shiny} module id in cases it is used as a \code{shiny} module +rather than a standalone \code{shiny} app. This is a legacy feature.} + +\item{modules}{(\code{list} or \code{teal_modules} or \code{teal_module}) +nested list of \code{teal_modules} or \code{teal_module} objects or a single +\code{teal_modules} or \code{teal_module} object. These are the specific output modules which +will be displayed in the \code{teal} application. See \code{\link[=modules]{modules()}} and \code{\link[=module]{module()}} for +more details.} + +\item{depth}{(\code{integer(1)}) +number which helps to determine depth of the modules nesting.} + +\item{data_rv}{(\code{reactive} returning \code{teal_data})} + +\item{datasets}{(\code{reactive} returning \code{FilteredData} or \code{NULL}) +When \code{datasets} is passed from the parent module (\code{srv_teal}) then \code{dataset} is a singleton +which implies in filter-panel to be "global". When \code{NULL} then filter-panel is "module-specific".} + +\item{slices_global}{(\code{reactiveVal} returning \code{modules_teal_slices}) +see \code{\link{module_filter_manager}}} +} +\value{ +output of currently active module. +\itemize{ +\item \code{srv_teal_module.teal_module} returns \code{reactiveVal} containing output of the called module. +\item \code{srv_teal_module.teal_modules} returns output of module selected by \code{input$active_tab}. +} +} +\description{ +On the UI side each \code{teal_modules} is translated to a \code{tabsetPanel} and each \code{teal_module} is a +\code{tabPanel}. Both, UI and server are called recursively so that each tab is a separate module and +reflect nested structure of \code{modules} argument. +} +\keyword{internal} diff --git a/man/module_teal_with_splash.Rd b/man/module_teal_with_splash.Rd index e274d2dae0..5d75a51b89 100644 --- a/man/module_teal_with_splash.Rd +++ b/man/module_teal_with_splash.Rd @@ -4,7 +4,7 @@ \alias{module_teal_with_splash} \alias{ui_teal_with_splash} \alias{srv_teal_with_splash} -\title{Add splash screen to \code{teal} application} +\title{UI and server modules of \code{teal}} \usage{ ui_teal_with_splash( id, @@ -17,11 +17,15 @@ ui_teal_with_splash( srv_teal_with_splash(id, data, modules, filter = teal_slices()) } \arguments{ -\item{id}{(\code{character(1)}) -module id} +\item{id}{(\code{character}) optional +string specifying the \code{shiny} module id in cases it is used as a \code{shiny} module +rather than a standalone \code{shiny} app. This is a legacy feature.} -\item{data}{(\code{teal_data} or \code{teal_data_module}) -For constructing the data object, refer to \code{\link[=teal_data]{teal_data()}} and \code{\link[=teal_data_module]{teal_data_module()}}.} +\item{data}{(\code{teal_data}, \code{teal_data_module}, or \code{reactive} returning \code{teal_data}) +The \code{ui} component of this module does not require \code{data} if \code{teal_data_module} is not provided. +The \code{data} argument in the \code{ui} is included solely for the \verb{$ui} function of the +\code{teal_data_module}. Otherwise, it can be disregarded, ensuring that \code{ui_teal} does not depend on +the reactive data of the enclosing application.} \item{title}{(\code{shiny.tag} or \code{character(1)}) The browser window title. Defaults to a title "teal app" with the icon of NEST. @@ -34,7 +38,9 @@ The header of the app.} \item{footer}{(\code{shiny.tag} or \code{character(1)}) The footer of the app.} -\item{modules}{(\code{teal_modules}) object containing the output modules which +\item{modules}{(\code{list} or \code{teal_modules} or \code{teal_module}) +nested list of \code{teal_modules} or \code{teal_module} objects or a single +\code{teal_modules} or \code{teal_module} object. These are the specific output modules which will be displayed in the \code{teal} application. See \code{\link[=modules]{modules()}} and \code{\link[=module]{module()}} for more details.} @@ -45,48 +51,6 @@ Specifies the initial filter using \code{\link[=teal_slices]{teal_slices()}}.} Returns a \code{reactive} expression containing a \code{teal_data} object when data is loaded or \code{NULL} when it is not. } \description{ -\ifelse{html}{\href{https://lifecycle.r-lib.org/articles/stages.html#stable}{\figure{lifecycle-stable.svg}{options: alt='[Stable]'}}}{\strong{[Stable]}} - -Displays custom splash screen during initial delayed data loading. -} -\details{ -This module pauses app initialization pending delayed data loading. -This is necessary because the filter panel and modules depend on the data to initialize. - -\code{teal_with_splash} follows the \code{shiny} module convention. -\code{\link[=init]{init()}} is a wrapper around this that assumes that \code{teal} is -the top-level module and cannot be embedded. - -Note: It is no longer recommended to embed \code{teal} in \code{shiny} apps as a module. -but rather use \code{init} to create a standalone application. -} -\section{Reproducibility}{ - -Reproducibility is supported by multiple features. \code{teal} includes a \code{utils::sessioInfo()} output to allow to compare -packages used in the session. It also allows to create \code{renv} lockfile to support project setup reproducibility. -For more information about lockfile creation visit \code{\link[=teal_lockfile]{teal_lockfile()}}. -} - -\examples{ -teal_modules <- modules(example_module()) -# Shiny app with modular integration of teal -ui <- fluidPage( - ui_teal_with_splash(id = "app1", data = teal_data()) -) - -server <- function(input, output, session) { - srv_teal_with_splash( - id = "app1", - data = teal_data(iris = iris), - modules = teal_modules - ) -} - -if (interactive()) { - shinyApp(ui, server) -} - -} -\seealso{ -\code{\link[=init]{init()}} +\ifelse{html}{\href{https://lifecycle.r-lib.org/articles/stages.html#deprecated}{\figure{lifecycle-deprecated.svg}{options: alt='[Deprecated]'}}}{\strong{[Deprecated]}} +Please use \code{\link{module_teal}} instead. } diff --git a/man/module_transform_data.Rd b/man/module_transform_data.Rd new file mode 100644 index 0000000000..5dbc480880 --- /dev/null +++ b/man/module_transform_data.Rd @@ -0,0 +1,27 @@ +% Generated by roxygen2: do not edit by hand +% Please edit documentation in R/module_transform_data.R +\name{module_transform_data} +\alias{module_transform_data} +\alias{ui_transform_data} +\alias{srv_transform_data} +\title{Module to transform \code{reactive} \code{teal_data}} +\usage{ +ui_transform_data(id, transforms, class = "well") + +srv_transform_data(id, data, transforms, modules) +} +\arguments{ +\item{id}{(\code{character(1)}) Module id} + +\item{data}{(\verb{reactive teal_data})} + +\item{modules}{(\code{teal_modules} or \code{teal_module}) For \code{datanames} validation purpose} +} +\value{ +\code{reactive} \code{teal_data} +} +\description{ +Module calls multiple \code{\link{module_teal_data}} in sequence so that \verb{reactive teal_data} output +from one module is handed over to the following module's input. +} +\keyword{internal} diff --git a/man/module_wunder_bar.Rd b/man/module_wunder_bar.Rd deleted file mode 100644 index 6e4f5b8fa1..0000000000 --- a/man/module_wunder_bar.Rd +++ /dev/null @@ -1,49 +0,0 @@ -% Generated by roxygen2: do not edit by hand -% Please edit documentation in R/module_wunder_bar.R -\name{module_wunder_bar} -\alias{module_wunder_bar} -\alias{wunder_bar_ui} -\alias{wunder_bar} -\alias{wunder_bar_module} -\alias{wunder_bar_srv} -\title{Manager bar module} -\usage{ -wunder_bar_ui(id) - -wunder_bar_srv(id, datasets, filter, modules) -} -\arguments{ -\item{id}{(\code{character(1)}) -module id} - -\item{datasets}{(named \code{list} of \code{FilteredData}) -object to store filter state and filtered datasets, shared across modules. For more -details see \code{\link[teal.slice:FilteredData]{teal.slice::FilteredData}}. Structure of the list must be the same as structure -of the \code{modules} argument and list names must correspond to the labels in \code{modules}. -When filter is not module-specific then list contains the same object in all elements.} - -\item{filter}{(\code{teal_slices}) -Specifies the initial filter using \code{\link[=teal_slices]{teal_slices()}}.} - -\item{modules}{(\code{teal_modules}) object containing the output modules which -will be displayed in the \code{teal} application. See \code{\link[=modules]{modules()}} and \code{\link[=module]{module()}} for -more details.} -} -\value{ -Nothing is returned. -} -\description{ -Bar of buttons that open modal dialogs. -} -\details{ -Creates a bar of buttons that open modal dialogs where manager modules reside. -Currently contains three modules: -\itemize{ -\item \code{\link{module_filter_manager}} -\item \code{\link{module_snapshot_manager}} -\item \code{\link{module_bookmark_manager}} -} - -The bar is placed in the \code{teal} app UI, next to the filter panel hamburger. -} -\keyword{internal} diff --git a/man/modules_datasets.Rd b/man/modules_datasets.Rd deleted file mode 100644 index 9c40dbff29..0000000000 --- a/man/modules_datasets.Rd +++ /dev/null @@ -1,34 +0,0 @@ -% Generated by roxygen2: do not edit by hand -% Please edit documentation in R/utils.R -\name{modules_datasets} -\alias{modules_datasets} -\title{Create filterable data for modules} -\usage{ -modules_datasets( - data, - modules, - filters, - filtered_data_singleton = teal_data_to_filtered_data(data), - progress = NULL -) -} -\arguments{ -\item{data}{(\code{teal_data})} - -\item{modules}{(\code{teal_modules}) object} - -\item{filters}{(\code{teal_slices}) object} - -\item{filtered_data_singleton}{A result of \code{teal_data_to_filtered_data} applied to \code{data}.} - -\item{progress}{(\code{Progress}) object from \code{shiny}, optional. -The progress bar will be filled during the (possibly recursive) call.} -} -\value{ -Returns list of same shape as \code{modules}, containing \code{FilteredData} at every leaf. -If module specific, each leaf contains different instance, otherwise every leaf contains \code{filtered_data_singleton}. -} -\description{ -Converts input data to a \code{FilteredData} object(s) to allow filtering before passing data to individual modules. -} -\keyword{internal} diff --git a/man/reexports.Rd b/man/reexports.Rd deleted file mode 100644 index 8394a23fcf..0000000000 --- a/man/reexports.Rd +++ /dev/null @@ -1,16 +0,0 @@ -% Generated by roxygen2: do not edit by hand -% Please edit documentation in R/teal.R -\docType{import} -\name{reexports} -\alias{reexports} -\alias{\%>\%} -\title{Objects exported from other packages} -\keyword{internal} -\description{ -These objects are imported from other packages. Follow the links -below to see their documentation. - -\describe{ - \item{magrittr}{\code{\link[magrittr:pipe]{\%>\%}}} -}} - diff --git a/man/tdata.Rd b/man/tdata.Rd index 2cdfb301e2..a1bab543cd 100644 --- a/man/tdata.Rd +++ b/man/tdata.Rd @@ -3,60 +3,37 @@ \name{tdata} \alias{tdata} \alias{new_tdata} +\alias{tdata2env} +\alias{get_code_tdata} +\alias{join_keys.tdata} +\alias{get_metadata} +\alias{as_tdata} \title{Create a \code{tdata} object} \usage{ -new_tdata(data, code = "", join_keys = NULL, metadata = NULL) -} -\arguments{ -\item{data}{(named \code{list}) A list of \code{data.frame} or \code{MultiAssayExperiment} objects, -which optionally can be \code{reactive}. -Inside this object all of these items will be made \code{reactive}.} +new_tdata(...) + +tdata2env(...) + +get_code_tdata(...) -\item{code}{(\code{character} or \code{reactive} which evaluates to a \code{character}) containing -the code used to generate the data. This should be \code{reactive} if the code is changing -during a reactive context (e.g. if filtering changes the code). Inside this -object \code{code} will be made reactive} +\method{join_keys}{tdata}(...) -\item{join_keys}{(\code{teal.data::join_keys}) object containing relationships between the -datasets.} +get_metadata(...) -\item{metadata}{(named \code{list}) each element contains a list of metadata about the named \code{data.frame} -Each element of these list should be atomic and length one.} +as_tdata(...) +} +\arguments{ +\item{...}{ignored} } \value{ -A \code{tdata} object. +nothing } \description{ -\ifelse{html}{\href{https://lifecycle.r-lib.org/articles/stages.html#deprecated}{\figure{lifecycle-deprecated.svg}{options: alt='[Deprecated]'}}}{\strong{[Deprecated]}} - -Create a new object called \code{tdata} which contains \code{data}, a \code{reactive} list of \code{data.frames} -(or \code{MultiAssayExperiment}), with attributes: -\itemize{ -\item \code{code} (\code{reactive}) containing code used to generate the data -\item join_keys (\code{join_keys}) containing the relationships between the data -\item metadata (named \code{list}) containing any metadata associated with the data frames -} -} -\examples{ - -data <- new_tdata( - data = list(iris = iris, mtcars = reactive(mtcars), dd = data.frame(x = 1:10)), - code = "iris <- iris - mtcars <- mtcars - dd <- data.frame(x = 1:10)", - metadata = list(dd = list(author = "NEST"), iris = list(version = 1)) -) - -# Extract a data.frame -isolate(data[["iris"]]()) +\ifelse{html}{\href{https://lifecycle.r-lib.org/articles/stages.html#superseded}{\figure{lifecycle-superseded.svg}{options: alt='[Superseded]'}}}{\strong{[Superseded]}} -# Get code -isolate(get_code_tdata(data)) - -# Get metadata -get_metadata(data, "iris") - -} -\seealso{ -\code{as_tdata} +Recent changes in \code{teal} cause modules to fail because modules expect a \code{tdata} object +to be passed to the \code{data} argument but instead they receive a \code{teal_data} object, +which is additionally wrapped in a reactive expression in the server functions. +In order to easily adapt such modules without a proper refactor, +use this function to downgrade the \code{data} argument. } diff --git a/man/tdata2env.Rd b/man/tdata2env.Rd deleted file mode 100644 index 61f0576320..0000000000 --- a/man/tdata2env.Rd +++ /dev/null @@ -1,28 +0,0 @@ -% Generated by roxygen2: do not edit by hand -% Please edit documentation in R/tdata.R -\name{tdata2env} -\alias{tdata2env} -\title{Function to convert a \code{tdata} object to an \code{environment}} -\usage{ -tdata2env(data) -} -\arguments{ -\item{data}{(\code{tdata}) object} -} -\value{ -An \code{environment}. -} -\description{ -Any \code{reactive} expressions inside \code{tdata} are evaluated first. -} -\examples{ - -data <- new_tdata( - data = list(iris = iris, mtcars = reactive(mtcars)), - code = "iris <- iris - mtcars = mtcars" -) - -my_env <- isolate(tdata2env(data)) - -} diff --git a/man/tdata_deprecation.Rd b/man/tdata_deprecation.Rd deleted file mode 100644 index 5f0b006de2..0000000000 --- a/man/tdata_deprecation.Rd +++ /dev/null @@ -1,32 +0,0 @@ -% Generated by roxygen2: do not edit by hand -% Please edit documentation in R/tdata.R -\name{as_tdata} -\alias{as_tdata} -\title{Downgrade \code{teal_data} objects in modules for compatibility} -\usage{ -as_tdata(x) -} -\arguments{ -\item{x}{data object, either \code{tdata} or \code{teal_data}, the latter possibly in a reactive expression} -} -\value{ -Object of class \code{tdata}. -} -\description{ -Convert \code{teal_data} to \code{tdata} in \code{teal} modules. -} -\details{ -Recent changes in \code{teal} cause modules to fail because modules expect a \code{tdata} object -to be passed to the \code{data} argument but instead they receive a \code{teal_data} object, -which is additionally wrapped in a reactive expression in the server functions. -In order to easily adapt such modules without a proper refactor, -use this function to downgrade the \code{data} argument. -} -\examples{ -td <- teal_data() -td <- within(td, iris <- iris) \%>\% within(mtcars <- mtcars) -td -as_tdata(td) -as_tdata(reactive(td)) - -} diff --git a/man/teal_data_datanames.Rd b/man/teal_data_datanames.Rd deleted file mode 100644 index 9c28c06aaf..0000000000 --- a/man/teal_data_datanames.Rd +++ /dev/null @@ -1,19 +0,0 @@ -% Generated by roxygen2: do not edit by hand -% Please edit documentation in R/utils.R -\name{teal_data_datanames} -\alias{teal_data_datanames} -\title{Wrapper on \code{teal.data::datanames}} -\usage{ -teal_data_datanames(data) -} -\arguments{ -\item{data}{(\code{teal_data})} -} -\value{ -\code{character} -} -\description{ -Special function used in internals of \code{teal} to return names of datasets even if \code{datanames} -has not been set. -} -\keyword{internal} diff --git a/man/teal_data_module.Rd b/man/teal_data_module.Rd index 631c2f2fd1..a6034cca13 100644 --- a/man/teal_data_module.Rd +++ b/man/teal_data_module.Rd @@ -11,7 +11,7 @@ \alias{within.teal_data_module} \title{Data module for \code{teal} applications} \usage{ -teal_data_module(ui, server) +teal_data_module(ui, server, label = "data module", once = TRUE) \S4method{eval_code}{teal_data_module,character}(object, code) @@ -25,6 +25,14 @@ teal_data_module(ui, server) \code{shiny} module server function; must only take \code{id} argument; must return reactive expression containing \code{teal_data} object} +\item{label}{(\code{character(1)}) Label of the module.} + +\item{once}{(\code{logical(1)}) +If \code{TRUE}, the data module will be shown only once and will disappear after successful data loading. +App user will no longer be able to interact with this module anymore. +If \code{FALSE}, the data module can be reused multiple times. +App user will be able to interact and change the data output from the module multiple times.} + \item{object}{(\code{teal_data_module})} \item{code}{(\code{character} or \code{language}) code to evaluate. If \code{character}, comments are retained.} @@ -36,7 +44,8 @@ must return reactive expression containing \code{teal_data} object} \item{...}{See \code{Details}.} } \value{ -\code{teal_data_module} returns an object of class \code{teal_data_module}. +\code{teal_data_module} returns a list of class \code{teal_data_module} containing two elements, \code{ui} and +\code{server} provided via arguments. \code{eval_code} returns a \code{teal_data_module} object with a delayed evaluation of \code{code} when the module is run. @@ -48,8 +57,8 @@ must return reactive expression containing \code{teal_data} object} Create a \code{teal_data_module} object and evaluate code on it with history tracking. } \details{ -\code{teal_data_module} creates a \code{shiny} module to supply or modify data in a \code{teal} application. -The module allows for running data pre-processing code (creation \emph{and} some modification) after the app starts. +\code{teal_data_module} creates a \code{shiny} module to interactively supply or modify data in a \code{teal} application. +The module allows for running any code (creation \emph{and} some modification) after the app starts or reloads. The body of the server function will be run in the app rather than in the global environment. This means it will be run every time the app starts, so use sparingly. diff --git a/man/teal_data_to_filtered_data.Rd b/man/teal_data_to_filtered_data.Rd index 5e7ac25b4f..d02e74ddfa 100644 --- a/man/teal_data_to_filtered_data.Rd +++ b/man/teal_data_to_filtered_data.Rd @@ -4,7 +4,7 @@ \alias{teal_data_to_filtered_data} \title{Create a \code{FilteredData}} \usage{ -teal_data_to_filtered_data(x, datanames = teal_data_datanames(x)) +teal_data_to_filtered_data(x, datanames = .teal_data_datanames(x)) } \arguments{ \item{x}{(\code{teal_data}) object} diff --git a/man/teal_data_utilities.Rd b/man/teal_data_utilities.Rd new file mode 100644 index 0000000000..7d5c4f3575 --- /dev/null +++ b/man/teal_data_utilities.Rd @@ -0,0 +1,46 @@ +% Generated by roxygen2: do not edit by hand +% Please edit documentation in R/teal_data_utils.R +\name{teal_data_utilities} +\alias{teal_data_utilities} +\alias{.append_evaluated_code} +\alias{.append_modified_data} +\alias{.subset_teal_data} +\alias{.teal_data_datanames} +\alias{.teal_data_ls} +\title{\code{teal_data} utils} +\usage{ +.append_evaluated_code(data, code) + +.append_modified_data(data, objects) + +.subset_teal_data(data, datanames) + +.teal_data_datanames(data) + +.teal_data_ls(data) +} +\arguments{ +\item{data}{(\code{teal_data})} + +\item{code}{(\code{character}) code to append to \code{data@code}} + +\item{objects}{(\code{list}) objects to append to \code{data@env}} + +\item{datanames}{(\code{character}) names of the datasets} +} +\value{ +modified \code{teal_data} +} +\description{ +In \code{teal} we need to recreate the \code{teal_data} object due to two operations:integer.max +\itemize{ +\item we need to append filter-data code and objects which have been evaluated in \code{FilteredData} and +we want to avoid double-evaluation. +\item we need to subset \code{teal_data} to \code{datanames} used by the module, to shorten obtainable R-code +} +} +\details{ +Due to above recreation of \code{teal_data} object can't be done simply by using public +\code{teal.code} and \code{teal.data} methods. +} +\keyword{internal} diff --git a/man/teal_modules.Rd b/man/teal_modules.Rd index 0e5d4aaf77..8c45a74b6a 100644 --- a/man/teal_modules.Rd +++ b/man/teal_modules.Rd @@ -13,19 +13,13 @@ \usage{ module( label = "module", - server = function(id, ...) { - moduleServer(id, function(input, output, session) { - - }) - }, - ui = function(id, ...) { - tags$p(paste0("This module has no UI (id: ", id, " )")) - - }, + server = function(id, ...) moduleServer(id, function(input, output, session) NULL), + ui = function(id, ...) tags$p(paste0("This module has no UI (id: ", id, " )")), filters, datanames = "all", server_args = NULL, - ui_args = NULL + ui_args = NULL, + transformers = list() ) modules(..., label = "root") @@ -79,6 +73,13 @@ a subset of datasets which are appended to the \code{data} argument in server fu \item{ui_args}{(named \code{list}) with additional arguments passed on to the UI function.} +\item{transformers}{(\code{list} of \code{teal_data_module}) that will be applied to transform the data. +Each transform module UI will appear in the \code{teal} application, unless the \code{custom_ui} attribute is set on the list. +If so, the module developer is responsible to display the UI in the module itself. + +When the transformation does not have sufficient input data, the resulting data will fallback +to the last successful transform or, in case there are none, to the filtered data.} + \item{...}{\itemize{ \item For \code{modules()}: (\code{teal_module} or \code{teal_modules}) Objects to wrap into a tab. \item For \code{format()} and \code{print()}: Arguments passed to other methods. @@ -100,7 +101,6 @@ their \code{label} attribute converted to a valid \code{shiny} id. } \description{ \ifelse{html}{\href{https://lifecycle.r-lib.org/articles/stages.html#stable}{\figure{lifecycle-stable.svg}{options: alt='[Stable]'}}}{\strong{[Stable]}} - Create a nested tab structure to embed modules in a \code{teal} application. } \details{ diff --git a/man/teal_transform_module.Rd b/man/teal_transform_module.Rd new file mode 100644 index 0000000000..0a4ed27a34 --- /dev/null +++ b/man/teal_transform_module.Rd @@ -0,0 +1,58 @@ +% Generated by roxygen2: do not edit by hand +% Please edit documentation in R/teal_data_module.R +\name{teal_transform_module} +\alias{teal_transform_module} +\title{Data module for \code{teal} transformers.} +\usage{ +teal_transform_module(ui, server, label = "transform module") +} +\arguments{ +\item{ui}{(\verb{function(id)}) +\code{shiny} module UI function; must only take \code{id} argument} + +\item{server}{(\verb{function(id, data)}) +\code{shiny} module server function; that takes \code{id} and \code{data} argument, +where the \code{id} is the module id and \code{data} is the reactive \code{teal_data} input. +The server function must return reactive expression containing \code{teal_data} object.} + +\item{label}{(\code{character(1)}) Label of the module.} +} +\description{ +\ifelse{html}{\href{https://lifecycle.r-lib.org/articles/stages.html#experimental}{\figure{lifecycle-experimental.svg}{options: alt='[Experimental]'}}}{\strong{[Experimental]}} + +Create a \code{teal_data_module} object for custom transformation of data for pre-processing +before passing the data into the module. +} +\details{ +\code{teal_transform_module} creates a \code{\link{teal_data_module}} object to transform data in a \code{teal} +application. This transformation happens after the data has passed through the filtering activity +in teal. The transformed data is then sent to the server of the \code{\link[=teal_module]{teal_module()}}. + +See vignette \code{vignette("data-transform-as-shiny-module", package = "teal")} for more details. +} +\examples{ +my_transformers <- list( + teal_transform_module( + label = "Custom transform for iris", + ui = function(id) { + ns <- NS(id) + tags$div( + numericInput(ns("n_rows"), "Subset n rows", value = 6, min = 1, max = 150, step = 1) + ) + }, + server = function(id, data) { + moduleServer(id, function(input, output, session) { + reactive({ + within(data(), + { + iris <- head(iris, num_rows) + }, + num_rows = input$n_rows + ) + }) + }) + } + ) +) + +} diff --git a/man/unfold_mapping.Rd b/man/unfold_mapping.Rd deleted file mode 100644 index 9a12a352d2..0000000000 --- a/man/unfold_mapping.Rd +++ /dev/null @@ -1,20 +0,0 @@ -% Generated by roxygen2: do not edit by hand -% Please edit documentation in R/module_snapshot_manager.R -\name{unfold_mapping} -\alias{unfold_mapping} -\title{Explicitly enumerate global filters.} -\usage{ -unfold_mapping(mapping, module_names) -} -\arguments{ -\item{mapping}{(named \code{list}) as stored in mapping parameter of \code{teal_slices}} - -\item{module_names}{(\code{character}) vector containing names of all modules in the app} -} -\value{ -A \code{named_list} with one element per module, each element containing all filters applied to that module. -} -\description{ -Transform module mapping such that global filters are explicitly specified for every module. -} -\keyword{internal} diff --git a/tests/testthat/test-filter_manager.R b/tests/testthat/test-filter_manager.R deleted file mode 100644 index 67550e3fa6..0000000000 --- a/tests/testthat/test-filter_manager.R +++ /dev/null @@ -1,68 +0,0 @@ -filter_global <- teal_slices( - teal.slice::teal_slice(dataname = "iris", varname = "Sepal.Length"), - teal.slice::teal_slice(dataname = "iris", varname = "Species"), - teal.slice::teal_slice(dataname = "mtcars", varname = "mpg"), - teal.slice::teal_slice(dataname = "women", varname = "height"), - module_specific = TRUE, - mapping = list( - m1 = c("iris Sepal.Length"), - m3 = c("women height"), - global_filters = "iris Species" - ) -) -filter_modular <- teal_slices( - teal.slice::teal_slice(dataname = "iris", varname = "Sepal.Length"), - teal.slice::teal_slice(dataname = "iris", varname = "Species"), - teal.slice::teal_slice(dataname = "mtcars", varname = "mpg"), - teal.slice::teal_slice(dataname = "women", varname = "height"), - module_specific = FALSE, - mapping = list( - m1 = c("iris Sepal.Length"), - m3 = c("women height"), - global_filters = "iris Species" - ) -) - -testthat::test_that("filter_manager_srv initializes properly processes input arguments", { - fd1 <- teal.slice::init_filtered_data(list(iris = list(dataset = iris))) - fd2 <- teal.slice::init_filtered_data( - list(iris = list(dataset = iris), mtcars = list(dataset = mtcars)) - ) - fd3 <- teal.slice::init_filtered_data( - list(iris = list(dataset = iris), women = list(dataset = women)) - ) - filtered_data_list <- list( - m1 = fd1, - tab = list(m2 = fd2, m3 = fd3) - ) - - # global filtering - shiny::testServer( - app = filter_manager_srv, - args = list( - id = "test", - datasets = filtered_data_list, - filter = filter_global - ), - expr = { - testthat::expect_named(datasets_flat, c("m1", "m2", "m3")) - - testthat::expect_identical(slices_global(), filter) - } - ) - - # modular filtering - shiny::testServer( - app = filter_manager_srv, - args = list( - id = "test", - datasets = filtered_data_list, - filter = filter_modular - ), - expr = { - testthat::expect_named(datasets_flat, "Global Filters") - - testthat::expect_identical(slices_global(), filter) - } - ) -}) diff --git a/tests/testthat/test-init.R b/tests/testthat/test-init.R index 0392763b5f..24efa4d119 100644 --- a/tests/testthat/test-init.R +++ b/tests/testthat/test-init.R @@ -56,14 +56,15 @@ testthat::test_that("init throws when an empty `data` is used", { ) }) -testthat::test_that("init throws when datanames in modules incompatible w/ datanames in data", { - msg <- "Module 'example teal module' uses datanames not available in 'data'" - testthat::expect_error( +testthat::test_that("init throws warning when datanames in modules incompatible w/ datanames in data", { + testthat::local_mocked_bindings(log_warn = warning, .package = "logger") + + testthat::expect_warning( init( data = teal.data::teal_data(mtcars = mtcars), modules = list(example_module(datanames = "iris")) ), - "Module 'example teal module' uses datanames not available in 'data'" + "Dataset \"iris\" is missing for tab 'example teal module'. Dataset available in data: \"mtcars\"." ) }) diff --git a/tests/testthat/test-module_nested_tabs.R b/tests/testthat/test-module_nested_tabs.R deleted file mode 100644 index 3468505249..0000000000 --- a/tests/testthat/test-module_nested_tabs.R +++ /dev/null @@ -1,472 +0,0 @@ -teal_data <- teal.data::teal_data() -teal_data <- within(teal_data, iris <- head(iris)) -datanames(teal_data) <- "iris" -filtered_data <- teal_data_to_filtered_data(teal_data) - -test_module1 <- module( - label = "test1", - ui = function(id, ...) NULL, - server = function(id) moduleServer(id, function(input, output, session) message("1")), - datanames = NULL -) -test_module2 <- module( - label = "test2", - ui = function(id) NULL, - server = function(id) moduleServer(id, function(input, output, session) message("2")), - datanames = NULL -) -test_module3 <- module( - label = "test3", - ui = function(id) NULL, - server = function(id) moduleServer(id, function(input, output, session) message("3")), - datanames = NULL -) -test_module4 <- module( - label = "test4", - ui = function(id) NULL, - server = function(id) moduleServer(id, function(input, output, session) message("4")), - datanames = NULL -) -test_module_wdata <- function(datanames) { - module( - label = "with_data", - ui = function(id) NULL, - server = function(id, data) moduleServer(id, function(input, output, session) message("module with data")), - datanames = datanames - ) -} - -get_example_filtered_data <- function() { - td <- teal.data::teal_data() - td <- within(td, d1 <- data.frame(id = 1:5, pk = c(2, 3, 2, 1, 4), val = 1:5)) - td <- within(td, d2 <- data.frame(id = 1:5, value = 1:5)) - datanames(td) <- c("d1", "d2") - teal.data::join_keys(td) <- teal.data::join_keys(teal.data::join_key("d1", "d2", c("pk" = "id"))) - teal_data_to_filtered_data(td) -} - - -testthat::test_that("srv_nested_tabs throws error if reporter is not inherited from class Reporter", { - testthat::expect_error( - srv_nested_tabs(id, datasets = filtered_data, modules = modules(test_module1), reporter = list()), - "Must inherit from class 'Reporter'" - ) -}) - -# server ------- -testthat::test_that("passed shiny module is initialized only when the UI is triggered", { - # module not initialized - testthat::expect_silent( - shiny::testServer( - app = srv_nested_tabs, - args = list( - id = "test", - datasets = list(test1 = filtered_data), - modules = modules(test_module1), - reporter = teal.reporter::Reporter$new() - ), - expr = NULL - ) - ) - - # module initialized - testthat::expect_message( - shiny::testServer( - app = srv_nested_tabs, - args = list( - id = "test", - datasets = list(test1 = filtered_data), - modules = modules(test_module1), - reporter = teal.reporter::Reporter$new() - ), - expr = { - session$setInputs() - } - ), - "1" - ) -}) - -testthat::test_that("nested teal-modules are initialized when the UI is triggered", { - # modules not initialized - testthat::expect_silent( - shiny::testServer( - app = srv_nested_tabs, - args = list( - id = "test", - datasets = list( - tab1 = list(test1 = filtered_data, test2 = filtered_data), - tab2 = list(test3 = filtered_data, test4 = filtered_data) - ), - modules = modules( - modules(label = "tab1", test_module1, test_module2), - modules(label = "tab2", test_module3, test_module4) - ), - reporter = teal.reporter::Reporter$new() - ), - expr = NULL - ) - ) - - # modules initialized - out <- testthat::capture_messages( - shiny::testServer( - app = srv_nested_tabs, - args = list( - id = "test", - datasets = list( - tab1 = list(test1 = filtered_data, test2 = filtered_data), - tab2 = list(test3 = filtered_data, test4 = filtered_data) - ), - modules = modules( - modules(label = "tab1", test_module1, test_module2), - modules(label = "tab2", test_module3, test_module4) - ), - reporter = teal.reporter::Reporter$new() - ), - expr = { - session$setInputs() - } - ) - ) - testthat::expect_identical(out, c("1\n", "2\n", "3\n", "4\n")) -}) - -out <- shiny::testServer( - app = srv_nested_tabs, - args = list( - id = "test", - datasets = list( - tab1 = list(test1 = filtered_data, test2 = filtered_data), - tab2 = list(test3 = filtered_data, test4 = filtered_data) - ), - modules = modules( - modules(label = "tab1", test_module1, test_module2), - modules(label = "tab2", test_module3, test_module4) - ), - reporter = teal.reporter::Reporter$new() - ), - expr = { - testthat::test_that("modules_reactive is a list of reactives", { - testthat::expect_is(modules_reactive, "list") - testthat::expect_is(modules_reactive$tab1, "reactive") - testthat::expect_is(modules_reactive$tab2, "reactive") - }) - - testthat::test_that("modules_reactive returns modules according to selection in the nested tabs", { - session$setInputs(`tab1-active_tab` = "test2") # active tab in tab1 - session$setInputs(`tab2-active_tab` = "test3") # active tab in tab2 - nested_active_modules <- lapply(modules_reactive, function(child) child()) - testthat::expect_identical(nested_active_modules, list(tab1 = test_module2, tab2 = test_module3)) - - session$setInputs(`tab1-active_tab` = "test1") # active tab in tab1 - session$setInputs(`tab2-active_tab` = "test4") # active tab in tab2 - nested_active_modules <- lapply(modules_reactive, function(child) child()) - testthat::expect_identical(nested_active_modules, list(tab1 = test_module1, tab2 = test_module4)) - }) - - testthat::test_that("Change of this tab returns active module from this tab", { - session$setInputs(`active_tab` = "tab1") - testthat::expect_identical(get_active_module(), test_module1) - - session$setInputs(`active_tab` = "tab2") - testthat::expect_identical(get_active_module(), test_module4) - }) - } -) - -testthat::test_that("srv_nested_tabs.teal_module does not pass data if not in the args explicitly", { - module <- module(server = function(id, ...) { - moduleServer(id, function(input, output, session) { - testthat::expect_null(list(...)$data) - }) - }) - - shiny::testServer( - app = srv_nested_tabs, - args = list( - id = "test", - datasets = list(module = filtered_data), - modules = modules(module), - reporter = teal.reporter::Reporter$new() - ), - expr = { - session$setInputs() - } - ) -}) - -testthat::test_that("srv_nested_tabs.teal_module does pass data if in the args explicitly", { - module <- module( - server = function(id, data, ...) { - moduleServer(id, function(input, output, session) { - checkmate::assert_class(data, "reactive") - checkmate::assert_class(data(), "teal_data") - }) - }, - datanames = NULL - ) - testthat::expect_no_error( - shiny::testServer( - app = srv_nested_tabs, - args = list( - id = "test", - datasets = list(module = filtered_data), - modules = modules(module), - reporter = teal.reporter::Reporter$new() - ), - expr = { - session$setInputs() - } - ) - ) -}) - -testthat::test_that("srv_nested_tabs.teal_module passes data to the server module", { - module <- module(datanames = NULL, server = function(id, data) { - moduleServer(id, function(input, output, session) checkmate::assert_list(data, "reactive")) - }) - - testthat::expect_no_error( - shiny::testServer( - app = srv_nested_tabs, - args = list( - id = "test", - datasets = list(module = filtered_data), - modules = modules(module), - reporter = teal.reporter::Reporter$new() - ), - expr = NULL - ) - ) -}) - -testthat::test_that("srv_nested_tabs.teal_module passes (deprecated) datasets to the server module", { - module <- lifecycle::expect_deprecated( - module(server = function(id, datasets) { - moduleServer(id, function(input, output, session) checkmate::assert_class(datasets, "FilteredData")) - }) - ) - - testthat::expect_no_error( - shiny::testServer( - app = srv_nested_tabs, - args = list( - id = "test", - datasets = list(module = filtered_data), - modules = modules(module), - reporter = teal.reporter::Reporter$new() - ), - expr = NULL - ) - ) -}) - -testthat::test_that("srv_nested_tabs.teal_module passes server_args to the ...", { - server_args <- list(a = 1, b = 2) - module <- module(server_args = server_args, server = function(id, ...) { - moduleServer(id, function(input, output, session) stopifnot(identical(list(...), server_args))) - }) - - testthat::expect_no_error( - shiny::testServer( - app = srv_nested_tabs, - args = list( - id = "test", - datasets = list(module = filtered_data), - modules = modules(module), - reporter = teal.reporter::Reporter$new() - ), - expr = NULL - ) - ) -}) - -fp_api <- teal.slice:::FilterPanelAPI$new(filtered_data) -testthat::test_that("srv_nested_tabs.teal_module doesn't pass filter_panel_api if not in the args explicitly", { - module <- module(server = function(id, ...) { - moduleServer(id, function(input, output, session) { - checkmate::assert_false( - tryCatch( - checkmate::test_class(filter_panel_api, "FilterPanelAPI"), - error = function(cond) FALSE - ) - ) - }) - }) - - testthat::expect_no_error( - shiny::testServer( - app = srv_nested_tabs, - args = list( - id = "test", - datasets = list(module = filtered_data), - modules = modules(module), - reporter = teal.reporter::Reporter$new() - ), - expr = { - session$setInputs() - } - ) - ) -}) - -testthat::test_that("srv_nested_tabs.teal_module passes filter_panel_api when passed in the args explicitly", { - module <- module(server = function(id, filter_panel_api = fp_api, ...) { - moduleServer(id, function(input, output, session) { - checkmate::assert_class(filter_panel_api, "FilterPanelAPI") - }) - }) - - testthat::expect_no_error( - shiny::testServer( - app = srv_nested_tabs, - args = list( - id = "test", - datasets = list(module = filtered_data), - modules = modules(module), - reporter = teal.reporter::Reporter$new() - ), - expr = { - session$setInputs() - } - ) - ) -}) - -testthat::test_that("srv_nested_tabs.teal_module passes filter_panel_api to the server module", { - module <- module(server = function(id, filter_panel_api) { - moduleServer(id, function(input, output, session) checkmate::assert_class(filter_panel_api, "FilterPanelAPI")) - }) - - testthat::expect_no_error( - shiny::testServer( - app = srv_nested_tabs, - args = list( - id = "test", - datasets = list(module = filtered_data), - modules = modules(module), - reporter = teal.reporter::Reporter$new() - ), - expr = NULL - ) - ) -}) - - -testthat::test_that(".datasets_to_data returns data which is filtered", { - datasets <- get_example_filtered_data() - datasets$set_filter_state( - teal.slice::teal_slices( - teal.slice::teal_slice(dataname = "d1", varname = "val", selected = c(1, 2)) - ) - ) - module <- test_module_wdata(datanames = c("d1", "d2")) - data <- shiny::isolate(.datasets_to_data(module, datasets)) - - d1_filtered <- data[["d1"]] - testthat::expect_equal(d1_filtered, data.frame(id = 1:2, pk = 2:3, val = 1:2)) - d2_filtered <- data[["d2"]] - testthat::expect_equal(d2_filtered, data.frame(id = 2:3, value = 2:3)) -}) - - -testthat::test_that(".datasets_to_data returns only data requested by modules$datanames", { - datasets <- get_example_filtered_data() - module <- test_module_wdata(datanames = "d1") - data <- shiny::isolate(.datasets_to_data(module, datasets)) - testthat::expect_equal(datanames(data), "d1") -}) - -testthat::test_that(".datasets_to_data returns teal_data object", { - datasets <- get_example_filtered_data() - module <- test_module_wdata(datanames = c("d1", "d2")) - data <- shiny::isolate(.datasets_to_data(module, datasets)) - - testthat::expect_s4_class(data, "teal_data") - - # join_keys - testthat::expect_equal( - join_keys(data), - teal.data::join_keys(teal.data::join_key("d1", "d2", c("pk" = "id"))) - ) - - # code - testthat::expect_equal( - teal.code::get_code(data), - paste( - c( - get_rcode_str_install(), - get_rcode_libraries(), - "d1 <- data.frame(id = 1:5, pk = c(2, 3, 2, 1, 4), val = 1:5)", - "d2 <- data.frame(id = 1:5, value = 1:5)", - "", - "stopifnot(rlang::hash(d1) == \"f6f90d2c133ca4abdeb2f7a7d85b731e\")", - "stopifnot(rlang::hash(d2) == \"6e30be195b7d914a1311672c3ebf4e4f\")", - "", - "d2 <- dplyr::inner_join(x = d2, y = d1[, c(\"pk\"), drop = FALSE], by = c(id = \"pk\"))", - "" - ), - collapse = "\n" - ) - ) -}) - -testthat::test_that("calculate_hashes takes a FilteredData and vector of datanames as input", { - adsl <- data.frame(STUDYID = 1, USUBJID = 1) - adae <- data.frame(STUDYID = 1, USUBJID = 1, ASTDTM = 1, AETERM = 1, AESEQ = 1) - adtte <- data.frame(STUDYID = 1, USUBJID = 1, PARAMCD = 1) - - datasets <- teal.slice::init_filtered_data( - list( - ADSL = list(dataset = adsl), - ADAE = list(dataset = adae), - ADTTE = list(dataset = adtte) - ) - ) - - testthat::expect_no_error(calculate_hashes(datanames = c("ADSL", "ADAE", "ADTTE"), datasets = datasets)) -}) - -testthat::test_that("calculate_hashes returns a named list", { - adsl <- data.frame(STUDYID = 1, USUBJID = 1) - adae <- data.frame(STUDYID = 1, USUBJID = 1, ASTDTM = 1, AETERM = 1, AESEQ = 1) - adtte <- data.frame(STUDYID = 1, USUBJID = 1, PARAMCD = 1) - - datasets <- teal.slice::init_filtered_data( - list( - ADSL = list(dataset = adsl), - ADAE = list(dataset = adae), - ADTTE = list(dataset = adtte) - ) - ) - - hashes <- calculate_hashes(datanames = c("ADSL", "ADAE", "ADTTE"), datasets = datasets) - testthat::expect_identical( - hashes, - list( - "ADSL" = "e89f5271357822c78dd5cfddb60c0a95", - "ADAE" = "f71b576ecfd23075f7285841327515e0", - "ADTTE" = "c68c01c86b946a3dfe05150da040aa2a" - ) - ) - testthat::expect_is(hashes, "list") - testthat::expect_named(hashes) -}) - -testthat::test_that("calculate_hashes returns the hash of the non Filtered dataset", { - datasets <- teal.slice::init_filtered_data( - list(iris = list(dataset = iris)) - ) - - fs <- teal.slice::teal_slices( - teal.slice::teal_slice(dataname = "iris", varname = "Sepal.Length", selected = c(5.1, 6.4)), - teal.slice::teal_slice(dataname = "iris", varname = "Species", selected = c("setosa", "versicolor")) - ) - - shiny::isolate(datasets$set_filter_state(state = fs)) - - hashes <- calculate_hashes(datanames = c("iris"), datasets = datasets) - testthat::expect_identical(hashes, list("iris" = "34844aba7bde36f5a34f6d8e39803508")) - testthat::expect_false(hashes == rlang::hash(shiny::isolate(datasets$get_data("iris", filtered = TRUE)))) -}) diff --git a/tests/testthat/test-module_tabs_with_filters.R b/tests/testthat/test-module_tabs_with_filters.R deleted file mode 100644 index d732b2ef5b..0000000000 --- a/tests/testthat/test-module_tabs_with_filters.R +++ /dev/null @@ -1,100 +0,0 @@ -teal_data <- teal.data::teal_data() -teal_data <- within(teal_data, iris <- head(iris)) -teal_data <- within(teal_data, mtcars <- head(mtcars)) -datanames(teal_data) <- c("iris", "mtcars") -filtered_data <- teal_data_to_filtered_data(teal_data) - -test_module1 <- module( - label = "iris tab", - server = function(id, data, ...) { - moduleServer(id, function(input, output, session) { - }) - }, - datanames = "iris" -) -test_module2 <- module( - label = "mtcars tab", - server = function(id, data, ...) { - moduleServer(id, function(input, output, session) { - }) - }, - datanames = "mtcars" -) - -testthat::test_that("srv_tabs_with_filters throws error if reporter is not of class Reporter", { - testthat::expect_error( - srv_tabs_with_filters( - id, - datasets = list(`iris tab` = filtered_data), - modules = modules(test_module1), - reporter = list() - ), - "Assertion on 'reporter' failed" - ) -}) - -testthat::test_that("active_module() returns module specs from active tab when filter.module_specific = FALSE", { - shiny::testServer( - app = srv_tabs_with_filters, - args = list( - id = "test", - datasets = list(`iris tab` = filtered_data, `mtcars tab` = filtered_data), - modules = modules(test_module1, test_module2), - filter = teal_slices(module_specific = FALSE), - reporter = teal.reporter::Reporter$new() - ), - expr = { - session$setInputs(`root-active_tab` = "iris_tab") - testthat::expect_identical(active_module(), test_module1) - session$setInputs(`root-active_tab` = "mtcars_tab") - testthat::expect_identical(active_module(), test_module2) - } - ) -}) - -testthat::test_that("srv_tabs_with_filters throws error if reporter is not of class Reporter", { - testthat::expect_error( - srv_tabs_with_filters( - id, - datasets = list(`iris tab` = filtered_data), - modules = modules(test_module1), - reporter = list() - ), - "Assertion on 'reporter' failed" - ) -}) - -testthat::test_that("active_datanames() returns dataname from single tab", { - shiny::testServer( - app = srv_tabs_with_filters, - args = list( - id = "test", - datasets = list(`iris tab` = filtered_data), - modules = modules(test_module1), - filter = teal_slices() - ), - expr = { - testthat::expect_identical(active_datanames(), "iris") - } - ) -}) - -testthat::test_that("active_datanames() returns dataname from active tab after change", { - shiny::testServer( - app = srv_tabs_with_filters, - args = list( - id = "test", - datasets = list(`iris tab` = filtered_data, `mtcars tab` = filtered_data), - modules = modules(test_module1, test_module2), - filter = teal_slices(), - reporter = teal.reporter::Reporter$new() - ), - expr = { - testthat::expect_error(active_datanames()) # to trigger active_module - session$setInputs(`root-active_tab` = "iris_tab") - testthat::expect_identical(active_datanames(), "iris") - session$setInputs(`root-active_tab` = "mtcars_tab") - testthat::expect_identical(active_datanames(), "mtcars") - } - ) -}) diff --git a/tests/testthat/test-module_teal.R b/tests/testthat/test-module_teal.R index e7e6bb3cc6..b308406a54 100644 --- a/tests/testthat/test-module_teal.R +++ b/tests/testthat/test-module_teal.R @@ -1,97 +1,2204 @@ -testthat::test_that("srv_teal fails when teal_data_rv is not reactive", { - testthat::expect_error( +# comment: srv_teal is exported so the tests here are extensive and cover srv_data as well. +# testing of srv_data is not needed. +module_output_table <<- function(output, id) { + testthat::skip_if_not_installed("rvest") + table_id <- sprintf("teal_modules-%s-data_summary-table", id) + html <- output[[table_id]]$html + as.data.frame(rvest::html_table(rvest::read_html(html), header = TRUE)[[1]]) +} + +is_slices_equivalent <<- function(x, y, with_attrs = TRUE) { + x_list <- as.list(x, recursive = TRUE) + y_list <- as.list(y, recursive = TRUE) + attributes(x_list) <- NULL + attributes(y_list) <- NULL + if (with_attrs) { + attributes(x_list) <- attributes(x)[c("mapping", "module_specific")] + attributes(y_list) <- attributes(y)[c("mapping", "module_specific")] + } + identical(x_list, y_list) +} + +transform_list <<- list( + fail = teal_transform_module( + ui = function(id) NULL, + server = function(id, data) { + moduleServer(id, function(input, output, session) { + add_error <- reactiveVal(TRUE) + observeEvent(input$add_error, add_error(input$add_error)) + + reactive({ + if (add_error()) { + stop("Oh no") + } else { + within(data(), iris <- head(iris, n = floor(nrow(iris) / 2))) + } + }) + }) + } + ), + iris = teal_transform_module( + ui = function(id) NULL, + server = function(id, data) { + moduleServer(id, function(input, output, session) { + n <- reactiveVal(6) + observeEvent(input$n, n(input$n)) + + reactive({ + within(data(), iris <- head(iris, n = n_input), n_input = n()) + }) + }) + } + ), + mtcars = teal_transform_module( + ui = function(id) NULL, + server = function(id, data) { + moduleServer(id, function(input, output, session) { + n <- reactiveVal(6) + observeEvent(input$n, n(input$n)) + + reactive({ + within(data(), mtcars <- head(mtcars, n = n_input), n_input = n()) + }) + }) + } + ), + add_dataset = teal_transform_module( + ui = function(id) NULL, + server = function(id, data) { + moduleServer(id, function(input, output, session) { + reactive({ + new_data <- within(data(), { + new_dataset <- data.frame(a = 1:3, b = 4:6) + }) + teal.data::datanames(new_data) <- c(teal.data::datanames(new_data), "new_dataset") + new_data + }) + }) + } + ) +) + +testthat::describe("srv_teal arguments", { + testthat::it("accepts data to be teal_data", { + testthat::expect_no_error( + shiny::testServer( + app = srv_teal, + args = list( + id = "test", + data = teal.data::teal_data(iris = iris), + modules = modules(example_module()) + ), + expr = NULL + ) + ) + }) + + testthat::it("accepts data to be teal_data_module returning reactive teal_data", { + testthat::expect_no_error( + shiny::testServer( + app = srv_teal, + args = list( + id = "test", + data = teal_data_module(ui = function(id) NULL, server = function(id) reactive(teal_data(iris = iris))), + modules = modules(example_module()) + ), + expr = NULL + ) + ) + }) + + testthat::it("accepts data to a reactive or reactiveVal teal_data", { + testthat::expect_no_error( + shiny::testServer( + app = srv_teal, + args = list( + id = "test", + data = reactive(teal_data(iris = iris)), + modules = modules(example_module()) + ), + expr = NULL + ) + ) + + reactive_val <- reactiveVal(teal_data(iris = iris)) + testthat::expect_no_error( + shiny::testServer( + app = srv_teal, + args = list( + id = "test", + data = reactive_val, + modules = modules(example_module()) + ), + expr = NULL + ) + ) + }) + + testthat::it("fails when data is not teal_data or teal_data_module", { + testthat::expect_error( + shiny::testServer( + app = srv_teal, + args = list( + id = "test", + data = data.frame(), + modules = modules(example_module()) + ), + expr = NULL + ), + "Must inherit from class 'teal_data'/'teal_data_module'/'reactive'/'reactiveVal'" + ) + }) + + testthat::it("app fails when teal_data_module doesn't return a reactive", { + testthat::expect_error( + shiny::testServer( + app = srv_teal, + args = list( + id = "test", + data = teal_data_module(ui = function(id) NULL, server = function(id) teal_data(iris = iris)), + modules = modules(example_module()) + ), + expr = { + session$flushReact() + } + ), + "The `teal_data_module` passed to `data` must return a reactive expressio" + ) + }) +}) + +testthat::describe("srv_teal teal_modules", { + testthat::it("are not called by default", { shiny::testServer( app = srv_teal, args = list( id = "test", - teal_data_rv = teal.data::teal_data(iris = iris), - modules = modules(example_module()) + data = reactive(teal_data(iris = iris)), + modules = modules( + module("module_1", server = function(id, data) 101L), + module("module_2", server = function(id, data) 102L) + ) ), - expr = NULL - ), - regexp = "is.reactive\\(teal_data_rv\\)" - ) + expr = { + testthat::expect_s4_class(data_rv(), "teal_data") + testthat::expect_null(modules_output$module_1()) + testthat::expect_null(modules_output$module_2()) + } + ) + }) + + testthat::it("are called once their tab is selected and data is `teal_data`", { + shiny::testServer( + app = srv_teal, + args = list( + id = "test", + data = teal_data(iris = iris), + modules = modules( + module("module_1", server = function(id, data) 101L), + module("module_2", server = function(id, data) 102L) + ) + ), + expr = { + testthat::expect_s4_class(data_rv(), "teal_data") + session$setInputs(`teal_modules-active_tab` = "module_1") + testthat::expect_identical(modules_output$module_1(), 101L) + testthat::expect_null(modules_output$module_2()) + session$setInputs(`teal_modules-active_tab` = "module_2") + testthat::expect_identical(modules_output$module_1(), 101L) + testthat::expect_identical(modules_output$module_2(), 102L) + } + ) + }) + + testthat::it("are called once their tab is selected and data returns reactive `teal_data`", { + shiny::testServer( + app = srv_teal, + args = list( + id = "test", + data = reactive(teal_data(iris = iris)), + modules = modules( + module("module_1", server = function(id, data) 101L), + module("module_2", server = function(id, data) 102L) + ) + ), + expr = { + testthat::expect_s4_class(data_rv(), "teal_data") + session$setInputs(`teal_modules-active_tab` = "module_1") + testthat::expect_identical(modules_output$module_1(), 101L) + testthat::expect_null(modules_output$module_2()) + + session$setInputs(`teal_modules-active_tab` = "module_2") + testthat::expect_identical(modules_output$module_1(), 101L) + testthat::expect_identical(modules_output$module_2(), 102L) + } + ) + }) + + testthat::it("are called once their tab is selected and teal_data_module returns reactive `teal_data`", { + shiny::testServer( + app = srv_teal, + args = list( + id = "test", + data = teal_data_module( + ui = function(id) NULL, + server = function(id) { + moduleServer(id, function(input, output, session) { + reactive(teal_data(iris = iris)) + }) + } + ), + modules = modules( + module("module_1", server = function(id, data) 101L), + module("module_2", server = function(id, data) 102L) + ) + ), + expr = { + testthat::expect_s4_class(data_rv(), "teal_data") + session$setInputs(`teal_modules-active_tab` = "module_1") + testthat::expect_identical(modules_output$module_1(), 101L) + testthat::expect_null(modules_output$module_2()) + + session$setInputs(`teal_modules-active_tab` = "module_2") + testthat::expect_identical(modules_output$module_1(), 101L) + testthat::expect_identical(modules_output$module_2(), 102L) + } + ) + }) + + testthat::it("are called with data argument being `teal_data`", { + shiny::testServer( + app = srv_teal, + args = list( + id = "test", + data = teal_data(iris = iris), + modules = modules( + module("module_1", server = function(id, data) data) + ) + ), + expr = { + testthat::expect_s4_class(data_rv(), "teal_data") + session$setInputs(`teal_modules-active_tab` = "module_1") + testthat::expect_s4_class(modules_output$module_1()(), "teal_data") + } + ) + }) + + testthat::it("are not called when the teal_data_module doesn't return teal_data", { + shiny::testServer( + app = srv_teal, + args = list( + id = "test", + data = teal_data_module( + ui = function(id) NULL, + server = function(id) { + moduleServer(id, function(input, output, session) { + reactive("my error") + }) + } + ), + modules = modules( + module("module_1", server = function(id, data) 101L), + module("module_2", server = function(id, data) 102L) + ) + ), + expr = { + testthat::expect_null(modules_output$module_1()) + testthat::expect_error(data_rv()) + session$setInputs(`teal_modules-active_tab` = "module_1") + testthat::expect_null(modules_output$module_1()) + } + ) + }) + + testthat::it("are not called when the teal_data_module returns validation error", { + shiny::testServer( + app = srv_teal, + args = list( + id = "test", + data = teal_data_module( + ui = function(id) NULL, + server = function(id) { + moduleServer(id, function(input, output, session) { + reactive(validate(need(FALSE, "my error"))) + }) + } + ), + modules = modules( + module("module_1", server = function(id, data) 101L), + module("module_2", server = function(id, data) 102L) + ) + ), + expr = { + testthat::expect_null(modules_output$module_1()) + session$setInputs(`teal_modules-active_tab` = "module_1") + testthat::expect_null(modules_output$module_1()) + } + ) + }) + + testthat::it("are not called when the teal_data_module throw en error", { + shiny::testServer( + app = srv_teal, + args = list( + id = "test", + data = teal_data_module( + ui = function(id) NULL, + server = function(id) { + moduleServer(id, function(input, output, session) { + reactive(validate(need(FALSE, "my error"))) + }) + } + ), + modules = modules( + module("module_1", server = function(id, data) 101L), + module("module_2", server = function(id, data) 102L) + ) + ), + expr = { + testthat::expect_null(modules_output$module_1()) + testthat::expect_error(data_rv()) + session$setInputs(`teal_modules-active_tab` = "module_1") + testthat::expect_null(modules_output$module_1()) + } + ) + }) + + testthat::it("are not called when the teal_data_module returns qenv.error", { + shiny::testServer( + app = srv_teal, + args = list( + id = "test", + data = teal_data_module( + ui = function(id) NULL, + server = function(id) { + moduleServer(id, function(input, output, session) { + reactive(within(teal_data(), stop("my qenv error"))) + }) + } + ), + modules = modules( + module("module_1", server = function(id, data) 101L), + module("module_2", server = function(id, data) 102L) + ) + ), + expr = { + testthat::expect_null(modules_output$module_1()) + testthat::expect_error(data_rv()) + session$setInputs(`teal_modules-active_tab` = "module_1") + testthat::expect_null(modules_output$module_1()) + } + ) + }) + + testthat::it("are receiving reactive data which triggers on change", { + shiny::testServer( + app = srv_teal, + args = list( + id = "test", + data = teal_data_module( + ui = function(id) NULL, + server = function(id) { + moduleServer(id, function(input, output, session) { + eventReactive(input$dataset, { + if (input$dataset == "iris") { + teal_data(iris = iris) + } else if (input$dataset == "mtcars") { + teal_data(mtcars = mtcars) + } + }) + }) + } + ), + modules = modules( + module("module_1", server = function(id, data) data), + module("module_2", server = function(id, data) data) + ) + ), + expr = { + testthat::expect_null(modules_output$module_1()) + + session$setInputs(`data-teal_data_module-data-dataset` = "iris", `teal_modules-active_tab` = "module_1") + testthat::expect_identical( + ls(teal.code::get_env(modules_output$module_1()())), + c("iris", "iris._raw_") + ) + + # comment: can't trigger reactivity in testServer - the change in a reactive input data + # is not propagated to the teal_module(data). Instead we test if the modified data + # is sent to another teal_module + session$setInputs(`data-teal_data_module-data-dataset` = "mtcars", `teal_modules-active_tab` = "module_2") + session$flushReact() + testthat::expect_identical( + ls(teal.code::get_env(modules_output$module_2()())), + c("mtcars", "mtcars._raw_") + ) + } + ) + }) + + testthat::it("are not called again when data changes", { + shiny::testServer( + app = srv_teal, + args = list( + id = "test", + data = teal_data_module( + ui = function(id) NULL, + server = function(id) { + moduleServer(id, function(input, output, session) { + eventReactive(input$dataset, { + if (input$dataset == "iris") { + teal_data(iris = iris) + } else if (input$dataset == "mtcars") { + teal_data(mtcars = mtcars) + } + }) + }) + } + ), + modules = modules( + module("module_1", server = function(id, data) runif(1)) + ) + ), + expr = { + testthat::expect_null(modules_output$module_1()) + session$setInputs( + `data-teal_data_module-data-dataset` = "iris", + `teal_modules-active_tab` = "module_1" + ) + out <- modules_output$module_1() + testthat::expect_true(!is.null(out)) + session$setInputs(`data-teal_data_module-data-dataset` = "mtcars") + testthat::expect_identical(out, modules_output$module_1()) + } + ) + }) + + testthat::it("receives data with datasets == module$datanames", { + shiny::testServer( + app = srv_teal, + args = list( + id = "test", + data = reactive(teal_data(iris = iris, mtcars = mtcars)), + modules = modules( + module("module_1", server = function(id, data) data, datanames = c("iris")) + ) + ), + expr = { + session$setInputs(`teal_modules-active_tab` = "module_1") + testthat::expect_identical(teal.data::datanames(modules_output$module_1()()), "iris") + testthat::expect_identical(modules_output$module_1()()[["iris"]], iris) + } + ) + }) + + testthat::it("throws warning when dataname is not available", { + testthat::skip_if_not_installed("rvest") + shiny::testServer( + app = srv_teal, + args = list( + id = "test", + data = teal_data(mtcars = mtcars), + modules = modules( + module("module_1", server = function(id, data) data, datanames = c("iris")) + ) + ), + expr = { + session$setInputs(`teal_modules-active_tab` = "module_1") + + testthat::expect_equal( + trimws( + rvest::html_text2( + rvest::read_html( + output[["teal_modules-module_1-validate_datanames-shiny_warnings"]]$html + ) + ) + ), + "Dataset iris is missing. No datasets are available in data." + ) + } + ) + }) + + testthat::it("is called and receives data even if datanames in `teal_data` are not sufficient", { + data <- teal_data(iris = iris) + teal.data::datanames(data) <- "iris" + + shiny::testServer( + app = srv_teal, + args = list( + id = "test", + data = reactive(data), + modules = modules( + module("module_1", server = function(id, data) data, datanames = c("iris", "mtcars")) + ) + ), + expr = { + session$setInputs(`teal_modules-active_tab` = "module_1") + testthat::expect_identical(teal.data::datanames(modules_output$module_1()()), "iris") + } + ) + }) + + testthat::it("receives all objects from @env except `DATA._raw_` when `DATA` is present in the @env and module$datanames = \"all\" and @datanames is empty", { # nolint: line_length. + shiny::testServer( + app = srv_teal, + args = list( + id = "test", + data = reactive({ + td <- teal_data(iris = iris, mtcars = mtcars, swiss = swiss, iris_raw = iris) + teal.data::datanames(td) <- character(0) + td + }), + modules = modules( + module("module_1", server = function(id, data) data, datanames = "all") + ) + ), + expr = { + session$setInputs(`teal_modules-active_tab` = "module_1") + testthat::expect_identical( + teal.data::datanames(modules_output$module_1()()), + c("iris", "iris_raw", "mtcars", "swiss") + ) + } + ) + }) + + testthat::it("receives @datanames when module$datanames = \"all\"", { + shiny::testServer( + app = srv_teal, + args = list( + id = "test", + data = reactive({ + td <- teal_data(iris = iris, mtcars = mtcars, swiss = swiss) + teal.data::datanames(td) <- c("iris", "mtcars") + td + }), + modules = modules( + module("module_1", server = function(id, data) data, datanames = "all") + ) + ), + expr = { + session$setInputs(`teal_modules-active_tab` = "module_1") + testthat::expect_identical(teal.data::datanames(modules_output$module_1()()), c("iris", "mtcars")) + } + ) + }) + + testthat::it("receives parent data when module$datanames limited to a child data but join keys are provided", { + parent <- data.frame(id = 1:3, test = letters[1:3]) + child <- data.frame(id = 1:9, parent_id = rep(1:3, each = 3), test2 = letters[1:9]) + data <- teal_data(parent = parent, child = child) + teal.data::join_keys(data) <- teal.data::join_keys( + teal.data::join_key("parent", "child", c(id = "parent_id")) + ) + + shiny::testServer( + app = srv_teal, + args = list( + id = "test", + data = reactive(data), + modules = modules( + module("module_1", server = function(id, data) data, datanames = "child") + ) + ), + expr = { + session$setInputs(`teal_modules-active_tab` = "module_1") + testthat::expect_identical(modules_output$module_1()()[["parent"]], parent) + testthat::expect_identical(modules_output$module_1()()[["child"]], child) + } + ) + }) + + testthat::it("receives extra datanames added in a transform if specified in module$datanames", { + shiny::testServer( + app = srv_teal, + args = list( + id = "test", + data = reactive(teal_data(iris = iris, mtcars = mtcars)), + modules = modules( + module( + label = "module_1", + server = function(id, data) data, + transformers = list( + teal_transform_module( + label = "Dummy", + ui = function(id) div("(does nothing)"), + server = function(id, data) { + moduleServer(id, function(input, output, session) { + reactive(within(data(), swiss <- swiss)) + }) + } + ) + ), + datanames = c("mtcars", "iris", "swiss") + ) + ) + ), + expr = { + session$setInputs(`teal_modules-active_tab` = "module_1") + testthat::expect_identical(teal.data::datanames(modules_output$module_1()()), c("mtcars", "iris", "swiss")) + } + ) + }) + + testthat::it("doesn't receive extra transform datasets not set in @datanames if module$datanames == 'all'", { + shiny::testServer( + app = srv_teal, + args = list( + id = "test", + data = reactive({ + td <- within(teal_data(), { + iris <- iris + mtcars <- mtcars + }) + teal.data::datanames(td) <- c("mtcars", "iris") + td + }), + modules = modules( + module( + label = "module_1", + server = function(id, data) data, + transformers = list( + teal_transform_module( + label = "Dummy", + ui = function(id) div("(does nothing)"), + server = function(id, data) { + moduleServer(id, function(input, output, session) { + reactive(within(data(), swiss <- swiss)) + }) + } + ) + ), + datanames = "all" + ) + ) + ), + expr = { + session$setInputs(`teal_modules-active_tab` = "module_1") + testthat::expect_identical(teal.data::datanames(modules_output$module_1()()), c("mtcars", "iris")) + } + ) + }) + + testthat::it("receives extra transform datasets if module$datanames == 'all' and @datanames empty", { + shiny::testServer( + app = srv_teal, + args = list( + id = "test", + data = reactive({ + within(teal_data(), { + iris <- iris + mtcars <- mtcars + }) + }), + modules = modules( + module( + label = "module_1", + server = function(id, data) data, + transformers = list( + teal_transform_module( + label = "Dummy", + ui = function(id) div("(does nothing)"), + server = function(id, data) { + moduleServer(id, function(input, output, session) { + reactive(within(data(), swiss <- swiss)) + }) + } + ) + ), + datanames = "all" + ) + ) + ), + expr = { + session$setInputs(`teal_modules-active_tab` = "module_1") + testthat::expect_identical(teal.data::datanames(modules_output$module_1()()), c("iris", "mtcars", "swiss")) + } + ) + }) + + testthat::it("doesn't receive extra datanames in a transform if not specified in module$datanames", { + shiny::testServer( + app = srv_teal, + args = list( + id = "test", + data = reactive(teal_data(iris = iris, mtcars = mtcars)), + modules = modules( + module( + label = "module_1", + server = function(id, data) data, + transformers = list( + teal_transform_module( + label = "Dummy", + ui = function(id) div("(does nothing)"), + server = function(id, data) { + moduleServer(id, function(input, output, session) { + reactive(within(data(), swiss <- swiss)) + }) + } + ) + ), + datanames = c("iris", "mtcars") + ) + ) + ), + expr = { + session$setInputs(`teal_modules-active_tab` = "module_1") + testthat::expect_identical(teal.data::datanames(modules_output$module_1()()), c("iris", "mtcars")) + } + ) + }) + + testthat::it("srv_teal_module.teal_module does not pass data if not in the args explicitly", { + shiny::testServer( + app = srv_teal, + args = list( + id = "test", + data = teal.data::teal_data(iris = iris, mtcars = mtcars), + modules = modules( + module("module_1", server = function(id, ...) { + list(...)$data + }) + ) + ), + expr = { + session$setInputs(`teal_modules-active_tab` = "module_1") + session$flushReact() + testthat::expect_null(modules_output$module_1()) + } + ) + }) + + testthat::it("srv_teal_module.teal_module passes (deprecated) datasets to the server module", { + testthat::expect_warning( + shiny::testServer( + app = srv_teal, + args = list( + id = "test", + data = teal.data::teal_data(iris = iris, mtcars = mtcars), + modules = modules( + module("module_1", server = function(id, datasets) datasets) + ) + ), + expr = { + session$setInputs(`teal_modules-active_tab` = "module_1") + testthat::expect_s3_class(modules_output$module_1(), "FilteredData") + } + ), + "`datasets` argument in the server is deprecated and will be removed in the next release" + ) + }) + + testthat::it("srv_teal_module.teal_module passes server_args to the ...", { + shiny::testServer( + app = srv_teal, + args = list( + id = "test", + data = teal.data::teal_data(iris = iris, mtcars = mtcars), + modules = modules( + module( + "module_1", + server = function(id, data, ...) { + data + }, + server_args = list(x = 1L, y = 2L) + ) + ) + ), + expr = { + session$setInputs(`teal_modules-active_tab` = "module_1") + testthat::expect_identical( + modules$children$module_1$server_args, + list(x = 1L, y = 2L) + ) + } + ) + }) + + testthat::it("srv_teal_module.teal_module passes filter_panel_api if specified", { + shiny::testServer( + app = srv_teal, + args = list( + id = "test", + data = teal.data::teal_data(iris = iris, mtcars = mtcars), + modules = modules( + module("module_1", server = function(id, filter_panel_api) filter_panel_api) + ) + ), + expr = { + session$setInputs(`teal_modules-active_tab` = "module_1") + testthat::expect_s3_class(modules_output$module_1(), "FilterPanelAPI") + } + ) + }) + + testthat::it("srv_teal_module.teal_module passes Reporter if specified", { + shiny::testServer( + app = srv_teal, + args = list( + id = "test", + data = teal.data::teal_data(iris = iris, mtcars = mtcars), + modules = modules( + module("module_1", server = function(id, reporter) reporter) + ) + ), + expr = { + session$setInputs(`teal_modules-active_tab` = "module_1") + testthat::expect_s3_class(modules_output$module_1(), "Reporter") + } + ) + }) + + testthat::it("reveives code of datasets used in transform even if not specified explicitly", { + testthat::it("receives all possible objects while those not specified in module$datanames are unfiltered", { + shiny::testServer( + app = srv_teal, + args = list( + id = "test", + data = reactive(within(teal.data::teal_data(), { + iris <- iris + mtcars <- mtcars + })), + filter = teal_slices( + teal_slice(dataname = "mtcars", varname = "cyl", selected = "4"), + teal_slice(dataname = "iris", varname = "Species", selected = "versicolor") + ), + modules = modules( + module( + label = "module_1", + server = function(id, data) data, + datanames = c("new_list"), + transformers = list( + teal_transform_module( + ui = function(id) NULL, + server = function(id, data) { + moduleServer(id, function(input, output, session) { + reactive({ + within(data(), new_list <- list(iris = iris, mtcars = mtcars)) + }) + }) + } + ) + ) + ) + ) + ), + expr = { + session$setInputs(`teal_modules-active_tab` = "module_1") + session$flushReact() + testthat::expect_identical(teal.data::datanames(modules_output$module_1()()), "new_list") + testthat::expect_identical(modules_output$module_1()()[["new_list"]]$mtcars, mtcars) + testthat::expect_identical(modules_output$module_1()()[["new_list"]]$iris, iris) + testthat::expect_identical( + teal.code::get_code(modules_output$module_1()()), + paste( + c( + "iris <- iris", + "mtcars <- mtcars", + 'stopifnot(rlang::hash(iris) == "34844aba7bde36f5a34f6d8e39803508")', + 'stopifnot(rlang::hash(mtcars) == "d0487363db4e6cc64fdb740cb6617fc0")', + "new_list <- list(iris = iris, mtcars = mtcars)" + ), + collapse = "\n" + ) + ) + } + ) + }) + }) }) -testthat::test_that("srv_teal when teal_data_rv changes, datasets_reactive is initialized as list of FilteredData", { - data <- teal.data::teal_data(iris1 = iris, mtcars1 = mtcars) - shiny::testServer( - app = srv_teal, - args = list( - id = "test", - teal_data_rv = reactiveVal(NULL), - modules = modules( - example_module(label = "iris_tab"), - example_module(label = "mtcars_tab") - ) - ), - expr = { - teal_data_rv(data) - checkmate::expect_list(datasets_reactive(), types = "FilteredData") - } - ) +testthat::describe("srv_teal filters", { + testthat::describe("slicesGlobal", { + testthat::it("is set to initial filters when !module_specific", { + init_filter <- teal_slices( + teal_slice("iris", "Species"), + teal_slice("mtcars", "cyl"), + mapping = list( + global_filters = c("iris Species", "mtcars cyl") + ), + module_specific = FALSE + ) + shiny::testServer( + app = srv_teal, + args = list( + id = "test", + data = teal.data::teal_data(iris = iris, mtcars = mtcars), + modules = modules(example_module(label = "module-1"), example_module(label = "module-2")), + filter = init_filter + ), + expr = { + testthat::expect_identical(slices_global$all_slices(), init_filter) + } + ) + }) + testthat::it("is set to initial filters with resolved attr(, 'mapping')$ when `module_specific`", { + init_filter <- teal_slices( + teal_slice("iris", "Species"), + teal_slice("mtcars", "cyl"), + module_specific = TRUE, + mapping = list( + global_filters = c("iris Species", "mtcars cyl") + ) + ) + shiny::testServer( + app = srv_teal, + args = list( + id = "test", + data = teal.data::teal_data(iris = iris, mtcars = mtcars), + modules = modules(example_module(label = "module-1"), example_module(label = "module-2")), + filter = init_filter + ), + expr = { + setdiff_teal_slices <- getFromNamespace("setdiff_teal_slices", "teal.slice") + testthat::expect_length(setdiff_teal_slices(slices_global$all_slices(), init_filter), 0) + testthat::expect_identical( + attr(slices_global$all_slices(), "mapping"), + list( + `module-1` = c("iris Species", "mtcars cyl"), + `module-2` = c("iris Species", "mtcars cyl") + ) + ) + } + ) + }) + testthat::it("slices in slicesGlobal and in FilteredData refer to the same object", { + init_filter <- teal_slices( + teal_slice("iris", "Species"), + teal_slice("mtcars", "cyl"), + module_specific = TRUE, + mapping = list( + global_filters = c("iris Species", "mtcars cyl") + ) + ) + shiny::testServer( + app = srv_teal, + args = list( + id = "test", + data = teal.data::teal_data(iris = iris, mtcars = mtcars), + modules = modules(example_module(label = "module_1"), example_module(label = "module_2")), + filter = init_filter + ), + expr = { + session$setInputs(`teal_modules-active_tab` = "module_1") + session$setInputs(`teal_modules-active_tab` = "module_2") + testthat::expect_true(identical( + slices_global$all_slices()[[1]], + slices_global$module_slices_api[["module_1"]]$get_filter_state()[[1]] + )) + testthat::expect_true(identical( + slices_global$all_slices()[[1]], + slices_global$module_slices_api[["module_2"]]$get_filter_state()[[1]] + )) + } + ) + }) + testthat::it("appends new slice and activates in $global_filters when added in a module if !module_specific", { + shiny::testServer( + app = srv_teal, + args = list( + id = "test", + data = reactive(within(teal.data::teal_data(), { + iris <- iris + mtcars <- mtcars + })), + modules = modules( + module("module_1", server = function(id, data) data), + module("module_2", server = function(id, data) data) + ), + filter = teal_slices(module_specific = FALSE) + ), + expr = { + session$setInputs(`teal_modules-active_tab` = "module_1") + session$setInputs(`teal_modules-active_tab` = "module_2") + session$setInputs(`teal_modules-module_2-filter_panel-filters-iris-iris-filter-var_to_add` = "Species") + testthat::expect_true(is_slices_equivalent( + x = slices_global$all_slices(), + y = teal_slices( + teal_slice("iris", "Species", choices = unique(iris$Species), selected = unique(iris$Species)), + mapping = list(global_filters = "iris Species"), + module_specific = FALSE + ) + )) + } + ) + }) + testthat::it("deactivates in $global_filters when removed from module if !module_specific", { + shiny::testServer( + app = srv_teal, + args = list( + id = "test", + data = reactive(within(teal.data::teal_data(), { + iris <- iris + mtcars <- mtcars + })), + modules = modules( + module("module_1", server = function(id, data) data), + module("module_2", server = function(id, data) data) + ), + filter = teal_slices( + teal_slice("iris", varname = "Species", selected = "versicolor"), + module_specific = FALSE + ) + ), + expr = { + session$setInputs(`teal_modules-active_tab` = "module_1") + session$setInputs(`teal_modules-active_tab` = "module_2") + session$setInputs(`teal_modules-module_2-filter_panel-filters-iris-filter-iris_Species-remove` = "Species") + testthat::expect_true(is_slices_equivalent( + x = slices_global$all_slices(), + y = teal_slices( + teal_slice("iris", "Species", choices = unique(iris$Species), selected = "versicolor"), + mapping = list(global_filters = character(0)), + module_specific = FALSE + ) + )) + } + ) + }) + testthat::it("appends new slice and activates in $ when added in a module if module_specific", { + shiny::testServer( + app = srv_teal, + args = list( + id = "test", + data = reactive(within(teal.data::teal_data(), { + iris <- iris + mtcars <- mtcars + })), + modules = modules( + module("module_1", server = function(id, data) data), + module("module_2", server = function(id, data) data) + ), + filter = teal_slices(module_specific = TRUE) + ), + expr = { + session$setInputs(`teal_modules-active_tab` = "module_1") + session$setInputs(`teal_modules-active_tab` = "module_2") + session$setInputs(`teal_modules-module_2-filter_panel-filters-iris-iris-filter-var_to_add` = "Species") + testthat::expect_true(is_slices_equivalent( + x = slices_global$all_slices(), + y = teal_slices( + teal_slice("iris", "Species", choices = unique(iris$Species), selected = unique(iris$Species)), + mapping = list(module_1 = character(0), module_2 = "iris Species"), + module_specific = TRUE + ) + )) + } + ) + }) + testthat::it("appends added 'duplicated' slice and makes new-slice$id unique", { + shiny::testServer( + app = srv_teal, + args = list( + id = "test", + data = reactive(within(teal.data::teal_data(), { + iris <- iris + mtcars <- mtcars + })), + modules = modules( + module("module_1", server = function(id, data) data), + module("module_2", server = function(id, data) data) + ), + filter = teal_slices( + teal_slice("iris", "Species", choices = unique(iris$Species), selected = unique(iris$Species)), + mapping = list(global_filters = character(0)) + ) + ), + expr = { + session$setInputs(`teal_modules-active_tab` = "module_1") + session$setInputs(`teal_modules-module_1-filter_panel-filters-iris-iris-filter-var_to_add` = "Species") + session$flushReact() + testthat::expect_true(is_slices_equivalent( + x = slices_global$all_slices(), + y = teal_slices( + teal_slice("iris", "Species", choices = unique(iris$Species), selected = unique(iris$Species)), + teal_slice("iris", "Species", + choices = unique(iris$Species), selected = unique(iris$Species), + id = "iris Species_1" + ), + mapping = list(global_filters = "iris Species_1"), + module_specific = FALSE + ) + )) + } + ) + }) + testthat::it("deactivates in $ when removed from module if module_specific", { + shiny::testServer( + app = srv_teal, + args = list( + id = "test", + data = reactive(within(teal.data::teal_data(), { + iris <- iris + mtcars <- mtcars + })), + modules = modules( + module("module_1", server = function(id, data) data), + module("module_2", server = function(id, data) data) + ), + filter = teal_slices( + teal_slice("iris", varname = "Species", selected = "versicolor"), + mapping = list(global_filters = "iris Species"), + module_specific = TRUE + ) + ), + expr = { + session$setInputs(`teal_modules-active_tab` = "module_1") + session$setInputs(`teal_modules-active_tab` = "module_2") + session$setInputs(`teal_modules-module_2-filter_panel-filters-iris-filter-iris_Species-remove` = "Species") + testthat::expect_true(is_slices_equivalent( + x = slices_global$all_slices(), + y = teal_slices( + teal_slice("iris", "Species", choices = unique(iris$Species), selected = "versicolor"), + mapping = list(module_1 = "iris Species", module_2 = character(0)), + module_specific = TRUE + ) + )) + } + ) + }) + testthat::it("auto-resolves to mapping$ when setting slices with mapping$global_filters in module_specific ", { + shiny::testServer( + app = srv_teal, + args = list( + id = "test", + data = reactive(within(teal.data::teal_data(), { + iris <- iris + mtcars <- mtcars + })), + modules = modules( + module("module_1", server = function(id, data) data), + module("module_2", server = function(id, data) data) + ), + filter = teal_slices(module_specific = TRUE) + ), + expr = { + testthat::skip("need a fix in a .slicesGlobal") + session$setInputs(`teal_modules-active_tab` = "module_1") + session$setInputs(`teal_modules-active_tab` = "module_2") + slices_global$slices_set(teal_slices( + teal_slice("iris", "Species"), + mapping = list(global_filters = "iris Species") + )) + testthat::expect_true(is_slices_equivalent( + x = slices_global$all_slices(), + y = teal_slices( + teal_slice("iris", "Species", choices = unique(iris$Species), selected = unique(iris$Species)), + mapping = list(module_1 = "iris Species", module_2 = "iris Species"), + module_specific = TRUE + ) + )) + } + ) + }) + testthat::it("sets filters from mapping$ to all modules' FilteredData when !module_specific", { + shiny::testServer( + app = srv_teal, + args = list( + id = "test", + data = reactive(within(teal.data::teal_data(), { + iris <- iris + mtcars <- mtcars + })), + modules = modules( + module("module_1", server = function(id, data) data), + module("module_2", server = function(id, data) data) + ), + filter = teal_slices(module_specific = FALSE) + ), + expr = { + session$setInputs(`teal_modules-active_tab` = "module_1") + session$setInputs(`teal_modules-active_tab` = "module_2") + slices_global$slices_append(teal_slices(teal_slice("iris", "Species", selected = "versicolor"))) + slices_global$slices_active(list(global_filter = "iris Species")) + session$flushReact() + expected_slices <- slices_global$all_slices() + + testthat::expect_true(is_slices_equivalent( + x = slices_global$all_slices(), + y = slices_global$module_slices_api[["global_filters"]]$get_filter_state(), + with_attrs = FALSE + )) + } + ) + }) + testthat::it("sets filters from mapping$ to module's FilteredData when module_specific", { + shiny::testServer( + app = srv_teal, + args = list( + id = "test", + data = reactive(within(teal.data::teal_data(), { + iris <- iris + mtcars <- mtcars + })), + modules = modules( + module("module_1", server = function(id, data) data), + module("module_2", server = function(id, data) data) + ), + filter = teal_slices(module_specific = TRUE) + ), + expr = { + session$setInputs(`teal_modules-active_tab` = "module_1") + session$setInputs(`teal_modules-active_tab` = "module_2") + slices_global$slices_append(teal_slices(teal_slice("iris", "Species", selected = "versicolor"))) + slices_global$slices_active(list(module_1 = "iris Species")) + session$flushReact() + expected_slices <- slices_global$all_slices() + testthat::expect_true(is_slices_equivalent( + x = slices_global$all_slices(), + y = slices_global$module_slices_api[["module_1"]]$get_filter_state(), + with_attrs = FALSE + )) + testthat::expect_true(is_slices_equivalent( + x = teal_slices(), + y = slices_global$module_slices_api[["module_2"]]$get_filter_state(), + with_attrs = FALSE + )) + } + ) + }) + testthat::it("sets filters from mapping$global_filters to all modules' FilteredData when module_specific", { + shiny::testServer( + app = srv_teal, + args = list( + id = "test", + data = reactive(within(teal.data::teal_data(), { + iris <- iris + mtcars <- mtcars + })), + modules = modules( + module("module_1", server = function(id, data) data), + module("module_2", server = function(id, data) data) + ), + filter = teal_slices(module_specific = TRUE) + ), + expr = { + session$setInputs(`teal_modules-active_tab` = "module_1") + session$setInputs(`teal_modules-active_tab` = "module_2") + slices_global$slices_append(teal_slices(teal_slice("iris", "Species", selected = "versicolor"))) + slices_global$slices_active(list(global_filters = "iris Species")) + session$flushReact() + expected_slices <- slices_global$all_slices() + testthat::expect_true(is_slices_equivalent( + x = slices_global$all_slices(), + y = slices_global$module_slices_api[["module_1"]]$get_filter_state(), + with_attrs = FALSE + )) + testthat::expect_true(is_slices_equivalent( + x = slices_global$all_slices(), + y = slices_global$module_slices_api[["module_2"]]$get_filter_state(), + with_attrs = FALSE + )) + } + ) + }) + testthat::it("change in the slicesGlobal causes module's data filtering", { + existing_filters <- teal_slices( + teal_slice(dataname = "iris", varname = "Species", selected = "versicolor"), + teal_slice(dataname = "mtcars", varname = "cyl", selected = 6) + ) + shiny::testServer( + app = srv_teal, + args = list( + id = "test", + data = reactive(within(teal.data::teal_data(), { + iris <- iris + mtcars <- mtcars + })), + filter = existing_filters, + modules = modules(module("module_1", server = function(id, data) data)) + ), + expr = { + session$setInputs(`teal_modules-active_tab` = "module_1") + slices_global$slices_set( + teal_slices( + teal_slice("mtcars", varname = "cyl", selected = "4") + ) + ) + session$flushReact() + # iris is not active + testthat::expect_identical(modules_output$module_1()()[["iris"]], iris) + testthat::expect_identical(modules_output$module_1()()[["iris._raw_"]], iris) + # mtcars has been modified + expected_mtcars <- subset(mtcars, cyl == 4) + testthat::expect_identical(modules_output$module_1()()[["mtcars"]], expected_mtcars) + testthat::expect_identical(modules_output$module_1()()[["mtcars._raw_"]], mtcars) + + expected_code <- paste0( + c( + "iris <- iris", + "mtcars <- mtcars", + sprintf('stopifnot(rlang::hash(iris) == "%s")', rlang::hash(iris)), + sprintf('stopifnot(rlang::hash(mtcars) == "%s")', rlang::hash(mtcars)), + "iris._raw_ <- iris", + "mtcars._raw_ <- mtcars", + "mtcars <- dplyr::filter(mtcars, cyl == 4)" + ), + collapse = "\n" + ) + testthat::expect_identical(teal.code::get_code(modules_output$module_1()()), expected_code) + } + ) + }) + }) + + testthat::describe("mapping table", { + testthat::it("returns no rows if no filters set", { + shiny::testServer( + app = srv_teal, + args = list( + id = "test", + data = teal.data::teal_data(iris = iris, mtcars = mtcars), + modules = modules(module("module_1", server = function(id, data) data)) + ), + expr = { + session$setInputs("teal_modules-active_tab" = "module_1") + session$flushReact() + testthat::expect_equal( + mapping_table(), + data.frame( + `Global filters` = logical(0), + row.names = integer(0), + check.names = FALSE + ) + ) + } + ) + }) + testthat::it("returns global filters with active=true, inactive=false, unavailable=na", { + shiny::testServer( + app = srv_teal, + args = list( + id = "test", + data = teal.data::teal_data(iris = iris, mtcars = mtcars), + modules = modules( + module("module_1", server = function(id, data) data), + module("module_2", server = function(id, data) data) + ), + filter = teal_slices( + teal_slice("iris", "Species"), + teal_slice("mtcars", "cyl"), + teal_slice("unknown", "unavailable"), + mapping = list(global_filters = "iris Species") + ) + ), + expr = { + testthat::expect_warning( + session$setInputs("teal_modules-active_tab" = "module_1"), + "Filter 'unknown unavailable' refers to dataname not available in 'data'" + ) + session$flushReact() + testthat::expect_identical( + mapping_table(), + data.frame( + `Global filters` = c(TRUE, FALSE, NA), + row.names = c("iris Species", "mtcars cyl", "unknown unavailable"), + check.names = FALSE + ) + ) + } + ) + }) + + testthat::it("returns column per module with active=true, inactive=false, unavailable=na", { + shiny::testServer( + app = srv_teal, + args = list( + id = "test", + data = teal.data::teal_data(iris = iris, mtcars = mtcars), + modules = modules( + module("module_1", server = function(id, data) data), + module("module_2", server = function(id, data) data) + ), + filter = teal_slices( + teal_slice("iris", "Species"), + teal_slice("mtcars", "cyl"), + teal_slice("unknown", "unavailable"), + module_specific = TRUE, + mapping = list(module_1 = "iris Species", module_2 = "mtcars cyl") + ) + ), + expr = { + testthat::expect_warning( + session$setInputs("teal_modules-active_tab" = "module_1"), + "Filter 'unknown unavailable' refers to dataname not available in 'data'" + ) + session$flushReact() + testthat::expect_identical( + mapping_table(), + data.frame( + module_1 = c(TRUE, FALSE, NA), + module_2 = c(FALSE, TRUE, NA), + row.names = c("iris Species", "mtcars cyl", "unknown unavailable"), + check.names = FALSE + ) + ) + } + ) + }) + + testthat::it("what happens when module$label is duplicated (when nested modules)", { + testthat::skip("todo") + }) + }) }) -testthat::test_that("srv_teal initialized datasets_reactive (list) reflects modules structure", { - data <- teal.data::teal_data(iris1 = iris, mtcars1 = mtcars) - shiny::testServer( - app = srv_teal, - args = list( - id = "test", - teal_data_rv = reactiveVal(data), - modules = modules( - example_module("iris_tab"), - modules(label = "tab", example_module("iris_tab"), example_module("mtcars_tab")) - ) - ), - expr = { - teal_data_rv(data) - testthat::expect_named(datasets_reactive(), c("iris_tab", "tab")) - testthat::expect_named(datasets_reactive()$tab, c("iris_tab", "mtcars_tab")) - } - ) +testthat::describe("srv_teal data reload", { + testthat::it("sets back the same active filters in each module", { + testthat::skip("todo") + }) + testthat::it("doesn't fail when teal_data has no datasets", { + testthat::skip("todo") + }) }) -testthat::test_that("srv_teal initialized data containing same FilteredData when the filter is global", { - data <- teal.data::teal_data(iris1 = iris, mtcars1 = mtcars) - shiny::testServer( - app = srv_teal, - args = list( - id = "test", - teal_data_rv = reactiveVal(data), - modules = modules( - example_module("iris_tab"), - modules(label = "tab", example_module("iris_tab"), example_module("mtcars_tab")) - ), - filter = teal_slices(module_specific = FALSE) - ), - expr = { - teal_data_rv(data) - unlisted_fd <- unlist(datasets_reactive(), use.names = FALSE) - testthat::expect_identical(unlisted_fd[[1]], unlisted_fd[[2]]) - testthat::expect_identical(unlisted_fd[[2]], unlisted_fd[[3]]) - } - ) +testthat::describe("srv_teal teal_module(s) transformer", { + testthat::it("evaluates custom qenv call and pass update teal_data to the module", { + shiny::testServer( + app = srv_teal, + args = list( + id = "test", + data = teal.data::teal_data(iris = iris, mtcars = mtcars), + modules = modules( + module( + label = "module_1", + server = function(id, data) data, + transformers = transform_list[c("iris", "mtcars")] + ) + ) + ), + expr = { + session$setInputs(`teal_modules-active_tab` = "module_1") + testthat::expect_identical(modules_output$module_1()()[["iris"]], head(iris)) + testthat::expect_identical(modules_output$module_1()()[["iris._raw_"]], iris) + testthat::expect_identical(modules_output$module_1()()[["mtcars"]], head(mtcars)) + testthat::expect_identical(modules_output$module_1()()[["mtcars._raw_"]], mtcars) + } + ) + }) + + testthat::it("evaluates custom qenv call after filter is applied", { + shiny::testServer( + app = srv_teal, + args = list( + id = "test", + data = reactive(within(teal.data::teal_data(), { + iris <- iris + mtcars <- mtcars + })), + filter = teal_slices( + teal_slice(dataname = "iris", varname = "Species", selected = "versicolor"), + teal_slice(dataname = "mtcars", varname = "cyl", selected = 6) + ), + modules = modules( + module( + label = "module_1", + server = function(id, data) data, + transformers = transform_list[c("iris", "mtcars")] + ) + ) + ), + expr = { + session$setInputs(`teal_modules-active_tab` = "module_1") + + expected_iris <- subset(iris, Species == "versicolor") + rownames(expected_iris) <- NULL + expected_iris <- head(expected_iris) + testthat::expect_identical(modules_output$module_1()()[["iris"]], expected_iris) + testthat::expect_identical(modules_output$module_1()()[["iris._raw_"]], iris) + testthat::expect_identical(modules_output$module_1()()[["mtcars"]], head(subset(mtcars, cyl == 6))) + testthat::expect_identical(modules_output$module_1()()[["mtcars._raw_"]], mtcars) + + expected_code <- paste(collapse = "\n", c( + "iris <- iris", + "mtcars <- mtcars", + sprintf('stopifnot(rlang::hash(iris) == "%s")', rlang::hash(iris)), + sprintf('stopifnot(rlang::hash(mtcars) == "%s")', rlang::hash(mtcars)), + "iris._raw_ <- iris", + "mtcars._raw_ <- mtcars", + 'iris <- dplyr::filter(iris, Species == "versicolor")', + "mtcars <- dplyr::filter(mtcars, cyl == 6)", + "iris <- head(iris, n = 6)", + "mtcars <- head(mtcars, n = 6)" + )) + testthat::expect_identical( + teal.code::get_code(modules_output$module_1()()), + expected_code + ) + } + ) + }) + + testthat::it("is reactive to the filter changes", { + shiny::testServer( + app = srv_teal, + args = list( + id = "test", + data = reactive(within(teal.data::teal_data(), { + iris <- iris + mtcars <- mtcars + })), + modules = modules( + module( + label = "module_1", + server = function(id, data) data, + transformers = transform_list[c("iris", "mtcars")] + ) + ) + ), + expr = { + session$setInputs(`teal_modules-active_tab` = "module_1") + slices_global$slices_set( + teal_slices(teal_slice(dataname = "mtcars", varname = "cyl", selected = "4")) + ) + session$flushReact() + + testthat::expect_identical(modules_output$module_1()()[["iris"]], head(iris)) + testthat::expect_identical(modules_output$module_1()()[["iris._raw_"]], iris) + testthat::expect_identical(modules_output$module_1()()[["mtcars"]], head(subset(mtcars, cyl == 4))) + testthat::expect_identical(modules_output$module_1()()[["mtcars._raw_"]], mtcars) + + expected_code <- paste(collapse = "\n", c( + "iris <- iris", + "mtcars <- mtcars", + sprintf('stopifnot(rlang::hash(iris) == "%s")', rlang::hash(iris)), + sprintf('stopifnot(rlang::hash(mtcars) == "%s")', rlang::hash(mtcars)), + "iris._raw_ <- iris", + "mtcars._raw_ <- mtcars", + "mtcars <- dplyr::filter(mtcars, cyl == 4)", + "iris <- head(iris, n = 6)", + "mtcars <- head(mtcars, n = 6)" + )) + testthat::expect_identical( + teal.code::get_code(modules_output$module_1()()), + expected_code + ) + } + ) + }) + + testthat::it("receives all possible objects while those not specified in module$datanames are unfiltered", { + shiny::testServer( + app = srv_teal, + args = list( + id = "test", + data = reactive(within(teal.data::teal_data(), { + iris <- iris + mtcars <- mtcars + })), + filter = teal_slices( + teal_slice(dataname = "mtcars", varname = "cyl", selected = "4"), + teal_slice(dataname = "iris", varname = "Species", selected = "versicolor") + ), + modules = modules( + module( + label = "module_1", + server = function(id, data) data, + datanames = c("iris", "data_from_transform"), + transformers = list( + teal_transform_module( + ui = function(id) NULL, + server = function(id, data) { + moduleServer(id, function(input, output, session) { + reactive({ + within(data(), data_from_transform <- list(iris = iris, mtcars = mtcars)) + }) + }) + } + ) + ) + ) + ) + ), + expr = { + session$setInputs(`teal_modules-active_tab` = "module_1") + session$flushReact() + data_from_transform <- modules_output$module_1()()[["data_from_transform"]] + testthat::expect_identical(data_from_transform$mtcars, mtcars) + expected_iris <- iris[iris$Species == "versicolor", ] + rownames(expected_iris) <- NULL + testthat::expect_identical(data_from_transform$iris, expected_iris) + } + ) + }) + + testthat::it("fails when transformer doesn't return reactive", { + testthat::expect_error( + testServer( + app = srv_teal, + args = list( + id = "test", + data = teal.data::teal_data(iris = iris), + modules = modules( + module( + server = function(id, data) data, + transformers = list( + teal_transform_module( + ui = function(id) NULL, + server = function(id, data) "whatever" + ) + ) + ) + ) + ), + expr = {} + ), + "must return a reactive expression" + ) + }) + + testthat::it("continues when transformer throws validation error and returns unchanged data", { + shiny::testServer( + app = srv_teal, + args = list( + id = "test", + data = teal.data::teal_data(iris = iris), + modules = modules( + module( + label = "module_1", + server = function(id, data) data, + transformers = list( + teal_transform_module( + ui = function(id) NULL, + server = function(id, data) { + reactive(validate(need(FALSE, "my error"))) + } + ) + ) + ) + ) + ), + expr = { + session$setInputs(`teal_modules-active_tab` = "module_1") + testthat::expect_identical(modules_output$module_1()()[["iris"]], iris) + testthat::expect_identical(modules_output$module_1()()[["iris._raw_"]], iris) + } + ) + }) + + testthat::it("continues when transformer throws validation error and returns unchanged data", { + shiny::testServer( + app = srv_teal, + args = list( + id = "test", + data = teal.data::teal_data(iris = iris), + modules = modules( + module( + label = "module_1", + server = function(id, data) data, + transformers = list( + teal_transform_module( + ui = function(id) NULL, + server = function(id, data) { + reactive(validate(need(FALSE, "my error"))) + } + ) + ) + ) + ) + ), + expr = { + session$setInputs(`teal_modules-active_tab` = "module_1") + testthat::expect_identical(modules_output$module_1()()[["iris"]], iris) + testthat::expect_identical(modules_output$module_1()()[["iris._raw_"]], iris) + } + ) + }) + + testthat::it("continues when transformer throws qenv error and returns unchanged data", { + testthat::skip("todo") + }) + testthat::it("upstream data change is updated on transformer fallback", { + shiny::testServer( + app = srv_teal, + args = list( + id = "test", + data = teal.data::teal_data(iris = iris, mtcars = mtcars), + modules = modules( + module( + label = "module_1", + server = function(id, data) data, + transformers = transform_list[c("iris", "fail")] + ) + ) + ), + expr = { + session$setInputs("teal_modules-active_tab" = "module_1") + new_row_size <- 14 + session$setInputs("teal_modules-module_1-data_transform-transform_module-data-n" = new_row_size) + session$flushReact() + + testthat::expect_equal(nrow(modules_output$module_1()()[["iris"]]), new_row_size) + } + ) + }) + + testthat::it("upstream data change with double reactivity resolves with correct this/that", { + shiny::testServer( + app = srv_teal, + args = list( + id = "test", + data = teal.data::teal_data(iris = iris, mtcars = mtcars), + modules = modules( + module( + label = "module_1", + server = function(id, data) data, + transformers = transform_list[c("iris", "fail")] + ) + ) + ), + expr = { + session$setInputs("teal_modules-active_tab" = "module_1") + + session$setInputs( + "teal_modules-module_1-data_transform-transform_module-data-n" = 12, + "teal_modules-module_1-data_transform-transform_module_1-data-add_error" = FALSE + ) + session$flushReact() + + testthat::expect_equal(nrow(modules_output$module_1()()[["iris"]]), 6) + } + ) + }) + + testthat::it("continues when transformer throws qenv error and returns unchanged data") + + testthat::it("isn't called when `data` is not teal_data", { + testthat::skip("todo") + }) + # when reactive returned teal_data_module is not triggered (for example when button isn't clicked) }) -testthat::test_that("srv_teal initialized data containing different FilteredData when the filter is module_specific", { - data <- teal.data::teal_data(iris1 = iris, mtcars1 = mtcars) - shiny::testServer( - app = srv_teal, - args = list( - id = "test", - teal_data_rv = reactiveVal(data), - modules = modules( - example_module("iris_tab"), - modules(label = "tab", example_module("iris_tab"), example_module("mtcars_tab")) - ), - filter = teal_slices(module_specific = TRUE) - ), - expr = { - teal_data_rv(data) - unlisted_fd <- unlist(datasets_reactive(), use.names = FALSE) - testthat::expect_false(identical(unlisted_fd[[1]], unlisted_fd[[2]])) - testthat::expect_false(identical(unlisted_fd[[2]], unlisted_fd[[3]])) - } - ) +testthat::describe("srv_teal summary table", { + testthat::it("displays Obs only column if all datasets have no join keys", { + shiny::testServer( + app = srv_teal, + args = list( + id = "test", + data = reactive(within(teal.data::teal_data(), { + iris <- iris + mtcars <- mtcars + })), + modules = modules(module("module_1", server = function(id, data) data)) + ), + expr = { + session$setInputs("teal_modules-active_tab" = "module_1") + session$flushReact() + testthat::expect_identical( + module_output_table(output, "module_1"), + data.frame( + "Data Name" = c("iris", "mtcars"), + Obs = c("150/150", "32/32"), + check.names = FALSE + ) + ) + } + ) + }) + + testthat::it("displays Subjects with count based on foreign key column", { + data <- teal.data::teal_data( + a = data.frame(id = seq(3), name = letters[seq(3)]), + b = data.frame(id = rep(seq(3), 2), id2 = seq(6), value = letters[seq(6)]) + ) + teal.data::join_keys(data) <- teal.data::join_keys( + teal.data::join_key("a", "b", keys = "id") + ) + teal.data::datanames(data) <- c("a", "b") + + shiny::testServer( + app = srv_teal, + args = list( + id = "test", + data = data, + modules = modules(module("module_1", server = function(id, data) data)) + ), + expr = { + session$setInputs("teal_modules-active_tab" = "module_1") + session$flushReact() + testthat::expect_identical( + module_output_table(output, "module_1"), + data.frame( + "Data Name" = c("a", "b"), + Obs = c("3/3", "6/6"), + Subjects = c("", "3/3"), + check.names = FALSE + ) + ) + } + ) + }) + + testthat::it("displays parent's Subjects with count based on primary key", { + data <- teal.data::teal_data( + a = data.frame(id = seq(3), name = letters[seq(3)]), + b = data.frame(id = rep(seq(3), 2), id2 = seq(6), value = letters[seq(6)]) + ) + teal.data::join_keys(data) <- teal.data::join_keys( + teal.data::join_key("a", keys = "id"), + teal.data::join_key("b", keys = c("id", "id2")) + ) + teal.data::datanames(data) <- c("a", "b") + + shiny::testServer( + app = srv_teal, + args = list( + id = "test", + data = data, + modules = modules(module("module_1", server = function(id, data) data)) + ), + expr = { + session$setInputs("teal_modules-active_tab" = "module_1") + session$flushReact() + testthat::expect_identical( + module_output_table(output, "module_1"), + data.frame( + "Data Name" = c("a", "b"), + Obs = c("3/3", "6/6"), + Subjects = c("3/3", "6/6"), + check.names = FALSE + ) + ) + } + ) + }) + + testthat::it("displays parent's Subjects with count based on primary and foreign key", { + data <- teal.data::teal_data( + a = data.frame(id = seq(3), name = letters[seq(3)]), + b = data.frame(id = rep(seq(3), 2), id2 = seq(6), value = letters[seq(6)]) + ) + teal.data::join_keys(data) <- teal.data::join_keys( + teal.data::join_key("a", keys = "id"), + teal.data::join_key("b", keys = c("id", "id2")), + teal.data::join_key("a", "b", keys = "id") + ) + teal.data::datanames(data) <- c("a", "b") + + shiny::testServer( + app = srv_teal, + args = list( + id = "test", + data = data, + modules = modules(module("module_1", server = function(id, data) data)) + ), + expr = { + session$setInputs("teal_modules-active_tab" = "module_1") + session$flushReact() + testthat::expect_identical( + module_output_table(output, "module_1"), + data.frame( + "Data Name" = c("a", "b"), + Obs = c("3/3", "6/6"), + Subjects = c("3/3", "3/3"), + check.names = FALSE + ) + ) + } + ) + }) + + testthat::it("reflects filters and displays subjects by their unique id count", { + data <- teal.data::teal_data( + a = data.frame(id = seq(3), name = letters[seq(3)]), + b = data.frame(id = rep(seq(3), 2), id2 = seq(6), value = letters[seq(6)]) + ) + teal.data::join_keys(data) <- teal.data::join_keys( + teal.data::join_key("a", keys = "id"), + teal.data::join_key("b", keys = c("id", "id2")), + teal.data::join_key("a", "b", keys = "id") + ) + teal.data::datanames(data) <- c("a", "b") + + shiny::testServer( + app = srv_teal, + args = list( + id = "test", + data = data, + modules = modules(module("module_1", server = function(id, data) data)), + filter = teal_slices(teal_slice("a", "name", selected = "a")) + ), + expr = { + session$setInputs("teal_modules-active_tab" = "module_1") + session$flushReact() + testthat::expect_identical( + module_output_table(output, "module_1"), + data.frame( + "Data Name" = c("a", "b"), + Obs = c("1/3", "2/6"), + Subjects = c("1/3", "1/3"), + check.names = FALSE + ) + ) + } + ) + }) + + testthat::it("reflects added filters and displays subjects by their unique id count", { + data <- teal.data::teal_data( + a = data.frame(id = seq(3), name = letters[seq(3)]), + b = data.frame(id = rep(seq(3), 2), id2 = seq(6), value = letters[seq(6)]) + ) + teal.data::join_keys(data) <- teal.data::join_keys( + teal.data::join_key("a", keys = "id"), + teal.data::join_key("b", keys = c("id", "id2")), + teal.data::join_key("a", "b", keys = "id") + ) + teal.data::datanames(data) <- c("a", "b") + + shiny::testServer( + app = srv_teal, + args = list( + id = "test", + data = data, + modules = modules(module("module_1", server = function(id, data) data)) + ), + expr = { + session$setInputs("teal_modules-active_tab" = "module_1") + slices_global$slices_set( + teal_slices(teal_slice("a", "name", selected = "a")) + ) + session$flushReact() + testthat::expect_identical( + module_output_table(output, "module_1"), + data.frame( + "Data Name" = c("a", "b"), + Obs = c("1/3", "2/6"), + Subjects = c("1/3", "1/3"), + check.names = FALSE + ) + ) + } + ) + }) + + testthat::it("reflects transform adding new dataset", { + shiny::testServer( + app = srv_teal, + args = list( + id = "test", + data = teal.data::teal_data(iris = iris), + modules = modules( + module( + "module_1", + server = function(id, data) data, + transformers = transform_list["add_dataset"], + datanames = c("iris", "new_dataset") + ) + ) + ), + expr = { + session$setInputs("teal_modules-active_tab" = "module_1") + session$flushReact() + testthat::expect_identical( + module_output_table(output, "module_1"), + data.frame( + "Data Name" = c("iris", "new_dataset"), + Obs = c("150/150", "3"), + check.names = FALSE + ) + ) + } + ) + }) + + testthat::it("reflects transform filtering", { + testthat::it("displays parent's Subjects with count based on primary key", { + shiny::testServer( + app = srv_teal, + args = list( + id = "test", + data = teal.data::teal_data(iris = iris), + modules = modules( + module( + "module_1", + server = function(id, data) data, + transformers = transform_list["iris"] + ) + ) + ), + expr = { + session$setInputs("teal_modules-active_tab" = "module_1") + session$flushReact() + testthat::expect_identical( + module_output_table(output, "module_1"), + data.frame( + "Data Name" = c("iris"), + Obs = c("6/150"), + check.names = FALSE + ) + ) + } + ) + }) + }) + + testthat::it("displays only module$datanames", { + data <- teal.data::teal_data(iris = iris, mtcars = mtcars) + teal.data::datanames(data) <- c("iris", "mtcars") + + shiny::testServer( + app = srv_teal, + args = list( + id = "test", + data = data, + modules = modules(module("module_1", server = function(id, data) data, datanames = "iris")) + ), + expr = { + session$setInputs("teal_modules-active_tab" = "module_1") + session$flushReact() + testthat::expect_identical( + module_output_table(output, "module_1"), + data.frame( + "Data Name" = c("iris"), + Obs = c("150/150"), + check.names = FALSE + ) + ) + } + ) + }) + + testthat::it("displays parent before child when join_keys are provided", { + data <- teal.data::teal_data(mtcars1 = mtcars, mtcars2 = data.frame(am = c(0, 1), test = c("a", "b"))) + + teal.data::join_keys(data) <- teal.data::join_keys( + teal.data::join_key("mtcars2", "mtcars1", keys = c("am")) + ) + + shiny::testServer( + app = srv_teal, + args = list( + id = "test", + data = data, + modules = modules(module("module_1", server = function(id, data) data)) + ), + expr = { + session$setInputs("teal_modules-active_tab" = "module_1") + session$flushReact() + testthat::expect_identical( + module_output_table(output, "module_1")[["Data Name"]], + c("mtcars2", "mtcars1") + ) + } + ) + }) + + testthat::it("displays subset of module$datanames if not sufficient", { + data <- teal.data::teal_data(iris = iris, mtcars = mtcars) + teal.data::datanames(data) <- c("iris", "mtcars") + + shiny::testServer( + app = srv_teal, + args = list( + id = "test", + data = data, + modules = modules(module("module_1", server = function(id, data) data, datanames = c("iris", "iris2"))) + ), + expr = { + session$setInputs("teal_modules-active_tab" = "module_1") + session$flushReact() + testthat::expect_identical( + module_output_table(output, "module_1"), + data.frame( + "Data Name" = c("iris"), + Obs = c("150/150"), + check.names = FALSE + ) + ) + } + ) + }) +}) + +testthat::describe("srv_teal snapshot manager", { + testthat::it("clicking reset button restores initial filters state when !module_specific", { + shiny::testServer( + app = srv_teal, + args = list( + id = "test", + data = teal.data::teal_data(iris = iris, mtcars = mtcars), + modules = modules( + module("module_1", server = function(id, data) data), + module("module_2", server = function(id, data) data) + ), + filter = teal_slices( + teal_slice("iris", "Species"), + teal_slice("mtcars", "cyl"), + module_specific = FALSE + ) + ), + expr = { + initial_slices <- slices_global$all_slices() + session$setInputs("teal_modules-active_tab" = "module_1") + slices_global$slices_set(teal_slices()) + session$flushReact() + session$setInputs("snapshot_manager_panel-module-snapshot_reset" = TRUE) + session$flushReact() + testthat::expect_true( + is_slices_equivalent( + slices_global$all_slices(), + initial_slices + ) + ) + testthat::expect_true( + is_slices_equivalent( + slices_global$module_slices_api[["global_filters"]]$get_filter_state(), + initial_slices, + with_attrs = FALSE + ) + ) + } + ) + }) + + testthat::it("clicking reset button restores initial filters with respect to mapping state when module_specific", { + shiny::testServer( + app = srv_teal, + args = list( + id = "test", + data = teal.data::teal_data(iris = iris, mtcars = mtcars), + modules = modules( + module("module_1", server = function(id, data) data), + module("module_2", server = function(id, data) data) + ), + filter = teal_slices( + teal_slice("iris", "Species"), + teal_slice("mtcars", "cyl"), + mapping = list(module_1 = "iris Species", module_2 = "mtcars cyl"), + module_specific = TRUE + ) + ), + expr = { + initial_slices <- slices_global$all_slices() + session$setInputs("teal_modules-active_tab" = "module_1") + session$setInputs("teal_modules-active_tab" = "module_2") + slices_global$slices_set(teal_slices()) + session$flushReact() + session$setInputs("snapshot_manager_panel-module-snapshot_reset" = TRUE) + session$flushReact() + testthat::expect_true( + is_slices_equivalent( + slices_global$all_slices(), + initial_slices + ) + ) + testthat::expect_true( + is_slices_equivalent( + slices_global$module_slices_api[["module_1"]]$get_filter_state(), + initial_slices[1], + with_attrs = FALSE + ) + ) + testthat::expect_true( + is_slices_equivalent( + slices_global$module_slices_api[["module_2"]]$get_filter_state(), + initial_slices[2], + with_attrs = FALSE + ) + ) + } + ) + }) }) diff --git a/tests/testthat/test-module_teal_with_splash.R b/tests/testthat/test-module_teal_with_splash.R deleted file mode 100644 index 7c5806f386..0000000000 --- a/tests/testthat/test-module_teal_with_splash.R +++ /dev/null @@ -1,286 +0,0 @@ -testthat::test_that("srv_teal_with_splash data accepts a teal_data_module", { - testthat::expect_no_error( - shiny::testServer( - app = srv_teal_with_splash, - args = list( - id = "id", - data = teal_data_module(ui = function(id) tags$div(), server = function(id) reactive(NULL)), - modules = modules(example_module()) - ), - expr = {} - ) - ) -}) - -testthat::test_that("srv_teal_with_splash throws when teal_data_module doesn't return reactive", { - testthat::expect_error( - shiny::testServer( - app = srv_teal_with_splash, - args = list( - id = "id", - data = teal_data_module(ui = function(id) tags$div(), server = function(id) NULL), - modules = modules(example_module()) - ), - expr = {} - ), - "The `teal_data_module` passed to `data` must return a reactive expression." - ) -}) - -testthat::test_that("srv_teal_with_splash teal_data_rv evaluates the server of teal_data_module", { - shiny::testServer( - app = srv_teal_with_splash, - args = list( - id = "test", - data = teal_data_module(ui = function(id) tags$div(), server = function(id) reactive("whatever")), - modules = modules(example_module()) - ), - expr = { - testthat::expect_is(teal_data_rv, "reactive") - testthat::expect_identical(teal_data_rv(), "whatever") - } - ) -}) - -testthat::test_that("srv_teal_with_splash passes teal_data to reactive", { - shiny::testServer( - app = srv_teal_with_splash, - args = list( - id = "test", - data = teal.data::teal_data(iris = iris), - modules = modules(example_module()) - ), - expr = { - testthat::expect_is(teal_data_rv_validate, "reactive") - testthat::expect_s4_class(teal_data_rv_validate(), "teal_data") - } - ) -}) - -testthat::test_that("srv_teal_with_splash passes when datanames are empty with warning", { - testthat::expect_warning( - shiny::testServer( - app = srv_teal_with_splash, - args = list( - id = "test", - data = teal.data::teal_data(), - modules = modules(example_module()) - ), - expr = { - testthat::expect_is(teal_data_rv_validate, "reactive") - testthat::expect_s4_class(teal_data_rv_validate(), "teal_data") - } - ), - "`data` object has no datanames. Default datanames are set using `teal_data`'s environment." - ) -}) - -testthat::test_that("srv_teal_with_splash teal_data_rv_validate throws when teal_data_module returns error", { - shiny::testServer( - app = srv_teal_with_splash, - args = list( - id = "test", - data = teal_data_module( - ui = function(id) tags$div(), - server = function(id) reactive(stop("this error")) - ), - modules = modules(example_module()) - ), - expr = { - testthat::expect_is(teal_data_rv_validate, "reactive") - testthat::expect_error(teal_data_rv_validate(), "this error") - } - ) -}) - -testthat::test_that("srv_teal_with_splash teal_data_rv_validate throws then qenv.error occurs", { - shiny::testServer( - app = srv_teal_with_splash, - args = list( - id = "test", - data = teal_data_module( - ui = function(id) tags$div(), - server = function(id) reactive(within(teal.data::teal_data(), stop("not good"))) - ), - modules = modules(example_module()) - ), - expr = { - testthat::expect_is(teal_data_rv_validate, "reactive") - testthat::expect_error(teal_data_rv_validate(), "not good") - } - ) -}) - -testthat::test_that( - "srv_teal_with_splash teal_data_rv_validate throws when teal_data_module doesn't return teal_data", - { - shiny::testServer( - app = srv_teal_with_splash, - args = list( - id = "test", - data = teal_data_module( - ui = function(id) tags$div(), - server = function(id) reactive(data.frame()) - ), - modules = modules(example_module()) - ), - expr = { - testthat::expect_is(teal_data_rv_validate, "reactive") - testthat::expect_error(teal_data_rv_validate(), "failed to return `teal_data`") - } - ) - } -) - -testthat::test_that("srv_teal_with_splash teal_data_rv_validate throws when incompatible module's datanames", { - shiny::testServer( - app = srv_teal_with_splash, - args = list( - id = "test", - data = teal.data::teal_data(mtcars = mtcars, iris = iris, npk = npk), - modules = modules(example_module(datanames = "non-existing")) - ), - expr = { - testthat::expect_is(teal_data_rv_validate, "reactive") - testthat::expect_error( - teal_data_rv_validate(), - "Module 'example teal module' uses datanames not available in 'data'" - ) - } - ) -}) - -testthat::test_that("srv_teal_with_splash teal_data_rv_validate returns teal_data if incompatible filter's datanames", { - shiny::testServer( - app = srv_teal_with_splash, - args = list( - id = "test", - data = teal.data::teal_data(mtcars = mtcars), - modules = modules(example_module(datanames = "mtcars")), - filter = teal_slices(teal_slice(dataname = "iris", varname = "Species")) - ), - expr = { - testthat::expect_is(teal_data_rv_validate, "reactive") - testthat::expect_warning( - teal_data_rv_validate(), - "Filter 'iris Species' refers to dataname not available in 'data'" - ) - testthat::expect_s4_class(teal_data_rv_validate(), "teal_data") - } - ) -}) - -testthat::test_that("srv_teal_with_splash gets observe event from srv_teal", { - shiny::testServer( - app = srv_teal_with_splash, - args = list( - id = "test", - data = teal.data::teal_data(), - modules = modules(example_module()) - ), - expr = { - testthat::expect_is(res, "Observer") - } - ) -}) - -testthat::test_that("srv_teal_with_splash accepts data after within.teal_data_module", { - tdm <- teal_data_module(ui = function(id) tags$div(), server = function(id) reactive(teal_data(IRIS = iris))) - tdm2 <- within(tdm, IRIS$id <- seq_len(NROW(IRIS$Species))) # nolint: object_name. - - testthat::expect_no_error( - shiny::testServer( - app = srv_teal_with_splash, - args = list( - id = "id", - data = tdm2, - modules = modules(example_module()) - ), - expr = { - testthat::expect_s3_class(teal_data_rv, "reactive") - testthat::expect_s3_class(teal_data_rv_validate, "reactive") - testthat::expect_s4_class(teal_data_rv_validate(), "teal_data") - testthat::expect_identical( - teal_data_rv_validate()[["IRIS"]], - within(iris, id <- seq_len(NROW(Species))) - ) - } - ) - ) -}) - -testthat::test_that("srv_teal_with_splash throws error when within.teal_data_module returns qenv.error", { - tdm <- teal_data_module(ui = function(id) tags$div(), server = function(id) reactive(teal_data(IRIS = iris))) - tdm2 <- within(tdm, non_existing_var + 1) - - testthat::expect_no_error( - shiny::testServer( - app = srv_teal_with_splash, - args = list( - id = "id", - data = tdm2, - modules = modules(example_module()) - ), - expr = { - testthat::expect_s3_class(teal_data_rv, "reactive") - testthat::expect_s3_class(teal_data_rv(), "qenv.error") - testthat::expect_s3_class(teal_data_rv_validate, "reactive") - testthat::expect_error(teal_data_rv_validate(), "when evaluating qenv code") - } - ) - ) -}) - -testthat::test_that("srv_teal_with_splash throws error when within.teal_data_module returns NULL", { - tdm <- teal_data_module(ui = function(id) tags$div(), server = function(id) reactive(NULL)) - tdm2 <- within(tdm, within(1 + 1)) - testthat::expect_no_error( - shiny::testServer( - app = srv_teal_with_splash, - args = list( - id = "id", - data = tdm2, - modules = modules(example_module()) - ), - expr = { - testthat::expect_s3_class(teal_data_rv, "reactive") - testthat::expect_null(teal_data_rv()) - testthat::expect_s3_class(teal_data_rv_validate, "reactive") - testthat::expect_error( - teal_data_rv_validate(), - "`teal_data_module` passed to `data` failed to return `teal_data` object" - ) - } - ) - ) -}) - -testthat::test_that( - paste( - "srv_teal_with_splash throws error when within.teal_data_module returns arbitrary object", - "(other than `teal_data` or `qenv.error`)" - ), - { - tdm <- teal_data_module(ui = function(id) tags$div(), server = function(id) reactive(NULL)) - tdm2 <- within(tdm, 1 + 1) - testthat::expect_no_error( - shiny::testServer( - app = srv_teal_with_splash, - args = list( - id = "id", - data = tdm2, - modules = modules(example_module()) - ), - expr = { - testthat::expect_s3_class(teal_data_rv, "reactive") - testthat::expect_null(teal_data_rv()) - testthat::expect_s3_class(teal_data_rv_validate, "reactive") - testthat::expect_error( - teal_data_rv_validate(), - "`teal_data_module` passed to `data` failed to return `teal_data` object" - ) - } - ) - ) - } -) diff --git a/tests/testthat/test-modules.R b/tests/testthat/test-modules.R index 9ee6359174..a4a6a8651f 100644 --- a/tests/testthat/test-modules.R +++ b/tests/testthat/test-modules.R @@ -121,7 +121,7 @@ testthat::test_that("module() returns list of class 'teal_module' containing inp ui_args = NULL ) testthat::expect_s3_class(test_module, "teal_module") - testthat::expect_named(test_module, c("label", "server", "ui", "datanames", "server_args", "ui_args")) + testthat::expect_named(test_module, c("label", "server", "ui", "datanames", "server_args", "ui_args", "transformers")) testthat::expect_identical(test_module$label, "aaa1") testthat::expect_identical(test_module$server, call_module_server_fun) testthat::expect_identical(test_module$ui, ui_fun1) diff --git a/tests/testthat/test-report_previewer_module.R b/tests/testthat/test-report_previewer_module.R index 5636bed5cb..dc7e014459 100644 --- a/tests/testthat/test-report_previewer_module.R +++ b/tests/testthat/test-report_previewer_module.R @@ -1,3 +1,10 @@ +testthat::test_that("report_previewer_module has specific classes", { + testthat::expect_s3_class( + reporter_previewer_module(), + c("teal_module_previewer", "teal_module") + ) +}) + testthat::test_that("report_previewer_module throws error if label is not string", { testthat::expect_error( reporter_previewer_module(label = 5), "Assertion on 'label' failed: Must be of type 'string'" @@ -16,3 +23,18 @@ testthat::test_that("report_previewer_module default label is Report previewer " r_p_m <- reporter_previewer_module() testthat::expect_equal(r_p_m$label, "Report previewer") }) + +testthat::test_that( + "report_previewer_module does not accept server_args out of formals(teal.reporter::reporter_previewer_srv) ", + { + error_pattern <- ".*Assertion on \\'all\\(names\\(server_args" + testthat::expect_error( + reporter_previewer_module(server_args = list(x = "A")), + error_pattern + ) + testthat::expect_error( + reporter_previewer_module(server_args = list(reporter = "A", global_knitr = 5, d = 1)), + error_pattern + ) + } +) diff --git a/tests/testthat/test-shinytest2-data_summary.R b/tests/testthat/test-shinytest2-data_summary.R new file mode 100644 index 0000000000..c97c861110 --- /dev/null +++ b/tests/testthat/test-shinytest2-data_summary.R @@ -0,0 +1,154 @@ +testthat::test_that("e2e: data summary list only data names if there is no MAE or data.frames in teal_data", { + skip_if_too_deep(5) + app <- TealAppDriver$new( + data = teal.data::teal_data(x = 1), + modules = example_module() + ) + + testthat::expect_identical( + as.data.frame(app$get_active_data_summary_table()), + data.frame( + `Data Name` = c("x"), + check.names = FALSE + ) + ) + + app$stop() +}) + + +testthat::test_that("e2e: data summary is displayed with 2 columns data without keys", { + skip_if_too_deep(5) + app <- TealAppDriver$new( + data = simple_teal_data(), # iris, mtcars + modules = example_module() + ) + + testthat::expect_identical( + as.data.frame(app$get_active_data_summary_table()), + data.frame( + `Data Name` = c("iris", "mtcars"), + Obs = c("150/150", "32/32"), + check.names = FALSE + ) + ) + + app$stop() +}) + +testthat::test_that("e2e: data summary displays datasets by topological_sort of join_keys", { + skip_if_too_deep(5) + + data <- teal.data::teal_data(mtcars1 = mtcars, mtcars2 = data.frame(am = c(0, 1), test = c("a", "b"))) + + teal.data::join_keys(data) <- teal.data::join_keys( + teal.data::join_key("mtcars2", "mtcars1", keys = c("am")) + ) + + app <- TealAppDriver$new( + data = data, + modules = example_module() + ) + + testthat::expect_identical( + as.data.frame(app$get_active_data_summary_table())[["Data Name"]], + c("mtcars2", "mtcars1") + ) + + app$stop() +}) + +testthat::test_that("e2e: data summary is displayed with 3 columns for data with join keys", { + skip_if_too_deep(5) + + data <- teal.data::teal_data(mtcars1 = mtcars, mtcars2 = data.frame(am = c(0, 1), test = c("a", "b"))) + + teal.data::join_keys(data) <- teal.data::join_keys( + teal.data::join_key("mtcars2", "mtcars1", keys = c("am")) + ) + + app <- TealAppDriver$new( + data = data, + modules = example_module() + ) + + testthat::expect_identical( + as.data.frame(app$get_active_data_summary_table()), + data.frame( + `Data Name` = c("mtcars2", "mtcars1"), + Obs = c("2/2", "32/32"), + Subjects = c("", "2/2"), + check.names = FALSE + ) + ) + + app$stop() +}) + +testthat::test_that( + "e2e: data summary is displayed properly if teal_data include data.frames with join keys, MAE objects and vectors", + { + testthat::skip_if_not_installed("MultiAssayExperiment") + skip_if_too_deep(5) + + data <- within( + teal.data::teal_data(), + { + mtcars1 <- mtcars + mtcars2 <- data.frame(am = c(0, 1), test = c("a", "b")) + iris <- iris + library(MultiAssayExperiment) + data("miniACC", package = "MultiAssayExperiment", envir = environment()) + # nolint start: object_name. + CO2 <- CO2 + factors <- names(Filter(isTRUE, vapply(CO2, is.factor, logical(1L)))) + CO2[factors] <- lapply(CO2[factors], as.character) + # nolint end: object_name. + } + ) + + datanames(data) <- c("CO2", "iris", "miniACC", "mtcars2", "mtcars1", "factors") + + teal.data::join_keys(data) <- teal.data::join_keys( + teal.data::join_key("mtcars2", "mtcars1", keys = c("am")) + ) + + app <- TealAppDriver$new( + data = data, + modules = example_module() + ) + + testthat::expect_identical( + as.data.frame(app$get_active_data_summary_table()), + data.frame( + `Data Name` = c( + "CO2", "iris", "miniACC", "- RNASeq2GeneNorm", "- gistict", "- RPPAArray", "- Mutations", "- miRNASeqGene", + "mtcars2", "mtcars1", "factors" + ), + Obs = c("84/84", "150/150", "", "198/198", "198/198", "33/33", "97/97", "471/471", "2/2", "32/32", ""), + Subjects = c("", "", "92/92", "79/79", "90/90", "46/46", "90/90", "80/80", "", "2/2", ""), + check.names = FALSE + ) + ) + + app$stop() + } +) + +testthat::test_that("e2e: data summary displays datasets by datanames() order if no join_keys", { + skip_if_too_deep(5) + + data <- teal.data::teal_data(mtcars1 = mtcars, mtcars2 = data.frame(am = c(0, 1), test = c("a", "b"))) + + app <- TealAppDriver$new( + data = data, + modules = example_module() + ) + + testthat::expect_identical( + as.data.frame(app$get_active_data_summary_table())[["Data Name"]], + c("mtcars1", "mtcars2") + ) + + app$stop() +}) diff --git a/tests/testthat/test-shinytest2-filter_panel.R b/tests/testthat/test-shinytest2-filter_panel.R index 11e89094bd..aa65eb567d 100644 --- a/tests/testthat/test-shinytest2-filter_panel.R +++ b/tests/testthat/test-shinytest2-filter_panel.R @@ -28,7 +28,7 @@ testthat::test_that("e2e: module content is updated when a data is filtered in f app$stop() }) -testthat::test_that("e2e: filtering a module-specific filter is refected in other shared module", { +testthat::test_that("e2e: filtering a module-specific filter is reflected in other shared module", { skip_if_too_deep(5) app <- TealAppDriver$new( data = simple_teal_data(), diff --git a/tests/testthat/test-shinytest2-init.R b/tests/testthat/test-shinytest2-init.R index 4bc1dfb459..e0672e7e53 100644 --- a/tests/testthat/test-shinytest2-init.R +++ b/tests/testthat/test-shinytest2-init.R @@ -88,9 +88,10 @@ testthat::test_that("e2e: init creates UI containing specified title, favicon, h app_title ) testthat::expect_equal( - app$get_html_rvest("head > link[rel='icon']") %>% - rvest::html_elements("link") %>% - rvest::html_attr("href"), + rvest::html_attr( + rvest::html_elements(app$get_html_rvest("head > link[rel='icon']"), "link"), + "href" + ), app_favicon ) testthat::expect_match( diff --git a/tests/testthat/test-shinytest2-landing_popup.R b/tests/testthat/test-shinytest2-landing_popup.R index 70d1278333..d2e8ba3093 100644 --- a/tests/testthat/test-shinytest2-landing_popup.R +++ b/tests/testthat/test-shinytest2-landing_popup.R @@ -6,11 +6,11 @@ testthat::test_that("e2e: teal app with landing_popup_module initializes with no app <- TealAppDriver$new( data = simple_teal_data(), modules = modules( - landing_popup_module( - title = "Welcome", - content = tags$b("A welcome message!", style = "color: red;") - ), example_module() + ), + landing_popup = landing_popup_module( + title = "Welcome", + content = tags$b("A welcome message!", style = "color: red;") ) ) @@ -26,9 +26,9 @@ testthat::test_that("e2e: app with default landing_popup_module creates modal co app <- TealAppDriver$new( data = simple_teal_data(), modules = modules( - landing_popup_module(), example_module() - ) + ), + landing_popup = landing_popup_module() ) testthat::expect_equal( @@ -44,9 +44,9 @@ testthat::test_that("e2e: when default landing_popup_module is closed, it shows app <- TealAppDriver$new( data = simple_teal_data(), modules = modules( - landing_popup_module(), example_module() - ) + ), + landing_popup = landing_popup_module() ) # Button is clicked. @@ -94,12 +94,12 @@ testthat::test_that( app <- TealAppDriver$new( data = simple_teal_data(), modules = modules( - landing_popup_module( - title = modal_title, - content = modal_content, - buttons = modal_buttons - ), example_module() + ), + landing_popup = landing_popup_module( + title = modal_title, + content = modal_content, + buttons = modal_buttons ) ) @@ -148,10 +148,10 @@ testthat::test_that("e2e: when customized button in landing_popup_module is clic app <- TealAppDriver$new( data = simple_teal_data(), modules = modules( - landing_popup_module( - buttons = actionButton("read", "Read more", onclick = onclick_text) - ), example_module() + ), + landing_popup = landing_popup_module( + buttons = actionButton("read", "Read more", onclick = onclick_text) ) ) diff --git a/tests/testthat/test-shinytest2-module_bookmark_manager.R b/tests/testthat/test-shinytest2-module_bookmark_manager.R index 8eb27f39c2..7c58bd3a72 100644 --- a/tests/testthat/test-shinytest2-module_bookmark_manager.R +++ b/tests/testthat/test-shinytest2-module_bookmark_manager.R @@ -6,7 +6,7 @@ testthat::test_that("bookmark_manager_button is not rendered by default", { app <- TealAppDriver$new( data = simple_teal_data(), modules = example_module(label = "Example Module"), - options = options() + options = list() ) on.exit(app$stop()) testthat::expect_null( @@ -17,11 +17,10 @@ testthat::test_that("bookmark_manager_button is not rendered by default", { testthat::test_that("bookmark_manager_button is not rendered when enableBookmarking = 'url'", { skip_if_too_deep(5) - options(shiny.bookmarkStore = "url") app <- TealAppDriver$new( data = simple_teal_data(), modules = example_module(label = "Example Module"), - options = options() + options = list(shiny.bookmarkStore = "url") ) on.exit(app$stop()) testthat::expect_null( @@ -32,11 +31,10 @@ testthat::test_that("bookmark_manager_button is not rendered when enableBookmark testthat::test_that("bookmark_manager_button is rendered when enableBookmarking = 'server'", { skip_if_too_deep(5) - options(shiny.bookmarkStore = "server") app <- TealAppDriver$new( data = simple_teal_data(), modules = example_module(label = "Example Module"), - options = options() + options = list(shiny.bookmarkStore = "server") ) on.exit(app$stop()) testthat::expect_true(!is.null(app$get_html(".bookmark_manager_button"))) @@ -44,11 +42,10 @@ testthat::test_that("bookmark_manager_button is rendered when enableBookmarking testthat::test_that("bookmark_manager_button shows modal with url containing state_id when clicked", { skip_if_too_deep(5) - options(shiny.bookmarkStore = "server") app <- TealAppDriver$new( data = simple_teal_data(), modules = example_module(label = "Example Module"), - options = options() + options = list(shiny.bookmarkStore = "server") ) bookmark_button_id <- app$get_attr(".bookmark_manager_button", "id") app$click(bookmark_button_id) diff --git a/tests/testthat/test-shinytest2-modules.R b/tests/testthat/test-shinytest2-modules.R index 91479e90bf..7b2584aa9d 100644 --- a/tests/testthat/test-shinytest2-modules.R +++ b/tests/testthat/test-shinytest2-modules.R @@ -72,24 +72,6 @@ testthat::test_that("e2e: filter panel shows all the datasets when datanames is app$stop() }) -testthat::test_that("e2e: filter panel is not displayed when datanames is NULL", { - skip_if_too_deep(5) - app <- TealAppDriver$new( - data = simple_teal_data(), - modules = modules( - example_module(label = "NULL", datanames = NULL) - ) - ) - - testthat::expect_identical( - app$get_html_rvest(".teal_secondary_col") %>% - rvest::html_element("div") %>% - rvest::html_attr("style"), - "display: none;" - ) - - app$stop() -}) testthat::test_that("e2e: all the nested teal modules are initiated as expected", { skip_if_too_deep(5) diff --git a/tests/testthat/test-shinytest2-reporter.R b/tests/testthat/test-shinytest2-reporter.R index 321637c6b6..9d4d8372b8 100644 --- a/tests/testthat/test-shinytest2-reporter.R +++ b/tests/testthat/test-shinytest2-reporter.R @@ -8,8 +8,7 @@ testthat::test_that("e2e: reporter tab is created when a module has reporter", { modules = report_module(label = "Module with Reporter") ) - teal_tabs <- app$get_html_rvest(selector = "#teal-main_ui-root-active_tab") %>% - rvest::html_elements("a") + teal_tabs <- rvest::html_elements(app$get_html_rvest(selector = "#teal-teal_modules-active_tab"), "a") tab_names <- setNames( rvest::html_attr(teal_tabs, "data-value"), rvest::html_text(teal_tabs) @@ -28,8 +27,10 @@ testthat::test_that("e2e: reporter tab is not created when a module has no repor data = simple_teal_data(), modules = example_module(label = "Example Module") ) - teal_tabs <- app$get_html_rvest(selector = "#teal-main_ui-root-active_tab") %>% - rvest::html_elements("a") + teal_tabs <- rvest::html_elements( + app$get_html_rvest(selector = "#teal-teal_modules-active_tab"), + "a" + ) tab_names <- setNames( rvest::html_attr(teal_tabs, "data-value"), rvest::html_text(teal_tabs) @@ -52,10 +53,6 @@ testthat::test_that("e2e: adding a report card in a module adds it in the report app$click(NS(app$active_module_ns(), "reporter-add_report_card_simple-add_report_card_button")) - app$set_input( - NS(app$active_module_ns(), "reporter-add_report_card_simple-label"), - "Card name" - ) app$set_input( NS(app$active_module_ns(), "reporter-add_report_card_simple-label"), "Card name" @@ -85,3 +82,58 @@ testthat::test_that("e2e: adding a report card in a module adds it in the report app$stop() }) + +testthat::test_that("e2e: reporter_previewer_module do not show data_summary nor filter_panel", { + skip_if_too_deep(5) + app <- teal:::TealAppDriver$new( + data = simple_teal_data(), + modules = report_module(label = "Module with Reporter") + ) + + app$navigate_teal_tab("Report previewer") + + testthat::expect_null(app$is_visible(app$active_data_summary_element("table"))) + + testthat::expect_null(app$get_active_filter_vars()) + + app$stop() +}) + +testthat::test_that("e2e: reporter does not show the secondary column that shows filter_panel", { + skip_if_too_deep(5) + app <- teal:::TealAppDriver$new( + data = simple_teal_data(), + modules = report_module(label = "Module with Reporter") + ) + + app$click(NS(app$active_module_ns(), "reporter-add_report_card_simple-add_report_card_button")) + + app$set_input( + NS(app$active_module_ns(), "reporter-add_report_card_simple-label"), + "Card name" + ) + app$set_input( + NS(app$active_module_ns(), "reporter-add_report_card_simple-comment"), + "Card comment" + ) + + app$click(NS(app$active_module_ns(), "reporter-add_report_card_simple-add_card_ok")) + + secondary_col <- paste0( + "#", + gsub("-module$", "", app$active_module_ns()), + " > div > div.col-sm-3.teal_secondary_col" + ) + + testthat::expect_true(app$is_visible(secondary_col)) + app$navigate_teal_tab("Report previewer") + + secondary_col <- paste0( + "#", + gsub("-module$", "", app$active_module_ns()), + " > div > div.col-sm-3.teal_secondary_col" + ) + testthat::expect_null(app$is_visible(secondary_col)) + + app$stop() +}) diff --git a/tests/testthat/test-shinytest2-show-rcode.R b/tests/testthat/test-shinytest2-show-rcode.R index 0e6f23b325..0e6be7c128 100644 --- a/tests/testthat/test-shinytest2-show-rcode.R +++ b/tests/testthat/test-shinytest2-show-rcode.R @@ -33,23 +33,19 @@ testthat::test_that("e2e: teal app initializes with Show R Code modal", { ) # Check for Copy buttons. testthat::expect_equal( - app$active_module_element("rcode-copy_button1") %>% - app$get_text(), + app$get_text(app$active_module_element("rcode-copy_button1")), "Copy to Clipboard" ) testthat::expect_equal( - app$active_module_element("rcode-copy_button2") %>% - app$get_text(), + app$get_text(app$active_module_element("rcode-copy_button2")), "Copy to Clipboard" ) # Check R code output. - r_code <- - app$active_module_element("rcode-verbatim_content") %>% - app$get_text() + r_code <- app$get_text(app$active_module_element("rcode-verbatim_content")) - testthat::expect_match(r_code, "# Add any code to install/load your NEST environment here", fixed = TRUE) - testthat::expect_match(r_code, "library(teal.code)", fixed = TRUE) + testthat::expect_match(r_code, "iris <- iris", fixed = TRUE) + testthat::expect_match(r_code, "iris._raw_ <- iris", fixed = TRUE) testthat::expect_match(r_code, "stopifnot(rlang::hash(", fixed = TRUE) app$stop() diff --git a/tests/testthat/test-shinytest2-teal_data_module.R b/tests/testthat/test-shinytest2-teal_data_module.R index b1a2272957..9cba14c73c 100644 --- a/tests/testthat/test-shinytest2-teal_data_module.R +++ b/tests/testthat/test-shinytest2-teal_data_module.R @@ -31,7 +31,8 @@ testthat::test_that("e2e: teal_data_module will have a delayed load of datasets" modules = example_module(label = "Example Module") ) - app$click("teal_data_module-submit") + app$click("teal-data-teal_data_module-data-submit") + app$navigate_teal_tab("Example Module") testthat::expect_setequal(app$get_active_filter_vars(), c("dataset1", "dataset2")) app$stop() @@ -66,7 +67,7 @@ testthat::test_that("e2e: teal_data_module shows validation errors", { modules = example_module(label = "Example Module") ) - app$click("teal_data_module-submit") + app$click("teal-data-teal_data_module-data-submit") app$expect_validation_error() @@ -110,8 +111,9 @@ testthat::test_that("e2e: teal_data_module inputs change teal_data object that i modules = example_module(label = "Example Module") ) - app$set_input("teal_data_module-new_column", "A_New_Column") - app$click("teal_data_module-submit") + app$set_input("teal-data-teal_data_module-data-new_column", "A_New_Column") + app$click("teal-data-teal_data_module-data-submit") + app$navigate_teal_tab("Example Module") # This may fail if teal_data_module does not perform the transformation testthat::expect_no_error(app$add_filter_var("dataset1", "A_New_Column")) @@ -123,3 +125,152 @@ testthat::test_that("e2e: teal_data_module inputs change teal_data object that i app$stop() }) + +testthat::test_that("e2e: teal_data_module gets removed after successful data load, when once = TRUE", { + skip_if_too_deep(5) + tdm <- teal_data_module( + ui = function(id) { + ns <- shiny::NS(id) + shiny::actionButton(ns("submit"), label = "Load data") + }, + server = function(id) { + shiny::moduleServer(id, function(input, output, session) { + shiny::eventReactive(input$submit, { + data <- within( + teal_data(), + { + dataset1 <- iris + dataset2 <- mtcars + } + ) + datanames(data) <- c("dataset1", "dataset2") + + data + }) + }) + }, + once = TRUE + ) + + app <- TealAppDriver$new( + data = tdm, + modules = example_module(label = "Example Module") + ) + + submit <- "teal-data-teal_data_module-data-submit" + app$click(submit) + + testthat::expect_false( + app$is_visible('#teal-teal_modules-active_tab a[data-value="teal_data_module"]') + ) + + testthat::expect_false( + app$is_visible(sprintf("#%s", submit)) + ) + + app$stop() +}) + +testthat::test_that("e2e: teal_data_module is still visible after successful data load, when once = FALSE", { + skip_if_too_deep(5) + tdm <- teal_data_module( + ui = function(id) { + ns <- shiny::NS(id) + shiny::actionButton(ns("submit"), label = "Load data") + }, + server = function(id) { + shiny::moduleServer(id, function(input, output, session) { + shiny::eventReactive(input$submit, { + data <- within( + teal_data(), + { + dataset1 <- iris + dataset2 <- mtcars + } + ) + datanames(data) <- c("dataset1", "dataset2") + + data + }) + }) + }, + once = FALSE + ) + + app <- TealAppDriver$new( + data = tdm, + modules = example_module(label = "Example Module") + ) + + app$click("teal-data-teal_data_module-data-submit") + + testthat::expect_true( + app$is_visible('#teal-teal_modules-active_tab a[data-value="teal_data_module"]') + ) + + app$stop() +}) + +testthat::test_that("e2e: teal_data_module will make other tabs inactive before successful data load", { + skip_if_too_deep(5) + tdm <- teal_data_module( + ui = function(id) { + ns <- shiny::NS(id) + shiny::actionButton(ns("submit"), label = "Load data") + }, + server = function(id) { + shiny::moduleServer(id, function(input, output, session) { + shiny::eventReactive(input$submit, { + data <- within( + teal_data(), + { + dataset1 <- iris + dataset2 <- mtcars + } + ) + datanames(data) <- c("dataset1", "dataset2") + + data + }) + }) + }, + once = FALSE + ) + + app <- TealAppDriver$new( + data = tdm, + modules = modules( + example_module(label = "Example Module 1"), + example_module(label = "Example Module 2") + ) + ) + + testthat::expect_equal( + rvest::html_attr( + rvest::html_nodes( + app$get_html_rvest("#teal-teal_modules-active_tab"), + "a[data-value*='example_module']" + ), + "disabled" + ), + c("disabled", "disabled") + ) + + app$click("teal-data-teal_data_module-data-submit") + + testthat::expect_true( + is.na( + unique( + rvest::html_attr( + rvest::html_nodes( + app$get_html_rvest("#teal-teal_modules-active_tab"), + "a[data-value*='example_module']" + ), + "disabled" + ) + ) + ) + ) + + app$stop() +}) diff --git a/tests/testthat/test-shinytest2-utils.R b/tests/testthat/test-shinytest2-utils.R index 9e1d8cff4c..6e783ac9e2 100644 --- a/tests/testthat/test-shinytest2-utils.R +++ b/tests/testthat/test-shinytest2-utils.R @@ -9,8 +9,7 @@ testthat::test_that("e2e: show/hide hamburger works as expected", { ) get_class_attributes <- function(app, selector) { - element <- app$get_html_rvest(selector = selector) %>% - rvest::html_elements(selector) + element <- rvest::html_elements(app$get_html_rvest(selector = selector), selector) list( class = rvest::html_attr(element, "class"), style = rvest::html_attr(element, "style") @@ -21,7 +20,7 @@ testthat::test_that("e2e: show/hide hamburger works as expected", { secondary_attrs <- get_class_attributes(app, ".teal_secondary_col") testthat::expect_true(grepl("col-sm-9", primary_attrs$class)) - testthat::expect_true(grepl("display: block;", secondary_attrs$style)) + testthat::expect_true(is.na(secondary_attrs$style)) app$click(selector = ".btn.action-button.filter_hamburger") primary_attrs <- get_class_attributes(app, ".teal_primary_col") diff --git a/tests/testthat/test-snapshot_manager.R b/tests/testthat/test-snapshot_manager.R deleted file mode 100644 index 4954fe4b41..0000000000 --- a/tests/testthat/test-snapshot_manager.R +++ /dev/null @@ -1,46 +0,0 @@ -testthat::test_that("snapshot manager holds initial state in history", { - filter <- teal_slices( - teal.slice::teal_slice(dataname = "iris", varname = "Sepal.Length"), - teal.slice::teal_slice(dataname = "iris", varname = "Species"), - teal.slice::teal_slice(dataname = "mtcars", varname = "mpg"), - teal.slice::teal_slice(dataname = "women", varname = "height"), - mapping = list( - m1 = c("iris Sepal.Length"), - m3 = c("women height"), - global_filters = "iris Species" - ) - ) - - fd1 <- teal.slice::init_filtered_data(list(iris = list(dataset = iris))) - fd2 <- teal.slice::init_filtered_data(list(iris = list(dataset = iris), mtcars = list(dataset = mtcars))) - fd3 <- teal.slice::init_filtered_data(list(iris = list(dataset = iris), women = list(dataset = women))) - datasets_flat <- list(m1 = fd1, m2 = fd2, m3 = fd3) - - slices_global <- reactiveVal(shiny::isolate(filter)) - - mapping_matrix <- reactive({ - module_states <- lapply(datasets_flat, function(x) x$get_filter_state()) - mapping_ragged <- lapply(module_states, function(x) vapply(x, `[[`, character(1L), "id")) - all_names <- vapply(slices_global(), `[[`, character(1L), "id") - mapping_smooth <- lapply(mapping_ragged, is.element, el = all_names) - as.data.frame(mapping_smooth, row.names = all_names, check.names = FALSE) - }) - - shiny::testServer( - app = snapshot_manager_srv, - args = list( - id = "test", - slices_global = slices_global, - mapping_matrix = mapping_matrix, - datasets = datasets_flat - ), - expr = { - testthat::expect_true("Initial application state" %in% names(snapshot_history())) - - snapshot <- snapshot_history()[["Initial application state"]] - snapshot_state <- as.teal_slices(snapshot) - - testthat::expect_equal(as.list(snapshot_state, recursive = TRUE), as.list(filter, recursive = TRUE)) - } - ) -}) diff --git a/tests/testthat/test-tdata.R b/tests/testthat/test-tdata.R deleted file mode 100644 index faef6351d3..0000000000 --- a/tests/testthat/test-tdata.R +++ /dev/null @@ -1,261 +0,0 @@ -withr::local_options(lifecycle_verbosity = "quiet") - -# ---- constructor ---- -testthat::test_that("new_tdata accepts reactive and not reactive MAE and data.frames", { - testthat::skip_if_not_installed("MultiAssayExperiment") - utils::data(miniACC, package = "MultiAssayExperiment") - - testthat::expect_no_error( - new_tdata( - list( - a = reactive(data.frame(x = 1:10)), - b = data.frame(y = 1:10), - c = reactive(miniACC), - d = miniACC - ) - ) - ) -}) - -testthat::test_that("new_tdata throws error if data is not a list with unique names", { - testthat::expect_error( - new_tdata(data.frame(1:10)), "Must be of type 'list'" - ) - - testthat::expect_error( - new_tdata(list(data.frame(1:10))), "Must have names" - ) - - testthat::expect_error( - new_tdata(list(x = data.frame(1:10), x = data.frame(1:5))), "Must have unique names" - ) -}) - -testthat::test_that("new_tdata throws error if contents of data list are not of correct type", { - testthat::expect_error( - new_tdata(list(x = 1)), "May only contain the following types: \\{data.frame,reactive,MultiAssayExperiment\\}" - ) -}) - -testthat::test_that("new_tdata throws error if code is not character or reactive character", { - testthat::expect_error( - new_tdata(list(x = iris), code = 5), - "Assertion on 'code' failed: Must inherit from class 'character'/'reactive'" - ) - - testthat::expect_error( - new_tdata(list(x = iris), code = reactive(5)), - "Assertion on 'code' failed: Must inherit from class 'character'" - ) -}) - -testthat::test_that("new_tdata accepts character and reactive characters for code argument", { - testthat::expect_no_error( - new_tdata(list(x = iris, y = mtcars), code = c("x <- iris", "y <- mtcars")) - ) - - testthat::expect_no_error( - new_tdata(list(x = iris, y = mtcars), code = reactive(c("x <- iris", "y <- mtcars"))) - ) -}) - -testthat::test_that("new_tdata throws error if join_keys is not of class join_keys", { - testthat::expect_error( - new_tdata(list(x = iris), join_keys = "x"), - "Assertion on 'join_keys' failed: Must inherit from class 'join_keys'" - ) -}) - -testthat::test_that("new_tdata throws no error if join_keys is of class join_keys", { - testthat::expect_no_error( - new_tdata(list(x = iris), join_keys = teal.data::join_keys()) - ) -}) - -# note not testing the contents of metadata elements are good as we are relying on -# the (tested) function in teal.data to do this -testthat::test_that( - "new_tdata throws error if metadata is not a list with unique names a subset of the names of data", - { - testthat::expect_error( - new_tdata(list(x = iris, y = mtcars), metadata = 1:3), - "Assertion on 'metadata' failed: Must be of type 'list' \\(or 'NULL'\\)" - ) - - testthat::expect_error( - new_tdata(list(x = iris, y = mtcars), metadata = list(1, 2, 3)), - "Assertion on 'metadata' failed: Must have names." - ) - - testthat::expect_error( - new_tdata(list(x = iris, y = mtcars), metadata = list(x = list(A = 1), z = list(B = 1))), - "Must be a subset of \\{'x','y'\\}, but has additional elements \\{'z'\\}." - ) - } -) - -testthat::test_that("new_tdata does not throw error with valid metadata", { - testthat::expect_no_error( - new_tdata(list(x = iris, y = mtcars), metadata = list(x = list(A = 1), y = list(B = 1))) - ) -}) - -# ---- get_metadata ---- -testthat::test_that("get_metadata returns NULL if no metadata", { - my_tdata <- new_tdata(data = list(iris = iris, mtcars = mtcars)) - testthat::expect_null(get_metadata(my_tdata, "iris")) -}) - -testthat::test_that("get_metadata returns NULL if no metadata for given dataset", { - my_tdata <- new_tdata( - data = list(iris = iris, mtcars = mtcars), - metadata = list(mtcars = list(A = 1)) - ) - testthat::expect_null(get_metadata(my_tdata, "iris")) -}) - -testthat::test_that("get_metadata returns metadata for given dataset", { - my_tdata <- new_tdata( - data = list(iris = iris, mtcars = mtcars), - metadata = list(mtcars = list(A = 1, B = 2)) - ) - testthat::expect_equal(get_metadata(my_tdata, "mtcars"), list(A = 1, B = 2)) -}) - -testthat::test_that("get_metadata returns NULL if dataset doesn't exist", { - my_tdata <- new_tdata( - data = list(iris = iris, mtcars = mtcars), - metadata = list(mtcars = list(A = 1, B = 2)) - ) - testthat::expect_null(get_metadata(my_tdata, "not_existing_df")) -}) - -# ---- get_code ---- -testthat::test_that("get_code returns empty character if tdata object has no code", { - my_tdata <- new_tdata(data = list(iris = iris, mtcars = mtcars)) - testthat::expect_equal("", isolate(get_code_tdata(my_tdata))) -}) - -testthat::test_that("get_code returns character of code if tdata object has code", { - code_string <- c("iris <- head(iris)", "mtcars <- head(mtcars)") - - # reactive case (for constructor) - my_tdata <- new_tdata( - data = list(x = iris, mtcars = head(mtcars)), - code = reactive(code_string) - ) - testthat::expect_equal(isolate(get_code_tdata(my_tdata)), code_string) - - # not reactive case (for constructor) - my_tdata <- new_tdata( - data = list(x = iris, mtcars = head(mtcars)), - code = code_string - ) - testthat::expect_equal(isolate(get_code_tdata(my_tdata)), code_string) -}) - -# ---- get_code wrapper ---- - -testthat::test_that("get_code_tdata accepts tdata", { - data <- new_tdata(data = list(iris = iris), code = "iris <- iris") - testthat::expect_no_error(isolate(get_code_tdata(data))) -}) - -testthat::test_that("get_code_tdata throws error when input is not tdata", { - testthat::expect_error( - isolate(get_code_tdata(iris)), - "Assertion on 'data' failed: Must inherit from class 'tdata', but has class 'data.frame'." - ) - - testthat::expect_error( - isolate(get_code_tdata("iris")), - "Assertion on 'data' failed: Must inherit from class 'tdata', but has class 'character'." - ) -}) - -testthat::test_that("get_code_tdata returns character code", { - data <- new_tdata(data = list(iris = iris), code = "iris <- iris") - testthat::expect_identical(isolate(get_code_tdata(data)), "iris <- iris") -}) - -# ---- tdata2env ---- -testthat::test_that("tdata2env returns environment containing tdata contents ", { - testthat::skip_if_not_installed("MultiAssayExperiment") - utils::data(miniACC, package = "MultiAssayExperiment") - my_tdata <- new_tdata(data = list(iris = iris, mae = reactive(miniACC))) - - my_env <- isolate(tdata2env(my_tdata)) - my_env_as_list <- as.list(my_env) - testthat::expect_setequal(names(my_env_as_list), c("iris", "mae")) - testthat::expect_equal(iris, my_env_as_list$iris) - testthat::expect_equal(miniACC, my_env_as_list$mae) -}) - -testthat::test_that("tdata2env throws error if argument is not tdata", { - testthat::expect_error(tdata2env(iris), "Must inherit from class 'tdata'") -}) - -# ---- join_keys ---- -testthat::test_that("join_keys returns NULL if no join_keys object exists inside tdata", { - my_tdata <- new_tdata(data = list(iris = iris, mae = reactive(miniACC))) - testthat::expect_null(join_keys(my_tdata)) -}) - -testthat::test_that("join_keys returns join_keys object if it exists inside tdata", { - jk <- teal.data::join_keys(teal.data::join_key("A", "B", c("id" = "fk"))) - - my_tdata <- new_tdata( - data = list( - A = data.frame(id = 1:10, val = 1:10), - B = data.frame(id = 1:10, val = 1:10, fk = 10:1) - ), - join_keys = jk - ) - - testthat::expect_equal(join_keys(my_tdata), jk) -}) - - -# as_tdata ---- -code <- c("iris <- iris", "mtcars <- mtcars") -data_tdata <- teal::new_tdata(list(iris = iris, mtcars = mtcars), code) -data_teal_data <- teal.data::teal_data(iris = iris, mtcars = mtcars, code = code) -data_reactive <- shiny::reactive(teal.data::teal_data(iris = iris, mtcars = mtcars, code = code)) - -testthat::test_that("as_tdata accepts all possible inputs", { - testthat::expect_no_error(as_tdata(data_tdata)) - testthat::expect_no_error(as_tdata(data_teal_data)) - testthat::expect_no_error(as_tdata(data_reactive)) -}) - -testthat::test_that("as_tdata always returns tdata object", { - data_tdata_downgraded <- as_tdata(data_tdata) - data_teal_data_downgraded <- as_tdata(data_teal_data) - data_reactive_downgraded <- as_tdata(data_teal_data) - - testthat::expect_s3_class(data_tdata_downgraded, "tdata") - testthat::expect_s3_class(data_teal_data_downgraded, "tdata") - testthat::expect_s3_class(data_reactive_downgraded, "tdata") -}) - -testthat::test_that("datasets are maintained during conversion", { - data_tdata_downgraded <- as_tdata(data_teal_data) - - datanames_teal_data <- sort(teal.data::datanames(data_teal_data)) - datanames_tdata <- sort(names(data_tdata_downgraded)) - - testthat::expect_identical(datanames_teal_data, datanames_tdata) - - datasets_teal_data <- sapply(datanames_teal_data, function(x) teal.code::get_var(data_teal_data, x)) - datasets_tdata <- sapply(datanames_tdata, function(x) shiny::isolate(data_tdata_downgraded[[x]]())) - - testthat::expect_identical(datasets_teal_data, datasets_tdata) -}) - -testthat::test_that("as_tdata maintains code during conversion", { - data_teal_data_downgraded <- as_tdata(data_teal_data) - testthat::expect_identical( - teal.code::get_code(data_teal_data), - shiny::isolate(attr(data_teal_data_downgraded, "code")()) - ) -}) diff --git a/tests/testthat/test-teal_data_module.R b/tests/testthat/test-teal_data_module.R index 4578649eb1..d751cdc000 100644 --- a/tests/testthat/test-teal_data_module.R +++ b/tests/testthat/test-teal_data_module.R @@ -15,6 +15,10 @@ testthat::test_that("teal_data_module throws when ui has other formals than id o testthat::test_that("teal_data_module throws when server has other formals than id only", { testthat::expect_error( teal_data_module(ui = function(id) tags$div(), server = function(id, x) NULL), - "Must have exactly 1 formal arguments" + ".*exactly 1 formal.*" + ) + testthat::expect_error( + teal_data_module(ui = function(id) tags$div(), server = function(id, x) NULL), + ".*formal arguments.*" ) }) diff --git a/tests/testthat/test-utils.R b/tests/testthat/test-utils.R index 505cc14fec..628f389162 100644 --- a/tests/testthat/test-utils.R +++ b/tests/testthat/test-utils.R @@ -51,7 +51,7 @@ test_that("teal_data_datanames returns names of the @env's objects when dataname iris <- head(iris) mtcars <- head(mtcars) }) - testthat::expect_setequal(teal_data_datanames(teal_data), c("mtcars", "iris")) + testthat::expect_setequal(.teal_data_datanames(teal_data), c("mtcars", "iris")) }) test_that("teal_data_datanames returns datanames which are set by teal.data::datanames", { @@ -61,64 +61,7 @@ test_that("teal_data_datanames returns datanames which are set by teal.data::dat mtcars <- head(mtcars) }) datanames(teal_data) <- "iris" - testthat::expect_equal(teal_data_datanames(teal_data), "iris") -}) - -test_that("modules_datasets returns correct structure", { - data <- teal_data() %>% - within({ - iris <- iris - mtcars <- mtcars - x <- 5 - }) - - modules <- modules( - label = "one", - modules( - label = "two", - example_module("example two", "all"), - modules( - label = "three", - example_module("example three", "iris"), - example_module("example four", "mtcars") - ) - ), - example_module("example one", "iris") - ) - - filters <- teal_slices( - teal_slice("iris", "Species"), - teal_slice("iris", "Sepal.Length"), - teal_slice("mtcars", "mpg"), - teal_slice("mtcars", "cyl"), - teal_slice("mtcars", "gear"), - module_specific = TRUE, - mapping = list( - "example one" = "iris Species", - "example four" = "mtcars mpg", - global_filters = "mtcars cyl" - ) - ) - - modules_structure <- rapply( - modules_datasets(data, modules, filters), - function(x) { - isolate(sapply(x$get_filter_state(), `[[`, "id")) - }, - how = "replace" - ) - expected_structure <- list( - two = list( - `example two` = "mtcars cyl", - three = list( - `example three` = list(), - `example four` = c("mtcars mpg", "mtcars cyl") - ) - ), - `example one` = "iris Species" - ) - - testthat::expect_identical(modules_structure, expected_structure) + testthat::expect_equal(.teal_data_datanames(teal_data), "iris") }) test_that("validate_app_title_tag works on validating the title tag", { @@ -263,3 +206,20 @@ testthat::test_that("create_renv_lockfile creates a lock file during the executi testthat::expect_true(file.exists(renv_file_name)) }) + +testthat::test_that("check_modules_datanames message is the same in html tags and in string", { + testthat::skip_if_not_installed("rvest") + modules <- module(datanames = c("iris", "mtcars"), ui = function(id) NULL, server = function(id, data) NULL) + + message <- check_modules_datanames(modules, "missing") + + # Compares 2 strings (removes quotations and empty space surrounding tags) + testthat::expect_identical( + gsub("\"", "", message$string), + trimws( + rvest::html_text2( + rvest::read_html(as.character(message$html(with_module_name = TRUE))) + ) + ) + ) +}) diff --git a/vignettes/adding-support-for-reporting.Rmd b/vignettes/adding-support-for-reporting.Rmd index f5ebb24d16..91d3c2bc40 100644 --- a/vignettes/adding-support-for-reporting.Rmd +++ b/vignettes/adding-support-for-reporting.Rmd @@ -31,7 +31,7 @@ The entire life cycle of objects involved in creating the report and configuring Let us consider an example module, based on the example module from `teal`: ```{r, message=FALSE} library(teal) -example_module <- function(label = "example teal module") { +my_module <- function(label = "example teal module") { module( label = label, server = function(id, data) { @@ -62,7 +62,7 @@ Using `teal`, you can launch this example module with the following: ```{r, eval = FALSE} app <- init( data = teal_data(IRIS = iris, MTCARS = mtcars), - modules = example_module() + modules = my_module() ) if (interactive()) shinyApp(app$ui, app$server) @@ -77,7 +77,7 @@ This informs `teal` that the module requires `reporter`, and it will be included See below: ```{r} -example_module_with_reporting <- function(label = "example teal module") { +my_module_with_reporting <- function(label = "example teal module") { module( label = label, server = function(id, data, reporter) { @@ -105,7 +105,7 @@ With these modifications, the module is now ready to be launched with `teal`: ```{r} app <- init( data = teal_data(IRIS = iris, MTCARS = mtcars), - modules = example_module_with_reporting() + modules = my_module_with_reporting() ) if (interactive()) shinyApp(app$ui, app$server) @@ -117,10 +117,10 @@ That requires inserting UI and server elements of the `teal.reporter` module int ### Insert `teal.reporter` module -The UI and the server logic necessary for adding cards from `example_module_with_reporting` to the report are provided by `teal.reporter::simple_reporter_ui` and `teal.reporter::simple_reporter_srv`. +The UI and the server logic necessary for adding cards from `my_module_with_reporting` to the report are provided by `teal.reporter::simple_reporter_ui` and `teal.reporter::simple_reporter_srv`. ```{r} -example_module_with_reporting <- function(label = "example teal module") { +my_module_with_reporting <- function(label = "example teal module") { module( label = label, server = function(id, data, reporter) { @@ -156,7 +156,7 @@ This updated module is now ready to be launched: ```{r} app <- init( data = teal_data(IRIS = iris, MTCARS = mtcars), - modules = example_module_with_reporting() + modules = my_module_with_reporting() ) if (interactive()) shinyApp(app$ui, app$server) @@ -182,7 +182,7 @@ custom_function <- function(card = teal.reporter::ReportCard$new()) { card } -example_module_with_reporting <- function(label = "example teal module") { +my_module_with_reporting <- function(label = "example teal module") { module( label = label, server = function(id, data, reporter) { @@ -216,7 +216,7 @@ example_module_with_reporting <- function(label = "example teal module") { ```{r} app <- init( data = teal_data(IRIS = iris, MTCARS = mtcars), - modules = example_module_with_reporting() + modules = my_module_with_reporting() ) if (interactive()) shinyApp(app$ui, app$server) @@ -353,7 +353,7 @@ app <- init( data = teal_data(AIR = airquality, IRIS = iris), modules = list( example_reporter_module(label = "with Reporter"), - example_module(label = "without Reporter") + my_module(label = "without Reporter") ), filter = teal_slices(teal_slice(dataname = "AIR", varname = "Temp", selected = c(72, 85))), header = "Example teal app with reporter" diff --git a/vignettes/bootstrap-themes-in-teal.Rmd b/vignettes/bootstrap-themes-in-teal.Rmd index 937e90212c..8ea255e4c6 100644 --- a/vignettes/bootstrap-themes-in-teal.Rmd +++ b/vignettes/bootstrap-themes-in-teal.Rmd @@ -75,7 +75,10 @@ The most important HTML tags in `teal` have a specific id or class, so they can ``` library(magrittr) -options("teal.bs_theme" = bslib::bs_theme(version = "5") %>% bslib::bs_add_rules("Anything understood by sass::as_sass()")) +options("teal.bs_theme" = bslib::bs_add_rules( + bslib::bs_theme(version = "5"), + "Anything understood by sass::as_sass()" +)) ``` Other `bslib::bs_add_*` family functions could be used to specify low-level Bootstrap elements. diff --git a/vignettes/creating-custom-modules.Rmd b/vignettes/creating-custom-modules.Rmd index 6353512e57..0d1cea359c 100644 --- a/vignettes/creating-custom-modules.Rmd +++ b/vignettes/creating-custom-modules.Rmd @@ -56,7 +56,7 @@ Note that dataset choices are specified by the `datanames` property of the `teal ```{r, message=FALSE} library(teal) -example_module <- function(label = "example teal module") { +my_module <- function(label = "example teal module") { checkmate::assert_string(label) module( @@ -149,10 +149,8 @@ srv_histogram_example <- function(id, data) { }) # view code - output$code <- renderPrint({ - plot_code_q() %>% - get_code() %>% - cat() + output$code <- renderText({ + get_code(plot_code_q()) }) }) } diff --git a/vignettes/data-transform-as-shiny-module.Rmd b/vignettes/data-transform-as-shiny-module.Rmd new file mode 100644 index 0000000000..306dd3217c --- /dev/null +++ b/vignettes/data-transform-as-shiny-module.Rmd @@ -0,0 +1,235 @@ +--- +title: "Data Transformations as shiny Module" +author: "NEST CoreDev" +output: + rmarkdown::html_vignette: + toc: true +vignette: > + %\VignetteIndexEntry{Data Transformations as shiny Module} + %\VignetteEngine{knitr::rmarkdown} + %\VignetteEncoding{UTF-8} +--- + +## Introduction + +`teal` version `0.16` introduced new argument in `teal::module` called `transformers`. +This argument allows to pass a `list` of `teal_data_module` objects that are created using `teal_transform_module()` function. + +The main benefit of `teal_transform_module()` is the ability to transform data before passing it +to the module. This feature allows to extend the regular behavior of existing modules by specifying custom data operations on data inside this module. + +`teal_transform_module()` is a Shiny module that takes `ui` and `server` arguments. When provided, `teal` will execute data transformations for the specified module when it is loaded and whenever the data changes. `server` extend the logic behind data manipulations, where `ui` extends filter panel with new UI elements that orchestrate the transformer inputs. + +This vignette presents the way on how to manage custom data transformations in `teal` apps. + +## Creating your first custom data transformation module + +We initialize a simple `teal` app where we pass `iris` and `mtcars` as the input datasets. + +```{r, message = FALSE, warning = FALSE} +library(teal) +``` + +```{r} +data <- within(teal_data(), { + iris <- iris + mtcars <- mtcars +}) + +app <- init( + data = data, + modules = teal::example_module() +) + +if (interactive()) { + shinyApp(app$ui, app$server) +} +``` + +### Single Transformer + +Let's create a simple `teal_transform_module` that returns the first `n` number of rows of `iris` based on the user input. + +We do this by creating the `ui` with the `numericInput` for the user to input the number of rows to be displayed. +In the `server` function we take in the reactive `data` and perform this transformation and return the new reactive `data`. + +```{r} +data <- within(teal_data(), { + iris <- iris + mtcars <- mtcars +}) +datanames(data) <- c("iris", "mtcars") + +my_transformers <- list( + teal_transform_module( + label = "Custom transform for iris", + ui = function(id) { + ns <- NS(id) + tags$div( + numericInput(ns("n_rows"), "Number of rows to subset", value = 6, min = 1, max = 150, step = 1) + ) + }, + server = function(id, data) { + moduleServer(id, function(input, output, session) { + reactive({ + within(data(), + { + iris <- head(iris, num_rows) + }, + num_rows = input$n_rows + ) + }) + }) + } + ) +) + +app <- init( + data = data, + modules = teal::example_module(transformers = my_transformers) +) + +if (interactive()) { + shinyApp(app$ui, app$server) +} +``` + +### Multiple Transformers + +Note that we can add multiple `teal` transformers by including `teal_transform_module` in a list. + +Let's add another transformation to the `mtcars` dataset that creates a column with `rownames` of `mtcars`. +Note that this module does not have interactive UI elements. + +```{r} +data <- within(teal_data(), { + iris <- iris + mtcars <- mtcars +}) +datanames(data) <- c("iris", "mtcars") + +my_transformers <- list( + teal_transform_module( + label = "Custom transform for iris", + ui = function(id) { + ns <- NS(id) + tags$div( + numericInput(ns("n_rows"), "Number of rows to subset", value = 6, min = 1, max = 150, step = 1) + ) + }, + server = function(id, data) { + moduleServer(id, function(input, output, session) { + reactive({ + within(data(), + { + iris <- head(iris, num_rows) + }, + num_rows = input$n_rows + ) + }) + }) + } + ), + teal_transform_module( + label = "Custom transform for mtcars", + ui = function(id) { + ns <- NS(id) + tags$div( + "Adding rownames column to mtcars" + ) + }, + server = function(id, data) { + moduleServer(id, function(input, output, session) { + reactive({ + within(data(), { + mtcars$rownames <- rownames(mtcars) + rownames(mtcars) <- NULL + }) + }) + }) + } + ) +) + +app <- init( + data = data, + modules = teal::example_module(transformers = my_transformers) +) + +if (interactive()) { + shinyApp(app$ui, app$server) +} +``` + +## Custom placement of the transform UI + +When a custom transformation is used, the UI for the transformation is placed below the filter panel. +However, there is a way to customize the placement of the UI inside the module content. + +In order to place the transformation UI inside the module there are few things one has to do: +1. Create a custom module wrapper function. +2. Call the desired module in the module wrapper function and store it in a variable so it's UI can be modified. +3. Modify the UI of the module with the transform UI at the desired location by calling the `ui_transform_data`. Note that in order for the transform to work you need to change the namespace of the `id` by passing `NS(gsub("-module$", "", id), "data_transform")`. +4. Set the `custom_ui` attribute of the `module$transformers` to `TRUE`. + +Now the custom module should embed the transformation UI inside the module content. + +Here is an example of a custom module wrapper function that modifies the `example_module` module. +```{r} +example_module_encoding <- function(label = "example module (on encoding)", datanames = "all", transformers = list()) { + mod <- example_module(label, datanames, transformers) + mod$ui <- function(id) { + ns <- NS(id) + teal.widgets::standard_layout( + output = verbatimTextOutput(ns("text")), + encoding = tags$div( + ui_transform_data(NS(gsub("-module$", "", id), "data_transform"), transformers), + selectInput(ns("dataname"), "Choose a dataset", choices = NULL), + teal.widgets::verbatim_popup_ui(ns("rcode"), "Show R code") + ) + ) + } + attr(mod$transformers, "custom_ui") <- TRUE + mod +} + +data <- within(teal_data(), { + iris <- iris + mtcars <- mtcars +}) +datanames(data) <- c("iris", "mtcars") + +my_transformers <- list( + teal_transform_module( + label = "Custom transform for iris", + ui = function(id) { + ns <- NS(id) + tags$div( + numericInput(ns("n_rows"), "Number of rows to subset", value = 6, min = 1, max = 150, step = 1) + ) + }, + server = function(id, data) { + moduleServer(id, function(input, output, session) { + reactive({ + within(data(), + { + iris <- head(iris, num_rows) + }, + num_rows = input$n_rows + ) + }) + }) + } + ) +) + +app <- init( + data = data, + modules = example_module_encoding(transformers = my_transformers) +) + +if (interactive()) { + shinyApp(app$ui, app$server) +} +``` + diff --git a/vignettes/including-data-in-teal-applications.Rmd b/vignettes/including-data-in-teal-applications.Rmd index 723210cc07..16f33f7e22 100644 --- a/vignettes/including-data-in-teal-applications.Rmd +++ b/vignettes/including-data-in-teal-applications.Rmd @@ -189,12 +189,11 @@ For convenience, an empty `datanames` property is considered to mean "all object ```{r} data_with_objects <- teal_data(iris = iris, cars = mtcars) -data_with_code <- teal_data() %>% - within({ - iris <- iris - cars <- mtcars - not_a_dataset <- "data source credits" - }) +data_with_code <- within(teal_data(), { + iris <- iris + cars <- mtcars + not_a_dataset <- "data source credits" +}) datanames(data_with_objects) datanames(data_with_code) datanames(data_with_code) <- c("iris", "cars")