diff --git a/.github/ISSUE_TEMPLATE/cran-release.yaml b/.github/ISSUE_TEMPLATE/cran-release.yaml new file mode 100644 index 000000000..bab34fcff --- /dev/null +++ b/.github/ISSUE_TEMPLATE/cran-release.yaml @@ -0,0 +1,124 @@ +--- +name: 🎉 CRAN Release +description: Template for release to CRAN +title: "[Release]: " +labels: ["release"] +assignees: + - KlaudiaBB + - cicdguy + - shajoezhu +body: + - type: markdown + attributes: + value: | + ⚠️ Please do not link or mention any internal references in this issue. This includes internal URLs, intellectual property and references. + - type: textarea + id: blocked-by + attributes: + label: Blocked by + description: Any PRs or issues that this release is blocked by. + placeholder: Add a list of blocking PRs or issues here. + value: | + ### PRs + + - [ ] PR 1 + + ### Issues + + - [ ] Issue 1 + validations: + required: true + - type: textarea + id: pre-release + attributes: + label: Pre-release + description: Pre-requisites that must be fulfilled before initiating the release process. + placeholder: Add your list of pre-requisites here. + value: | + - [ ] Make sure you adhere to CRAN submission policy: https://cran.r-project.org/web/packages/submission_checklist.html; https://cran.r-project.org/web/packages/policies.html. + - [ ] Make sure that high priority bugs (label "priority" + "bug") have been resolved before going into the release. + - [ ] Review old/hanging PRs before going into the release (Optional). + - [ ] Revisit R-package's lifecycle badges (Optional). + - [ ] Make sure that all upstream dependencies of this package that need to be submitted to CRAN were accepted before going into release activities. + - [ ] Make sure integration tests are green 2-3 days before the release. Look carefully through logs (check for warnings and notes). + - [ ] Decide what gets merged in before starting release activities. + - type: textarea + id: release + attributes: + label: Release + description: The steps to be taken in order to create a release. + placeholder: Steps to create a release. + value: | + ### Prepare the release + + - [ ] Create a new release candidate branch + `git checkout -b release-candidate-vX.Y.Z` + - [ ] Update NEWS.md file: make sure it reflects a holistic summary of what has changed in the package. + - [ ] Remove the additional fields (`Remotes`) from the DESCRIPTION file where applicable. + - [ ] Make sure that the minimum dependency versions are updated in the DESCRIPTION file for the package and its reverse dependencies (Optional). + - [ ] Increase versioned dependency on {package name} to >=X.Y.Z (Optional). + - [ ] Commit your changes and create the PR on GitHub (add "[skip vbump]" in the PR title). Add all updates, commit, and push changes: + `# Make the necessary modifications to your files + # Stage the changes + git add + # Commit the changes + git commit -m "[skip vbump] " + git push origin release-candidate-vX.Y.Z` + + ### Test the release + + - [ ] Execute the manual tests on Shiny apps that are deployed on various hosting providers (Posit connect and shinyapps.io) - track the results in GitHub issue (Applicable only for frameworks that use Shiny). + - [ ] Monitor integration tests, if integration fails, create priority issues on the board. + - [ ] Execute UAT tests (Optional). + + ### CRAN submission + + - [ ] Tag the update(s) as a release candidate vX.Y.Z-rc (e.g. v0.5.3-rc1) on the release candidate branch (release-candidate-vX.Y.Z). + `# Create rc tag for submission for internal validation + git tag vX.Y.Z-rc + git push origin vX.Y.Z-rc` + - [ ] Build the package locally using the command:`R CMD build .` which will generate a .tar.gz file necessary for the CRAN submission. + - [ ] Submit the package to https://win-builder.r-project.org/upload.aspx for testing, for more details please see "Building and checking R source packages for Windows": https://win-builder.r-project.org/. + - [ ] Once tested, send the package that was built in the previous steps to CRAN via this form: https://cran.r-project.org/submit.html. + - [ ] Address CRAN feedback, tag the package vX.Y.Z-rc(n+1) and repeat the submission to CRAN whenever necessary. + - [ ] Get the package accepted and published on CRAN. + + ### Tag the release + + - [ ] If the additional fields were removed, add them back in a separate PR, and then merge the PR back to main (add "[skip vbump]" in the PR title). If nothing was removed just merge the PR you created in the "Prepare the release" section to 'main'. Note the commit hash of the merged commit. **Note:** additional commits might be added to the `main` branch by a bot or an automation - we do **NOT** want to tag this commit. + + ### Make sure of the following before continuing + + - [ ] CI checks are passing in GH before releasing the package. + - [ ] Shiny apps are deployable and there are no errors/warnings (Applicable only for frameworks that use Shiny). + + - [ ] Create a git tag with the final version set to vX.Y.Z on the main branch. In order to do this: + 1. Checkout the commit hash. + `git checkout ` + 2. Tag the hash with the release version (vX.Y.Z). + `git tag vX.Y.Z` + 3. Push the tag to make the final release. + `git push origin vX.Y.Z` + - [ ] Update downstream package dependencies to (>=X.Y.Z) in {package name}. + Note: Once the release tag is created, the package is automatically published to internal repositories. + - type: textarea + id: post-release + attributes: + label: Post-release + description: The list of activities to be completed after the release. + placeholder: The steps that must be taken after the release. + value: | + - [ ] Ensure that CRAN checks are passing for the package. + - [ ] Make sure that the package is published to internal repositories. + - [ ] Make sure internal documentation is up to date. + - [ ] Review and update installation instructions for the package wherever needed (Optional). + - [ ] Update all integration tests to reference the new release. + - [ ] Announce the release on ________. + - type: textarea + id: decision-tree + attributes: + label: Decision tree + description: Any decision tree(s) that would aid release management + placeholder: Any decision tree(s) that would aid release management. + value: | + Click [here](https://github.com/insightsengineering/.github/blob/main/.github/ISSUE_TEMPLATE/RELEASE_DECISION_TREE.md) to see the release decision tree. diff --git a/DESCRIPTION b/DESCRIPTION index 1638c3ced..2a7133cf3 100644 --- a/DESCRIPTION +++ b/DESCRIPTION @@ -71,4 +71,4 @@ Encoding: UTF-8 Language: en-US LazyData: true Roxygen: list(markdown = TRUE) -RoxygenNote: 7.2.3 +RoxygenNote: 7.3.1 diff --git a/NEWS.md b/NEWS.md index 6c08bc306..ac6fad305 100644 --- a/NEWS.md +++ b/NEWS.md @@ -26,7 +26,7 @@ * It is now possible to initialize the filter panel without the "Add filter variables" panel through `allow_add` in `teal_slices`. * It is now possible to set a filter that cannot be removed by the app user. See `anchored` argument in `teal_slice`. * It is now possible to set a filter whose selection cannot be changed. See `fixed` argument in `teal_slice`. -* It is now possible to limit choices within a variable to a single value only. See `multuple` argument in `teal_slice` . +* It is now possible to limit choices within a variable to a single value only. See `multiple` argument in `teal_slice` . * Changed appearance of filter cards to a collapsible accordion. * Replaced `sliderInput` with interactive `plotly` chart to allow the user to zoom in on the variable distribution. * Implemented reactive counts in single filter cards to compare filtered and unfiltered variable distributions. See `count_type` argument in `teal_slices`. @@ -59,7 +59,7 @@ ### Bug fixes * Fixed an error where the `RangeFilterState` produced an error when using `bootstrap 4`. -* Fixed a bug that caused the range slider to omit values selected programmatically through the filter API. +* Fixed a bug that caused the range slider to omit values selected by the filter API. * Fixed a bug where setting incorrect values for Date and Date time ranges caused the app to crash. ### Miscellaneous diff --git a/R/FilterPanelAPI.R b/R/FilterPanelAPI.R index 0dc974a29..288497832 100644 --- a/R/FilterPanelAPI.R +++ b/R/FilterPanelAPI.R @@ -1,20 +1,23 @@ +# FilterPanelAPI ------ + #' @name FilterPanelAPI #' @docType class #' #' @title Class to encapsulate the API of the filter panel of a teal app #' -#' @details -#' The purpose of this class is to encapsulate the API of the filter panel in a new class `FilterPanelAPI` so -#' that it can be passed and used in the `server` call of any module instead of passing the whole `FilteredData` -#' object. +#' @description +#' An API class for managing filter states in a teal application's filter panel. #' -#' This class is supported by methods to set, get, remove filter states in the filter panel API. +#' @details +#' The purpose of this class is to encapsulate the API of the filter panel in a +#' new class `FilterPanelAPI` so that it can be passed and used in the server +#' call of any module instead of passing the whole `FilteredData` object. #' -#' @export +#' This class is supported by methods to set, get, remove filter states in the +#' filter panel API. #' #' @examples -#' library(teal.slice) -#' fd <- teal.slice::init_filtered_data(list(iris = list(dataset = iris))) +#' fd <- init_filtered_data(list(iris = list(dataset = iris))) #' fpa <- FilterPanelAPI$new(fd) #' #' # get the actual filter state --> empty named list @@ -37,13 +40,15 @@ #' # get the actual filter state --> empty named list #' isolate(fpa$get_filter_state()) #' +#' @export +#' FilterPanelAPI <- R6::R6Class( # nolint "FilterPanelAPI", - ## __Public Methods ==== + # public methods ---- public = list( #' @description - #' Initialize a `FilterPanelAPI` object - #' @param datasets (`FilteredData`) object. + #' Initialize a `FilterPanelAPI` object. + #' @param datasets (`FilteredData`) #' initialize = function(datasets) { checkmate::assert_class(datasets, "FilteredData") @@ -76,7 +81,7 @@ FilterPanelAPI <- R6::R6Class( # nolint #' @description #' Remove one or more `FilterState` of a `FilteredDataset` in the `FilteredData` object. #' - #' @param filter (`teal_slices`)\cr + #' @param filter (`teal_slices`) #' specifying `FilterState` objects to remove; #' `teal_slice`s may contain only `dataname` and `varname`, other elements are ignored #' @@ -87,9 +92,10 @@ FilterPanelAPI <- R6::R6Class( # nolint invisible(NULL) }, - #' @description Remove all `FilterStates` of the `FilteredData` object. + #' @description + #' Remove all `FilterStates` of the `FilteredData` object. #' - #' @param datanames (`character`)\cr + #' @param datanames (`character`) #' `datanames` to remove their `FilterStates`; #' omit to remove all `FilterStates` in the `FilteredData` object #' @@ -101,7 +107,7 @@ FilterPanelAPI <- R6::R6Class( # nolint invisible(NULL) } ), - ## __Private Methods ==== + # private methods ---- private = list( filtered_data = NULL ) diff --git a/R/FilterState-utils.R b/R/FilterState-utils.R index bfe2372ce..dc035e537 100644 --- a/R/FilterState-utils.R +++ b/R/FilterState-utils.R @@ -1,28 +1,29 @@ #' Initializes `FilterState` #' -#' Initializes `FilterState` depending on a variable class.\cr -#' @param x (`vector`)\cr +#' Initializes `FilterState` depending on a variable class. +#' +#' @param x (`vector`) #' values of the variable used in filter -#' @param x_reactive (`reactive`)\cr +#' @param x_reactive (`reactive`) #' returning vector of the same type as `x`. Is used to update #' counts following the change in values of the filtered dataset. #' If it is set to `reactive(NULL)` then counts based on filtered #' dataset are not shown. -#' @param slice (`teal_slice`)\cr +#' @param slice (`teal_slice`) #' object created using [teal_slice()]. -#' @param extract_type (`character(0)`, `character(1)`)\cr +#' @param extract_type (`character`) #' specifying whether condition calls should be prefixed by `dataname`. Possible values: -#' \itemize{ -#' \item{`character(0)` (default)}{ `varname` in the condition call will not be prefixed} -#' \item{`"list"`}{ `varname` in the condition call will be returned as `$`} -#' \item{`"matrix"`}{ `varname` in the condition call will be returned as `[, ]`} -#' } +#' - `character(0)` (default) `varname` in the condition call will not be prefixed +#' - `"list"` `varname` in the condition call will be returned as `$` +#' - `"matrix"` `varname` in the condition call will be returned as `[, ]` #' @param ... additional arguments to be saved as a list in `private$extras` field #' -#' @keywords internal -#' #' @examples -#' filter_state <- teal.slice:::init_filter_state( +#' # use non-exported function from teal.slice +#' include_js_files <- getFromNamespace("include_js_files", "teal.slice") +#' init_filter_state <- getFromNamespace("init_filter_state", "teal.slice") +#' +#' filter_state <- init_filter_state( #' x = c(1:10, NA, Inf), #' x_reactive = reactive(c(1:10, NA, Inf)), #' slice = teal_slice( @@ -32,24 +33,26 @@ #' extract_type = "matrix" #' ) #' -#' shiny::isolate(filter_state$get_call()) -#' app <- shinyApp( -#' ui = fluidPage( -#' filter_state$ui(id = "app"), -#' verbatimTextOutput("call") -#' ), -#' server = function(input, output, session) { -#' filter_state$server("app") +#' isolate(filter_state$get_call()) #' -#' output$call <- renderText( -#' deparse1(filter_state$get_call(), collapse = "\n") -#' ) -#' } +#' ui <- fluidPage( +#' filter_state$ui(id = "app"), +#' verbatimTextOutput("call") #' ) +#' server <- function(input, output, session) { +#' filter_state$server("app") +#' +#' output$call <- renderText( +#' deparse1(filter_state$get_call(), collapse = "\n") +#' ) +#' } +#' #' if (interactive()) { -#' shinyApp(app$ui, app$server) +#' shinyApp(ui, server) #' } +#' #' @return `FilterState` object +#' @keywords internal init_filter_state <- function(x, x_reactive = reactive(NULL), slice, @@ -214,8 +217,7 @@ init_filter_state.POSIXlt <- function(x, #' Initialize a `FilterStateExpr` object #' -#' Initialize a `FilterStateExpr` object -#' @param slice (`teal_slice_expr`)\cr +#' @param slice (`teal_slice_expr`) #' object created using [teal_slice()]. `teal_slice` is stored #' in the class and `set_state` directly manipulates values within `teal_slice`. `get_state` #' returns `teal_slice` object which can be reused in other places. Beware, that `teal_slice` @@ -233,18 +235,12 @@ init_filter_state_expr <- function(slice) { #' #' Determines the color specification for the currently active Bootstrap color theme and returns one queried color. #' -#' @param color `character(1)` naming one of the available theme colors +#' @param color (`character(1)`) naming one of the available theme colors #' @param alpha either a `numeric(1)` or `character(1)` specifying transparency #' in the range of `0-1` or a hexadecimal value `00-ff`, respectively; #' set to NULL to omit adding the alpha channel #' #' @return Named `character(1)` containing a hexadecimal color representation. -#' -#' @examples -#' teal.slice:::fetch_bs_color("primary") -#' teal.slice:::fetch_bs_color("danger", 0.35) -#' teal.slice:::fetch_bs_color("danger", "80") -#' #' @keywords internal #' fetch_bs_color <- function(color, alpha = NULL) { diff --git a/R/FilterState.R b/R/FilterState.R index 28788396c..cbcb738d2 100644 --- a/R/FilterState.R +++ b/R/FilterState.R @@ -1,14 +1,15 @@ +# FilterState ------ + #' @name FilterState #' @docType class #' +#' @title `FilterState` abstract Class #' -#' @title `FilterState` Abstract Class -#' -#' @description Abstract class to encapsulate single filter state +#' @description Abstract class to encapsulate single filter state. #' #' @details #' This class is responsible for managing single filter item within -#' `FilteredData` class. Filter states depend on the variable type: +#' `FilteredData` class object. Filter states depend on the variable type: #' (`logical`, `integer`, `numeric`, `factor`, `character`, `Date`, `POSIXct`, `POSIXlt`) #' and returns `FilterState` object with class corresponding to input variable. #' Class controls single filter entry in `module_single_filter_item` and returns @@ -20,22 +21,21 @@ #' - `POSIXct`, `POSIXlt`: `class = DatetimeFilterState` #' - all `NA` entries: `class: FilterState`, cannot be filtered #' - default: `FilterState`, cannot be filtered -#' \cr +#' #' Each variable's filter state is an `R6` object which contains `choices`, #' `selected`, `varname`, `dataname`, `labels`, `na_count`, `keep_na` and other #' variable type specific fields (`keep_inf`, `inf_count`, `timezone`). -#' Object contains also shiny module (`ui` and `server`) which manages +#' Object also contains a `shiny` module (UI and server) which manages the #' state of the filter through reactive values `selected`, `keep_na`, `keep_inf` -#' which trigger `get_call()` and every R function call up in reactive chain. -#' \cr -#' \cr +#' which trigger `get_call()` and every `R` function call up in reactive chain. +#' #' @section Modifying state: #' Modifying a `FilterState` object is possible in three scenarios: -#' * In the interactive session by passing an appropriate `teal_slice` +#' - In the interactive session by passing an appropriate `teal_slice` #' to the `set_state` method, or using #' `set_selected`, `set_keep_na` or `set_keep_inf` methods. -#' * In a running application by changing appropriate inputs. -#' * In a running application by using [filter_state_api] which directly uses +#' - In a running application by changing appropriate inputs. +#' - In a running application by using [filter_state_api] which directly uses #' `set_state` method of the `InteractiveFilterState` object. #' #' @keywords internal @@ -46,26 +46,24 @@ FilterState <- R6::R6Class( # nolint public = list( #' @description - #' Initialize a `FilterState` object - #' @param x (`vector`)\cr + #' Initialize a `FilterState` object. + #' @param x (`vector`) #' values of the variable used in filter - #' @param x_reactive (`reactive`)\cr + #' @param x_reactive (`reactive`) #' returning vector of the same type as `x`. Is used to update #' counts following the change in values of the filtered dataset. #' If it is set to `reactive(NULL)` then counts based on filtered #' dataset are not shown. - #' @param slice (`teal_slice`)\cr + #' @param slice (`teal_slice`) #' object created by [teal_slice()] - #' @param extract_type (`character(0)`, `character(1)`)\cr + #' @param extract_type (`character`) #' specifying whether condition calls should be prefixed by `dataname`. Possible values: - #' \itemize{ - #' \item{`character(0)` (default)}{ `varname` in the condition call will not be prefixed} - #' \item{`"list"`}{ `varname` in the condition call will be returned as `$`} - #' \item{`"matrix"`}{ `varname` in the condition call will be returned as `[, ]`} - #' } + #' - `character(0)` (default) `varname` in the condition call will not be prefixed + #' - `"list"` `varname` in the condition call will be returned as `$` + #' - `"matrix"` `varname` in the condition call will be returned as `[, ]` #' @param ... additional arguments to be saved as a list in `private$extras` field #' - #' @return self invisibly + #' @return `self` invisibly #' initialize = function(x, x_reactive = reactive(NULL), @@ -114,8 +112,8 @@ FilterState <- R6::R6Class( # nolint #' @description #' Returns a formatted string representing this `FilterState` object. #' - #' @param show_all `logical(1)` passed to `format.teal_slice` - #' @param trim_lines `logical(1)` passed to `format.teal_slice` + #' @param show_all (`logical(1)`) passed to `format.teal_slice` + #' @param trim_lines (`logical(1)`) passed to `format.teal_slice` #' #' @return `character(1)` the formatted string #' @@ -141,7 +139,7 @@ FilterState <- R6::R6Class( # nolint #' - `fixed` state is prevented from changing state #' - `anchored` state is prevented from removing state #' - #' @param state a `teal_slice` object + #' @param state (`teal_slice`) #' #' @return `self` invisibly #' @@ -194,10 +192,10 @@ FilterState <- R6::R6Class( # nolint }, #' @description - #' Shiny module server. + #' `shiny` module server. #' - #' @param id (`character(1)`)\cr - #' shiny module instance id + #' @param id (`character(1)`) + #' `shiny` module instance id #' #' @return `moduleServer` function which returns reactive value #' signaling that remove button has been clicked @@ -282,10 +280,10 @@ FilterState <- R6::R6Class( # nolint }, #' @description - #' Shiny module UI. + #' `shiny` UI module. #' - #' @param id (`character(1)`)\cr - #' shiny element (module instance) id; + #' @param id (`character(1)`) + #' `shiny` element (module instance) id; #' the UI for this class contains simple message stating that it is not supported #' @param parent_id (`character(1)`) id of the `FilterStates` card container ui = function(id, parent_id = "cards") { @@ -377,7 +375,7 @@ FilterState <- R6::R6Class( # nolint #' @description #' Destroy observers stored in `private$observers`. #' - #' @return NULL invisibly + #' @return `NULL` invisibly #' destroy_observers = function() { if (!is.null(private$destroy_shiny)) { @@ -404,7 +402,7 @@ FilterState <- R6::R6Class( # nolint # private methods ---- - ## setters for state features ---- + # setters for state features ---- # @description # Set values that can be selected from. @@ -415,7 +413,7 @@ FilterState <- R6::R6Class( # nolint # @description # Set selection. # - # @param value (`vector`)\cr + # @param value (`vector`) # value(s) that come from filter selection; values are set in the # module server after a selection is made in the app interface; # values are stored in `teal_slice$selected` which is reactive; @@ -450,7 +448,7 @@ FilterState <- R6::R6Class( # nolint # @description # Set whether to keep NAs. # - # @param value `logical(1)`\cr + # @param value (`logical(1)`) # value(s) which come from the filter selection. Value is set in `server` # modules after selecting check-box-input in the shiny interface. Values are set to # `private$teal_slice$keep_na` @@ -474,7 +472,7 @@ FilterState <- R6::R6Class( # nolint # @description # Set whether to keep Infs # - # @param value (`logical(1)`)\cr + # @param value (`logical(1)`) # Value(s) which come from the filter selection. Value is set in `server` # modules after selecting check-box-input in the shiny interface. Values are set to # `private$teal_slice$keep_inf` @@ -494,7 +492,7 @@ FilterState <- R6::R6Class( # nolint invisible(NULL) }, - ## getters for state features ---- + # getters for state features ---- # @description # Returns dataname. @@ -563,7 +561,7 @@ FilterState <- R6::R6Class( # nolint shiny::isolate(isTRUE(private$teal_slice$multiple)) }, - ## other ---- + # other ---- # @description # Returns variable label. @@ -659,7 +657,7 @@ FilterState <- R6::R6Class( # nolint } }, - ## shiny modules ----- + # shiny modules ----- # @description # Server module to display filter summary diff --git a/R/FilterStateChoices.R b/R/FilterStateChoices.R index 9ef1d85d9..54f693eb9 100644 --- a/R/FilterStateChoices.R +++ b/R/FilterStateChoices.R @@ -1,16 +1,23 @@ +# ChoicesFilterState ------ + #' @name ChoicesFilterState -#' @title `FilterState` object for factor or character variable -#' @description Manages choosing elements from a set #' @docType class -#' @keywords internal #' +#' @title `FilterState` object for factor or character variable +#' +#' @description Manages choosing elements from a set. #' #' @examples -#' filter_state <- teal.slice:::ChoicesFilterState$new( +#' # use non-exported function from teal.slice +#' include_css_files <- getFromNamespace("include_css_files", "teal.slice") +#' include_js_files <- getFromNamespace("include_js_files", "teal.slice") +#' ChoicesFilterState <- getFromNamespace("ChoicesFilterState", "teal.slice") +#' +#' filter_state <- ChoicesFilterState$new( #' x = c(LETTERS, NA), #' slice = teal_slice(varname = "x", dataname = "data") #' ) -#' shiny::isolate(filter_state$get_call()) +#' isolate(filter_state$get_call()) #' filter_state$set_state( #' teal_slice( #' dataname = "data", @@ -19,15 +26,14 @@ #' keep_na = TRUE #' ) #' ) -#' shiny::isolate(filter_state$get_call()) +#' isolate(filter_state$get_call()) #' #' # working filter in an app -#' library(shiny) #' library(shinyjs) #' #' data_choices <- c(sample(letters[1:4], 100, replace = TRUE), NA) #' attr(data_choices, "label") <- "lowercase letters" -#' fs <- teal.slice:::ChoicesFilterState$new( +#' fs <- ChoicesFilterState$new( #' x = data_choices, #' slice = teal_slice( #' dataname = "data", varname = "variable", selected = c("a", "c"), keep_na = TRUE @@ -36,8 +42,8 @@ #' #' ui <- fluidPage( #' useShinyjs(), -#' teal.slice:::include_css_files(pattern = "filter-panel"), -#' teal.slice:::include_js_files(pattern = "count-bar-labels"), +#' include_css_files(pattern = "filter-panel"), +#' include_js_files(pattern = "count-bar-labels"), #' column(4, div( #' h4("ChoicesFilterState"), #' fs$ui("fs") @@ -102,6 +108,8 @@ #' shinyApp(ui, server) #' } #' +#' @keywords internal +#' ChoicesFilterState <- R6::R6Class( # nolint "ChoicesFilterState", inherit = FilterState, @@ -111,28 +119,27 @@ ChoicesFilterState <- R6::R6Class( # nolint public = list( #' @description - #' Initialize a `InteractiveFilterState` object + #' Initialize a `InteractiveFilterState` object. #' - #' @param x (`vector`)\cr + #' @param x (`vector`) #' values of the variable used in filter - #' @param x_reactive (`reactive`)\cr + #' @param x_reactive (`reactive`) #' returning vector of the same type as `x`. Is used to update #' counts following the change in values of the filtered dataset. #' If it is set to `reactive(NULL)` then counts based on filtered #' dataset are not shown. - #' @param slice (`teal_slice`)\cr + #' @param slice (`teal_slice`) #' object created using [teal_slice()]. `teal_slice` is stored #' in the class and `set_state` directly manipulates values within `teal_slice`. `get_state` #' returns `teal_slice` object which can be reused in other places. Beware, that `teal_slice` #' is a `reactiveValues` which means that changes in particular object are automatically #' reflected in all places which refer to the same `teal_slice`. - #' @param extract_type (`character(0)`, `character(1)`)\cr + #' @param extract_type (`character`) #' whether condition calls should be prefixed by `dataname`. Possible values: - #' \itemize{ - #' \item{`character(0)` (default)}{ `varname` in the condition call will not be prefixed} - #' \item{`"list"`}{ `varname` in the condition call will be returned as `$`} - #' \item{`"matrix"`}{ `varname` in the condition call will be returned as `[, ]`} - #' } + #' - `character(0)` (default) `varname` in the condition call will not be prefixed + #' - `"list"` `varname` in the condition call will be returned as `$` + #' - `"matrix"` `varname` in the condition call will be returned as `[, ]` + #' #' @param ... additional arguments to be saved as a list in `private$extras` field #' initialize = function(x, @@ -191,7 +198,7 @@ ChoicesFilterState <- R6::R6Class( # nolint #' ` %in% c()` with #' optional `is.na()`. #' @param dataname name of data set; defaults to `private$get_dataname()` - #' @return (`call`) or `NULL` + #' @return `call` or `NULL` #' get_call = function(dataname) { if (isFALSE(private$is_any_filtered())) { @@ -316,7 +323,7 @@ ChoicesFilterState <- R6::R6Class( # nolint values <- as.character(values) if (anyNA(values)) stop() }, - error = function(e) stop("The vactor of set values must contain values coercible to character.") + error = function(e) stop("The vector of set values must contain values coercible to character.") ) values }, @@ -348,7 +355,7 @@ ChoicesFilterState <- R6::R6Class( # nolint # UI Module for `ChoicesFilterState`. # This UI element contains available choices selection and # checkbox whether to keep or not keep the `NA` values. - # @param id (`character(1)`)\cr + # @param id (`character(1)`) # id of shiny element ui_inputs = function(id) { ns <- NS(id) @@ -421,7 +428,7 @@ ChoicesFilterState <- R6::R6Class( # nolint # @description # Server module - # @param id (`character(1)`)\cr + # @param id (`character(1)`) # an ID string that corresponds with the ID used to call the module's UI function. # @return `moduleServer` function which returns `NULL` server_inputs = function(id) { diff --git a/R/FilterStateDate.R b/R/FilterStateDate.R index 273bb6fc5..e78ea298e 100644 --- a/R/FilterStateDate.R +++ b/R/FilterStateDate.R @@ -1,17 +1,24 @@ +# DateFilterState ------ + #' @name DateFilterState -#' @title `FilterState` object for Date variable -#' @description Manages choosing a range of Dates #' @docType class -#' @keywords internal #' +#' @title `FilterState` object for `Date` variable +#' +#' @description Manages choosing a range of `Date`s. #' #' @examples -#' filter_state <- teal.slice:::DateFilterState$new( +#' # use non-exported function from teal.slice +#' include_css_files <- getFromNamespace("include_css_files", "teal.slice") +#' include_js_files <- getFromNamespace("include_js_files", "teal.slice") +#' DateFilterState <- getFromNamespace("DateFilterState", "teal.slice") +#' +#' filter_state <- DateFilterState$new( #' x = c(Sys.Date() + seq(1:10), NA), #' slice = teal_slice(varname = "x", dataname = "data"), #' extract_type = character(0) #' ) -#' shiny::isolate(filter_state$get_call()) +#' isolate(filter_state$get_call()) #' filter_state$set_state( #' teal_slice( #' dataname = "data", @@ -20,15 +27,14 @@ #' keep_na = TRUE #' ) #' ) -#' shiny::isolate(filter_state$get_call()) +#' isolate(filter_state$get_call()) #' #' # working filter in an app -#' library(shiny) #' library(shinyjs) #' #' dates <- c(Sys.Date() - 100, Sys.Date()) #' data_date <- c(seq(from = dates[1], to = dates[2], length.out = 100), NA) -#' fs <- teal.slice:::DateFilterState$new( +#' fs <- DateFilterState$new( #' x = data_date, #' slice = teal_slice( #' dataname = "data", varname = "x", selected = data_date[c(47, 98)], keep_na = TRUE @@ -37,8 +43,8 @@ #' #' ui <- fluidPage( #' useShinyjs(), -#' teal.slice:::include_css_files(pattern = "filter-panel"), -#' teal.slice:::include_js_files(pattern = "count-bar-labels"), +#' include_css_files(pattern = "filter-panel"), +#' include_js_files(pattern = "count-bar-labels"), #' column(4, div( #' h4("DateFilterState"), #' fs$ui("fs") @@ -96,6 +102,8 @@ #' shinyApp(ui, server) #' } #' +#' @keywords internal +#' DateFilterState <- R6::R6Class( # nolint "DateFilterState", inherit = FilterState, @@ -105,28 +113,27 @@ DateFilterState <- R6::R6Class( # nolint public = list( #' @description - #' Initialize a `FilterState` object + #' Initialize a `FilterState` object. #' - #' @param x (`Date`)\cr + #' @param x (`Date`) #' values of the variable used in filter - #' @param x_reactive (`reactive`)\cr + #' @param x_reactive (`reactive`) #' returning vector of the same type as `x`. Is used to update #' counts following the change in values of the filtered dataset. #' If it is set to `reactive(NULL)` then counts based on filtered #' dataset are not shown. - #' @param slice (`teal_slice`)\cr + #' @param slice (`teal_slice`) #' object created using [teal_slice()]. `teal_slice` is stored #' in the class and `set_state` directly manipulates values within `teal_slice`. `get_state` #' returns `teal_slice` object which can be reused in other places. Beware, that `teal_slice` #' is a `reactiveValues` which means that changes in particular object are automatically #' reflected in all places which refer to the same `teal_slice`. - #' @param extract_type (`character(0)`, `character(1)`)\cr + #' @param extract_type (`character`) #' whether condition calls should be prefixed by `dataname`. Possible values: - #' \itemize{ - #' \item{`character(0)` (default)}{ `varname` in the condition call will not be prefixed} - #' \item{`"list"`}{ `varname` in the condition call will be returned as `$`} - #' \item{`"matrix"`}{ `varname` in the condition call will be returned as `[, ]`} - #' } + #' - `character(0)` (default) `varname` in the condition call will not be prefixed + #' - `"list"` `varname` in the condition call will be returned as `$` + #' - `"matrix"` `varname` in the condition call will be returned as `[, ]` + #' #' @param ... additional arguments to be saved as a list in `private$extras` field #' initialize = function(x, @@ -157,8 +164,8 @@ DateFilterState <- R6::R6Class( # nolint #' For this class returned call looks like #' ` >= & <= ` with #' optional `is.na()`. - #' @param dataname `character(1)` containing possibly prefixed name of data set - #' @return (`call`) + #' @param dataname (`character(1)`) containing possibly prefixed name of data set + #' @return `call` #' get_call = function(dataname) { if (isFALSE(private$is_any_filtered())) { @@ -266,7 +273,7 @@ DateFilterState <- R6::R6Class( # nolint # UI Module for `DateFilterState`. # This UI element contains two date selections for `min` and `max` # of the range and a checkbox whether to keep the `NA` values. - # @param id (`character(1)`)\cr + # @param id (`character(1)`) # id of shiny element ui_inputs = function(id) { ns <- NS(id) @@ -306,7 +313,7 @@ DateFilterState <- R6::R6Class( # nolint # @description # Server module - # @param id (`character(1)`)\cr + # @param id (`character(1)`) # an ID string that corresponds with the ID used to call the module's UI function. # @return `moduleServer` function which returns `NULL` server_inputs = function(id) { diff --git a/R/FilterStateDatettime.R b/R/FilterStateDatettime.R index d8282be04..17f290e81 100644 --- a/R/FilterStateDatettime.R +++ b/R/FilterStateDatettime.R @@ -1,17 +1,24 @@ +# DatetimeFilterState ------ + #' @rdname DatetimeFilterState -#' @title `FilterState` object for `POSIXct` variable -#' @description Manages choosing a range of date-times #' @docType class -#' @keywords internal #' +#' @title `FilterState` object for `POSIXct` variable +#' +#' @description Manages choosing a range of date-times. #' #' @examples -#' filter_state <- teal.slice:::DatetimeFilterState$new( +#' # use non-exported function from teal.slice +#' include_css_files <- getFromNamespace("include_css_files", "teal.slice") +#' include_js_files <- getFromNamespace("include_js_files", "teal.slice") +#' DatetimeFilterState <- getFromNamespace("DatetimeFilterState", "teal.slice") +#' +#' filter_state <- DatetimeFilterState$new( #' x = c(Sys.time() + seq(0, by = 3600, length.out = 10), NA), #' slice = teal_slice(varname = "x", dataname = "data"), #' extract_type = character(0) #' ) -#' shiny::isolate(filter_state$get_call()) +#' isolate(filter_state$get_call()) #' filter_state$set_state( #' teal_slice( #' dataname = "data", @@ -20,15 +27,14 @@ #' keep_na = TRUE #' ) #' ) -#' shiny::isolate(filter_state$get_call()) +#' isolate(filter_state$get_call()) #' #' # working filter in an app -#' library(shiny) #' library(shinyjs) #' #' datetimes <- as.POSIXct(c("2012-01-01 12:00:00", "2020-01-01 12:00:00")) #' data_datetime <- c(seq(from = datetimes[1], to = datetimes[2], length.out = 100), NA) -#' fs <- teal.slice:::DatetimeFilterState$new( +#' fs <- DatetimeFilterState$new( #' x = data_datetime, #' slice = teal_slice( #' varname = "x", dataname = "data", selected = data_datetime[c(47, 98)], keep_na = TRUE @@ -37,8 +43,8 @@ #' #' ui <- fluidPage( #' useShinyjs(), -#' teal.slice:::include_css_files(pattern = "filter-panel"), -#' teal.slice:::include_js_files(pattern = "count-bar-labels"), +#' include_css_files(pattern = "filter-panel"), +#' include_js_files(pattern = "count-bar-labels"), #' column(4, div( #' h4("DatetimeFilterState"), #' fs$ui("fs") @@ -102,6 +108,8 @@ #' shinyApp(ui, server) #' } #' +#' @keywords internal +#' DatetimeFilterState <- R6::R6Class( # nolint "DatetimeFilterState", inherit = FilterState, @@ -117,26 +125,26 @@ DatetimeFilterState <- R6::R6Class( # nolint #' timezone of the app user. App user timezone is taken from `session$userData$timezone` #' and is set only if object is initialized in `shiny`. #' - #' @param x (`POSIXct` or `POSIXlt`)\cr + #' @param x (`POSIXct` or `POSIXlt`) #' values of the variable used in filter - #' @param x_reactive (`reactive`)\cr + #' @param x_reactive (`reactive`) #' returning vector of the same type as `x`. Is used to update #' counts following the change in values of the filtered dataset. #' If it is set to `reactive(NULL)` then counts based on filtered #' dataset are not shown. - #' @param slice (`teal_slice`)\cr + #' @param slice (`teal_slice`) #' object created using [teal_slice()]. `teal_slice` is stored #' in the class and `set_state` directly manipulates values within `teal_slice`. `get_state` #' returns `teal_slice` object which can be reused in other places. Beware, that `teal_slice` #' is a `reactiveValues` which means that changes in particular object are automatically #' reflected in all places which refer to the same `teal_slice`. - #' @param extract_type (`character(0)`, `character(1)`)\cr + #' @param extract_type (`character`) #' whether condition calls should be prefixed by `dataname`. Possible values: - #' \itemize{ - #' \item{`character(0)` (default)}{ `varname` in the condition call will not be prefixed} - #' \item{`"list"`}{ `varname` in the condition call will be returned as `$`} - #' \item{`"matrix"`}{ `varname` in the condition call will be returned as `[, ]`} - #' } + #' + #' - `character(0)` (default) `varname` in the condition call will not be prefixed + #' - `"list"` `varname` in the condition call will be returned as `$` + #' - `"matrix"` `varname` in the condition call will be returned as `[, ]` + #' #' @param ... additional arguments to be saved as a list in `private$extras` field #' initialize = function(x, @@ -168,7 +176,7 @@ DatetimeFilterState <- R6::R6Class( # nolint #' ` >= as.POSIXct() & <= )` #' with optional `is.na()`. #' @param dataname name of data set; defaults to `private$get_dataname()` - #' @return (`call`) + #' @return `call` #' get_call = function(dataname) { if (isFALSE(private$is_any_filtered())) { @@ -236,8 +244,10 @@ DatetimeFilterState <- R6::R6Class( # nolint private$set_is_choice_limited(private$x, choices) private$x <- private$x[ - (as.POSIXct(trunc(private$x, units = "secs")) >= choices[1L] & - as.POSIXct(trunc(private$x, units = "secs")) <= choices[2L]) | is.na(private$x) + ( + as.POSIXct(trunc(private$x, units = "secs")) >= choices[1L] & + as.POSIXct(trunc(private$x, units = "secs")) <= choices[2L] + ) | is.na(private$x) ] private$teal_slice$choices <- choices invisible(NULL) @@ -303,7 +313,7 @@ DatetimeFilterState <- R6::R6Class( # nolint # UI Module for `DatetimeFilterState`. # This UI element contains two date-time selections for `min` and `max` # of the range and a checkbox whether to keep the `NA` values. - # @param id (`character(1)`)\cr + # @param id (`character(1)`) # id of shiny element ui_inputs = function(id) { ns <- NS(id) @@ -369,7 +379,7 @@ DatetimeFilterState <- R6::R6Class( # nolint # @description # Server module - # @param id (`character(1)`)\cr + # @param id (`character(1)`) # an ID string that corresponds with the ID used to call the module's UI function. # @return `moduleServer` function which returns `NULL` server_inputs = function(id) { diff --git a/R/FilterStateEmpty.R b/R/FilterStateEmpty.R index fef117ef4..2d45b504c 100644 --- a/R/FilterStateEmpty.R +++ b/R/FilterStateEmpty.R @@ -1,19 +1,27 @@ +# EmptyFilterState ------ + #' @name EmptyFilterState -#' @title `FilterState` object for empty variable -#' @description `FilterState` subclass representing an empty variable #' @docType class -#' @keywords internal #' +#' @title `FilterState` object for empty variable +#' +#' @description `FilterState` subclass representing an empty variable. #' #' @examples -#' filter_state <- teal.slice:::EmptyFilterState$new( +#' # use non-exported function from teal.slice +#' include_js_files <- getFromNamespace("include_js_files", "teal.slice") +#' EmptyFilterState <- getFromNamespace("EmptyFilterState", "teal.slice") +#' +#' filter_state <- EmptyFilterState$new( #' x = NA, #' slice = teal_slice(varname = "x", dataname = "data"), #' extract_type = character(0) #' ) -#' shiny::isolate(filter_state$get_call()) +#' isolate(filter_state$get_call()) #' filter_state$set_state(teal_slice(dataname = "data", varname = "x", keep_na = TRUE)) -#' shiny::isolate(filter_state$get_call()) +#' isolate(filter_state$get_call()) +#' +#' @keywords internal #' EmptyFilterState <- R6::R6Class( # nolint "EmptyFilterState", @@ -25,26 +33,25 @@ EmptyFilterState <- R6::R6Class( # nolint #' @description #' Initialize `EmptyFilterState` object. #' - #' @param x (`vector`)\cr + #' @param x (`vector`) #' values of the variable used in filter - #' @param x_reactive (`reactive`)\cr + #' @param x_reactive (`reactive`) #' returning vector of the same type as `x`. Is used to update #' counts following the change in values of the filtered dataset. #' If it is set to `reactive(NULL)` then counts based on filtered #' dataset are not shown. - #' @param slice (`teal_slice`)\cr + #' @param slice (`teal_slice`) #' object created using [teal_slice()]. `teal_slice` is stored #' in the class and `set_state` directly manipulates values within `teal_slice`. `get_state` #' returns `teal_slice` object which can be reused in other places. Beware, that `teal_slice` #' is a `reactiveValues` which means that changes in particular object are automatically #' reflected in all places which refer to the same `teal_slice`. - #' @param extract_type (`character(0)`, `character(1)`)\cr + #' @param extract_type (`character`) #' whether condition calls should be prefixed by `dataname`. Possible values: - #' \itemize{ - #' \item{`character(0)` (default)}{ `varname` in the condition call will not be prefixed} - #' \item{`"list"`}{ `varname` in the condition call will be returned as `$`} - #' \item{`"matrix"`}{ `varname` in the condition call will be returned as `[, ]`} - #' } + #' - `character(0)` (default) `varname` in the condition call will not be prefixed + #' - `"list"` `varname` in the condition call will be returned as `$` + #' - `"matrix"` `varname` in the condition call will be returned as `[, ]` + #' #' @param ... additional arguments to be saved as a list in `private$extras` field #' initialize = function(x, @@ -118,7 +125,7 @@ EmptyFilterState <- R6::R6Class( # nolint # UI Module for `EmptyFilterState`. # This UI element contains a checkbox input to filter or keep missing values. # - # @param id (`character(1)`)\cr + # @param id (`character(1)`) # shiny element (module instance) id # ui_inputs = function(id) { @@ -134,7 +141,7 @@ EmptyFilterState <- R6::R6Class( # nolint # @description # Controls state of the `keep_na` checkbox input. # - # @param id (`character(1)`)\cr + # @param id (`character(1)`) # shiny module instance id # # @return `moduleServer` function which returns `NULL` diff --git a/R/FilterStateExpr.R b/R/FilterStateExpr.R index b4fe6214c..7d64c1f13 100644 --- a/R/FilterStateExpr.R +++ b/R/FilterStateExpr.R @@ -1,19 +1,22 @@ +# FilterStateExpr ------ + #' @name FilterStateExpr #' @docType class #' -#' -#' @title `FilterStateExpr` Class +#' @title `FilterStateExpr` `R6` class #' #' @description Class to handle filter expression. #' -#' #' @details -#' This class is responsible for displaying filter card and returning filter expression -#' -#' @keywords internal +#' This class is responsible for displaying filter card and returning filter expression. #' #' @examples -#' filter_state <- teal.slice:::FilterStateExpr$new( +#' # use non-exported function from teal.slice +#' include_js_files <- getFromNamespace("include_js_files", "teal.slice") +#' include_css_files <- getFromNamespace("include_css_files", "teal.slice") +#' FilterStateExpr <- getFromNamespace("FilterStateExpr", "teal.slice") +#' +#' filter_state <- FilterStateExpr$new( #' slice = teal_slice( #' dataname = "x", #' id = "FA", @@ -24,13 +27,12 @@ #' filter_state$get_call() #' #' # working filter in an app -#' library(shiny) #' library(shinyjs) #' #' ui <- fluidPage( #' useShinyjs(), -#' teal.slice:::include_css_files(pattern = "filter-panel"), -#' teal.slice:::include_js_files(pattern = "count-bar-labels"), +#' include_css_files(pattern = "filter-panel"), +#' include_js_files(pattern = "count-bar-labels"), #' column(4, div( #' h4("ChoicesFilterState"), #' filter_state$ui("fs") @@ -55,13 +57,16 @@ #' if (interactive()) { #' shinyApp(ui, server) #' } +#' +#' @keywords internal +#' FilterStateExpr <- R6::R6Class( # nolint classname = "FilterStateExpr", # public methods ---- public = list( #' @description - #' Initialize a `FilterStateExpr` object - #' @param slice (`teal_slice_expr`)\cr + #' Initialize a `FilterStateExpr` object. + #' @param slice (`teal_slice_expr`) #' object created by [teal_slice()] #' @return `FilterStateExpr` initialize = function(slice) { @@ -73,8 +78,8 @@ FilterStateExpr <- R6::R6Class( # nolint #' @description #' Returns a formatted string representing this `FilterStateExpr` object. #' - #' @param show_all `logical(1)` passed to `format.teal_slice` - #' @param trim_lines `logical(1)` passed to `format.teal_slice` + #' @param show_all (`logical(1)`) passed to `format.teal_slice` + #' @param trim_lines (`logical(1)`) passed to `format.teal_slice` #' #' @return `character(1)` the formatted string #' @@ -106,7 +111,7 @@ FilterStateExpr <- R6::R6Class( # nolint #' @description #' Sets filtering state. #' - #' @param state a `teal_slice` object + #' @param state (`teal_slice`) #' #' @return `self` invisibly #' @@ -116,7 +121,7 @@ FilterStateExpr <- R6::R6Class( # nolint }, #' @description - #' Get reproducible call + #' Get reproducible call. #' #' @param dataname (`ignored`) for a consistency with `FilterState` #' @@ -132,7 +137,7 @@ FilterStateExpr <- R6::R6Class( # nolint #' @description #' Destroy observers stored in `private$observers`. #' - #' @return NULL invisibly + #' @return `NULL` invisibly #' destroy_observers = function() { lapply(private$observers, function(x) x$destroy()) @@ -146,10 +151,10 @@ FilterStateExpr <- R6::R6Class( # nolint # public shiny modules ---- #' @description - #' Shiny module server. + #' `shiny` module server. #' - #' @param id (`character(1)`)\cr - #' shiny module instance id + #' @param id (`character(1)`) + #' `shiny` module instance id #' #' @return `moduleServer` function which returns reactive value #' signaling that remove button has been clicked @@ -172,10 +177,10 @@ FilterStateExpr <- R6::R6Class( # nolint }, #' @description - #' Shiny module UI. + #' `shiny` module UI. #' - #' @param id (`character(1)`)\cr - #' shiny element (module instance) id; + #' @param id (`character(1)`) + #' `shiny` element (module instance) id; #' the UI for this class contains simple message stating that it is not supported #' @param parent_id (`character(1)`) id of the `FilterStates` card container ui = function(id, parent_id = "cards") { diff --git a/R/FilterStateLogical.R b/R/FilterStateLogical.R index 73f6133da..091a14a25 100644 --- a/R/FilterStateLogical.R +++ b/R/FilterStateLogical.R @@ -1,35 +1,42 @@ +# LogicalFilterState ------ + #' @name LogicalFilterState -#' @title `FilterState` object for logical variable -#' @description Manages choosing a logical state #' @docType class -#' @keywords internal #' +#' @title `FilterState` object for logical variable +#' +#' @description Manages choosing a logical state. #' #' @examples -#' filter_state <- teal.slice:::LogicalFilterState$new( +#' # use non-exported function from teal.slice +#' include_css_files <- getFromNamespace("include_css_files", "teal.slice") +#' include_js_files <- getFromNamespace("include_js_files", "teal.slice") +#' LogicalFilterState <- getFromNamespace("LogicalFilterState", "teal.slice") +#' +#' +#' filter_state <- LogicalFilterState$new( #' x = sample(c(TRUE, FALSE, NA), 10, replace = TRUE), #' slice = teal_slice(varname = "x", dataname = "data") #' ) -#' shiny::isolate(filter_state$get_call()) +#' isolate(filter_state$get_call()) #' filter_state$set_state( #' teal_slice(dataname = "data", varname = "x", selected = TRUE, keep_na = TRUE) #' ) -#' shiny::isolate(filter_state$get_call()) +#' isolate(filter_state$get_call()) #' #' # working filter in an app -#' library(shiny) #' library(shinyjs) #' #' data_logical <- c(sample(c(TRUE, FALSE), 10, replace = TRUE), NA) -#' fs <- teal.slice:::LogicalFilterState$new( +#' fs <- LogicalFilterState$new( #' x = data_logical, #' slice = teal_slice(dataname = "data", varname = "x", selected = FALSE, keep_na = TRUE) #' ) #' #' ui <- fluidPage( #' useShinyjs(), -#' teal.slice:::include_css_files(pattern = "filter-panel"), -#' teal.slice:::include_js_files(pattern = "count-bar-labels"), +#' include_css_files(pattern = "filter-panel"), +#' include_js_files(pattern = "count-bar-labels"), #' column(4, div( #' h4("LogicalFilterState"), #' fs$ui("fs") @@ -82,6 +89,8 @@ #' shinyApp(ui, server) #' } #' +#' @keywords internal +#' LogicalFilterState <- R6::R6Class( # nolint "LogicalFilterState", inherit = FilterState, @@ -90,28 +99,27 @@ LogicalFilterState <- R6::R6Class( # nolint public = list( #' @description - #' Initialize a `FilterState` object + #' Initialize a `FilterState` object. #' - #' @param x (`logical`)\cr + #' @param x (`logical`) #' values of the variable used in filter - #' @param x_reactive (`reactive`)\cr + #' @param x_reactive (`reactive`) #' returning vector of the same type as `x`. Is used to update #' counts following the change in values of the filtered dataset. #' If it is set to `reactive(NULL)` then counts based on filtered #' dataset are not shown. - #' @param slice (`teal_slice`)\cr + #' @param slice (`teal_slice`) #' object created using [teal_slice()]. `teal_slice` is stored #' in the class and `set_state` directly manipulates values within `teal_slice`. `get_state` #' returns `teal_slice` object which can be reused in other places. Beware, that `teal_slice` #' is a `reactiveValues` which means that changes in particular object are automatically #' reflected in all places which refer to the same `teal_slice`. - #' @param extract_type (`character(0)`, `character(1)`)\cr + #' @param extract_type (`character`) #' whether condition calls should be prefixed by `dataname`. Possible values: - #' \itemize{ - #' \item{`character(0)` (default)}{ `varname` in the condition call will not be prefixed} - #' \item{`"list"`}{ `varname` in the condition call will be returned as `$`} - #' \item{`"matrix"`}{ `varname` in the condition call will be returned as `[, ]`} - #' } + #' - `character(0)` (default) `varname` in the condition call will not be prefixed + #' - `"list"` `varname` in the condition call will be returned as `$` + #' - `"matrix"` `varname` in the condition call will be returned as `[, ]` + #' #' @param ... additional arguments to be saved as a list in `private$extras` field #' initialize = function(x, @@ -143,7 +151,7 @@ LogicalFilterState <- R6::R6Class( # nolint #' For `LogicalFilterState` it's a `!` or `` and optionally #' `is.na()` #' @param dataname name of data set; defaults to `private$get_dataname()` - #' @return (`call`) + #' @return `call` #' get_call = function(dataname) { if (isFALSE(private$is_any_filtered())) { @@ -210,8 +218,10 @@ LogicalFilterState <- R6::R6Class( # nolint TRUE } else if (all(private$choices_counts > 0)) { TRUE - } else if (setequal(private$get_selected(), private$get_choices()) && - !anyNA(private$get_selected(), private$get_choices())) { + } else if ( + setequal(private$get_selected(), private$get_choices()) && + !anyNA(private$get_selected(), private$get_choices()) + ) { TRUE } else if (!isTRUE(private$get_keep_na()) && private$na_count > 0) { TRUE @@ -226,7 +236,7 @@ LogicalFilterState <- R6::R6Class( # nolint # UI Module for `EmptyFilterState`. # This UI element contains available choices selection and # checkbox whether to keep or not keep the `NA` values. - # @param id (`character(1)`)\cr + # @param id (`character(1)`) # id of shiny element ui_inputs = function(id) { ns <- NS(id) @@ -277,7 +287,7 @@ LogicalFilterState <- R6::R6Class( # nolint # @description # Server module # - # @param id (`character(1)`)\cr + # @param id (`character(1)`) # an ID string that corresponds with the ID used to call the module's UI function. # @return `moduleServer` function which returns `NULL` server_inputs = function(id) { diff --git a/R/FilterStateRange.R b/R/FilterStateRange.R index 0e2f2e458..f18e5d4e8 100644 --- a/R/FilterStateRange.R +++ b/R/FilterStateRange.R @@ -1,16 +1,23 @@ +# RangeFilterState ------ + #' @name RangeFilterState -#' @title `FilterState` object for numeric variable -#' @description Manages choosing a numeric range #' @docType class -#' @keywords internal #' +#' @title `FilterState` object for numeric variable +#' +#' @description Manages choosing a numeric range. #' #' @examples -#' filter_state <- teal.slice:::RangeFilterState$new( +#' # use non-exported function from teal.slice +#' include_css_files <- getFromNamespace("include_css_files", "teal.slice") +#' include_js_files <- getFromNamespace("include_js_files", "teal.slice") +#' RangeFilterState <- getFromNamespace("RangeFilterState", "teal.slice") +#' +#' filter_state <- RangeFilterState$new( #' x = c(NA, Inf, seq(1:10)), #' slice = teal_slice(varname = "x", dataname = "data") #' ) -#' shiny::isolate(filter_state$get_call()) +#' isolate(filter_state$get_call()) #' filter_state$set_state( #' teal_slice( #' dataname = "data", @@ -20,14 +27,13 @@ #' keep_inf = TRUE #' ) #' ) -#' shiny::isolate(filter_state$get_call()) +#' isolate(filter_state$get_call()) #' #' # working filter in an app -#' library(shiny) #' library(shinyjs) #' #' data_range <- c(runif(100, 0, 1), NA, Inf) -#' fs <- teal.slice:::RangeFilterState$new( +#' fs <- RangeFilterState$new( #' x = data_range, #' slice = teal_slice( #' dataname = "data", @@ -40,8 +46,8 @@ #' #' ui <- fluidPage( #' useShinyjs(), -#' teal.slice:::include_css_files(pattern = "filter-panel"), -#' teal.slice:::include_js_files(pattern = "count-bar-labels"), +#' include_css_files(pattern = "filter-panel"), +#' include_js_files(pattern = "count-bar-labels"), #' column(4, div( #' h4("RangeFilterState"), #' fs$ui("fs") @@ -110,6 +116,7 @@ #' if (interactive()) { #' shinyApp(ui, server) #' } +#' @keywords internal #' RangeFilterState <- R6::R6Class( # nolint "RangeFilterState", @@ -119,27 +126,26 @@ RangeFilterState <- R6::R6Class( # nolint public = list( #' @description - #' Initialize a `FilterState` object for range selection - #' @param x (`numeric`)\cr + #' Initialize a `FilterState` object for range selection. + #' @param x (`numeric`) #' values of the variable used in filter - #' @param x_reactive (`reactive`)\cr + #' @param x_reactive (`reactive`) #' returning vector of the same type as `x`. Is used to update #' counts following the change in values of the filtered dataset. #' If it is set to `reactive(NULL)` then counts based on filtered #' dataset are not shown. - #' @param slice (`teal_slice`)\cr + #' @param slice (`teal_slice`) #' object created using [teal_slice()]. `teal_slice` is stored #' in the class and `set_state` directly manipulates values within `teal_slice`. `get_state` #' returns `teal_slice` object which can be reused in other places. Beware, that `teal_slice` #' is a `reactiveValues` which means that changes in particular object are automatically #' reflected in all places which refer to the same `teal_slice`. - #' @param extract_type (`character(0)`, `character(1)`)\cr + #' @param extract_type (`character`) #' whether condition calls should be prefixed by `dataname`. Possible values: - #' \itemize{ - #' \item{`character(0)` (default)}{ `varname` in the condition call will not be prefixed} - #' \item{`"list"`}{ `varname` in the condition call will be returned as `$`} - #' \item{`"matrix"`}{ `varname` in the condition call will be returned as `[, ]`} - #' } + #' - `character(0)` (default) `varname` in the condition call will not be prefixed + #' - `"list"` `varname` in the condition call will be returned as `$` + #' - `"matrix"` `varname` in the condition call will be returned as `[, ]` + #' #' @param ... additional arguments to be saved as a list in `private$extras` field #' initialize = function(x, @@ -230,7 +236,7 @@ RangeFilterState <- R6::R6Class( # nolint #' ` >= & <= ` with #' optional `is.na()` and `is.finite()`. #' @param dataname name of data set; defaults to `private$get_dataname()` - #' @return (`call`) + #' @return `call` #' get_call = function(dataname) { if (isFALSE(private$is_any_filtered())) { @@ -248,8 +254,8 @@ RangeFilterState <- R6::R6Class( # nolint }, #' @description - #' Returns current `keep_inf` selection - #' @return (`logical(1)`) + #' Returns current `keep_inf` selection. + #' @return `logical(1)` get_keep_inf = function() { private$teal_slice$keep_inf } @@ -397,7 +403,7 @@ RangeFilterState <- R6::R6Class( # nolint # UI Module for `RangeFilterState`. # This UI element contains two values for `min` and `max` # of the range and two checkboxes whether to keep the `NA` or `Inf` values. - # @param id (`character(1)`)\cr + # @param id (`character(1)`) # id of shiny element ui_inputs = function(id) { ns <- NS(id) @@ -469,7 +475,7 @@ RangeFilterState <- R6::R6Class( # nolint # @description # Server module - # @param id (`character(1)`)\cr + # @param id (`character(1)`) # an ID string that corresponds with the ID used to call the module's UI function. # return `moduleServer` function which returns `NULL` server_inputs = function(id) { diff --git a/R/FilterStates-utils.R b/R/FilterStates-utils.R index 27b33245b..2301e7f0e 100644 --- a/R/FilterStates-utils.R +++ b/R/FilterStates-utils.R @@ -1,55 +1,59 @@ #' Initialize `FilterStates` object #' -#' Initialize `FilterStates` object -#' @param data (`data.frame`, `MultiAssayExperiment`, `SummarizedExperiment`, `matrix`)\cr -#' the R object which `subset` function is applied on. -#' @param data_reactive (`function(sid)`)\cr +#' @param data (`data.frame` or `MultiAssayExperiment` or `SummarizedExperiment` or `matrix`) +#' the `R` object which `subset` function is applied on. +#' @param data_reactive (`function(sid)`) #' should return an object of the same type as `data` or `NULL`. -#' This object is needed for the `FilterState` shiny module to update +#' This object is needed for the `FilterState` `shiny` module to update #' counts if filtered data changes. #' If function returns `NULL` then filtered counts #' are not shown. Function has to have `sid` argument being a character which #' is related to `sid` argument in the `get_call` method. -#' @param dataname (`character(1)`)\cr +#' @param dataname (`character(1)`) #' name of the data used in the expression #' specified to the function argument attached to this `FilterStates`. -#' @param datalabel (`character(0)` or `character(1)`)\cr +#' @param datalabel (`NULL` or `character(1)`) #' text label value. #' @param ... (optional) #' additional arguments for specific classes: keys. #' @keywords internal -#' @export #' @examples -#' library(shiny) +#' # use non-exported function from teal.slice +#' init_filter_states <- getFromNamespace("init_filter_states", "teal.slice") +#' #' df <- data.frame( #' character = letters, #' numeric = seq_along(letters), #' date = seq(Sys.Date(), length.out = length(letters), by = "1 day"), #' datetime = seq(Sys.time(), length.out = length(letters), by = "33.33 hours") #' ) -#' rf <- teal.slice:::init_filter_states( +#' rf <- init_filter_states( #' data = df, #' dataname = "DF" #' ) -#' app <- shinyApp( -#' ui = fluidPage( -#' actionButton("clear", span(icon("xmark"), "Remove all filters")), -#' rf$ui_add(id = "add"), -#' rf$ui_active("states"), -#' verbatimTextOutput("expr"), -#' ), -#' server = function(input, output, session) { -#' rf$srv_add(id = "add") -#' rf$srv_active(id = "states") -#' output$expr <- renderText({ -#' deparse1(rf$get_call(), collapse = "\n") -#' }) -#' observeEvent(input$clear, rf$state_list_empty()) -#' } +#' +#' ui <- fluidPage( +#' actionButton("clear", span(icon("xmark"), "Remove all filters")), +#' rf$ui_add(id = "add"), +#' rf$ui_active("states"), +#' verbatimTextOutput("expr"), #' ) +#' +#' server <- function(input, output, session) { +#' rf$srv_add(id = "add") +#' rf$srv_active(id = "states") +#' output$expr <- renderText({ +#' deparse1(rf$get_call(), collapse = "\n") +#' }) +#' observeEvent(input$clear, rf$clear_filter_states()) +#' } +#' #' if (interactive()) { -#' shinyApp(app$ui, app$server) +#' shinyApp(ui, server) #' } +#' +#' @export +#' init_filter_states <- function(data, data_reactive = reactive(NULL), dataname, @@ -132,10 +136,13 @@ init_filter_states.SummarizedExperiment <- function(data, # nolint #' #' Gets filterable variable names from a given object. The names match variables #' of classes in an vector `teal.slice:::.filterable_class`. -#' @param data (`object`)\cr -#' the R object containing elements which class can be checked through `vapply` or `apply`. -#' +#' @param data +#' the `R` object containing elements which class can be checked through `vapply` or `apply`. +#' @return `character` vector of matched element names #' @examples +#' # use non-exported function from teal.slice +#' get_supported_filter_varnames <- getFromNamespace("get_supported_filter_varnames", "teal.slice") +#' #' df <- data.frame( #' a = letters[1:3], #' b = 1:3, @@ -143,8 +150,7 @@ init_filter_states.SummarizedExperiment <- function(data, # nolint #' d = Sys.time() + 1:3, #' z = complex(3) #' ) -#' teal.slice:::get_supported_filter_varnames(df) -#' @return `character` vector of matched element names +#' get_supported_filter_varnames(df) #' @keywords internal get_supported_filter_varnames <- function(data) { UseMethod("get_supported_filter_varnames") @@ -186,16 +192,16 @@ get_supported_filter_varnames.MultiAssayExperiment <- function(data) { # nolint } } -#' @title Returns a `choices_labeled` object +#' Returns a `choices_labeled` object #' -#' @param data (`data.frame`, `DFrame`, `list`)\cr +#' @param data (`data.frame` or `DFrame` or `list`) #' where labels can be taken from in case when `varlabels` is not specified. #' `data` must be specified if `varlabels` is not specified. -#' @param choices (`character`)\cr +#' @param choices (`character`) #' the vector of chosen variables -#' @param varlabels (`character`)\cr +#' @param varlabels (`character`) #' the labels of variables in data -#' @param keys (`character`)\cr +#' @param keys (`character`) #' the names of the key columns in data #' @return `character(0)` if choices are empty; a `choices_labeled` object otherwise #' @keywords internal @@ -216,6 +222,8 @@ data_choices_labeled <- function(data, ) } +#' @noRd +#' @keywords internal get_varlabels <- function(data) { if (!is.array(data)) { vapply( diff --git a/R/FilterStates.R b/R/FilterStates.R index cf22d87bd..ebec24e41 100644 --- a/R/FilterStates.R +++ b/R/FilterStates.R @@ -1,16 +1,20 @@ -#' @title `FilterStates` R6 class +# FilterStates ------ + +#' @name FilterStates +#' @docType class +#' @title `FilterStates` `R6` class #' #' @description #' Abstract class that manages adding and removing `FilterState` objects -#' and builds a \emph{subset expression}. +#' and builds a *subset expression*. #' #' A `FilterStates` object tracks all subsetting expressions #' (logical predicates that limit observations) associated with a given dataset -#' and composes them into a single reproducible R expression +#' and composes them into a single reproducible `R` expression #' that will assign a subset of the original data to a new variable. -#' This expression is hereafter referred to as \emph{subset expression}. +#' This expression is hereafter referred to as *subset expression*. #' -#' The \emph{subset expression} is constructed differently for different +#' The *subset expression* is constructed differently for different #' classes of the underlying data object and `FilterStates` sub-classes. #' Currently implemented for `data.frame`, `matrix`, #' `SummarizedExperiment`, and `MultiAssayExperiment`. @@ -23,22 +27,20 @@ FilterStates <- R6::R6Class( # nolint # public members ---- public = list( #' @description - #' Initializes `FilterStates` object. - #' #' Initializes `FilterStates` object by setting #' `dataname`, and `datalabel`. #' - #' @param data (`data.frame`, `MultiAssayExperiment`, `SummarizedExperiment`, `matrix`)\cr - #' the R object which `subset` function is applied on. - #' @param data_reactive (`function(sid)`)\cr + #' @param data (`data.frame` or `MultiAssayExperiment` or `SummarizedExperiment` or `matrix`) + #' the `R` object which `subset` function is applied on. + #' @param data_reactive (`function(sid)`) #' should return an object of the same type as `data` object or `NULL`. #' This object is needed for the `FilterState` counts being updated #' on a change in filters. If function returns `NULL` then filtered counts are not shown. #' Function has to have `sid` argument being a character. - #' @param dataname (`character(1)`)\cr + #' @param dataname (`character(1)`) #' name of the data used in the expression #' specified to the function argument attached to this `FilterStates` - #' @param datalabel (`NULL` or `character(1)`)\cr + #' @param datalabel (`NULL` or `character(1)`) #' text label value #' #' @return @@ -67,8 +69,8 @@ FilterStates <- R6::R6Class( # nolint #' @description #' Returns a formatted string representing this `FilterStates` object. #' - #' @param show_all `logical(1)` passed to `format.teal_slices` - #' @param trim_lines `logical(1)` passed to `format.teal_slices` + #' @param show_all (`logical(1)`) passed to `format.teal_slices` + #' @param trim_lines (`logical(1)`) passed to `format.teal_slices` #' #' @return `character(1)` the formatted string #' @@ -83,7 +85,7 @@ FilterStates <- R6::R6Class( # nolint #' @description #' Filter call #' - #' Builds \emph{subset expression} from condition calls generated by `FilterState`. + #' Builds *subset expression* from condition calls generated by `FilterState`. #' The `lhs` of the expression is a `dataname_prefixed`, where word prefixed refers to #' situation when call is evaluated on elements of the original data, for example `dataname[[x]]`. #' By default `dataname_prefixed = dataname` and it's not alterable through class methods. @@ -115,7 +117,7 @@ FilterStates <- R6::R6Class( # nolint #' #' If no filters are applied, `NULL` is returned to avoid no-op calls such as `dataname <- dataname`. #' - #' @param sid (`character`)\cr + #' @param sid (`character`) #' when specified then method returns code containing filter conditions of #' `FilterState` objects which `"sid"` attribute is different than this `sid` argument. #' @@ -192,7 +194,7 @@ FilterStates <- R6::R6Class( # nolint #' @description #' Remove one or more `FilterState`s from the `state_list` along with their UI elements. #' - #' @param state (`teal_slices`)\cr + #' @param state (`teal_slices`) #' specifying `FilterState` objects to remove; #' `teal_slice`s may contain only `dataname` and `varname`, other elements are ignored #' @@ -242,12 +244,9 @@ FilterStates <- R6::R6Class( # nolint #' @description #' Sets active `FilterState` objects. #' - #' @param data (`data.frame`)\cr + #' @param data (`data.frame`) #' data which are supposed to be filtered - #' @param state (`named list`)\cr - #' should contain values which are initial selection in the `FilterState`. - #' Names of the `list` element should correspond to the name of the - #' column in `data`. + #' @param state (`teal_slices`) #' @return function which throws an error set_filter_state = function(state) { shiny::isolate({ @@ -293,7 +292,7 @@ FilterStates <- R6::R6Class( # nolint #' @description #' Remove all `FilterState` objects from this `FilterStates` object. #' - #' @param force (`logical(1)`)\cr + #' @param force (`logical(1)`) #' include locked filter states #' #' @return `NULL`, invisibly @@ -306,20 +305,18 @@ FilterStates <- R6::R6Class( # nolint # shiny modules ---- #' @description - #' Shiny module UI - #' - #' Shiny UI element that stores `FilterState` UI elements. + #' `shiny` UI definition that stores `FilterState` UI elements. #' Populated with elements created with `renderUI` in the module server. #' - #' @param id (`character(1)`)\cr - #' shiny element (module instance) id + #' @param id (`character(1)`) + #' `shiny` element (module instance) id #' #' @return `shiny.tag` #' ui_active = function(id) { ns <- NS(id) tagList( - teal.slice:::include_css_files(pattern = "filter-panel"), + include_css_files(pattern = "filter-panel"), uiOutput(ns("trigger_visible_state_change"), inline = TRUE), uiOutput( ns("cards"), @@ -330,10 +327,10 @@ FilterStates <- R6::R6Class( # nolint }, #' @description - #' Shiny server module. + #' `shiny` server module. #' - #' @param id (`character(1)`)\cr - #' shiny module instance id + #' @param id (`character(1)`) + #' `shiny` module instance id #' #' @return `moduleServer` function which returns `NULL` #' @@ -399,10 +396,10 @@ FilterStates <- R6::R6Class( # nolint }, #' @description - #' Shiny UI module to add filter variable. + #' `shiny` UI module to add filter variable. #' - #' @param id (`character(1)`)\cr - #' shiny element (module instance) id + #' @param id (`character(1)`) + #' `shiny` element (module instance) id #' #' @return `shiny.tag` #' @@ -422,14 +419,14 @@ FilterStates <- R6::R6Class( # nolint }, #' @description - #' Shiny server module to add filter variable. + #' `shiny` server module to add filter variable. #' #' This module controls available choices to select as a filter variable. #' Once selected, a variable is removed from available choices. #' Removing a filter variable adds it back to available choices. #' - #' @param id (`character(1)`)\cr - #' an ID string that corresponds with the ID used to call the module's UI function. + #' @param id (`character(1)`) + #' an id string that corresponds with the id used to call the module's `ui_add` function. #' #' @return `moduleServer` function which returns `NULL` srv_add = function(id) { @@ -583,7 +580,7 @@ FilterStates <- R6::R6Class( # nolint # @description # Returns a list of `FilterState` objects stored in this `FilterStates`. # - # @param state_id (`character(1)`)\cr + # @param state_id (`character(1)`) # name of element in a filter state (which is a `reactiveVal` containing a list) # # @return `list` of `FilterState` objects @@ -599,12 +596,12 @@ FilterStates <- R6::R6Class( # nolint }, # @description - # Adds a new `FilterState` object to this `FilterStates`.\cr + # Adds a new `FilterState` object to this `FilterStates`. # Raises error if the length of `x` does not match the length of `state_id`. # - # @param x (`FilterState`)\cr + # @param x (`FilterState`) # object to be added to filter state list - # @param state_id (`character(1)`)\cr + # @param state_id (`character(1)`) # name of element in a filter state (which is a `reactiveVal` containing a list) # # @return NULL @@ -625,14 +622,14 @@ FilterStates <- R6::R6Class( # nolint }, # @description - # Removes a single filter state with all associated shiny elements:\cr + # Removes a single filter state with all associated shiny elements: # * specified `FilterState` from `private$state_list` # * UI card created for this filter # * observers tracking the selection and remove button # - # @param state_id (`character`)\cr + # @param state_id (`character`) # names of element in a filter state (which is a `reactiveVal` containing a list) - # @param force (`logical(1)`)\cr + # @param force (`logical(1)`) # include locked filter states # # @return NULL @@ -671,7 +668,7 @@ FilterStates <- R6::R6Class( # nolint # @description # Remove all `FilterState` objects from this `FilterStates` object. - # @param force (`logical(1)`)\cr + # @param force (`logical(1)`) # include locked filter states # @return invisible NULL # diff --git a/R/FilterStatesDF.R b/R/FilterStatesDF.R index 674cbabc3..c99953675 100644 --- a/R/FilterStatesDF.R +++ b/R/FilterStatesDF.R @@ -1,12 +1,18 @@ +# DFFilterStates ------ + +#' @name DFFilterStates +#' @docType class +#' #' @title `FilterStates` subclass for data frames -#' @description Handles filter states in a `data.frame` -#' @keywords internal #' +#' @description Handles filter states in a `data.frame`. #' #' @examples -#' # working filters in an app +#' # use non-exported function from teal.slice +#' include_css_files <- getFromNamespace("include_css_files", "teal.slice") +#' include_js_files <- getFromNamespace("include_js_files", "teal.slice") +#' init_filter_states <- getFromNamespace("init_filter_states", "teal.slice") #' -#' library(shiny) #' library(shinyjs) #' #' # create data frame to filter @@ -28,7 +34,6 @@ #' ) #' data_df <- rbind(data_df, data_na) #' -#' #' # initiate `FilterStates` object #' filter_states_df <- init_filter_states( #' data = data_df, @@ -38,8 +43,8 @@ #' #' ui <- fluidPage( #' useShinyjs(), -#' teal.slice:::include_css_files(pattern = "filter-panel"), -#' teal.slice:::include_js_files(pattern = "count-bar-labels"), +#' include_css_files(pattern = "filter-panel"), +#' include_js_files(pattern = "count-bar-labels"), #' column(4, div( #' h4("Active filters"), #' filter_states_df$ui_active("fsdf") @@ -107,17 +112,43 @@ #' ) #' filter_states_df$set_filter_state(state = filter_state) #' }) -#' observeEvent(input$button7_df, filter_states_df$remove_filter_state(state_id = "NUM1")) -#' observeEvent(input$button8_df, filter_states_df$remove_filter_state(state_id = "NUM2")) -#' observeEvent(input$button9_df, filter_states_df$remove_filter_state(state_id = "CHAR1")) -#' observeEvent(input$button10_df, filter_states_df$remove_filter_state(state_id = "CHAR2")) -#' observeEvent(input$button11_df, filter_states_df$remove_filter_state(state_id = "DATE")) -#' observeEvent(input$button12_df, filter_states_df$remove_filter_state(state_id = "DATETIME")) +#' +#' observeEvent(input$button7_df, { +#' filter_state <- teal_slices(teal_slice("dataset", "NUM1")) +#' filter_states_df$remove_filter_state(filter_state) +#' }) +#' observeEvent(input$button8_df, { +#' filter_state <- teal_slices(teal_slice("dataset", "NUM2")) +#' filter_states_df$remove_filter_state(filter_state) +#' }) +#' observeEvent(input$button9_df, { +#' filter_state <- teal_slices(teal_slice("dataset", "CHAR1")) +#' filter_states_df$remove_filter_state(filter_state) +#' }) +#' observeEvent(input$button10_df, { +#' filter_state <- teal_slices(teal_slice("dataset", "CHAR2")) +#' filter_states_df$remove_filter_state(filter_state) +#' }) +#' observeEvent(input$button11_df, { +#' filter_state <- teal_slices( +#' teal_slice("dataset", "DATE") +#' ) +#' filter_states_df$remove_filter_state(filter_state) +#' }) +#' observeEvent(input$button12_df, { +#' filter_state <- teal_slices( +#' teal_slice("dataset", "DATETIME", selected = as.POSIXct(c("2020-01-01", "2020-02-02"))) +#' ) +#' filter_states_df$remove_filter_state(filter_state) +#' }) +#' #' observeEvent(input$button0_df, filter_states_df$clear_filter_states()) #' } +#' #' if (interactive()) { #' shinyApp(ui, server) #' } +#' @keywords internal #' DFFilterStates <- R6::R6Class( # nolint classname = "DFFilterStates", @@ -125,27 +156,26 @@ DFFilterStates <- R6::R6Class( # nolint # public methods ---- public = list( - #' @description Initializes `DFFilterStates` object. - #' + #' @description #' Initializes `DFFilterStates` object by setting `dataname` #' and initializing `state_list` (`shiny::reactiveVal`). #' This class contains a single `state_list` with no specified name, #' which means that when calling the subset function associated with this class #' (`dplyr::filter`), a list of conditions is passed to unnamed arguments (`...`). #' - #' @param data (`data.frame`)\cr - #' the R object which `dplyr::filter` function is applied on. - #' @param data_reactive (`function(sid)`)\cr + #' @param data (`data.frame`) + #' the `R` object which `dplyr::filter` function is applied on. + #' @param data_reactive (`function(sid)`) #' should return a `data.frame` object or `NULL`. #' This object is needed for the `FilterState` counts being updated #' on a change in filters. If function returns `NULL` then filtered counts are not shown. #' Function has to have `sid` argument being a character. - #' @param dataname (`character`)\cr - #' name of the data used in the \emph{subset expression} + #' @param dataname (`character`) + #' name of the data used in the *subset expression* #' specified to the function argument attached to this `FilterStates` - #' @param datalabel (`NULL` or `character(1)`)\cr + #' @param datalabel (`NULL` or `character(1)`) #' text label value - #' @param keys (`character`)\cr + #' @param keys (`character`) #' key columns names #' initialize = function(data, diff --git a/R/FilterStatesMAE.R b/R/FilterStatesMAE.R index 904058958..f17ba3cf0 100644 --- a/R/FilterStatesMAE.R +++ b/R/FilterStatesMAE.R @@ -1,33 +1,34 @@ +# MAEFilterStates ------ + +#' @name MAEFilterStates +#' @docType class #' @title `FilterStates` subclass for `MultiAssayExperiments` -#' @description Handles filter states in a `MultiAssayExperiment` +#' @description Handles filter states in a `MultiAssayExperiment`. #' @keywords internal #' -#' MAEFilterStates <- R6::R6Class( # nolint classname = "MAEFilterStates", inherit = FilterStates, + # public methods ---- public = list( - # public methods ---- - - #' @description Initializes `MAEFilterStates` object - #' - #' Initialize `MAEFilterStates` object + #' @description + #' Initialize `MAEFilterStates` object. #' - #' @param data (`MultiAssayExperiment`)\cr - #' the R object which `MultiAssayExperiment::subsetByColData` function is applied on. - #' @param data_reactive (`function(sid)`)\cr + #' @param data (`MultiAssayExperiment`) + #' the `R` object which `MultiAssayExperiment::subsetByColData` function is applied on. + #' @param data_reactive (`function(sid)`) #' should return a `MultiAssayExperiment` object or `NULL`. #' This object is needed for the `FilterState` counts being updated #' on a change in filters. If function returns `NULL` then filtered counts are not shown. #' Function has to have `sid` argument being a character. - #' @param dataname (`character(1)`)\cr + #' @param dataname (`character(1)`) #' name of the data used in the expression #' specified to the function argument attached to this `FilterStates`. - #' @param datalabel (`NULL` or `character(1)`)\cr + #' @param datalabel (`NULL` or `character(1)`) #' text label value - #' @param varlabels (`character`)\cr + #' @param varlabels (`character`) #' labels of the variables used in this object - #' @param keys (`character`)\cr + #' @param keys (`character`) #' key columns names #' initialize = function(data, diff --git a/R/FilterStatesMatrix.R b/R/FilterStatesMatrix.R index da9109558..1cd0faf69 100644 --- a/R/FilterStatesMatrix.R +++ b/R/FilterStatesMatrix.R @@ -1,5 +1,9 @@ +# MatrixFilterStates ------ + +#' @name MatrixFilterStates +#' @docType class #' @title `FilterStates` subclass for matrices -#' @description Handles filter states in a `matrix` +#' @description Handles filter states in a `matrix`. #' @keywords internal #' #' @@ -9,21 +13,20 @@ MatrixFilterStates <- R6::R6Class( # nolint # public methods ---- public = list( - #' @description Initialize `MatrixFilterStates` object - #' - #' Initialize `MatrixFilterStates` object + #' @description + #' Initialize `MatrixFilterStates` object. #' - #' @param data (`matrix`)\cr - #' the R object which `subset` function is applied on. - #' @param data_reactive (`function(sid)`)\cr + #' @param data (`matrix`) + #' the `R` object which `subset` function is applied on. + #' @param data_reactive (`function(sid)`) #' should return a `matrix` object or `NULL`. #' This object is needed for the `FilterState` counts being updated #' on a change in filters. If function returns `NULL` then filtered counts are not shown. #' Function has to have `sid` argument being a character. - #' @param dataname (`character(1)`)\cr + #' @param dataname (`character(1)`) #' name of the data used in the expression #' specified to the function argument attached to this `FilterStates`. - #' @param datalabel (`NULL` or `character(1)`)\cr + #' @param datalabel (`NULL` or `character(1)`) #' text label value. Should be a name of experiment. #' initialize = function(data, diff --git a/R/FilterStatesSE.R b/R/FilterStatesSE.R index cef85be6f..56ee031ab 100644 --- a/R/FilterStatesSE.R +++ b/R/FilterStatesSE.R @@ -1,29 +1,31 @@ +# SEFilterStates ------ + +#' @name SEFilterStates +#' @docType class #' @title `FilterStates` subclass for `SummarizedExperiments` -#' @description Handles filter states in a `SummaryExperiment` +#' @description Handles filter states in a `SummaryExperiment`. #' @keywords internal #' -#' SEFilterStates <- R6::R6Class( # nolint classname = "SEFilterStates", inherit = FilterStates, # public methods ---- public = list( - #' @description Initialize `SEFilterStates` object - #' - #' Initialize `SEFilterStates` object + #' @description + #' Initialize `SEFilterStates` object. #' - #' @param data (`SummarizedExperiment`)\cr - #' the R object which `subset` function is applied on. - #' @param data_reactive (`function(sid)`)\cr + #' @param data (`SummarizedExperiment`) + #' the `R` object which `subset` function is applied on. + #' @param data_reactive (`function(sid)`) #' should return a `SummarizedExperiment` object or `NULL`. #' This object is needed for the `FilterState` counts being updated #' on a change in filters. If function returns `NULL` then filtered counts are not shown. #' Function has to have `sid` argument being a character. - #' @param dataname (`character(1)`)\cr + #' @param dataname (`character(1)`) #' name of the data used in the expression #' specified to the function argument attached to this `FilterStates`. - #' @param datalabel (`character(0)` or `character(1)`)\cr + #' @param datalabel (`NULL` or `character(1)`) #' text label value. Should be a name of experiment #' initialize = function(data, @@ -42,9 +44,9 @@ SEFilterStates <- R6::R6Class( # nolint }, #' @description - #' Set filter state + #' Set filter state. #' - #' @param state (`teal_slices`)\cr + #' @param state (`teal_slices`) #' `teal_slice` objects should contain the field `arg %in% c("subset", "select")` #' #' @return `NULL` invisibly @@ -91,10 +93,10 @@ SEFilterStates <- R6::R6Class( # nolint }, #' @description - #' Shiny UI module to add filter variable - #' @param id (`character(1)`)\cr - #' id of shiny module - #' @return shiny.tag + #' `shiny` UI module to add filter variable. + #' @param id (`character(1)`) + #' id of `shiny` module + #' @return `shiny.tag` ui_add = function(id) { data <- private$data checkmate::assert_string(id) @@ -136,7 +138,7 @@ SEFilterStates <- R6::R6Class( # nolint }, #' @description - #' Shiny server module to add filter variable + #' `shiny` server module to add filter variable. #' #' Module controls available choices to select as a filter variable. #' Selected filter variable is being removed from available choices. @@ -145,8 +147,8 @@ SEFilterStates <- R6::R6Class( # nolint #' sets of filter variables - one for `colData` and another for #' `rowData`. #' - #' @param id (`character(1)`)\cr - #' an ID string that corresponds with the ID used to call the module's UI function. + #' @param id (`character(1)`) + #' an id string that corresponds with the id used to call the module's `ui_add` function. #' @return `moduleServer` function which returns `NULL` srv_add = function(id) { data <- private$data diff --git a/R/FilteredData-utils.R b/R/FilteredData-utils.R index 8f084e93d..d0d819b53 100644 --- a/R/FilteredData-utils.R +++ b/R/FilteredData-utils.R @@ -1,13 +1,14 @@ #' Initialize `FilteredData` #' -#' Initialize `FilteredData` +#' Function creates a `FilteredData` object. +#' #' @param x (named `list`) of datasets. -#' @param join_keys (`join_keys`) see [teal.data::join_keys()]. -#' @param code (deprecated) -#' @param check (deprecated) +#' @param join_keys (`join_keys`) see [`teal.data::join_keys()`]. +#' @param code `r lifecycle::badge("deprecated")` +#' @param check `r lifecycle::badge("deprecated")` #' @examples -#' library(shiny) -#' datasets <- teal.slice::init_filtered_data(list(iris = iris, mtcars = mtcars)) +#' datasets <- init_filtered_data(list(iris = iris, mtcars = mtcars)) +#' datasets #' @export init_filtered_data <- function(x, join_keys = teal.data::join_keys(), code, check) { # nolint checkmate::assert_list(x, any.missing = FALSE, names = "unique") @@ -29,12 +30,13 @@ init_filtered_data <- function(x, join_keys = teal.data::join_keys(), code, chec #' Evaluate expression with meaningful message #' -#' Method created for the `FilteredData` to execute filter call with +#' Method created for the `FilteredData` object to execute filter call with #' meaningful message. After evaluation used environment should contain #' all necessary bindings. +#' #' @param expr (`language`) #' @param env (`environment`) where expression is evaluated. -#' @return invisible `NULL`. +#' @return `NULL` invisibly. #' @keywords internal eval_expr_with_msg <- function(expr, env) { lapply( @@ -69,21 +71,21 @@ eval_expr_with_msg <- function(expr, env) { #' `removeClass` and `addClass` methods (when `one_way = TRUE`) to change icons. #' `toggle_title` calls the `attr` method to modify the `Title` attribute of the button. #' -#' @param input_id `character(1)` (name-spaced) id of the button -#' @param icons,titles `character(2)` vector specifying values between which to toggle -#' @param one_way `logical(1)` flag specifying whether to keep toggling; +#' @param input_id (`character(1)`) (name-spaced) id of the button +#' @param icons,titles (`character(2)`) vector specifying values between which to toggle +#' @param one_way (`logical(1)`) flag specifying whether to keep toggling; #' if TRUE, the target will be changed #' from the first element of `icons`/`titles` to the second #' -#' @return Invisible NULL. -#' -#' @name toggle_button -#' +#' @return `NULL` invisibly #' @examples -#' library(shiny) +#' # use non-exported function from teal.slice +#' toggle_icon <- getFromNamespace("toggle_icon", "teal.slice") +#' +#' library(shinyjs) #' #' ui <- fluidPage( -#' shinyjs::useShinyjs(), +#' useShinyjs(), #' actionButton("hide_content", label = "hide", icon = icon("xmark")), #' actionButton("show_content", label = "show", icon = icon("check")), #' actionButton("toggle_content", label = "toggle", icon = icon("angle-down")), @@ -97,7 +99,7 @@ eval_expr_with_msg <- function(expr, env) { #' server <- function(input, output, session) { #' observeEvent(input$hide_content, #' { -#' shinyjs::hide("content") +#' hide("content") #' toggle_icon("toggle_content", c("fa-angle-down", "fa-angle-right"), one_way = TRUE) #' }, #' ignoreInit = TRUE @@ -105,7 +107,7 @@ eval_expr_with_msg <- function(expr, env) { #' #' observeEvent(input$show_content, #' { -#' shinyjs::show("content") +#' show("content") #' toggle_icon("toggle_content", c("fa-angle-right", "fa-angle-down"), one_way = TRUE) #' }, #' ignoreInit = TRUE @@ -113,7 +115,7 @@ eval_expr_with_msg <- function(expr, env) { #' #' observeEvent(input$toggle_content, #' { -#' shinyjs::toggle("content") +#' toggle("content") #' toggle_icon("toggle_content", c("fa-angle-right", "fa-angle-down")) #' }, #' ignoreInit = TRUE @@ -126,7 +128,7 @@ eval_expr_with_msg <- function(expr, env) { #' if (interactive()) { #' shinyApp(ui, server) #' } -#' +#' @name toggle_button #' @rdname toggle_button #' @keywords internal toggle_icon <- function(input_id, icons, one_way = FALSE) { @@ -180,63 +182,15 @@ toggle_title <- function(input_id, titles, one_way = FALSE) { invisible(NULL) } -#' Topological graph sort -#' -#' Graph is a list which for each node contains a vector of child nodes -#' in the returned list, parents appear before their children. -#' -#' Implementation of `Kahn` algorithm with a modification to maintain the order of input elements. +#' @inherit teal.data::topological_sort description details params title +#' @examples +#' # use non-exported function from teal.slice +#' topological_sort <- getFromNamespace("topological_sort", "teal.slice") #' -#' @param graph (named `list`) list with node vector elements +#' topological_sort(list(A = c(), B = c("A"), C = c("B"), D = c("A"))) +#' topological_sort(list(D = c("A"), A = c(), B = c("A"), C = c("B"))) +#' topological_sort(list(D = c("A"), B = c("A"), C = c("B"), A = c())) #' @keywords internal -#' -#' @examples -#' teal.slice:::topological_sort(list(A = c(), B = c("A"), C = c("B"), D = c("A"))) -#' teal.slice:::topological_sort(list(D = c("A"), A = c(), B = c("A"), C = c("B"))) -#' teal.slice:::topological_sort(list(D = c("A"), B = c("A"), C = c("B"), A = c())) topological_sort <- function(graph) { - # compute in-degrees - in_degrees <- list() - for (node in names(graph)) { - in_degrees[[node]] <- 0 - for (to_edge in graph[[node]]) { - in_degrees[[to_edge]] <- 0 - } - } - - for (node in graph) { - for (to_edge in node) { - in_degrees[[to_edge]] <- in_degrees[[to_edge]] + 1 - } - } - - # sort - visited <- 0 - sorted <- list() - zero_in <- list() - for (node in names(in_degrees)) { - if (in_degrees[[node]] == 0) zero_in <- append(zero_in, node) - } - zero_in <- rev(zero_in) - - while (length(zero_in) != 0) { - visited <- visited + 1 - sorted <- c(zero_in[[1]], sorted) - for (edge_to in graph[[zero_in[[1]]]]) { - in_degrees[[edge_to]] <- in_degrees[[edge_to]] - 1 - if (in_degrees[[edge_to]] == 0) { - zero_in <- append(zero_in, edge_to, 1) - } - } - zero_in[[1]] <- NULL - } - - if (visited != length(in_degrees)) { - stop( - "Graph is not a directed acyclic graph. Cycles involving nodes: ", - paste0(setdiff(names(in_degrees), sorted), collapse = " ") - ) - } else { - return(sorted) - } + utils::getFromNamespace("topological_sort", ns = "teal.data")(graph) } diff --git a/R/FilteredData.R b/R/FilteredData.R index b8063cc09..4ab600d01 100644 --- a/R/FilteredData.R +++ b/R/FilteredData.R @@ -1,8 +1,13 @@ +# FilteredData ------ + #' @name FilteredData #' @docType class #' #' @title Class to encapsulate filtered datasets #' +#' @description +#' Class is designed to manage and encapsulate filtered datasets. +#' #' @details #' The main purpose of this class is to provide a collection of reactive datasets, #' each dataset having a filter state that determines how it is filtered. @@ -10,7 +15,7 @@ #' For each dataset, `get_filter_expr` returns the call to filter the dataset according #' to the filter state. The data itself can be obtained through `get_data`. #' -#' The datasets are filtered lazily, i.e. only when requested / needed in a Shiny app. +#' The datasets are filtered lazily, i.e. only when requested / needed in a `shiny` app. #' #' By design, any `dataname` set through `set_dataset` cannot be removed because #' other code may already depend on it. As a workaround, the underlying @@ -28,14 +33,14 @@ #' #' Common arguments are: #' 1. `filtered`: whether to return a filtered result or not -#' 2. `dataname`: the name of one of the datasets in this `FilteredData` +#' 2. `dataname`: the name of one of the datasets in this `FilteredData` object #' 3. `varname`: one of the columns in a dataset #' -#' @keywords internal -#' #' @examples -#' library(shiny) -#' datasets <- teal.slice:::FilteredData$new(list(iris = iris, mtcars = mtcars)) +#' # use non-exported function from teal.slice +#' FilteredData <- getFromNamespace("FilteredData", "teal.slice") +#' +#' datasets <- FilteredData$new(list(iris = iris, mtcars = mtcars)) #' #' # get datanames #' datasets$datanames() @@ -52,13 +57,42 @@ #' isolate(datasets$get_filter_state()) #' isolate(datasets$get_call("iris")) #' isolate(datasets$get_call("mtcars")) +#' @examplesIf requireNamespace("MultiAssayExperiment") +#' ### set_filter_state +#' +#' utils::data(miniACC, package = "MultiAssayExperiment") +#' datasets <- FilteredData$new(list(iris = iris, mae = miniACC)) +#' fs <- teal_slices( +#' teal_slice( +#' dataname = "iris", varname = "Sepal.Length", selected = c(5.1, 6.4), +#' keep_na = TRUE, keep_inf = FALSE +#' ), +#' teal_slice( +#' dataname = "iris", varname = "Species", selected = c("setosa", "versicolor"), +#' keep_na = FALSE +#' ), +#' teal_slice( +#' dataname = "mae", varname = "years_to_birth", selected = c(30, 50), +#' keep_na = TRUE, keep_inf = FALSE +#' ), +#' teal_slice(dataname = "mae", varname = "vital_status", selected = "1", keep_na = FALSE), +#' teal_slice(dataname = "mae", varname = "gender", selected = "female", keep_na = TRUE), +#' teal_slice( +#' dataname = "mae", varname = "ARRAY_TYPE", +#' selected = "", keep_na = TRUE, experiment = "RPPAArray", arg = "subset" +#' ) +#' ) +#' datasets$set_filter_state(state = fs) +#' isolate(datasets$get_filter_state()) +#' +#' @keywords internal #' FilteredData <- R6::R6Class( # nolint "FilteredData", - ## __Public Methods ==== + # public methods ---- public = list( #' @description - #' Initialize a `FilteredData` object + #' Initialize a `FilteredData` object. #' @param data_objects (`list`) #' Named list of data objects. #' Names of the list will serve as `dataname`. @@ -98,35 +132,36 @@ FilteredData <- R6::R6Class( # nolint }, #' @description - #' Gets `datanames` - #' + #' Gets `datanames`. + #' @details #' The `datanames` are returned in the order in which they must be #' evaluated (in case of dependencies). - #' @return (`character` vector) of `datanames` + #' @return `character` vector of `datanames` datanames = function() { names(private$filtered_datasets) }, - #' Gets data label for the dataset - #' + #' @description + #' Gets data label for the dataset. #' Useful to display in `Show R Code`. #' #' @param dataname (`character(1)`) name of the dataset - #' @return (`character`) keys of dataset + #' @return `character` keys of dataset get_datalabel = function(dataname) { private$get_filtered_dataset(dataname)$get_dataset_label() }, + #' @description #' Set list of external filter states available for activation. - #' + #' @details #' Unlike adding new filter from the column, these filters can come with some prespecified settings. #' `teal_slices` are wrapped in a `reactive` so they can be updated from elsewhere in the app. - #' Filters passed in `x` are limited to those that can be set for this `FilteredData`, + #' Filters passed in `x` are limited to those that can be set for this `FilteredData` object, #' i.e. they have the correct `dataname` and `varname` (waived `teal_slice_fixed` as they do not have `varname`). #' List is accessible in `ui/srv_active` through `ui/srv_available_filters`. - #' @param x (`reactive`)\cr + #' @param x (`reactive`) #' should return `teal_slices` - #' @return invisible `NULL` + #' @return `NULL` invisibly set_available_teal_slices = function(x) { checkmate::assert_class(x, "reactive") private$available_teal_slices <- reactive({ @@ -151,8 +186,9 @@ FilteredData <- R6::R6Class( # nolint invisible(NULL) }, + #' @description #' Get list of filter states available for this object. - #' + #' @details #' All `teal_slice` objects that have been created since the beginning of the app session #' are stored in one `teal_slices` object. This returns a subset of that `teal_slices`, #' describing filter states that can be set for this object. @@ -165,7 +201,7 @@ FilteredData <- R6::R6Class( # nolint #' @description #' Gets a `call` to filter the dataset according to the filter state. - #' + #' @details #' It returns a `call` to filter the dataset only, assuming the #' other (filtered) datasets it depends on are available. #' @@ -174,7 +210,7 @@ FilteredData <- R6::R6Class( # nolint #' `FilteredData$get_filter_code`. #' #' For the return type, note that `rlang::is_expression` returns `TRUE` on the - #' return type, both for base R expressions and calls (single expression, + #' return type, both for base `R` expressions and calls (single expression, #' capturing a function call). #' #' The filtered dataset has the name given by `self$filtered_dataname(dataname)` @@ -209,21 +245,21 @@ FilteredData <- R6::R6Class( # nolint #' @description #' Get join keys between two datasets. #' - #' @return (`join_keys`) + #' @return `join_keys` #' get_join_keys = function() { return(private$join_keys) }, #' @description - #' Get filter overview table in form of X (filtered) / Y (non-filtered). + #' Get filter overview table in form of `X (filtered) / Y (non-filtered)`. #' #' This is intended to be presented in the application. #' The content for each of the data names is defined in `get_filter_overview_info` method. #' - #' @param datanames (`character` vector) names of the dataset + #' @param datanames (`character`) vector of dataset names #' - #' @return (`matrix`) matrix of observations and subjects of all datasets + #' @return `matrix` of observations and subjects of all datasets #' get_filter_overview = function(datanames) { rows <- lapply( @@ -241,7 +277,7 @@ FilteredData <- R6::R6Class( # nolint #' #' @param dataname (`character(1)`) name of the dataset #' - #' @return (`character`) keys of dataset + #' @return `character` keys of dataset #' get_keys = function(dataname) { private$get_filtered_dataset(dataname)$get_keys() @@ -257,13 +293,13 @@ FilteredData <- R6::R6Class( # nolint #' "Child" dataset return filtered data then dependent on the reactive filtered data of the #' "parent". See more in documentation of `parent` argument in `DataframeFilteredDataset` constructor. #' - #' @param data (`data.frame`, `MultiAssayExperiment`)\cr + #' @param data (`data.frame` or `MultiAssayExperiment`) #' data to be filtered. #' - #' @param dataname (`string`)\cr + #' @param dataname (`string`) #' the name of the `dataset` to be added to this object #' - #' @return (`self`) invisibly this `FilteredData` + #' @return `self` invisibly #' set_dataset = function(data, dataname) { checkmate::assert_string(dataname) @@ -303,7 +339,7 @@ FilteredData <- R6::R6Class( # nolint #' #' @param join_keys (`join_keys`) join_key (converted to a nested list) #' - #' @return (`self`) invisibly this `FilteredData` + #' @return `self` invisibly #' set_join_keys = function(join_keys) { checkmate::assert_class(join_keys, "join_keys") @@ -331,8 +367,8 @@ FilteredData <- R6::R6Class( # nolint #' @description #' Returns a formatted string representing this `FilteredData` object. #' - #' @param show_all `logical(1)` passed to `format.teal_slice` - #' @param trim_lines `logical(1)` passed to `format.teal_slice` + #' @param show_all (`logical(1)`) passed to `format.teal_slice` + #' @param trim_lines (`logical(1)`) passed to `format.teal_slice` #' #' @return `character(1)` the formatted string #' @@ -362,31 +398,9 @@ FilteredData <- R6::R6Class( # nolint #' @description #' Sets active filter states. #' - #' @param state (`teal_slices`) object + #' @param state (`teal_slices`) #' #' @return `NULL` invisibly - #' - #' @examples - #' utils::data(miniACC, package = "MultiAssayExperiment") - #' - #' datasets <- teal.slice:::FilteredData$new(list(iris = iris, mae = miniACC)) - #' - #' fs <- - #' teal_slices( - #' teal_slice(dataname = "iris", varname = "Sepal.Length", selected = c(5.1, 6.4), - #' keep_na = TRUE, keep_inf = FALSE), - #' teal_slice(dataname = "iris", varname = "Species", selected = c("setosa", "versicolor"), - #' keep_na = FALSE), - #' teal_slice(dataname = "mae", varname = "years_to_birth", selected = c(30, 50), - #' keep_na = TRUE, keep_inf = FALSE), - #' teal_slice(dataname = "mae", varname = "vital_status", selected = "1", keep_na = FALSE), - #' teal_slice(dataname = "mae", varname = "gender", selected = "female", keep_na = TRUE), - #' teal_slice(dataname = "mae", varname = "ARRAY_TYPE", - #' selected = "", keep_na = TRUE, datalabel = "RPPAArray", arg = "subset") - #' ) - #' datasets$set_filter_state(state = fs) - #' shiny::isolate(datasets$get_filter_state()) - #' set_filter_state = function(state) { shiny::isolate({ logger::log_trace("{ class(self)[1] }$set_filter_state initializing") @@ -410,7 +424,7 @@ FilteredData <- R6::R6Class( # nolint #' @description #' Removes one or more `FilterState` from a `FilteredData` object. #' - #' @param state (`teal_slices`)\cr + #' @param state (`teal_slices`) #' specifying `FilterState` objects to remove; #' `teal_slice`s may contain only `dataname` and `varname`, other elements are ignored #' @@ -443,10 +457,10 @@ FilteredData <- R6::R6Class( # nolint #' Remove all `FilterStates` of a `FilteredDataset` or all `FilterStates` #' of a `FilteredData` object. #' - #' @param datanames (`character`)\cr + #' @param datanames (`character`) #' `datanames` to remove their `FilterStates` or empty which removes #' all `FilterStates` in the `FilteredData` object - #' @param force (`logical(1)`)\cr + #' @param force (`logical(1)`) #' include locked filter states #' #' @return `NULL` invisibly @@ -474,13 +488,14 @@ FilteredData <- R6::R6Class( # nolint # shiny modules ----- + #' @description #' Module for the right filter panel in the teal app #' with a filter overview panel and a filter variable panel. #' #' This panel contains info about the number of observations left in #' the (active) datasets and allows to filter the datasets. #' - #' @param id (`character(1)`)\cr + #' @param id (`character(1)`) #' module id #' @return `shiny.tag` ui_filter_panel = function(id) { @@ -496,11 +511,13 @@ FilteredData <- R6::R6Class( # nolint ) }, - #' Server function for filter panel + #' @description + #' Server function for filter panel. #' - #' @param id (`character(1)`)\cr - #' an ID string that corresponds with the ID used to call the module's UI function. - #' @param active_datanames `function / reactive` returning `datanames` that + #' @param id (`character(1)`) + #' an id string that corresponds with the id used to call the module's + #' `ui_filter_panel` function. + #' @param active_datanames (`function` or `reactive`) returning `datanames` that #' should be shown on the filter panel, #' must be a subset of the `datanames` argument provided to `ui_filter_panel`; #' if the function returns `NULL` (as opposed to `character(0)`), the filter @@ -532,8 +549,8 @@ FilteredData <- R6::R6Class( # nolint #' @description #' Server module responsible for displaying active filters. - #' @param id (`character(1)`)\cr - #' an ID string that corresponds with the ID used to call the module's UI function. + #' @param id (`character(1)`) + #' an id string that corresponds with the id used to call the module's `srv_active` function. #' @return `shiny.tag` ui_active = function(id) { ns <- NS(id) @@ -582,9 +599,9 @@ FilteredData <- R6::R6Class( # nolint #' @description #' Server module responsible for displaying active filters. - #' @param id (`character(1)`)\cr - #' an ID string that corresponds with the ID used to call the module's UI function. - #' @param active_datanames (`reactive`)\cr + #' @param id (`character(1)`) + #' an id string that corresponds with the id used to call the module's `ui_active` function. + #' @param active_datanames (`reactive`) #' defining subset of `self$datanames()` to be displayed. #' @return `moduleServer` returning `NULL` srv_active = function(id, active_datanames = self$datanames) { @@ -651,8 +668,9 @@ FilteredData <- R6::R6Class( # nolint #' @description #' Server module responsible for displaying drop-downs with variables to add a filter. - #' @param id (`character(1)`)\cr - #' an ID string that corresponds with the ID used to call the module's UI function. + #' @param id (`character(1)`) + #' an id string that corresponds with the id used to call the module's + #' `ui_add` function. #' @return `shiny.tag` ui_add = function(id) { ns <- NS(id) @@ -693,9 +711,10 @@ FilteredData <- R6::R6Class( # nolint #' @description #' Server module responsible for displaying drop-downs with variables to add a filter. - #' @param id (`character(1)`)\cr - #' an ID string that corresponds with the ID used to call the module's UI function. - #' @param active_datanames (`reactive`)\cr + #' @param id (`character(1)`) + #' an id string that corresponds with the id used to call the module's + #' `ui_add` function. + #' @param active_datanames (`reactive`) #' defining subset of `self$datanames()` to be displayed. #' @return `moduleServer` returning `NULL` srv_add = function(id, active_datanames = reactive(self$datanames())) { @@ -732,8 +751,9 @@ FilteredData <- R6::R6Class( # nolint }) }, - #' Creates the UI for the module showing counts for each dataset - #' contrasting the filtered to the full unfiltered dataset + #' @description + #' Creates the UI definition for the module showing counts for each dataset + #' contrasting the filtered to the full unfiltered dataset. #' #' Per dataset, it displays #' the number of rows/observations in each dataset, @@ -772,12 +792,14 @@ FilteredData <- R6::R6Class( # nolint ) }, + #' @description #' Server function to display the number of records in the filtered and unfiltered - #' data + #' data. #' - #' @param id (`character(1)`)\cr - #' an ID string that corresponds with the ID used to call the module's UI function. - #' @param active_datanames (`reactive`)\cr + #' @param id (`character(1)`) + #' an id string that corresponds with the id used to call the module's + #' `ui_overview` function. + #' @param active_datanames (`reactive`) #' returning `datanames` that should be shown on the filter panel, #' must be a subset of the `datanames` argument provided to `ui_filter_panel`; #' if the function returns `NULL` (as opposed to `character(0)`), the filter @@ -875,7 +897,7 @@ FilteredData <- R6::R6Class( # nolint } ), - ## __Private Members ==== + # private members ---- private = list( # selectively hide / show to only show `active_datanames` out of all datanames @@ -900,7 +922,7 @@ FilteredData <- R6::R6Class( # nolint # Gets `FilteredDataset` object which contains all information # pertaining to the specified dataset. # - # @param dataname (`character(1)`)\cr + # @param dataname (`character(1)`) # name of the dataset # # @return `FilteredDataset` object or list of `FilteredDataset`s diff --git a/R/FilteredDataset-utils.R b/R/FilteredDataset-utils.R index 62a13b07e..7d1ed4edf 100644 --- a/R/FilteredDataset-utils.R +++ b/R/FilteredDataset-utils.R @@ -1,75 +1,74 @@ #' Initializes `FilteredDataset` #' -#' @keywords internal +#' @param dataset (`data.frame` or `MultiAssayExperiment`) +#' @param dataname (`character`) +#' A given name for the dataset it may not contain spaces +#' @param keys optional, (`character`) +#' Vector with primary keys +#' @param parent_name (`character(1)`) +#' Name of the parent dataset +#' @param parent (`reactive`) +#' object returned by this reactive is a filtered `data.frame` from other `FilteredDataset` +#' named `parent_name`. Consequence of passing `parent` is a `reactive` link which causes +#' causing re-filtering of this `dataset` based on the changes in `parent`. +#' @param join_keys (`character`) +#' Name of the columns in this dataset to join with `parent` +#' dataset. If the column names are different if both datasets +#' then the names of the vector define the `parent` columns. +#' @param label (`character`) +#' Label to describe the dataset #' @examples #' # DataframeFilteredDataset example -#' iris_fd <- teal.slice:::init_filtered_dataset(iris, dataname = "iris") -#' app <- shinyApp( -#' ui = fluidPage( -#' iris_fd$ui_add(id = "add"), -#' iris_fd$ui_active("dataset"), -#' verbatimTextOutput("call") -#' ), -#' server = function(input, output, session) { -#' iris_fd$srv_add(id = "add") -#' iris_fd$srv_active(id = "dataset") -#' -#' output$call <- renderText({ -#' paste( -#' vapply(iris_fd$get_call(), deparse1, character(1), collapse = "\n"), -#' collapse = "\n" -#' ) -#' }) -#' } +#' iris_fd <- init_filtered_dataset(iris, dataname = "iris") +#' ui <- fluidPage( +#' iris_fd$ui_add(id = "add"), +#' iris_fd$ui_active("dataset"), +#' verbatimTextOutput("call") #' ) +#' server <- function(input, output, session) { +#' iris_fd$srv_add(id = "add") +#' iris_fd$srv_active(id = "dataset") +#' +#' output$call <- renderText({ +#' paste( +#' vapply(iris_fd$get_call(), deparse1, character(1), collapse = "\n"), +#' collapse = "\n" +#' ) +#' }) +#' } #' if (interactive()) { -#' shinyApp(app$ui, app$server) +#' shinyApp(ui, server) #' } #' +#' @examplesIf requireNamespace("MultiAssayExperiment") #' # MAEFilteredDataset example #' library(MultiAssayExperiment) -#' data(miniACC) -#' MAE_fd <- teal.slice:::init_filtered_dataset(miniACC, "MAE") -#' app <- shinyApp( -#' ui = fluidPage( -#' MAE_fd$ui_add(id = "add"), -#' MAE_fd$ui_active("dataset"), -#' verbatimTextOutput("call") -#' ), -#' server = function(input, output, session) { -#' MAE_fd$srv_add(id = "add") -#' MAE_fd$srv_active(id = "dataset") -#' output$call <- renderText({ -#' paste( -#' vapply(MAE_fd$get_call(), deparse1, character(1), collapse = "\n"), -#' collapse = "\n" -#' ) -#' }) -#' } +#' data(miniACC, package = "MultiAssayExperiment") +#' MAE_fd <- init_filtered_dataset(miniACC, "MAE") +#' ui <- fluidPage( +#' MAE_fd$ui_add(id = "add"), +#' MAE_fd$ui_active("dataset"), +#' verbatimTextOutput("call") #' ) +#' server <- function(input, output, session) { +#' MAE_fd$srv_add(id = "add") +#' MAE_fd$srv_active(id = "dataset") +#' output$call <- renderText({ +#' paste( +#' vapply(MAE_fd$get_call(), deparse1, character(1), collapse = "\n"), +#' collapse = "\n" +#' ) +#' }) +#' } #' if (interactive()) { -#' shinyApp(app$ui, app$server) +#' shinyApp(ui, server) #' } -#' @param dataset (`data.frame` or `MultiAssayExperiment`)\cr -#' @param dataname (`character`)\cr -#' A given name for the dataset it may not contain spaces -#' @param keys optional, (`character`)\cr -#' Vector with primary keys -#' @param parent_name (`character(1)`)\cr -#' Name of the parent dataset -#' @param parent (`reactive`)\cr -#' object returned by this reactive is a filtered `data.frame` from other `FilteredDataset` -#' named `parent_name`. Consequence of passing `parent` is a `reactive` link which causes -#' causing re-filtering of this `dataset` based on the changes in `parent`. -#' @param join_keys (`character`)\cr -#' Name of the columns in this dataset to join with `parent` -#' dataset. If the column names are different if both datasets -#' then the names of the vector define the `parent` columns. -#' @param label (`character`)\cr -#' Label to describe the dataset +#' +#' @keywords internal #' @export #' @note Although this function is exported for use in other packages, it may be changed or removed in a future release #' at which point any code which relies on this exported function will need to be changed. +#' init_filtered_dataset <- function(dataset, # nolint dataname, keys = character(0), diff --git a/R/FilteredDataset.R b/R/FilteredDataset.R index 73d96be1f..1f421fc2e 100644 --- a/R/FilteredDataset.R +++ b/R/FilteredDataset.R @@ -1,5 +1,9 @@ # FilteredDataset abstract -------- -#' @title `FilterStates` R6 class + +#' @name FilteredDataset +#' @docType class +#' +#' @title `FilteredDataset` `R6` class #' @description #' `FilteredDataset` is a class which renders/controls `FilterStates`(s) #' Each `FilteredDataset` contains `filter_states` field - a `list` which contains one @@ -9,18 +13,18 @@ #' @keywords internal FilteredDataset <- R6::R6Class( # nolint "FilteredDataset", - ## __Public Methods ==== + # public methods ---- public = list( #' @description - #' Initializes this `FilteredDataset` object + #' Initializes this `FilteredDataset` object. #' - #' @param dataset (`data.frame` or `MultiAssayExperiment`)\cr + #' @param dataset (`data.frame` or `MultiAssayExperiment`) #' single dataset for which filters are rendered - #' @param dataname (`character(1)`)\cr + #' @param dataname (`character(1)`) #' A given name for the dataset it may not contain spaces - #' @param keys optional, (`character`)\cr + #' @param keys optional, (`character`) #' Vector with primary keys - #' @param label (`character(1)`)\cr + #' @param label (`character(1)`) #' Label to describe the dataset. initialize = function(dataset, dataname, keys = character(0), label = attr(dataset, "label")) { logger::log_trace("Instantiating { class(self)[1] }, dataname: { dataname }") @@ -59,8 +63,8 @@ FilteredDataset <- R6::R6Class( # nolint #' @description #' Returns a formatted string representing this `FilteredDataset` object. #' - #' @param show_all `logical(1)` passed to `format.teal_slice` - #' @param trim_lines `logical(1)` passed to `format.teal_slice` + #' @param show_all (`logical(1)`) passed to `format.teal_slice` + #' @param trim_lines (`logical(1)`) passed to `format.teal_slice` #' #' @return `character(1)` the formatted string #' @@ -82,8 +86,9 @@ FilteredDataset <- R6::R6Class( # nolint }, #' @description - #' Removes all active filter items applied to this dataset - #' @param force (`logical(1)`)\cr + #' Removes all active filter items applied to this dataset. + #' + #' @param force (`logical(1)`) #' include locked filter states #' #' @return NULL @@ -101,13 +106,13 @@ FilteredDataset <- R6::R6Class( # nolint # getters ---- #' @description - #' Gets a filter expression + #' Gets a filter expression. #' #' This functions returns filter calls equivalent to selected items #' within each of `filter_states`. Configuration of the calls is constant and #' depends on `filter_states` type and order which are set during initialization. #' - #' @param sid (`character`)\cr + #' @param sid (`character`) #' when specified then method returns code containing filter conditions of #' `FilterState` objects which `"sid"` attribute is different than this `sid` argument. #' @@ -124,7 +129,7 @@ FilteredDataset <- R6::R6Class( # nolint }, #' @description - #' Gets states of all active `FilterState` objects + #' Gets states of all active `FilterState` objects. #' #' @return A `teal_slices` object. #' @@ -134,9 +139,9 @@ FilteredDataset <- R6::R6Class( # nolint }, #' @description - #' Set filter state + #' Set filter state. #' - #' @param state (`teal_slice`) object + #' @param state (`teal_slices`) #' #' @return `NULL` invisibly #' @@ -152,7 +157,7 @@ FilteredDataset <- R6::R6Class( # nolint }, #' @description - #' Gets the name of the dataset + #' Gets the name of the dataset. #' #' @return `character(1)` as a name of this dataset get_dataname = function() { @@ -160,8 +165,9 @@ FilteredDataset <- R6::R6Class( # nolint }, #' @description - #' Gets the dataset object in this `FilteredDataset` - #' @param filtered (`logical(1)`)\cr + #' Gets the dataset object in this `FilteredDataset`. + #' + #' @param filtered (`logical(1)`) #' #' @return `data.frame` or `MultiAssayExperiment`, either raw #' or as a reactive with current filters applied @@ -175,9 +181,9 @@ FilteredDataset <- R6::R6Class( # nolint }, #' @description - #' Get filter overview rows of a dataset + #' Get filter overview rows of a dataset. #' The output shows the comparison between `filtered_dataset` - #' function parameter and the dataset inside self + #' function parameter and the dataset inside self. #' @param filtered_dataset comparison object, of the same class #' as `self$get_dataset()`, if `NULL` then `self$get_dataset()` #' is used. @@ -193,29 +199,29 @@ FilteredDataset <- R6::R6Class( # nolint }, #' @description - #' Gets the keys for the dataset of this `FilteredDataset` - #' @return (`character`) the keys of dataset + #' Gets the keys for the dataset of this `FilteredDataset`. + #' @return `character` the keys of dataset get_keys = function() { private$keys }, #' @description - #' Gets the dataset label - #' @return (`character`) the dataset label + #' Gets the dataset label. + #' @return `character` the dataset label get_dataset_label = function() { private$label }, # modules ------ #' @description - #' UI module for dataset active filters - #' + #' UI module for dataset active filters. + #' @details #' UI module containing dataset active filters along with #' title and remove button. - #' @param id (`character(1)`)\cr + #' @param id (`character(1)`) #' identifier of the element - preferably containing dataset name #' - #' @return function - shiny UI module + #' @return function - `shiny` UI module ui_active = function(id) { dataname <- self$get_dataname() checkmate::assert_string(dataname) @@ -278,11 +284,10 @@ FilteredDataset <- R6::R6Class( # nolint }, #' @description - #' Server module for a dataset active filters + #' Server module for a dataset active filters. #' - #' Server module managing a active filters. - #' @param id (`character(1)`)\cr - #' an ID string that corresponds with the ID used to call the module's UI function. + #' @param id (`character(1)`) + #' an id string that corresponds with the id used to call the module's `ui_active` function. #' @return `moduleServer` function which returns `NULL` srv_active = function(id) { moduleServer( @@ -333,28 +338,26 @@ FilteredDataset <- R6::R6Class( # nolint }, #' @description - #' UI module to add filter variable for this dataset + #' UI module to add filter variable for this dataset. #' - #' UI module to add filter variable for this dataset - #' @param id (`character(1)`)\cr + #' @param id (`character(1)`) #' identifier of the element - preferably containing dataset name #' - #' @return function - shiny UI module + #' @return function - `shiny` UI module ui_add = function(id) { stop("Pure virtual method") }, #' @description - #' Server module to add filter variable for this dataset - #' #' Server module to add filter variable for this dataset. #' For this class `srv_add` calls multiple modules #' of the same name from `FilterStates` as `MAEFilteredDataset` #' contains one `FilterStates` object for `colData` and one for each #' experiment. #' - #' @param id (`character(1)`)\cr - #' an ID string that corresponds with the ID used to call the module's UI function. + #' @param id (`character(1)`) + #' an id string that corresponds with the id used to call the module's + #' `ui_add` function. #' #' @return `moduleServer` function which returns `NULL` #' @@ -375,7 +378,7 @@ FilteredDataset <- R6::R6Class( # nolint ) } ), - ## __Private Fields ==== + # private fields ---- private = list( dataset = NULL, # data.frame or MultiAssayExperiment data_filtered = NULL, @@ -398,7 +401,7 @@ FilteredDataset <- R6::R6Class( # nolint # @description # Gets the active `FilterStates` objects. - # @param id (`character(1)`, `character(0)`)\cr + # @param id (`character(1)`, `character(0)`) # the id of the `private$filter_states` list element where `FilterStates` is kept. # @return `FilterStates` or `list` of `FilterStates` objects. get_filter_states = function() { diff --git a/R/FilteredDatasetDataframe.R b/R/FilteredDatasetDataframe.R index a58eccdd1..11d38967b 100644 --- a/R/FilteredDatasetDataframe.R +++ b/R/FilteredDatasetDataframe.R @@ -1,9 +1,13 @@ # DataframeFilteredDataset ------ -#' @title The `DataframeFilteredDataset` R6 class -#' @keywords internal + +#' @name DataframeFilteredDataset +#' @docType class +#' @title The `DataframeFilteredDataset` `R6` class #' @examples -#' library(shiny) -#' ds <- teal.slice:::DataframeFilteredDataset$new(iris, "iris") +#' # use non-exported function from teal.slice +#' DataframeFilteredDataset <- getFromNamespace("DataframeFilteredDataset", "teal.slice") +#' +#' ds <- DataframeFilteredDataset$new(iris, "iris") #' ds$set_filter_state( #' teal_slices( #' teal_slice(dataname = "iris", varname = "Species", selected = "virginica"), @@ -12,34 +16,45 @@ #' ) #' isolate(ds$get_filter_state()) #' isolate(ds$get_call()) +#' +#' ## set_filter_state +#' dataset <- DataframeFilteredDataset$new(iris, "iris") +#' fs <- teal_slices( +#' teal_slice(dataname = "iris", varname = "Species", selected = "virginica"), +#' teal_slice(dataname = "iris", varname = "Petal.Length", selected = c(2.0, 5)) +#' ) +#' dataset$set_filter_state(state = fs) +#' isolate(dataset$get_filter_state()) +#' @keywords internal +#' DataframeFilteredDataset <- R6::R6Class( # nolint classname = "DataframeFilteredDataset", inherit = FilteredDataset, - ## Public Fields ---- + # public fields ---- public = list( #' @description - #' Initializes this `DataframeFilteredDataset` object + #' Initializes this `DataframeFilteredDataset` object. #' - #' @param dataset (`data.frame`)\cr + #' @param dataset (`data.frame`) #' single data.frame for which filters are rendered - #' @param dataname (`character`)\cr + #' @param dataname (`character`) #' A given name for the dataset it may not contain spaces - #' @param keys optional, (`character`)\cr + #' @param keys optional, (`character`) #' Vector with primary keys - #' @param parent_name (`character(1)`)\cr + #' @param parent_name (`character(1)`) #' Name of the parent dataset - #' @param parent (`reactive`)\cr + #' @param parent (`reactive`) #' object returned by this reactive is a filtered `data.frame` from other `FilteredDataset` #' named `parent_name`. Consequence of passing `parent` is a `reactive` link which causes #' causing re-filtering of this `dataset` based on the changes in `parent`. - #' @param join_keys (`character`)\cr + #' @param join_keys (`character`) #' Name of the columns in this dataset to join with `parent` #' dataset. If the column names are different if both datasets #' then the names of the vector define the `parent` columns. #' - #' @param label (`character`)\cr + #' @param label (`character`) #' Label to describe the dataset initialize = function(dataset, dataname, @@ -100,7 +115,7 @@ DataframeFilteredDataset <- R6::R6Class( # nolint }, #' @description - #' Gets the filter expression + #' Gets the filter expression. #' #' This functions returns filter calls equivalent to selected items #' within each of `filter_states`. Configuration of the calls is constant and @@ -109,7 +124,7 @@ DataframeFilteredDataset <- R6::R6Class( # nolint #' which contains single `state_list` and all `FilterState` objects #' applies to one argument (`...`) in `dplyr::filter` call. #' - #' @param sid (`character`)\cr + #' @param sid (`character`) #' when specified then method returns code containing filter conditions of #' `FilterState` objects which `"sid"` attribute is different than this `sid` argument. #' @@ -163,19 +178,9 @@ DataframeFilteredDataset <- R6::R6Class( # nolint }, #' @description - #' Set filter state - #' - #' @param state (`teal_slice`) object - #' - #' @examples - #' dataset <- teal.slice:::DataframeFilteredDataset$new(iris, "iris") - #' fs <- teal_slices( - #' teal_slice(dataname = "iris", varname = "Species", selected = "virginica"), - #' teal_slice(dataname = "iris", varname = "Petal.Length", selected = c(2.0, 5)) - #' ) - #' dataset$set_filter_state(state = fs) - #' shiny::isolate(dataset$get_filter_state()) + #' Set filter state. #' + #' @param state (`teal_slices`) #' @return `NULL` invisibly #' set_filter_state = function(state) { @@ -191,9 +196,9 @@ DataframeFilteredDataset <- R6::R6Class( # nolint }, #' @description - #' Remove one or more `FilterState` form a `FilteredDataset` + #' Remove one or more `FilterState` form a `FilteredDataset`. #' - #' @param state (`teal_slices`)\cr + #' @param state (`teal_slices`) #' specifying `FilterState` objects to remove; #' `teal_slice`s may contain only `dataname` and `varname`, other elements are ignored #' @@ -215,13 +220,12 @@ DataframeFilteredDataset <- R6::R6Class( # nolint }, #' @description - #' UI module to add filter variable for this dataset + #' UI module to add filter variable for this dataset. #' - #' UI module to add filter variable for this dataset - #' @param id (`character(1)`)\cr + #' @param id (`character(1)`) #' identifier of the element - preferably containing dataset name #' - #' @return function - shiny UI module + #' @return function - `shiny` UI module ui_add = function(id) { ns <- NS(id) tagList( @@ -231,7 +235,7 @@ DataframeFilteredDataset <- R6::R6Class( # nolint }, #' @description - #' Get number of observations based on given keys + #' Get number of observations based on given keys. #' The output shows the comparison between `filtered_dataset` #' function parameter and the dataset inside self #' @return `list` containing character `#filtered/#not_filtered` @@ -264,7 +268,7 @@ DataframeFilteredDataset <- R6::R6Class( # nolint } ), - ## Private Fields ---- + # private fields ---- private = list( parent_name = character(0), join_keys = character(0) diff --git a/R/FilteredDatasetDefault.R b/R/FilteredDatasetDefault.R index f9052b800..a5157a28c 100644 --- a/R/FilteredDatasetDefault.R +++ b/R/FilteredDatasetDefault.R @@ -1,28 +1,32 @@ # DefaultFilteredDataset ---- -#' @title The `DefaultFilteredDataset` R6 class +#' @name DefaultFilteredDataset +#' @docType class +#' @title `DefaultFilteredDataset` `R6` class #' @description Stores any object as inert entity. Filtering is not supported. -#' @keywords internal #' @examples -#' library(shiny) -#' ds <- teal.slice:::DefaultFilteredDataset$new(letters, "letters") +#' # use non-exported function from teal.slice +#' DefaultFilteredDataset <- getFromNamespace("DefaultFilteredDataset", "teal.slice") +#' +#' ds <- DefaultFilteredDataset$new(letters, "letters") #' isolate(ds$get_filter_state()) #' isolate(ds$get_call()) +#' @keywords internal DefaultFilteredDataset <- R6::R6Class( # nolint classname = "DefaultFilteredDataset", inherit = FilteredDataset, - ## Public Methods ---- + # public methods ---- public = list( #' @description - #' Initializes this `DefaultFilteredDataset` object + #' Initializes this `DefaultFilteredDataset` object. #' #' @param dataset #' Any type of object; will not be filtered. - #' @param dataname (`character(1)`)\cr + #' @param dataname (`character(1)`) #' Name given to the dataset; must not contain spaces. - #' @param label (`character(1)`)\cr + #' @param label (`character(1)`) #' Label to describe the dataset. initialize = function(dataset, dataname, label = character(0)) { super$initialize(dataset = dataset, dataname = dataname, label = label) @@ -77,13 +81,13 @@ DefaultFilteredDataset <- R6::R6Class( # nolint data.frame(dataname = private$dataname, obs = NA, obs_filtered = NA) }, - ### shiny modules ---- + # shiny modules ---- #' @description #' Overwrites parent method. #' @details - #' Blank module UI that would list active filter states for this dataset. - #' @param id (`character(1)`)\cr + #' Blank UI module that would list active filter states for this dataset. + #' @param id (`character(1)`) #' `shiny` module id #' @return empty `div` ui_active = function(id) { @@ -94,8 +98,8 @@ DefaultFilteredDataset <- R6::R6Class( # nolint #' @description #' Overwrites parent method. #' @details - #' Blank module UI that would list active filter states for this dataset. - #' @param id (`character(1)`)\cr + #' Blank UI module that would list active filter states for this dataset. + #' @param id (`character(1)`) #' `shiny` module id #' @return empty `div` ui_add = function(id) { @@ -104,7 +108,7 @@ DefaultFilteredDataset <- R6::R6Class( # nolint } ), private = list( - ## Private Methods ---- - ## Private Fields ---- + # private methods ---- + # private fields ---- ) ) diff --git a/R/FilteredDatasetMAE.R b/R/FilteredDatasetMAE.R index 9629243b1..c689befdf 100644 --- a/R/FilteredDatasetMAE.R +++ b/R/FilteredDatasetMAE.R @@ -1,6 +1,32 @@ # MAEFilteredDataset ------ + +#' @name MAEFilteredDataset +#' @docType class +#' @title `MAEFilteredDataset` `R6` class #' @keywords internal -#' @title `MAEFilteredDataset` R6 class +#' @examplesIf requireNamespace("MultiAssayExperiment") +#' # use non-exported function from teal.slice +#' MAEFilteredDataset <- getFromNamespace("MAEFilteredDataset", "teal.slice") +#' +#' utils::data(miniACC, package = "MultiAssayExperiment") +#' dataset <- MAEFilteredDataset$new(miniACC, "MAE") +#' fs <- teal_slices( +#' teal_slice( +#' dataname = "MAE", varname = "years_to_birth", selected = c(30, 50), keep_na = TRUE +#' ), +#' teal_slice( +#' dataname = "MAE", varname = "vital_status", selected = "1", keep_na = FALSE +#' ), +#' teal_slice( +#' dataname = "MAE", varname = "gender", selected = "female", keep_na = TRUE +#' ), +#' teal_slice( +#' dataname = "MAE", varname = "ARRAY_TYPE", selected = "", keep_na = TRUE +#' ) +#' ) +#' dataset$set_filter_state(state = fs) +#' isolate(dataset$get_filter_state()) +#' MAEFilteredDataset <- R6::R6Class( # nolint classname = "MAEFilteredDataset", inherit = FilteredDataset, @@ -8,15 +34,15 @@ MAEFilteredDataset <- R6::R6Class( # nolint # public methods ---- public = list( #' @description - #' Initialize `MAEFilteredDataset` object + #' Initialize `MAEFilteredDataset` object. #' - #' @param dataset (`MulitiAssayExperiment`)\cr + #' @param dataset (`MulitiAssayExperiment`) #' a single `MultiAssayExperiment` for which to define a subset - #' @param dataname (`character`)\cr + #' @param dataname (`character`) #' a given name for the dataset it may not contain spaces - #' @param keys optional, (`character`)\cr + #' @param keys optional, (`character`) #' vector with primary keys - #' @param label (`character`)\cr + #' @param label (`character`) #' label to describe the dataset initialize = function(dataset, dataname, keys = character(0), label = character(0)) { if (!requireNamespace("MultiAssayExperiment", quietly = TRUE)) { @@ -58,34 +84,9 @@ MAEFilteredDataset <- R6::R6Class( # nolint }, #' @description - #' Set filter state - #' - #' @param state (`named list`)\cr - #' names of the list should correspond to the names of the initialized `FilterStates` - #' kept in `private$filter_states`. For this object they are `"subjects"` and - #' names of the experiments. Values of initial state should be relevant - #' to the referred column. - #' - #' @examples - #' utils::data(miniACC, package = "MultiAssayExperiment") - #' dataset <- teal.slice:::MAEFilteredDataset$new(miniACC, "MAE") - #' fs <- teal_slices( - #' teal_slice( - #' dataname = "MAE", varname = "years_to_birth", selected = c(30, 50), keep_na = TRUE - #' ), - #' teal_slice( - #' dataname = "MAE", varname = "vital_status", selected = "1", keep_na = FALSE - #' ), - #' teal_slice( - #' dataname = "MAE", varname = "gender", selected = "female", keep_na = TRUE - #' ), - #' teal_slice( - #' dataname = "MAE", varname = "ARRAY_TYPE", selected = "", keep_na = TRUE - #' ) - #' ) - #' dataset$set_filter_state(state = fs) - #' shiny::isolate(dataset$get_filter_state()) + #' Set filter state. #' + #' @param state (`teal_slices`) #' @return `NULL` invisibly #' set_filter_state = function(state) { @@ -127,9 +128,9 @@ MAEFilteredDataset <- R6::R6Class( # nolint }, #' @description - #' Remove one or more `FilterState` of a `MAEFilteredDataset` + #' Remove one or more `FilterState` of a `MAEFilteredDataset`. #' - #' @param state (`teal_slices`)\cr + #' @param state (`teal_slices`) #' specifying `FilterState` objects to remove; #' `teal_slice`s may contain only `dataname` and `varname`, other elements are ignored #' @@ -171,13 +172,11 @@ MAEFilteredDataset <- R6::R6Class( # nolint }, #' @description - #' UI module to add filter variable for this dataset - #' - #' UI module to add filter variable for this dataset - #' @param id (`character(1)`)\cr + #' UI module to add filter variable for this dataset. + #' @param id (`character(1)`) #' identifier of the element - preferably containing dataset name #' - #' @return function - shiny UI module + #' @return function - `shiny` UI module #' ui_add = function(id) { ns <- NS(id) @@ -206,7 +205,7 @@ MAEFilteredDataset <- R6::R6Class( # nolint }, #' @description - #' Get filter overview rows of a dataset + #' Get filter overview rows of a dataset. #' @return (`matrix`) matrix of observations and subjects get_filter_overview = function() { data <- self$get_dataset() diff --git a/R/calls_combine_by.R b/R/calls_combine_by.R index 29fe47f6e..18c5e4583 100644 --- a/R/calls_combine_by.R +++ b/R/calls_combine_by.R @@ -1,32 +1,36 @@ -#' Combine calls by operator +#' Compose predicates #' -#' Combine list of calls by specific operator +#' Combines calls with a logical operator. #' -#' @param calls (`list` of calls)\cr -#' list containing calls to be combined by `operator`; -#' if empty, NULL is returned -#' @param operator (`character(1)`)\cr -#' name/symbol of the operator passed as character string +#' This function is used to combine logical predicates produced by `FilterState` objects +#' to build a complete subsetting expression. #' -#' @return call or NULL, if `calls` is an empty list +#' @param calls (`list`) +#' containing calls (or symbols) to be combined by `operator` +#' @param operator (`character(1)`) +#' infix operator to use in predicate composition, _e.g._ `"&"` +#' +#' @return +#' A `call` where elements of `calls` are composed with `operator` or `NULL` if `calls` is an empty list. #' #' @examples +#' # use non-exported function from teal.slice +#' calls_combine_by <- getFromNamespace("calls_combine_by", "teal.slice") +#' #' calls <- list( #' quote(SEX == "F"), # subsetting on factor #' quote(AGE >= 20 & AGE <= 50), # subsetting on range #' quote(!SURV) # subsetting on logical #' ) -#' teal.slice:::calls_combine_by(calls, "&") +#' calls_combine_by(calls, "&") #' -#' @return a combined `call` #' @keywords internal +#' calls_combine_by <- function(calls, operator) { checkmate::assert_list(calls) if (length(calls) > 0L) checkmate::assert_list(calls, types = c("call", "name")) checkmate::assert_string(operator) - calls <- Filter(x = calls, f = Negate(is.null)) - Reduce( x = calls, f = function(x, y) call(operator, x, y) diff --git a/R/choices_labeled.R b/R/choices_labeled.R index 4c7da4763..99ba1e291 100644 --- a/R/choices_labeled.R +++ b/R/choices_labeled.R @@ -1,17 +1,20 @@ -#' Set "`: