diff --git a/ddl@main/coverage-report/index.html b/ddl@main/coverage-report/index.html new file mode 100644 index 0000000000..59db899473 --- /dev/null +++ b/ddl@main/coverage-report/index.html @@ -0,0 +1,30241 @@ + + +
+ + + + + + + + + + + + + + + + + + + + + + +1 | ++ |
+ #' Get Client Timezone+ |
+
2 | ++ |
+ #'+ |
+
3 | ++ |
+ #' Local timezone in the browser may differ from the system timezone from the server.+ |
+
4 | ++ |
+ #' This script can be run to register a shiny input which contains information about+ |
+
5 | ++ |
+ #' the timezone in the browser.+ |
+
6 | ++ |
+ #'+ |
+
7 | ++ |
+ #' @param ns (`function`) namespace function passed from the `session` object in the+ |
+
8 | ++ |
+ #' Shiny server. For Shiny modules this will allow for proper name spacing of the+ |
+
9 | ++ |
+ #' registered input.+ |
+
10 | ++ |
+ #'+ |
+
11 | ++ |
+ #' @return (`Shiny`) input variable accessible with `input$tz` which is a (`character`)+ |
+
12 | ++ |
+ #' string containing the timezone of the browser/client.+ |
+
13 | ++ |
+ #' @keywords internal+ |
+
14 | ++ |
+ get_client_timezone <- function(ns) {+ |
+
15 | +16x | +
+ script <- sprintf(+ |
+
16 | +16x | +
+ "Shiny.setInputValue(`%s`, Intl.DateTimeFormat().resolvedOptions().timeZone)",+ |
+
17 | +16x | +
+ ns("timezone")+ |
+
18 | ++ |
+ )+ |
+
19 | +16x | +
+ shinyjs::runjs(script) # function does not return anything+ |
+
20 | +16x | +
+ return(invisible(NULL))+ |
+
21 | ++ |
+ }+ |
+
22 | ++ | + + | +
23 | ++ |
+ #' Resolve the expected bootstrap theme+ |
+
24 | ++ |
+ #' @keywords internal+ |
+
25 | ++ |
+ get_teal_bs_theme <- function() {+ |
+
26 | +26x | +
+ bs_theme <- getOption("teal.bs_theme")+ |
+
27 | +26x | +
+ if (is.null(bs_theme)) {+ |
+
28 | +23x | +
+ NULL+ |
+
29 | +3x | +
+ } else if (!inherits(bs_theme, "bs_theme")) {+ |
+
30 | +2x | +
+ warning("teal.bs_theme has to be of a bslib::bs_theme class, the default shiny bootstrap is used.")+ |
+
31 | +2x | +
+ NULL+ |
+
32 | ++ |
+ } else {+ |
+
33 | +1x | +
+ bs_theme+ |
+
34 | ++ |
+ }+ |
+
35 | ++ |
+ }+ |
+
36 | ++ | + + | +
37 | ++ |
+ include_parent_datanames <- function(dataname, join_keys) {+ |
+
38 | +3x | +
+ parents <- character(0)+ |
+
39 | +3x | +
+ for (i in dataname) {+ |
+
40 | +6x | +
+ while (length(i) > 0) {+ |
+
41 | +6x | +
+ parent_i <- teal.data::parent(join_keys, i)+ |
+
42 | +6x | +
+ parents <- c(parent_i, parents)+ |
+
43 | +6x | +
+ i <- parent_i+ |
+
44 | ++ |
+ }+ |
+
45 | ++ |
+ }+ |
+
46 | ++ | + + | +
47 | +3x | +
+ return(unique(c(parents, dataname)))+ |
+
48 | ++ |
+ }+ |
+
49 | ++ | + + | +
50 | ++ | + + | +
51 | ++ | + + | +
52 | ++ |
+ #' Create a `FilteredData`+ |
+
53 | ++ |
+ #'+ |
+
54 | ++ |
+ #' Create a `FilteredData` object from a `teal_data` object+ |
+
55 | ++ |
+ #' @param x (`teal_data`) object+ |
+
56 | ++ |
+ #' @param datanames (`character`) vector of data set names to include; must be subset of `datanames(x)`+ |
+
57 | ++ |
+ #' @return (`FilteredData`) object+ |
+
58 | ++ |
+ #' @keywords internal+ |
+
59 | ++ |
+ teal_data_to_filtered_data <- function(x, datanames = teal.data::datanames(x)) {+ |
+
60 | +15x | +
+ checkmate::assert_class(x, "teal_data")+ |
+
61 | +15x | +
+ checkmate::assert_character(datanames, min.len = 1L, any.missing = FALSE)+ |
+
62 | +15x | +
+ checkmate::assert_subset(datanames, teal.data::datanames(x))+ |
+
63 | ++ | + + | +
64 | +15x | +
+ ans <- teal.slice::init_filtered_data(+ |
+
65 | +15x | +
+ x = sapply(datanames, function(dn) x[[dn]], simplify = FALSE),+ |
+
66 | +15x | +
+ join_keys = teal.data::join_keys(x)+ |
+
67 | ++ |
+ )+ |
+
68 | ++ |
+ # Piggy-back entire pre-processing code so that filtering code can be appended later.+ |
+
69 | +15x | +
+ attr(ans, "preprocessing_code") <- teal.code::get_code(x)+ |
+
70 | +15x | +
+ ans+ |
+
71 | ++ |
+ }+ |
+
72 | ++ | + + | +
73 | ++ |
+ #' Template Function for `TealReportCard` Creation and Customization+ |
+
74 | ++ |
+ #'+ |
+
75 | ++ |
+ #' This function generates a report card with a title,+ |
+
76 | ++ |
+ #' an optional description, and the option to append the filter state list.+ |
+
77 | ++ |
+ #'+ |
+
78 | ++ |
+ #' @param title (`character(1)`) title of the card (unless overwritten by label)+ |
+
79 | ++ |
+ #' @param label (`character(1)`) label provided by the user when adding the card+ |
+
80 | ++ |
+ #' @param description (`character(1)`) optional additional description+ |
+
81 | ++ |
+ #' @param with_filter (`logical(1)`) flag indicating to add filter state+ |
+
82 | ++ |
+ #' @param filter_panel_api (`FilterPanelAPI`) object with API that allows the generation+ |
+
83 | ++ |
+ #' of the filter state in the report+ |
+
84 | ++ |
+ #'+ |
+
85 | ++ |
+ #' @return (`TealReportCard`) populated with a title, description and filter state+ |
+
86 | ++ |
+ #'+ |
+
87 | ++ |
+ #' @export+ |
+
88 | ++ |
+ report_card_template <- function(title, label, description = NULL, with_filter, filter_panel_api) {+ |
+
89 | +2x | +
+ checkmate::assert_string(title)+ |
+
90 | +2x | +
+ checkmate::assert_string(label)+ |
+
91 | +2x | +
+ checkmate::assert_string(description, null.ok = TRUE)+ |
+
92 | +2x | +
+ checkmate::assert_flag(with_filter)+ |
+
93 | +2x | +
+ checkmate::assert_class(filter_panel_api, classes = "FilterPanelAPI")+ |
+
94 | ++ | + + | +
95 | +2x | +
+ card <- teal::TealReportCard$new()+ |
+
96 | +2x | +
+ title <- if (label == "") title else label+ |
+
97 | +2x | +
+ card$set_name(title)+ |
+
98 | +2x | +
+ card$append_text(title, "header2")+ |
+
99 | +1x | +
+ if (!is.null(description)) card$append_text(description, "header3")+ |
+
100 | +1x | +
+ if (with_filter) card$append_fs(filter_panel_api$get_filter_state())+ |
+
101 | +2x | +
+ card+ |
+
102 | ++ |
+ }+ |
+
103 | ++ |
+ #' Resolve `datanames` for the modules+ |
+
104 | ++ |
+ #'+ |
+
105 | ++ |
+ #' Modifies `module$datanames` to include names of the parent dataset (taken from `join_keys`).+ |
+
106 | ++ |
+ #' When `datanames` is set to `"all"` it is replaced with all available datasets names.+ |
+
107 | ++ |
+ #' @param modules (`teal_modules`) object+ |
+
108 | ++ |
+ #' @param datanames (`character`) names of datasets available in the `data` object+ |
+
109 | ++ |
+ #' @param join_keys (`join_keys`) object+ |
+
110 | ++ |
+ #' @return `teal_modules` with resolved `datanames`+ |
+
111 | ++ |
+ #' @keywords internal+ |
+
112 | ++ |
+ resolve_modules_datanames <- function(modules, datanames, join_keys) {+ |
+
113 | +! | +
+ if (inherits(modules, "teal_modules")) {+ |
+
114 | +! | +
+ modules$children <- sapply(+ |
+
115 | +! | +
+ modules$children,+ |
+
116 | +! | +
+ resolve_modules_datanames,+ |
+
117 | +! | +
+ simplify = FALSE,+ |
+
118 | +! | +
+ datanames = datanames,+ |
+
119 | +! | +
+ join_keys = join_keys+ |
+
120 | ++ |
+ )+ |
+
121 | +! | +
+ modules+ |
+
122 | ++ |
+ } else {+ |
+
123 | +! | +
+ modules$datanames <- if (identical(modules$datanames, "all")) {+ |
+
124 | +! | +
+ datanames+ |
+
125 | +! | +
+ } else if (is.character(modules$datanames)) {+ |
+
126 | +! | +
+ extra_datanames <- setdiff(modules$datanames, datanames)+ |
+
127 | +! | +
+ if (length(extra_datanames)) {+ |
+
128 | +! | +
+ stop(+ |
+
129 | +! | +
+ sprintf(+ |
+
130 | +! | +
+ "Module %s has datanames that are not available in a 'data':\n %s not in %s",+ |
+
131 | +! | +
+ modules$label,+ |
+
132 | +! | +
+ toString(extra_datanames),+ |
+
133 | +! | +
+ toString(datanames)+ |
+
134 | ++ |
+ )+ |
+
135 | ++ |
+ )+ |
+
136 | ++ |
+ }+ |
+
137 | +! | +
+ datanames_adjusted <- intersect(modules$datanames, datanames)+ |
+
138 | +! | +
+ include_parent_datanames(dataname = datanames_adjusted, join_keys = join_keys)+ |
+
139 | ++ |
+ }+ |
+
140 | +! | +
+ modules+ |
+
141 | ++ |
+ }+ |
+
142 | ++ |
+ }+ |
+
143 | ++ | + + | +
144 | ++ |
+ #' Check `datanames` in modules+ |
+
145 | ++ |
+ #'+ |
+
146 | ++ |
+ #' This function ensures specified `datanames` in modules match those in the data object,+ |
+
147 | ++ |
+ #' returning error messages or `TRUE` for successful validation.+ |
+
148 | ++ |
+ #'+ |
+
149 | ++ |
+ #' @param modules (`teal_modules`) object+ |
+
150 | ++ |
+ #' @param datanames (`character`) names of datasets available in the `data` object+ |
+
151 | ++ |
+ #'+ |
+
152 | ++ |
+ #' @return A `character(1)` containing error message or `TRUE` if validation passes.+ |
+
153 | ++ |
+ #' @keywords internal+ |
+
154 | ++ |
+ check_modules_datanames <- function(modules, datanames) {+ |
+
155 | +7x | +
+ checkmate::assert_class(modules, "teal_modules")+ |
+
156 | +7x | +
+ checkmate::assert_character(datanames)+ |
+
157 | ++ | + + | +
158 | +7x | +
+ recursive_check_datanames <- function(modules, datanames) {+ |
+
159 | ++ |
+ # check teal_modules against datanames+ |
+
160 | +14x | +
+ if (inherits(modules, "teal_modules")) {+ |
+
161 | +7x | +
+ sapply(modules$children, function(module) recursive_check_datanames(module, datanames = datanames))+ |
+
162 | ++ |
+ } else {+ |
+
163 | +7x | +
+ extra_datanames <- setdiff(modules$datanames, c("all", datanames))+ |
+
164 | +7x | +
+ if (length(extra_datanames)) {+ |
+
165 | +2x | +
+ sprintf(+ |
+
166 | +2x | +
+ "- Module '%s' uses datanames not available in 'data': (%s) not in (%s)",+ |
+
167 | +2x | +
+ modules$label,+ |
+
168 | +2x | +
+ toString(dQuote(extra_datanames, q = FALSE)),+ |
+
169 | +2x | +
+ toString(dQuote(datanames, q = FALSE))+ |
+
170 | ++ |
+ )+ |
+
171 | ++ |
+ }+ |
+
172 | ++ |
+ }+ |
+
173 | ++ |
+ }+ |
+
174 | +7x | +
+ check_datanames <- unlist(recursive_check_datanames(modules, datanames))+ |
+
175 | +7x | +
+ if (length(check_datanames)) {+ |
+
176 | +2x | +
+ paste(check_datanames, collapse = "\n")+ |
+
177 | ++ |
+ } else {+ |
+
178 | +5x | +
+ TRUE+ |
+
179 | ++ |
+ }+ |
+
180 | ++ |
+ }+ |
+
181 | ++ | + + | +
182 | ++ |
+ #' Check `datanames` in filters+ |
+
183 | ++ |
+ #'+ |
+
184 | ++ |
+ #' This function checks whether `datanames` in filters correspond to those in `data`,+ |
+
185 | ++ |
+ #' returning character vector with error messages or TRUE if all checks pass.+ |
+
186 | ++ |
+ #'+ |
+
187 | ++ |
+ #' @param filters (`teal_slices`) object+ |
+
188 | ++ |
+ #' @param datanames (`character`) names of datasets available in the `data` object+ |
+
189 | ++ |
+ #'+ |
+
190 | ++ |
+ #' @return A `character(1)` containing error message or TRUE if validation passes.+ |
+
191 | ++ |
+ #' @keywords internal+ |
+
192 | ++ |
+ check_filter_datanames <- function(filters, datanames) {+ |
+
193 | +5x | +
+ checkmate::assert_class(filters, "teal_slices")+ |
+
194 | +5x | +
+ checkmate::assert_character(datanames)+ |
+
195 | ++ | + + | +
196 | ++ |
+ # check teal_slices against datanames+ |
+
197 | +5x | +
+ out <- unlist(sapply(+ |
+
198 | +5x | +
+ filters, function(filter) {+ |
+
199 | +2x | +
+ dataname <- shiny::isolate(filter$dataname)+ |
+
200 | +2x | +
+ if (!dataname %in% datanames) {+ |
+
201 | +2x | +
+ sprintf(+ |
+
202 | +2x | +
+ "- Filter '%s' refers to dataname not available in 'data':\n %s not in (%s)",+ |
+
203 | +2x | +
+ shiny::isolate(filter$id),+ |
+
204 | +2x | +
+ dQuote(dataname, q = FALSE),+ |
+
205 | +2x | +
+ toString(dQuote(datanames, q = FALSE))+ |
+
206 | ++ |
+ )+ |
+
207 | ++ |
+ }+ |
+
208 | ++ |
+ }+ |
+
209 | ++ |
+ ))+ |
+
210 | ++ | + + | +
211 | ++ | + + | +
212 | +5x | +
+ if (length(out)) {+ |
+
213 | +2x | +
+ paste(out, collapse = "\n")+ |
+
214 | ++ |
+ } else {+ |
+
215 | +3x | +
+ TRUE+ |
+
216 | ++ |
+ }+ |
+
217 | ++ |
+ }+ |
+
1 | ++ |
+ #' Filter settings for teal applications+ |
+
2 | ++ |
+ #'+ |
+
3 | ++ |
+ #' Specify initial filter states and filtering settings for a `teal` app.+ |
+
4 | ++ |
+ #'+ |
+
5 | ++ |
+ #' Produces a `teal_slices` object.+ |
+
6 | ++ |
+ #' The `teal_slice` components will specify filter states that will be active when the app starts.+ |
+
7 | ++ |
+ #' Attributes (created with the named arguments) will configure the way the app applies filters.+ |
+
8 | ++ |
+ #' See argument descriptions for details.+ |
+
9 | ++ |
+ #'+ |
+
10 | ++ |
+ #' @inheritParams teal.slice::teal_slices+ |
+
11 | ++ |
+ #'+ |
+
12 | ++ |
+ #' @param module_specific optional (`logical(1)`)\cr+ |
+
13 | ++ |
+ #' - `FALSE` (default) when one filter panel applied to all modules.+ |
+
14 | ++ |
+ #' All filters will be shared by all modules.+ |
+
15 | ++ |
+ #' - `TRUE` when filter panel module-specific.+ |
+
16 | ++ |
+ #' Modules can have different set of filters specified - see `mapping` argument.+ |
+
17 | ++ |
+ #' @param mapping `r lifecycle::badge("experimental")` _This is a new feature. Do kindly share your opinions.\cr_+ |
+
18 | ++ |
+ #' (`named list`)\cr+ |
+
19 | ++ |
+ #' Specifies which filters will be active in which modules on app start.+ |
+
20 | ++ |
+ #' Elements should contain character vector of `teal_slice` `id`s (see [teal.slice::teal_slice()]).+ |
+
21 | ++ |
+ #' Names of the list should correspond to `teal_module` `label` set in [module()] function.+ |
+
22 | ++ |
+ #' `id`s listed under `"global_filters` will be active in all modules.+ |
+
23 | ++ |
+ #' If missing, all filters will be applied to all modules.+ |
+
24 | ++ |
+ #' If empty list, all filters will be available to all modules but will start inactive.+ |
+
25 | ++ |
+ #' If `module_specific` is `FALSE`, only `global_filters` will be active on start.+ |
+
26 | ++ |
+ #' @param app_id (`character(1)`)\cr+ |
+
27 | ++ |
+ #' For internal use only, do not set manually.+ |
+
28 | ++ |
+ #' Added by `init` so that a `teal_slices` can be matched to the app in which it was used.+ |
+
29 | ++ |
+ #' Used for verifying snapshots uploaded from file. See `snapshot`.+ |
+
30 | ++ |
+ #'+ |
+
31 | ++ |
+ #' @param x (`list`) of lists to convert to `teal_slices`+ |
+
32 | ++ |
+ #'+ |
+
33 | ++ |
+ #' @return+ |
+
34 | ++ |
+ #' A `teal_slices` object.+ |
+
35 | ++ |
+ #'+ |
+
36 | ++ |
+ #' @seealso [`teal.slice::teal_slices`], [`teal.slice::teal_slice`], [`slices_store`]+ |
+
37 | ++ |
+ #'+ |
+
38 | ++ |
+ #' @examples+ |
+
39 | ++ |
+ #' filter <- teal_slices(+ |
+
40 | ++ |
+ #' teal.slice::teal_slice(dataname = "iris", varname = "Species", id = "species"),+ |
+
41 | ++ |
+ #' teal.slice::teal_slice(dataname = "iris", varname = "Sepal.Length", id = "sepal_length"),+ |
+
42 | ++ |
+ #' teal.slice::teal_slice(+ |
+
43 | ++ |
+ #' dataname = "iris", id = "long_petals", title = "Long petals", expr = "Petal.Length > 5"+ |
+
44 | ++ |
+ #' ),+ |
+
45 | ++ |
+ #' teal.slice::teal_slice(dataname = "mtcars", varname = "mpg", id = "mtcars_mpg"),+ |
+
46 | ++ |
+ #' mapping = list(+ |
+
47 | ++ |
+ #' module1 = c("species", "sepal_length"),+ |
+
48 | ++ |
+ #' module2 = c("mtcars_mpg"),+ |
+
49 | ++ |
+ #' global_filters = "long_petals"+ |
+
50 | ++ |
+ #' )+ |
+
51 | ++ |
+ #' )+ |
+
52 | ++ |
+ #'+ |
+
53 | ++ |
+ #' app <- teal::init(+ |
+
54 | ++ |
+ #' modules = list(+ |
+
55 | ++ |
+ #' module("module1"),+ |
+
56 | ++ |
+ #' module("module2")+ |
+
57 | ++ |
+ #' ),+ |
+
58 | ++ |
+ #' data = list(iris, mtcars),+ |
+
59 | ++ |
+ #' filter = filter+ |
+
60 | ++ |
+ #' )+ |
+
61 | ++ |
+ #'+ |
+
62 | ++ |
+ #' if (interactive()) {+ |
+
63 | ++ |
+ #' shinyApp(app$ui, app$server)+ |
+
64 | ++ |
+ #' }+ |
+
65 | ++ |
+ #'+ |
+
66 | ++ |
+ #' @export+ |
+
67 | ++ |
+ teal_slices <- function(...,+ |
+
68 | ++ |
+ exclude_varnames = NULL,+ |
+
69 | ++ |
+ include_varnames = NULL,+ |
+
70 | ++ |
+ count_type = NULL,+ |
+
71 | ++ |
+ allow_add = TRUE,+ |
+
72 | ++ |
+ module_specific = FALSE,+ |
+
73 | ++ |
+ mapping,+ |
+
74 | ++ |
+ app_id = NULL) {+ |
+
75 | +90x | +
+ shiny::isolate({+ |
+
76 | +90x | +
+ checkmate::assert_flag(allow_add)+ |
+
77 | +90x | +
+ checkmate::assert_flag(module_specific)+ |
+
78 | +44x | +
+ if (!missing(mapping)) checkmate::assert_list(mapping, types = c("character", "NULL"), names = "named")+ |
+
79 | +87x | +
+ checkmate::assert_string(app_id, null.ok = TRUE)+ |
+
80 | ++ | + + | +
81 | +87x | +
+ slices <- list(...)+ |
+
82 | +87x | +
+ all_slice_id <- vapply(slices, `[[`, character(1L), "id")+ |
+
83 | ++ | + + | +
84 | +87x | +
+ if (missing(mapping)) {+ |
+
85 | +46x | +
+ mapping <- list(global_filters = all_slice_id)+ |
+
86 | ++ |
+ }+ |
+
87 | +87x | +
+ if (!module_specific) {+ |
+
88 | +83x | +
+ mapping[setdiff(names(mapping), "global_filters")] <- NULL+ |
+
89 | ++ |
+ }+ |
+
90 | ++ | + + | +
91 | +87x | +
+ failed_slice_id <- setdiff(unlist(mapping), all_slice_id)+ |
+
92 | +87x | +
+ if (length(failed_slice_id)) {+ |
+
93 | +1x | +
+ stop(sprintf(+ |
+
94 | +1x | +
+ "Filters in mapping don't match any available filter.\n %s not in %s",+ |
+
95 | +1x | +
+ toString(failed_slice_id),+ |
+
96 | +1x | +
+ toString(all_slice_id)+ |
+
97 | ++ |
+ ))+ |
+
98 | ++ |
+ }+ |
+
99 | ++ | + + | +
100 | +86x | +
+ tss <- teal.slice::teal_slices(+ |
+
101 | ++ |
+ ...,+ |
+
102 | +86x | +
+ exclude_varnames = exclude_varnames,+ |
+
103 | +86x | +
+ include_varnames = include_varnames,+ |
+
104 | +86x | +
+ count_type = count_type,+ |
+
105 | +86x | +
+ allow_add = allow_add+ |
+
106 | ++ |
+ )+ |
+
107 | +86x | +
+ attr(tss, "mapping") <- mapping+ |
+
108 | +86x | +
+ attr(tss, "module_specific") <- module_specific+ |
+
109 | +86x | +
+ attr(tss, "app_id") <- app_id+ |
+
110 | +86x | +
+ class(tss) <- c("modules_teal_slices", class(tss))+ |
+
111 | +86x | +
+ tss+ |
+
112 | ++ |
+ })+ |
+
113 | ++ |
+ }+ |
+
114 | ++ | + + | +
115 | ++ | + + | +
116 | ++ |
+ #' @rdname teal_slices+ |
+
117 | ++ |
+ #' @export+ |
+
118 | ++ |
+ #' @keywords internal+ |
+
119 | ++ |
+ #'+ |
+
120 | ++ |
+ as.teal_slices <- function(x) { # nolint+ |
+
121 | +25x | +
+ checkmate::assert_list(x)+ |
+
122 | +25x | +
+ lapply(x, checkmate::assert_list, names = "named", .var.name = "list element")+ |
+
123 | ++ | + + | +
124 | +25x | +
+ attrs <- attributes(unclass(x))+ |
+
125 | +25x | +
+ ans <- lapply(x, function(x) if (is.teal_slice(x)) x else as.teal_slice(x))+ |
+
126 | +25x | +
+ do.call(teal_slices, c(ans, attrs))+ |
+
127 | ++ |
+ }+ |
+
128 | ++ | + + | +
129 | ++ | + + | +
130 | ++ |
+ #' @rdname teal_slices+ |
+
131 | ++ |
+ #' @export+ |
+
132 | ++ |
+ #' @keywords internal+ |
+
133 | ++ |
+ #'+ |
+
134 | ++ |
+ c.teal_slices <- function(...) {+ |
+
135 | +! | +
+ x <- list(...)+ |
+
136 | +! | +
+ checkmate::assert_true(all(vapply(x, is.teal_slices, logical(1L))), .var.name = "all arguments are teal_slices")+ |
+
137 | ++ | + + | +
138 | +! | +
+ all_attributes <- lapply(x, attributes)+ |
+
139 | +! | +
+ all_attributes <- coalesce_r(all_attributes)+ |
+
140 | +! | +
+ all_attributes <- all_attributes[names(all_attributes) != "class"]+ |
+
141 | ++ | + + | +
142 | +! | +
+ do.call(+ |
+
143 | +! | +
+ teal_slices,+ |
+
144 | +! | +
+ c(+ |
+
145 | +! | +
+ unique(unlist(x, recursive = FALSE)),+ |
+
146 | +! | +
+ all_attributes+ |
+
147 | ++ |
+ )+ |
+
148 | ++ |
+ )+ |
+
149 | ++ |
+ }+ |
+
150 | ++ | + + | +
151 | ++ | + + | +
152 | ++ |
+ #' Deep copy `teal_slices`+ |
+
153 | ++ |
+ #'+ |
+
154 | ++ |
+ #' it's important to create a new copy of `teal_slices` when+ |
+
155 | ++ |
+ #' starting a new `shiny` session. Otherwise, object will be shared+ |
+
156 | ++ |
+ #' by multiple users as it is created in global environment before+ |
+
157 | ++ |
+ #' `shiny` session starts.+ |
+
158 | ++ |
+ #' @param filter (`teal_slices`)+ |
+
159 | ++ |
+ #' @return `teal_slices`+ |
+
160 | ++ |
+ #' @keywords internal+ |
+
161 | ++ |
+ deep_copy_filter <- function(filter) {+ |
+
162 | +1x | +
+ checkmate::assert_class(filter, "teal_slices")+ |
+
163 | +1x | +
+ shiny::isolate({+ |
+
164 | +1x | +
+ filter_copy <- lapply(filter, function(slice) {+ |
+
165 | +2x | +
+ teal.slice::as.teal_slice(as.list(slice))+ |
+
166 | ++ |
+ })+ |
+
167 | +1x | +
+ attributes(filter_copy) <- attributes(filter)+ |
+
168 | +1x | +
+ filter_copy+ |
+
169 | ++ |
+ })+ |
+
170 | ++ |
+ }+ |
+
1 | ++ |
+ #' Filter state snapshot management.+ |
+
2 | ++ |
+ #'+ |
+
3 | ++ |
+ #' Capture and restore snapshots of the global (app) filter state.+ |
+
4 | ++ |
+ #'+ |
+
5 | ++ |
+ #' This module introduces snapshots: stored descriptions of the filter state of the entire application.+ |
+
6 | ++ |
+ #' Snapshots allow the user to save the current filter state of the application for later use in the session,+ |
+
7 | ++ |
+ #' as well as to save it to file in order to share it with an app developer or other users,+ |
+
8 | ++ |
+ #' who in turn can upload it to their own session.+ |
+
9 | ++ |
+ #'+ |
+
10 | ++ |
+ #' The snapshot manager is accessed through the filter manager, with the cog icon in the top right corner.+ |
+
11 | ++ |
+ #' At the beginning of a session it presents three icons: a camera, an upload, and an circular arrow.+ |
+
12 | ++ |
+ #' Clicking the camera captures a snapshot, clicking the upload adds a snapshot from a file+ |
+
13 | ++ |
+ #' and applies the filter states therein, and clicking the arrow resets initial application state.+ |
+
14 | ++ |
+ #' As snapshots are added, they will show up as rows in a table and each will have a select button and a save button.+ |
+
15 | ++ |
+ #'+ |
+
16 | ++ |
+ #' @section Server logic:+ |
+
17 | ++ |
+ #' Snapshots are basically `teal_slices` objects, however, since each module is served by a separate instance+ |
+
18 | ++ |
+ #' of `FilteredData` and these objects require shared state, `teal_slice` is a `reactiveVal` so `teal_slices`+ |
+
19 | ++ |
+ #' cannot be stored as is. Therefore, `teal_slices` are reversibly converted to a list of lists representation+ |
+
20 | ++ |
+ #' (attributes are maintained).+ |
+
21 | ++ |
+ #'+ |
+
22 | ++ |
+ #' Snapshots are stored in a `reactiveVal` as a named list.+ |
+
23 | ++ |
+ #' The first snapshot is the initial state of the application and the user can add a snapshot whenever they see fit.+ |
+
24 | ++ |
+ #'+ |
+
25 | ++ |
+ #' For every snapshot except the initial one, a piece of UI is generated that contains+ |
+
26 | ++ |
+ #' the snapshot name, a select button to restore that snapshot, and a save button to save it to a file.+ |
+
27 | ++ |
+ #' The initial snapshot is restored by a separate "reset" button.+ |
+
28 | ++ |
+ #' It cannot be saved directly but a user is welcome to capture the initial state as a snapshot and save that.+ |
+
29 | ++ |
+ #'+ |
+
30 | ++ |
+ #' @section Snapshot mechanics:+ |
+
31 | ++ |
+ #' When a snapshot is captured, the user is prompted to name it.+ |
+
32 | ++ |
+ #' Names are displayed as is but since they are used to create button ids,+ |
+
33 | ++ |
+ #' under the hood they are converted to syntactically valid strings.+ |
+
34 | ++ |
+ #' New snapshot names are validated so that their valid versions are unique.+ |
+
35 | ++ |
+ #' Leading and trailing white space is trimmed.+ |
+
36 | ++ |
+ #'+ |
+
37 | ++ |
+ #' The module can read the global state of the application from `slices_global` and `mapping_matrix`.+ |
+
38 | ++ |
+ #' The former provides a list of all existing `teal_slice`s and the latter says which slice is active in which module.+ |
+
39 | ++ |
+ #' Once a name has been accepted, `slices_global` is converted to a list of lists - a snapshot.+ |
+
40 | ++ |
+ #' The snapshot contains the `mapping` attribute of the initial application state+ |
+
41 | ++ |
+ #' (or one that has been restored), which may not reflect the current one,+ |
+
42 | ++ |
+ #' so `mapping_matrix` is transformed to obtain the current mapping, i.e. a list that,+ |
+
43 | ++ |
+ #' when passed to the `mapping` argument of [`teal::teal_slices`], would result in the current mapping.+ |
+
44 | ++ |
+ #' This is substituted as the snapshot's `mapping` attribute and the snapshot is added to the snapshot list.+ |
+
45 | ++ |
+ #'+ |
+
46 | ++ |
+ #' To restore app state, a snapshot is retrieved from storage and rebuilt into a `teal_slices` object.+ |
+
47 | ++ |
+ #' Then state of all `FilteredData` objects (provided in `filtered_data_list`) is cleared+ |
+
48 | ++ |
+ #' and set anew according to the `mapping` attribute of the snapshot.+ |
+
49 | ++ |
+ #' The snapshot is then set as the current content of `slices_global`.+ |
+
50 | ++ |
+ #'+ |
+
51 | ++ |
+ #' To save a snapshot, the snapshot is retrieved and reassembled just like for restoring,+ |
+
52 | ++ |
+ #' and then saved to file with [`slices_store`].+ |
+
53 | ++ |
+ #'+ |
+
54 | ++ |
+ #' When a snapshot is uploaded, it will first be added to storage just like a newly created one,+ |
+
55 | ++ |
+ #' and then used to restore app state much like a snapshot taken from storage.+ |
+
56 | ++ |
+ #' Upon clicking the upload icon the user will be prompted for a file to upload+ |
+
57 | ++ |
+ #' and may choose to name the new snapshot. The name defaults to the name of the file (the extension is dropped)+ |
+
58 | ++ |
+ #' and normal naming rules apply. Loading the file yields a `teal_slices` object,+ |
+
59 | ++ |
+ #' which is disassembled for storage and used directly for restoring app state.+ |
+
60 | ++ |
+ #'+ |
+
61 | ++ |
+ #' @section Transferring snapshots:+ |
+
62 | ++ |
+ #' Snapshots uploaded from disk should only be used in the same application they come from,+ |
+
63 | ++ |
+ #' _i.e._ an application that uses the same data and the same modules.+ |
+
64 | ++ |
+ #' To ensure this is the case, `init` stamps `teal_slices` with an app id that is stored in the `app_id` attribute of+ |
+
65 | ++ |
+ #' a `teal_slices` object. When a snapshot is restored from file, its `app_id` is compared to that+ |
+
66 | ++ |
+ #' of the current app state and only if the match is the snapshot admitted to the session.+ |
+
67 | ++ |
+ #'+ |
+
68 | ++ |
+ #' @param id (`character(1)`) `shiny` module id+ |
+
69 | ++ |
+ #' @param slices_global (`reactiveVal`) that contains a `teal_slices` object+ |
+
70 | ++ |
+ #' containing all `teal_slice`s existing in the app, both active and inactive+ |
+
71 | ++ |
+ #' @param mapping_matrix (`reactive`) that contains a `data.frame` representation+ |
+
72 | ++ |
+ #' of the mapping of filter state ids (rows) to modules labels (columns);+ |
+
73 | ++ |
+ #' all columns are `logical` vectors+ |
+
74 | ++ |
+ #' @param filtered_data_list non-nested (`named list`) that contains `FilteredData` objects+ |
+
75 | ++ |
+ #'+ |
+
76 | ++ |
+ #' @return Nothing is returned.+ |
+
77 | ++ |
+ #'+ |
+
78 | ++ |
+ #' @name snapshot_manager_module+ |
+
79 | ++ |
+ #' @aliases snapshot snapshot_manager+ |
+
80 | ++ |
+ #'+ |
+
81 | ++ |
+ #' @author Aleksander Chlebowski+ |
+
82 | ++ |
+ #'+ |
+
83 | ++ |
+ #' @rdname snapshot_manager_module+ |
+
84 | ++ |
+ #' @keywords internal+ |
+
85 | ++ |
+ #'+ |
+
86 | ++ |
+ snapshot_manager_ui <- function(id) {+ |
+
87 | +! | +
+ ns <- NS(id)+ |
+
88 | +! | +
+ div(+ |
+
89 | +! | +
+ class = "snapshot_manager_content",+ |
+
90 | +! | +
+ div(+ |
+
91 | +! | +
+ class = "snapshot_table_row",+ |
+
92 | +! | +
+ span(tags$b("Snapshot manager")),+ |
+
93 | +! | +
+ actionLink(ns("snapshot_add"), label = NULL, icon = icon("camera"), title = "add snapshot"),+ |
+
94 | +! | +
+ actionLink(ns("snapshot_load"), label = NULL, icon = icon("upload"), title = "upload snapshot"),+ |
+
95 | +! | +
+ actionLink(ns("snapshot_reset"), label = NULL, icon = icon("undo"), title = "reset initial state"),+ |
+
96 | +! | +
+ NULL+ |
+
97 | ++ |
+ ),+ |
+
98 | +! | +
+ uiOutput(ns("snapshot_list"))+ |
+
99 | ++ |
+ )+ |
+
100 | ++ |
+ }+ |
+
101 | ++ | + + | +
102 | ++ |
+ #' @rdname snapshot_manager_module+ |
+
103 | ++ |
+ #' @keywords internal+ |
+
104 | ++ |
+ #'+ |
+
105 | ++ |
+ snapshot_manager_srv <- function(id, slices_global, mapping_matrix, filtered_data_list) {+ |
+
106 | +7x | +
+ checkmate::assert_character(id)+ |
+
107 | +7x | +
+ checkmate::assert_true(is.reactive(slices_global))+ |
+
108 | +7x | +
+ checkmate::assert_class(isolate(slices_global()), "teal_slices")+ |
+
109 | +7x | +
+ checkmate::assert_true(is.reactive(mapping_matrix))+ |
+
110 | +7x | +
+ checkmate::assert_data_frame(isolate(mapping_matrix()), null.ok = TRUE)+ |
+
111 | +7x | +
+ checkmate::assert_list(filtered_data_list, types = "FilteredData", any.missing = FALSE, names = "named")+ |
+
112 | ++ | + + | +
113 | +7x | +
+ moduleServer(id, function(input, output, session) {+ |
+
114 | +7x | +
+ ns <- session$ns+ |
+
115 | ++ | + + | +
116 | ++ |
+ # Store global filter states ----+ |
+
117 | +7x | +
+ filter <- isolate(slices_global())+ |
+
118 | +7x | +
+ snapshot_history <- reactiveVal({+ |
+
119 | +7x | +
+ list(+ |
+
120 | +7x | +
+ "Initial application state" = as.list(filter, recursive = TRUE)+ |
+
121 | ++ |
+ )+ |
+
122 | ++ |
+ })+ |
+
123 | ++ | + + | +
124 | ++ |
+ # Snapshot current application state ----+ |
+
125 | ++ |
+ # Name snaphsot.+ |
+
126 | +7x | +
+ observeEvent(input$snapshot_add, {+ |
+
127 | +! | +
+ showModal(+ |
+
128 | +! | +
+ modalDialog(+ |
+
129 | +! | +
+ textInput(ns("snapshot_name"), "Name the snapshot", width = "100%", placeholder = "Meaningful, unique name"),+ |
+
130 | +! | +
+ footer = tagList(+ |
+
131 | +! | +
+ actionButton(ns("snapshot_name_accept"), "Accept", icon = icon("thumbs-up")),+ |
+
132 | +! | +
+ modalButton(label = "Cancel", icon = icon("thumbs-down"))+ |
+
133 | ++ |
+ ),+ |
+
134 | +! | +
+ size = "s"+ |
+
135 | ++ |
+ )+ |
+
136 | ++ |
+ )+ |
+
137 | ++ |
+ })+ |
+
138 | ++ |
+ # Store snaphsot.+ |
+
139 | +7x | +
+ observeEvent(input$snapshot_name_accept, {+ |
+
140 | +! | +
+ snapshot_name <- trimws(input$snapshot_name)+ |
+
141 | +! | +
+ if (identical(snapshot_name, "")) {+ |
+
142 | +! | +
+ showNotification(+ |
+
143 | +! | +
+ "Please name the snapshot.",+ |
+
144 | +! | +
+ type = "message"+ |
+
145 | ++ |
+ )+ |
+
146 | +! | +
+ updateTextInput(inputId = "snapshot_name", value = "", placeholder = "Meaningful, unique name")+ |
+
147 | +! | +
+ } else if (is.element(make.names(snapshot_name), make.names(names(snapshot_history())))) {+ |
+
148 | +! | +
+ showNotification(+ |
+
149 | +! | +
+ "This name is in conflict with other snapshot names. Please choose a different one.",+ |
+
150 | +! | +
+ type = "message"+ |
+
151 | ++ |
+ )+ |
+
152 | +! | +
+ updateTextInput(inputId = "snapshot_name", value = "", placeholder = "Meaningful, unique name")+ |
+
153 | ++ |
+ } else {+ |
+
154 | +! | +
+ snapshot <- as.list(slices_global(), recursive = TRUE)+ |
+
155 | +! | +
+ attr(snapshot, "mapping") <- matrix_to_mapping(mapping_matrix())+ |
+
156 | +! | +
+ snapshot_update <- c(snapshot_history(), list(snapshot))+ |
+
157 | +! | +
+ names(snapshot_update)[length(snapshot_update)] <- snapshot_name+ |
+
158 | +! | +
+ snapshot_history(snapshot_update)+ |
+
159 | +! | +
+ removeModal()+ |
+
160 | ++ |
+ # Reopen filter manager modal by clicking button in the main application.+ |
+
161 | +! | +
+ shinyjs::click(id = "teal-main_ui-filter_manager-show", asis = TRUE)+ |
+
162 | ++ |
+ }+ |
+
163 | ++ |
+ })+ |
+
164 | ++ | + + | +
165 | ++ |
+ # Upload a snapshot file ----+ |
+
166 | ++ |
+ # Select file.+ |
+
167 | +7x | +
+ observeEvent(input$snapshot_load, {+ |
+
168 | +! | +
+ showModal(+ |
+
169 | +! | +
+ modalDialog(+ |
+
170 | +! | +
+ fileInput(ns("snapshot_file"), "Choose snapshot file", accept = ".json", width = "100%"),+ |
+
171 | +! | +
+ textInput(+ |
+
172 | +! | +
+ ns("snapshot_name"),+ |
+
173 | +! | +
+ "Name the snapshot (optional)",+ |
+
174 | +! | +
+ width = "100%",+ |
+
175 | +! | +
+ placeholder = "Meaningful, unique name"+ |
+
176 | ++ |
+ ),+ |
+
177 | +! | +
+ footer = tagList(+ |
+
178 | +! | +
+ actionButton(ns("snaphot_file_accept"), "Accept", icon = icon("thumbs-up")),+ |
+
179 | +! | +
+ modalButton(label = "Cancel", icon = icon("thumbs-down"))+ |
+
180 | ++ |
+ )+ |
+
181 | ++ |
+ )+ |
+
182 | ++ |
+ )+ |
+
183 | ++ |
+ })+ |
+
184 | ++ |
+ # Store new snapshot to list and restore filter states.+ |
+
185 | +7x | +
+ observeEvent(input$snaphot_file_accept, {+ |
+
186 | +! | +
+ snapshot_name <- trimws(input$snapshot_name)+ |
+
187 | +! | +
+ if (identical(snapshot_name, "")) {+ |
+
188 | +! | +
+ snapshot_name <- tools::file_path_sans_ext(input$snapshot_file$name)+ |
+
189 | ++ |
+ }+ |
+
190 | +! | +
+ if (is.element(make.names(snapshot_name), make.names(names(snapshot_history())))) {+ |
+
191 | +! | +
+ showNotification(+ |
+
192 | +! | +
+ "This name is in conflict with other snapshot names. Please choose a different one.",+ |
+
193 | +! | +
+ type = "message"+ |
+
194 | ++ |
+ )+ |
+
195 | +! | +
+ updateTextInput(inputId = "snapshot_name", value = "", placeholder = "Meaningful, unique name")+ |
+
196 | ++ |
+ } else {+ |
+
197 | ++ |
+ # Restore snapshot and verify app compatibility.+ |
+
198 | +! | +
+ snapshot_state <- try(slices_restore(input$snapshot_file$datapath))+ |
+
199 | +! | +
+ if (!inherits(snapshot_state, "modules_teal_slices")) {+ |
+
200 | +! | +
+ showNotification(+ |
+
201 | +! | +
+ "File appears to be corrupt.",+ |
+
202 | +! | +
+ type = "error"+ |
+
203 | ++ |
+ )+ |
+
204 | +! | +
+ } else if (!identical(attr(snapshot_state, "app_id"), attr(slices_global(), "app_id"))) {+ |
+
205 | +! | +
+ showNotification(+ |
+
206 | +! | +
+ "This snapshot file is not compatible with the app and cannot be loaded.",+ |
+
207 | +! | +
+ type = "warning"+ |
+
208 | ++ |
+ )+ |
+
209 | ++ |
+ } else {+ |
+
210 | ++ |
+ # Add to snapshot history.+ |
+
211 | +! | +
+ snapshot <- as.list(snapshot_state, recursive = TRUE)+ |
+
212 | +! | +
+ snapshot_update <- c(snapshot_history(), list(snapshot))+ |
+
213 | +! | +
+ names(snapshot_update)[length(snapshot_update)] <- snapshot_name+ |
+
214 | +! | +
+ snapshot_history(snapshot_update)+ |
+
215 | ++ |
+ ### Begin simplified restore procedure. ###+ |
+
216 | +! | +
+ mapping_unfolded <- unfold_mapping(attr(snapshot_state, "mapping"), names(filtered_data_list))+ |
+
217 | +! | +
+ mapply(+ |
+
218 | +! | +
+ function(filtered_data, filter_ids) {+ |
+
219 | +! | +
+ filtered_data$clear_filter_states(force = TRUE)+ |
+
220 | +! | +
+ slices <- Filter(function(x) x$id %in% filter_ids, snapshot_state)+ |
+
221 | +! | +
+ filtered_data$set_filter_state(slices)+ |
+
222 | ++ |
+ },+ |
+
223 | +! | +
+ filtered_data = filtered_data_list,+ |
+
224 | +! | +
+ filter_ids = mapping_unfolded+ |
+
225 | ++ |
+ )+ |
+
226 | +! | +
+ slices_global(snapshot_state)+ |
+
227 | +! | +
+ removeModal()+ |
+
228 | ++ |
+ ### End simplified restore procedure. ###+ |
+
229 | ++ |
+ }+ |
+
230 | ++ |
+ }+ |
+
231 | ++ |
+ })+ |
+
232 | ++ |
+ # Apply newly added snapshot.+ |
+
233 | ++ | + + | +
234 | ++ |
+ # Restore initial state ----+ |
+
235 | +7x | +
+ observeEvent(input$snapshot_reset, {+ |
+
236 | +! | +
+ s <- "Initial application state"+ |
+
237 | ++ |
+ ### Begin restore procedure. ###+ |
+
238 | +! | +
+ snapshot <- snapshot_history()[[s]]+ |
+
239 | +! | +
+ snapshot_state <- as.teal_slices(snapshot)+ |
+
240 | +! | +
+ mapping_unfolded <- unfold_mapping(attr(snapshot_state, "mapping"), names(filtered_data_list))+ |
+
241 | +! | +
+ mapply(+ |
+
242 | +! | +
+ function(filtered_data, filter_ids) {+ |
+
243 | +! | +
+ filtered_data$clear_filter_states(force = TRUE)+ |
+
244 | +! | +
+ slices <- Filter(function(x) x$id %in% filter_ids, snapshot_state)+ |
+
245 | +! | +
+ filtered_data$set_filter_state(slices)+ |
+
246 | ++ |
+ },+ |
+
247 | +! | +
+ filtered_data = filtered_data_list,+ |
+
248 | +! | +
+ filter_ids = mapping_unfolded+ |
+
249 | ++ |
+ )+ |
+
250 | +! | +
+ slices_global(snapshot_state)+ |
+
251 | +! | +
+ removeModal()+ |
+
252 | ++ |
+ ### End restore procedure. ###+ |
+
253 | ++ |
+ })+ |
+
254 | ++ | + + | +
255 | ++ |
+ # Build snapshot table ----+ |
+
256 | ++ |
+ # Create UI elements and server logic for the snapshot table.+ |
+
257 | ++ |
+ # Observers must be tracked to avoid duplication and excess reactivity.+ |
+
258 | ++ |
+ # Remaining elements are tracked likewise for consistency and a slight speed margin.+ |
+
259 | +7x | +
+ observers <- reactiveValues()+ |
+
260 | +7x | +
+ handlers <- reactiveValues()+ |
+
261 | +7x | +
+ divs <- reactiveValues()+ |
+
262 | ++ | + + | +
263 | +7x | +
+ observeEvent(snapshot_history(), {+ |
+
264 | +3x | +
+ lapply(names(snapshot_history())[-1L], function(s) {+ |
+
265 | +! | +
+ id_pickme <- sprintf("pickme_%s", make.names(s))+ |
+
266 | +! | +
+ id_saveme <- sprintf("saveme_%s", make.names(s))+ |
+
267 | +! | +
+ id_rowme <- sprintf("rowme_%s", make.names(s))+ |
+
268 | ++ | + + | +
269 | ++ |
+ # Observer for restoring snapshot.+ |
+
270 | +! | +
+ if (!is.element(id_pickme, names(observers))) {+ |
+
271 | +! | +
+ observers[[id_pickme]] <- observeEvent(input[[id_pickme]], {+ |
+
272 | ++ |
+ ### Begin restore procedure. ###+ |
+
273 | +! | +
+ snapshot <- snapshot_history()[[s]]+ |
+
274 | +! | +
+ snapshot_state <- as.teal_slices(snapshot)+ |
+
275 | +! | +
+ mapping_unfolded <- unfold_mapping(attr(snapshot_state, "mapping"), names(filtered_data_list))+ |
+
276 | +! | +
+ mapply(+ |
+
277 | +! | +
+ function(filtered_data, filter_ids) {+ |
+
278 | +! | +
+ filtered_data$clear_filter_states(force = TRUE)+ |
+
279 | +! | +
+ slices <- Filter(function(x) x$id %in% filter_ids, snapshot_state)+ |
+
280 | +! | +
+ filtered_data$set_filter_state(slices)+ |
+
281 | ++ |
+ },+ |
+
282 | +! | +
+ filtered_data = filtered_data_list,+ |
+
283 | +! | +
+ filter_ids = mapping_unfolded+ |
+
284 | ++ |
+ )+ |
+
285 | +! | +
+ slices_global(snapshot_state)+ |
+
286 | +! | +
+ removeModal()+ |
+
287 | ++ |
+ ### End restore procedure. ###+ |
+
288 | ++ |
+ })+ |
+
289 | ++ |
+ }+ |
+
290 | ++ |
+ # Create handler for downloading snapshot.+ |
+
291 | +! | +
+ if (!is.element(id_saveme, names(handlers))) {+ |
+
292 | +! | +
+ output[[id_saveme]] <- downloadHandler(+ |
+
293 | +! | +
+ filename = function() {+ |
+
294 | +! | +
+ sprintf("teal_snapshot_%s_%s.json", s, Sys.Date())+ |
+
295 | ++ |
+ },+ |
+
296 | +! | +
+ content = function(file) {+ |
+
297 | +! | +
+ snapshot <- snapshot_history()[[s]]+ |
+
298 | +! | +
+ snapshot_state <- as.teal_slices(snapshot)+ |
+
299 | +! | +
+ slices_store(tss = snapshot_state, file = file)+ |
+
300 | ++ |
+ }+ |
+
301 | ++ |
+ )+ |
+
302 | +! | +
+ handlers[[id_saveme]] <- id_saveme+ |
+
303 | ++ |
+ }+ |
+
304 | ++ |
+ # Create a row for the snapshot table.+ |
+
305 | +! | +
+ if (!is.element(id_rowme, names(divs))) {+ |
+
306 | +! | +
+ divs[[id_rowme]] <- div(+ |
+
307 | +! | +
+ class = "snapshot_table_row",+ |
+
308 | +! | +
+ span(h5(s)),+ |
+
309 | +! | +
+ actionLink(inputId = ns(id_pickme), label = icon("circle-check"), title = "select"),+ |
+
310 | +! | +
+ downloadLink(outputId = ns(id_saveme), label = icon("save"), title = "save to file")+ |
+
311 | ++ |
+ )+ |
+
312 | ++ |
+ }+ |
+
313 | ++ |
+ })+ |
+
314 | ++ |
+ })+ |
+
315 | ++ | + + | +
316 | ++ |
+ # Create table to display list of snapshots and their actions.+ |
+
317 | +7x | +
+ output$snapshot_list <- renderUI({+ |
+
318 | +3x | +
+ rows <- lapply(rev(reactiveValuesToList(divs)), function(d) d)+ |
+
319 | +3x | +
+ if (length(rows) == 0L) {+ |
+
320 | +3x | +
+ div(+ |
+
321 | +3x | +
+ class = "snapshot_manager_placeholder",+ |
+
322 | +3x | +
+ "Snapshots will appear here."+ |
+
323 | ++ |
+ )+ |
+
324 | ++ |
+ } else {+ |
+
325 | +! | +
+ rows+ |
+
326 | ++ |
+ }+ |
+
327 | ++ |
+ })+ |
+
328 | ++ |
+ })+ |
+
329 | ++ |
+ }+ |
+
330 | ++ | + + | +
331 | ++ | + + | +
332 | ++ | + + | +
333 | ++ | + + | +
334 | ++ |
+ ### utility functions ----+ |
+
335 | ++ | + + | +
336 | ++ |
+ #' Explicitly enumerate global filters.+ |
+
337 | ++ |
+ #'+ |
+
338 | ++ |
+ #' Transform module mapping such that global filters are explicitly specified for every module.+ |
+
339 | ++ |
+ #'+ |
+
340 | ++ |
+ #' @param mapping (`named list`) as stored in mapping parameter of `teal_slices`+ |
+
341 | ++ |
+ #' @param module_names (`character`) vector containing names of all modules in the app+ |
+
342 | ++ |
+ #' @return A `named_list` with one element per module, each element containing all filters applied to that module.+ |
+
343 | ++ |
+ #' @keywords internal+ |
+
344 | ++ |
+ #'+ |
+
345 | ++ |
+ unfold_mapping <- function(mapping, module_names) {+ |
+
346 | +! | +
+ module_names <- structure(module_names, names = module_names)+ |
+
347 | +! | +
+ lapply(module_names, function(x) c(mapping[[x]], mapping[["global_filters"]]))+ |
+
348 | ++ |
+ }+ |
+
349 | ++ | + + | +
350 | ++ |
+ #' Convert mapping matrix to filter mapping specification.+ |
+
351 | ++ |
+ #'+ |
+
352 | ++ |
+ #' Transform a mapping matrix, i.e. a data frame that maps each filter state to each module,+ |
+
353 | ++ |
+ #' to a list specification like the one used in the `mapping` attribute of `teal_slices`.+ |
+
354 | ++ |
+ #' Global filters are gathered in one list element.+ |
+
355 | ++ |
+ #' If a module has no active filters but the global ones, it will not be mentioned in the output.+ |
+
356 | ++ |
+ #'+ |
+
357 | ++ |
+ #' @param mapping_matrix (`data.frame`) of logical vectors where+ |
+
358 | ++ |
+ #' columns represent modules and row represent `teal_slice`s+ |
+
359 | ++ |
+ #' @return `named list` like that in the `mapping` attribute of a `teal_slices` object.+ |
+
360 | ++ |
+ #' @keywords internal+ |
+
361 | ++ |
+ #'+ |
+
362 | ++ |
+ matrix_to_mapping <- function(mapping_matrix) {+ |
+
363 | +! | +
+ mapping_matrix[] <- lapply(mapping_matrix, function(x) x | is.na(x))+ |
+
364 | +! | +
+ global <- vapply(as.data.frame(t(mapping_matrix)), all, logical(1L))+ |
+
365 | +! | +
+ global_filters <- names(global[global])+ |
+
366 | +! | +
+ local_filters <- mapping_matrix[!rownames(mapping_matrix) %in% global_filters, ]+ |
+
367 | ++ | + + | +
368 | +! | +
+ mapping <- c(lapply(local_filters, function(x) rownames(local_filters)[x]), list(global_filters = global_filters))+ |
+
369 | +! | +
+ Filter(function(x) length(x) != 0L, mapping)+ |
+
370 | ++ |
+ }+ |
+
1 | ++ |
+ setOldClass("teal_data_module")+ |
+
2 | ++ | + + | +
3 | ++ |
+ #' @name eval_code+ |
+
4 | ++ |
+ #' @description+ |
+
5 | ++ |
+ #' Given code is evaluated in the `qenv` environment of `teal_data` reactive defined in `teal_data_module`.+ |
+
6 | ++ |
+ #'+ |
+
7 | ++ |
+ #' @param object (`teal_data_module`)+ |
+
8 | ++ |
+ #' @inherit teal.code::eval_code+ |
+
9 | ++ |
+ #' @importMethodsFrom teal.code eval_code+ |
+
10 | ++ |
+ #' @export+ |
+
11 | ++ |
+ #' @examples+ |
+
12 | ++ |
+ #' tdm <- teal_data_module(+ |
+
13 | ++ |
+ #' ui = function(id) div(id = shiny::NS(id)("div_id")),+ |
+
14 | ++ |
+ #' server = function(id) {+ |
+
15 | ++ |
+ #' shiny::moduleServer(id, function(input, output, session) {+ |
+
16 | ++ |
+ #' shiny::reactive(teal_data(IRIS = iris))+ |
+
17 | ++ |
+ #' })+ |
+
18 | ++ |
+ #' }+ |
+
19 | ++ |
+ #' )+ |
+
20 | ++ |
+ #' \dontrun{+ |
+
21 | ++ |
+ #' eval_code(tdm, "IRIS <- subset(IRIS, Species == 'virginica')")+ |
+
22 | ++ |
+ #' }+ |
+
23 | ++ |
+ setMethod("eval_code", signature = c("teal_data_module", "character"), function(object, code) {+ |
+
24 | +5x | +
+ teal_data_module(+ |
+
25 | +5x | +
+ ui = function(id) {+ |
+
26 | +1x | +
+ ns <- NS(id)+ |
+
27 | +1x | +
+ object$ui(ns("mutate_inner"))+ |
+
28 | ++ |
+ },+ |
+
29 | +5x | +
+ server = function(id) {+ |
+
30 | +4x | +
+ moduleServer(id, function(input, output, session) {+ |
+
31 | +4x | +
+ data <- object$server("mutate_inner")+ |
+
32 | +4x | +
+ reactive(eval_code(data(), code))+ |
+
33 | ++ |
+ })+ |
+
34 | ++ |
+ }+ |
+
35 | ++ |
+ )+ |
+
36 | ++ |
+ })+ |
+
37 | ++ | + + | +
38 | ++ |
+ #' @rdname eval_code+ |
+
39 | ++ |
+ #' @export+ |
+
40 | ++ |
+ setMethod("eval_code", signature = c("teal_data_module", "language"), function(object, code) {+ |
+
41 | +1x | +
+ eval_code(object, code = format_expression(code))+ |
+
42 | ++ |
+ })+ |
+
43 | ++ | + + | +
44 | ++ |
+ #' @rdname eval_code+ |
+
45 | ++ |
+ #' @export+ |
+
46 | ++ |
+ setMethod("eval_code", signature = c("teal_data_module", "expression"), function(object, code) {+ |
+
47 | +4x | +
+ eval_code(object, code = format_expression(code))+ |
+
48 | ++ |
+ })+ |
+
49 | ++ | + + | +
50 | ++ |
+ #' @inherit teal.code::within.qenv params title details+ |
+
51 | ++ |
+ #' @description+ |
+
52 | ++ |
+ #' Convenience function for evaluating inline code inside the environment of a+ |
+
53 | ++ |
+ #' `teal_data_module`+ |
+
54 | ++ |
+ #'+ |
+
55 | ++ |
+ #' @param data (`teal_data_module`) object+ |
+
56 | ++ |
+ #' @return Returns a `teal_data_module` object with a delayed evaluation of `expr`+ |
+
57 | ++ |
+ #' when module.+ |
+
58 | ++ |
+ #' @export+ |
+
59 | ++ |
+ #' @seealso [base::within()], [teal_data_module()]+ |
+
60 | ++ |
+ #' @examples+ |
+
61 | ++ |
+ #' tdm <- teal_data_module(+ |
+
62 | ++ |
+ #' ui = function(id) div(id = shiny::NS(id)("div_id")),+ |
+
63 | ++ |
+ #' server = function(id) {+ |
+
64 | ++ |
+ #' shiny::moduleServer(id, function(input, output, session) {+ |
+
65 | ++ |
+ #' shiny::reactive(teal_data(IRIS = iris))+ |
+
66 | ++ |
+ #' })+ |
+
67 | ++ |
+ #' }+ |
+
68 | ++ |
+ #' )+ |
+
69 | ++ |
+ #' \dontrun{+ |
+
70 | ++ |
+ #' within(tdm, IRIS <- subset(IRIS, Species == "virginica"))+ |
+
71 | ++ |
+ #' }+ |
+
72 | ++ |
+ within.teal_data_module <- function(data, expr, ...) {+ |
+
73 | +3x | +
+ expr <- substitute(expr)+ |
+
74 | +3x | +
+ extras <- list(...)+ |
+
75 | ++ | + + | +
76 | ++ |
+ # Add braces for consistency.+ |
+
77 | +3x | +
+ if (!identical(as.list(expr)[[1L]], as.symbol("{"))) {+ |
+
78 | +3x | +
+ expr <- call("{", expr)+ |
+
79 | ++ |
+ }+ |
+
80 | ++ | + + | +
81 | +3x | +
+ calls <- as.list(expr)[-1]+ |
+
82 | ++ | + + | +
83 | ++ |
+ # Inject extra values into expressions.+ |
+
84 | +3x | +
+ calls <- lapply(calls, function(x) do.call(substitute, list(x, env = extras)))+ |
+
85 | ++ | + + | +
86 | +3x | +
+ eval_code(object = data, code = as.expression(calls))+ |
+
87 | ++ |
+ }+ |
+
1 | ++ |
+ # This module is the main teal module that puts everything together.+ |
+
2 | ++ | + + | +
3 | ++ |
+ #' teal main app module+ |
+
4 | ++ |
+ #'+ |
+
5 | ++ |
+ #' This is the main teal app that puts everything together.+ |
+
6 | ++ |
+ #'+ |
+
7 | ++ |
+ #' It displays the splash UI which is used to fetch the data, possibly+ |
+
8 | ++ |
+ #' prompting for a password input to fetch the data. Once the data is ready,+ |
+
9 | ++ |
+ #' the splash screen is replaced by the actual teal UI that is tabsetted and+ |
+
10 | ++ |
+ #' has a filter panel with `datanames` that are relevant for the current tab.+ |
+
11 | ++ |
+ #' Nested tabs are possible, but we limit it to two nesting levels for reasons+ |
+
12 | ++ |
+ #' of clarity of the UI.+ |
+
13 | ++ |
+ #'+ |
+
14 | ++ |
+ #' The splash screen functionality can also be used+ |
+
15 | ++ |
+ #' for non-delayed data which takes time to load into memory, avoiding+ |
+
16 | ++ |
+ #' Shiny session timeouts.+ |
+
17 | ++ |
+ #'+ |
+
18 | ++ |
+ #' Server evaluates the `teal_data_rv` (delayed data mechanism) and creates the+ |
+
19 | ++ |
+ #' `datasets` object that is shared across modules.+ |
+
20 | ++ |
+ #' Once it is ready and non-`NULL`, the splash screen is replaced by the+ |
+
21 | ++ |
+ #' main teal UI that depends on the data.+ |
+
22 | ++ |
+ #' The currently active tab is tracked and the right filter panel+ |
+
23 | ++ |
+ #' updates the displayed datasets to filter for according to the active `datanames`+ |
+
24 | ++ |
+ #' of the tab.+ |
+
25 | ++ |
+ #'+ |
+
26 | ++ |
+ #' It is written as a Shiny module so it can be added into other apps as well.+ |
+
27 | ++ |
+ #'+ |
+
28 | ++ |
+ #' @name module_teal+ |
+
29 | ++ |
+ #'+ |
+
30 | ++ |
+ #' @inheritParams ui_teal_with_splash+ |
+
31 | ++ |
+ #'+ |
+
32 | ++ |
+ #' @param splash_ui (`shiny.tag`)\cr UI to display initially,+ |
+
33 | ++ |
+ #' can be a splash screen or a Shiny module UI. For the latter, see+ |
+
34 | ++ |
+ #' [init()] about how to call the corresponding server function.+ |
+
35 | ++ |
+ #'+ |
+
36 | ++ |
+ #' @param teal_data_rv (`reactive`)\cr+ |
+
37 | ++ |
+ #' returns the `teal_data`, only evaluated once, `NULL` value is ignored+ |
+
38 | ++ |
+ #'+ |
+
39 | ++ |
+ #' @return+ |
+
40 | ++ |
+ #' `ui_teal` returns `HTML` for Shiny module UI.+ |
+
41 | ++ |
+ #' `srv_teal` returns `reactive` which returns the currently active module.+ |
+
42 | ++ |
+ #'+ |
+
43 | ++ |
+ #' @keywords internal+ |
+
44 | ++ |
+ #'+ |
+
45 | ++ |
+ #' @examples+ |
+
46 | ++ |
+ #' mods <- teal:::example_modules()+ |
+
47 | ++ |
+ #' teal_data_rv <- reactive(teal:::example_cdisc_data())+ |
+
48 | ++ |
+ #' app <- shinyApp(+ |
+
49 | ++ |
+ #' ui = function() {+ |
+
50 | ++ |
+ #' teal:::ui_teal("dummy")+ |
+
51 | ++ |
+ #' },+ |
+
52 | ++ |
+ #' server = function(input, output, session) {+ |
+
53 | ++ |
+ #' active_module <- teal:::srv_teal(id = "dummy", modules = mods, teal_data_rv = teal_data_rv)+ |
+
54 | ++ |
+ #' }+ |
+
55 | ++ |
+ #' )+ |
+
56 | ++ |
+ #' if (interactive()) {+ |
+
57 | ++ |
+ #' shinyApp(app$ui, app$server)+ |
+
58 | ++ |
+ #' }+ |
+
59 | ++ |
+ NULL+ |
+
60 | ++ | + + | +
61 | ++ |
+ #' @rdname module_teal+ |
+
62 | ++ |
+ ui_teal <- function(id,+ |
+
63 | ++ |
+ splash_ui = tags$h2("Starting the Teal App"),+ |
+
64 | ++ |
+ title = NULL,+ |
+
65 | ++ |
+ header = tags$p(""),+ |
+
66 | ++ |
+ footer = tags$p("")) {+ |
+
67 | +22x | +
+ if (checkmate::test_string(header)) {+ |
+
68 | +! | +
+ header <- tags$h1(header)+ |
+
69 | ++ |
+ }+ |
+
70 | +22x | +
+ if (checkmate::test_string(footer)) {+ |
+
71 | +! | +
+ footer <- tags$p(footer)+ |
+
72 | ++ |
+ }+ |
+
73 | +22x | +
+ checkmate::assert(+ |
+
74 | +22x | +
+ checkmate::check_class(splash_ui, "shiny.tag"),+ |
+
75 | +22x | +
+ checkmate::check_class(splash_ui, "shiny.tag.list"),+ |
+
76 | +22x | +
+ checkmate::check_class(splash_ui, "html")+ |
+
77 | ++ |
+ )+ |
+
78 | +22x | +
+ checkmate::assert(+ |
+
79 | +22x | +
+ checkmate::check_class(header, "shiny.tag"),+ |
+
80 | +22x | +
+ checkmate::check_class(header, "shiny.tag.list"),+ |
+
81 | +22x | +
+ checkmate::check_class(header, "html")+ |
+
82 | ++ |
+ )+ |
+
83 | +22x | +
+ checkmate::assert(+ |
+
84 | +22x | +
+ checkmate::check_class(footer, "shiny.tag"),+ |
+
85 | +22x | +
+ checkmate::check_class(footer, "shiny.tag.list"),+ |
+
86 | +22x | +
+ checkmate::check_class(footer, "html")+ |
+
87 | ++ |
+ )+ |
+
88 | ++ | + + | +
89 | +22x | +
+ ns <- NS(id)+ |
+
90 | ++ |
+ # Once the data is loaded, we will remove this element and add the real teal UI instead+ |
+
91 | +22x | +
+ splash_ui <- div(+ |
+
92 | ++ |
+ # id so we can remove the splash screen once ready, which is the first child of this container+ |
+
93 | +22x | +
+ id = ns("main_ui_container"),+ |
+
94 | ++ |
+ # we put it into a div, so it can easily be removed as a whole, also when it is a tagList (and not+ |
+
95 | ++ |
+ # just the first item of the tagList)+ |
+
96 | +22x | +
+ div(splash_ui)+ |
+
97 | ++ |
+ )+ |
+
98 | ++ | + + | +
99 | ++ |
+ # show busy icon when shiny session is busy computing stuff+ |
+
100 | ++ |
+ # based on https://stackoverflow.com/questions/17325521/r-shiny-display-loading-message-while-function-is-running/22475216#22475216 #nolint+ |
+
101 | +22x | +
+ shiny_busy_message_panel <- conditionalPanel(+ |
+
102 | +22x | +
+ condition = "(($('html').hasClass('shiny-busy')) && (document.getElementById('shiny-notification-panel') == null))", # nolint+ |
+
103 | +22x | +
+ div(+ |
+
104 | +22x | +
+ icon("arrows-rotate", "spin fa-spin"),+ |
+
105 | +22x | +
+ "Computing ...",+ |
+
106 | ++ |
+ # CSS defined in `custom.css`+ |
+
107 | +22x | +
+ class = "shinybusymessage"+ |
+
108 | ++ |
+ )+ |
+
109 | ++ |
+ )+ |
+
110 | ++ | + + | +
111 | +22x | +
+ res <- fluidPage(+ |
+
112 | +22x | +
+ title = title,+ |
+
113 | +22x | +
+ theme = get_teal_bs_theme(),+ |
+
114 | +22x | +
+ include_teal_css_js(),+ |
+
115 | +22x | +
+ tags$header(header),+ |
+
116 | +22x | +
+ tags$hr(class = "my-2"),+ |
+
117 | +22x | +
+ shiny_busy_message_panel,+ |
+
118 | +22x | +
+ splash_ui,+ |
+
119 | +22x | +
+ tags$hr(),+ |
+
120 | +22x | +
+ tags$footer(+ |
+
121 | +22x | +
+ div(+ |
+
122 | +22x | +
+ footer,+ |
+
123 | +22x | +
+ teal.widgets::verbatim_popup_ui(ns("sessionInfo"), "Session Info", type = "link"),+ |
+
124 | +22x | +
+ textOutput(ns("identifier"))+ |
+
125 | ++ |
+ )+ |
+
126 | ++ |
+ )+ |
+
127 | ++ |
+ )+ |
+
128 | +22x | +
+ return(res)+ |
+
129 | ++ |
+ }+ |
+
130 | ++ | + + | +
131 | ++ | + + | +
132 | ++ |
+ #' @rdname module_teal+ |
+
133 | ++ |
+ srv_teal <- function(id, modules, teal_data_rv, filter = teal_slices()) {+ |
+
134 | +17x | +
+ stopifnot(is.reactive(teal_data_rv))+ |
+
135 | +16x | +
+ moduleServer(id, function(input, output, session) {+ |
+
136 | +16x | +
+ logger::log_trace("srv_teal initializing the module.")+ |
+
137 | ++ | + + | +
138 | +16x | +
+ output$identifier <- renderText(+ |
+
139 | +16x | +
+ paste0("Pid:", Sys.getpid(), " Token:", substr(session$token, 25, 32))+ |
+
140 | ++ |
+ )+ |
+
141 | ++ | + + | +
142 | +16x | +
+ teal.widgets::verbatim_popup_srv(+ |
+
143 | +16x | +
+ "sessionInfo",+ |
+
144 | +16x | +
+ verbatim_content = utils::capture.output(utils::sessionInfo()),+ |
+
145 | +16x | +
+ title = "SessionInfo"+ |
+
146 | ++ |
+ )+ |
+
147 | ++ | + + | +
148 | ++ |
+ # `JavaScript` code+ |
+
149 | +16x | +
+ run_js_files(files = "init.js") # `JavaScript` code to make the clipboard accessible+ |
+
150 | ++ |
+ # set timezone in shiny app+ |
+
151 | ++ |
+ # timezone is set in the early beginning so it will be available also+ |
+
152 | ++ |
+ # for `DDL` and all shiny modules+ |
+
153 | +16x | +
+ get_client_timezone(session$ns)+ |
+
154 | +16x | +
+ observeEvent(+ |
+
155 | +16x | +
+ eventExpr = input$timezone,+ |
+
156 | +16x | +
+ once = TRUE,+ |
+
157 | +16x | +
+ handlerExpr = {+ |
+
158 | +! | +
+ session$userData$timezone <- input$timezone+ |
+
159 | +! | +
+ logger::log_trace("srv_teal@1 Timezone set to client's timezone: { input$timezone }.")+ |
+
160 | ++ |
+ }+ |
+
161 | ++ |
+ )+ |
+
162 | ++ | + + | +
163 | +16x | +
+ reporter <- teal.reporter::Reporter$new()+ |
+
164 | +16x | +
+ if (is_arg_used(modules, "reporter") && length(extract_module(modules, "teal_module_previewer")) == 0) {+ |
+
165 | +! | +
+ modules <- append_module(modules, reporter_previewer_module())+ |
+
166 | ++ |
+ }+ |
+
167 | ++ | + + | +
168 | +16x | +
+ env <- environment()+ |
+
169 | +16x | +
+ datasets_reactive <- eventReactive(teal_data_rv(), {+ |
+
170 | +5x | +
+ env$progress <- shiny::Progress$new(session)+ |
+
171 | +5x | +
+ env$progress$set(0.25, message = "Setting data")+ |
+
172 | ++ | + + | +
173 | ++ |
+ # create a list of data following structure of the nested modules list structure.+ |
+
174 | ++ |
+ # Because it's easier to unpack modules and datasets when they follow the same nested structure.+ |
+
175 | +5x | +
+ datasets_singleton <- teal_data_to_filtered_data(teal_data_rv())+ |
+
176 | ++ | + + | +
177 | ++ |
+ # Singleton starts with only global filters active.+ |
+
178 | +5x | +
+ filter_global <- Filter(function(x) x$id %in% attr(filter, "mapping")$global_filters, filter)+ |
+
179 | +5x | +
+ datasets_singleton$set_filter_state(filter_global)+ |
+
180 | ++ | + + | +
181 | +5x | +
+ module_datasets <- function(modules) {+ |
+
182 | +20x | +
+ if (inherits(modules, "teal_modules")) {+ |
+
183 | +8x | +
+ datasets <- lapply(modules$children, module_datasets)+ |
+
184 | +8x | +
+ labels <- vapply(modules$children, `[[`, character(1), "label")+ |
+
185 | +8x | +
+ names(datasets) <- labels+ |
+
186 | +8x | +
+ datasets+ |
+
187 | +12x | +
+ } else if (isTRUE(attr(filter, "module_specific"))) {+ |
+
188 | ++ |
+ # we should create FilteredData even if modules$datanames is null+ |
+
189 | ++ |
+ # null controls a display of filter panel but data should be still passed+ |
+
190 | +3x | +
+ datanames <- if (is.null(modules$datanames) || modules$datanames == "all") {+ |
+
191 | +3x | +
+ include_parent_datanames(+ |
+
192 | +3x | +
+ teal.data::datanames(teal_data_rv()),+ |
+
193 | +3x | +
+ teal_data_rv()@join_keys+ |
+
194 | ++ |
+ )+ |
+
195 | ++ |
+ } else {+ |
+
196 | +! | +
+ modules$datanames+ |
+
197 | ++ |
+ }+ |
+
198 | ++ |
+ # todo: subset teal_data to datanames+ |
+
199 | +3x | +
+ datasets_module <- teal_data_to_filtered_data(teal_data_rv(), datanames = datanames)+ |
+
200 | ++ | + + | +
201 | ++ |
+ # set initial filters+ |
+
202 | ++ |
+ # - filtering filters for this module+ |
+
203 | +3x | +
+ slices <- Filter(x = filter, f = function(x) {+ |
+
204 | +! | +
+ x$id %in% unique(unlist(attr(filter, "mapping")[c(modules$label, "global_filters")])) &&+ |
+
205 | +! | +
+ x$dataname %in% datanames+ |
+
206 | ++ |
+ })+ |
+
207 | +3x | +
+ include_varnames <- attr(slices, "include_varnames")[names(attr(slices, "include_varnames")) %in% datanames]+ |
+
208 | +3x | +
+ exclude_varnames <- attr(slices, "exclude_varnames")[names(attr(slices, "exclude_varnames")) %in% datanames]+ |
+
209 | +3x | +
+ slices$include_varnames <- include_varnames+ |
+
210 | +3x | +
+ slices$exclude_varnames <- exclude_varnames+ |
+
211 | +3x | +
+ datasets_module$set_filter_state(slices)+ |
+
212 | +3x | +
+ datasets_module+ |
+
213 | ++ |
+ } else {+ |
+
214 | +9x | +
+ datasets_singleton+ |
+
215 | ++ |
+ }+ |
+
216 | ++ |
+ }+ |
+
217 | +5x | +
+ module_datasets(modules)+ |
+
218 | ++ |
+ })+ |
+
219 | ++ | + + | +
220 | ++ |
+ # Replace splash / welcome screen once data is loaded ----+ |
+
221 | ++ |
+ # ignoreNULL to not trigger at the beginning when data is NULL+ |
+
222 | ++ |
+ # just handle it once because data obtained through delayed loading should+ |
+
223 | ++ |
+ # usually not change afterwards+ |
+
224 | ++ |
+ # if restored from bookmarked state, `filter` is ignored+ |
+
225 | ++ | + + | +
226 | +16x | +
+ observeEvent(datasets_reactive(), once = TRUE, {+ |
+
227 | +1x | +
+ logger::log_trace("srv_teal@5 setting main ui after data was pulled")+ |
+
228 | +1x | +
+ on.exit(env$progress$close())+ |
+
229 | +1x | +
+ env$progress$set(0.5, message = "Setting up main UI")+ |
+
230 | +1x | +
+ datasets <- datasets_reactive()+ |
+
231 | ++ | + + | +
232 | ++ |
+ # main_ui_container contains splash screen first and we remove it and replace it by the real UI+ |
+
233 | +1x | +
+ removeUI(sprintf("#%s > div:nth-child(1)", session$ns("main_ui_container")))+ |
+
234 | +1x | +
+ insertUI(+ |
+
235 | +1x | +
+ selector = paste0("#", session$ns("main_ui_container")),+ |
+
236 | +1x | +
+ where = "beforeEnd",+ |
+
237 | ++ |
+ # we put it into a div, so it can easily be removed as a whole, also when it is a tagList (and not+ |
+
238 | ++ |
+ # just the first item of the tagList)+ |
+
239 | +1x | +
+ ui = div(ui_tabs_with_filters(+ |
+
240 | +1x | +
+ session$ns("main_ui"),+ |
+
241 | +1x | +
+ modules = modules,+ |
+
242 | +1x | +
+ datasets = datasets,+ |
+
243 | +1x | +
+ filter = filter+ |
+
244 | ++ |
+ )),+ |
+
245 | ++ |
+ # needed so that the UI inputs are available and can be immediately updated, otherwise, updating may not+ |
+
246 | ++ |
+ # have any effect as they are ignored when not present+ |
+
247 | +1x | +
+ immediate = TRUE+ |
+
248 | ++ |
+ )+ |
+
249 | ++ | + + | +
250 | ++ |
+ # must make sure that this is only executed once as modules assume their observers are only+ |
+
251 | ++ |
+ # registered once (calling server functions twice would trigger observers twice each time)+ |
+
252 | +1x | +
+ active_module <- srv_tabs_with_filters(+ |
+
253 | +1x | +
+ id = "main_ui",+ |
+
254 | +1x | +
+ datasets = datasets,+ |
+
255 | +1x | +
+ modules = modules,+ |
+
256 | +1x | +
+ reporter = reporter,+ |
+
257 | +1x | +
+ filter = filter+ |
+
258 | ++ |
+ )+ |
+
259 | +1x | +
+ return(active_module)+ |
+
260 | ++ |
+ })+ |
+
261 | ++ |
+ })+ |
+
262 | ++ |
+ }+ |
+
1 | ++ |
+ #' Generates library calls from current session info+ |
+
2 | ++ |
+ #'+ |
+
3 | ++ |
+ #' Function to create multiple library calls out of current session info to make reproducible code works.+ |
+
4 | ++ |
+ #'+ |
+
5 | ++ |
+ #' @return Character object contain code+ |
+
6 | ++ |
+ #' @keywords internal+ |
+
7 | ++ |
+ get_rcode_libraries <- function() {+ |
+
8 | +16x | +
+ vapply(+ |
+
9 | +16x | +
+ utils::sessionInfo()$otherPkgs,+ |
+
10 | +16x | +
+ function(x) {+ |
+
11 | +272x | +
+ paste0("library(", x$Package, ")")+ |
+
12 | ++ |
+ },+ |
+
13 | +16x | +
+ character(1)+ |
+
14 | ++ |
+ ) %>%+ |
+
15 | ++ |
+ # put it into reverse order to correctly simulate executed code+ |
+
16 | +16x | +
+ rev() %>%+ |
+
17 | +16x | +
+ paste0(sep = "\n") %>%+ |
+
18 | +16x | +
+ paste0(collapse = "")+ |
+
19 | ++ |
+ }+ |
+
20 | ++ | + + | +
21 | ++ | + + | +
22 | ++ | + + | +
23 | ++ |
+ get_rcode_str_install <- function() {+ |
+
24 | +20x | +
+ code_string <- getOption("teal.load_nest_code")+ |
+
25 | ++ | + + | +
26 | +20x | +
+ if (!is.null(code_string) && is.character(code_string)) {+ |
+
27 | +2x | +
+ return(code_string)+ |
+
28 | ++ |
+ }+ |
+
29 | ++ | + + | +
30 | +18x | +
+ return("# Add any code to install/load your NEST environment here\n")+ |
+
31 | ++ |
+ }+ |
+
32 | ++ | + + | +
33 | ++ |
+ #' Get datasets code+ |
+
34 | ++ |
+ #'+ |
+
35 | ++ |
+ #' Get combined code from `FilteredData` and from `CodeClass` object.+ |
+
36 | ++ |
+ #'+ |
+
37 | ++ |
+ #' @param datanames (`character`) names of datasets to extract code from+ |
+
38 | ++ |
+ #' @param datasets (`FilteredData`) object+ |
+
39 | ++ |
+ #' @param hashes named (`list`) of hashes per dataset+ |
+
40 | ++ |
+ #'+ |
+
41 | ++ |
+ #' @return `character(3)` containing the following elements:+ |
+
42 | ++ |
+ #' - data pre-processing code (from `data` argument in `init`)+ |
+
43 | ++ |
+ #' - hash check of loaded objects+ |
+
44 | ++ |
+ #' - filter code+ |
+
45 | ++ |
+ #'+ |
+
46 | ++ |
+ #' @keywords internal+ |
+
47 | ++ |
+ get_datasets_code <- function(datanames, datasets, hashes) {+ |
+
48 | +14x | +
+ str_prepro <- teal.data:::get_code_dependency(attr(datasets, "preprocessing_code"), names = datanames)+ |
+
49 | +14x | +
+ if (length(str_prepro) == 0) {+ |
+
50 | +! | +
+ str_prepro <- "message('Preprocessing is empty')"+ |
+
51 | +14x | +
+ } else if (length(str_prepro) > 0) {+ |
+
52 | +14x | +
+ str_prepro <- paste0(str_prepro, "\n\n")+ |
+
53 | ++ |
+ }+ |
+
54 | ++ | + + | +
55 | +14x | +
+ str_hash <- paste(+ |
+
56 | +14x | +
+ paste0(+ |
+
57 | +14x | +
+ vapply(+ |
+
58 | +14x | +
+ datanames,+ |
+
59 | +14x | +
+ function(dataname) {+ |
+
60 | +17x | +
+ sprintf(+ |
+
61 | +17x | +
+ "stopifnot(%s == %s)",+ |
+
62 | +17x | +
+ deparse1(bquote(rlang::hash(.(as.name(dataname))))),+ |
+
63 | +17x | +
+ deparse1(hashes[[dataname]])+ |
+
64 | ++ |
+ )+ |
+
65 | ++ |
+ },+ |
+
66 | +14x | +
+ character(1)+ |
+
67 | ++ |
+ ),+ |
+
68 | +14x | +
+ collapse = "\n"+ |
+
69 | ++ |
+ ),+ |
+
70 | +14x | +
+ "\n\n"+ |
+
71 | ++ |
+ )+ |
+
72 | ++ | + + | +
73 | +14x | +
+ str_filter <- teal.slice::get_filter_expr(datasets, datanames)+ |
+
74 | ++ | + + | +
75 | +14x | +
+ c(str_prepro, str_hash, str_filter)+ |
+
76 | ++ |
+ }+ |
+
1 | ++ |
+ #' Validate that dataset has a minimum number of observations+ |
+
2 | ++ |
+ #'+ |
+
3 | ++ |
+ #' @description `r lifecycle::badge("stable")`+ |
+
4 | ++ |
+ #' @param x a data.frame+ |
+
5 | ++ |
+ #' @param min_nrow minimum number of rows in \code{x}+ |
+
6 | ++ |
+ #' @param complete \code{logical} default \code{FALSE} when set to \code{TRUE} then complete cases are checked.+ |
+
7 | ++ |
+ #' @param allow_inf \code{logical} default \code{TRUE} when set to \code{FALSE} then error thrown if any values are+ |
+
8 | ++ |
+ #' infinite.+ |
+
9 | ++ |
+ #' @param msg (`character(1)`) additional message to display alongside the default message.+ |
+
10 | ++ |
+ #'+ |
+
11 | ++ |
+ #' @details This function is a wrapper for `shiny::validate`.+ |
+
12 | ++ |
+ #'+ |
+
13 | ++ |
+ #' @export+ |
+
14 | ++ |
+ #'+ |
+
15 | ++ |
+ #' @examples+ |
+
16 | ++ |
+ #' library(teal)+ |
+
17 | ++ |
+ #' ui <- fluidPage(+ |
+
18 | ++ |
+ #' sliderInput("len", "Max Length of Sepal",+ |
+
19 | ++ |
+ #' min = 4.3, max = 7.9, value = 5+ |
+
20 | ++ |
+ #' ),+ |
+
21 | ++ |
+ #' plotOutput("plot")+ |
+
22 | ++ |
+ #' )+ |
+
23 | ++ |
+ #'+ |
+
24 | ++ |
+ #' server <- function(input, output) {+ |
+
25 | ++ |
+ #' output$plot <- renderPlot({+ |
+
26 | ++ |
+ #' df <- iris[iris$Sepal.Length <= input$len, ]+ |
+
27 | ++ |
+ #' validate_has_data(+ |
+
28 | ++ |
+ #' iris_f,+ |
+
29 | ++ |
+ #' min_nrow = 10,+ |
+
30 | ++ |
+ #' complete = FALSE,+ |
+
31 | ++ |
+ #' msg = "Please adjust Max Length of Sepal"+ |
+
32 | ++ |
+ #' )+ |
+
33 | ++ |
+ #'+ |
+
34 | ++ |
+ #' hist(iris_f$Sepal.Length, breaks = 5)+ |
+
35 | ++ |
+ #' })+ |
+
36 | ++ |
+ #' }+ |
+
37 | ++ |
+ #' if (interactive()) {+ |
+
38 | ++ |
+ #' shinyApp(ui, server)+ |
+
39 | ++ |
+ #' }+ |
+
40 | ++ |
+ #'+ |
+
41 | ++ |
+ validate_has_data <- function(x,+ |
+
42 | ++ |
+ min_nrow = NULL,+ |
+
43 | ++ |
+ complete = FALSE,+ |
+
44 | ++ |
+ allow_inf = TRUE,+ |
+
45 | ++ |
+ msg = NULL) {+ |
+
46 | +17x | +
+ checkmate::assert_string(msg, null.ok = TRUE)+ |
+
47 | +15x | +
+ checkmate::assert_data_frame(x)+ |
+
48 | +15x | +
+ if (!is.null(min_nrow)) {+ |
+
49 | +15x | +
+ if (complete) {+ |
+
50 | +5x | +
+ complete_index <- stats::complete.cases(x)+ |
+
51 | +5x | +
+ validate(need(+ |
+
52 | +5x | +
+ sum(complete_index) > 0 && nrow(x[complete_index, , drop = FALSE]) >= min_nrow,+ |
+
53 | +5x | +
+ paste(c(paste("Number of complete cases is less than:", min_nrow), msg), collapse = "\n")+ |
+
54 | ++ |
+ ))+ |
+
55 | ++ |
+ } else {+ |
+
56 | +10x | +
+ validate(need(+ |
+
57 | +10x | +
+ nrow(x) >= min_nrow,+ |
+
58 | +10x | +
+ paste(+ |
+
59 | +10x | +
+ c(paste("Minimum number of records not met: >=", min_nrow, "records required."), msg),+ |
+
60 | +10x | +
+ collapse = "\n"+ |
+
61 | ++ |
+ )+ |
+
62 | ++ |
+ ))+ |
+
63 | ++ |
+ }+ |
+
64 | ++ | + + | +
65 | +10x | +
+ if (!allow_inf) {+ |
+
66 | +6x | +
+ validate(need(+ |
+
67 | +6x | +
+ all(vapply(x, function(col) !is.numeric(col) || !any(is.infinite(col)), logical(1))),+ |
+
68 | +6x | +
+ "Dataframe contains Inf values which is not allowed."+ |
+
69 | ++ |
+ ))+ |
+
70 | ++ |
+ }+ |
+
71 | ++ |
+ }+ |
+
72 | ++ |
+ }+ |
+
73 | ++ | + + | +
74 | ++ |
+ #' Validate that dataset has unique rows for key variables+ |
+
75 | ++ |
+ #'+ |
+
76 | ++ |
+ #' @description `r lifecycle::badge("stable")`+ |
+
77 | ++ |
+ #' @param x a data.frame+ |
+
78 | ++ |
+ #' @param key a vector of ID variables from \code{x} that identify unique records+ |
+
79 | ++ |
+ #'+ |
+
80 | ++ |
+ #' @details This function is a wrapper for `shiny::validate`.+ |
+
81 | ++ |
+ #'+ |
+
82 | ++ |
+ #' @export+ |
+
83 | ++ |
+ #'+ |
+
84 | ++ |
+ #' @examples+ |
+
85 | ++ |
+ #' iris$id <- rep(1:50, times = 3)+ |
+
86 | ++ |
+ #' ui <- fluidPage(+ |
+
87 | ++ |
+ #' selectInput(+ |
+
88 | ++ |
+ #' inputId = "species",+ |
+
89 | ++ |
+ #' label = "Select species",+ |
+
90 | ++ |
+ #' choices = c("setosa", "versicolor", "virginica"),+ |
+
91 | ++ |
+ #' selected = "setosa",+ |
+
92 | ++ |
+ #' multiple = TRUE+ |
+
93 | ++ |
+ #' ),+ |
+
94 | ++ |
+ #' plotOutput("plot")+ |
+
95 | ++ |
+ #' )+ |
+
96 | ++ |
+ #' server <- function(input, output) {+ |
+
97 | ++ |
+ #' output$plot <- renderPlot({+ |
+
98 | ++ |
+ #' iris_f <- iris[iris$Species %in% input$species, ]+ |
+
99 | ++ |
+ #' validate_one_row_per_id(iris_f, key = c("id"))+ |
+
100 | ++ |
+ #'+ |
+
101 | ++ |
+ #' hist(iris_f$Sepal.Length, breaks = 5)+ |
+
102 | ++ |
+ #' })+ |
+
103 | ++ |
+ #' }+ |
+
104 | ++ |
+ #' if (interactive()) {+ |
+
105 | ++ |
+ #' shinyApp(ui, server)+ |
+
106 | ++ |
+ #' }+ |
+
107 | ++ |
+ #'+ |
+
108 | ++ |
+ validate_one_row_per_id <- function(x, key = c("USUBJID", "STUDYID")) {+ |
+
109 | +! | +
+ validate(need(!any(duplicated(x[key])), paste("Found more than one row per id.")))+ |
+
110 | ++ |
+ }+ |
+
111 | ++ | + + | +
112 | ++ |
+ #' Validates that vector includes all expected values+ |
+
113 | ++ |
+ #'+ |
+
114 | ++ |
+ #' @description `r lifecycle::badge("stable")`+ |
+
115 | ++ |
+ #' @param x values to test. All must be in \code{choices}+ |
+
116 | ++ |
+ #' @param choices a vector to test for values of \code{x}+ |
+
117 | ++ |
+ #' @param msg warning message to display+ |
+
118 | ++ |
+ #'+ |
+
119 | ++ |
+ #' @details This function is a wrapper for `shiny::validate`.+ |
+
120 | ++ |
+ #'+ |
+
121 | ++ |
+ #' @export+ |
+
122 | ++ |
+ #'+ |
+
123 | ++ |
+ #' @examples+ |
+
124 | ++ |
+ #' ui <- fluidPage(+ |
+
125 | ++ |
+ #' selectInput(+ |
+
126 | ++ |
+ #' "species",+ |
+
127 | ++ |
+ #' "Select species",+ |
+
128 | ++ |
+ #' choices = c("setosa", "versicolor", "virginica", "unknown species"),+ |
+
129 | ++ |
+ #' selected = "setosa",+ |
+
130 | ++ |
+ #' multiple = FALSE+ |
+
131 | ++ |
+ #' ),+ |
+
132 | ++ |
+ #' verbatimTextOutput("summary")+ |
+
133 | ++ |
+ #' )+ |
+
134 | ++ |
+ #'+ |
+
135 | ++ |
+ #' server <- function(input, output) {+ |
+
136 | ++ |
+ #' output$summary <- renderPrint({+ |
+
137 | ++ |
+ #' validate_in(input$species, iris$Species, "Species does not exist.")+ |
+
138 | ++ |
+ #' nrow(iris[iris$Species == input$species, ])+ |
+
139 | ++ |
+ #' })+ |
+
140 | ++ |
+ #' }+ |
+
141 | ++ |
+ #' if (interactive()) {+ |
+
142 | ++ |
+ #' shinyApp(ui, server)+ |
+
143 | ++ |
+ #' }+ |
+
144 | ++ |
+ #'+ |
+
145 | ++ |
+ validate_in <- function(x, choices, msg) {+ |
+
146 | +! | +
+ validate(need(length(x) > 0 && length(choices) > 0 && all(x %in% choices), msg))+ |
+
147 | ++ |
+ }+ |
+
148 | ++ | + + | +
149 | ++ |
+ #' Validates that vector has length greater than 0+ |
+
150 | ++ |
+ #'+ |
+
151 | ++ |
+ #' @description `r lifecycle::badge("stable")`+ |
+
152 | ++ |
+ #' @param x vector+ |
+
153 | ++ |
+ #' @param msg message to display+ |
+
154 | ++ |
+ #'+ |
+
155 | ++ |
+ #' @details This function is a wrapper for `shiny::validate`.+ |
+
156 | ++ |
+ #'+ |
+
157 | ++ |
+ #' @export+ |
+
158 | ++ |
+ #'+ |
+
159 | ++ |
+ #' @examples+ |
+
160 | ++ |
+ #' data <- data.frame(+ |
+
161 | ++ |
+ #' id = c(1:10, 11:20, 1:10),+ |
+
162 | ++ |
+ #' strata = rep(c("A", "B"), each = 15)+ |
+
163 | ++ |
+ #' )+ |
+
164 | ++ |
+ #' ui <- fluidPage(+ |
+
165 | ++ |
+ #' selectInput("ref1", "Select strata1 to compare",+ |
+
166 | ++ |
+ #' choices = c("A", "B", "C"), selected = "A"+ |
+
167 | ++ |
+ #' ),+ |
+
168 | ++ |
+ #' selectInput("ref2", "Select strata2 to compare",+ |
+
169 | ++ |
+ #' choices = c("A", "B", "C"), selected = "B"+ |
+
170 | ++ |
+ #' ),+ |
+
171 | ++ |
+ #' verbatimTextOutput("arm_summary")+ |
+
172 | ++ |
+ #' )+ |
+
173 | ++ |
+ #'+ |
+
174 | ++ |
+ #' server <- function(input, output) {+ |
+
175 | ++ |
+ #' output$arm_summary <- renderText({+ |
+
176 | ++ |
+ #' sample_1 <- data$id[data$strata == input$ref1]+ |
+
177 | ++ |
+ #' sample_2 <- data$id[data$strata == input$ref2]+ |
+
178 | ++ |
+ #'+ |
+
179 | ++ |
+ #' validate_has_elements(sample_1, "No subjects in strata1.")+ |
+
180 | ++ |
+ #' validate_has_elements(sample_2, "No subjects in strata2.")+ |
+
181 | ++ |
+ #'+ |
+
182 | ++ |
+ #' paste0(+ |
+
183 | ++ |
+ #' "Number of samples in: strata1=", length(sample_1),+ |
+
184 | ++ |
+ #' " comparions strata2=", length(sample_2)+ |
+
185 | ++ |
+ #' )+ |
+
186 | ++ |
+ #' })+ |
+
187 | ++ |
+ #' }+ |
+
188 | ++ |
+ #' if (interactive()) {+ |
+
189 | ++ |
+ #' shinyApp(ui, server)+ |
+
190 | ++ |
+ #' }+ |
+
191 | ++ |
+ validate_has_elements <- function(x, msg) {+ |
+
192 | +! | +
+ validate(need(length(x) > 0, msg))+ |
+
193 | ++ |
+ }+ |
+
194 | ++ | + + | +
195 | ++ |
+ #' Validates no intersection between two vectors+ |
+
196 | ++ |
+ #'+ |
+
197 | ++ |
+ #' @description `r lifecycle::badge("stable")`+ |
+
198 | ++ |
+ #' @param x vector+ |
+
199 | ++ |
+ #' @param y vector+ |
+
200 | ++ |
+ #' @param msg message to display if \code{x} and \code{y} intersect+ |
+
201 | ++ |
+ #'+ |
+
202 | ++ |
+ #' @details This function is a wrapper for `shiny::validate`.+ |
+
203 | ++ |
+ #'+ |
+
204 | ++ |
+ #' @export+ |
+
205 | ++ |
+ #'+ |
+
206 | ++ |
+ #' @examples+ |
+
207 | ++ |
+ #' data <- data.frame(+ |
+
208 | ++ |
+ #' id = c(1:10, 11:20, 1:10),+ |
+
209 | ++ |
+ #' strata = rep(c("A", "B", "C"), each = 10)+ |
+
210 | ++ |
+ #' )+ |
+
211 | ++ |
+ #'+ |
+
212 | ++ |
+ #' ui <- fluidPage(+ |
+
213 | ++ |
+ #' selectInput("ref1", "Select strata1 to compare",+ |
+
214 | ++ |
+ #' choices = c("A", "B", "C"),+ |
+
215 | ++ |
+ #' selected = "A"+ |
+
216 | ++ |
+ #' ),+ |
+
217 | ++ |
+ #' selectInput("ref2", "Select strata2 to compare",+ |
+
218 | ++ |
+ #' choices = c("A", "B", "C"),+ |
+
219 | ++ |
+ #' selected = "B"+ |
+
220 | ++ |
+ #' ),+ |
+
221 | ++ |
+ #' verbatimTextOutput("summary")+ |
+
222 | ++ |
+ #' )+ |
+
223 | ++ |
+ #'+ |
+
224 | ++ |
+ #' server <- function(input, output) {+ |
+
225 | ++ |
+ #' output$summary <- renderText({+ |
+
226 | ++ |
+ #' sample_1 <- data$id[data$strata == input$ref1]+ |
+
227 | ++ |
+ #' sample_2 <- data$id[data$strata == input$ref2]+ |
+
228 | ++ |
+ #'+ |
+
229 | ++ |
+ #' validate_no_intersection(+ |
+
230 | ++ |
+ #' sample_1, sample_2,+ |
+
231 | ++ |
+ #' "subjects within strata1 and strata2 cannot overlap"+ |
+
232 | ++ |
+ #' )+ |
+
233 | ++ |
+ #' paste0(+ |
+
234 | ++ |
+ #' "Number of subject in: reference treatment=", length(sample_1),+ |
+
235 | ++ |
+ #' " comparions treatment=", length(sample_2)+ |
+
236 | ++ |
+ #' )+ |
+
237 | ++ |
+ #' })+ |
+
238 | ++ |
+ #' }+ |
+
239 | ++ |
+ #' if (interactive()) {+ |
+
240 | ++ |
+ #' shinyApp(ui, server)+ |
+
241 | ++ |
+ #' }+ |
+
242 | ++ |
+ #'+ |
+
243 | ++ |
+ validate_no_intersection <- function(x, y, msg) {+ |
+
244 | +! | +
+ validate(need(length(intersect(x, y)) == 0, msg))+ |
+
245 | ++ |
+ }+ |
+
246 | ++ | + + | +
247 | ++ | + + | +
248 | ++ |
+ #' Validates that dataset contains specific variable+ |
+
249 | ++ |
+ #'+ |
+
250 | ++ |
+ #' @description `r lifecycle::badge("stable")`+ |
+
251 | ++ |
+ #' @param data a data.frame+ |
+
252 | ++ |
+ #' @param varname name of variable in \code{data}+ |
+
253 | ++ |
+ #' @param msg message to display if \code{data} does not include \code{varname}+ |
+
254 | ++ |
+ #'+ |
+
255 | ++ |
+ #' @details This function is a wrapper for `shiny::validate`.+ |
+
256 | ++ |
+ #'+ |
+
257 | ++ |
+ #' @export+ |
+
258 | ++ |
+ #'+ |
+
259 | ++ |
+ #' @examples+ |
+
260 | ++ |
+ #' data <- data.frame(+ |
+
261 | ++ |
+ #' one = rep("a", length.out = 20),+ |
+
262 | ++ |
+ #' two = rep(c("a", "b"), length.out = 20)+ |
+
263 | ++ |
+ #' )+ |
+
264 | ++ |
+ #' ui <- fluidPage(+ |
+
265 | ++ |
+ #' selectInput(+ |
+
266 | ++ |
+ #' "var",+ |
+
267 | ++ |
+ #' "Select variable",+ |
+
268 | ++ |
+ #' choices = c("one", "two", "three", "four"),+ |
+
269 | ++ |
+ #' selected = "one"+ |
+
270 | ++ |
+ #' ),+ |
+
271 | ++ |
+ #' verbatimTextOutput("summary")+ |
+
272 | ++ |
+ #' )+ |
+
273 | ++ |
+ #'+ |
+
274 | ++ |
+ #' server <- function(input, output) {+ |
+
275 | ++ |
+ #' output$summary <- renderText({+ |
+
276 | ++ |
+ #' validate_has_variable(data, input$var)+ |
+
277 | ++ |
+ #' paste0("Selected treatment variables: ", paste(input$var, collapse = ", "))+ |
+
278 | ++ |
+ #' })+ |
+
279 | ++ |
+ #' }+ |
+
280 | ++ |
+ #' if (interactive()) {+ |
+
281 | ++ |
+ #' shinyApp(ui, server)+ |
+
282 | ++ |
+ #' }+ |
+
283 | ++ |
+ validate_has_variable <- function(data, varname, msg) {+ |
+
284 | +! | +
+ if (length(varname) != 0) {+ |
+
285 | +! | +
+ has_vars <- varname %in% names(data)+ |
+
286 | ++ | + + | +
287 | +! | +
+ if (!all(has_vars)) {+ |
+
288 | +! | +
+ if (missing(msg)) {+ |
+
289 | +! | +
+ msg <- sprintf(+ |
+
290 | +! | +
+ "%s does not have the required variables: %s.",+ |
+
291 | +! | +
+ deparse(substitute(data)),+ |
+
292 | +! | +
+ toString(varname[!has_vars])+ |
+
293 | ++ |
+ )+ |
+
294 | ++ |
+ }+ |
+
295 | +! | +
+ validate(need(FALSE, msg))+ |
+
296 | ++ |
+ }+ |
+
297 | ++ |
+ }+ |
+
298 | ++ |
+ }+ |
+
299 | ++ | + + | +
300 | ++ |
+ #' Validate that variables has expected number of levels+ |
+
301 | ++ |
+ #'+ |
+
302 | ++ |
+ #' @description `r lifecycle::badge("stable")`+ |
+
303 | ++ |
+ #' @param x variable name. If \code{x} is not a factor, the unique values+ |
+
304 | ++ |
+ #' are treated as levels.+ |
+
305 | ++ |
+ #' @param min_levels cutoff for minimum number of levels of \code{x}+ |
+
306 | ++ |
+ #' @param max_levels cutoff for maximum number of levels of \code{x}+ |
+
307 | ++ |
+ #' @param var_name name of variable being validated for use in+ |
+
308 | ++ |
+ #' validation message+ |
+
309 | ++ |
+ #'+ |
+
310 | ++ |
+ #' @details If the number of levels of \code{x} is less than \code{min_levels}+ |
+
311 | ++ |
+ #' or greater than \code{max_levels} the validation will fail.+ |
+
312 | ++ |
+ #' This function is a wrapper for `shiny::validate`.+ |
+
313 | ++ |
+ #'+ |
+
314 | ++ |
+ #' @export+ |
+
315 | ++ |
+ #' @examples+ |
+
316 | ++ |
+ #' data <- data.frame(+ |
+
317 | ++ |
+ #' one = rep("a", length.out = 20),+ |
+
318 | ++ |
+ #' two = rep(c("a", "b"), length.out = 20),+ |
+
319 | ++ |
+ #' three = rep(c("a", "b", "c"), length.out = 20),+ |
+
320 | ++ |
+ #' four = rep(c("a", "b", "c", "d"), length.out = 20),+ |
+
321 | ++ |
+ #' stringsAsFactors = TRUE+ |
+
322 | ++ |
+ #' )+ |
+
323 | ++ |
+ #' ui <- fluidPage(+ |
+
324 | ++ |
+ #' selectInput(+ |
+
325 | ++ |
+ #' "var",+ |
+
326 | ++ |
+ #' "Select variable",+ |
+
327 | ++ |
+ #' choices = c("one", "two", "three", "four"),+ |
+
328 | ++ |
+ #' selected = "one"+ |
+
329 | ++ |
+ #' ),+ |
+
330 | ++ |
+ #' verbatimTextOutput("summary")+ |
+
331 | ++ |
+ #' )+ |
+
332 | ++ |
+ #'+ |
+
333 | ++ |
+ #' server <- function(input, output) {+ |
+
334 | ++ |
+ #' output$summary <- renderText({+ |
+
335 | ++ |
+ #' validate_n_levels(data[[input$var]], min_levels = 2, max_levels = 15, var_name = input$var)+ |
+
336 | ++ |
+ #' paste0(+ |
+
337 | ++ |
+ #' "Levels of selected treatment variable: ",+ |
+
338 | ++ |
+ #' paste(levels(data[[input$var]]),+ |
+
339 | ++ |
+ #' collapse = ", "+ |
+
340 | ++ |
+ #' )+ |
+
341 | ++ |
+ #' )+ |
+
342 | ++ |
+ #' })+ |
+
343 | ++ |
+ #' }+ |
+
344 | ++ |
+ #' if (interactive()) {+ |
+
345 | ++ |
+ #' shinyApp(ui, server)+ |
+
346 | ++ |
+ #' }+ |
+
347 | ++ |
+ validate_n_levels <- function(x, min_levels = 1, max_levels = 12, var_name) {+ |
+
348 | +! | +
+ x_levels <- if (is.factor(x)) {+ |
+
349 | +! | +
+ levels(x)+ |
+
350 | ++ |
+ } else {+ |
+
351 | +! | +
+ unique(x)+ |
+
352 | ++ |
+ }+ |
+
353 | ++ | + + | +
354 | +! | +
+ if (!is.null(min_levels) && !(is.null(max_levels))) {+ |
+
355 | +! | +
+ validate(need(+ |
+
356 | +! | +
+ length(x_levels) >= min_levels && length(x_levels) <= max_levels,+ |
+
357 | +! | +
+ sprintf(+ |
+
358 | +! | +
+ "%s variable needs minimum %s level(s) and maximum %s level(s).",+ |
+
359 | +! | +
+ var_name, min_levels, max_levels+ |
+
360 | ++ |
+ )+ |
+
361 | ++ |
+ ))+ |
+
362 | +! | +
+ } else if (!is.null(min_levels)) {+ |
+
363 | +! | +
+ validate(need(+ |
+
364 | +! | +
+ length(x_levels) >= min_levels,+ |
+
365 | +! | +
+ sprintf("%s variable needs minimum %s levels(s)", var_name, min_levels)+ |
+
366 | ++ |
+ ))+ |
+
367 | +! | +
+ } else if (!is.null(max_levels)) {+ |
+
368 | +! | +
+ validate(need(+ |
+
369 | +! | +
+ length(x_levels) <= max_levels,+ |
+
370 | +! | +
+ sprintf("%s variable needs maximum %s level(s)", var_name, max_levels)+ |
+
371 | ++ |
+ ))+ |
+
372 | ++ |
+ }+ |
+
373 | ++ |
+ }+ |
+
1 | ++ |
+ #' Get dummy `CDISC` data+ |
+
2 | ++ |
+ #'+ |
+
3 | ++ |
+ #' Get dummy `CDISC` data including `ADSL`, `ADAE` and `ADLB`.+ |
+
4 | ++ |
+ #' Some NAs are also introduced to stress test.+ |
+
5 | ++ |
+ #'+ |
+
6 | ++ |
+ #' @return `cdisc_data`+ |
+
7 | ++ |
+ #' @keywords internal+ |
+
8 | ++ |
+ example_cdisc_data <- function() { # nolint+ |
+
9 | +! | +
+ ADSL <- data.frame( # nolint+ |
+
10 | +! | +
+ STUDYID = "study",+ |
+
11 | +! | +
+ USUBJID = 1:10,+ |
+
12 | +! | +
+ SEX = sample(c("F", "M"), 10, replace = TRUE),+ |
+
13 | +! | +
+ AGE = stats::rpois(10, 40)+ |
+
14 | ++ |
+ )+ |
+
15 | +! | +
+ ADTTE <- rbind(ADSL, ADSL, ADSL) # nolint+ |
+
16 | +! | +
+ ADTTE$PARAMCD <- rep(c("OS", "EFS", "PFS"), each = 10) # nolint+ |
+
17 | +! | +
+ ADTTE$AVAL <- c( # nolint+ |
+
18 | +! | +
+ stats::rnorm(10, mean = 700, sd = 200), # dummy OS level+ |
+
19 | +! | +
+ stats::rnorm(10, mean = 400, sd = 100), # dummy EFS level+ |
+
20 | +! | +
+ stats::rnorm(10, mean = 450, sd = 200) # dummy PFS level+ |
+
21 | ++ |
+ )+ |
+
22 | ++ | + + | +
23 | +! | +
+ ADSL$logical_test <- sample(c(TRUE, FALSE, NA), size = nrow(ADSL), replace = TRUE) # nolint+ |
+
24 | +! | +
+ ADSL$SEX[c(2, 5)] <- NA # nolint+ |
+
25 | ++ | + + | +
26 | +! | +
+ res <- teal.data::cdisc_data(+ |
+
27 | +! | +
+ ADSL = ADSL,+ |
+
28 | +! | +
+ ADTTE = ADTTE,+ |
+
29 | +! | +
+ code = '+ |
+
30 | +! | +
+ ADSL <- data.frame(+ |
+
31 | +! | +
+ STUDYID = "study",+ |
+
32 | +! | +
+ USUBJID = 1:10,+ |
+
33 | +! | +
+ SEX = sample(c("F", "M"), 10, replace = TRUE),+ |
+
34 | +! | +
+ AGE = rpois(10, 40)+ |
+
35 | ++ |
+ )+ |
+
36 | +! | +
+ ADTTE <- rbind(ADSL, ADSL, ADSL)+ |
+
37 | +! | +
+ ADTTE$PARAMCD <- rep(c("OS", "EFS", "PFS"), each = 10)+ |
+
38 | +! | +
+ ADTTE$AVAL <- c(+ |
+
39 | +! | +
+ rnorm(10, mean = 700, sd = 200),+ |
+
40 | +! | +
+ rnorm(10, mean = 400, sd = 100),+ |
+
41 | +! | +
+ rnorm(10, mean = 450, sd = 200)+ |
+
42 | ++ |
+ )+ |
+
43 | ++ | + + | +
44 | +! | +
+ ADSL$logical_test <- sample(c(TRUE, FALSE, NA), size = nrow(ADSL), replace = TRUE)+ |
+
45 | +! | +
+ ADSL$SEX[c(2, 5)] <- NA+ |
+
46 | ++ |
+ '+ |
+
47 | ++ |
+ )+ |
+
48 | +! | +
+ return(res)+ |
+
49 | ++ |
+ }+ |
+
50 | ++ | + + | +
51 | ++ |
+ #' Get datasets to go with example modules.+ |
+
52 | ++ |
+ #'+ |
+
53 | ++ |
+ #' Creates a nested list, the structure of which matches the module hierarchy created by `example_modules`.+ |
+
54 | ++ |
+ #' Each list leaf is the same `FilteredData` object.+ |
+
55 | ++ |
+ #'+ |
+
56 | ++ |
+ #' @return named list of `FilteredData` objects, each with `ADSL` set.+ |
+
57 | ++ |
+ #' @keywords internal+ |
+
58 | ++ |
+ example_datasets <- function() { # nolint+ |
+
59 | +! | +
+ dummy_cdisc_data <- example_cdisc_data()+ |
+
60 | +! | +
+ datasets <- teal_data_to_filtered_data(dummy_cdisc_data)+ |
+
61 | +! | +
+ list(+ |
+
62 | +! | +
+ "d2" = list(+ |
+
63 | +! | +
+ "d3" = list(+ |
+
64 | +! | +
+ "aaa1" = datasets,+ |
+
65 | +! | +
+ "aaa2" = datasets,+ |
+
66 | +! | +
+ "aaa3" = datasets+ |
+
67 | ++ |
+ ),+ |
+
68 | +! | +
+ "bbb" = datasets+ |
+
69 | ++ |
+ ),+ |
+
70 | +! | +
+ "ccc" = datasets+ |
+
71 | ++ |
+ )+ |
+
72 | ++ |
+ }+ |
+
73 | ++ | + + | +
74 | ++ |
+ #' An example `teal` module+ |
+
75 | ++ |
+ #'+ |
+
76 | ++ |
+ #' @description `r lifecycle::badge("experimental")`+ |
+
77 | ++ |
+ #' @inheritParams module+ |
+
78 | ++ |
+ #' @return A `teal` module which can be included in the `modules` argument to [teal::init()].+ |
+
79 | ++ |
+ #' @examples+ |
+
80 | ++ |
+ #' app <- init(+ |
+
81 | ++ |
+ #' data = teal_data(+ |
+
82 | ++ |
+ #' dataset("IRIS", iris),+ |
+
83 | ++ |
+ #' dataset("MTCARS", mtcars)+ |
+
84 | ++ |
+ #' ),+ |
+
85 | ++ |
+ #' modules = example_module()+ |
+
86 | ++ |
+ #' )+ |
+
87 | ++ |
+ #' if (interactive()) {+ |
+
88 | ++ |
+ #' shinyApp(app$ui, app$server)+ |
+
89 | ++ |
+ #' }+ |
+
90 | ++ |
+ #' @export+ |
+
91 | ++ |
+ example_module <- function(label = "example teal module", datanames = "all") {+ |
+
92 | +64x | +
+ checkmate::assert_string(label)+ |
+
93 | +64x | +
+ module(+ |
+
94 | +64x | +
+ label,+ |
+
95 | +64x | +
+ server = function(id, data) {+ |
+
96 | +1x | +
+ checkmate::assert_class(data, "tdata")+ |
+
97 | +1x | +
+ moduleServer(id, function(input, output, session) {+ |
+
98 | +1x | +
+ output$text <- renderPrint(data[[input$dataname]]())+ |
+
99 | +1x | +
+ teal.widgets::verbatim_popup_srv(+ |
+
100 | +1x | +
+ id = "rcode",+ |
+
101 | +1x | +
+ verbatim_content = attr(data, "code")(),+ |
+
102 | +1x | +
+ title = "Association Plot"+ |
+
103 | ++ |
+ )+ |
+
104 | ++ |
+ })+ |
+
105 | ++ |
+ },+ |
+
106 | +64x | +
+ ui = function(id, data) {+ |
+
107 | +1x | +
+ ns <- NS(id)+ |
+
108 | +1x | +
+ teal.widgets::standard_layout(+ |
+
109 | +1x | +
+ output = verbatimTextOutput(ns("text")),+ |
+
110 | +1x | +
+ encoding = div(+ |
+
111 | +1x | +
+ selectInput(ns("dataname"), "Choose a dataset", choices = names(data)),+ |
+
112 | +1x | +
+ teal.widgets::verbatim_popup_ui(ns("rcode"), "Show R code")+ |
+
113 | ++ |
+ )+ |
+
114 | ++ |
+ )+ |
+
115 | ++ |
+ },+ |
+
116 | +64x | +
+ datanames = datanames+ |
+
117 | ++ |
+ )+ |
+
118 | ++ |
+ }+ |
+
119 | ++ | + + | +
120 | ++ | + + | +
121 | ++ |
+ #' Get example modules.+ |
+
122 | ++ |
+ #'+ |
+
123 | ++ |
+ #' Creates an example hierarchy of `teal_modules` from which a `teal` app can be created.+ |
+
124 | ++ |
+ #' @param datanames (`character`)\cr+ |
+
125 | ++ |
+ #' names of the datasets to be used in the example modules. Possible choices are `ADSL`, `ADTTE`.+ |
+
126 | ++ |
+ #' @return `teal_modules`+ |
+
127 | ++ |
+ #' @keywords internal+ |
+
128 | ++ |
+ example_modules <- function(datanames = c("ADSL", "ADTTE")) {+ |
+
129 | +3x | +
+ checkmate::assert_subset(datanames, c("ADSL", "ADTTE"))+ |
+
130 | +3x | +
+ mods <- modules(+ |
+
131 | +3x | +
+ label = "d1",+ |
+
132 | +3x | +
+ modules(+ |
+
133 | +3x | +
+ label = "d2",+ |
+
134 | +3x | +
+ modules(+ |
+
135 | +3x | +
+ label = "d3",+ |
+
136 | +3x | +
+ example_module(label = "aaa1", datanames = datanames),+ |
+
137 | +3x | +
+ example_module(label = "aaa2", datanames = datanames),+ |
+
138 | +3x | +
+ example_module(label = "aaa3", datanames = datanames)+ |
+
139 | ++ |
+ ),+ |
+
140 | +3x | +
+ example_module(label = "bbb", datanames = datanames)+ |
+
141 | ++ |
+ ),+ |
+
142 | +3x | +
+ example_module(label = "ccc", datanames = datanames)+ |
+
143 | ++ |
+ )+ |
+
144 | +3x | +
+ return(mods)+ |
+
145 | ++ |
+ }+ |
+
1 | ++ |
+ #' Send input validation messages to output.+ |
+
2 | ++ |
+ #'+ |
+
3 | ++ |
+ #' Captures messages from `InputValidator` objects and collates them+ |
+
4 | ++ |
+ #' into one message passed to `validate`.+ |
+
5 | ++ |
+ #'+ |
+
6 | ++ |
+ #' `shiny::validate` is used to withhold rendering of an output element until+ |
+
7 | ++ |
+ #' certain conditions are met and to print a validation message in place+ |
+
8 | ++ |
+ #' of the output element.+ |
+
9 | ++ |
+ #' `shinyvalidate::InputValidator` allows to validate input elements+ |
+
10 | ++ |
+ #' and to display specific messages in their respective input widgets.+ |
+
11 | ++ |
+ #' `validate_inputs` provides a hybrid solution.+ |
+
12 | ++ |
+ #' Given an `InputValidator` object, messages corresponding to inputs that fail validation+ |
+
13 | ++ |
+ #' are extracted and placed in one validation message that is passed to a `validate`/`need` call.+ |
+
14 | ++ |
+ #' This way the input `validator` messages are repeated in the output.+ |
+
15 | ++ |
+ #'+ |
+
16 | ++ |
+ #' The `...` argument accepts any number of `InputValidator` objects+ |
+
17 | ++ |
+ #' or a nested list of such objects.+ |
+
18 | ++ |
+ #' If `validators` are passed directly, all their messages are printed together+ |
+
19 | ++ |
+ #' under one (optional) header message specified by `header`. If a list is passed,+ |
+
20 | ++ |
+ #' messages are grouped by `validator`. The list's names are used as headers+ |
+
21 | ++ |
+ #' for their respective message groups.+ |
+
22 | ++ |
+ #' If neither of the nested list elements is named, a header message is taken from `header`.+ |
+
23 | ++ |
+ #'+ |
+
24 | ++ |
+ #' @param ... either any number of `InputValidator` objects+ |
+
25 | ++ |
+ #' or an optionally named, possibly nested `list` of `InputValidator`+ |
+
26 | ++ |
+ #' objects, see `Details`+ |
+
27 | ++ |
+ #' @param header `character(1)` generic validation message; set to NULL to omit+ |
+
28 | ++ |
+ #'+ |
+
29 | ++ |
+ #' @return+ |
+
30 | ++ |
+ #' Returns NULL if the final validation call passes and a `shiny.silent.error` if it fails.+ |
+
31 | ++ |
+ #'+ |
+
32 | ++ |
+ #' @seealso [`shinyvalidate::InputValidator`], [`shiny::validate`]+ |
+
33 | ++ |
+ #'+ |
+
34 | ++ |
+ #' @examples+ |
+
35 | ++ |
+ #' library(shiny)+ |
+
36 | ++ |
+ #' library(shinyvalidate)+ |
+
37 | ++ |
+ #'+ |
+
38 | ++ |
+ #' ui <- fluidPage(+ |
+
39 | ++ |
+ #' selectInput("method", "validation method", c("sequential", "combined", "grouped")),+ |
+
40 | ++ |
+ #' sidebarLayout(+ |
+
41 | ++ |
+ #' sidebarPanel(+ |
+
42 | ++ |
+ #' selectInput("letter", "select a letter:", c(letters[1:3], LETTERS[4:6])),+ |
+
43 | ++ |
+ #' selectInput("number", "select a number:", 1:6),+ |
+
44 | ++ |
+ #' br(),+ |
+
45 | ++ |
+ #' selectInput("color", "select a color:",+ |
+
46 | ++ |
+ #' c("black", "indianred2", "springgreen2", "cornflowerblue"),+ |
+
47 | ++ |
+ #' multiple = TRUE+ |
+
48 | ++ |
+ #' ),+ |
+
49 | ++ |
+ #' sliderInput("size", "select point size:",+ |
+
50 | ++ |
+ #' min = 0.1, max = 4, value = 0.25+ |
+
51 | ++ |
+ #' )+ |
+
52 | ++ |
+ #' ),+ |
+
53 | ++ |
+ #' mainPanel(plotOutput("plot"))+ |
+
54 | ++ |
+ #' )+ |
+
55 | ++ |
+ #' )+ |
+
56 | ++ |
+ #'+ |
+
57 | ++ |
+ #' server <- function(input, output) {+ |
+
58 | ++ |
+ #' # set up input validation+ |
+
59 | ++ |
+ #' iv <- InputValidator$new()+ |
+
60 | ++ |
+ #' iv$add_rule("letter", sv_in_set(LETTERS, "choose a capital letter"))+ |
+
61 | ++ |
+ #' iv$add_rule("number", ~ if (as.integer(.) %% 2L == 1L) "choose an even number")+ |
+
62 | ++ |
+ #' iv$enable()+ |
+
63 | ++ |
+ #' # more input validation+ |
+
64 | ++ |
+ #' iv_par <- InputValidator$new()+ |
+
65 | ++ |
+ #' iv_par$add_rule("color", sv_required(message = "choose a color"))+ |
+
66 | ++ |
+ #' iv_par$add_rule("color", ~ if (length(.) > 1L) "choose only one color")+ |
+
67 | ++ |
+ #' iv_par$add_rule(+ |
+
68 | ++ |
+ #' "size",+ |
+
69 | ++ |
+ #' sv_between(+ |
+
70 | ++ |
+ #' left = 0.5, right = 3,+ |
+
71 | ++ |
+ #' message_fmt = "choose a value between {left} and {right}"+ |
+
72 | ++ |
+ #' )+ |
+
73 | ++ |
+ #' )+ |
+
74 | ++ |
+ #' iv_par$enable()+ |
+
75 | ++ |
+ #'+ |
+
76 | ++ |
+ #' output$plot <- renderPlot({+ |
+
77 | ++ |
+ #' # validate output+ |
+
78 | ++ |
+ #' switch(input[["method"]],+ |
+
79 | ++ |
+ #' "sequential" = {+ |
+
80 | ++ |
+ #' validate_inputs(iv)+ |
+
81 | ++ |
+ #' validate_inputs(iv_par, header = "Set proper graphical parameters")+ |
+
82 | ++ |
+ #' },+ |
+
83 | ++ |
+ #' "combined" = validate_inputs(iv, iv_par),+ |
+
84 | ++ |
+ #' "grouped" = validate_inputs(list(+ |
+
85 | ++ |
+ #' "Some inputs require attention" = iv,+ |
+
86 | ++ |
+ #' "Set proper graphical parameters" = iv_par+ |
+
87 | ++ |
+ #' ))+ |
+
88 | ++ |
+ #' )+ |
+
89 | ++ |
+ #'+ |
+
90 | ++ |
+ #' plot(eruptions ~ waiting, faithful,+ |
+
91 | ++ |
+ #' las = 1, pch = 16,+ |
+
92 | ++ |
+ #' col = input[["color"]], cex = input[["size"]]+ |
+
93 | ++ |
+ #' )+ |
+
94 | ++ |
+ #' })+ |
+
95 | ++ |
+ #' }+ |
+
96 | ++ |
+ #'+ |
+
97 | ++ |
+ #' if (interactive()) {+ |
+
98 | ++ |
+ #' shinyApp(ui, server)+ |
+
99 | ++ |
+ #' }+ |
+
100 | ++ |
+ #'+ |
+
101 | ++ |
+ #' @export+ |
+
102 | ++ |
+ #'+ |
+
103 | ++ |
+ validate_inputs <- function(..., header = "Some inputs require attention") {+ |
+
104 | +36x | +
+ dots <- list(...)+ |
+
105 | +2x | +
+ if (!is_validators(dots)) stop("validate_inputs accepts validators or a list thereof")+ |
+
106 | ++ | + + | +
107 | +34x | +
+ messages <- extract_validator(dots, header)+ |
+
108 | +34x | +
+ failings <- if (!any_names(dots)) {+ |
+
109 | +29x | +
+ add_header(messages, header)+ |
+
110 | ++ |
+ } else {+ |
+
111 | +5x | +
+ unlist(messages)+ |
+
112 | ++ |
+ }+ |
+
113 | ++ | + + | +
114 | +34x | +
+ shiny::validate(shiny::need(is.null(failings), failings))+ |
+
115 | ++ |
+ }+ |
+
116 | ++ | + + | +
117 | ++ |
+ ### internal functions+ |
+
118 | ++ | + + | +
119 | ++ |
+ #' @keywords internal+ |
+
120 | ++ |
+ # recursive object type test+ |
+
121 | ++ |
+ # returns logical of length 1+ |
+
122 | ++ |
+ is_validators <- function(x) {+ |
+
123 | +118x | +
+ all(if (is.list(x)) unlist(lapply(x, is_validators)) else inherits(x, "InputValidator"))+ |
+
124 | ++ |
+ }+ |
+
125 | ++ | + + | +
126 | ++ |
+ #' @keywords internal+ |
+
127 | ++ |
+ # test if an InputValidator object is enabled+ |
+
128 | ++ |
+ # returns logical of length 1+ |
+
129 | ++ |
+ # official method requested at https://github.com/rstudio/shinyvalidate/issues/64+ |
+
130 | ++ |
+ validator_enabled <- function(x) {+ |
+
131 | +49x | +
+ x$.__enclos_env__$private$enabled+ |
+
132 | ++ |
+ }+ |
+
133 | ++ | + + | +
134 | ++ |
+ #' @keywords internal+ |
+
135 | ++ |
+ # recursively extract messages from validator list+ |
+
136 | ++ |
+ # returns character vector or a list of character vectors, possibly nested and named+ |
+
137 | ++ |
+ extract_validator <- function(iv, header) {+ |
+
138 | +113x | +
+ if (inherits(iv, "InputValidator")) {+ |
+
139 | +49x | +
+ add_header(gather_messages(iv), header)+ |
+
140 | ++ |
+ } else {+ |
+
141 | +58x | +
+ if (is.null(names(iv))) names(iv) <- rep("", length(iv))+ |
+
142 | +64x | +
+ mapply(extract_validator, iv = iv, header = names(iv), SIMPLIFY = FALSE)+ |
+
143 | ++ |
+ }+ |
+
144 | ++ |
+ }+ |
+
145 | ++ | + + | +
146 | ++ |
+ #' @keywords internal+ |
+
147 | ++ |
+ # collate failing messages from validator+ |
+
148 | ++ |
+ # returns list+ |
+
149 | ++ |
+ gather_messages <- function(iv) {+ |
+
150 | +49x | +
+ if (validator_enabled(iv)) {+ |
+
151 | +46x | +
+ status <- iv$validate()+ |
+
152 | +46x | +
+ failing_inputs <- Filter(Negate(is.null), status)+ |
+
153 | +46x | +
+ unique(lapply(failing_inputs, function(x) x[["message"]]))+ |
+
154 | ++ |
+ } else {+ |
+
155 | +3x | +
+ logger::log_warn("Validator is disabled and will be omitted.")+ |
+
156 | +3x | +
+ list()+ |
+
157 | ++ |
+ }+ |
+
158 | ++ |
+ }+ |
+
159 | ++ | + + | +
160 | ++ |
+ #' @keywords internal+ |
+
161 | ++ |
+ # add optional header to failing messages+ |
+
162 | ++ |
+ add_header <- function(messages, header = "") {+ |
+
163 | +78x | +
+ ans <- unlist(messages)+ |
+
164 | +78x | +
+ if (length(ans) != 0L && isTRUE(nchar(header) > 0L)) {+ |
+
165 | +31x | +
+ ans <- c(paste0(header, "\n"), ans, "\n")+ |
+
166 | ++ |
+ }+ |
+
167 | +78x | +
+ ans+ |
+
168 | ++ |
+ }+ |
+
169 | ++ | + + | +
170 | ++ |
+ #' @keywords internal+ |
+
171 | ++ |
+ # recursively check if the object contains a named list+ |
+
172 | ++ |
+ any_names <- function(x) {+ |
+
173 | +103x | +
+ any(+ |
+
174 | +103x | +
+ if (is.list(x)) {+ |
+
175 | +58x | +
+ if (!is.null(names(x)) && any(names(x) != "")) TRUE else unlist(lapply(x, any_names))+ |
+
176 | ++ |
+ } else {+ |
+
177 | +40x | +
+ FALSE+ |
+
178 | ++ |
+ }+ |
+
179 | ++ |
+ )+ |
+
180 | ++ |
+ }+ |
+
1 | ++ |
+ #' Create a `tdata` Object+ |
+
2 | ++ |
+ #'+ |
+
3 | ++ |
+ #' Create a new object called `tdata` which contains `data`, a `reactive` list of data.frames+ |
+
4 | ++ |
+ #' (or `MultiAssayExperiment`), with attributes:+ |
+
5 | ++ |
+ #' \itemize{+ |
+
6 | ++ |
+ #' \item{`code` (`reactive`) containing code used to generate the data}+ |
+
7 | ++ |
+ #' \item{join_keys (`join_keys`) containing the relationships between the data}+ |
+
8 | ++ |
+ #' \item{metadata (`named list`) containing any metadata associated with the data frames}+ |
+
9 | ++ |
+ #' }+ |
+
10 | ++ |
+ #' @name tdata+ |
+
11 | ++ |
+ #' @param data A `named list` of `data.frames` (or `MultiAssayExperiment`)+ |
+
12 | ++ |
+ #' which optionally can be `reactive`.+ |
+
13 | ++ |
+ #' Inside this object all of these items will be made `reactive`.+ |
+
14 | ++ |
+ #' @param code A `character` (or `reactive` which evaluates to a `character`) containing+ |
+
15 | ++ |
+ #' the code used to generate the data. This should be `reactive` if the code is changing+ |
+
16 | ++ |
+ #' during a reactive context (e.g. if filtering changes the code). Inside this+ |
+
17 | ++ |
+ #' object `code` will be made reactive+ |
+
18 | ++ |
+ #' @param join_keys A `teal.data::join_keys` object containing relationships between the+ |
+
19 | ++ |
+ #' datasets.+ |
+
20 | ++ |
+ #' @param metadata A `named list` each element contains a list of metadata about the named data.frame+ |
+
21 | ++ |
+ #' Each element of these list should be atomic and length one.+ |
+
22 | ++ |
+ #' @return A `tdata` object+ |
+
23 | ++ |
+ #' @examples+ |
+
24 | ++ |
+ #'+ |
+
25 | ++ |
+ #' data <- new_tdata(+ |
+
26 | ++ |
+ #' data = list(iris = iris, mtcars = reactive(mtcars), dd = data.frame(x = 1:10)),+ |
+
27 | ++ |
+ #' code = "iris <- iris+ |
+
28 | ++ |
+ #' mtcars <- mtcars+ |
+
29 | ++ |
+ #' dd <- data.frame(x = 1:10)",+ |
+
30 | ++ |
+ #' metadata = list(dd = list(author = "NEST"), iris = list(version = 1))+ |
+
31 | ++ |
+ #' )+ |
+
32 | ++ |
+ #'+ |
+
33 | ++ |
+ #' # Extract a data.frame+ |
+
34 | ++ |
+ #' isolate(data[["iris"]]())+ |
+
35 | ++ |
+ #'+ |
+
36 | ++ |
+ #' # Get code+ |
+
37 | ++ |
+ #' isolate(get_code(data))+ |
+
38 | ++ |
+ #'+ |
+
39 | ++ |
+ #' # Get metadata+ |
+
40 | ++ |
+ #' get_metadata(data, "iris")+ |
+
41 | ++ |
+ #'+ |
+
42 | ++ |
+ #' @export+ |
+
43 | ++ |
+ new_tdata <- function(data, code = "", join_keys = NULL, metadata = NULL) {+ |
+
44 | +42x | +
+ checkmate::assert_list(+ |
+
45 | +42x | +
+ data,+ |
+
46 | +42x | +
+ any.missing = FALSE, names = "unique",+ |
+
47 | +42x | +
+ types = c("data.frame", "reactive", "MultiAssayExperiment")+ |
+
48 | ++ |
+ )+ |
+
49 | +38x | +
+ checkmate::assert_class(join_keys, "join_keys", null.ok = TRUE)+ |
+
50 | +37x | +
+ checkmate::assert_multi_class(code, c("character", "reactive"))+ |
+
51 | ++ | + + | +
52 | +36x | +
+ checkmate::assert_list(metadata, names = "unique", null.ok = TRUE)+ |
+
53 | +34x | +
+ checkmate::assert_subset(names(metadata), names(data))+ |
+
54 | +22x | +
+ for (m in metadata) teal.data::validate_metadata(m)+ |
+
55 | ++ | + + | +
56 | +33x | +
+ if (is.reactive(code)) {+ |
+
57 | +17x | +
+ isolate(checkmate::assert_class(code(), "character", .var.name = "code"))+ |
+
58 | ++ |
+ }+ |
+
59 | ++ | + + | +
60 | ++ |
+ # create reactive data.frames+ |
+
61 | +32x | +
+ for (x in names(data)) {+ |
+
62 | +51x | +
+ if (!is.reactive(data[[x]])) {+ |
+
63 | +29x | +
+ data[[x]] <- do.call(reactive, list(as.name(x)), envir = list2env(data[x]))+ |
+
64 | ++ |
+ } else {+ |
+
65 | +22x | +
+ isolate(+ |
+
66 | +22x | +
+ checkmate::assert_multi_class(+ |
+
67 | +22x | +
+ data[[x]](), c("data.frame", "MultiAssayExperiment"),+ |
+
68 | +22x | +
+ .var.name = "data"+ |
+
69 | ++ |
+ )+ |
+
70 | ++ |
+ )+ |
+
71 | ++ |
+ }+ |
+
72 | ++ |
+ }+ |
+
73 | ++ | + + | +
74 | ++ |
+ # set attributes+ |
+
75 | +31x | +
+ attr(data, "code") <- if (is.reactive(code)) code else reactive(code)+ |
+
76 | +31x | +
+ attr(data, "join_keys") <- join_keys+ |
+
77 | +31x | +
+ attr(data, "metadata") <- metadata+ |
+
78 | ++ | + + | +
79 | ++ |
+ # set class+ |
+
80 | +31x | +
+ class(data) <- c("tdata", class(data))+ |
+
81 | +31x | +
+ data+ |
+
82 | ++ |
+ }+ |
+
83 | ++ | + + | +
84 | ++ |
+ #' Function to convert a `tdata` object to an `environment`+ |
+
85 | ++ |
+ #' Any `reactives` inside `tdata` are first evaluated+ |
+
86 | ++ |
+ #' @param data a `tdata` object+ |
+
87 | ++ |
+ #' @return an `environment`+ |
+
88 | ++ |
+ #' @examples+ |
+
89 | ++ |
+ #'+ |
+
90 | ++ |
+ #' data <- new_tdata(+ |
+
91 | ++ |
+ #' data = list(iris = iris, mtcars = reactive(mtcars)),+ |
+
92 | ++ |
+ #' code = "iris <- iris+ |
+
93 | ++ |
+ #' mtcars = mtcars"+ |
+
94 | ++ |
+ #' )+ |
+
95 | ++ |
+ #'+ |
+
96 | ++ |
+ #' my_env <- isolate(tdata2env(data))+ |
+
97 | ++ |
+ #'+ |
+
98 | ++ |
+ #' @export+ |
+
99 | ++ |
+ tdata2env <- function(data) { # nolint+ |
+
100 | +2x | +
+ checkmate::assert_class(data, "tdata")+ |
+
101 | +1x | +
+ list2env(lapply(data, function(x) if (is.reactive(x)) x() else x))+ |
+
102 | ++ |
+ }+ |
+
103 | ++ | + + | +
104 | ++ |
+ #' @rdname tdata+ |
+
105 | ++ |
+ #' @param x a `tdata` object+ |
+
106 | ++ |
+ #' @param ... additional arguments for the generic+ |
+
107 | ++ |
+ #' @export+ |
+
108 | ++ |
+ get_code.tdata <- function(x, ...) { # nolint+ |
+
109 | ++ |
+ # note teal.data which teal depends on defines the get_code method+ |
+
110 | +6x | +
+ attr(x, "code")()+ |
+
111 | ++ |
+ }+ |
+
112 | ++ | + + | +
113 | ++ | + + | +
114 | ++ |
+ #' Wrapper for `get_code.tdata`+ |
+
115 | ++ |
+ #' This wrapper is to be used by downstream packages to extract the code of a `tdata` object+ |
+
116 | ++ |
+ #'+ |
+
117 | ++ |
+ #' @param data (`tdata`) object+ |
+
118 | ++ |
+ #'+ |
+
119 | ++ |
+ #' @return (`character`) code used in the `tdata` object.+ |
+
120 | ++ |
+ #' @export+ |
+
121 | ++ |
+ get_code_tdata <- function(data) {+ |
+
122 | +4x | +
+ checkmate::assert_class(data, "tdata")+ |
+
123 | +2x | +
+ get_code(data)+ |
+
124 | ++ |
+ }+ |
+
125 | ++ | + + | +
126 | ++ |
+ #' Extract `join_keys` from `tdata`+ |
+
127 | ++ |
+ #' @param data A `tdata` object+ |
+
128 | ++ |
+ #' @param ... Additional arguments (not used)+ |
+
129 | ++ |
+ #' @export+ |
+
130 | ++ |
+ join_keys.tdata <- function(data, ...) {+ |
+
131 | +3x | +
+ attr(data, "join_keys")+ |
+
132 | ++ |
+ }+ |
+
133 | ++ | + + | +
134 | ++ | + + | +
135 | ++ |
+ #' Function to get metadata from a `tdata` object+ |
+
136 | ++ |
+ #' @param data `tdata` - object to extract the data from+ |
+
137 | ++ |
+ #' @param dataname `character(1)` the dataset name whose metadata is requested+ |
+
138 | ++ |
+ #' @return Either list of metadata or NULL if no metadata+ |
+
139 | ++ |
+ #' @export+ |
+
140 | ++ |
+ get_metadata <- function(data, dataname) {+ |
+
141 | +4x | +
+ checkmate::assert_string(dataname)+ |
+
142 | +4x | +
+ UseMethod("get_metadata", data)+ |
+
143 | ++ |
+ }+ |
+
144 | ++ | + + | +
145 | ++ |
+ #' @rdname get_metadata+ |
+
146 | ++ |
+ #' @export+ |
+
147 | ++ |
+ get_metadata.tdata <- function(data, dataname) {+ |
+
148 | +4x | +
+ metadata <- attr(data, "metadata")+ |
+
149 | +4x | +
+ if (is.null(metadata)) {+ |
+
150 | +1x | +
+ return(NULL)+ |
+
151 | ++ |
+ }+ |
+
152 | +3x | +
+ metadata[[dataname]]+ |
+
153 | ++ |
+ }+ |
+
154 | ++ | + + | +
155 | ++ |
+ #' @rdname get_metadata+ |
+
156 | ++ |
+ #' @export+ |
+
157 | ++ |
+ get_metadata.default <- function(data, dataname) {+ |
+
158 | +! | +
+ stop("get_metadata function not implemented for this object")+ |
+
159 | ++ |
+ }+ |
+
1 | ++ |
+ #' Filter manager modal+ |
+
2 | ++ |
+ #'+ |
+
3 | ++ |
+ #' Opens modal containing the filter manager UI.+ |
+
4 | ++ |
+ #'+ |
+
5 | ++ |
+ #' @name module_filter_manager_modal+ |
+
6 | ++ |
+ #' @inheritParams filter_manager_srv+ |
+
7 | ++ |
+ #' @examples+ |
+
8 | ++ |
+ #' fd1 <- teal.slice::init_filtered_data(list(iris = list(dataset = iris)))+ |
+
9 | ++ |
+ #' fd2 <- teal.slice::init_filtered_data(+ |
+
10 | ++ |
+ #' list(iris = list(dataset = iris), mtcars = list(dataset = mtcars))+ |
+
11 | ++ |
+ #' )+ |
+
12 | ++ |
+ #' fd3 <- teal.slice::init_filtered_data(+ |
+
13 | ++ |
+ #' list(iris = list(dataset = iris), women = list(dataset = women))+ |
+
14 | ++ |
+ #' )+ |
+
15 | ++ |
+ #' filter <- teal_slices(+ |
+
16 | ++ |
+ #' teal.slice::teal_slice(dataname = "iris", varname = "Sepal.Length"),+ |
+
17 | ++ |
+ #' teal.slice::teal_slice(dataname = "iris", varname = "Species"),+ |
+
18 | ++ |
+ #' teal.slice::teal_slice(dataname = "mtcars", varname = "mpg"),+ |
+
19 | ++ |
+ #' teal.slice::teal_slice(dataname = "women", varname = "height"),+ |
+
20 | ++ |
+ #' mapping = list(+ |
+
21 | ++ |
+ #' module2 = c("mtcars mpg"),+ |
+
22 | ++ |
+ #' module3 = c("women height"),+ |
+
23 | ++ |
+ #' global_filters = "iris Species"+ |
+
24 | ++ |
+ #' )+ |
+
25 | ++ |
+ #' )+ |
+
26 | ++ |
+ #'+ |
+
27 | ++ |
+ #' app <- shinyApp(+ |
+
28 | ++ |
+ #' ui = fluidPage(+ |
+
29 | ++ |
+ #' teal:::filter_manager_modal_ui("manager")+ |
+
30 | ++ |
+ #' ),+ |
+
31 | ++ |
+ #' server = function(input, output, session) {+ |
+
32 | ++ |
+ #' teal:::filter_manager_modal_srv(+ |
+
33 | ++ |
+ #' "manager",+ |
+
34 | ++ |
+ #' filtered_data_list = list(module1 = fd1, module2 = fd2, module3 = fd3),+ |
+
35 | ++ |
+ #' filter = filter+ |
+
36 | ++ |
+ #' )+ |
+
37 | ++ |
+ #' }+ |
+
38 | ++ |
+ #' )+ |
+
39 | ++ |
+ #' if (interactive()) {+ |
+
40 | ++ |
+ #' shinyApp(app$ui, app$server)+ |
+
41 | ++ |
+ #' }+ |
+
42 | ++ |
+ #'+ |
+
43 | ++ |
+ #' @keywords internal+ |
+
44 | ++ |
+ #'+ |
+
45 | ++ |
+ NULL+ |
+
46 | ++ | + + | +
47 | ++ |
+ #' @rdname module_filter_manager_modal+ |
+
48 | ++ |
+ filter_manager_modal_ui <- function(id) {+ |
+
49 | +1x | +
+ ns <- NS(id)+ |
+
50 | +1x | +
+ tags$button(+ |
+
51 | +1x | +
+ id = ns("show"),+ |
+
52 | +1x | +
+ class = "btn action-button filter_manager_button",+ |
+
53 | +1x | +
+ title = "Show filters manager modal",+ |
+
54 | +1x | +
+ icon("gear")+ |
+
55 | ++ |
+ )+ |
+
56 | ++ |
+ }+ |
+
57 | ++ | + + | +
58 | ++ |
+ #' @rdname module_filter_manager_modal+ |
+
59 | ++ |
+ filter_manager_modal_srv <- function(id, filtered_data_list, filter) {+ |
+
60 | +4x | +
+ moduleServer(id, function(input, output, session) {+ |
+
61 | +4x | +
+ observeEvent(input$show, {+ |
+
62 | +! | +
+ logger::log_trace("filter_manager_modal_srv@1 show button has been clicked.")+ |
+
63 | +! | +
+ showModal(+ |
+
64 | +! | +
+ modalDialog(+ |
+
65 | +! | +
+ filter_manager_ui(session$ns("filter_manager")),+ |
+
66 | +! | +
+ size = "l",+ |
+
67 | +! | +
+ footer = NULL,+ |
+
68 | +! | +
+ easyClose = TRUE+ |
+
69 | ++ |
+ )+ |
+
70 | ++ |
+ )+ |
+
71 | ++ |
+ })+ |
+
72 | ++ | + + | +
73 | +4x | +
+ filter_manager_srv("filter_manager", filtered_data_list, filter)+ |
+
74 | ++ |
+ })+ |
+
75 | ++ |
+ }+ |
+
76 | ++ | + + | +
77 | ++ |
+ #' @rdname module_filter_manager+ |
+
78 | ++ |
+ filter_manager_ui <- function(id) {+ |
+
79 | +! | +
+ ns <- NS(id)+ |
+
80 | +! | +
+ div(+ |
+
81 | +! | +
+ class = "filter_manager_content",+ |
+
82 | +! | +
+ tableOutput(ns("slices_table")),+ |
+
83 | +! | +
+ snapshot_manager_ui(ns("snapshot_manager"))+ |
+
84 | ++ |
+ )+ |
+
85 | ++ |
+ }+ |
+
86 | ++ | + + | +
87 | ++ |
+ #' Manage multiple `FilteredData` objects+ |
+
88 | ++ |
+ #'+ |
+
89 | ++ |
+ #' Oversee filter states in the whole application.+ |
+
90 | ++ |
+ #'+ |
+
91 | ++ |
+ #' @rdname module_filter_manager+ |
+
92 | ++ |
+ #' @details+ |
+
93 | ++ |
+ #' This module observes the changes of the filters in each `FilteredData` object+ |
+
94 | ++ |
+ #' and keeps track of all filters used. A mapping of filters to modules+ |
+
95 | ++ |
+ #' is kept in the `mapping_matrix` object (which is actually a `data.frame`)+ |
+
96 | ++ |
+ #' that tracks which filters (rows) are active in which modules (columns).+ |
+
97 | ++ |
+ #'+ |
+
98 | ++ |
+ #' @param id (`character(1)`)\cr+ |
+
99 | ++ |
+ #' `shiny` module id.+ |
+
100 | ++ |
+ #' @param filtered_data_list (`named list`)\cr+ |
+
101 | ++ |
+ #' A list, possibly nested, of `FilteredData` objects.+ |
+
102 | ++ |
+ #' Each `FilteredData` will be served to one module in the `teal` application.+ |
+
103 | ++ |
+ #' The structure of the list must reflect the nesting of modules in tabs+ |
+
104 | ++ |
+ #' and names of the list must be the same as labels of their respective modules.+ |
+
105 | ++ |
+ #' @inheritParams init+ |
+
106 | ++ |
+ #' @return A list of `reactive`s, each holding a `teal_slices`, as returned by `filter_manager_module_srv`.+ |
+
107 | ++ |
+ #' @keywords internal+ |
+
108 | ++ |
+ #'+ |
+
109 | ++ |
+ filter_manager_srv <- function(id, filtered_data_list, filter) {+ |
+
110 | +6x | +
+ moduleServer(id, function(input, output, session) {+ |
+
111 | +6x | +
+ logger::log_trace("filter_manager_srv initializing for: { paste(names(filtered_data_list), collapse = ', ')}.")+ |
+
112 | ++ | + + | +
113 | +6x | +
+ is_module_specific <- isTRUE(attr(filter, "module_specific"))+ |
+
114 | ++ | + + | +
115 | ++ |
+ # Create global list of slices.+ |
+
116 | ++ |
+ # Contains all available teal_slice objects available to all modules.+ |
+
117 | ++ |
+ # Passed whole to instances of FilteredData used for individual modules.+ |
+
118 | ++ |
+ # Down there a subset that pertains to the data sets used in that module is applied and displayed.+ |
+
119 | +6x | +
+ slices_global <- reactiveVal(filter)+ |
+
120 | ++ | + + | +
121 | +6x | +
+ filtered_data_list <-+ |
+
122 | +6x | +
+ if (!is_module_specific) {+ |
+
123 | ++ |
+ # Retrieve the first FilteredData from potentially nested list.+ |
+
124 | ++ |
+ # List of length one is named "global_filters" because that name is forbidden for a module label.+ |
+
125 | +5x | +
+ list(global_filters = unlist(filtered_data_list)[[1]])+ |
+
126 | ++ |
+ } else {+ |
+
127 | ++ |
+ # Flatten potentially nested list of FilteredData objects while maintaining useful names.+ |
+
128 | ++ |
+ # Simply using `unlist` would result in concatenated names.+ |
+
129 | +1x | +
+ flatten_nested <- function(x, name = NULL) {+ |
+
130 | +5x | +
+ if (inherits(x, "FilteredData")) {+ |
+
131 | +3x | +
+ setNames(list(x), name)+ |
+
132 | ++ |
+ } else {+ |
+
133 | +2x | +
+ unlist(lapply(names(x), function(name) flatten_nested(x[[name]], name)))+ |
+
134 | ++ |
+ }+ |
+
135 | ++ |
+ }+ |
+
136 | +1x | +
+ flatten_nested(filtered_data_list)+ |
+
137 | ++ |
+ }+ |
+
138 | ++ | + + | +
139 | ++ |
+ # Create mapping fo filters to modules in matrix form (presented as data.frame).+ |
+
140 | ++ |
+ # Modules get NAs for filters that cannot be set for them.+ |
+
141 | +6x | +
+ mapping_matrix <- reactive({+ |
+
142 | +6x | +
+ state_ids_global <- vapply(slices_global(), `[[`, character(1L), "id")+ |
+
143 | +6x | +
+ mapping_smooth <- lapply(filtered_data_list, function(x) {+ |
+
144 | +8x | +
+ state_ids_local <- vapply(x$get_filter_state(), `[[`, character(1L), "id")+ |
+
145 | +8x | +
+ state_ids_allowed <- vapply(x$get_available_teal_slices()(), `[[`, character(1L), "id")+ |
+
146 | +8x | +
+ states_active <- state_ids_global %in% state_ids_local+ |
+
147 | +8x | +
+ ifelse(state_ids_global %in% state_ids_allowed, states_active, NA)+ |
+
148 | ++ |
+ })+ |
+
149 | ++ | + + | +
150 | +6x | +
+ as.data.frame(mapping_smooth, row.names = state_ids_global, check.names = FALSE)+ |
+
151 | ++ |
+ })+ |
+
152 | ++ | + + | +
153 | +6x | +
+ output$slices_table <- renderTable(+ |
+
154 | +6x | +
+ expr = {+ |
+
155 | ++ |
+ # Display logical values as UTF characters.+ |
+
156 | +3x | +
+ mm <- mapping_matrix()+ |
+
157 | +3x | +
+ mm[] <- lapply(mm, ifelse, yes = intToUtf8(9989), no = intToUtf8(10060))+ |
+
158 | +3x | +
+ mm[] <- lapply(mm, function(x) ifelse(is.na(x), intToUtf8(128306), x))+ |
+
159 | +3x | +
+ if (!is_module_specific) colnames(mm) <- "Global Filters"+ |
+
160 | ++ | + + | +
161 | ++ |
+ # Display placeholder if no filters defined.+ |
+
162 | +3x | +
+ if (nrow(mm) == 0L) {+ |
+
163 | +3x | +
+ mm <- data.frame(`Filter manager` = "No filters specified.", check.names = FALSE)+ |
+
164 | +3x | +
+ rownames(mm) <- ""+ |
+
165 | ++ |
+ }+ |
+
166 | ++ | + + | +
167 | ++ |
+ # Report Previewer will not be displayed.+ |
+
168 | +3x | +
+ mm[names(mm) != "Report previewer"]+ |
+
169 | ++ |
+ },+ |
+
170 | +6x | +
+ align = paste(c("l", rep("c", sum(names(filtered_data_list) != "Report previewer"))), collapse = ""),+ |
+
171 | +6x | +
+ rownames = TRUE+ |
+
172 | ++ |
+ )+ |
+
173 | ++ | + + | +
174 | ++ |
+ # Create list of module calls.+ |
+
175 | +6x | +
+ modules_out <- lapply(names(filtered_data_list), function(module_name) {+ |
+
176 | +8x | +
+ filter_manager_module_srv(+ |
+
177 | +8x | +
+ id = module_name,+ |
+
178 | +8x | +
+ module_fd = filtered_data_list[[module_name]],+ |
+
179 | +8x | +
+ slices_global = slices_global+ |
+
180 | ++ |
+ )+ |
+
181 | ++ |
+ })+ |
+
182 | ++ | + + | +
183 | ++ |
+ # Call snapshot manager.+ |
+
184 | +6x | +
+ snapshot_manager_srv("snapshot_manager", slices_global, mapping_matrix, filtered_data_list)+ |
+
185 | ++ | + + | +
186 | +6x | +
+ modules_out # returned for testing purpose+ |
+
187 | ++ |
+ })+ |
+
188 | ++ |
+ }+ |
+
189 | ++ | + + | +
190 | ++ |
+ #' Module specific filter manager+ |
+
191 | ++ |
+ #'+ |
+
192 | ++ |
+ #' Track filter states in single module.+ |
+
193 | ++ |
+ #'+ |
+
194 | ++ |
+ #' This module tracks the state of a single `FilteredData` object and global `teal_slices`+ |
+
195 | ++ |
+ #' and updates both objects as necessary. Filter states added in different modules+ |
+
196 | ++ |
+ #' Filter states added any individual module are added to global `teal_slices`+ |
+
197 | ++ |
+ #' and from there become available in other modules+ |
+
198 | ++ |
+ #' by setting `private$available_teal_slices` in each `FilteredData`.+ |
+
199 | ++ |
+ #'+ |
+
200 | ++ |
+ #' @param id (`character(1)`)\cr+ |
+
201 | ++ |
+ #' `shiny` module id.+ |
+
202 | ++ |
+ #' @param module_fd (`FilteredData`)\cr+ |
+
203 | ++ |
+ #' object to filter data in the teal-module+ |
+
204 | ++ |
+ #' @param slices_global (`reactiveVal`)\cr+ |
+
205 | ++ |
+ #' stores `teal_slices` with all available filters; allows the following actions:+ |
+
206 | ++ |
+ #' - to disable/enable a specific filter in a module+ |
+
207 | ++ |
+ #' - to restore saved filter settings+ |
+
208 | ++ |
+ #' - to save current filter panel settings+ |
+
209 | ++ |
+ #' @return A `reactive` expression containing the slices active in this module.+ |
+
210 | ++ |
+ #' @keywords internal+ |
+
211 | ++ |
+ #'+ |
+
212 | ++ |
+ filter_manager_module_srv <- function(id, module_fd, slices_global) {+ |
+
213 | +8x | +
+ moduleServer(id, function(input, output, session) {+ |
+
214 | ++ |
+ # Only operate on slices that refer to data sets present in this module.+ |
+
215 | +8x | +
+ module_fd$set_available_teal_slices(reactive(slices_global()))+ |
+
216 | ++ | + + | +
217 | ++ |
+ # Track filter state of this module.+ |
+
218 | +8x | +
+ slices_module <- reactive(module_fd$get_filter_state())+ |
+
219 | ++ | + + | +
220 | ++ |
+ # Reactive values for comparing states.+ |
+
221 | +8x | +
+ previous_slices <- reactiveVal(isolate(slices_module()))+ |
+
222 | +8x | +
+ slices_added <- reactiveVal(NULL)+ |
+
223 | ++ | + + | +
224 | ++ |
+ # Observe changes in module filter state and trigger appropriate actions.+ |
+
225 | +8x | +
+ observeEvent(slices_module(), ignoreNULL = FALSE, {+ |
+
226 | +3x | +
+ logger::log_trace("filter_manager_srv@1 detecting states deltas in module: { id }.")+ |
+
227 | +3x | +
+ added <- setdiff_teal_slices(slices_module(), slices_global())+ |
+
228 | +! | +
+ if (length(added)) slices_added(added)+ |
+
229 | +3x | +
+ previous_slices(slices_module())+ |
+
230 | ++ |
+ })+ |
+
231 | ++ | + + | +
232 | +8x | +
+ observeEvent(slices_added(), ignoreNULL = TRUE, {+ |
+
233 | +! | +
+ logger::log_trace("filter_manager_srv@2 added filter in module: { id }.")+ |
+
234 | ++ |
+ # In case the new state has the same id as an existing state, add a suffix to it.+ |
+
235 | +! | +
+ global_ids <- vapply(slices_global(), `[[`, character(1L), "id")+ |
+
236 | +! | +
+ lapply(+ |
+
237 | +! | +
+ slices_added(),+ |
+
238 | +! | +
+ function(slice) {+ |
+
239 | +! | +
+ if (slice$id %in% global_ids) {+ |
+
240 | +! | +
+ slice$id <- utils::tail(make.unique(c(global_ids, slice$id), sep = "_"), 1)+ |
+
241 | ++ |
+ }+ |
+
242 | ++ |
+ }+ |
+
243 | ++ |
+ )+ |
+
244 | +! | +
+ slices_global_new <- c(slices_global(), slices_added())+ |
+
245 | +! | +
+ slices_global(slices_global_new)+ |
+
246 | +! | +
+ slices_added(NULL)+ |
+
247 | ++ |
+ })+ |
+
248 | ++ | + + | +
249 | +8x | +
+ slices_module # returned for testing purpose+ |
+
250 | ++ |
+ })+ |
+
251 | ++ |
+ }+ |
+
1 | ++ |
+ #' Include `CSS` files from `/inst/css/` package directory to application header+ |
+
2 | ++ |
+ #'+ |
+
3 | ++ |
+ #' `system.file` should not be used to access files in other packages, it does+ |
+
4 | ++ |
+ #' not work with `devtools`. Therefore, we redefine this method in each package+ |
+
5 | ++ |
+ #' as needed. Thus, we do not export this method+ |
+
6 | ++ |
+ #'+ |
+
7 | ++ |
+ #' @param pattern (`character`) pattern of files to be included+ |
+
8 | ++ |
+ #'+ |
+
9 | ++ |
+ #' @return HTML code that includes `CSS` files+ |
+
10 | ++ |
+ #' @keywords internal+ |
+
11 | ++ |
+ include_css_files <- function(pattern = "*") {+ |
+
12 | +22x | +
+ css_files <- list.files(+ |
+
13 | +22x | +
+ system.file("css", package = "teal", mustWork = TRUE),+ |
+
14 | +22x | +
+ pattern = pattern, full.names = TRUE+ |
+
15 | ++ |
+ )+ |
+
16 | +22x | +
+ return(+ |
+
17 | +22x | +
+ shiny::singleton(+ |
+
18 | +22x | +
+ shiny::tags$head(lapply(css_files, shiny::includeCSS))+ |
+
19 | ++ |
+ )+ |
+
20 | ++ |
+ )+ |
+
21 | ++ |
+ }+ |
+
22 | ++ | + + | +
23 | ++ |
+ #' Include `JS` files from `/inst/js/` package directory to application header+ |
+
24 | ++ |
+ #'+ |
+
25 | ++ |
+ #' `system.file` should not be used to access files in other packages, it does+ |
+
26 | ++ |
+ #' not work with `devtools`. Therefore, we redefine this method in each package+ |
+
27 | ++ |
+ #' as needed. Thus, we do not export this method+ |
+
28 | ++ |
+ #'+ |
+
29 | ++ |
+ #' @param pattern (`character`) pattern of files to be included, passed to `system.file`+ |
+
30 | ++ |
+ #' @param except (`character`) vector of basename filenames to be excluded+ |
+
31 | ++ |
+ #'+ |
+
32 | ++ |
+ #' @return HTML code that includes `JS` files+ |
+
33 | ++ |
+ #' @keywords internal+ |
+
34 | ++ |
+ include_js_files <- function(pattern = NULL, except = NULL) {+ |
+
35 | +22x | +
+ checkmate::assert_character(except, min.len = 1, any.missing = FALSE, null.ok = TRUE)+ |
+
36 | +22x | +
+ js_files <- list.files(system.file("js", package = "teal", mustWork = TRUE), pattern = pattern, full.names = TRUE)+ |
+
37 | +22x | +
+ js_files <- js_files[!(basename(js_files) %in% except)] # no-op if except is NULL+ |
+
38 | ++ | + + | +
39 | +22x | +
+ return(singleton(lapply(js_files, includeScript)))+ |
+
40 | ++ |
+ }+ |
+
41 | ++ | + + | +
42 | ++ |
+ #' Run `JS` file from `/inst/js/` package directory+ |
+
43 | ++ |
+ #'+ |
+
44 | ++ |
+ #' This is triggered from the server to execute on the client+ |
+
45 | ++ |
+ #' rather than triggered directly on the client.+ |
+
46 | ++ |
+ #' Unlike `include_js_files` which includes `JavaScript` functions,+ |
+
47 | ++ |
+ #' the `run_js` actually executes `JavaScript` functions.+ |
+
48 | ++ |
+ #'+ |
+
49 | ++ |
+ #' `system.file` should not be used to access files in other packages, it does+ |
+
50 | ++ |
+ #' not work with `devtools`. Therefore, we redefine this method in each package+ |
+
51 | ++ |
+ #' as needed. Thus, we do not export this method+ |
+
52 | ++ |
+ #'+ |
+
53 | ++ |
+ #' @param files (`character`) vector of filenames+ |
+
54 | ++ |
+ #' @keywords internal+ |
+
55 | ++ |
+ run_js_files <- function(files) {+ |
+
56 | +16x | +
+ checkmate::assert_character(files, min.len = 1, any.missing = FALSE)+ |
+
57 | +16x | +
+ lapply(files, function(file) {+ |
+
58 | +16x | +
+ shinyjs::runjs(paste0(readLines(system.file("js", file, package = "teal", mustWork = TRUE)), collapse = "\n"))+ |
+
59 | ++ |
+ })+ |
+
60 | +16x | +
+ return(invisible(NULL))+ |
+
61 | ++ |
+ }+ |
+
62 | ++ | + + | +
63 | ++ |
+ #' Code to include teal `CSS` and `JavaScript` files+ |
+
64 | ++ |
+ #'+ |
+
65 | ++ |
+ #' This is useful when you want to use the same `JavaScript` and `CSS` files that are+ |
+
66 | ++ |
+ #' used with the teal application.+ |
+
67 | ++ |
+ #' This is also useful for running standalone modules in teal with the correct+ |
+
68 | ++ |
+ #' styles.+ |
+
69 | ++ |
+ #' Also initializes `shinyjs` so you can use it.+ |
+
70 | ++ |
+ #'+ |
+
71 | ++ |
+ #' @return HTML code to include+ |
+
72 | ++ |
+ #' @examples+ |
+
73 | ++ |
+ #' shiny_ui <- tagList(+ |
+
74 | ++ |
+ #' teal:::include_teal_css_js(),+ |
+
75 | ++ |
+ #' p("Hello")+ |
+
76 | ++ |
+ #' )+ |
+
77 | ++ |
+ #' @keywords internal+ |
+
78 | ++ |
+ include_teal_css_js <- function() {+ |
+
79 | +22x | +
+ tagList(+ |
+
80 | +22x | +
+ shinyjs::useShinyjs(),+ |
+
81 | +22x | +
+ include_css_files(),+ |
+
82 | ++ |
+ # init.js is executed from the server+ |
+
83 | +22x | +
+ include_js_files(except = "init.js"),+ |
+
84 | +22x | +
+ shinyjs::hidden(icon("gear")), # add hidden icon to load font-awesome css for icons+ |
+
85 | ++ |
+ )+ |
+
86 | ++ |
+ }+ |
+
1 | ++ |
+ # This is the main function from teal to be used by the end-users. Although it delegates+ |
+
2 | ++ |
+ # directly to `module_teal_with_splash.R`, we keep it in a separate file because its doc is quite large+ |
+
3 | ++ |
+ # and it is very end-user oriented. It may also perform more argument checking with more informative+ |
+
4 | ++ |
+ # error messages.+ |
+
5 | ++ | + + | +
6 | ++ | + + | +
7 | ++ |
+ #' Create the Server and UI Function For the Shiny App+ |
+
8 | ++ |
+ #'+ |
+
9 | ++ |
+ #' @description `r lifecycle::badge("stable")`+ |
+
10 | ++ |
+ #' End-users: This is the most important function for you to start a+ |
+
11 | ++ |
+ #' teal app that is composed out of teal modules.+ |
+
12 | ++ |
+ #'+ |
+
13 | ++ |
+ #' @param data (`TealData` or `TealDataset` or `TealDatasetConnector` or `list` or `data.frame`+ |
+
14 | ++ |
+ #' or `MultiAssayExperiment`, `teal_data`, `teal_data_module`)\cr+ |
+
15 | ++ |
+ #' `R6` object as returned by [teal.data::cdisc_data()], [teal.data::teal_data()],+ |
+
16 | ++ |
+ #' [teal.data::cdisc_dataset()], [teal.data::dataset()], [teal.data::dataset_connector()] or+ |
+
17 | ++ |
+ #' [teal.data::cdisc_dataset_connector()] or [teal_data_module()] or a single `data.frame` or+ |
+
18 | ++ |
+ #' a `MultiAssayExperiment`+ |
+
19 | ++ |
+ #' or a list of the previous objects or function returning a named list.+ |
+
20 | ++ |
+ #' NOTE: teal does not guarantee reproducibility of the code when names of the list elements+ |
+
21 | ++ |
+ #' do not match the original object names. To ensure reproducibility please use [teal.data::teal_data()]+ |
+
22 | ++ |
+ #' or [teal.data::cdisc_data()] with `check = TRUE` enabled.+ |
+
23 | ++ |
+ #' @param modules (`list`, `teal_modules` or `teal_module`)\cr+ |
+
24 | ++ |
+ #' nested list of `teal_modules` or `teal_module` objects or a single+ |
+
25 | ++ |
+ #' `teal_modules` or `teal_module` object. These are the specific output modules which+ |
+
26 | ++ |
+ #' will be displayed in the teal application. See [modules()] and [module()] for+ |
+
27 | ++ |
+ #' more details.+ |
+
28 | ++ |
+ #' @param title (`NULL` or `character`)\cr+ |
+
29 | ++ |
+ #' The browser window title (defaults to the host URL of the page).+ |
+
30 | ++ |
+ #' @param filter (`teal_slices`)\cr+ |
+
31 | ++ |
+ #' Specification of initial filter. Filters can be specified using [teal::teal_slices()].+ |
+
32 | ++ |
+ #' Old way of specifying filters through a list is deprecated and will be removed in the+ |
+
33 | ++ |
+ #' next release. Please fix your applications to use [teal::teal_slices()].+ |
+
34 | ++ |
+ #' @param header (`shiny.tag` or `character`) \cr+ |
+
35 | ++ |
+ #' the header of the app. Note shiny code placed here (and in the footer+ |
+
36 | ++ |
+ #' argument) will be placed in the app's `ui` function so code which needs to be placed in the `ui` function+ |
+
37 | ++ |
+ #' (such as loading `CSS` via [htmltools::htmlDependency()]) should be included here.+ |
+
38 | ++ |
+ #' @param footer (`shiny.tag` or `character`)\cr+ |
+
39 | ++ |
+ #' the footer of the app+ |
+
40 | ++ |
+ #' @param id (`character`)\cr+ |
+
41 | ++ |
+ #' module id to embed it, if provided,+ |
+
42 | ++ |
+ #' the server function must be called with [shiny::moduleServer()];+ |
+
43 | ++ |
+ #' See the vignette for an example. However, [ui_teal_with_splash()]+ |
+
44 | ++ |
+ #' is then preferred to this function.+ |
+
45 | ++ |
+ #'+ |
+
46 | ++ |
+ #' @return named list with `server` and `ui` function+ |
+
47 | ++ |
+ #'+ |
+
48 | ++ |
+ #' @export+ |
+
49 | ++ |
+ #'+ |
+
50 | ++ |
+ #' @include modules.R+ |
+
51 | ++ |
+ #'+ |
+
52 | ++ |
+ #' @examples+ |
+
53 | ++ |
+ #' new_iris <- transform(iris, id = seq_len(nrow(iris)))+ |
+
54 | ++ |
+ #' new_mtcars <- transform(mtcars, id = seq_len(nrow(mtcars)))+ |
+
55 | ++ |
+ #'+ |
+
56 | ++ |
+ #' app <- init(+ |
+
57 | ++ |
+ #' data = teal_data(+ |
+
58 | ++ |
+ #' dataset("new_iris", new_iris),+ |
+
59 | ++ |
+ #' dataset("new_mtcars", new_mtcars),+ |
+
60 | ++ |
+ #' code = "+ |
+
61 | ++ |
+ #' new_iris <- transform(iris, id = seq_len(nrow(iris)))+ |
+
62 | ++ |
+ #' new_mtcars <- transform(mtcars, id = seq_len(nrow(mtcars)))+ |
+
63 | ++ |
+ #' "+ |
+
64 | ++ |
+ #' ),+ |
+
65 | ++ |
+ #' modules = modules(+ |
+
66 | ++ |
+ #' module(+ |
+
67 | ++ |
+ #' label = "data source",+ |
+
68 | ++ |
+ #' server = function(input, output, session, data) {},+ |
+
69 | ++ |
+ #' ui = function(id, ...) div(p("information about data source")),+ |
+
70 | ++ |
+ #' datanames = "all"+ |
+
71 | ++ |
+ #' ),+ |
+
72 | ++ |
+ #' example_module(label = "example teal module"),+ |
+
73 | ++ |
+ #' module(+ |
+
74 | ++ |
+ #' "Iris Sepal.Length histogram",+ |
+
75 | ++ |
+ #' server = function(input, output, session, data) {+ |
+
76 | ++ |
+ #' output$hist <- renderPlot(+ |
+
77 | ++ |
+ #' hist(data[["new_iris"]]()$Sepal.Length)+ |
+
78 | ++ |
+ #' )+ |
+
79 | ++ |
+ #' },+ |
+
80 | ++ |
+ #' ui = function(id, ...) {+ |
+
81 | ++ |
+ #' ns <- NS(id)+ |
+
82 | ++ |
+ #' plotOutput(ns("hist"))+ |
+
83 | ++ |
+ #' },+ |
+
84 | ++ |
+ #' datanames = "new_iris"+ |
+
85 | ++ |
+ #' )+ |
+
86 | ++ |
+ #' ),+ |
+
87 | ++ |
+ #' title = "App title",+ |
+
88 | ++ |
+ #' filter = teal_slices(+ |
+
89 | ++ |
+ #' teal_slice(dataname = "new_iris", varname = "Species"),+ |
+
90 | ++ |
+ #' teal_slice(dataname = "new_iris", varname = "Sepal.Length"),+ |
+
91 | ++ |
+ #' teal_slice(dataname = "new_mtcars", varname = "cyl"),+ |
+
92 | ++ |
+ #' exclude_varnames = list(new_iris = c("Sepal.Width", "Petal.Width")),+ |
+
93 | ++ |
+ #' mapping = list(+ |
+
94 | ++ |
+ #' `example teal module` = "new_iris Species",+ |
+
95 | ++ |
+ #' `Iris Sepal.Length histogram` = "new_iris Species",+ |
+
96 | ++ |
+ #' global_filters = "new_mtcars cyl"+ |
+
97 | ++ |
+ #' )+ |
+
98 | ++ |
+ #' ),+ |
+
99 | ++ |
+ #' header = tags$h1("Sample App"),+ |
+
100 | ++ |
+ #' footer = tags$p("Copyright 2017 - 2023")+ |
+
101 | ++ |
+ #' )+ |
+
102 | ++ |
+ #' if (interactive()) {+ |
+
103 | ++ |
+ #' shinyApp(app$ui, app$server)+ |
+
104 | ++ |
+ #' }+ |
+
105 | ++ |
+ #'+ |
+
106 | ++ |
+ init <- function(data,+ |
+
107 | ++ |
+ modules,+ |
+
108 | ++ |
+ title = NULL,+ |
+
109 | ++ |
+ filter = teal_slices(),+ |
+
110 | ++ |
+ header = tags$p(),+ |
+
111 | ++ |
+ footer = tags$p(),+ |
+
112 | ++ |
+ id = character(0)) {+ |
+
113 | +30x | +
+ logger::log_trace("init initializing teal app with: data ({ class(data)[1] }).")+ |
+
114 | +30x | +
+ if (!inherits(data, c("TealData", "teal_data", "teal_data_module"))) {+ |
+
115 | +24x | +
+ data <- teal.data::to_relational_data(data = data)+ |
+
116 | ++ |
+ }+ |
+
117 | ++ | + + | +
118 | +25x | +
+ checkmate::assert_multi_class(data, c("TealData", "teal_data", "teal_data_module"))+ |
+
119 | +25x | +
+ checkmate::assert_multi_class(modules, c("teal_module", "list", "teal_modules"))+ |
+
120 | +25x | +
+ checkmate::assert_string(title, null.ok = TRUE)+ |
+
121 | +25x | +
+ checkmate::assert(+ |
+
122 | +25x | +
+ checkmate::check_class(filter, "teal_slices"),+ |
+
123 | +25x | +
+ checkmate::check_list(filter, names = "named")+ |
+
124 | ++ |
+ )+ |
+
125 | +24x | +
+ checkmate::assert_multi_class(header, c("shiny.tag", "character"))+ |
+
126 | +24x | +
+ checkmate::assert_multi_class(footer, c("shiny.tag", "character"))+ |
+
127 | +24x | +
+ checkmate::assert_character(id, max.len = 1, any.missing = FALSE)+ |
+
128 | ++ | + + | +
129 | +24x | +
+ teal.logger::log_system_info()+ |
+
130 | ++ | + + | +
131 | +24x | +
+ if (inherits(modules, "teal_module")) {+ |
+
132 | +1x | +
+ modules <- list(modules)+ |
+
133 | ++ |
+ }+ |
+
134 | +24x | +
+ if (inherits(modules, "list")) {+ |
+
135 | +4x | +
+ modules <- do.call(teal::modules, modules)+ |
+
136 | ++ |
+ }+ |
+
137 | ++ | + + | +
138 | +24x | +
+ landing <- extract_module(modules, "teal_module_landing")+ |
+
139 | +! | +
+ if (length(landing) > 1L) stop("Only one `landing_popup_module` can be used.")+ |
+
140 | +24x | +
+ modules <- drop_module(modules, "teal_module_landing")+ |
+
141 | ++ | + + | +
142 | ++ |
+ # Calculate app id that will be used to stamp filter state snapshots.+ |
+
143 | ++ |
+ # App id is a hash of the app's data and modules.+ |
+
144 | ++ |
+ # See "transferring snapshots" section in ?snapshot.+ |
+
145 | +24x | +
+ hashables <- mget(c("data", "modules"))+ |
+
146 | +24x | +
+ hashables$data <- if (inherits(hashables$data, "teal_data")) {+ |
+
147 | +4x | +
+ as.list(hashables$data@env)+ |
+
148 | +24x | +
+ } else if (inherits(data, "teal_data_module")) {+ |
+
149 | +1x | +
+ body(data$server)+ |
+
150 | +24x | +
+ } else if (hashables$data$is_pulled()) {+ |
+
151 | +17x | +
+ sapply(get_dataname(hashables$data), simplify = FALSE, function(dn) {+ |
+
152 | +23x | +
+ hashables$data$get_dataset(dn)$get_raw_data()+ |
+
153 | ++ |
+ })+ |
+
154 | ++ |
+ } else {+ |
+
155 | +2x | +
+ hashables$data$get_code()+ |
+
156 | ++ |
+ }+ |
+
157 | ++ | + + | +
158 | +24x | +
+ attr(filter, "app_id") <- rlang::hash(hashables)+ |
+
159 | ++ | + + | +
160 | ++ |
+ # convert teal.slice::teal_slices to teal::teal_slices+ |
+
161 | +24x | +
+ filter <- as.teal_slices(as.list(filter))+ |
+
162 | ++ | + + | +
163 | +24x | +
+ if (isTRUE(attr(filter, "module_specific"))) {+ |
+
164 | +! | +
+ module_names <- unlist(c(module_labels(modules), "global_filters"))+ |
+
165 | +! | +
+ failed_mod_names <- setdiff(names(attr(filter, "mapping")), module_names)+ |
+
166 | +! | +
+ if (length(failed_mod_names)) {+ |
+
167 | +! | +
+ stop(+ |
+
168 | +! | +
+ sprintf(+ |
+
169 | +! | +
+ "Some module names in the mapping arguments don't match module labels.\n %s not in %s",+ |
+
170 | +! | +
+ toString(failed_mod_names),+ |
+
171 | +! | +
+ toString(unique(module_names))+ |
+
172 | ++ |
+ )+ |
+
173 | ++ |
+ )+ |
+
174 | ++ |
+ }+ |
+
175 | ++ | + + | +
176 | +! | +
+ if (anyDuplicated(module_names)) {+ |
+
177 | ++ |
+ # In teal we are able to set nested modules with duplicated label.+ |
+
178 | ++ |
+ # Because mapping argument bases on the relationship between module-label and filter-id,+ |
+
179 | ++ |
+ # it is possible that module-label in mapping might refer to multiple teal_module (identified by the same label)+ |
+
180 | +! | +
+ stop(+ |
+
181 | +! | +
+ sprintf(+ |
+
182 | +! | +
+ "Module labels should be unique when teal_slices(mapping = TRUE). Duplicated labels:\n%s ",+ |
+
183 | +! | +
+ toString(module_names[duplicated(module_names)])+ |
+
184 | ++ |
+ )+ |
+
185 | ++ |
+ )+ |
+
186 | ++ |
+ }+ |
+
187 | ++ |
+ }+ |
+
188 | ++ | + + | +
189 | +24x | +
+ if (inherits(data, "teal_data")) {+ |
+
190 | +4x | +
+ if (length(teal.data::datanames(data)) == 0) {+ |
+
191 | +1x | +
+ stop("`data` object has no datanames. Specify `datanames(data)` and try again.")+ |
+
192 | ++ |
+ }+ |
+
193 | ++ | + + | +
194 | ++ |
+ # in case of teal_data_module this check is postponed to the srv_teal_with_splash+ |
+
195 | +3x | +
+ is_modules_ok <- check_modules_datanames(modules, teal.data::datanames(data))+ |
+
196 | +3x | +
+ if (!isTRUE(is_modules_ok)) {+ |
+
197 | +1x | +
+ logger::log_error(is_modules_ok)+ |
+
198 | +1x | +
+ checkmate::assert(is_modules_ok, .var.name = "modules")+ |
+
199 | ++ |
+ }+ |
+
200 | ++ | + + | +
201 | ++ | + + | +
202 | +2x | +
+ is_filter_ok <- check_filter_datanames(filter, teal.data::datanames(data))+ |
+
203 | +2x | +
+ if (!isTRUE(is_filter_ok)) {+ |
+
204 | +1x | +
+ logger::log_warn(is_filter_ok)+ |
+
205 | ++ |
+ # we allow app to continue if applied filters are outside+ |
+
206 | ++ |
+ # of possible data range+ |
+
207 | ++ |
+ }+ |
+
208 | ++ |
+ }+ |
+
209 | ++ | + + | +
210 | ++ |
+ # Note regarding case `id = character(0)`:+ |
+
211 | ++ |
+ # rather than using `callModule` and creating a submodule of this module, we directly modify+ |
+
212 | ++ |
+ # the `ui` and `server` with `id = character(0)` and calling the server function directly+ |
+
213 | ++ |
+ # rather than through `callModule`+ |
+
214 | +22x | +
+ res <- list(+ |
+
215 | +22x | +
+ ui = ui_teal_with_splash(id = id, data = data, title = title, header = header, footer = footer),+ |
+
216 | +22x | +
+ server = function(input, output, session) {+ |
+
217 | +! | +
+ if (length(landing) == 1L) {+ |
+
218 | +! | +
+ landing_module <- landing[[1L]]+ |
+
219 | +! | +
+ do.call(landing_module$server, c(list(id = "landing_module_shiny_id"), landing_module$server_args))+ |
+
220 | ++ |
+ }+ |
+
221 | +! | +
+ if (inherits(data, "TealDataAbstract")) {+ |
+
222 | ++ |
+ # copy TealData so that load won't be shared between the session+ |
+
223 | +! | +
+ data <- data$copy(deep = TRUE)+ |
+
224 | ++ |
+ }+ |
+
225 | +! | +
+ filter <- deep_copy_filter(filter)+ |
+
226 | +! | +
+ srv_teal_with_splash(id = id, data = data, modules = modules, filter = filter)+ |
+
227 | ++ |
+ }+ |
+
228 | ++ |
+ )+ |
+
229 | +22x | +
+ logger::log_trace("init teal app has been initialized.")+ |
+
230 | +22x | +
+ return(res)+ |
+
231 | ++ |
+ }+ |
+
1 | ++ |
+ #' Create a UI of nested tabs of `teal_modules`+ |
+
2 | ++ |
+ #'+ |
+
3 | ++ |
+ #' @section `ui_nested_tabs`:+ |
+
4 | ++ |
+ #' Each `teal_modules` is translated to a `tabsetPanel` and each+ |
+
5 | ++ |
+ #' of its children is another tab-module called recursively. The UI of a+ |
+
6 | ++ |
+ #' `teal_module` is obtained by calling the `ui` function on it.+ |
+
7 | ++ |
+ #'+ |
+
8 | ++ |
+ #' The `datasets` argument is required to resolve the teal arguments in an+ |
+
9 | ++ |
+ #' isolated context (with respect to reactivity)+ |
+
10 | ++ |
+ #'+ |
+
11 | ++ |
+ #' @section `srv_nested_tabs`:+ |
+
12 | ++ |
+ #' This module calls recursively all elements of the `modules` returns one which+ |
+
13 | ++ |
+ #' is currently active.+ |
+
14 | ++ |
+ #' - `teal_module` returns self as a active module.+ |
+
15 | ++ |
+ #' - `teal_modules` also returns module active within self which is determined by the `input$active_tab`.+ |
+
16 | ++ |
+ #'+ |
+
17 | ++ |
+ #' @name module_nested_tabs+ |
+
18 | ++ |
+ #'+ |
+
19 | ++ |
+ #' @inheritParams module_tabs_with_filters+ |
+
20 | ++ |
+ #'+ |
+
21 | ++ |
+ #' @param depth (`integer(1)`)\cr+ |
+
22 | ++ |
+ #' number which helps to determine depth of the modules nesting.+ |
+
23 | ++ |
+ #' @param is_module_specific (`logical(1)`)\cr+ |
+
24 | ++ |
+ #' flag determining if the filter panel is global or module-specific.+ |
+
25 | ++ |
+ #' When set to `TRUE`, a filter panel is called inside of each module tab.+ |
+
26 | ++ |
+ #' @return depending on class of `modules`, `ui_nested_tabs` returns:+ |
+
27 | ++ |
+ #' - `teal_module`: instantiated UI of the module+ |
+
28 | ++ |
+ #' - `teal_modules`: `tabsetPanel` with each tab corresponding to recursively+ |
+
29 | ++ |
+ #' calling this function on it.\cr+ |
+
30 | ++ |
+ #' `srv_nested_tabs` returns a reactive which returns the active module that corresponds to the selected tab.+ |
+
31 | ++ |
+ #'+ |
+
32 | ++ |
+ #' @examples+ |
+
33 | ++ |
+ #' mods <- teal:::example_modules()+ |
+
34 | ++ |
+ #' datasets <- teal:::example_datasets()+ |
+
35 | ++ |
+ #' app <- shinyApp(+ |
+
36 | ++ |
+ #' ui = function() {+ |
+
37 | ++ |
+ #' tagList(+ |
+
38 | ++ |
+ #' teal:::include_teal_css_js(),+ |
+
39 | ++ |
+ #' textOutput("info"),+ |
+
40 | ++ |
+ #' fluidPage( # needed for nice tabs+ |
+
41 | ++ |
+ #' teal:::ui_nested_tabs("dummy", modules = mods, datasets = datasets)+ |
+
42 | ++ |
+ #' )+ |
+
43 | ++ |
+ #' )+ |
+
44 | ++ |
+ #' },+ |
+
45 | ++ |
+ #' server = function(input, output, session) {+ |
+
46 | ++ |
+ #' active_module <- teal:::srv_nested_tabs(+ |
+
47 | ++ |
+ #' "dummy",+ |
+
48 | ++ |
+ #' datasets = datasets,+ |
+
49 | ++ |
+ #' modules = mods+ |
+
50 | ++ |
+ #' )+ |
+
51 | ++ |
+ #' output$info <- renderText({+ |
+
52 | ++ |
+ #' paste0("The currently active tab name is ", active_module()$label)+ |
+
53 | ++ |
+ #' })+ |
+
54 | ++ |
+ #' }+ |
+
55 | ++ |
+ #' )+ |
+
56 | ++ |
+ #' if (interactive()) {+ |
+
57 | ++ |
+ #' shinyApp(app$ui, app$server)+ |
+
58 | ++ |
+ #' }+ |
+
59 | ++ |
+ #' @keywords internal+ |
+
60 | ++ |
+ NULL+ |
+
61 | ++ | + + | +
62 | ++ |
+ #' @rdname module_nested_tabs+ |
+
63 | ++ |
+ ui_nested_tabs <- function(id, modules, datasets, depth = 0L, is_module_specific = FALSE) {+ |
+
64 | +2x | +
+ checkmate::assert_multi_class(modules, c("teal_modules", "teal_module"))+ |
+
65 | +2x | +
+ checkmate::assert_count(depth)+ |
+
66 | +2x | +
+ UseMethod("ui_nested_tabs", modules)+ |
+
67 | ++ |
+ }+ |
+
68 | ++ | + + | +
69 | ++ |
+ #' @rdname module_nested_tabs+ |
+
70 | ++ |
+ #' @export+ |
+
71 | ++ |
+ ui_nested_tabs.default <- function(id, modules, datasets, depth = 0L, is_module_specific = FALSE) {+ |
+
72 | +! | +
+ stop("Modules class not supported: ", paste(class(modules), collapse = " "))+ |
+
73 | ++ |
+ }+ |
+
74 | ++ | + + | +
75 | ++ |
+ #' @rdname module_nested_tabs+ |
+
76 | ++ |
+ #' @export+ |
+
77 | ++ |
+ ui_nested_tabs.teal_modules <- function(id, modules, datasets, depth = 0L, is_module_specific = FALSE) {+ |
+
78 | +1x | +
+ checkmate::assert_list(datasets, types = c("list", "FilteredData"))+ |
+
79 | +1x | +
+ ns <- NS(id)+ |
+
80 | +1x | +
+ do.call(+ |
+
81 | +1x | +
+ tabsetPanel,+ |
+
82 | +1x | +
+ c(+ |
+
83 | ++ |
+ # by giving an id, we can reactively respond to tab changes+ |
+
84 | +1x | +
+ list(+ |
+
85 | +1x | +
+ id = ns("active_tab"),+ |
+
86 | +1x | +
+ type = if (modules$label == "root") "pills" else "tabs"+ |
+
87 | ++ |
+ ),+ |
+
88 | +1x | +
+ lapply(+ |
+
89 | +1x | +
+ names(modules$children),+ |
+
90 | +1x | +
+ function(module_id) {+ |
+
91 | +1x | +
+ module_label <- modules$children[[module_id]]$label+ |
+
92 | +1x | +
+ tabPanel(+ |
+
93 | +1x | +
+ title = module_label,+ |
+
94 | +1x | +
+ value = module_id, # when clicked this tab value changes input$<tabset panel id>+ |
+
95 | +1x | +
+ ui_nested_tabs(+ |
+
96 | +1x | +
+ id = ns(module_id),+ |
+
97 | +1x | +
+ modules = modules$children[[module_id]],+ |
+
98 | +1x | +
+ datasets = datasets[[module_label]],+ |
+
99 | +1x | +
+ depth = depth + 1L,+ |
+
100 | +1x | +
+ is_module_specific = is_module_specific+ |
+
101 | ++ |
+ )+ |
+
102 | ++ |
+ )+ |
+
103 | ++ |
+ }+ |
+
104 | ++ |
+ )+ |
+
105 | ++ |
+ )+ |
+
106 | ++ |
+ )+ |
+
107 | ++ |
+ }+ |
+
108 | ++ | + + | +
109 | ++ |
+ #' @rdname module_nested_tabs+ |
+
110 | ++ |
+ #' @export+ |
+
111 | ++ |
+ ui_nested_tabs.teal_module <- function(id, modules, datasets, depth = 0L, is_module_specific = FALSE) {+ |
+
112 | +1x | +
+ checkmate::assert_class(datasets, classes = "FilteredData")+ |
+
113 | +1x | +
+ ns <- NS(id)+ |
+
114 | ++ | + + | +
115 | +1x | +
+ args <- isolate(teal.transform::resolve_delayed(modules$ui_args, datasets))+ |
+
116 | +1x | +
+ args <- c(list(id = ns("module")), args)+ |
+
117 | ++ | + + | +
118 | +1x | +
+ if (is_arg_used(modules$ui, "datasets")) {+ |
+
119 | +! | +
+ args <- c(args, datasets = datasets)+ |
+
120 | ++ |
+ }+ |
+
121 | ++ | + + | +
122 | +1x | +
+ if (is_arg_used(modules$ui, "data")) {+ |
+
123 | +1x | +
+ data <- .datasets_to_data(modules, datasets)+ |
+
124 | +1x | +
+ args <- c(args, data = list(data))+ |
+
125 | ++ |
+ }+ |
+
126 | ++ | + + | +
127 | +1x | +
+ teal_ui <- tags$div(+ |
+
128 | +1x | +
+ id = id,+ |
+
129 | +1x | +
+ class = "teal_module",+ |
+
130 | +1x | +
+ uiOutput(ns("data_reactive"), inline = TRUE),+ |
+
131 | +1x | +
+ tagList(+ |
+
132 | +1x | +
+ if (depth >= 2L) div(style = "mt-6"),+ |
+
133 | +1x | +
+ do.call(modules$ui, args)+ |
+
134 | ++ |
+ )+ |
+
135 | ++ |
+ )+ |
+
136 | ++ | + + | +
137 | +1x | +
+ if (!is.null(modules$datanames) && is_module_specific) {+ |
+
138 | +! | +
+ fluidRow(+ |
+
139 | +! | +
+ column(width = 9, teal_ui, class = "teal_primary_col"),+ |
+
140 | +! | +
+ column(+ |
+
141 | +! | +
+ width = 3,+ |
+
142 | +! | +
+ datasets$ui_filter_panel(ns("module_filter_panel")),+ |
+
143 | +! | +
+ class = "teal_secondary_col"+ |
+
144 | ++ |
+ )+ |
+
145 | ++ |
+ )+ |
+
146 | ++ |
+ } else {+ |
+
147 | +1x | +
+ teal_ui+ |
+
148 | ++ |
+ }+ |
+
149 | ++ |
+ }+ |
+
150 | ++ | + + | +
151 | ++ |
+ #' @rdname module_nested_tabs+ |
+
152 | ++ |
+ srv_nested_tabs <- function(id, datasets, modules, is_module_specific = FALSE,+ |
+
153 | ++ |
+ reporter = teal.reporter::Reporter$new()) {+ |
+
154 | +54x | +
+ checkmate::assert_multi_class(modules, c("teal_modules", "teal_module"))+ |
+
155 | +54x | +
+ checkmate::assert_class(reporter, "Reporter")+ |
+
156 | +53x | +
+ UseMethod("srv_nested_tabs", modules)+ |
+
157 | ++ |
+ }+ |
+
158 | ++ | + + | +
159 | ++ |
+ #' @rdname module_nested_tabs+ |
+
160 | ++ |
+ #' @export+ |
+
161 | ++ |
+ srv_nested_tabs.default <- function(id, datasets, modules, is_module_specific = FALSE,+ |
+
162 | ++ |
+ reporter = teal.reporter::Reporter$new()) {+ |
+
163 | +! | +
+ stop("Modules class not supported: ", paste(class(modules), collapse = " "))+ |
+
164 | ++ |
+ }+ |
+
165 | ++ | + + | +
166 | ++ |
+ #' @rdname module_nested_tabs+ |
+
167 | ++ |
+ #' @export+ |
+
168 | ++ |
+ srv_nested_tabs.teal_modules <- function(id, datasets, modules, is_module_specific = FALSE,+ |
+
169 | ++ |
+ reporter = teal.reporter::Reporter$new()) {+ |
+
170 | +24x | +
+ checkmate::assert_list(datasets, types = c("list", "FilteredData"))+ |
+
171 | ++ | + + | +
172 | +24x | +
+ moduleServer(id = id, module = function(input, output, session) {+ |
+
173 | +24x | +
+ logger::log_trace("srv_nested_tabs.teal_modules initializing the module { deparse1(modules$label) }.")+ |
+
174 | ++ | + + | +
175 | +24x | +
+ labels <- vapply(modules$children, `[[`, character(1), "label")+ |
+
176 | +24x | +
+ modules_reactive <- sapply(+ |
+
177 | +24x | +
+ names(modules$children),+ |
+
178 | +24x | +
+ function(module_id) {+ |
+
179 | +35x | +
+ srv_nested_tabs(+ |
+
180 | +35x | +
+ id = module_id,+ |
+
181 | +35x | +
+ datasets = datasets[[labels[module_id]]],+ |
+
182 | +35x | +
+ modules = modules$children[[module_id]],+ |
+
183 | +35x | +
+ is_module_specific = is_module_specific,+ |
+
184 | +35x | +
+ reporter = reporter+ |
+
185 | ++ |
+ )+ |
+
186 | ++ |
+ },+ |
+
187 | +24x | +
+ simplify = FALSE+ |
+
188 | ++ |
+ )+ |
+
189 | ++ | + + | +
190 | ++ |
+ # when not ready input$active_tab would return NULL - this would fail next reactive+ |
+
191 | +24x | +
+ input_validated <- eventReactive(input$active_tab, input$active_tab, ignoreNULL = TRUE)+ |
+
192 | +24x | +
+ get_active_module <- reactive({+ |
+
193 | +13x | +
+ if (length(modules$children) == 1L) {+ |
+
194 | ++ |
+ # single tab is active by default+ |
+
195 | +2x | +
+ modules_reactive[[1]]()+ |
+
196 | ++ |
+ } else {+ |
+
197 | ++ |
+ # switch to active tab+ |
+
198 | +11x | +
+ modules_reactive[[input_validated()]]()+ |
+
199 | ++ |
+ }+ |
+
200 | ++ |
+ })+ |
+
201 | ++ | + + | +
202 | +24x | +
+ get_active_module+ |
+
203 | ++ |
+ })+ |
+
204 | ++ |
+ }+ |
+
205 | ++ | + + | +
206 | ++ |
+ #' @rdname module_nested_tabs+ |
+
207 | ++ |
+ #' @export+ |
+
208 | ++ |
+ srv_nested_tabs.teal_module <- function(id, datasets, modules, is_module_specific = TRUE,+ |
+
209 | ++ |
+ reporter = teal.reporter::Reporter$new()) {+ |
+
210 | +29x | +
+ checkmate::assert_class(datasets, "FilteredData")+ |
+
211 | +29x | +
+ logger::log_trace("srv_nested_tabs.teal_module initializing the module: { deparse1(modules$label) }.")+ |
+
212 | ++ | + + | +
213 | +29x | +
+ moduleServer(id = id, module = function(input, output, session) {+ |
+
214 | +29x | +
+ modules$server_args <- teal.transform::resolve_delayed(modules$server_args, datasets)+ |
+
215 | +29x | +
+ if (!is.null(modules$datanames) && is_module_specific) {+ |
+
216 | +! | +
+ datasets$srv_filter_panel("module_filter_panel")+ |
+
217 | ++ |
+ }+ |
+
218 | ++ | + + | +
219 | ++ |
+ # Create two triggers to limit reactivity between filter-panel and modules.+ |
+
220 | ++ |
+ # We want to recalculate only visible modules+ |
+
221 | ++ |
+ # - trigger the data when the tab is selected+ |
+
222 | ++ |
+ # - trigger module to be called when the tab is selected for the first time+ |
+
223 | +29x | +
+ trigger_data <- reactiveVal(1L)+ |
+
224 | +29x | +
+ trigger_module <- reactiveVal(NULL)+ |
+
225 | +29x | +
+ output$data_reactive <- renderUI({+ |
+
226 | +18x | +
+ lapply(datasets$datanames(), function(x) {+ |
+
227 | +22x | +
+ datasets$get_data(x, filtered = TRUE)+ |
+
228 | ++ |
+ })+ |
+
229 | +18x | +
+ isolate(trigger_data(trigger_data() + 1))+ |
+
230 | +18x | +
+ isolate(trigger_module(TRUE))+ |
+
231 | ++ | + + | +
232 | +18x | +
+ NULL+ |
+
233 | ++ |
+ })+ |
+
234 | ++ | + + | +
235 | ++ |
+ # collect arguments to run teal_module+ |
+
236 | +29x | +
+ args <- c(list(id = "module"), modules$server_args)+ |
+
237 | +29x | +
+ if (is_arg_used(modules$server, "reporter")) {+ |
+
238 | +! | +
+ args <- c(args, list(reporter = reporter))+ |
+
239 | ++ |
+ }+ |
+
240 | ++ | + + | +
241 | +29x | +
+ if (is_arg_used(modules$server, "datasets")) {+ |
+
242 | +2x | +
+ args <- c(args, datasets = datasets)+ |
+
243 | ++ |
+ }+ |
+
244 | ++ | + + | +
245 | +29x | +
+ if (is_arg_used(modules$server, "data")) {+ |
+
246 | +9x | +
+ data <- .datasets_to_data(modules, datasets, trigger_data)+ |
+
247 | +9x | +
+ args <- c(args, data = list(data))+ |
+
248 | ++ |
+ }+ |
+
249 | ++ | + + | +
250 | +29x | +
+ if (is_arg_used(modules$server, "filter_panel_api")) {+ |
+
251 | +2x | +
+ filter_panel_api <- teal.slice::FilterPanelAPI$new(datasets)+ |
+
252 | +2x | +
+ args <- c(args, filter_panel_api = filter_panel_api)+ |
+
253 | ++ |
+ }+ |
+
254 | ++ | + + | +
255 | +29x | +
+ if (is_arg_used(modules$server, "datasets") && is_arg_used(modules$server, "data")) {+ |
+
256 | +1x | +
+ warning(+ |
+
257 | +1x | +
+ "Module '", modules$label, "' has `data` and `datasets` arguments in the formals.",+ |
+
258 | +1x | +
+ "\nIt's recommended to use `data` to work with filtered objects."+ |
+
259 | ++ |
+ )+ |
+
260 | ++ |
+ }+ |
+
261 | ++ | + + | +
262 | ++ |
+ # observe the trigger_module above to induce the module once the renderUI is triggered+ |
+
263 | +29x | +
+ observeEvent(+ |
+
264 | +29x | +
+ ignoreNULL = TRUE,+ |
+
265 | +29x | +
+ once = TRUE,+ |
+
266 | +29x | +
+ eventExpr = trigger_module(),+ |
+
267 | +29x | +
+ handlerExpr = {+ |
+
268 | +18x | +
+ module_output <- if (is_arg_used(modules$server, "id")) {+ |
+
269 | +18x | +
+ do.call(modules$server, args)+ |
+
270 | ++ |
+ } else {+ |
+
271 | +! | +
+ do.call(callModule, c(args, list(module = modules$server)))+ |
+
272 | ++ |
+ }+ |
+
273 | ++ |
+ }+ |
+
274 | ++ |
+ )+ |
+
275 | ++ | + + | +
276 | +29x | +
+ reactive(modules)+ |
+
277 | ++ |
+ })+ |
+
278 | ++ |
+ }+ |
+
279 | ++ | + + | +
280 | ++ |
+ #' Convert `FilteredData` to reactive list of datasets of the `tdata` type.+ |
+
281 | ++ |
+ #'+ |
+
282 | ++ |
+ #' Converts `FilteredData` object to `tdata` object containing datasets needed for a specific module.+ |
+
283 | ++ |
+ #' Please note that if module needs dataset which has a parent, then parent will be also returned.+ |
+
284 | ++ |
+ #' A hash per `dataset` is calculated internally and returned in the code.+ |
+
285 | ++ |
+ #'+ |
+
286 | ++ |
+ #' @param module (`teal_module`) module where needed filters are taken from+ |
+
287 | ++ |
+ #' @param datasets (`FilteredData`) object where needed data are taken from+ |
+
288 | ++ |
+ #' @param trigger_data (`reactiveVal`) to trigger getting the filtered data+ |
+
289 | ++ |
+ #' @return list of reactive datasets with following attributes:+ |
+
290 | ++ |
+ #' - `code` (`character`) containing datasets reproducible code.+ |
+
291 | ++ |
+ #' - `join_keys` (`join_keys`) containing relationships between datasets.+ |
+
292 | ++ |
+ #' - `metadata` (`list`) containing metadata of datasets.+ |
+
293 | ++ |
+ #'+ |
+
294 | ++ |
+ #' @keywords internal+ |
+
295 | ++ |
+ .datasets_to_data <- function(module, datasets, trigger_data = reactiveVal(1L)) {+ |
+
296 | +15x | +
+ checkmate::assert_class(module, "teal_module")+ |
+
297 | +15x | +
+ checkmate::assert_class(datasets, "FilteredData")+ |
+
298 | +15x | +
+ checkmate::assert_class(trigger_data, "reactiveVal")+ |
+
299 | ++ | + + | +
300 | +14x | +
+ datanames <- if (is.null(module$datanames) || identical(module$datanames, "all")) {+ |
+
301 | +5x | +
+ datasets$datanames()+ |
+
302 | ++ |
+ } else {+ |
+
303 | +9x | +
+ unique(module$datanames) # todo: include parents! unique shouldn't be needed here!+ |
+
304 | ++ |
+ }+ |
+
305 | ++ | + + | +
306 | ++ |
+ # list of reactive filtered data+ |
+
307 | +14x | +
+ data <- sapply(+ |
+
308 | +14x | +
+ datanames,+ |
+
309 | +14x | +
+ function(x) eventReactive(trigger_data(), datasets$get_data(x, filtered = TRUE)),+ |
+
310 | +14x | +
+ simplify = FALSE+ |
+
311 | ++ |
+ )+ |
+
312 | ++ | + + | +
313 | +14x | +
+ hashes <- calculate_hashes(datanames, datasets)+ |
+
314 | +14x | +
+ metadata <- sapply(datanames, datasets$get_metadata, simplify = FALSE)+ |
+
315 | ++ | + + | +
316 | +14x | +
+ new_tdata(+ |
+
317 | +14x | +
+ data,+ |
+
318 | +14x | +
+ eventReactive(+ |
+
319 | +14x | +
+ trigger_data(),+ |
+
320 | ++ |
+ {+ |
+
321 | +14x | +
+ c(+ |
+
322 | +14x | +
+ get_rcode_str_install(),+ |
+
323 | +14x | +
+ get_rcode_libraries(),+ |
+
324 | +14x | +
+ get_datasets_code(datanames, datasets, hashes)+ |
+
325 | ++ |
+ )+ |
+
326 | ++ |
+ }+ |
+
327 | ++ |
+ ),+ |
+
328 | +14x | +
+ datasets$get_join_keys(),+ |
+
329 | +14x | +
+ metadata+ |
+
330 | ++ |
+ )+ |
+
331 | ++ |
+ }+ |
+
332 | ++ | + + | +
333 | ++ |
+ #' Get the hash of a dataset+ |
+
334 | ++ |
+ #'+ |
+
335 | ++ |
+ #' @param datanames (`character`) names of datasets+ |
+
336 | ++ |
+ #' @param datasets (`FilteredData`) object holding the data+ |
+
337 | ++ |
+ #'+ |
+
338 | ++ |
+ #' @return A list of hashes per dataset+ |
+
339 | ++ |
+ #' @keywords internal+ |
+
340 | ++ |
+ #'+ |
+
341 | ++ |
+ calculate_hashes <- function(datanames, datasets) {+ |
+
342 | +17x | +
+ sapply(datanames, function(x) rlang::hash(datasets$get_data(x, filtered = FALSE)), simplify = FALSE)+ |
+
343 | ++ |
+ }+ |
+
1 | ++ |
+ #' @title `TealReportCard`+ |
+
2 | ++ |
+ #' @description `r lifecycle::badge("experimental")`+ |
+
3 | ++ |
+ #' A child of [`ReportCard`] that is used for teal specific applications.+ |
+
4 | ++ |
+ #' In addition to the parent methods, it supports rendering teal specific elements such as+ |
+
5 | ++ |
+ #' the source code, the encodings panel content and the filter panel content as part of the+ |
+
6 | ++ |
+ #' meta data.+ |
+
7 | ++ |
+ #' @export+ |
+
8 | ++ |
+ #'+ |
+
9 | ++ |
+ TealReportCard <- R6::R6Class( # nolint: object_name_linter.+ |
+
10 | ++ |
+ classname = "TealReportCard",+ |
+
11 | ++ |
+ inherit = teal.reporter::ReportCard,+ |
+
12 | ++ |
+ public = list(+ |
+
13 | ++ |
+ #' @description Appends the source code to the `content` meta data of this `TealReportCard`.+ |
+
14 | ++ |
+ #'+ |
+
15 | ++ |
+ #' @param src (`character(1)`) code as text.+ |
+
16 | ++ |
+ #' @param ... any `rmarkdown` R chunk parameter and its value.+ |
+
17 | ++ |
+ #' But `eval` parameter is always set to `FALSE`.+ |
+
18 | ++ |
+ #' @return invisibly self+ |
+
19 | ++ |
+ #' @examples+ |
+
20 | ++ |
+ #' card <- TealReportCard$new()$append_src(+ |
+
21 | ++ |
+ #' "plot(iris)"+ |
+
22 | ++ |
+ #' )+ |
+
23 | ++ |
+ #' card$get_content()[[1]]$get_content()+ |
+
24 | ++ |
+ append_src = function(src, ...) {+ |
+
25 | +4x | +
+ checkmate::assert_character(src, min.len = 0, max.len = 1)+ |
+
26 | +4x | +
+ params <- list(...)+ |
+
27 | +4x | +
+ params$eval <- FALSE+ |
+
28 | +4x | +
+ rblock <- RcodeBlock$new(src)+ |
+
29 | +4x | +
+ rblock$set_params(params)+ |
+
30 | +4x | +
+ self$append_content(rblock)+ |
+
31 | +4x | +
+ self$append_metadata("SRC", src)+ |
+
32 | +4x | +
+ invisible(self)+ |
+
33 | ++ |
+ },+ |
+
34 | ++ |
+ #' @description Appends the filter state list to the `content` and `metadata` of this `TealReportCard`.+ |
+
35 | ++ |
+ #' If the filter state list has an attribute named `formatted`, it appends it to the card otherwise it uses+ |
+
36 | ++ |
+ #' the default `yaml::as.yaml` to format the list.+ |
+
37 | ++ |
+ #' If the filter state list is empty, nothing is appended to the `content`.+ |
+
38 | ++ |
+ #'+ |
+
39 | ++ |
+ #' @param fs (`teal_slices`) object returned from [teal_slices()] function.+ |
+
40 | ++ |
+ #' @return invisibly self+ |
+
41 | ++ |
+ append_fs = function(fs) {+ |
+
42 | +5x | +
+ checkmate::assert_class(fs, "teal_slices")+ |
+
43 | +4x | +
+ self$append_text("Filter State", "header3")+ |
+
44 | +4x | +
+ self$append_content(TealSlicesBlock$new(fs))+ |
+
45 | +4x | +
+ invisible(self)+ |
+
46 | ++ |
+ },+ |
+
47 | ++ |
+ #' @description Appends the encodings list to the `content` and `metadata` of this `TealReportCard`.+ |
+
48 | ++ |
+ #'+ |
+
49 | ++ |
+ #' @param encodings (`list`) list of encodings selections of the teal app.+ |
+
50 | ++ |
+ #' @return invisibly self+ |
+
51 | ++ |
+ #' @examples+ |
+
52 | ++ |
+ #' card <- TealReportCard$new()$append_encodings(list(variable1 = "X"))+ |
+
53 | ++ |
+ #' card$get_content()[[1]]$get_content()+ |
+
54 | ++ |
+ #'+ |
+
55 | ++ |
+ append_encodings = function(encodings) {+ |
+
56 | +4x | +
+ checkmate::assert_list(encodings)+ |
+
57 | +4x | +
+ self$append_text("Selected Options", "header3")+ |
+
58 | +4x | +
+ if (requireNamespace("yaml", quietly = TRUE)) {+ |
+
59 | +4x | +
+ self$append_text(yaml::as.yaml(encodings, handlers = list(+ |
+
60 | +4x | +
+ POSIXct = function(x) format(x, "%Y-%m-%d"),+ |
+
61 | +4x | +
+ POSIXlt = function(x) format(x, "%Y-%m-%d"),+ |
+
62 | +4x | +
+ Date = function(x) format(x, "%Y-%m-%d")+ |
+
63 | +4x | +
+ )), "verbatim")+ |
+
64 | ++ |
+ } else {+ |
+
65 | +! | +
+ stop("yaml package is required to format the encodings list")+ |
+
66 | ++ |
+ }+ |
+
67 | +4x | +
+ self$append_metadata("Encodings", encodings)+ |
+
68 | +4x | +
+ invisible(self)+ |
+
69 | ++ |
+ }+ |
+
70 | ++ |
+ ),+ |
+
71 | ++ |
+ private = list()+ |
+
72 | ++ |
+ )+ |
+
73 | ++ | + + | +
74 | ++ |
+ #' @title `RcodeBlock`+ |
+
75 | ++ |
+ #' @keywords internal+ |
+
76 | ++ |
+ TealSlicesBlock <- R6::R6Class( # nolint: object_name_linter.+ |
+
77 | ++ |
+ classname = "TealSlicesBlock",+ |
+
78 | ++ |
+ inherit = teal.reporter:::TextBlock,+ |
+
79 | ++ |
+ public = list(+ |
+
80 | ++ |
+ #' @description Returns a `TealSlicesBlock` object.+ |
+
81 | ++ |
+ #'+ |
+
82 | ++ |
+ #' @details Returns a `TealSlicesBlock` object with no content and no parameters.+ |
+
83 | ++ |
+ #'+ |
+
84 | ++ |
+ #' @param content (`teal_slices`) object returned from [teal_slices()] function.+ |
+
85 | ++ |
+ #' @param style (`character(1)`) string specifying style to apply.+ |
+
86 | ++ |
+ #'+ |
+
87 | ++ |
+ #' @return `TealSlicesBlock`+ |
+
88 | ++ |
+ #' @examples+ |
+
89 | ++ |
+ #' block <- teal:::TealSlicesBlock$new()+ |
+
90 | ++ |
+ #'+ |
+
91 | ++ |
+ initialize = function(content = teal_slices(), style = "verbatim") {+ |
+
92 | +10x | +
+ self$set_content(content)+ |
+
93 | +9x | +
+ self$set_style(style)+ |
+
94 | +9x | +
+ invisible(self)+ |
+
95 | ++ |
+ },+ |
+
96 | ++ | + + | +
97 | ++ |
+ #' @description Sets content of this `TealSlicesBlock`.+ |
+
98 | ++ |
+ #' Sets content as `YAML` text which represents a list generated from `teal_slices`.+ |
+
99 | ++ |
+ #' The list displays limited number of fields from `teal_slice` objects, but this list is+ |
+
100 | ++ |
+ #' sufficient to conclude which filters were applied.+ |
+
101 | ++ |
+ #' When `selected` field in `teal_slice` object is a range, then it is displayed as a "min"+ |
+
102 | ++ |
+ #'+ |
+
103 | ++ |
+ #'+ |
+
104 | ++ |
+ #' @param content (`teal_slices`) object returned from [teal_slices()] function.+ |
+
105 | ++ |
+ #' @return invisibly self+ |
+
106 | ++ |
+ set_content = function(content) {+ |
+
107 | +11x | +
+ checkmate::assert_class(content, "teal_slices")+ |
+
108 | +10x | +
+ if (length(content) != 0) {+ |
+
109 | +7x | +
+ states_list <- lapply(content, function(x) {+ |
+
110 | +7x | +
+ x_list <- shiny::isolate(as.list(x))+ |
+
111 | +7x | +
+ if (+ |
+
112 | +7x | +
+ inherits(x_list$choices, c("integer", "numeric", "Date", "POSIXct", "POSIXlt")) &&+ |
+
113 | +7x | +
+ length(x_list$choices) == 2 &&+ |
+
114 | +7x | +
+ length(x_list$selected) == 2+ |
+
115 | ++ |
+ ) {+ |
+
116 | +! | +
+ x_list$range <- paste(x_list$selected, collapse = " - ")+ |
+
117 | +! | +
+ x_list["selected"] <- NULL+ |
+
118 | ++ |
+ }+ |
+
119 | +7x | +
+ if (!is.null(x_list$arg)) {+ |
+
120 | +! | +
+ x_list$arg <- if (x_list$arg == "subset") "Genes" else "Samples"+ |
+
121 | ++ |
+ }+ |
+
122 | ++ | + + | +
123 | +7x | +
+ x_list <- x_list[+ |
+
124 | +7x | +
+ c("dataname", "varname", "experiment", "arg", "expr", "selected", "range", "keep_na", "keep_inf")+ |
+
125 | ++ |
+ ]+ |
+
126 | +7x | +
+ names(x_list) <- c(+ |
+
127 | +7x | +
+ "Dataset name", "Variable name", "Experiment", "Filtering by", "Applied expression",+ |
+
128 | +7x | +
+ "Selected Values", "Selected range", "Include NA values", "Include Inf values"+ |
+
129 | ++ |
+ )+ |
+
130 | ++ | + + | +
131 | +7x | +
+ Filter(Negate(is.null), x_list)+ |
+
132 | ++ |
+ })+ |
+
133 | ++ | + + | +
134 | +7x | +
+ if (requireNamespace("yaml", quietly = TRUE)) {+ |
+
135 | +7x | +
+ super$set_content(yaml::as.yaml(states_list))+ |
+
136 | ++ |
+ } else {+ |
+
137 | +! | +
+ stop("yaml package is required to format the filter state list")+ |
+
138 | ++ |
+ }+ |
+
139 | ++ |
+ }+ |
+
140 | +10x | +
+ private$teal_slices <- content+ |
+
141 | +10x | +
+ invisible(self)+ |
+
142 | ++ |
+ },+ |
+
143 | ++ |
+ #' @description Create the `RcodeBlock` from a list.+ |
+
144 | ++ |
+ #' @param x `named list` with two fields `c("text", "params")`.+ |
+
145 | ++ |
+ #' Use the `get_available_params` method to get all possible parameters.+ |
+
146 | ++ |
+ #' @return invisibly self+ |
+
147 | ++ |
+ from_list = function(x) {+ |
+
148 | +1x | +
+ checkmate::assert_list(x)+ |
+
149 | +1x | +
+ checkmate::assert_names(names(x), must.include = c("teal_slices"))+ |
+
150 | +1x | +
+ self$set_content(x$teal_slices)+ |
+
151 | +1x | +
+ invisible(self)+ |
+
152 | ++ |
+ },+ |
+
153 | ++ |
+ #' @description Convert the `RcodeBlock` to a list.+ |
+
154 | ++ |
+ #' @return `named list` with a text and `params`.+ |
+
155 | ++ | + + | +
156 | ++ |
+ to_list = function() {+ |
+
157 | +2x | +
+ list(teal_slices = private$teal_slices)+ |
+
158 | ++ |
+ }+ |
+
159 | ++ |
+ ),+ |
+
160 | ++ |
+ private = list(+ |
+
161 | ++ |
+ style = "verbatim",+ |
+
162 | ++ |
+ teal_slices = NULL # teal_slices+ |
+
163 | ++ |
+ )+ |
+
164 | ++ |
+ )+ |
+
1 | ++ |
+ #' Add right filter panel into each of the top-level `teal_modules` UIs.+ |
+
2 | ++ |
+ #'+ |
+
3 | ++ |
+ #' The [ui_nested_tabs] function returns a nested tabbed UI corresponding+ |
+
4 | ++ |
+ #' to the nested modules.+ |
+
5 | ++ |
+ #' This function adds the right filter panel to each main tab.+ |
+
6 | ++ |
+ #'+ |
+
7 | ++ |
+ #' The right filter panel's filter choices affect the `datasets` object. Therefore,+ |
+
8 | ++ |
+ #' all modules using the same `datasets` share the same filters.+ |
+
9 | ++ |
+ #'+ |
+
10 | ++ |
+ #' This works with nested modules of depth greater than 2, though the filter+ |
+
11 | ++ |
+ #' panel is inserted at the right of the modules at depth 1 and not at the leaves.+ |
+
12 | ++ |
+ #'+ |
+
13 | ++ |
+ #' @name module_tabs_with_filters+ |
+
14 | ++ |
+ #'+ |
+
15 | ++ |
+ #' @inheritParams module_teal+ |
+
16 | ++ |
+ #'+ |
+
17 | ++ |
+ #' @param datasets (`named list` of `FilteredData`)\cr+ |
+
18 | ++ |
+ #' object to store filter state and filtered datasets, shared across modules. For more+ |
+
19 | ++ |
+ #' details see [`teal.slice::FilteredData`]. Structure of the list must be the same as structure+ |
+
20 | ++ |
+ #' of the `modules` argument and list names must correspond to the labels in `modules`.+ |
+
21 | ++ |
+ #' When filter is not module-specific then list contains the same object in all elements.+ |
+
22 | ++ |
+ #' @param reporter (`Reporter`) object from `teal.reporter`+ |
+
23 | ++ |
+ #'+ |
+
24 | ++ |
+ #' @return A `tagList` of The main menu, place holders for filters and+ |
+
25 | ++ |
+ #' place holders for the teal modules+ |
+
26 | ++ |
+ #'+ |
+
27 | ++ |
+ #'+ |
+
28 | ++ |
+ #' @keywords internal+ |
+
29 | ++ |
+ #'+ |
+
30 | ++ |
+ #' @examples+ |
+
31 | ++ |
+ #'+ |
+
32 | ++ |
+ #' mods <- teal:::example_modules()+ |
+
33 | ++ |
+ #' datasets <- teal:::example_datasets()+ |
+
34 | ++ |
+ #'+ |
+
35 | ++ |
+ #' app <- shinyApp(+ |
+
36 | ++ |
+ #' ui = function() {+ |
+
37 | ++ |
+ #' tagList(+ |
+
38 | ++ |
+ #' teal:::include_teal_css_js(),+ |
+
39 | ++ |
+ #' textOutput("info"),+ |
+
40 | ++ |
+ #' fluidPage( # needed for nice tabs+ |
+
41 | ++ |
+ #' ui_tabs_with_filters("dummy", modules = mods, datasets = datasets)+ |
+
42 | ++ |
+ #' )+ |
+
43 | ++ |
+ #' )+ |
+
44 | ++ |
+ #' },+ |
+
45 | ++ |
+ #' server = function(input, output, session) {+ |
+
46 | ++ |
+ #' output$info <- renderText({+ |
+
47 | ++ |
+ #' paste0("The currently active tab name is ", active_module()$label)+ |
+
48 | ++ |
+ #' })+ |
+
49 | ++ |
+ #' active_module <- srv_tabs_with_filters(id = "dummy", datasets = datasets, modules = mods)+ |
+
50 | ++ |
+ #' }+ |
+
51 | ++ |
+ #' )+ |
+
52 | ++ |
+ #' if (interactive()) {+ |
+
53 | ++ |
+ #' shinyApp(app$ui, app$server)+ |
+
54 | ++ |
+ #' }+ |
+
55 | ++ |
+ #'+ |
+
56 | ++ |
+ NULL+ |
+
57 | ++ | + + | +
58 | ++ |
+ #' @rdname module_tabs_with_filters+ |
+
59 | ++ |
+ ui_tabs_with_filters <- function(id, modules, datasets, filter = teal_slices()) {+ |
+
60 | +1x | +
+ checkmate::assert_class(modules, "teal_modules")+ |
+
61 | +1x | +
+ checkmate::assert_list(datasets, types = c("list", "FilteredData"))+ |
+
62 | +1x | +
+ checkmate::assert_class(filter, "teal_slices")+ |
+
63 | ++ | + + | +
64 | +1x | +
+ ns <- NS(id)+ |
+
65 | +1x | +
+ is_module_specific <- isTRUE(attr(filter, "module_specific"))+ |
+
66 | ++ | + + | +
67 | +1x | +
+ teal_ui <- ui_nested_tabs(ns("root"), modules = modules, datasets, is_module_specific = is_module_specific)+ |
+
68 | +1x | +
+ filter_panel_btns <- tags$li(+ |
+
69 | +1x | +
+ class = "flex-grow",+ |
+
70 | +1x | +
+ tags$button(+ |
+
71 | +1x | +
+ class = "btn action-button filter_hamburger", # see sidebar.css for style filter_hamburger+ |
+
72 | +1x | +
+ href = "javascript:void(0)",+ |
+
73 | +1x | +
+ onclick = "toggleFilterPanel();", # see sidebar.js+ |
+
74 | +1x | +
+ title = "Toggle filter panels",+ |
+
75 | +1x | +
+ icon("fas fa-bars")+ |
+
76 | ++ |
+ ),+ |
+
77 | +1x | +
+ filter_manager_modal_ui(ns("filter_manager"))+ |
+
78 | ++ |
+ )+ |
+
79 | +1x | +
+ teal_ui$children[[1]] <- tagAppendChild(teal_ui$children[[1]], filter_panel_btns)+ |
+
80 | ++ | + + | +
81 | +1x | +
+ if (!is_module_specific) {+ |
+
82 | ++ |
+ # need to rearrange html so that filter panel is within tabset+ |
+
83 | +1x | +
+ tabset_bar <- teal_ui$children[[1]]+ |
+
84 | +1x | +
+ teal_modules <- teal_ui$children[[2]]+ |
+
85 | +1x | +
+ filter_ui <- unlist(datasets)[[1]]$ui_filter_panel(ns("filter_panel"))+ |
+
86 | +1x | +
+ list(+ |
+
87 | +1x | +
+ tabset_bar,+ |
+
88 | +1x | +
+ tags$hr(class = "my-2"),+ |
+
89 | +1x | +
+ fluidRow(+ |
+
90 | +1x | +
+ column(width = 9, teal_modules, class = "teal_primary_col"),+ |
+
91 | +1x | +
+ column(width = 3, filter_ui, class = "teal_secondary_col")+ |
+
92 | ++ |
+ )+ |
+
93 | ++ |
+ )+ |
+
94 | ++ |
+ } else {+ |
+
95 | +! | +
+ teal_ui+ |
+
96 | ++ |
+ }+ |
+
97 | ++ |
+ }+ |
+
98 | ++ | + + | +
99 | ++ |
+ #' @rdname module_tabs_with_filters+ |
+
100 | ++ |
+ srv_tabs_with_filters <- function(id,+ |
+
101 | ++ |
+ datasets,+ |
+
102 | ++ |
+ modules,+ |
+
103 | ++ |
+ reporter = teal.reporter::Reporter$new(),+ |
+
104 | ++ |
+ filter = teal_slices()) {+ |
+
105 | +6x | +
+ checkmate::assert_class(modules, "teal_modules")+ |
+
106 | +6x | +
+ checkmate::assert_list(datasets, types = c("list", "FilteredData"))+ |
+
107 | +6x | +
+ checkmate::assert_class(reporter, "Reporter")+ |
+
108 | +4x | +
+ checkmate::assert_class(filter, "teal_slices")+ |
+
109 | ++ | + + | +
110 | +4x | +
+ moduleServer(id, function(input, output, session) {+ |
+
111 | +4x | +
+ logger::log_trace("srv_tabs_with_filters initializing the module.")+ |
+
112 | ++ | + + | +
113 | +4x | +
+ is_module_specific <- isTRUE(attr(filter, "module_specific"))+ |
+
114 | +4x | +
+ manager_out <- filter_manager_modal_srv("filter_manager", filtered_data_list = datasets, filter = filter)+ |
+
115 | ++ | + + | +
116 | +4x | +
+ active_module <- srv_nested_tabs(+ |
+
117 | +4x | +
+ id = "root",+ |
+
118 | +4x | +
+ datasets = datasets,+ |
+
119 | +4x | +
+ modules = modules,+ |
+
120 | +4x | +
+ reporter = reporter,+ |
+
121 | +4x | +
+ is_module_specific = is_module_specific+ |
+
122 | ++ |
+ )+ |
+
123 | ++ | + + | +
124 | +4x | +
+ if (!is_module_specific) {+ |
+
125 | +4x | +
+ active_datanames <- reactive({+ |
+
126 | +7x | +
+ if (identical(active_module()$datanames, "all")) {+ |
+
127 | +1x | +
+ singleton$datanames()+ |
+
128 | ++ |
+ } else {+ |
+
129 | +5x | +
+ active_module()$datanames+ |
+
130 | ++ |
+ }+ |
+
131 | ++ |
+ })+ |
+
132 | +4x | +
+ singleton <- unlist(datasets)[[1]]+ |
+
133 | +4x | +
+ singleton$srv_filter_panel("filter_panel", active_datanames = active_datanames)+ |
+
134 | ++ | + + | +
135 | +4x | +
+ observeEvent(+ |
+
136 | +4x | +
+ eventExpr = active_datanames(),+ |
+
137 | +4x | +
+ handlerExpr = {+ |
+
138 | +5x | +
+ script <- if (length(active_datanames()) == 0 || is.null(active_datanames())) {+ |
+
139 | ++ |
+ # hide the filter panel and disable the burger button+ |
+
140 | +! | +
+ "handleNoActiveDatasets();"+ |
+
141 | ++ |
+ } else {+ |
+
142 | ++ |
+ # show the filter panel and enable the burger button+ |
+
143 | +5x | +
+ "handleActiveDatasetsPresent();"+ |
+
144 | ++ |
+ }+ |
+
145 | +5x | +
+ shinyjs::runjs(script)+ |
+
146 | ++ |
+ },+ |
+
147 | +4x | +
+ ignoreNULL = FALSE+ |
+
148 | ++ |
+ )+ |
+
149 | ++ |
+ }+ |
+
150 | ++ | + + | +
151 | +4x | +
+ showNotification("Data loaded - App fully started up")+ |
+
152 | +4x | +
+ logger::log_trace("srv_tabs_with_filters initialized the module")+ |
+
153 | +4x | +
+ return(active_module)+ |
+
154 | ++ |
+ })+ |
+
155 | ++ |
+ }+ |
+
1 | ++ |
+ # This file adds a splash screen for delayed data loading on top of teal+ |
+
2 | ++ | + + | +
3 | ++ |
+ #' UI to show a splash screen in the beginning, then delegate to [srv_teal()]+ |
+
4 | ++ |
+ #'+ |
+
5 | ++ |
+ #' @description `r lifecycle::badge("stable")`+ |
+
6 | ++ |
+ #' The splash screen could be used to query for a password to fetch the data.+ |
+
7 | ++ |
+ #' [init()] is a very thin wrapper around this module useful for end-users which+ |
+
8 | ++ |
+ #' assumes that it is a top-level module and cannot be embedded.+ |
+
9 | ++ |
+ #' This function instead adheres to the Shiny module conventions.+ |
+
10 | ++ |
+ #'+ |
+
11 | ++ |
+ #' If data is obtained through delayed loading, its splash screen is used. Otherwise,+ |
+
12 | ++ |
+ #' a default splash screen is shown.+ |
+
13 | ++ |
+ #'+ |
+
14 | ++ |
+ #' Please also refer to the doc of [init()].+ |
+
15 | ++ |
+ #'+ |
+
16 | ++ |
+ #' @param id (`character(1)`)\cr+ |
+
17 | ++ |
+ #' module id+ |
+
18 | ++ |
+ #' @inheritParams init+ |
+
19 | ++ |
+ #' @export+ |
+
20 | ++ |
+ ui_teal_with_splash <- function(id,+ |
+
21 | ++ |
+ data,+ |
+
22 | ++ |
+ title,+ |
+
23 | ++ |
+ header = tags$p("Add Title Here"),+ |
+
24 | ++ |
+ footer = tags$p("Add Footer Here")) {+ |
+
25 | +22x | +
+ checkmate::assert_multi_class(data, c("TealData", "teal_data", "teal_data_module"))+ |
+
26 | +22x | +
+ ns <- NS(id)+ |
+
27 | ++ | + + | +
28 | ++ |
+ # Startup splash screen for delayed loading+ |
+
29 | ++ |
+ # We use delayed loading in all cases, even when the data does not need to be fetched.+ |
+
30 | ++ |
+ # This has the benefit that when filtering the data takes a lot of time initially, the+ |
+
31 | ++ |
+ # Shiny app does not time out.+ |
+
32 | ++ | + + | +
33 | +22x | +
+ splash_ui <- if (inherits(data, "teal_data_module")) {+ |
+
34 | +1x | +
+ data$ui(ns("teal_data_module"))+ |
+
35 | +22x | +
+ } else if (inherits(data, "teal_data")) {+ |
+
36 | +2x | +
+ div()+ |
+
37 | +22x | +
+ } else if (inherits(data, "TealDataAbstract") && teal.data::is_pulled(data)) {+ |
+
38 | +17x | +
+ div()+ |
+
39 | ++ |
+ } else {+ |
+
40 | +2x | +
+ message("App was initialized with delayed data loading.")+ |
+
41 | +2x | +
+ data$get_ui(ns("startapp_module"))+ |
+
42 | ++ |
+ }+ |
+
43 | +22x | +
+ ui_teal(+ |
+
44 | +22x | +
+ id = ns("teal"),+ |
+
45 | +22x | +
+ splash_ui = div(splash_ui, uiOutput(ns("error"))),+ |
+
46 | +22x | +
+ title = title,+ |
+
47 | +22x | +
+ header = header,+ |
+
48 | +22x | +
+ footer = footer+ |
+
49 | ++ |
+ )+ |
+
50 | ++ |
+ }+ |
+
51 | ++ | + + | +
52 | ++ |
+ #' Server function that loads the data through reactive loading and then delegates+ |
+
53 | ++ |
+ #' to [srv_teal()].+ |
+
54 | ++ |
+ #'+ |
+
55 | ++ |
+ #' @description `r lifecycle::badge("stable")`+ |
+
56 | ++ |
+ #' Please also refer to the doc of [init()].+ |
+
57 | ++ |
+ #'+ |
+
58 | ++ |
+ #' @inheritParams init+ |
+
59 | ++ |
+ #' @param modules `teal_modules` object containing the output modules which+ |
+
60 | ++ |
+ #' will be displayed in the teal application. See [modules()] and [module()] for+ |
+
61 | ++ |
+ #' more details.+ |
+
62 | ++ |
+ #' @inheritParams shiny::moduleServer+ |
+
63 | ++ |
+ #' @return `reactive` containing `teal_data` object when data is loaded.+ |
+
64 | ++ |
+ #' If data is not loaded yet, `reactive` returns `NULL`.+ |
+
65 | ++ |
+ #' @export+ |
+
66 | ++ |
+ srv_teal_with_splash <- function(id, data, modules, filter = teal_slices()) {+ |
+
67 | +13x | +
+ checkmate::check_multi_class(data, c("TealData", "teal_data", "teal_data_module"))+ |
+
68 | ++ | + + | +
69 | +13x | +
+ moduleServer(id, function(input, output, session) {+ |
+
70 | +13x | +
+ logger::log_trace("srv_teal_with_splash initializing module with data.")+ |
+
71 | ++ | + + | +
72 | +13x | +
+ if (getOption("teal.show_js_log", default = FALSE)) {+ |
+
73 | +! | +
+ shinyjs::showLog()+ |
+
74 | ++ |
+ }+ |
+
75 | ++ | + + | +
76 | ++ |
+ # teal_data_rv contains teal_data object+ |
+
77 | ++ |
+ # either passed to teal::init or returned from teal_data_module+ |
+
78 | +13x | +
+ teal_data_rv <- if (inherits(data, "teal_data_module")) {+ |
+
79 | +6x | +
+ data <- data$server(id = "teal_data_module")+ |
+
80 | +6x | +
+ if (!is.reactive(data)) {+ |
+
81 | +1x | +
+ stop("The `teal_data_module` must return a reactive expression.", call. = FALSE)+ |
+
82 | ++ |
+ }+ |
+
83 | +5x | +
+ data+ |
+
84 | +13x | +
+ } else if (inherits(data, "teal_data")) {+ |
+
85 | +5x | +
+ reactiveVal(data)+ |
+
86 | +13x | +
+ } else if (inherits(data, "TealDataAbstract") && teal.data::is_pulled(data)) {+ |
+
87 | +! | +
+ new_data <- do.call(+ |
+
88 | +! | +
+ teal.data::teal_data,+ |
+
89 | +! | +
+ c(+ |
+
90 | +! | +
+ lapply(data$get_datasets(), function(x) x$get_raw_data()),+ |
+
91 | +! | +
+ list(code = data$get_code()),+ |
+
92 | +! | +
+ list(join_keys = teal.data::join_keys(data))+ |
+
93 | ++ |
+ )+ |
+
94 | ++ |
+ )+ |
+
95 | +! | +
+ reactiveVal(new_data) # will trigger by setting it+ |
+
96 | ++ |
+ } else {+ |
+
97 | +2x | +
+ raw_data_old <- data$get_server()(id = "startapp_module")+ |
+
98 | +2x | +
+ raw_data <- reactive({+ |
+
99 | +3x | +
+ data <- raw_data_old()+ |
+
100 | +3x | +
+ if (!is.null(data)) {+ |
+
101 | ++ |
+ # raw_data is a reactive which returns data only when submit button clicked+ |
+
102 | ++ |
+ # otherwise it returns NULL+ |
+
103 | +1x | +
+ do.call(+ |
+
104 | +1x | +
+ teal.data::teal_data,+ |
+
105 | +1x | +
+ c(+ |
+
106 | +1x | +
+ lapply(data$get_datasets(), function(x) x$get_raw_data()),+ |
+
107 | +1x | +
+ list(code = data$get_code()),+ |
+
108 | +1x | +
+ list(join_keys = teal.data::join_keys(data))+ |
+
109 | ++ |
+ )+ |
+
110 | ++ |
+ )+ |
+
111 | ++ |
+ }+ |
+
112 | ++ |
+ })+ |
+
113 | +2x | +
+ raw_data+ |
+
114 | ++ |
+ }+ |
+
115 | ++ | + + | +
116 | +12x | +
+ teal_data_rv_validate <- reactive({+ |
+
117 | ++ |
+ # custom module can return error+ |
+
118 | +8x | +
+ data <- tryCatch(teal_data_rv(), error = function(e) e)+ |
+
119 | ++ | + + | +
120 | ++ |
+ # there is an empty reactive event on init!+ |
+
121 | +8x | +
+ if (inherits(data, "shiny.silent.error") && identical(data$message, "")) {+ |
+
122 | +! | +
+ return(NULL)+ |
+
123 | ++ |
+ }+ |
+
124 | ++ | + + | +
125 | ++ |
+ # to handle qenv.error+ |
+
126 | +8x | +
+ if (inherits(data, "qenv.error")) {+ |
+
127 | +1x | +
+ validate(+ |
+
128 | +1x | +
+ need(+ |
+
129 | +1x | +
+ FALSE,+ |
+
130 | +1x | +
+ paste(+ |
+
131 | +1x | +
+ "Error when executing `teal_data_module`:\n ",+ |
+
132 | +1x | +
+ paste(data$message, collapse = "\n"),+ |
+
133 | +1x | +
+ "\n Check your inputs or contact app developer if error persists."+ |
+
134 | ++ |
+ )+ |
+
135 | ++ |
+ )+ |
+
136 | ++ |
+ )+ |
+
137 | ++ |
+ }+ |
+
138 | ++ | + + | +
139 | ++ |
+ # to handle module non-qenv errors+ |
+
140 | +7x | +
+ if (inherits(data, "error")) {+ |
+
141 | +1x | +
+ validate(+ |
+
142 | +1x | +
+ need(+ |
+
143 | +1x | +
+ FALSE,+ |
+
144 | +1x | +
+ paste(+ |
+
145 | +1x | +
+ "Error when executing `teal_data_module`:\n ",+ |
+
146 | +1x | +
+ paste(data$message, collpase = "\n"),+ |
+
147 | +1x | +
+ "\n Check your inputs or contact app developer if error persists."+ |
+
148 | ++ |
+ )+ |
+
149 | ++ |
+ )+ |
+
150 | ++ |
+ )+ |
+
151 | ++ |
+ }+ |
+
152 | ++ | + + | +
153 | +6x | +
+ validate(+ |
+
154 | +6x | +
+ need(+ |
+
155 | +6x | +
+ inherits(data, "teal_data"),+ |
+
156 | +6x | +
+ paste(+ |
+
157 | +6x | +
+ "Error: `teal_data_module` did not return `teal_data` object",+ |
+
158 | +6x | +
+ "\n Check your inputs or contact app developer if error persists"+ |
+
159 | ++ |
+ )+ |
+
160 | ++ |
+ )+ |
+
161 | ++ |
+ )+ |
+
162 | ++ | + + | +
163 | +5x | +
+ validate(need(teal.data::datanames(data), "Data has no datanames. Contact app developer."))+ |
+
164 | ++ | + + | +
165 | ++ | + + | +
166 | +4x | +
+ is_modules_ok <- check_modules_datanames(modules, teal.data::datanames(data))+ |
+
167 | +4x | +
+ validate(need(isTRUE(is_modules_ok), is_modules_ok))+ |
+
168 | ++ | + + | +
169 | +3x | +
+ is_filter_ok <- check_filter_datanames(filter, teal.data::datanames(data))+ |
+
170 | +3x | +
+ if (!isTRUE(is_filter_ok)) {+ |
+
171 | +1x | +
+ showNotification(+ |
+
172 | +1x | +
+ "Some filters were not applied because of incompatibility with data. Contact app developer.",+ |
+
173 | +1x | +
+ type = "warning",+ |
+
174 | +1x | +
+ duration = 10+ |
+
175 | ++ |
+ )+ |
+
176 | +1x | +
+ logger::log_warn(is_filter_ok)+ |
+
177 | ++ |
+ }+ |
+
178 | ++ | + + | +
179 | +3x | +
+ teal_data_rv()+ |
+
180 | ++ |
+ })+ |
+
181 | ++ | + + | +
182 | +12x | +
+ output$error <- renderUI({+ |
+
183 | +1x | +
+ teal_data_rv_validate()+ |
+
184 | +1x | +
+ NULL+ |
+
185 | ++ |
+ })+ |
+
186 | ++ | + + | +
187 | ++ | + + | +
188 | ++ | + + | +
189 | +12x | +
+ res <- srv_teal(id = "teal", modules = modules, teal_data_rv = teal_data_rv_validate, filter = filter)+ |
+
190 | +12x | +
+ logger::log_trace("srv_teal_with_splash initialized module with data.")+ |
+
191 | +12x | +
+ return(res)+ |
+
192 | ++ |
+ })+ |
+
193 | ++ |
+ }+ |
+
1 | ++ |
+ #' Creates a `teal_modules` object.+ |
+
2 | ++ |
+ #'+ |
+
3 | ++ |
+ #' @description `r lifecycle::badge("stable")`+ |
+
4 | ++ |
+ #' This function collects a list of `teal_modules` and `teal_module` objects and returns a `teal_modules` object+ |
+
5 | ++ |
+ #' containing the passed objects.+ |
+
6 | ++ |
+ #'+ |
+
7 | ++ |
+ #' This function dictates what modules are included in a `teal` application. The internal structure of `teal_modules`+ |
+
8 | ++ |
+ #' shapes the navigation panel of a `teal` application.+ |
+
9 | ++ |
+ #'+ |
+
10 | ++ |
+ #' @param ... (`teal_module` or `teal_modules`) see [module()] and [modules()] for more details+ |
+
11 | ++ |
+ #' @param label (`character(1)`) label of modules collection (default `"root"`).+ |
+
12 | ++ |
+ #' If using the `label` argument then it must be explicitly named.+ |
+
13 | ++ |
+ #' For example `modules("lab", ...)` should be converted to `modules(label = "lab", ...)`+ |
+
14 | ++ |
+ #'+ |
+
15 | ++ |
+ #' @export+ |
+
16 | ++ |
+ #'+ |
+
17 | ++ |
+ #' @return object of class \code{teal_modules}. Object contains following fields+ |
+
18 | ++ |
+ #' - `label`: taken from the `label` argument+ |
+
19 | ++ |
+ #' - `children`: a list containing objects passed in `...`. List elements are named after+ |
+
20 | ++ |
+ #' their `label` attribute converted to a valid `shiny` id.+ |
+
21 | ++ |
+ #' @examples+ |
+
22 | ++ |
+ #' library(shiny)+ |
+
23 | ++ |
+ #'+ |
+
24 | ++ |
+ #' app <- init(+ |
+
25 | ++ |
+ #' data = teal_data(dataset("iris", iris)),+ |
+
26 | ++ |
+ #' modules = modules(+ |
+
27 | ++ |
+ #' label = "Modules",+ |
+
28 | ++ |
+ #' modules(+ |
+
29 | ++ |
+ #' label = "Module",+ |
+
30 | ++ |
+ #' module(+ |
+
31 | ++ |
+ #' label = "Inner module",+ |
+
32 | ++ |
+ #' server = function(id, data) {+ |
+
33 | ++ |
+ #' moduleServer(+ |
+
34 | ++ |
+ #' id,+ |
+
35 | ++ |
+ #' module = function(input, output, session) {+ |
+
36 | ++ |
+ #' output$data <- renderDataTable(data[["iris"]]())+ |
+
37 | ++ |
+ #' }+ |
+
38 | ++ |
+ #' )+ |
+
39 | ++ |
+ #' },+ |
+
40 | ++ |
+ #' ui = function(id) {+ |
+
41 | ++ |
+ #' ns <- NS(id)+ |
+
42 | ++ |
+ #' tagList(dataTableOutput(ns("data")))+ |
+
43 | ++ |
+ #' },+ |
+
44 | ++ |
+ #' datanames = "all"+ |
+
45 | ++ |
+ #' )+ |
+
46 | ++ |
+ #' ),+ |
+
47 | ++ |
+ #' module(+ |
+
48 | ++ |
+ #' label = "Another module",+ |
+
49 | ++ |
+ #' server = function(id) {+ |
+
50 | ++ |
+ #' moduleServer(+ |
+
51 | ++ |
+ #' id,+ |
+
52 | ++ |
+ #' module = function(input, output, session) {+ |
+
53 | ++ |
+ #' output$text <- renderText("Another module")+ |
+
54 | ++ |
+ #' }+ |
+
55 | ++ |
+ #' )+ |
+
56 | ++ |
+ #' },+ |
+
57 | ++ |
+ #' ui = function(id) {+ |
+
58 | ++ |
+ #' ns <- NS(id)+ |
+
59 | ++ |
+ #' tagList(textOutput(ns("text")))+ |
+
60 | ++ |
+ #' },+ |
+
61 | ++ |
+ #' datanames = NULL+ |
+
62 | ++ |
+ #' )+ |
+
63 | ++ |
+ #' )+ |
+
64 | ++ |
+ #' )+ |
+
65 | ++ |
+ #' if (interactive()) {+ |
+
66 | ++ |
+ #' shinyApp(app$ui, app$server)+ |
+
67 | ++ |
+ #' }+ |
+
68 | ++ |
+ modules <- function(..., label = "root") {+ |
+
69 | +140x | +
+ checkmate::assert_string(label)+ |
+
70 | +138x | +
+ submodules <- list(...)+ |
+
71 | +138x | +
+ if (any(vapply(submodules, is.character, FUN.VALUE = logical(1)))) {+ |
+
72 | +2x | +
+ stop(+ |
+
73 | +2x | +
+ "The only character argument to modules() must be 'label' and it must be named, ",+ |
+
74 | +2x | +
+ "change modules('lab', ...) to modules(label = 'lab', ...)"+ |
+
75 | ++ |
+ )+ |
+
76 | ++ |
+ }+ |
+
77 | ++ | + + | +
78 | +136x | +
+ checkmate::assert_list(submodules, min.len = 1, any.missing = FALSE, types = c("teal_module", "teal_modules"))+ |
+
79 | ++ |
+ # name them so we can more easily access the children+ |
+
80 | ++ |
+ # beware however that the label of the submodules should not be changed as it must be kept synced+ |
+
81 | +133x | +
+ labels <- vapply(submodules, function(submodule) submodule$label, character(1))+ |
+
82 | +133x | +
+ names(submodules) <- make.unique(gsub("[^[:alnum:]]+", "_", labels), sep = "_")+ |
+
83 | +133x | +
+ structure(+ |
+
84 | +133x | +
+ list(+ |
+
85 | +133x | +
+ label = label,+ |
+
86 | +133x | +
+ children = submodules+ |
+
87 | ++ |
+ ),+ |
+
88 | +133x | +
+ class = "teal_modules"+ |
+
89 | ++ |
+ )+ |
+
90 | ++ |
+ }+ |
+
91 | ++ | + + | +
92 | ++ |
+ #' Append a `teal_module` to `children` of a `teal_modules` object+ |
+
93 | ++ |
+ #' @keywords internal+ |
+
94 | ++ |
+ #' @param modules `teal_modules`+ |
+
95 | ++ |
+ #' @param module `teal_module` object to be appended onto the children of `modules`+ |
+
96 | ++ |
+ #' @return `teal_modules` object with `module` appended+ |
+
97 | ++ |
+ append_module <- function(modules, module) {+ |
+
98 | +7x | +
+ checkmate::assert_class(modules, "teal_modules")+ |
+
99 | +5x | +
+ checkmate::assert_class(module, "teal_module")+ |
+
100 | +3x | +
+ modules$children <- c(modules$children, list(module))+ |
+
101 | +3x | +
+ labels <- vapply(modules$children, function(submodule) submodule$label, character(1))+ |
+
102 | +3x | +
+ names(modules$children) <- make.unique(gsub("[^[:alnum:]]", "_", tolower(labels)), sep = "_")+ |
+
103 | +3x | +
+ modules+ |
+
104 | ++ |
+ }+ |
+
105 | ++ | + + | +
106 | ++ |
+ #' Extract/Remove module(s) of specific class+ |
+
107 | ++ |
+ #'+ |
+
108 | ++ |
+ #' Given a `teal_module` or a `teal_modules`, return the elements of the structure according to `class`.+ |
+
109 | ++ |
+ #'+ |
+
110 | ++ |
+ #' @param modules `teal_modules`+ |
+
111 | ++ |
+ #' @param class The class name of `teal_module` to be extracted or dropped.+ |
+
112 | ++ |
+ #' @keywords internal+ |
+
113 | ++ |
+ #' @return+ |
+
114 | ++ |
+ #' For `extract_module`, a `teal_module` of class `class` or `teal_modules` containing modules of class `class`.+ |
+
115 | ++ |
+ #' For `drop_module`, the opposite, which is all `teal_modules` of class other than `class`.+ |
+
116 | ++ |
+ #' @rdname module_management+ |
+
117 | ++ |
+ extract_module <- function(modules, class) {+ |
+
118 | +68x | +
+ if (inherits(modules, class)) {+ |
+
119 | +! | +
+ modules+ |
+
120 | +68x | +
+ } else if (inherits(modules, "teal_module")) {+ |
+
121 | +38x | +
+ NULL+ |
+
122 | +30x | +
+ } else if (inherits(modules, "teal_modules")) {+ |
+
123 | +30x | +
+ Filter(function(x) length(x) > 0L, lapply(modules$children, extract_module, class))+ |
+
124 | ++ |
+ }+ |
+
125 | ++ |
+ }+ |
+
126 | ++ | + + | +
127 | ++ |
+ #' @keywords internal+ |
+
128 | ++ |
+ #' @return `teal_modules`+ |
+
129 | ++ |
+ #' @rdname module_management+ |
+
130 | ++ |
+ drop_module <- function(modules, class) {+ |
+
131 | +68x | +
+ if (inherits(modules, class)) {+ |
+
132 | +! | +
+ NULL+ |
+
133 | +68x | +
+ } else if (inherits(modules, "teal_module")) {+ |
+
134 | +38x | +
+ modules+ |
+
135 | +30x | +
+ } else if (inherits(modules, "teal_modules")) {+ |
+
136 | +30x | +
+ do.call(+ |
+
137 | +30x | +
+ "modules",+ |
+
138 | +30x | +
+ c(Filter(function(x) length(x) > 0L, lapply(modules$children, drop_module, class)), label = modules$label)+ |
+
139 | ++ |
+ )+ |
+
140 | ++ |
+ }+ |
+
141 | ++ |
+ }+ |
+
142 | ++ | + + | +
143 | ++ |
+ #' Does the object make use of the `arg`+ |
+
144 | ++ |
+ #'+ |
+
145 | ++ |
+ #' @param modules (`teal_module` or `teal_modules`) object+ |
+
146 | ++ |
+ #' @param arg (`character(1)`) names of the arguments to be checked against formals of `teal` modules.+ |
+
147 | ++ |
+ #' @return `logical` whether the object makes use of `arg`+ |
+
148 | ++ |
+ #' @rdname is_arg_used+ |
+
149 | ++ |
+ #' @keywords internal+ |
+
150 | ++ |
+ is_arg_used <- function(modules, arg) {+ |
+
151 | +320x | +
+ checkmate::assert_string(arg)+ |
+
152 | +317x | +
+ if (inherits(modules, "teal_modules")) {+ |
+
153 | +27x | +
+ any(unlist(lapply(modules$children, is_arg_used, arg)))+ |
+
154 | +290x | +
+ } else if (inherits(modules, "teal_module")) {+ |
+
155 | +41x | +
+ is_arg_used(modules$server, arg) || is_arg_used(modules$ui, arg)+ |
+
156 | +249x | +
+ } else if (is.function(modules)) {+ |
+
157 | +247x | +
+ isTRUE(arg %in% names(formals(modules)))+ |
+
158 | ++ |
+ } else {+ |
+
159 | +2x | +
+ stop("is_arg_used function not implemented for this object")+ |
+
160 | ++ |
+ }+ |
+
161 | ++ |
+ }+ |
+
162 | ++ | + + | +
163 | ++ | + + | +
164 | ++ |
+ #' Creates a `teal_module` object.+ |
+
165 | ++ |
+ #'+ |
+
166 | ++ |
+ #' @description `r lifecycle::badge("stable")`+ |
+
167 | ++ |
+ #' This function embeds a `shiny` module inside a `teal` application. One `teal_module` maps to one `shiny` module.+ |
+
168 | ++ |
+ #'+ |
+
169 | ++ |
+ #' @param label (`character(1)`) Label shown in the navigation item for the module. Any label possible except+ |
+
170 | ++ |
+ #' `"global_filters"` - read more in `mapping` argument of [teal::teal_slices].+ |
+
171 | ++ |
+ #' @param server (`function`) `shiny` module with following arguments:+ |
+
172 | ++ |
+ #' - `id` - teal will set proper shiny namespace for this module (see [shiny::moduleServer()]).+ |
+
173 | ++ |
+ #' - `input`, `output`, `session` - (not recommended) then [shiny::callModule()] will be used to call a module.+ |
+
174 | ++ |
+ #' - `data` (optional) module will receive a `tdata` object, a list of reactive (filtered) data specified in+ |
+
175 | ++ |
+ #' the `filters` argument.+ |
+
176 | ++ |
+ #' - `datasets` (optional) module will receive `FilteredData`. (See `[teal.slice::FilteredData]`).+ |
+
177 | ++ |
+ #' - `reporter` (optional) module will receive `Reporter`. (See [teal.reporter::Reporter]).+ |
+
178 | ++ |
+ # - `filter_panel_api` (optional) module will receive `FilterPanelAPI`. (See [teal.slice::FilterPanelAPI]).+ |
+
179 | ++ |
+ #' - `...` (optional) `server_args` elements will be passed to the module named argument or to the `...`.+ |
+
180 | ++ |
+ #' @param ui (`function`) Shiny `ui` module function with following arguments:+ |
+
181 | ++ |
+ #' - `id` - teal will set proper shiny namespace for this module.+ |
+
182 | ++ |
+ #' - `data` (optional) module will receive list of reactive (filtered) data specified in the `filters` argument.+ |
+
183 | ++ |
+ #' - `datasets` (optional) module will receive `FilteredData`. (See `[teal.slice::FilteredData]`).+ |
+
184 | ++ |
+ #' - `...` (optional) `ui_args` elements will be passed to the module named argument or to the `...`.+ |
+
185 | ++ |
+ #' @param filters (`character`) Deprecated. Use `datanames` instead.+ |
+
186 | ++ |
+ #' @param datanames (`character`) A vector with `datanames` that are relevant for the item. The+ |
+
187 | ++ |
+ #' filter panel will automatically update the shown filters to include only+ |
+
188 | ++ |
+ #' filters in the listed datasets. `NULL` will hide the filter panel,+ |
+
189 | ++ |
+ #' and the keyword `'all'` will show filters of all datasets. `datanames` also determines+ |
+
190 | ++ |
+ #' a subset of datasets which are appended to the `data` argument in `server` function.+ |
+
191 | ++ |
+ #' @param server_args (named `list`) with additional arguments passed on to the+ |
+
192 | ++ |
+ #' `server` function.+ |
+
193 | ++ |
+ #' @param ui_args (named `list`) with additional arguments passed on to the+ |
+
194 | ++ |
+ #' `ui` function.+ |
+
195 | ++ |
+ #'+ |
+
196 | ++ |
+ #' @return object of class `teal_module`.+ |
+
197 | ++ |
+ #' @export+ |
+
198 | ++ |
+ #' @examples+ |
+
199 | ++ |
+ #' library(shiny)+ |
+
200 | ++ |
+ #'+ |
+
201 | ++ |
+ #' app <- init(+ |
+
202 | ++ |
+ #' data = teal_data(dataset("iris", iris)),+ |
+
203 | ++ |
+ #' modules = list(+ |
+
204 | ++ |
+ #' module(+ |
+
205 | ++ |
+ #' label = "Module",+ |
+
206 | ++ |
+ #' server = function(id, data) {+ |
+
207 | ++ |
+ #' moduleServer(+ |
+
208 | ++ |
+ #' id,+ |
+
209 | ++ |
+ #' module = function(input, output, session) {+ |
+
210 | ++ |
+ #' output$data <- renderDataTable(data[["iris"]]())+ |
+
211 | ++ |
+ #' }+ |
+
212 | ++ |
+ #' )+ |
+
213 | ++ |
+ #' },+ |
+
214 | ++ |
+ #' ui = function(id) {+ |
+
215 | ++ |
+ #' ns <- NS(id)+ |
+
216 | ++ |
+ #' tagList(dataTableOutput(ns("data")))+ |
+
217 | ++ |
+ #' }+ |
+
218 | ++ |
+ #' )+ |
+
219 | ++ |
+ #' )+ |
+
220 | ++ |
+ #' )+ |
+
221 | ++ |
+ #' if (interactive()) {+ |
+
222 | ++ |
+ #' shinyApp(app$ui, app$server)+ |
+
223 | ++ |
+ #' }+ |
+
224 | ++ |
+ module <- function(label = "module",+ |
+
225 | ++ |
+ server = function(id, ...) {+ |
+
226 | +! | +
+ moduleServer(id, function(input, output, session) {}) # nolint+ |
+
227 | ++ |
+ },+ |
+
228 | ++ |
+ ui = function(id, ...) {+ |
+
229 | +! | +
+ tags$p(paste0("This module has no UI (id: ", id, " )"))+ |
+
230 | ++ |
+ },+ |
+
231 | ++ |
+ filters,+ |
+
232 | ++ |
+ datanames = "all",+ |
+
233 | ++ |
+ server_args = NULL,+ |
+
234 | ++ |
+ ui_args = NULL) {+ |
+
235 | +155x | +
+ checkmate::assert_string(label)+ |
+
236 | +152x | +
+ checkmate::assert_function(server)+ |
+
237 | +152x | +
+ checkmate::assert_function(ui)+ |
+
238 | +152x | +
+ checkmate::assert_character(datanames, min.len = 1, null.ok = TRUE, any.missing = FALSE)+ |
+
239 | +151x | +
+ checkmate::assert_list(server_args, null.ok = TRUE, names = "named")+ |
+
240 | +149x | +
+ checkmate::assert_list(ui_args, null.ok = TRUE, names = "named")+ |
+
241 | ++ | + + | +
242 | +147x | +
+ if (!missing(filters)) {+ |
+
243 | +! | +
+ checkmate::assert_character(filters, min.len = 1, null.ok = TRUE, any.missing = FALSE)+ |
+
244 | +! | +
+ datanames <- filters+ |
+
245 | +! | +
+ msg <-+ |
+
246 | +! | +
+ "The `filters` argument is deprecated and will be removed in the next release. Please use `datanames` instead."+ |
+
247 | +! | +
+ logger::log_warn(msg)+ |
+
248 | +! | +
+ warning(msg)+ |
+
249 | ++ |
+ }+ |
+
250 | ++ | + + | +
251 | +147x | +
+ if (label == "global_filters") {+ |
+
252 | +1x | +
+ stop(+ |
+
253 | +1x | +
+ sprintf("module(label = \"%s\", ...\n ", label),+ |
+
254 | +1x | +
+ "Label 'global_filters' is reserved in teal. Please change to something else.",+ |
+
255 | +1x | +
+ call. = FALSE+ |
+
256 | ++ |
+ )+ |
+
257 | ++ |
+ }+ |
+
258 | +146x | +
+ if (label == "Report previewer") {+ |
+
259 | +! | +
+ stop(+ |
+
260 | +! | +
+ sprintf("module(label = \"%s\", ...\n ", label),+ |
+
261 | +! | +
+ "Label 'Report previewer' is reserved in teal.",+ |
+
262 | +! | +
+ call. = FALSE+ |
+
263 | ++ |
+ )+ |
+
264 | ++ |
+ }+ |
+
265 | +146x | +
+ server_formals <- names(formals(server))+ |
+
266 | +146x | +
+ if (!(+ |
+
267 | +146x | +
+ "id" %in% server_formals ||+ |
+
268 | +146x | +
+ all(c("input", "output", "session") %in% server_formals)+ |
+
269 | ++ |
+ )) {+ |
+
270 | +2x | +
+ stop(+ |
+
271 | +2x | +
+ "\nmodule() `server` argument requires a function with following arguments:",+ |
+
272 | +2x | +
+ "\n - id - teal will set proper shiny namespace for this module.",+ |
+
273 | +2x | +
+ "\n - input, output, session (not recommended) - then shiny::callModule will be used to call a module.",+ |
+
274 | +2x | +
+ "\n\nFollowing arguments can be used optionaly:",+ |
+
275 | +2x | +
+ "\n - `data` - module will receive list of reactive (filtered) data specified in the `filters` argument",+ |
+
276 | +2x | +
+ "\n - `datasets` - module will receive `FilteredData`. See `help(teal.slice::FilteredData)`",+ |
+
277 | +2x | +
+ "\n - `reporter` - module will receive `Reporter`. See `help(teal.reporter::Reporter)`",+ |
+
278 | +2x | +
+ "\n - `filter_panel_api` - module will receive `FilterPanelAPI`. (See [teal.slice::FilterPanelAPI]).",+ |
+
279 | +2x | +
+ "\n - `...` server_args elements will be passed to the module named argument or to the `...`"+ |
+
280 | ++ |
+ )+ |
+
281 | ++ |
+ }+ |
+
282 | ++ | + + | +
283 | +144x | +
+ if (!is.element("data", server_formals) && !is.null(datanames)) {+ |
+
284 | +61x | +
+ message(sprintf("module \"%s\" server function takes no data so \"datanames\" will be ignored", label))+ |
+
285 | +61x | +
+ datanames <- NULL+ |
+
286 | ++ |
+ }+ |
+
287 | ++ | + + | +
288 | +144x | +
+ srv_extra_args <- setdiff(names(server_args), server_formals)+ |
+
289 | +144x | +
+ if (length(srv_extra_args) > 0 && !"..." %in% server_formals) {+ |
+
290 | +1x | +
+ stop(+ |
+
291 | +1x | +
+ "\nFollowing `server_args` elements have no equivalent in the formals of the `server`:\n",+ |
+
292 | +1x | +
+ paste(paste(" -", srv_extra_args), collapse = "\n"),+ |
+
293 | +1x | +
+ "\n\nUpdate the `server` arguments by including above or add `...`"+ |
+
294 | ++ |
+ )+ |
+
295 | ++ |
+ }+ |
+
296 | ++ | + + | +
297 | +143x | +
+ ui_formals <- names(formals(ui))+ |
+
298 | +143x | +
+ if (!"id" %in% ui_formals) {+ |
+
299 | +1x | +
+ stop(+ |
+
300 | +1x | +
+ "\nmodule() `ui` argument requires a function with following arguments:",+ |
+
301 | +1x | +
+ "\n - id - teal will set proper shiny namespace for this module.",+ |
+
302 | +1x | +
+ "\n\nFollowing arguments can be used optionaly:",+ |
+
303 | +1x | +
+ "\n - `data` - module will receive list of reactive (filtered) data specied in the `filters` argument",+ |
+
304 | +1x | +
+ "\n - `datasets` - module will receive `FilteredData`. See `help(teal.slice::FilteredData)`",+ |
+
305 | +1x | +
+ "\n - `...` ui_args elements will be passed to the module argument of the same name or to the `...`"+ |
+
306 | ++ |
+ )+ |
+
307 | ++ |
+ }+ |
+
308 | ++ | + + | +
309 | +142x | +
+ ui_extra_args <- setdiff(names(ui_args), ui_formals)+ |
+
310 | +142x | +
+ if (length(ui_extra_args) > 0 && !"..." %in% ui_formals) {+ |
+
311 | +1x | +
+ stop(+ |
+
312 | +1x | +
+ "\nFollowing `ui_args` elements have no equivalent in the formals of `ui`:\n",+ |
+
313 | +1x | +
+ paste(paste(" -", ui_extra_args), collapse = "\n"),+ |
+
314 | +1x | +
+ "\n\nUpdate the `ui` arguments by including above or add `...`"+ |
+
315 | ++ |
+ )+ |
+
316 | ++ |
+ }+ |
+
317 | ++ | + + | +
318 | +141x | +
+ structure(+ |
+
319 | +141x | +
+ list(+ |
+
320 | +141x | +
+ label = label,+ |
+
321 | +141x | +
+ server = server, ui = ui, datanames = unique(datanames),+ |
+
322 | +141x | +
+ server_args = server_args, ui_args = ui_args+ |
+
323 | ++ |
+ ),+ |
+
324 | +141x | +
+ class = "teal_module"+ |
+
325 | ++ |
+ )+ |
+
326 | ++ |
+ }+ |
+
327 | ++ | + + | +
328 | ++ | + + | +
329 | ++ |
+ #' Get module depth+ |
+
330 | ++ |
+ #'+ |
+
331 | ++ |
+ #' Depth starts at 0, so a single `teal.module` has depth 0.+ |
+
332 | ++ |
+ #' Nesting it increases overall depth by 1.+ |
+
333 | ++ |
+ #'+ |
+
334 | ++ |
+ #' @inheritParams init+ |
+
335 | ++ |
+ #' @param depth optional, integer determining current depth level+ |
+
336 | ++ |
+ #'+ |
+
337 | ++ |
+ #' @return depth level for given module+ |
+
338 | ++ |
+ #' @keywords internal+ |
+
339 | ++ |
+ #'+ |
+
340 | ++ |
+ #' @examples+ |
+
341 | ++ |
+ #' mods <- modules(+ |
+
342 | ++ |
+ #' label = "d1",+ |
+
343 | ++ |
+ #' modules(+ |
+
344 | ++ |
+ #' label = "d2",+ |
+
345 | ++ |
+ #' modules(+ |
+
346 | ++ |
+ #' label = "d3",+ |
+
347 | ++ |
+ #' module(label = "aaa1"), module(label = "aaa2"), module(label = "aaa3")+ |
+
348 | ++ |
+ #' ),+ |
+
349 | ++ |
+ #' module(label = "bbb")+ |
+
350 | ++ |
+ #' ),+ |
+
351 | ++ |
+ #' module(label = "ccc")+ |
+
352 | ++ |
+ #' )+ |
+
353 | ++ |
+ #' stopifnot(teal:::modules_depth(mods) == 3L)+ |
+
354 | ++ |
+ #'+ |
+
355 | ++ |
+ #' mods <- modules(+ |
+
356 | ++ |
+ #' label = "a",+ |
+
357 | ++ |
+ #' modules(+ |
+
358 | ++ |
+ #' label = "b1", module(label = "c")+ |
+
359 | ++ |
+ #' ),+ |
+
360 | ++ |
+ #' module(label = "b2")+ |
+
361 | ++ |
+ #' )+ |
+
362 | ++ |
+ #' stopifnot(teal:::modules_depth(mods) == 2L)+ |
+
363 | ++ |
+ modules_depth <- function(modules, depth = 0L) {+ |
+
364 | +12x | +
+ checkmate::assert(+ |
+
365 | +12x | +
+ checkmate::check_class(modules, "teal_module"),+ |
+
366 | +12x | +
+ checkmate::check_class(modules, "teal_modules")+ |
+
367 | ++ |
+ )+ |
+
368 | +12x | +
+ checkmate::assert_int(depth, lower = 0)+ |
+
369 | +11x | +
+ if (inherits(modules, "teal_modules")) {+ |
+
370 | +4x | +
+ max(vapply(modules$children, modules_depth, integer(1), depth = depth + 1L))+ |
+
371 | ++ |
+ } else {+ |
+
372 | +7x | +
+ depth+ |
+
373 | ++ |
+ }+ |
+
374 | ++ |
+ }+ |
+
375 | ++ | + + | +
376 | ++ | + + | +
377 | ++ |
+ module_labels <- function(modules) {+ |
+
378 | +! | +
+ if (inherits(modules, "teal_modules")) {+ |
+
379 | +! | +
+ lapply(modules$children, module_labels)+ |
+
380 | ++ |
+ } else {+ |
+
381 | +! | +
+ modules$label+ |
+
382 | ++ |
+ }+ |
+
383 | ++ |
+ }+ |
+
384 | ++ | + + | +
385 | ++ |
+ #' Converts `teal_modules` to a string+ |
+
386 | ++ |
+ #'+ |
+
387 | ++ |
+ #' @param x (`teal_modules`) to print+ |
+
388 | ++ |
+ #' @param indent (`integer`) indent level;+ |
+
389 | ++ |
+ #' each `submodule` is indented one level more+ |
+
390 | ++ |
+ #' @param ... (optional) additional parameters to pass to recursive calls of `toString`+ |
+
391 | ++ |
+ #' @return (`character`)+ |
+
392 | ++ |
+ #' @export+ |
+
393 | ++ |
+ #' @rdname modules+ |
+
394 | ++ |
+ toString.teal_modules <- function(x, indent = 0, ...) { # nolint+ |
+
395 | ++ |
+ # argument must be `x` to be consistent with base method+ |
+
396 | +! | +
+ paste(c(+ |
+
397 | +! | +
+ paste0(rep(" ", indent), "+ ", x$label),+ |
+
398 | +! | +
+ unlist(lapply(x$children, toString, indent = indent + 1, ...))+ |
+
399 | +! | +
+ ), collapse = "\n")+ |
+
400 | ++ |
+ }+ |
+
401 | ++ | + + | +
402 | ++ |
+ #' Converts `teal_module` to a string+ |
+
403 | ++ |
+ #'+ |
+
404 | ++ |
+ #' @inheritParams toString.teal_modules+ |
+
405 | ++ |
+ #' @param x `teal_module`+ |
+
406 | ++ |
+ #' @param ... ignored+ |
+
407 | ++ |
+ #' @export+ |
+
408 | ++ |
+ #' @rdname module+ |
+
409 | ++ |
+ toString.teal_module <- function(x, indent = 0, ...) { # nolint+ |
+
410 | +! | +
+ paste0(paste(rep(" ", indent), collapse = ""), "+ ", x$label, collapse = "")+ |
+
411 | ++ |
+ }+ |
+
412 | ++ | + + | +
413 | ++ |
+ #' Prints `teal_modules`+ |
+
414 | ++ |
+ #' @param x `teal_modules`+ |
+
415 | ++ |
+ #' @param ... parameters passed to `toString`+ |
+
416 | ++ |
+ #' @export+ |
+
417 | ++ |
+ #' @rdname modules+ |
+
418 | ++ |
+ print.teal_modules <- function(x, ...) {+ |
+
419 | +! | +
+ s <- toString(x, ...)+ |
+
420 | +! | +
+ cat(s)+ |
+
421 | +! | +
+ return(invisible(s))+ |
+
422 | ++ |
+ }+ |
+
423 | ++ | + + | +
424 | ++ |
+ #' Prints `teal_module`+ |
+
425 | ++ |
+ #' @param x `teal_module`+ |
+
426 | ++ |
+ #' @param ... parameters passed to `toString`+ |
+
427 | ++ |
+ #' @export+ |
+
428 | ++ |
+ #' @rdname module+ |
+
429 | ++ |
+ print.teal_module <- print.teal_modules+ |
+
1 | ++ |
+ #' Create a `teal` module for previewing a report+ |
+
2 | ++ |
+ #'+ |
+
3 | ++ |
+ #' @description `r lifecycle::badge("experimental")`+ |
+
4 | ++ |
+ #' This function wraps [teal.reporter::reporter_previewer_ui()] and+ |
+
5 | ++ |
+ #' [teal.reporter::reporter_previewer_srv()] into a `teal_module` to be+ |
+
6 | ++ |
+ #' used in `teal` applications.+ |
+
7 | ++ |
+ #'+ |
+
8 | ++ |
+ #' If you are creating a `teal` application using [teal::init()] then this+ |
+
9 | ++ |
+ #' module will be added to your application automatically if any of your `teal modules`+ |
+
10 | ++ |
+ #' support report generation.+ |
+
11 | ++ |
+ #'+ |
+
12 | ++ |
+ #' @inheritParams module+ |
+
13 | ++ |
+ #' @param server_args (`named list`)\cr+ |
+
14 | ++ |
+ #' Arguments passed to [teal.reporter::reporter_previewer_srv()].+ |
+
15 | ++ |
+ #' @return `teal_module` (extended with `teal_module_previewer` class) containing the `teal.reporter` previewer+ |
+
16 | ++ |
+ #' functionality.+ |
+
17 | ++ |
+ #' @export+ |
+
18 | ++ |
+ reporter_previewer_module <- function(label = "Report previewer", server_args = list()) {+ |
+
19 | +4x | +
+ checkmate::assert_string(label)+ |
+
20 | +2x | +
+ checkmate::assert_list(server_args, names = "named")+ |
+
21 | +2x | +
+ checkmate::assert_true(all(names(server_args) %in% names(formals(teal.reporter::reporter_previewer_srv))))+ |
+
22 | ++ | + + | +
23 | +2x | +
+ logger::log_info("Initializing reporter_previewer_module")+ |
+
24 | ++ | + + | +
25 | +2x | +
+ srv <- function(id, reporter, ...) {+ |
+
26 | +! | +
+ teal.reporter::reporter_previewer_srv(id, reporter, ...)+ |
+
27 | ++ |
+ }+ |
+
28 | ++ | + + | +
29 | +2x | +
+ ui <- function(id, ...) {+ |
+
30 | +! | +
+ teal.reporter::reporter_previewer_ui(id, ...)+ |
+
31 | ++ |
+ }+ |
+
32 | ++ | + + | +
33 | +2x | +
+ module <- module(+ |
+
34 | +2x | +
+ label = "temporary label",+ |
+
35 | +2x | +
+ server = srv, ui = ui,+ |
+
36 | +2x | +
+ server_args = server_args, ui_args = list(), datanames = NULL+ |
+
37 | ++ |
+ )+ |
+
38 | ++ |
+ # Module is created with a placeholder label and the label is changed later.+ |
+
39 | ++ |
+ # This is to prevent another module being labeled "Report previewer".+ |
+
40 | +2x | +
+ class(module) <- c("teal_module_previewer", class(module))+ |
+
41 | +2x | +
+ module$label <- label+ |
+
42 | +2x | +
+ module+ |
+
43 | ++ |
+ }+ |
+
1 | ++ |
+ .onLoad <- function(libname, pkgname) { # nolint+ |
+
2 | ++ |
+ # adapted from https://github.com/r-lib/devtools/blob/master/R/zzz.R+ |
+
3 | +! | +
+ teal_default_options <- list(teal.show_js_log = FALSE)+ |
+
4 | ++ | + + | +
5 | +! | +
+ op <- options()+ |
+
6 | +! | +
+ toset <- !(names(teal_default_options) %in% names(op))+ |
+
7 | +! | +
+ if (any(toset)) options(teal_default_options[toset])+ |
+
8 | ++ | + + | +
9 | +! | +
+ options("shiny.sanitize.errors" = FALSE)+ |
+
10 | ++ | + + | +
11 | ++ |
+ # Set up the teal logger instance+ |
+
12 | +! | +
+ teal.logger::register_logger("teal")+ |
+
13 | ++ | + + | +
14 | +! | +
+ invisible()+ |
+
15 | ++ |
+ }+ |
+
16 | ++ | + + | +
17 | ++ |
+ .onAttach <- function(libname, pkgname) { # nolint+ |
+
18 | +2x | +
+ packageStartupMessage(+ |
+
19 | +2x | +
+ "\nYou are using teal version ",+ |
+
20 | ++ |
+ # `system.file` uses the `shim` of `system.file` by `teal`+ |
+
21 | ++ |
+ # we avoid `desc` dependency here to get the version+ |
+
22 | +2x | +
+ read.dcf(system.file("DESCRIPTION", package = "teal"))[, "Version"]+ |
+
23 | ++ |
+ )+ |
+
24 | ++ |
+ }+ |
+
25 | ++ | + + | +
26 | ++ |
+ # Use non-exported function(s) from teal.slice.+ |
+
27 | ++ |
+ # This is a temporary measure and will be removed two release cycles from now (now meaning 0.13.0).+ |
+
28 | ++ |
+ list_to_teal_slices <- getFromNamespace("list_to_teal_slices", "teal.slice")+ |
+
29 | ++ |
+ # This one is here because setdiff_teal_slice should not be exported from teal.slice.+ |
+
30 | ++ |
+ setdiff_teal_slices <- getFromNamespace("setdiff_teal_slices", "teal.slice")+ |
+
31 | ++ |
+ # This one is here because it is needed by c.teal_slices but we don't want it exported from teal.slice.+ |
+
32 | ++ |
+ coalesce_r <- getFromNamespace("coalesce_r", "teal.slice")+ |
+
33 | ++ |
+ # all *Block objects are private in teal.reporter+ |
+
34 | ++ |
+ RcodeBlock <- getFromNamespace("RcodeBlock", "teal.reporter") # nolint+ |
+
35 | ++ | + + | +
36 | ++ |
+ # Use non-exported function(s) from teal.code+ |
+
37 | ++ |
+ # This one is here because format_expression should not be exported from teal.code+ |
+
38 | ++ |
+ format_expression <- getFromNamespace("format_expression", "teal.code")+ |
+
1 | ++ |
+ #' Store teal_slices object to a file+ |
+
2 | ++ |
+ #'+ |
+
3 | ++ |
+ #' This function takes a `teal_slices` object and saves it to a file in `JSON` format.+ |
+
4 | ++ |
+ #' The `teal_slices` object contains information about filter states and can be used to+ |
+
5 | ++ |
+ #' create, modify, and delete filter states. The saved file can be later loaded using+ |
+
6 | ++ |
+ #' the `slices_restore` function.+ |
+
7 | ++ |
+ #'+ |
+
8 | ++ |
+ #' @param tss (`teal_slices`) object to be stored.+ |
+
9 | ++ |
+ #' @param file (`character(1)`) The file path where `teal_slices` object will be saved.+ |
+
10 | ++ |
+ #' The file extension should be `".json"`.+ |
+
11 | ++ |
+ #'+ |
+
12 | ++ |
+ #' @details `Date` class is stored in `"ISO8601"` format (`YYYY-MM-DD`). `POSIX*t` classes are converted to a+ |
+
13 | ++ |
+ #' character by using `format.POSIX*t(usetz = TRUE, tz = "UTC")` (`YYYY-MM-DD {N}{N}:{N}{N}:{N}{N} UTC`, where+ |
+
14 | ++ |
+ #' `{N} = [0-9]` is a number and `UTC` is `Coordinated Universal Time` timezone short-code).+ |
+
15 | ++ |
+ #' This format is assumed during `slices_restore`. All `POSIX*t` objects in `selected` or `choices` fields of+ |
+
16 | ++ |
+ #' `teal_slice` objects are always printed in `UTC` timezone as well.+ |
+
17 | ++ |
+ #'+ |
+
18 | ++ |
+ #' @return `NULL`, invisibly.+ |
+
19 | ++ |
+ #'+ |
+
20 | ++ |
+ #' @keywords internal+ |
+
21 | ++ |
+ #'+ |
+
22 | ++ |
+ #' @examples+ |
+
23 | ++ |
+ #' # Create a teal_slices object+ |
+
24 | ++ |
+ #' tss <- teal_slices(+ |
+
25 | ++ |
+ #' teal_slice(dataname = "data", varname = "var"),+ |
+
26 | ++ |
+ #' teal_slice(dataname = "data", expr = "x > 0", id = "positive_x", title = "Positive x")+ |
+
27 | ++ |
+ #' )+ |
+
28 | ++ |
+ #'+ |
+
29 | ++ |
+ #' if (interactive()) {+ |
+
30 | ++ |
+ #' # Store the teal_slices object to a file+ |
+
31 | ++ |
+ #' slices_store(tss, "path/to/file.json")+ |
+
32 | ++ |
+ #' }+ |
+
33 | ++ |
+ #'+ |
+
34 | ++ |
+ slices_store <- function(tss, file) {+ |
+
35 | +6x | +
+ checkmate::assert_class(tss, "teal_slices")+ |
+
36 | +6x | +
+ checkmate::assert_path_for_output(file, overwrite = TRUE, extension = "json")+ |
+
37 | ++ | + + | +
38 | +6x | +
+ cat(format(tss, trim_lines = FALSE), "\n", file = file)+ |
+
39 | ++ |
+ }+ |
+
40 | ++ | + + | +
41 | ++ |
+ #' Restore teal_slices object from a file+ |
+
42 | ++ |
+ #'+ |
+
43 | ++ |
+ #' This function takes a file path to a `JSON` file containing a `teal_slices` object+ |
+
44 | ++ |
+ #' and restores it to its original form. The restored `teal_slices` object can be used+ |
+
45 | ++ |
+ #' to access filter states and their corresponding attributes.+ |
+
46 | ++ |
+ #'+ |
+
47 | ++ |
+ #' @param file Path to file where `teal_slices` is stored. Must have a `.json` extension and read access.+ |
+
48 | ++ |
+ #'+ |
+
49 | ++ |
+ #' @return A `teal_slices` object restored from the file.+ |
+
50 | ++ |
+ #'+ |
+
51 | ++ |
+ #' @keywords internal+ |
+
52 | ++ |
+ #'+ |
+
53 | ++ |
+ #' @examples+ |
+
54 | ++ |
+ #' if (interactive()) {+ |
+
55 | ++ |
+ #' # Restore a teal_slices object from a file+ |
+
56 | ++ |
+ #' tss_restored <- slices_restore("path/to/file.json")+ |
+
57 | ++ |
+ #' }+ |
+
58 | ++ |
+ #'+ |
+
59 | ++ |
+ slices_restore <- function(file) {+ |
+
60 | +6x | +
+ checkmate::assert_file_exists(file, access = "r", extension = "json")+ |
+
61 | ++ | + + | +
62 | +6x | +
+ tss_json <- jsonlite::fromJSON(file, simplifyDataFrame = FALSE)+ |
+
63 | +6x | +
+ tss_json$slices <-+ |
+
64 | +6x | +
+ lapply(tss_json$slices, function(slice) {+ |
+
65 | +6x | +
+ for (field in c("selected", "choices")) {+ |
+
66 | +12x | +
+ if (!is.null(slice[[field]])) {+ |
+
67 | +9x | +
+ date_partial_regex <- "^[0-9]{4}-[0-9]{2}-[0-9]{2}"+ |
+
68 | +9x | +
+ time_stamp_regex <- paste0(date_partial_regex, "\\s[0-9]{2}:[0-9]{2}:[0-9]{2}\\sUTC$")+ |
+
69 | ++ | + + | +
70 | +9x | +
+ slice[[field]] <-+ |
+
71 | +9x | +
+ if (all(grepl(paste0(date_partial_regex, "$"), slice[[field]]))) {+ |
+
72 | +3x | +
+ as.Date(slice[[field]])+ |
+
73 | +9x | +
+ } else if (all(grepl(time_stamp_regex, slice[[field]]))) {+ |
+
74 | +3x | +
+ as.POSIXct(slice[[field]], tz = "UTC")+ |
+
75 | ++ |
+ } else {+ |
+
76 | +3x | +
+ slice[[field]]+ |
+
77 | ++ |
+ }+ |
+
78 | ++ |
+ }+ |
+
79 | ++ |
+ }+ |
+
80 | +6x | +
+ slice+ |
+
81 | ++ |
+ })+ |
+
82 | ++ | + + | +
83 | +6x | +
+ tss_elements <- lapply(tss_json$slices, as.teal_slice)+ |
+
84 | ++ | + + | +
85 | +6x | +
+ do.call(teal_slices, c(tss_elements, tss_json$attributes))+ |
+
86 | ++ |
+ }+ |
+
1 | ++ |
+ #' Landing Popup Module+ |
+
2 | ++ |
+ #'+ |
+
3 | ++ |
+ #' @description Creates a landing welcome popup for `teal` applications.+ |
+
4 | ++ |
+ #'+ |
+
5 | ++ |
+ #' This module is used to display a popup dialog when the application starts.+ |
+
6 | ++ |
+ #' The dialog blocks the access to the application and must be closed with a button before the application is viewed.+ |
+
7 | ++ |
+ #'+ |
+
8 | ++ |
+ #' @param label `character(1)` the label of the module.+ |
+
9 | ++ |
+ #' @param title `character(1)` the text to be displayed as a title of the popup.+ |
+
10 | ++ |
+ #' @param content The content of the popup. Passed to `...` of `shiny::modalDialog`. Can be a `character`+ |
+
11 | ++ |
+ #' or a list of `shiny.tag`s. See examples.+ |
+
12 | ++ |
+ #' @param buttons `shiny.tag` or a list of tags (`tagList`). Typically a `modalButton` or `actionButton`. See examples.+ |
+
13 | ++ |
+ #'+ |
+
14 | ++ |
+ #' @return A `teal_module` (extended with `teal_landing_module` class) to be used in `teal` applications.+ |
+
15 | ++ |
+ #'+ |
+
16 | ++ |
+ #' @examples+ |
+
17 | ++ |
+ #' app1 <- teal::init(+ |
+
18 | ++ |
+ #' data = teal_data(iris = iris),+ |
+
19 | ++ |
+ #' modules = teal::modules(+ |
+
20 | ++ |
+ #' teal::landing_popup_module(+ |
+
21 | ++ |
+ #' content = "A place for the welcome message or a disclaimer statement.",+ |
+
22 | ++ |
+ #' buttons = modalButton("Proceed")+ |
+
23 | ++ |
+ #' ),+ |
+
24 | ++ |
+ #' example_module()+ |
+
25 | ++ |
+ #' )+ |
+
26 | ++ |
+ #' )+ |
+
27 | ++ |
+ #' if (interactive()) {+ |
+
28 | ++ |
+ #' shinyApp(app1$ui, app1$server)+ |
+
29 | ++ |
+ #' }+ |
+
30 | ++ |
+ #'+ |
+
31 | ++ |
+ #' app2 <- teal::init(+ |
+
32 | ++ |
+ #' data = teal_data(iris = iris),+ |
+
33 | ++ |
+ #' modules = teal::modules(+ |
+
34 | ++ |
+ #' teal::landing_popup_module(+ |
+
35 | ++ |
+ #' title = "Welcome",+ |
+
36 | ++ |
+ #' content = tags$b(+ |
+
37 | ++ |
+ #' "A place for the welcome message or a disclaimer statement.",+ |
+
38 | ++ |
+ #' style = "color: red;"+ |
+
39 | ++ |
+ #' ),+ |
+
40 | ++ |
+ #' buttons = tagList(+ |
+
41 | ++ |
+ #' modalButton("Proceed"),+ |
+
42 | ++ |
+ #' actionButton("read", "Read more",+ |
+
43 | ++ |
+ #' onclick = "window.open('http://google.com', '_blank')"+ |
+
44 | ++ |
+ #' ),+ |
+
45 | ++ |
+ #' actionButton("close", "Reject", onclick = "window.close()")+ |
+
46 | ++ |
+ #' )+ |
+
47 | ++ |
+ #' ),+ |
+
48 | ++ |
+ #' example_module()+ |
+
49 | ++ |
+ #' )+ |
+
50 | ++ |
+ #' )+ |
+
51 | ++ |
+ #'+ |
+
52 | ++ |
+ #' if (interactive()) {+ |
+
53 | ++ |
+ #' shinyApp(app2$ui, app2$server)+ |
+
54 | ++ |
+ #' }+ |
+
55 | ++ |
+ #'+ |
+
56 | ++ |
+ #' @export+ |
+
57 | ++ |
+ landing_popup_module <- function(label = "Landing Popup",+ |
+
58 | ++ |
+ title = NULL,+ |
+
59 | ++ |
+ content = NULL,+ |
+
60 | ++ |
+ buttons = modalButton("Accept")) {+ |
+
61 | +! | +
+ checkmate::assert_string(label)+ |
+
62 | +! | +
+ checkmate::assert_string(title, null.ok = TRUE)+ |
+
63 | +! | +
+ checkmate::assert_multi_class(+ |
+
64 | +! | +
+ content,+ |
+
65 | +! | +
+ classes = c("character", "shiny.tag", "shiny.tag.list", "html"), null.ok = TRUE+ |
+
66 | ++ |
+ )+ |
+
67 | +! | +
+ checkmate::assert_multi_class(buttons, classes = c("shiny.tag", "shiny.tag.list"))+ |
+
68 | ++ | + + | +
69 | +! | +
+ logger::log_info("Initializing landing_popup_module")+ |
+
70 | ++ | + + | +
71 | +! | +
+ module <- module(+ |
+
72 | +! | +
+ label = label,+ |
+
73 | +! | +
+ server = function(id) {+ |
+
74 | +! | +
+ moduleServer(id, function(input, output, session) {+ |
+
75 | +! | +
+ showModal(+ |
+
76 | +! | +
+ modalDialog(+ |
+
77 | +! | +
+ id = "landingpopup",+ |
+
78 | +! | +
+ title = title,+ |
+
79 | +! | +
+ content,+ |
+
80 | +! | +
+ footer = buttons+ |
+
81 | ++ |
+ )+ |
+
82 | ++ |
+ )+ |
+
83 | ++ |
+ })+ |
+
84 | ++ |
+ }+ |
+
85 | ++ |
+ )+ |
+
86 | +! | +
+ class(module) <- c("teal_module_landing", class(module))+ |
+
87 | +! | +
+ module+ |
+
88 | ++ |
+ }+ |
+
1 | ++ |
+ #' Show R Code Modal+ |
+
2 | ++ |
+ #'+ |
+
3 | ++ |
+ #' @export+ |
+
4 | ++ |
+ #' @description `r lifecycle::badge("stable")`+ |
+
5 | ++ |
+ #' Use the [shiny::showModal()] function to show the R code inside.+ |
+
6 | ++ |
+ #'+ |
+
7 | ++ |
+ #' @param title (`character(1)`)\cr+ |
+
8 | ++ |
+ #' Title of the modal, displayed in the first comment of the R-code.+ |
+
9 | ++ |
+ #' @param rcode (`character`)\cr+ |
+
10 | ++ |
+ #' vector with R code to show inside the modal.+ |
+
11 | ++ |
+ #' @param session (`ShinySession` optional)\cr+ |
+
12 | ++ |
+ #' `shiny` Session object, if missing then [shiny::getDefaultReactiveDomain()] is used.+ |
+
13 | ++ |
+ #'+ |
+
14 | ++ |
+ #' @references [shiny::showModal()]+ |
+
15 | ++ |
+ show_rcode_modal <- function(title = NULL, rcode, session = getDefaultReactiveDomain()) {+ |
+
16 | +! | +
+ rcode <- paste(rcode, collapse = "\n")+ |
+
17 | ++ | + + | +
18 | +! | +
+ ns <- session$ns+ |
+
19 | +! | +
+ showModal(modalDialog(+ |
+
20 | +! | +
+ tagList(+ |
+
21 | +! | +
+ tags$div(+ |
+
22 | +! | +
+ actionButton(ns("copyRCode"), "Copy to Clipboard", `data-clipboard-target` = paste0("#", ns("r_code"))),+ |
+
23 | +! | +
+ modalButton("Dismiss"),+ |
+
24 | +! | +
+ style = "mb-4"+ |
+
25 | ++ |
+ ),+ |
+
26 | +! | +
+ tags$div(tags$pre(id = ns("r_code"), rcode)),+ |
+
27 | ++ |
+ ),+ |
+
28 | +! | +
+ title = title,+ |
+
29 | +! | +
+ footer = tagList(+ |
+
30 | +! | +
+ actionButton(ns("copyRCode"), "Copy to Clipboard", `data-clipboard-target` = paste0("#", ns("r_code"))),+ |
+
31 | +! | +
+ modalButton("Dismiss")+ |
+
32 | ++ |
+ ),+ |
+
33 | +! | +
+ size = "l",+ |
+
34 | +! | +
+ easyClose = TRUE+ |
+
35 | ++ |
+ ))+ |
+
36 | ++ | + + | +
37 | +! | +
+ return(NULL)+ |
+
38 | ++ |
+ }+ |
+
1 | ++ |
+ #' Data module for `teal` applications+ |
+
2 | ++ |
+ #'+ |
+
3 | ++ |
+ #' Creates `teal_data_module` object - a `shiny` module to supply or modify data in a `teal` application.+ |
+
4 | ++ |
+ #'+ |
+
5 | ++ |
+ #' This function creates a `shiny` module that allows for running data pre-processing code after the app starts.+ |
+
6 | ++ |
+ #' The body of the server function will be run in the app rather than in the global environment.+ |
+
7 | ++ |
+ #' This means it will be run every time the app starts, so use sparingly.+ |
+
8 | ++ |
+ #'+ |
+
9 | ++ |
+ #' Pass this module instead of a `teal_data` object in a call to [init()].+ |
+
10 | ++ |
+ #'+ |
+
11 | ++ |
+ #' See vignette \code{vignette("data-as-shiny-module", package = "teal")} for more details.+ |
+
12 | ++ |
+ #'+ |
+
13 | ++ |
+ #' @param ui (`function(id)`)\cr+ |
+
14 | ++ |
+ #' `shiny` module `ui` function; must only take `id` argument+ |
+
15 | ++ |
+ #' @param server (`function(id)`)\cr+ |
+
16 | ++ |
+ #' `shiny` module `ui` function; must only take `id` argument;+ |
+
17 | ++ |
+ #' must return reactive expression containing `teal_data` object+ |
+
18 | ++ |
+ #'+ |
+
19 | ++ |
+ #' @return Object of class `teal_data_module`.+ |
+
20 | ++ |
+ #'+ |
+
21 | ++ |
+ #' @examples+ |
+
22 | ++ |
+ #' data <- teal_data_module(+ |
+
23 | ++ |
+ #' ui = function(id) {+ |
+
24 | ++ |
+ #' ns <- NS(id)+ |
+
25 | ++ |
+ #' actionButton(ns("submit"), label = "Load data")+ |
+
26 | ++ |
+ #' },+ |
+
27 | ++ |
+ #' server = function(id) {+ |
+
28 | ++ |
+ #' moduleServer(id, function(input, output, session) {+ |
+
29 | ++ |
+ #' eventReactive(input$submit, {+ |
+
30 | ++ |
+ #' data <- within(+ |
+
31 | ++ |
+ #' teal_data(),+ |
+
32 | ++ |
+ #' {+ |
+
33 | ++ |
+ #' dataset1 <- iris+ |
+
34 | ++ |
+ #' dataset2 <- mtcars+ |
+
35 | ++ |
+ #' }+ |
+
36 | ++ |
+ #' )+ |
+
37 | ++ |
+ #' datanames(data) <- c("iris", "mtcars")+ |
+
38 | ++ |
+ #'+ |
+
39 | ++ |
+ #' data+ |
+
40 | ++ |
+ #' })+ |
+
41 | ++ |
+ #' })+ |
+
42 | ++ |
+ #' }+ |
+
43 | ++ |
+ #' )+ |
+
44 | ++ |
+ #' @export+ |
+
45 | ++ |
+ teal_data_module <- function(ui, server) {+ |
+
46 | +20x | +
+ checkmate::assert_function(ui, args = "id", nargs = 1)+ |
+
47 | +19x | +
+ checkmate::assert_function(server, args = "id", nargs = 1)+ |
+
48 | +18x | +
+ structure(+ |
+
49 | +18x | +
+ list(ui = ui, server = server),+ |
+
50 | +18x | +
+ class = "teal_data_module"+ |
+
51 | ++ |
+ )+ |
+
52 | ++ |
+ }+ |
+
1 | ++ |
+ # This file contains Shiny modules useful for debugging and developing teal.+ |
+
2 | ++ |
+ # We do not export the functions in this file. They are for+ |
+
3 | ++ |
+ # developers only and can be accessed via `:::`.+ |
+
4 | ++ | + + | +
5 | ++ |
+ #' Dummy module to show the filter calls generated by the right encoding panel+ |
+
6 | ++ |
+ #'+ |
+
7 | ++ |
+ #'+ |
+
8 | ++ |
+ #' Please do not remove, this is useful for debugging teal without+ |
+
9 | ++ |
+ #' dependencies and simplifies `\link[devtools]{load_all}` which otherwise fails+ |
+
10 | ++ |
+ #' and avoids session restarts!+ |
+
11 | ++ |
+ #'+ |
+
12 | ++ |
+ #' @param label `character` label of module+ |
+
13 | ++ |
+ #' @keywords internal+ |
+
14 | ++ |
+ #'+ |
+
15 | ++ |
+ #' @examples+ |
+
16 | ++ |
+ #' app <- init(+ |
+
17 | ++ |
+ #' data = list(iris = iris, mtcars = mtcars),+ |
+
18 | ++ |
+ #' modules = teal:::filter_calls_module(),+ |
+
19 | ++ |
+ #' header = "Simple teal app"+ |
+
20 | ++ |
+ #' )+ |
+
21 | ++ |
+ #' if (interactive()) {+ |
+
22 | ++ |
+ #' shinyApp(app$ui, app$server)+ |
+
23 | ++ |
+ #' }+ |
+
24 | ++ |
+ filter_calls_module <- function(label = "Filter Calls Module") { # nolint+ |
+
25 | +! | +
+ checkmate::assert_string(label)+ |
+
26 | ++ | + + | +
27 | +! | +
+ module(+ |
+
28 | +! | +
+ label = label,+ |
+
29 | +! | +
+ server = function(input, output, session, data) {+ |
+
30 | +! | +
+ checkmate::assert_class(data, "tdata")+ |
+
31 | ++ | + + | +
32 | +! | +
+ output$filter_calls <- renderText({+ |
+
33 | +! | +
+ get_code_tdata(data)+ |
+
34 | ++ |
+ })+ |
+
35 | ++ |
+ },+ |
+
36 | +! | +
+ ui = function(id, ...) {+ |
+
37 | +! | +
+ ns <- NS(id)+ |
+
38 | +! | +
+ div(+ |
+
39 | +! | +
+ h2("The following filter calls are generated:"),+ |
+
40 | +! | +
+ verbatimTextOutput(ns("filter_calls"))+ |
+
41 | ++ |
+ )+ |
+
42 | ++ |
+ },+ |
+
43 | +! | +
+ datanames = "all"+ |
+
44 | ++ |
+ )+ |
+
45 | ++ |
+ }+ |
+
"+y.value+"
";t=p.firstChild.firstChild;p.firstChild.cN=s.cN;s.parentNode.replaceChild(p.firstChild,s)}else{t.innerHTML=y.value}t.className=u;t.result={language:v,kw:y.keyword_count,re:y.r};if(y.second_best){t.second_best={language:y.second_best.language,kw:y.second_best.keyword_count,re:y.second_best.r}}}function o(){if(o.called){return}o.called=true;var r=document.getElementsByTagName("pre");for(var p=0;p