diff --git a/DESCRIPTION b/DESCRIPTION index e88e29a11e..84f77760f8 100644 --- a/DESCRIPTION +++ b/DESCRIPTION @@ -93,12 +93,14 @@ Collate: 'modules.R' 'init.R' 'landing_popup_module.R' + 'module_bookmark_manager.R' 'module_filter_manager.R' 'module_nested_tabs.R' 'module_snapshot_manager.R' 'module_tabs_with_filters.R' 'module_teal.R' 'module_teal_with_splash.R' + 'module_wunder_bar.R' 'reporter_previewer_module.R' 'show_rcode_modal.R' 'tdata.R' diff --git a/NEWS.md b/NEWS.md index afb7101b1c..401182f502 100644 --- a/NEWS.md +++ b/NEWS.md @@ -1,5 +1,8 @@ # teal 0.15.2.9018 +### Miscellaneous +* Filter mapping display is no longer coupled to the snapshot manager. + # teal 0.15.2 ### Bug fixes diff --git a/R/dummy_functions.R b/R/dummy_functions.R index 19abbaa16d..4624b783d9 100644 --- a/R/dummy_functions.R +++ b/R/dummy_functions.R @@ -15,12 +15,16 @@ #' @export example_module <- function(label = "example teal module", datanames = "all") { checkmate::assert_string(label) - module( + ans <- module( label, server = function(id, data) { checkmate::assert_class(data(), "teal_data") moduleServer(id, function(input, output, session) { - updateSelectInput(session, "dataname", choices = isolate(teal.data::datanames(data()))) + updateSelectInput( + inputId = "dataname", + choices = isolate(teal.data::datanames(data())), + selected = restoreInput(session$ns("dataname"), NULL) + ) output$text <- renderPrint({ req(input$dataname) data()[[input$dataname]] @@ -44,4 +48,6 @@ example_module <- function(label = "example teal module", datanames = "all") { }, datanames = datanames ) + attr(ans, "teal_bookmarkable") <- TRUE + ans } diff --git a/R/init.R b/R/init.R index 445d9432b0..8bea52b69a 100644 --- a/R/init.R +++ b/R/init.R @@ -35,7 +35,7 @@ #' 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. #' -#' @return Named list with server and UI functions. +#' @return Named list containing server and UI functions. #' #' @export #' @@ -164,8 +164,8 @@ init <- function(data, stop("Only one `landing_popup_module` can be used.") } - ## `filter` - app_id attribute - attr(filter, "app_id") <- create_app_id(data, modules) + ## `filter` - set app_id attribute unless present (when restoring bookmark) + if (is.null(attr(filter, "app_id", exact = TRUE))) attr(filter, "app_id") <- create_app_id(data, modules) ## `filter` - convert teal.slice::teal_slices to teal::teal_slices filter <- as.teal_slices(as.list(filter)) @@ -221,8 +221,9 @@ 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 + # Note: UI must be a function to support bookmarking. res <- list( - ui = ui_teal_with_splash(id = id, data = data, title = title, header = header, footer = footer), + ui = function(request) ui_teal_with_splash(id = id, data = data, 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)) diff --git a/R/module_bookmark_manager.R b/R/module_bookmark_manager.R new file mode 100644 index 0000000000..594392ba00 --- /dev/null +++ b/R/module_bookmark_manager.R @@ -0,0 +1,313 @@ +#' App state management. +#' +#' @description +#' `r lifecycle::badge("experimental")` +#' +#' Capture and restore the global (app) input state. +#' +#' @details +#' 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`]. +#' 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. +#' Those that are, have a `teal_bookmarkable` attribute set to `TRUE`. If any modules are not bookmarkable, +#' the bookmark manager modal displays a warning and the bookmark button displays a flag. +#' In order to communicate that a external module is bookmarkable, the module developer +#' should set the `teal_bookmarkable` attribute to `TRUE`. +#' +#' @section Server logic: +#' A bookmark is a URL that contains the app address with a `/?_state_id_=` suffix. +#' `` is a directory created on the server, where the state of the application is saved. +#' Accessing the bookmark URL opens a new session of the app that starts in the previously saved state. +#' +#' @section Note: +#' To enable bookmarking use either: +#' - `shiny` app by using `shinyApp(..., enableBookmarking = "server")` (not supported in `shinytest2`) +#' - set `options(shiny.bookmarkStore = "server")` before running the app +#' +#' +#' @inheritParams module_wunder_bar +#' +#' @return Invisible `NULL`. +#' +#' @aliases bookmark bookmark_manager bookmark_manager_module +#' +#' @name module_bookmark_manager +#' @keywords internal +#' +bookmark_manager_ui <- function(id) { + ns <- NS(id) + uiOutput(ns("bookmark_button"), inline = TRUE) +} + +#' @rdname module_bookmark_manager +#' @keywords internal +#' +bookmark_manager_srv <- 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") + 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" + ) + } + ) + ) + } + }) + + # 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$onBookmarked(function(url) { + logger::log_trace("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", + bookmark_option, + "Only server-side bookmarking is supported.", + "Please contact your app developer." + ) + tags$div( + tags$p(msg, class = "text-warning") + ) + } else { + tags$div( + tags$span( + tags$pre(url) + ), + if (any(is_unbookmarkable)) { + bkmb_summary <- rapply2( + modules_bookmarkable(modules), + function(x) { + if (isTRUE(x)) { + "\u2705" # check mark + } else if (isFALSE(x)) { + "\u274C" # cross mark + } else { + "\u2753" # question mark + } + } + ) + tags$div( + tags$p( + icon("fas fa-exclamation-triangle"), + "Some modules will not be restored when using this bookmark.", + tags$br(), + "Check the list below to see which modules are not bookmarkable.", + class = "text-warning" + ), + tags$pre(yaml::as.yaml(bkmb_summary)) + ) + } + ) + } + + showModal( + modalDialog( + id = ns("bookmark_modal"), + title = "Bookmarked teal app url", + modal_content, + easyClose = TRUE + ) + ) + }) + + # 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.") + session$doBookmark() + }) + + invisible(NULL) + }) +} + +# utilities ---- + +#' Restore value from bookmark. +#' +#' Get value from bookmark or return default. +#' +#' Bookmarks can store not only inputs but also arbitrary values. +#' These values are stored by `onBookmark` callbacks and restored by `onBookmarked` callbacks, +#' and they are placed in the `values` environment in the `session$restoreContext` field. +#' Using `teal_data_module` makes it impossible to run the callbacks +#' because the app becomes ready before modules execute and callbacks are registered. +#' In those cases the stored values can still be recovered from the `session` object directly. +#' +#' Note that variable names in the `values` environment are prefixed with module name space names, +#' therefore, when using this function in modules, `value` must be run through the name space function. +#' +#' @param value (`character(1)`) name of value to restore +#' @param default fallback value +#' +#' @return +#' In an application restored from a server-side bookmark, +#' the variable specified by `value` from the `values` environment. +#' Otherwise `default`. +#' +#' @keywords internal +#' +restoreValue <- function(value, default) { # nolint: object_name. + checkmate::assert_character("value") + session_default <- shiny::getDefaultReactiveDomain() + session_parent <- .subset2(session_default, "parent") + session <- if (is.null(session_parent)) session_default else session_parent + + if (isTRUE(session$restoreContext$active) && exists(value, session$restoreContext$values, inherits = FALSE)) { + session$restoreContext$values[[value]] + } else { + default + } +} + +#' Compare bookmarks. +#' +#' Test if two bookmarks store identical state. +#' +#' `input` environments are compared one variable at a time and if not identical, +#' values in both bookmarks are reported. States of `datatable`s are stripped +#' of the `time` element before comparing because the time stamp is always different. +#' The contents themselves are not printed as they are large and the contents are not informative. +#' Elements present in one bookmark and absent in the other are also reported. +#' Differences are printed as messages. +#' +#' `values` environments are compared with `all.equal`. +#' +#' @section How to use: +#' Open an application, change relevant inputs (typically, all of them), and create a bookmark. +#' Then open that bookmark and immediately create a bookmark of that. +#' If restoring bookmarks occurred properly, the two bookmarks should store the same state. +#' +#' +#' @param book1,book2 bookmark directories stored in `shiny_bookmarks/`; +#' default to the two most recently modified directories +#' +#' @return +#' Invisible `NULL` if bookmarks are identical or if there are no bookmarks to test. +#' `FALSE` if inconsistencies are detected. +#' +#' @keywords internal +#' +bookmarks_identical <- function(book1, book2) { + if (!dir.exists("shiny_bookmarks")) { + message("no bookmark directory") + return(invisible(NULL)) + } + + ans <- TRUE + + if (missing(book1) && missing(book2)) { + dirs <- list.dirs("shiny_bookmarks", recursive = FALSE) + bookmarks_sorted <- basename(rev(dirs[order(file.mtime(dirs))])) + if (length(bookmarks_sorted) < 2L) { + message("no bookmarks to compare") + return(invisible(NULL)) + } + book1 <- bookmarks_sorted[2L] + book2 <- bookmarks_sorted[1L] + } else { + if (!dir.exists(file.path("shiny_bookmarks", book1))) stop(book1, " not found") + if (!dir.exists(file.path("shiny_bookmarks", book2))) stop(book2, " not found") + } + + book1_input <- readRDS(file.path("shiny_bookmarks", book1, "input.rds")) + book2_input <- readRDS(file.path("shiny_bookmarks", book2, "input.rds")) + + elements_common <- intersect(names(book1_input), names(book2_input)) + dt_states <- grepl("_state$", elements_common) + if (any(dt_states)) { + for (el in elements_common[dt_states]) { + book1_input[[el]][["time"]] <- NULL + book2_input[[el]][["time"]] <- NULL + } + } + + identicals <- mapply(identical, book1_input[elements_common], book2_input[elements_common]) + non_identicals <- names(identicals[!identicals]) + compares <- sprintf("$ %s:\t%s --- %s", non_identicals, book1_input[non_identicals], book2_input[non_identicals]) + if (length(compares) != 0L) { + message("common elements not identical: \n", paste(compares, collapse = "\n")) + ans <- FALSE + } + + elements_boook1 <- setdiff(names(book1_input), names(book2_input)) + if (length(elements_boook1) != 0L) { + dt_states <- grepl("_state$", elements_boook1) + if (any(dt_states)) { + for (el in elements_boook1[dt_states]) { + if (is.list(book1_input[[el]])) book1_input[[el]] <- "--- data table state ---" + } + } + excess1 <- sprintf("$ %s:\t%s", elements_boook1, book1_input[elements_boook1]) + message("elements only in book1: \n", paste(excess1, collapse = "\n")) + ans <- FALSE + } + + elements_boook2 <- setdiff(names(book2_input), names(book1_input)) + if (length(elements_boook2) != 0L) { + dt_states <- grepl("_state$", elements_boook1) + if (any(dt_states)) { + for (el in elements_boook1[dt_states]) { + if (is.list(book2_input[[el]])) book2_input[[el]] <- "--- data table state ---" + } + } + excess2 <- sprintf("$ %s:\t%s", elements_boook2, book2_input[elements_boook2]) + message("elements only in book2: \n", paste(excess2, collapse = "\n")) + ans <- FALSE + } + + book1_values <- readRDS(file.path("shiny_bookmarks", book1, "values.rds")) + book2_values <- readRDS(file.path("shiny_bookmarks", book2, "values.rds")) + + if (!isTRUE(all.equal(book1_values, book2_values))) { + message("different values detected") + message("choices for numeric filters MAY be different, see RangeFilterState$set_choices") + ans <- FALSE + } + + if (ans) message("perfect!") + invisible(NULL) +} + + +# Replacement for [base::rapply] which doesn't handle NULL values - skips the evaluation +# of the function and returns NULL for given element. +rapply2 <- function(x, f) { + if (inherits(x, "list")) { + lapply(x, rapply2, f = f) + } else { + f(x) + } +} diff --git a/R/module_filter_manager.R b/R/module_filter_manager.R index bf3fbd0c2a..611a1bcc38 100644 --- a/R/module_filter_manager.R +++ b/R/module_filter_manager.R @@ -7,75 +7,47 @@ #' is kept in the `mapping_matrix` object (which is actually a `data.frame`) #' that tracks which filters (rows) are active in which modules (columns). #' -#' @name module_filter_manager -#' #' @param id (`character(1)`) -#' `shiny` module id. -#' @param filtered_data_list (named `list`) +#' `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 of `reactive`s, each holding a `teal_slices`, as returned by `filter_manager_module_srv`. -#' @keywords internal #' -NULL - -#' Filter manager modal +#' @return +#' A `list` containing: #' -#' Opens a modal containing the filter manager UI. +#' 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, #' -#' @name module_filter_manager_modal -#' @inheritParams module_filter_manager -#' @keywords internal +#' objects used for testing +#' - modules_out: `list` of `reactive`s, each holding a `teal_slices`, as returned by `filter_manager_module_srv`. +#' +#' @name module_filter_manager +#' @aliases filter_manager filter_manager_module #' -NULL - -#' @rdname module_filter_manager_modal -filter_manager_modal_ui <- function(id) { - ns <- NS(id) - tags$button( - id = ns("show"), - class = "btn action-button filter_manager_button", - title = "Show filters manager modal", - icon("gear") - ) -} - -#' @rdname module_filter_manager_modal -filter_manager_modal_srv <- function(id, filtered_data_list, filter) { - moduleServer(id, function(input, output, session) { - observeEvent(input$show, { - logger::log_trace("filter_manager_modal_srv@1 show button has been clicked.") - showModal( - modalDialog( - filter_manager_ui(session$ns("filter_manager")), - size = "l", - footer = NULL, - easyClose = TRUE - ) - ) - }) - - filter_manager_srv("filter_manager", filtered_data_list, filter) - }) -} #' @rdname module_filter_manager +#' @keywords internal +#' filter_manager_ui <- function(id) { ns <- NS(id) tags$div( class = "filter_manager_content", - tableOutput(ns("slices_table")), - snapshot_manager_ui(ns("snapshot_manager")) + tableOutput(ns("slices_table")) ) } #' @rdname module_filter_manager -filter_manager_srv <- function(id, filtered_data_list, filter) { +#' @keywords internal +#' +filter_manager_srv <- function(id, datasets, filter) { moduleServer(id, function(input, output, session) { - logger::log_trace("filter_manager_srv initializing for: { paste(names(filtered_data_list), collapse = ', ')}.") + logger::log_trace("filter_manager_srv initializing for: { paste(names(datasets), collapse = ', ')}.") is_module_specific <- isTRUE(attr(filter, "module_specific")) @@ -85,29 +57,18 @@ filter_manager_srv <- function(id, filtered_data_list, filter) { # Down there a subset that pertains to the data sets used in that module is applied and displayed. slices_global <- reactiveVal(filter) - filtered_data_list <- + datasets_flat <- if (!is_module_specific) { - # Retrieve the first FilteredData from potentially nested list. - # List of length one is named "global_filters" because that name is forbidden for a module label. - list(global_filters = unlist(filtered_data_list)[[1]]) + flatten_datasets(unlist(datasets)[[1]]) } else { - # Flatten potentially nested list of FilteredData objects while maintaining useful names. - # Simply using `unlist` would result in concatenated names. - flatten_nested <- function(x, name = NULL) { - if (inherits(x, "FilteredData")) { - setNames(list(x), name) - } else { - unlist(lapply(names(x), function(name) flatten_nested(x[[name]], name))) - } - } - flatten_nested(filtered_data_list) + flatten_datasets(datasets) } # 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(filtered_data_list, function(x) { + 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 @@ -123,7 +84,6 @@ filter_manager_srv <- function(id, filtered_data_list, filter) { 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)) - if (!is_module_specific) colnames(mm) <- "Global Filters" # Display placeholder if no filters defined. if (nrow(mm) == 0L) { @@ -134,23 +94,25 @@ filter_manager_srv <- function(id, filtered_data_list, filter) { # Report Previewer will not be displayed. mm[names(mm) != "Report previewer"] }, - align = paste(c("l", rep("c", sum(names(filtered_data_list) != "Report previewer"))), collapse = ""), + align = paste(c("l", rep("c", sum(names(datasets_flat) != "Report previewer"))), collapse = ""), rownames = TRUE ) # Create list of module calls. - modules_out <- lapply(names(filtered_data_list), function(module_name) { + modules_out <- lapply(names(datasets_flat), function(module_name) { filter_manager_module_srv( id = module_name, - module_fd = filtered_data_list[[module_name]], + module_fd = datasets_flat[[module_name]], slices_global = slices_global ) }) - # Call snapshot manager. - snapshot_manager_srv("snapshot_manager", slices_global, mapping_matrix, filtered_data_list) - - modules_out # returned for testing purpose + list( + slices_global = slices_global, + mapping_matrix = mapping_matrix, + datasets_flat = datasets_flat, + modules_out = modules_out # returned for testing purpose + ) }) } @@ -173,7 +135,7 @@ filter_manager_srv <- function(id, filtered_data_list, filter) { #' - 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 the slices active in this module. +#' @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) { @@ -216,3 +178,25 @@ filter_manager_module_srv <- function(id, module_fd, slices_global) { slices_module # returned for testing purpose }) } + + + +# utilities ---- + +#' 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))) + } +} diff --git a/R/module_nested_tabs.R b/R/module_nested_tabs.R index 07825917ca..71f0cd22fd 100644 --- a/R/module_nested_tabs.R +++ b/R/module_nested_tabs.R @@ -216,19 +216,34 @@ srv_nested_tabs.teal_module <- function(id, datasets, modules, is_module_specifi args <- c(args, filter_panel_api = filter_panel_api) } - # observe the trigger_module above to induce the module once the renderUI is triggered - observeEvent( - ignoreNULL = TRUE, - once = TRUE, - eventExpr = trigger_module(), - handlerExpr = { - module_output <- if (is_arg_used(modules$server, "id")) { - do.call(modules$server, args) - } else { - do.call(callModule, c(args, list(module = modules$server))) - } + # 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))) } - ) + } + + # 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( + ignoreNULL = TRUE, + once = TRUE, + eventExpr = trigger_module(), + handlerExpr = call_module() + ) + } reactive(modules) }) diff --git a/R/module_snapshot_manager.R b/R/module_snapshot_manager.R index 6335b2a69e..bbdfdea2f5 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 through the filter manager, with the cog icon in the top right corner. +#' The snapshot manager is accessed with the camera icon in the [`wunder_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. @@ -44,7 +44,7 @@ #' This is substituted as the snapshot's `mapping` attribute and the snapshot is added to the snapshot list. #' #' To restore app state, a snapshot is retrieved from storage and rebuilt into a `teal_slices` object. -#' Then state of all `FilteredData` objects (provided in `filtered_data_list`) is cleared +#' Then state of all `FilteredData` objects (provided in `datasets`) is cleared #' and set anew according to the `mapping` attribute of the snapshot. #' The snapshot is then set as the current content of `slices_global`. #' @@ -65,30 +65,38 @@ #' a `teal_slices` object. When a snapshot is restored from file, its `app_id` is compared to that #' of the current app state and only if the match is the snapshot admitted to the session. #' -#' @param id (`character(1)`) `shiny` module id +#' @section Bookmarks: +#' An `onBookmark` callback creates a snapshot of the current filter state. +#' This is done on the app session, not the module session. +#' (The snapshot will be retrieved by `module_teal` in order to set initial app state in a restored app.) +#' Then that snapshot, and the previous snapshot history are dumped into the `values.rds` file in ``. +#' +#' @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 +#' 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 filtered_data_list non-nested (named `list`) that contains `FilteredData` objects +#' all columns are `logical` vectors. +#' @param datasets non-nested (named `list`) of `FilteredData` objects. #' -#' @return Nothing is returned. +#' @return `list` containing the snapshot history, where each element is an unlisted `teal_slices` object. #' -#' @name snapshot_manager_module -#' @aliases snapshot snapshot_manager +#' @name module_snapshot_manager +#' @aliases snapshot snapshot_manager snapshot_manager_module #' #' @author Aleksander Chlebowski #' -#' @rdname snapshot_manager_module + + +#' @rdname module_snapshot_manager #' @keywords internal #' snapshot_manager_ui <- function(id) { ns <- NS(id) tags$div( - class = "snapshot_manager_content", + class = "manager_content", tags$div( - class = "snapshot_table_row", + 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"), @@ -99,31 +107,56 @@ snapshot_manager_ui <- function(id) { ) } -#' @rdname snapshot_manager_module +#' @rdname module_snapshot_manager #' @keywords internal #' -snapshot_manager_srv <- function(id, slices_global, mapping_matrix, filtered_data_list) { +snapshot_manager_srv <- function(id, slices_global, mapping_matrix, datasets) { 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(filtered_data_list, types = "FilteredData", any.missing = FALSE, names = "named") + checkmate::assert_list(datasets, types = "FilteredData", any.missing = FALSE, names = "named") moduleServer(id, function(input, output, session) { + logger::log_trace("snapshot_manager_srv initializing") + + # Set up bookmarking callbacks ---- + # Register bookmark exclusions (all buttons and text fields). + setBookmarkExclude(c( + "snapshot_add", "snapshot_load", "snapshot_reset", + "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") + state$values$snapshot_history <- snapshot_history() # isolate this? + }) + ns <- session$ns - # Store global filter states ---- + # Track global filter states ---- filter <- isolate(slices_global()) snapshot_history <- reactiveVal({ - list( - "Initial application state" = as.list(filter, recursive = TRUE) - ) + # Restore directly from bookmarked state, if applicable. + restoreValue(ns("snapshot_history"), list("Initial application state" = as.list(filter, recursive = TRUE))) }) # Snapshot current application state ---- # Name snaphsot. observeEvent(input$snapshot_add, { + logger::log_trace("snapshot_manager_srv: snapshot_add button clicked") showModal( modalDialog( textInput(ns("snapshot_name"), "Name the snapshot", width = "100%", placeholder = "Meaningful, unique name"), @@ -137,20 +170,24 @@ snapshot_manager_srv <- function(id, slices_global, mapping_matrix, filtered_dat }) # Store snaphsot. observeEvent(input$snapshot_name_accept, { + logger::log_trace("snapshot_manager_srv: snapshot_name_accept button clicked") snapshot_name <- trimws(input$snapshot_name) if (identical(snapshot_name, "")) { + logger::log_trace("snapshot_manager_srv: 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") 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()) snapshot_update <- c(snapshot_history(), list(snapshot)) @@ -158,13 +195,14 @@ snapshot_manager_srv <- function(id, slices_global, mapping_matrix, filtered_dat snapshot_history(snapshot_update) removeModal() # Reopen filter manager modal by clicking button in the main application. - shinyjs::click(id = "teal-main_ui-filter_manager-show", asis = TRUE) + shinyjs::click(id = "teal-main_ui-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") showModal( modalDialog( fileInput(ns("snapshot_file"), "Choose snapshot file", accept = ".json", width = "100%"), @@ -183,11 +221,14 @@ snapshot_manager_srv <- function(id, slices_global, mapping_matrix, filtered_dat }) # 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") snapshot_name <- trimws(input$snapshot_name) if (identical(snapshot_name, "")) { + logger::log_trace("snapshot_manager_srv: 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") showNotification( "This name is in conflict with other snapshot names. Please choose a different one.", type = "message" @@ -195,32 +236,37 @@ snapshot_manager_srv <- function(id, slices_global, mapping_matrix, filtered_dat 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") 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") 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") 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) snapshot_update <- c(snapshot_history(), list(snapshot)) names(snapshot_update)[length(snapshot_update)] <- snapshot_name snapshot_history(snapshot_update) ### Begin simplified restore procedure. ### - mapping_unfolded <- unfold_mapping(attr(snapshot_state, "mapping"), names(filtered_data_list)) + 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 = filtered_data_list, + filtered_data = datasets, filter_ids = mapping_unfolded ) slices_global(snapshot_state) @@ -233,18 +279,19 @@ snapshot_manager_srv <- function(id, slices_global, mapping_matrix, filtered_dat # Restore initial state ---- observeEvent(input$snapshot_reset, { + logger::log_trace("snapshot_manager_srv: snapshot_reset button clicked, restoring snapshot") s <- "Initial application state" ### Begin restore procedure. ### snapshot <- snapshot_history()[[s]] snapshot_state <- as.teal_slices(snapshot) - mapping_unfolded <- unfold_mapping(attr(snapshot_state, "mapping"), names(filtered_data_list)) + 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 = filtered_data_list, + filtered_data = datasets, filter_ids = mapping_unfolded ) slices_global(snapshot_state) @@ -261,6 +308,7 @@ snapshot_manager_srv <- function(id, slices_global, mapping_matrix, filtered_dat divs <- reactiveValues() observeEvent(snapshot_history(), { + logger::log_trace("snapshot_manager_srv: 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)) @@ -272,14 +320,14 @@ snapshot_manager_srv <- function(id, slices_global, mapping_matrix, filtered_dat ### Begin restore procedure. ### snapshot <- snapshot_history()[[s]] snapshot_state <- as.teal_slices(snapshot) - mapping_unfolded <- unfold_mapping(attr(snapshot_state, "mapping"), names(filtered_data_list)) + 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 = filtered_data_list, + filtered_data = datasets, filter_ids = mapping_unfolded ) slices_global(snapshot_state) @@ -304,7 +352,7 @@ snapshot_manager_srv <- function(id, slices_global, mapping_matrix, filtered_dat # Create a row for the snapshot table. if (!is.element(id_rowme, names(divs))) { divs[[id_rowme]] <- tags$div( - class = "snapshot_table_row", + 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") @@ -315,16 +363,18 @@ snapshot_manager_srv <- function(id, slices_global, mapping_matrix, filtered_dat # Create table to display list of snapshots and their actions. output$snapshot_list <- renderUI({ - rows <- lapply(rev(reactiveValuesToList(divs)), function(d) d) + rows <- rev(reactiveValuesToList(divs)) if (length(rows) == 0L) { tags$div( - class = "snapshot_manager_placeholder", + class = "manager_placeholder", "Snapshots will appear here." ) } else { rows } }) + + snapshot_history }) } @@ -337,6 +387,7 @@ snapshot_manager_srv <- function(id, slices_global, mapping_matrix, filtered_dat #' @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) { @@ -354,6 +405,7 @@ unfold_mapping <- function(mapping, module_names) { #' @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) { diff --git a/R/module_tabs_with_filters.R b/R/module_tabs_with_filters.R index 4cf2f58fdc..a9651c7ff4 100644 --- a/R/module_tabs_with_filters.R +++ b/R/module_tabs_with_filters.R @@ -44,10 +44,10 @@ ui_tabs_with_filters <- function(id, modules, datasets, filter = teal_slices()) 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 panels", + title = "Toggle filter panel", icon("fas fa-bars") ), - filter_manager_modal_ui(ns("filter_manager")) + wunder_bar_ui(ns("wunder_bar")) ) teal_ui$children[[1]] <- tagAppendChild(teal_ui$children[[1]], filter_panel_btns) @@ -84,7 +84,7 @@ srv_tabs_with_filters <- function(id, logger::log_trace("srv_tabs_with_filters initializing the module.") is_module_specific <- isTRUE(attr(filter, "module_specific")) - manager_out <- filter_manager_modal_srv("filter_manager", filtered_data_list = datasets, filter = filter) + wunder_bar_out <- wunder_bar_srv("wunder_bar", datasets, filter, modules) active_module <- srv_nested_tabs( id = "root", diff --git a/R/module_teal.R b/R/module_teal.R index cb12b09fd8..924077a79f 100644 --- a/R/module_teal.R +++ b/R/module_teal.R @@ -162,8 +162,13 @@ srv_teal <- function(id, modules, teal_data_rv, filter = teal_slices()) { env$progress <- shiny::Progress$new(session) env$progress$set(0.25, message = "Setting data") + # 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, teal_data_to_filtered_data(teal_data_rv())) + modules_datasets(teal_data_rv(), modules, filter_restored, teal_data_to_filtered_data(teal_data_rv())) }) # Replace splash / welcome screen once data is loaded ---- diff --git a/R/module_wunder_bar.R b/R/module_wunder_bar.R new file mode 100644 index 0000000000..28b245f7bb --- /dev/null +++ b/R/module_wunder_bar.R @@ -0,0 +1,93 @@ +#' 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 751d34fb74..4ec76a2ca7 100644 --- a/R/modules.R +++ b/R/modules.R @@ -421,3 +421,21 @@ module_labels <- function(modules) { modules$label } } + +#' Retrieve `teal_bookmarkable` attribute from `teal_modules` +#' +#' @param modules (`teal_modules` or `teal_module`) object +#' @return named list of the same structure as `modules` with `TRUE` or `FALSE` values indicating +#' whether the module is bookmarkable. +#' @keywords internal +modules_bookmarkable <- function(modules) { + checkmate::assert_multi_class(modules, c("teal_modules", "teal_module")) + if (inherits(modules, "teal_modules")) { + setNames( + lapply(modules$children, modules_bookmarkable), + vapply(modules$children, `[[`, "label", FUN.VALUE = character(1)) + ) + } else { + attr(modules, "teal_bookmarkable", exact = TRUE) + } +} diff --git a/R/reporter_previewer_module.R b/R/reporter_previewer_module.R index eeba694a7f..ba84f3173f 100644 --- a/R/reporter_previewer_module.R +++ b/R/reporter_previewer_module.R @@ -43,5 +43,6 @@ reporter_previewer_module <- function(label = "Report previewer", server_args = # This is to prevent another module being labeled "Report previewer". class(module) <- c("teal_module_previewer", class(module)) module$label <- label + attr(module, "teal_bookmarkable") <- TRUE module } diff --git a/inst/WORDLIST b/inst/WORDLIST index bfba486c17..20dae96097 100644 --- a/inst/WORDLIST +++ b/inst/WORDLIST @@ -1,26 +1,30 @@ Biomarker +bookmarkable CDISC -Forkers -Hoffmann -MAEs -ORCID -Reproducibility -TLG -UI -UIs -UX cloneable customizable +dialog favicon favicons +Forkers funder +Hoffmann +href +JSON +MAEs omics +ORCID pre programmatically repo reproducibility +Reproducibility summarization tabsetted themer theming +TLG +UI +UIs uncheck +UX diff --git a/inst/css/sidebar.css b/inst/css/sidebar.css index f0d30a0677..60a9380f7d 100644 --- a/inst/css/sidebar.css +++ b/inst/css/sidebar.css @@ -1,13 +1,26 @@ /* teal sidebar css */ -.filter_hamburger, .filter_manager_button { +.filter_hamburger, +.wunder_bar_button { font-size: 16px; padding: 8px !important; float: right !important; background-color: transparent !important; } +.badge-count { + padding-left: 1em; + padding-right: 1em; + -webkit-border-radius: 1em; + -moz-border-radius: 1em; + border-radius: 1em; + font-size: 0.7em; + padding: 0 .5em; + vertical-align: top; + margin-left: -1em; +} + /* disable any anchor with the disabled class */ a.disabled { pointer-events: none; @@ -17,29 +30,25 @@ a.disabled { .filter_manager_content { display: flex; - flex-direction: row; flex-wrap: wrap; - align-items: flex-start; justify-content: center; } -.filter_manager_content > * { - flex: 1 1 auto; - padding: 0em 1em; - width: min-content; -} -.snapshot_table_row { +.manager_table_row { display: flex; flex-direction: row; align-items: center; } -.snapshot_table_row *:first-child { + +.manager_table_row *:first-child { flex: 1 1 80%; } -.snapshot_table_row * + * { - flex: 1 0 50px; - padding: 0em 1em; + +.manager_table_row *+* { + flex: 0 0 0px; + padding: 0em 1.5em; } -.snapshot_manager_placeholder { + +.manager_placeholder { margin-top: 1em; } diff --git a/man/bookmarks_identical.Rd b/man/bookmarks_identical.Rd new file mode 100644 index 0000000000..a61169362f --- /dev/null +++ b/man/bookmarks_identical.Rd @@ -0,0 +1,37 @@ +% Generated by roxygen2: do not edit by hand +% Please edit documentation in R/module_bookmark_manager.R +\name{bookmarks_identical} +\alias{bookmarks_identical} +\title{Compare bookmarks.} +\usage{ +bookmarks_identical(book1, book2) +} +\arguments{ +\item{book1, book2}{bookmark directories stored in \verb{shiny_bookmarks/}; +default to the two most recently modified directories} +} +\value{ +Invisible \code{NULL} if bookmarks are identical or if there are no bookmarks to test. +\code{FALSE} if inconsistencies are detected. +} +\description{ +Test if two bookmarks store identical state. +} +\details{ +\code{input} environments are compared one variable at a time and if not identical, +values in both bookmarks are reported. States of \code{datatable}s are stripped +of the \code{time} element before comparing because the time stamp is always different. +The contents themselves are not printed as they are large and the contents are not informative. +Elements present in one bookmark and absent in the other are also reported. +Differences are printed as messages. + +\code{values} environments are compared with \code{all.equal}. +} +\section{How to use}{ + +Open an application, change relevant inputs (typically, all of them), and create a bookmark. +Then open that bookmark and immediately create a bookmark of that. +If restoring bookmarks occurred properly, the two bookmarks should store the same state. +} + +\keyword{internal} diff --git a/man/filter_manager_module_srv.Rd b/man/filter_manager_module_srv.Rd index e00afd6e38..e00216a5d1 100644 --- a/man/filter_manager_module_srv.Rd +++ b/man/filter_manager_module_srv.Rd @@ -22,7 +22,7 @@ stores \code{teal_slices} with all available filters; allows the following actio }} } \value{ -A \code{reactive} expression containing the slices active in this module. +A \code{reactive} expression containing a \code{teal_slices} with the slices active in this module. } \description{ Tracks filter states in a single module. diff --git a/man/init.Rd b/man/init.Rd index 991a7db62c..2f0e34d2f5 100644 --- a/man/init.Rd +++ b/man/init.Rd @@ -43,7 +43,7 @@ string specifying the \code{shiny} module id in cases it is used as a \code{shin rather than a standalone \code{shiny} app. This is a legacy feature.} } \value{ -Named list with server and UI functions. +Named list containing server and UI functions. } \description{ \ifelse{html}{\href{https://lifecycle.r-lib.org/articles/stages.html#stable}{\figure{lifecycle-stable.svg}{options: alt='[Stable]'}}}{\strong{[Stable]}} diff --git a/man/module_bookmark_manager.Rd b/man/module_bookmark_manager.Rd new file mode 100644 index 0000000000..ca7a892cb5 --- /dev/null +++ b/man/module_bookmark_manager.Rd @@ -0,0 +1,61 @@ +% Generated by roxygen2: do not edit by hand +% 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} +\title{App state management.} +\usage{ +bookmark_manager_ui(id) + +bookmark_manager_srv(id, modules) +} +\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.} +} +\value{ +Invisible \code{NULL}. +} +\description{ +\ifelse{html}{\href{https://lifecycle.r-lib.org/articles/stages.html#experimental}{\figure{lifecycle-experimental.svg}{options: alt='[Experimental]'}}}{\strong{[Experimental]}} + +Capture and restore the global (app) input state. +} +\details{ +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}}. +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. +Those that are, have a \code{teal_bookmarkable} attribute set to \code{TRUE}. If any modules are not bookmarkable, +the bookmark manager modal displays a warning and the bookmark button displays a flag. +In order to communicate that a external module is bookmarkable, the module developer +should set the \code{teal_bookmarkable} attribute to \code{TRUE}. +} +\section{Server logic}{ + +A bookmark is a URL that contains the app address with a \verb{/?_state_id_=} suffix. +\verb{} is a directory created on the server, where the state of the application is saved. +Accessing the bookmark URL opens a new session of the app that starts in the previously saved state. +} + +\section{Note}{ + +To enable bookmarking use either: +\itemize{ +\item \code{shiny} app by using \code{shinyApp(..., enableBookmarking = "server")} (not supported in \code{shinytest2}) +\item set \code{options(shiny.bookmarkStore = "server")} before running the app +} +} + +\keyword{internal} diff --git a/man/module_filter_manager.Rd b/man/module_filter_manager.Rd index 3d487118fd..1d0e5af2ef 100644 --- a/man/module_filter_manager.Rd +++ b/man/module_filter_manager.Rd @@ -3,18 +3,20 @@ \name{module_filter_manager} \alias{module_filter_manager} \alias{filter_manager_ui} +\alias{filter_manager} +\alias{filter_manager_module} \alias{filter_manager_srv} \title{Manage multiple \code{FilteredData} objects} \usage{ filter_manager_ui(id) -filter_manager_srv(id, filtered_data_list, filter) +filter_manager_srv(id, datasets, filter) } \arguments{ \item{id}{(\code{character(1)}) -\code{shiny} module id.} +\code{shiny} module instance id.} -\item{filtered_data_list}{(named \code{list}) +\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 @@ -24,7 +26,19 @@ and the names of the list must match the labels of their respective modules.} Specifies the initial filter using \code{\link[=teal_slices]{teal_slices()}}.} } \value{ -A list of \code{reactive}s, each holding a \code{teal_slices}, as returned by \code{filter_manager_module_srv}. +A \code{list} containing: + +objects used by other manager modules +\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, +} + +objects used for testing +\itemize{ +\item modules_out: \code{list} of \code{reactive}s, each holding a \code{teal_slices}, as returned by \code{filter_manager_module_srv}. +} } \description{ Oversee filter states across the entire application. diff --git a/man/module_filter_manager_modal.Rd b/man/module_filter_manager_modal.Rd deleted file mode 100644 index e3f5c5225f..0000000000 --- a/man/module_filter_manager_modal.Rd +++ /dev/null @@ -1,29 +0,0 @@ -% Generated by roxygen2: do not edit by hand -% Please edit documentation in R/module_filter_manager.R -\name{module_filter_manager_modal} -\alias{module_filter_manager_modal} -\alias{filter_manager_modal_ui} -\alias{filter_manager_modal_srv} -\title{Filter manager modal} -\usage{ -filter_manager_modal_ui(id) - -filter_manager_modal_srv(id, filtered_data_list, filter) -} -\arguments{ -\item{id}{(\code{character(1)}) -\code{shiny} module id.} - -\item{filtered_data_list}{(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{filter}{(\code{teal_slices}) -Specifies the initial filter using \code{\link[=teal_slices]{teal_slices()}}.} -} -\description{ -Opens a modal containing the filter manager UI. -} -\keyword{internal} diff --git a/man/snapshot_manager_module.Rd b/man/module_snapshot_manager.Rd similarity index 84% rename from man/snapshot_manager_module.Rd rename to man/module_snapshot_manager.Rd index 307abaf06a..80aa9fb44b 100644 --- a/man/snapshot_manager_module.Rd +++ b/man/module_snapshot_manager.Rd @@ -1,31 +1,32 @@ % Generated by roxygen2: do not edit by hand % Please edit documentation in R/module_snapshot_manager.R -\name{snapshot_manager_module} -\alias{snapshot_manager_module} +\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} \title{Filter state snapshot management} \usage{ snapshot_manager_ui(id) -snapshot_manager_srv(id, slices_global, mapping_matrix, filtered_data_list) +snapshot_manager_srv(id, slices_global, mapping_matrix, datasets) } \arguments{ -\item{id}{(\code{character(1)}) \code{shiny} module id} +\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} +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} +all columns are \code{logical} vectors.} -\item{filtered_data_list}{non-nested (named \code{list}) that contains \code{FilteredData} objects} +\item{datasets}{non-nested (named \code{list}) of \code{FilteredData} objects.} } \value{ -Nothing is returned. +\code{list} containing the snapshot history, where each element is an unlisted \code{teal_slices} object. } \description{ Capture and restore snapshots of the global (app) filter state. @@ -36,7 +37,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 through the filter manager, with the cog icon in the top right corner. +The snapshot manager is accessed with the camera icon in the \code{\link{wunder_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. @@ -76,7 +77,7 @@ when passed to the \code{mapping} argument of \code{\link[=teal_slices]{teal_sli This is substituted as the snapshot's \code{mapping} attribute and the snapshot is added to the snapshot list. To restore app state, a snapshot is retrieved from storage and rebuilt into a \code{teal_slices} object. -Then state of all \code{FilteredData} objects (provided in \code{filtered_data_list}) is cleared +Then state of all \code{FilteredData} objects (provided in \code{datasets}) is cleared and set anew according to the \code{mapping} attribute of the snapshot. The snapshot is then set as the current content of \code{slices_global}. @@ -100,6 +101,14 @@ a \code{teal_slices} object. When a snapshot is restored from file, its \code{ap of the current app state and only if the match is the snapshot admitted to the session. } +\section{Bookmarks}{ + +An \code{onBookmark} callback creates a snapshot of the current filter state. +This is done on the app session, not the module session. +(The snapshot will be retrieved by \code{module_teal} in order to set initial app state in a restored app.) +Then that snapshot, and the previous snapshot history are dumped into the \code{values.rds} file in \verb{}. +} + \author{ Aleksander Chlebowski } diff --git a/man/module_wunder_bar.Rd b/man/module_wunder_bar.Rd new file mode 100644 index 0000000000..6e4f5b8fa1 --- /dev/null +++ b/man/module_wunder_bar.Rd @@ -0,0 +1,49 @@ +% 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_bookmarkable.Rd b/man/modules_bookmarkable.Rd new file mode 100644 index 0000000000..2b33647a2b --- /dev/null +++ b/man/modules_bookmarkable.Rd @@ -0,0 +1,19 @@ +% Generated by roxygen2: do not edit by hand +% Please edit documentation in R/modules.R +\name{modules_bookmarkable} +\alias{modules_bookmarkable} +\title{Retrieve \code{teal_bookmarkable} attribute from \code{teal_modules}} +\usage{ +modules_bookmarkable(modules) +} +\arguments{ +\item{modules}{(\code{teal_modules} or \code{teal_module}) object} +} +\value{ +named list of the same structure as \code{modules} with \code{TRUE} or \code{FALSE} values indicating +whether the module is bookmarkable. +} +\description{ +Retrieve \code{teal_bookmarkable} attribute from \code{teal_modules} +} +\keyword{internal} diff --git a/man/restoreValue.Rd b/man/restoreValue.Rd new file mode 100644 index 0000000000..ebfce5b446 --- /dev/null +++ b/man/restoreValue.Rd @@ -0,0 +1,33 @@ +% Generated by roxygen2: do not edit by hand +% Please edit documentation in R/module_bookmark_manager.R +\name{restoreValue} +\alias{restoreValue} +\title{Restore value from bookmark.} +\usage{ +restoreValue(value, default) +} +\arguments{ +\item{value}{(\code{character(1)}) name of value to restore} + +\item{default}{fallback value} +} +\value{ +In an application restored from a server-side bookmark, +the variable specified by \code{value} from the \code{values} environment. +Otherwise \code{default}. +} +\description{ +Get value from bookmark or return default. +} +\details{ +Bookmarks can store not only inputs but also arbitrary values. +These values are stored by \code{onBookmark} callbacks and restored by \code{onBookmarked} callbacks, +and they are placed in the \code{values} environment in the \code{session$restoreContext} field. +Using \code{teal_data_module} makes it impossible to run the callbacks +because the app becomes ready before modules execute and callbacks are registered. +In those cases the stored values can still be recovered from the \code{session} object directly. + +Note that variable names in the \code{values} environment are prefixed with module name space names, +therefore, when using this function in modules, \code{value} must be run through the name space function. +} +\keyword{internal} diff --git a/tests/testthat/test-filter_manager.R b/tests/testthat/test-filter_manager.R index 9cc0bd8d91..67550e3fa6 100644 --- a/tests/testthat/test-filter_manager.R +++ b/tests/testthat/test-filter_manager.R @@ -41,11 +41,11 @@ testthat::test_that("filter_manager_srv initializes properly processes input arg app = filter_manager_srv, args = list( id = "test", - filtered_data_list = filtered_data_list, + datasets = filtered_data_list, filter = filter_global ), expr = { - testthat::expect_named(filtered_data_list, c("m1", "m2", "m3")) + testthat::expect_named(datasets_flat, c("m1", "m2", "m3")) testthat::expect_identical(slices_global(), filter) } @@ -56,11 +56,11 @@ testthat::test_that("filter_manager_srv initializes properly processes input arg app = filter_manager_srv, args = list( id = "test", - filtered_data_list = filtered_data_list, + datasets = filtered_data_list, filter = filter_modular ), expr = { - testthat::expect_named(filtered_data_list, "global_filters") + testthat::expect_named(datasets_flat, "Global Filters") testthat::expect_identical(slices_global(), filter) } diff --git a/tests/testthat/test-shinytest2-module_bookmark_manager.R b/tests/testthat/test-shinytest2-module_bookmark_manager.R new file mode 100644 index 0000000000..237e53a99b --- /dev/null +++ b/tests/testthat/test-shinytest2-module_bookmark_manager.R @@ -0,0 +1,57 @@ +testthat::test_that("bookmark_manager_button is not rendered by default", { + skip_if_too_deep(5) + app <- TealAppDriver$new( + data = simple_teal_data(), + modules = example_module(label = "Example Module"), + options = options() + ) + on.exit(app$stop()) + testthat::expect_null( + app$get_html(".bookmark_manager_button") + ) +}) + + +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() + ) + on.exit(app$stop()) + testthat::expect_null( + app$get_html(".bookmark_manager_button") + ) +}) + + +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() + ) + on.exit(app$stop()) + testthat::expect_true(!is.null(app$get_html(".bookmark_manager_button"))) +}) + +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() + ) + bookmark_button_id <- app$get_attr(".bookmark_manager_button", "id") + app$click(bookmark_button_id) + + testthat::expect_match( + rvest::html_text(app$get_html_rvest("div[id$=bookmark_modal] pre")), + "_state_id_" + ) +}) diff --git a/tests/testthat/test-shinytest2-wunder_bar.R b/tests/testthat/test-shinytest2-wunder_bar.R new file mode 100644 index 0000000000..dedb7c244d --- /dev/null +++ b/tests/testthat/test-shinytest2-wunder_bar.R @@ -0,0 +1,36 @@ +testthat::test_that("wunder_bar_srv clicking filter icon opens filter-manager modal", { + skip_if_too_deep(5) + app <- TealAppDriver$new( + data = simple_teal_data(), + modules = example_module(label = "Example Module") + ) + + filter_manager_btn_id <- grep( + "filter_manager", + x = app$get_attr(".wunder_bar_button", "id"), + value = TRUE + ) + + testthat::expect_true(is.null(app$get_text(".filter_manager_modal"))) + app$click(filter_manager_btn_id) + testthat::expect_true(!is.null(app$get_text(".filter_manager_modal"))) +}) + + +testthat::test_that("wunder_bar_srv clicking snapshot icon opens snapshot-manager modal", { + skip_if_too_deep(5) + app <- TealAppDriver$new( + data = simple_teal_data(), + modules = example_module(label = "Example Module") + ) + + snapshot_manager_btn_id <- grep( + "snapshot_manager", + x = app$get_attr(".wunder_bar_button", "id"), + value = TRUE + ) + + testthat::expect_true(is.null(app$get_text(".snapshot_manager_modal"))) + app$click(snapshot_manager_btn_id) + testthat::expect_true(!is.null(app$get_text(".snapshot_manager_modal"))) +}) diff --git a/tests/testthat/test-snapshot_manager.R b/tests/testthat/test-snapshot_manager.R index 800ece54b3..4954fe4b41 100644 --- a/tests/testthat/test-snapshot_manager.R +++ b/tests/testthat/test-snapshot_manager.R @@ -14,12 +14,12 @@ testthat::test_that("snapshot manager holds initial state in history", { 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, m2 = fd2, m3 = fd3) + datasets_flat <- list(m1 = fd1, m2 = fd2, m3 = fd3) slices_global <- reactiveVal(shiny::isolate(filter)) mapping_matrix <- reactive({ - module_states <- lapply(filtered_data_list, function(x) x$get_filter_state()) + 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) @@ -32,7 +32,7 @@ testthat::test_that("snapshot manager holds initial state in history", { id = "test", slices_global = slices_global, mapping_matrix = mapping_matrix, - filtered_data_list = filtered_data_list + datasets = datasets_flat ), expr = { testthat::expect_true("Initial application state" %in% names(snapshot_history()))