Skip to content
New issue

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

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

Already on GitHub? Sign in to your account

shinyvalidate improvements #199

Merged
merged 78 commits into from
Jan 3, 2023
Merged
Show file tree
Hide file tree
Changes from 14 commits
Commits
Show all changes
78 commits
Select commit Hold shift + click to select a range
bc13fa9
add function gather_fails
Nov 28, 2022
7473a83
gather_fails in tm_g_a_oview
Nov 28, 2022
07dc66e
update examples
Nov 29, 2022
f589883
replace validate with shinyvalidate
Nov 29, 2022
3e7e800
use validations in tm_g_ae_sub
Nov 29, 2022
3d80d64
use validations in tm_g_butterfly
Nov 29, 2022
c77cbb6
use validations in tm_g_events_ternm_id
Nov 29, 2022
d803659
use validations in tm_g_spiderplot
Nov 29, 2022
cedcbb5
use validations in tm_g_swimlane
Nov 29, 2022
2917c8b
use validations in tm_g_waterfall
Nov 29, 2022
de1c988
minor edits
Nov 29, 2022
976252c
use validations in tm_g_heat_bygrade
Nov 30, 2022
fc1bfcf
use validations in tm_g_patient_profile
Nov 30, 2022
dad8481
fix validations of selected values
Nov 30, 2022
b29540c
apply review and clean up
Dec 1, 2022
b6b54f5
add namespace prefixes for gather_fails
Dec 1, 2022
b2ee629
move gather_fails to teal package
Dec 1, 2022
3380b6b
update DESCRIPTION
Dec 1, 2022
3250430
update NEWS
Dec 7, 2022
78a7203
changed 'gather_fails' to 'validate_inputs'
Dec 7, 2022
a509412
Merge 78a7203100605a85e60ce414a5afdeef697d979d into f07ee20511a66e052…
chlebowa Dec 7, 2022
221b250
[skip actions] Restyle files
github-actions[bot] Dec 7, 2022
cb9d781
trigger
Dec 8, 2022
133bbe1
apply code review
Dec 8, 2022
1f63bf4
update DESCRIPTION
Dec 8, 2022
f667bac
add workflow badges to README
Dec 9, 2022
21b6b43
update code review
Dec 9, 2022
2c1052e
Merge 21b6b439ba057b0936121908dd934057b4a57efc into f07ee20511a66e052…
chlebowa Dec 9, 2022
a7608a1
[skip actions] Restyle files
github-actions[bot] Dec 9, 2022
905d4f8
rework patient_profile module
Dec 9, 2022
2e18889
Merge 905d4f86d50a0f8a34937b9e4298410d79de65bf into f07ee20511a66e052…
chlebowa Dec 9, 2022
9f1de3e
[skip actions] Restyle files
github-actions[bot] Dec 9, 2022
0512830
adjust patient_profile module
Dec 9, 2022
2f7580b
Merge 05128304c53cad3cae9ce348f25d520086a0e837 into f07ee20511a66e052…
chlebowa Dec 9, 2022
fa5bf0a
[skip actions] Restyle files
github-actions[bot] Dec 9, 2022
ba68b65
trigger
Dec 9, 2022
980b4a2
move ivs to reactives, fix bugs
Dec 12, 2022
f46296b
amend DEWSCRIPTION and NEWS
Dec 12, 2022
385cbc0
Merge branch 'main' into 185_gather_fails@main
chlebowa Dec 12, 2022
337a0bb
Merge 385cbc0d6bf0c8783664458089763e6eba80f6b0 into 531c64f03499b10e1…
chlebowa Dec 12, 2022
2e0d4ea
[skip actions] Restyle files
github-actions[bot] Dec 12, 2022
dbd52a3
apply code review
Dec 14, 2022
127d9a2
apply more code review
Dec 14, 2022
532b4c7
linter
Dec 14, 2022
a448b02
more review of patient_profile
Dec 14, 2022
9bbf16e
apply review of ~heat_bygrade
Dec 14, 2022
bb06f3c
apply review to spiderplot
Dec 14, 2022
24eb0f1
more review patient_profile
Dec 14, 2022
54706cf
Merge 24eb0f11aade9f5f36a52fc692c7a2391ff47f49 into 531c64f03499b10e1…
chlebowa Dec 14, 2022
8295598
[skip actions] Restyle files
github-actions[bot] Dec 14, 2022
ded2d75
update NEWS
Dec 14, 2022
6e069b7
Move checkbox init to UI
gogonzo Dec 15, 2022
9103936
fix calls
gogonzo Dec 15, 2022
0018423
Merge 9103936031492fa18239a64c5dec71a0af9ff436 into 531c64f03499b10e1…
chlebowa Dec 15, 2022
94cd612
[skip actions] Restyle files
github-actions[bot] Dec 15, 2022
6f8f767
Fixing sl_dataname in the g_patient_profile call
gogonzo Dec 15, 2022
64c9687
Merge branch '185_gather_fails@main' of github.com:insightsengineerin…
gogonzo Dec 15, 2022
bfe2d76
fix
gogonzo Dec 15, 2022
fdefc39
fix
gogonzo Dec 15, 2022
4c675ed
fix heat_by_grade and reload docs
gogonzo Dec 15, 2022
14738f4
fix checkboxes error
gogonzo Dec 15, 2022
dac01ad
fix example
Dec 21, 2022
adb5b68
indentation
Dec 21, 2022
1080a54
add debouce to plot in patient_profile
Dec 21, 2022
52b9b5e
add debouce to plot in heat_bygrade
Dec 21, 2022
27b19dd
add validation to heat_bygrade
Dec 21, 2022
74c5da4
move debounce in patient_profile
Dec 21, 2022
4fd69ce
update validation in events_term_id
Dec 21, 2022
800ccf7
add debounce in butterfly
Dec 21, 2022
82f7c80
add debounce in ae_sub
Dec 21, 2022
56416fa
add debounce in ae_oview
Dec 21, 2022
650f8f6
Merge 56416fadac040a067a31a106f18e718de86e557e into 531c64f03499b10e1…
chlebowa Dec 21, 2022
aeb6563
[skip actions] Restyle files
github-actions[bot] Dec 21, 2022
5d71863
trigger
Dec 21, 2022
0ae47bc
bug fix and NEWS update
Dec 22, 2022
65a1daf
[skip actions] Roxygen Man Pages Auto Update
dependabot-preview[bot] Dec 22, 2022
146305c
trigger checks
Jan 3, 2023
83e2354
trigger checks again
Jan 3, 2023
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
169 changes: 169 additions & 0 deletions R/gather_fails.R
Original file line number Diff line number Diff line change
@@ -0,0 +1,169 @@

#' send input validation messages to output
#'
#' Captures messages from `InputValidator` objects and collates them
#' into one message passed to `validate`.
#'
BLAZEWIM marked this conversation as resolved.
Show resolved Hide resolved
#' `shiny::validate` is used to withhold rendering of an output element until
#' certain conditions are met and a print a validation message in place
#' of the output element.
#' `shinyvalidate` allows to validate input elements and display specific messages
#' in their respective input widgets.
#' This function is a hybrid solution. Given an `InputValidator` object,
#' it extracts messages from inputs that fail validation and places them all in one
#' validation message that is passed to a `validate`/`need` call.
#' This way the input validator messages are repeated in the output.
#'
#' \code{gather_fails} accepts one `InputValidator`
#' and can add a header to its validation messages.
#' \code{gather_fails_com} accepts an arbitrary number of `InputValidator`s
#' and prints all messages together under one header.
#' \code{gather_fails_grp} accepts a \strong{list} of `InputValidator`s
BLAZEWIM marked this conversation as resolved.
Show resolved Hide resolved
#' and prints messages in groups. If elements of \code{validators} are named,
#' the names are used as headers for their respective message groups.
#'
#'
#' @name gather_fails
BLAZEWIM marked this conversation as resolved.
Show resolved Hide resolved
#'
#' @param iv object of class `InputValidator`
#' @param header `character(1)` optional generic validation message
#' @param ... arguments passed to `shiny::validate`
#' @param validators optionally named `list` of `InputValidator` objects, see\code{Details}
#'
#' @return
#' Returns NULL if the final validation call passes and a `shiny.silent.error` if it fails.
#'
#' @seealso \code{\link{[shinyvalidate::InputValidator]}} \code{\link{[shiny::validate]}}
#'
#' @examples
#' library(shiny)
#' library(shinyvalidate)
#'
#' ui <- fluidPage(
#' selectInput("method", "validation method", c("hierarchical", "combined", "grouped")),
#' sidebarLayout(
#' sidebarPanel(
#' selectInput("letter", "select a letter:", c(letters[1:3], LETTERS[4:6])),
#' selectInput("number", "select a number:", 1:6),
#' br(),
#' selectInput("color", "select a color:",
#' c("black", "indianred2", "springgreen2", "cornflowerblue"),
#' multiple = TRUE),
#' sliderInput("size", "select point size:",
#' min = 0.1, max = 4, value = 0.25)
#' ),
#' mainPanel(plotOutput('plot'))
#' )
#' )
#'
#' server <- function(input, output) {
#' # set up input validation
#' iv <- InputValidator$new()
#' iv$add_rule("letter", sv_in_set(LETTERS, "choose a capital letter"))
#' iv$add_rule("number", ~ if (as.integer(.) %% 2L == 1L) "choose an even number")
#' iv$enable()
#' # more input validation
#' iv_par <- InputValidator$new()
#' iv_par$add_rule("color", sv_required(message = "choose a color"))
#' iv_par$add_rule("color", ~ if (length(.) > 1L) "choose only one color")
#' iv_par$add_rule("size", sv_between(left = 0.5, right = 3,
#' message_fmt = "choose a value between {left} and {right}"))
#' iv_par$enable()
#'
#'
#' output$plot <- renderPlot({
#' # validate output
#' switch(input[["method"]],
#' "hierarchical" = {
#' gather_fails(iv)
#' gather_fails(iv_par, "Set proper graphical parameters")
#' },
#' "combined" = gather_fails_com(iv, iv_par),
#' "grouped" = gather_fails_grp(list(
#' "Some inputs require attention" = iv,
#' "Set proper graphical parameters" = iv_par
#' )))
#'
#' plot(eruptions ~ waiting, faithful, las = 1, pch = 16,
#' col = input[["color"]], cex = input[["size"]])
#' })
#' }
#'
#' if (interactive()) {
#' shinyApp(ui, server)
#' }


#' @rdname gather_fials
chlebowa marked this conversation as resolved.
Show resolved Hide resolved
#' @export
gather_fails <- function(iv, header = "Some inputs require attention", ...) {
checkmate::assert_class(iv, "InputValidator")
checkmate::assert_string(header, null.ok = TRUE)

fail_messages <- gather_messages(iv)
failings <- add_header(fail_messages, header)

shiny::validate(shiny::need(is.null(failings), failings), ...)
}


#' @rdname gather_fials
chlebowa marked this conversation as resolved.
Show resolved Hide resolved
#' @export
gather_fails_com <- function(..., header = "Some inputs require attention") {
vals <- list(...)
lapply(vals, checkmate::assert_class, "InputValidator")
checkmate::assert_string(header, null.ok = TRUE)

fail_messages <- unlist(lapply(vals, gather_messages))
failings <- add_header(fail_messages, header)

shiny::validate(shiny::need(is.null(failings), failings))
}


#' @rdname gather_fials
chlebowa marked this conversation as resolved.
Show resolved Hide resolved
#' @export
gather_fails_grp <- function(validators, ...) {
checkmate::assert_list(validators, types = "InputValidator")

# Since some or all names may be NULL, mapply cannot be used here, a loop is required.
fail_messages <- vector("list", length(validators))
for (v in seq_along(validators)) {
fail_messages[[v]] <- gather_and_add(validators[[v]], names(validators)[v])
}

failings <- unlist(fail_messages)

shiny::validate(shiny::need(is.null(failings), failings), ...)
}


### internal functions

#' @keywords internal
# internal used by all methods
# collate failing messages from validator
gather_messages <- function(iv) {
status <- iv$validate()
failing_inputs <- Filter(Negate(is.null), status)
unique(lapply(failing_inputs, function(x) x[["message"]]))
}


#' @keywords internal
# internal used by all hierarchical and combined methods
# format failing messages with optional header message
add_header <- function(messages, header) {
if (length(messages) > 0L) {
c(paste0(header, "\n"), unlist(messages), "\n")
} else NULL
}

#' @keywords internal
# collate failing messages with optional header message
# internal used by grouped method
gather_and_add <- function(iv, header) {
BLAZEWIM marked this conversation as resolved.
Show resolved Hide resolved
fail_messages <- gather_messages(iv)
failings <- add_header(fail_messages, header)
failings
}
63 changes: 30 additions & 33 deletions R/tm_g_ae_oview.R
Original file line number Diff line number Diff line change
Expand Up @@ -120,7 +120,8 @@ tm_g_ae_oview <- function(label,
)
)
checkmate::assert_numeric(plot_height, len = 3, any.missing = FALSE, finite = TRUE)
checkmate::assert_numeric(plot_height[1], lower = plot_height[2], upper = plot_height[3], .var.name = "plot_height")
checkmate::assert_numeric(plot_height[1], lower = plot_height[2], upper = plot_height[3],
.var.name = "plot_height")
checkmate::assert_numeric(plot_width, len = 3, any.missing = FALSE, null.ok = TRUE, finite = TRUE)
checkmate::assert_numeric(
plot_width[1],
Expand Down Expand Up @@ -232,14 +233,9 @@ srv_g_ae_oview <- function(id,
checkmate::assert_class(data, "tdata")

moduleServer(id, function(input, output, session) {
iv <- shinyvalidate::InputValidator$new()
iv$add_rule("arm_var", shinyvalidate::sv_required())
iv$add_rule("flag_var_anl", shinyvalidate::sv_required(
message = "Please select at least one flag"
))
iv$enable()

decorate_output <- srv_g_decorate(id = NULL, plt = plot_r, plot_height = plot_height, plot_width = plot_width)
decorate_output <- srv_g_decorate(id = NULL, plt = plot_r,
plot_height = plot_height, plot_width = plot_width)
font_size <- decorate_output$font_size
pws <- decorate_output$pws

Expand Down Expand Up @@ -285,34 +281,35 @@ srv_g_ae_oview <- function(id,

output_q <- reactive({
ANL <- data[[dataname]]() # nolint
validate(need(iv$is_valid(), "Misspecification error: please observe red flags in the encodings."))
validate(need(
is.factor(ANL[[input$arm_var]]),
"Selected arm variable needs to be a factor."
))
validate(need(input$flag_var_anl, "Please select at least one flag."))
iv_comp <- shinyvalidate::InputValidator$new()
iv_comp$add_rule("arm_trt", shinyvalidate::sv_not_equal(

teal::validate_has_data(ANL, min_nrow = 10, msg = sprintf("%s has not enough data", dataname))

# set up and enable input validator(s)
iv <- shinyvalidate::InputValidator$new()
nikolas-burkoff marked this conversation as resolved.
Show resolved Hide resolved
chlebowa marked this conversation as resolved.
Show resolved Hide resolved
iv$add_rule("arm_var", shinyvalidate::sv_required(
message = "Arm Variable is required"))
iv$add_rule("arm_var", ~ if (!is.factor(ANL[[req(.)]]))
"Arm Var must be a factor variable")
iv$add_rule("arm_var", ~ if (length(unique(ANL[[req(.)]])) < 2)
"Selected Arm Var has not enough treatments to compare")
iv$add_rule("flag_var_anl", shinyvalidate::sv_required(
message = "At least one Flag is required"))
iv$add_rule("arm_trt", shinyvalidate::sv_not_equal(
input$arm_ref,
message_fmt = "Must not be equal to Control"
))
iv_comp$add_rule("arm_ref", shinyvalidate::sv_not_equal(
message_fmt = "Control and Treatment must be different"))
iv$add_rule("arm_ref", shinyvalidate::sv_not_equal(
input$arm_trt,
message_fmt = "Must not be equal to Treatment"
))
iv_comp$enable()
message_fmt = "Control and Treatment must be different"))
iv$add_rule("arm_trt", shinyvalidate::sv_in_set(
set = unique(ANL[[req(input$arm_var)]]),
message_fmt = "Treatment not found in Arm Variable"))
iv$add_rule("arm_ref", shinyvalidate::sv_in_set(
set = unique(ANL[[req(input$arm_var)]]),
message_fmt = "Control not found in Arm Variable"))
iv$enable()

validate(need(iv_comp$is_valid(), "Misspecification error: please observe red flags in the encodings."))
validate(need(nlevels(ANL[[input$arm_var]]) > 1, "Arm needs to have at least 2 levels"))
validate_has_data(ANL, min_nrow = 10)
if (all(c(input$arm_trt, input$arm_ref) %in% ANL[[input$arm_var]])) {
iv_an <- shinyvalidate::InputValidator$new()
iv_an$add_rule("arm_ref", shinyvalidate::sv_in_set(set = ANL[[input$arm_var]]))
iv_an$add_rule("arm_trt", shinyvalidate::sv_in_set(set = ANL[[input$arm_var]]))
iv_an$enable()
validate(need(iv_an$is_valid(), "Misspecification error: please observe red flags in the encodings."))
}
validate(need(all(c(input$arm_trt, input$arm_ref) %in% unique(ANL[[input$arm_var]])), "Plot loading"))
# collate validator messages
gather_fails(iv)

q1 <- teal.code::eval_code(
teal.code::new_qenv(tdata2env(data), code = get_code_tdata(data)),
Expand Down
62 changes: 28 additions & 34 deletions R/tm_g_ae_sub.R
Original file line number Diff line number Diff line change
Expand Up @@ -184,9 +184,6 @@ srv_g_ae_sub <- function(id,
checkmate::assert_class(data, "tdata")

moduleServer(id, function(input, output, session) {
iv <- shinyvalidate::InputValidator$new()
iv$add_rule("arm_var", shinyvalidate::sv_required())
iv$enable()

decorate_output <- srv_g_decorate(
id = NULL,
Expand Down Expand Up @@ -289,40 +286,37 @@ srv_g_ae_sub <- function(id,
output_q <- reactive({
ANL <- data[[dataname]]() # nolint
ADSL <- data[["ADSL"]]() # nolint
validate_has_data(ANL, min_nrow = 10)
iv_comp <- shinyvalidate::InputValidator$new()
iv_comp$add_rule("arm_trt", shinyvalidate::sv_not_equal(

teal::validate_has_data(ANL, min_nrow = 10, msg = sprintf("%s has not enough data", dataname))

# set up and enable input validator(s)
iv <- shinyvalidate::InputValidator$new()
iv$add_rule("arm_var", shinyvalidate::sv_required(
message = "Arm Variable is required"))
iv$add_rule("arm_var", ~ if (!is.factor(ANL[[req(.)]]))
"Arm Var must be a factor variable, contact developer")
iv$add_rule("arm_trt", shinyvalidate::sv_not_equal(
input$arm_ref,
message_fmt = "Must not be equal to Control"
))
iv_comp$add_rule("arm_ref", shinyvalidate::sv_not_equal(
message_fmt = "Control and Treatment must be different"))
iv$add_rule("arm_ref", shinyvalidate::sv_not_equal(
input$arm_trt,
message_fmt = "Must not be equal to Treatment"
))
iv_comp$enable()
validate(need(iv_comp$is_valid(), "Misspecification error: please observe red flags in the encodings."))
message_fmt = "Control and Treatment must be different"))
iv$add_rule("groups", shinyvalidate::sv_in_set(
names(ANL),
message_fmt = "Groups must be a variable in ANL"))
iv$add_rule("groups", shinyvalidate::sv_in_set(
names(ADSL),
message_fmt = "Groups must be a variable in ADSL"))
iv$add_rule("arm_trt", shinyvalidate::sv_in_set(
set = unique(ANL[[req(input$arm_var)]]),
message_fmt = "Treatment not found in Arm Variable"))
iv$add_rule("arm_ref", shinyvalidate::sv_in_set(
set = unique(ANL[[req(input$arm_var)]]),
message_fmt = "Control not found in Arm Variable"))
iv$enable()

validate(need(iv$is_valid(), "Misspecification error: please observe red flags in the encodings."))
validate(need(
is.factor(ANL[[input$arm_var]]),
"Selected arm variable needs to be a factor. Contact the app developer."
))
validate(
need(
all(c(input$arm_trt, input$arm_ref) %in% levels(ADSL[[input$arm_var]])),
"Updating treatment and control selections."
)
)
validate(
need(
all(c(input$arm_trt, input$arm_ref) %in% levels(ANL[[input$arm_var]])),
"The dataset does not contain subjects with AE events from both the control and treatment arms."
),
need(
all(input$groups %in% names(ANL)) & all(input$groups %in% names(ADSL)),
"Check all selected subgroups are columns in ADAE and ADSL."
)
)
# collate validator messages
gather_fails(iv)

group_labels <- lapply(seq_along(input$groups), function(x) {
items <- input[[sprintf("groups__%s", x)]]
Expand Down
Loading