From 0b566c34c79cf43132c9b15908bae3ff1728a36d Mon Sep 17 00:00:00 2001 From: Emily de la Rua <59304861+edelarua@users.noreply.github.com> Date: Thu, 28 Sep 2023 12:14:19 -0400 Subject: [PATCH] Display Patient ID in relevant templates, clean up data tables in reports (#834) Closes #816 --- DESCRIPTION | 1 + NEWS.md | 2 ++ R/tm_g_pp_adverse_events.R | 29 ++++++++++++++---- R/tm_g_pp_therapy.R | 30 ++++++++++++++----- R/tm_t_pp_basic_info.R | 29 ++++++++++++++---- R/tm_t_pp_laboratory.R | 25 +++++++++++++--- R/tm_t_pp_medical_history.R | 12 ++++++-- man/template_basic_info.Rd | 4 ++- man/template_laboratory.Rd | 3 ++ man/template_medical_history.Rd | 5 +++- staged_dependencies.yaml | 3 ++ .../_snaps/tm_t_pp_medical_history.md | 1 + 12 files changed, 117 insertions(+), 27 deletions(-) diff --git a/DESCRIPTION b/DESCRIPTION index 917036a923..80f1cfffc3 100644 --- a/DESCRIPTION +++ b/DESCRIPTION @@ -43,6 +43,7 @@ Imports: magrittr, methods, rlang, + rlistings (>= 0.2.4), rmarkdown, rtables (>= 0.6.1), scales, diff --git a/NEWS.md b/NEWS.md index d947126acc..67212744f8 100644 --- a/NEWS.md +++ b/NEWS.md @@ -10,6 +10,8 @@ * Updated `tm_t_events` to maintain indentation after pruning. * Updated default reference/comparison arm level selection to work when arm variable levels are filtered out. * Updated `tm_t_coxreg` to drop factor covariate variable levels that are not present to avoid errors when filtering. +* Updated `tm_t_pp_basic_info`, `tm_t_pp_medical_history`, `tm_g_pp_therapy`, `tm_g_pp_adverse_events`, and `tm_t_pp_laboratory` to print patient ID above table. +* Updated `tm_t_pp_basic_info`, `tm_g_pp_therapy`, `tm_g_pp_adverse_events`, and `tm_t_pp_laboratory` to use `rlistings` to print data neatly in reports. ### Bug fixes diff --git a/R/tm_g_pp_adverse_events.R b/R/tm_g_pp_adverse_events.R index 59d3cd1b43..ea37176a7e 100644 --- a/R/tm_g_pp_adverse_events.R +++ b/R/tm_g_pp_adverse_events.R @@ -53,6 +53,14 @@ template_adverse_events <- function(dataname = "ANL", ) %>% dplyr::arrange(dplyr::desc(tox_grade)) %>% `colnames<-`(get_labels(dataname)$column_labels[vars]) + + table <- rlistings::as_listing( + table, + key_cols = NULL, + default_formatting = list(all = fmt_config(align = "left")) + ) + main_title(table) <- paste("Patient ID:", patient_id) + table }, env = list( @@ -64,7 +72,8 @@ template_adverse_events <- function(dataname = "ANL", action = as.name(action), time = as.name(time), decod = `if`(is.null(decod), NULL, as.name(decod)), - vars = c(aeterm, tox_grade, causality, outcome, action, time, decod) + vars = c(aeterm, tox_grade, causality, outcome, action, time, decod), + patient_id = patient_id ) ) ) @@ -311,6 +320,7 @@ ui_g_adverse_events <- function(id, ...) { ns <- shiny::NS(id) teal.widgets::standard_layout( output = shiny::div( + shiny::htmlOutput(ns("title")), teal.widgets::get_dt_rows(ns("table"), ns("table_rows")), DT::DTOutput(outputId = ns("table")), teal.widgets::plot_with_settings_ui(id = ns("chart")) @@ -502,8 +512,10 @@ srv_g_adverse_events <- function(id, qenv2 <- teal.code::eval_code( qenv, substitute( - expr = ANL <- ANL[ANL[[patient_col]] == patient_id, ], # nolint - env = list( + expr = { + pt_id <- patient_id + ANL <- ANL[ANL[[patient_col]] == patient_id, ] # nolint + }, env = list( patient_col = patient_col, patient_id = patient_id() ) @@ -526,6 +538,11 @@ srv_g_adverse_events <- function(id, teal.code::eval_code(qenv2, as.expression(calls)) }) + + output$title <- shiny::renderText({ + paste("
Patient ID:", all_q()[["pt_id"]], "
") + }) + output$table <- DT::renderDataTable( expr = teal.code::dev_suppress(all_q()[["table"]]), options = list(pageLength = input$table_rows) @@ -560,11 +577,13 @@ srv_g_adverse_events <- function(id, if (with_reporter) { card_fun <- function(comment) { card <- teal::TealReportCard$new() - card$set_name("Patient Profile Adverse Events Plot") - card$append_text("Patient Profile Adverse Events Plot", "header2") + card$set_name("Patient Profile Adverse Events") + card$append_text("Patient Profile Adverse Events", "header2") if (with_filter) { card$append_fs(filter_panel_api$get_filter_state()) } + card$append_text("Table", "header3") + card$append_table(teal.code::dev_suppress(all_q()[["table"]])) card$append_text("Plot", "header3") card$append_plot(plot_r(), dim = pws$dim()) if (!comment == "") { diff --git a/R/tm_g_pp_therapy.R b/R/tm_g_pp_therapy.R index c5356c8be4..adf42b587d 100644 --- a/R/tm_g_pp_therapy.R +++ b/R/tm_g_pp_therapy.R @@ -87,6 +87,14 @@ template_therapy <- function(dataname = "ANL", get_labels(dataname)$column_labels[c(cmindc_char, cmdecod_char)], "Dosage", get_labels(dataname)$column_labels[c(cmstdy_char, cmendy_char)] )) + + therapy_table <- rlistings::as_listing( + therapy_table, + key_cols = NULL, + default_formatting = list(all = fmt_config(align = "left")) + ) + main_title(therapy_table) <- paste("Patient ID:", patient_id) + therapy_table }, env = list( dataname = as.name(dataname), @@ -108,7 +116,8 @@ template_therapy <- function(dataname = "ANL", cmroute_char = cmroute, cmdosfrq_char = cmdosfrq, cmendy_char = cmendy, - cmstdy_char = cmstdy + cmstdy_char = cmstdy, + patient_id = patient_id )) ) @@ -172,10 +181,9 @@ template_therapy <- function(dataname = "ANL", ggplot2::ggplot(data = data, ggplot2::aes(fill = cmindc, color = cmindc, y = CMDECOD, x = CMSTDY)) + ggplot2::geom_segment(ggplot2::aes(xend = CMENDY, yend = CMDECOD), size = 2) + ggplot2::geom_text( - data = - data %>% - dplyr::select(CMDECOD, cmindc, CMSTDY) %>% - dplyr::distinct(), + data = data %>% + dplyr::select(CMDECOD, cmindc, CMSTDY) %>% + dplyr::distinct(), ggplot2::aes(x = CMSTDY, label = CMDECOD), color = "black", hjust = "left", vjust = "bottom", @@ -411,6 +419,7 @@ ui_g_therapy <- function(id, ...) { ns <- shiny::NS(id) teal.widgets::standard_layout( output = shiny::div( + shiny::htmlOutput(ns("title")), teal.widgets::get_dt_rows(ns("therapy_table"), ns("therapy_table_rows")), DT::DTOutput(outputId = ns("therapy_table")), teal.widgets::plot_with_settings_ui(id = ns("therapy_plot")) @@ -638,6 +647,7 @@ srv_g_therapy <- function(id, merged$anl_q(), substitute( expr = { + pt_id <- patient_id ANL <- ANL[ANL[[patient_col]] == patient_id, ] # nolint }, env = list( patient_col = patient_col, @@ -648,6 +658,10 @@ srv_g_therapy <- function(id, teal.code::eval_code(as.expression(my_calls)) }) + output$title <- shiny::renderText({ + paste("
Patient ID:", all_q()[["pt_id"]], "
") + }) + output$therapy_table <- DT::renderDataTable( expr = { teal.code::dev_suppress(all_q()[["therapy_table"]]) @@ -684,11 +698,13 @@ srv_g_therapy <- function(id, if (with_reporter) { card_fun <- function(comment) { card <- teal::TealReportCard$new() - card$set_name("Patient Profile Therapy Plot") - card$append_text("Patient Profile Therapy Plot", "header2") + card$set_name("Patient Profile Therapy") + card$append_text("Patient Profile Therapy", "header2") if (with_filter) { card$append_fs(filter_panel_api$get_filter_state()) } + card$append_text("Table", "header3") + card$append_table(teal.code::dev_suppress(all_q()[["therapy_table"]])) card$append_text("Plot", "header3") card$append_plot(plot_r(), dim = pws$dim()) if (!comment == "") { diff --git a/R/tm_t_pp_basic_info.R b/R/tm_t_pp_basic_info.R index f535393aea..d6a112f223 100644 --- a/R/tm_t_pp_basic_info.R +++ b/R/tm_t_pp_basic_info.R @@ -4,10 +4,12 @@ #' #' @inheritParams template_arguments #' @param vars (`character`)\cr variable names to be shown in Basic Info tab. +#' @param patient_id (`character`)\cr patient ID. #' @keywords internal #' template_basic_info <- function(dataname = "ANL", - vars) { + vars, + patient_id = NULL) { checkmate::assert_string(dataname) checkmate::assert_character(vars, min.len = 1) @@ -27,13 +29,21 @@ template_basic_info <- function(dataname = "ANL", key <- get_labels(dataname)$column_labels[rownames(values)] result <- - data.frame(key = key, value = values) %>% - dplyr::select(key, value) %>% - dplyr::rename(` ` = key, ` ` = value) + data.frame(var = rownames(values), key = key, value = values) %>% + dplyr::select(var, key, value) %>% + dplyr::rename(` ` = var, ` ` = key, ` ` = value) + + result <- rlistings::as_listing( + result, + default_formatting = list(all = fmt_config(align = "left")) + ) + main_title(result) <- paste("Patient ID:", patient_id) + result }, env = list( dataname = as.name(dataname), - vars = vars + vars = vars, + patient_id = patient_id ) ) ) @@ -119,6 +129,7 @@ ui_t_basic_info <- function(id, ...) { ns <- shiny::NS(id) teal.widgets::standard_layout( output = shiny::div( + shiny::htmlOutput(ns("title")), DT::DTOutput(outputId = ns("basic_info_table")) ), encoding = shiny::div( @@ -221,13 +232,15 @@ srv_t_basic_info <- function(id, teal::validate_inputs(iv_r()) my_calls <- template_basic_info( dataname = "ANL", - vars = anl_inputs()$columns_source$vars + vars = anl_inputs()$columns_source$vars, + patient_id = patient_id() ) teal.code::eval_code( anl_q(), substitute( expr = { + pt_id <- patient_id ANL <- ANL[ANL[[patient_col]] == patient_id, ] # nolint }, env = list( patient_col = patient_col, @@ -238,6 +251,10 @@ srv_t_basic_info <- function(id, teal.code::eval_code(as.expression(my_calls)) }) + output$title <- shiny::renderText({ + paste("
Patient ID:", all_q()[["pt_id"]], "
") + }) + table_r <- shiny::reactive(all_q()[["result"]]) output$basic_info_table <- DT::renderDataTable( diff --git a/R/tm_t_pp_laboratory.R b/R/tm_t_pp_laboratory.R index 84b961e776..16f7f301d1 100644 --- a/R/tm_t_pp_laboratory.R +++ b/R/tm_t_pp_laboratory.R @@ -9,6 +9,7 @@ #' @param anrind (`character`)\cr name of the analysis reference range indicator variable. #' @param aval (`character`)\cr name of the analysis value variable. #' @param avalu (`character`)\cr name of the analysis value unit variable. +#' @param patient_id (`character`)\cr patient ID. #' @param round_value (`numeric`)\cr number of decimal places to be used when rounding. #' @keywords internal #' @@ -19,6 +20,7 @@ template_laboratory <- function(dataname = "ANL", timepoints = "ADY", aval = "AVAL", avalu = "AVALU", + patient_id = NULL, round_value = 0L) { assertthat::assert_that( assertthat::is.string(dataname), @@ -49,16 +51,23 @@ template_laboratory <- function(dataname = "ANL", dplyr::mutate(aval_anrind = paste(aval, anrind)) %>% dplyr::select(-c(aval, anrind)) + labor_table_raw <- labor_table_base %>% + tidyr::pivot_wider(names_from = INDEX, values_from = aval_anrind) %>% + dplyr::mutate(param_char := clean_description(.data[[param_char]])) + + labor_table_raw <- rlistings::as_listing( + labor_table_raw, + key_cols = NULL, + default_formatting = list(all = fmt_config(align = "left")) + ) + main_title(labor_table_raw) <- paste("Patient ID:", patient_id) + labor_table_html <- labor_table_base %>% dplyr::mutate(aval_anrind_col = color_lab_values(aval_anrind)) %>% dplyr::select(-aval_anrind) %>% tidyr::pivot_wider(names_from = INDEX, values_from = aval_anrind_col) %>% dplyr::mutate(param_char := clean_description(.data[[param_char]])) - labor_table_raw <- labor_table_base %>% - tidyr::pivot_wider(names_from = INDEX, values_from = aval_anrind) %>% - dplyr::mutate(param_char := clean_description(.data[[param_char]])) - labor_table_html_dt <- DT::datatable(labor_table_html, escape = FALSE) labor_table_html_dt$dependencies <- c( labor_table_html_dt$dependencies, @@ -76,6 +85,7 @@ template_laboratory <- function(dataname = "ANL", avalu = as.name(avalu), timepoints = as.name(timepoints), anrind = as.name(anrind), + patient_id = patient_id, round_value = round_value ) ) @@ -214,6 +224,7 @@ ui_g_laboratory <- function(id, ...) { ns <- shiny::NS(id) teal.widgets::standard_layout( output = shiny::div( + shiny::htmlOutput(ns("title")), DT::DTOutput(outputId = ns("lab_values_table")) ), encoding = shiny::div( @@ -386,6 +397,7 @@ srv_g_laboratory <- function(id, param = input[[extract_input("param", dataname)]], paramcd = input[[extract_input("paramcd", dataname)]], anrind = input[[extract_input("anrind", dataname)]], + patient_id = patient_id(), round_value = as.integer(input$round_value) ) @@ -393,6 +405,7 @@ srv_g_laboratory <- function(id, anl_q(), substitute( expr = { + pt_id <- patient_id ANL <- ANL[ANL[[patient_col]] == patient_id, ] # nolint }, env = list( patient_col = patient_col, @@ -403,6 +416,10 @@ srv_g_laboratory <- function(id, teal.code::eval_code(as.expression(labor_calls)) }) + output$title <- shiny::renderText({ + paste("
Patient ID:", all_q()[["pt_id"]], "
") + }) + table_r <- shiny::reactive({ q <- all_q() list( diff --git a/R/tm_t_pp_medical_history.R b/R/tm_t_pp_medical_history.R index f77bd6f17e..917ca2127c 100644 --- a/R/tm_t_pp_medical_history.R +++ b/R/tm_t_pp_medical_history.R @@ -6,12 +6,14 @@ #' @param mhterm (`character`)\cr name of the reported name for medical history variable. #' @param mhbodsys (`character`)\cr name of the body system or organ class variable. #' @param mhdistat (`character`)\cr name of the status of the disease variable. +#' @param patient_id (`character`)\cr patient ID. #' @keywords internal #' template_medical_history <- function(dataname = "ANL", mhterm = "MHTERM", mhbodsys = "MHBODSYS", - mhdistat = "MHDISTAT") { + mhdistat = "MHDISTAT", + patient_id = NULL) { assertthat::assert_that( assertthat::is.string(dataname), assertthat::is.string(mhterm), @@ -51,6 +53,8 @@ template_medical_history <- function(dataname = "ANL", rtables::analyze_colvars(function(x) x[seq_along(x)]) %>% rtables::build_table(result_raw) + main_title(result) <- paste("Patient ID:", patient_id) + result }, env = list( dataname = as.name(dataname), @@ -59,7 +63,8 @@ template_medical_history <- function(dataname = "ANL", mhdistat = as.name(mhdistat), mhbodsys_char = mhbodsys, mhterm_char = mhterm, - mhdistat_char = mhdistat + mhdistat_char = mhdistat, + patient_id = patient_id )) ) @@ -297,7 +302,8 @@ srv_t_medical_history <- function(id, dataname = "ANL", mhterm = input[[extract_input("mhterm", dataname)]], mhbodsys = input[[extract_input("mhbodsys", dataname)]], - mhdistat = input[[extract_input("mhdistat", dataname)]] + mhdistat = input[[extract_input("mhdistat", dataname)]], + patient_id = patient_id() ) teal.code::eval_code( diff --git a/man/template_basic_info.Rd b/man/template_basic_info.Rd index 65ba3c18d2..e11bd7f154 100644 --- a/man/template_basic_info.Rd +++ b/man/template_basic_info.Rd @@ -4,13 +4,15 @@ \alias{template_basic_info} \title{Template: Basic Info} \usage{ -template_basic_info(dataname = "ANL", vars) +template_basic_info(dataname = "ANL", vars, patient_id = NULL) } \arguments{ \item{dataname}{(\code{character})\cr analysis data used in teal module.} \item{vars}{(\code{character})\cr variable names to be shown in Basic Info tab.} + +\item{patient_id}{(\code{character})\cr patient ID.} } \description{ Creates a basic info template. diff --git a/man/template_laboratory.Rd b/man/template_laboratory.Rd index aacf45c07a..cb60cd682b 100644 --- a/man/template_laboratory.Rd +++ b/man/template_laboratory.Rd @@ -12,6 +12,7 @@ template_laboratory( timepoints = "ADY", aval = "AVAL", avalu = "AVALU", + patient_id = NULL, round_value = 0L ) } @@ -32,6 +33,8 @@ the laboratory table.} \item{avalu}{(\code{character})\cr name of the analysis value unit variable.} +\item{patient_id}{(\code{character})\cr patient ID.} + \item{round_value}{(\code{numeric})\cr number of decimal places to be used when rounding.} } \description{ diff --git a/man/template_medical_history.Rd b/man/template_medical_history.Rd index 37aeaf8edd..bc6fff058a 100644 --- a/man/template_medical_history.Rd +++ b/man/template_medical_history.Rd @@ -8,7 +8,8 @@ template_medical_history( dataname = "ANL", mhterm = "MHTERM", mhbodsys = "MHBODSYS", - mhdistat = "MHDISTAT" + mhdistat = "MHDISTAT", + patient_id = NULL ) } \arguments{ @@ -20,6 +21,8 @@ analysis data used in teal module.} \item{mhbodsys}{(\code{character})\cr name of the body system or organ class variable.} \item{mhdistat}{(\code{character})\cr name of the status of the disease variable.} + +\item{patient_id}{(\code{character})\cr patient ID.} } \description{ Creates medical history template. diff --git a/staged_dependencies.yaml b/staged_dependencies.yaml index a4e5149826..affdcc2199 100644 --- a/staged_dependencies.yaml +++ b/staged_dependencies.yaml @@ -11,6 +11,9 @@ upstream_repos: insightsengineering/tern.gee: repo: insightsengineering/tern.gee host: https://github.com + insightsengineering/rlistings: + repo: insightsengineering/rlistings + host: https://github.com insightsengineering/rtables: repo: insightsengineering/rtables host: https://github.com diff --git a/tests/testthat/_snaps/tm_t_pp_medical_history.md b/tests/testthat/_snaps/tm_t_pp_medical_history.md index ff7f73dbab..7156efd45c 100644 --- a/tests/testthat/_snaps/tm_t_pp_medical_history.md +++ b/tests/testthat/_snaps/tm_t_pp_medical_history.md @@ -16,6 +16,7 @@ rtables::split_rows_by(colnames(result_raw)[2], split_fun = rtables::drop_split_levels, child_labels = "hidden") %>% rtables::analyze_colvars(function(x) x[seq_along(x)]) %>% rtables::build_table(result_raw) + main_title(result) <- paste("Patient ID:", NULL) result }