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

init simple report #23

Merged
merged 51 commits into from
Apr 29, 2022
Merged
Changes from 1 commit
Commits
Show all changes
51 commits
Select commit Hold shift + click to select a range
deb145a
init simple report
Polkas Apr 20, 2022
d81155c
add first testServer
Polkas Apr 21, 2022
b6068c8
testServer
Polkas Apr 21, 2022
6f7c05f
styler + lintr
Polkas Apr 21, 2022
65a4a50
utils
Polkas Apr 21, 2022
b5b7886
Merge branch 'main' into 1_simple_app@main
Polkas Apr 21, 2022
88747e3
docs
Polkas Apr 21, 2022
2973923
docs and tests
Polkas Apr 21, 2022
5765dfa
tests
Polkas Apr 21, 2022
856aa1f
extract input
Polkas Apr 22, 2022
67c726f
extract input
Polkas Apr 22, 2022
04f4146
polish
Polkas Apr 22, 2022
f52f090
lintr actions
Polkas Apr 22, 2022
b49cf1c
spelling user interface
Polkas Apr 22, 2022
bc9e614
docs
Polkas Apr 22, 2022
a4f1e25
fix
Polkas Apr 26, 2022
1587e43
fix
Polkas Apr 26, 2022
fed131b
Mahmoud rev
Polkas Apr 26, 2022
b1cd6a7
Mahmoud rev 2
Polkas Apr 26, 2022
2dbf120
Mahmoud rev 2b
Polkas Apr 26, 2022
ba29264
Update R/utils.R
Polkas Apr 26, 2022
b3d132c
Mahmoud rev 2c
Polkas Apr 26, 2022
6dfa710
small update
Polkas Apr 28, 2022
60c80eb
Apply suggestions from code review
Polkas Apr 28, 2022
af9e25b
buttons specified by user (#28)
gogonzo Apr 28, 2022
d7f4f97
docs
Polkas Apr 28, 2022
934e7f5
style
Polkas Apr 28, 2022
d3013ba
revert update
Polkas Apr 28, 2022
2157e0b
Revert "docs"
Polkas Apr 28, 2022
30954da
Revert "buttons specified by user (#28)"
Polkas Apr 28, 2022
073c0f2
callback
Polkas Apr 28, 2022
553bdb3
Apply suggestions from code review
Polkas Apr 28, 2022
0f95dfd
comments
Polkas Apr 28, 2022
6ebb603
style
Polkas Apr 28, 2022
7d96517
docs
Polkas Apr 28, 2022
2951746
docs
Polkas Apr 28, 2022
bfbae65
docs
Polkas Apr 28, 2022
1e23c5d
download fun
Polkas Apr 28, 2022
8746c1a
docs
Polkas Apr 28, 2022
994a86f
tests
Polkas Apr 28, 2022
a05ccef
call back magic, comment
Polkas Apr 28, 2022
eabeb33
no pdf render
Polkas Apr 29, 2022
34200dc
lintr
Polkas Apr 29, 2022
f485b3e
even better
Polkas Apr 29, 2022
be355dd
validate
Polkas Apr 29, 2022
9268853
Merge branch 'main' into 1_simple_app@main
Polkas Apr 29, 2022
a165a36
Apply suggestions from code review
Polkas Apr 29, 2022
61f77cb
Apply suggestions from code review
Polkas Apr 29, 2022
867c673
updates
Polkas Apr 29, 2022
293887f
empty comment
Polkas Apr 29, 2022
a7702f3
docs
Polkas Apr 29, 2022
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
Next Next commit
init simple report
Polkas committed Apr 20, 2022
commit deb145ad3c0802eca6c3bd85fe401faf6432a050
5 changes: 4 additions & 1 deletion DESCRIPTION
Original file line number Diff line number Diff line change
@@ -11,7 +11,10 @@ BugReports: https://github.com/insightsengineering/teal.reporter/issues
Imports:
checkmate,
R6,
yaml
yaml,
shiny,
shinyWidgets,
zip
Suggests:
ggplot2,
knitr,
7 changes: 7 additions & 0 deletions NAMESPACE
Original file line number Diff line number Diff line change
@@ -2,6 +2,13 @@

export(ReportCard)
export(Reporter)
export(add_card_button_srv)
export(add_card_button_ui)
export(download_report_button_srv)
export(download_report_button_ui)
export(extract_addcard_id)
export(simple_reporter)
export(simple_reporter_ui)
importFrom(R6,R6Class)
importFrom(checkmate,assert_string)
importFrom(yaml,as.yaml)
83 changes: 83 additions & 0 deletions R/AddCardModule.R
Original file line number Diff line number Diff line change
@@ -0,0 +1,83 @@
#' Add Card Reporter UI
#' @description button for adding views/cards to the Report. Part of the simple Reporter UI.
#' @param id character
Polkas marked this conversation as resolved.
Show resolved Hide resolved
#' @return shiny `tagList`
Polkas marked this conversation as resolved.
Show resolved Hide resolved
#' @export
add_card_button_ui <- function(id) {
ns <- shiny::NS(id)
shiny::tagList(
shiny::tags$button(
id = ns("addReportCard"),
type = "button",
class = "btn btn-primary action-button",
`data-val` = shiny::restoreInput(id = ns("addReportCard"), default = NULL),
NULL,
"Add Card"
)
)
}

Polkas marked this conversation as resolved.
Show resolved Hide resolved
#' Add Card Button Server
#' @description server for adding views/cards the Report. Part of the simple Reporter.
Polkas marked this conversation as resolved.
Show resolved Hide resolved
#' @param id character
#' @param reporter `Reporter` instance.
#' @param card `ReportCard` instance
#' @return shiny `moduleServer`
#' @export
#' @export
Polkas marked this conversation as resolved.
Show resolved Hide resolved
add_card_button_srv <- function(id, reporter, card) {
shiny::moduleServer(
id,
function(input, output, session) {
ns <- session$ns
add_modal <- function(failed = FALSE) {
shiny::modalDialog(
easyClose = TRUE,
shiny::tags$h3("Add the Card to the Report"),
shiny::tags$hr(),
shiny::textInput(
ns("comment"),
"Comment",
value = "The idea behind",
width = "100%"
),
if (failed) {
shiny::tags$div(
shiny::tags$b("Invalid", style = "color: red;")
)
},
footer = shiny::tagList(
shiny::tags$button(
type = "button",
class = "btn btn-primary",
`data-dismiss` = "modal",
`data-bs-dismiss` = "modal",
NULL,
"Cancel"
),
shiny::tags$button(
id = ns("addCardOk"),
Polkas marked this conversation as resolved.
Show resolved Hide resolved
type = "button",
class = "btn btn-primary action-button",
`data-val` = shiny::restoreInput(id = ns("addCardOk"), default = NULL),
NULL,
"Add Card"
)
)
)
}

shiny::observeEvent(input$addReportCard, {
shiny::showModal(add_modal())
})

shiny::observeEvent(input$addCardOk, {
Polkas marked this conversation as resolved.
Show resolved Hide resolved
stopifnot(inherits(card(), "ReportCard"))
Polkas marked this conversation as resolved.
Show resolved Hide resolved
card()$append_text("Comment", "header3")
card()$append_text(input$comment)
reporter$append_cards(list(card()))
shiny::removeModal()
})
}
)
}
146 changes: 146 additions & 0 deletions R/DownloadModule.R
Original file line number Diff line number Diff line change
@@ -0,0 +1,146 @@
#' Download Button Reporter UI
#' @description button for downloading the Report. Part of the simple Reporter UI.
#' @param id character
#' @return shiny `tagList`
Polkas marked this conversation as resolved.
Show resolved Hide resolved
#' @export
download_report_button_ui <- function(id) {
ns <- shiny::NS(id)
shiny::tagList(
shiny::tags$button(
id = ns("download_button"),
type = "button",
class = "btn btn-primary action-button",
`data-val` = shiny::restoreInput(id = ns("download_button"), default = NULL),
NULL,
"Download Report"
)
)
}

#' Download Button Server
#' @description server for downloading the Report. Part of the simple Reporter.
Polkas marked this conversation as resolved.
Show resolved Hide resolved
#' @param id character
Polkas marked this conversation as resolved.
Show resolved Hide resolved
#' @param reporter `Reporter` instance.
#' @param notification logical if to add shiny notification about the download process.
#' @return shiny `moduleServer`
#' @export
download_report_button_srv <- function(id, reporter, notification = TRUE) {
shiny::moduleServer(
id,
function(input, output, session) {
ns <- session$ns
download_modal <- function(failed = FALSE) {
nr_cards <- length(reporter$get_cards())
downb <- shiny::tags$a(
id = ns("download_data"),
class = paste("btn btn-primary shiny-download-link", if (nr_cards) NULL else "disabled"),
style = if (nr_cards) NULL else "pointer-events: none;",
href = "",
target = "_blank",
download = NA,
shiny::icon("download"),
"Download Report"
)
shiny::modalDialog(
easyClose = TRUE,
shiny::tags$h3("Download the Report"),
shiny::tags$hr(),
if (length(reporter$get_cards()) == 0) {
shiny::tags$div(
shiny::tags$p(shiny::tags$strong("No Cards Added"), style = "color: red; margin-bottom:15px;")
)
} else {
shiny::tags$div(
style = "color: green; margin-bottom:15px;",
shiny::tags$p(
shiny::tags$strong(paste("Number of cards: ", nr_cards))
),
)
},
shiny::textInput(ns("docAuthor"), label = "Author:", value = "NEST"),
shiny::textInput(ns("docTitle"), label = "Title:", value = "NEST Report"),
shiny::tags$div(
shinyWidgets::pickerInput(
inputId = ns("docType"),
label = "Choose a document type: ",
choices = c("pdf document", "html document", "powerpoint presentation", "word document")
)
),
if (failed) {
shiny::tags$div(shiny::tags$b("Invalid", style = "color: red;"))
},
footer = shiny::tagList(
shiny::tags$button(
id = ns("reset_reporter"),
type = "button",
style = "float: left;",
class = "btn btn-danger action-button",
`data-val` = shiny::restoreInput(id = ns("reset_reporter"), default = NULL),
NULL,
"Reset Reporter"
),
shiny::tags$button(
type = "button",
class = "btn btn-primary",
`data-dismiss` = "modal",
`data-bs-dismiss` = "modal",
NULL,
"Cancel"
),
downb
)
)
}

shiny::observeEvent(input$download_button, {
shiny::showModal(download_modal())
})

shiny::observeEvent(input$reset_reporter, {
shiny::showModal(
shiny::modalDialog(
shiny::tags$h3("Reset the Report"),
shiny::tags$hr(),
shiny::tags$strong(shiny::tags$p("Are you sure you want to reset the report?")),
footer = shiny::tagList(
shiny::modalButton("Cancel"),
shiny::actionButton(ns("reset_reporter_ok"), "Reset")
)
)
)
})
Polkas marked this conversation as resolved.
Show resolved Hide resolved

shiny::observeEvent(input$reset_reporter_ok, {
reporter$reset()
shiny::removeModal()
})

output$download_data <- shiny::downloadHandler(
filename = function() {
paste("Report_", format(Sys.time(), "%y%m%d%H%M%S"), ".zip", sep = "")
},
content = function(file) {
renderer <- Renderer$new()

yaml <- list(
author = yaml_quoted(input$docAuthor),
title = yaml_quoted(input$docTitle),
date = yaml_quoted(as.character(Sys.Date()))
)

yaml[["output"]] <- gsub(" ", "_", input$docType)

yaml_header <- md_header(yaml::as.yaml(yaml))

if (notification) shiny::showNotification(sprintf("Rendering and Downloading\n%s.", input$docType))

renderer$render(reporter$get_blocks(), yaml_header)
temp_zip_file <- tempfile(fileext = ".zip")
zip::zipr(temp_zip_file, renderer$get_output_dir())
file.copy(temp_zip_file, file)
Polkas marked this conversation as resolved.
Show resolved Hide resolved
},
contentType = "application/zip"
)
}
)
}
8 changes: 8 additions & 0 deletions R/Reporter.R
Original file line number Diff line number Diff line change
@@ -107,6 +107,14 @@ Reporter <- R6::R6Class( # nolint: object_name_linter.
}
blocks
},
#' @description Reset the instance, remove already added cards.
Polkas marked this conversation as resolved.
Show resolved Hide resolved
#'
#' @return a `Reporter` object
Polkas marked this conversation as resolved.
Show resolved Hide resolved
#'
reset = function() {
private$cards <- list()
invisible(self)
},
#' @description The copy constructor.
#'
#' @param name the name of the field
30 changes: 30 additions & 0 deletions R/SimpleReporter.R
Original file line number Diff line number Diff line change
@@ -0,0 +1,30 @@
#' Simple Reporter UI
#' @description two buttons for adding views and downloading the Report
#' @param id character
#' @return shiny `tagList`
Polkas marked this conversation as resolved.
Show resolved Hide resolved
#' @export
simple_reporter_ui <- function(id) {
ns <- shiny::NS(id)
shiny::tagList(
add_card_button_ui(ns("addReportCard")),
download_report_button_ui(ns("downloadButton")),
)
}

#' Simple Reporter Server
#' @description two buttons for adding views and downloading the Report
#' @param id character
#' @param reporter `Reporter` instance.
#' @param card `ReportCard` instance
#' @param notification logical if to add shiny notification about the download process.
#' @return shiny `moduleServer`
Polkas marked this conversation as resolved.
Show resolved Hide resolved
#' @export
simple_reporter <- function(id, reporter, card, notification = TRUE) {
shiny::moduleServer(
id,
function(input, output, session) {
add_card_button_srv("addReportCard", reporter = reporter, card = card)
download_report_button_srv("downloadButton", reporter = reporter, notification = notification)
}
)
}
16 changes: 16 additions & 0 deletions R/utils.R
Original file line number Diff line number Diff line change
@@ -0,0 +1,16 @@
#' Extract Add Card Button id
#' @description extract Add Card Button id.
#' It is needed to know when trigger the reactivity cycle for the `ReportCard`.
#' @param input shiny input
#' @note has to be invoked inside the reactive call.
#' @export
extract_addcard_id <- function(input) {
nams <- names(input)
which_addcard <- grep("addReportCard-addReportCard$", nams)
res <- nams[which_addcard]
if (length(res) == 1) {
res
} else {
"NULL"
}
}
14 changes: 14 additions & 0 deletions man/Reporter.Rd

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

21 changes: 21 additions & 0 deletions man/add_card_button_srv.Rd

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

17 changes: 17 additions & 0 deletions man/add_card_button_ui.Rd

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

21 changes: 21 additions & 0 deletions man/download_report_button_srv.Rd
17 changes: 17 additions & 0 deletions man/download_report_button_ui.Rd
18 changes: 18 additions & 0 deletions man/extract_addcard_id.Rd
23 changes: 23 additions & 0 deletions man/simple_reporter.Rd
17 changes: 17 additions & 0 deletions man/simple_reporter_ui.Rd
2 changes: 2 additions & 0 deletions vignettes/.gitignore
Original file line number Diff line number Diff line change
@@ -0,0 +1,2 @@
*.html
*.R
204 changes: 204 additions & 0 deletions vignettes/simpleReporter.Rmd
Original file line number Diff line number Diff line change
@@ -0,0 +1,204 @@
---
title: "Simple Reporter"
output: rmarkdown::html_vignette
vignette: >
%\VignetteIndexEntry{Simple Reporter}
%\VignetteEngine{knitr::rmarkdown}
%\VignetteEncoding{UTF-8}
---

```{r, include = FALSE}
knitr::opts_chunk$set(
collapse = TRUE,
comment = "#>"
)
```

```{r setup}
library(shiny)
library(teal.reporter)
library(ggplot2)
library(rtables)
```

AddCardButton and downloadReportButton modules:

```{r}
ui <- fluidPage(
titlePanel(""),
sidebarLayout(
sidebarPanel(
uiOutput("encoding")
),
mainPanel(
### REPORTER
teal.reporter::add_card_button_ui("addReportCard"),
teal.reporter::download_report_button_ui("downloadButton"),
###
tags$br(),
tags$br(),
tabsetPanel(
id = "tabs",
tabPanel("Plot", plotOutput("distPlot")),
tabPanel("Table", verbatimTextOutput("table"))
)
)
)
)
server <- function(input, output, session) {
output$encoding <- renderUI({
if (input$tabs == "Plot") {
sliderInput(
"binwidth",
"binwidth",
min = 2,
max = 10,
value = 8
)
} else {
selectInput(
"stat",
label = "Statistic",
choices = c("mean", "median", "sd"),
"mean"
)
}
})
plot <- reactive({
req(input$binwidth)
x <- mtcars$mpg
ggplot2::ggplot(data = mtcars, ggplot2::aes(x = mpg)) +
ggplot2::geom_histogram(binwidth = input$binwidth)
})
output$distPlot <- renderPlot({
plot()
})
table <- reactive({
req(input$stat)
lyt <- basic_table() %>%
split_rows_by("Month", label_pos = "visible") %>%
analyze("Ozone", afun = eval(str2expression(input$stat)))
build_table(lyt, airquality)
})
output$table <- renderPrint({
table()
})
### REPORTER
reporter <- teal.reporter::Reporter$new()
card_r <- eventReactive(
input[[teal.reporter::extract_addcard_id(input)]], {
card <- teal.reporter::ReportCard$new()
if (input$tabs == "Plot") {
card$append_text("My plot", "header2")
card$append_plot(plot())
} else if (input$tabs == "Table") {
card$append_text("My Table", "header2")
card$append_table(table())
}
card
})
teal.reporter::add_card_button_srv("addReportCard", reporter = reporter, card = card_r)
teal.reporter::download_report_button_srv("downloadButton", reporter = reporter)
}
shinyApp(ui = ui, server = server)
```

Simple Reporter:

```{r}
ui <- fluidPage(
titlePanel(""),
sidebarLayout(
sidebarPanel(
uiOutput("encoding")
),
mainPanel(
### REPORTER
teal.reporter::simple_reporter_ui("simpleReporter"),
###
tags$br(),
tags$br(),
tabsetPanel(
id = "tabs",
tabPanel("Plot", plotOutput("distPlot")),
tabPanel("Table", verbatimTextOutput("table"))
)
)
)
)
server <- function(input, output, session) {
output$encoding <- renderUI({
if (input$tabs == "Plot") {
sliderInput(
"binwidth",
"binwidth",
min = 2,
max = 10,
value = 8
)
} else {
selectInput(
"stat",
label = "Statistic",
choices = c("mean", "median", "sd"),
"mean"
)
}
})
plot <- reactive({
req(input$binwidth)
x <- mtcars$mpg
ggplot2::ggplot(data = mtcars, ggplot2::aes(x = mpg)) +
ggplot2::geom_histogram(binwidth = input$binwidth)
})
output$distPlot <- renderPlot({
plot()
})
table <- reactive({
req(input$stat)
lyt <- basic_table() %>%
split_rows_by("Month", label_pos = "visible") %>%
analyze("Ozone", afun = eval(str2expression(input$stat)))
build_table(lyt, airquality)
})
output$table <- renderPrint({
table()
})
### REPORTER
reporter <- teal.reporter::Reporter$new()
card_r <- eventReactive(
input[[teal.reporter::extract_addcard_id(input)]], {
card <- teal.reporter::ReportCard$new()
if (input$tabs == "Plot") {
card$append_text("My plot", "header2")
card$append_plot(plot())
} else if (input$tabs == "Table") {
card$append_text("My Table", "header2")
card$append_table(table())
}
card
})
teal.reporter::simple_reporter("simpleReporter", reporter = reporter, card = card_r)
###
}
shinyApp(ui = ui, server = server)
```