class: center, middle, inverse, title-slide # R for Data Analysis ## RShiny ### Ayush Patel ### 05-Aug-2021 --- layout: true --- name: Introduction class: left,middle .pull-left[ ## Find me [__@ayushbipinpatel__](https://twitter.com/ayushbipinpatel) <img src="" width=5%> [__@AyushBipinPatel__](https://github.com/AyushBipinPatel) <img src="" width=5%> [__ayushpatel.netlify.app__](https://ayushpatel.netlify.app/) <img src="" width=5%> [__ayush.ap58@gmail.com__](ayush.ap58@gmail.com)<img src="" width=5%> ] .pull-right[ <img src = "https://images.metmuseum.org/CRDImages/ad/original/57258.jpg"> .small[ Image: [John Biglin in a Single Scull by Thomas Eakins](https://images.metmuseum.org/CRDImages/ad/original/57258.jpg) ] ] --- class: left, middle .pull-left[ # Pre-requisite .big[You....] understand __different types of objects, how to create objects and assign values to objects__. <br> __how to access specific values within an object.__ <br> know __what a function is and how to use a function.__ <br> know __basics of tidyverse__ <br> ] .pull-right[ <img src = "https://images.metmuseum.org/CRDImages/ad/original/DT84.jpg"> .small[ [Image: Lake George by John Frederick Kensett ](https://images.metmuseum.org/CRDImages/ad/original/DT84.jpg) ] ] --- # What is Shiny .left-column[ <img src="https://raw.githubusercontent.com/rstudio/hex-stickers/master/thumbs/shiny.png" width="250px" align = "center"> ] .right-coulmn[ .right-column[ .big[A R package] to help us: + Create .big[interactive web apps]<br> + .big[Without] requiring you to have deep knowledge of HTNL, CSS and JavaScript ] ] --- # What will you learn + The basics to User Interface<br><br> + Foundation of reactive programming<br><br> + Tidyevaluation<br><br> --- # A shiny app .pull-left[ ## What is it? A web page linked to a computer that is running a live R session. ] .pull-right[ ## Components + User Interface<br> + Server side code<br> ] --- # How are we proceeding? .big[Make a quick and trivial shiny app] .big[Learn about the basics of UI] .big[Learn about the basics of Reactive programming] .big[A little bit of tidyevaluation] --- # Creating a Shiny App There are multiple ways of doing this. We will focus on one. The simplest is to create a new directory for your app, and put a single file called `app.R` in it. This `app.R` file will be used to tell Shiny both how your app should look, and how it should behave. .big[OR] File> new file> Shiny web app ... > single file app> in new directory --- # The first shiny app .big[Copy this code in your app.R file and run the entire doc] ```r library(shiny) ui <- fluidPage( numericInput(inputId = "n", # What is the importance of inputID "Sample size", value = 25), plotOutput(outputId = "den") ) server <- function(input, output) { output$den <- renderPlot({ # Notice how the `$` sign is used plot(density(rnorm(input$n))) }) # Notice how the `$` sign is used } shinyApp(ui = ui, server = server) ``` --- # Changing the UI controls Could you change the code of the app to accept additional numeric value. Visit [https://www.rstudio.com/resources/cheatsheets/](https://www.rstudio.com/resources/cheatsheets/) and use the shiny cheatsheet --- # Adding Behaviour Could you use the following code to add a histogram to the web app, showing the same distribution as the density function (product of two inputs). Also change the density plot so that it shows the density of the product of two values. ```r hist(rnorm(input$)) ``` --- # Changing the UI Visit [https://www.rstudio.com/resources/cheatsheets/](https://www.rstudio.com/resources/cheatsheets/) and use the shiny cheatsheet .big[Could you refer to the layout section and change he UI to sidebarLayout] ```r ui <- fluidPage( sidebarLayout( sidebarPanel(), mainPanel() ) ) ``` The charts should appear in the main panel and the inputs should appear in the side panel --- # Question 1 ### Error object `x` not found ```r library(shiny) ui <- fluidPage( sliderInput("x", label = "If x is", min = 1, max = 50, value = 30), "then x times 5 is", textOutput("product") ) server <- function(input, output, session) { output$product <- renderText({ x * 5 }) } shinyApp(ui, server) ``` .big[What is generating the error] --- class: middle, center # You just made your first shiny app --- class: middle, center # Basics of UI --- # Inputs .left-column[ Functions that create UI components for users to input: + Text + Numbers + Dates and date ranges + Drop down section + and much more ] .right-column[ .big[Structure of these Input Functions] + First arg __inputID__: Connects UI to back end<br><br> + if inputID is "yellow", server function (back end) can access it by input$yellow<br><br> + Only letters, numbers and underscore<br><br> + Must be unique<br><br> + Second arg __label__: the text the user reads about the input, usually instructive ] --- # Various Inputs Refere to the cheat sheet and glance throught various inputs.<br> [https://www.rstudio.com/resources/cheatsheets/](https://www.rstudio.com/resources/cheatsheets/) --- # Outputs Outputs reserve space in the UI for the server function to fill later.<br> Like inputs the outputs also have ID. <br> In the server function these outputs can be accessed by output$plot, assuming the ID is "plot."<br> All output functions in the front end are coupled with a render function in the back end. --- # Various Outputs Refere to the cheat sheet and glance throught various outputs.<br> [https://www.rstudio.com/resources/cheatsheets/](https://www.rstudio.com/resources/cheatsheets/) --- class: center, middle # The Basics of reactivity --- # What is it? > "The key Idea of reactive programming is to specify a graph of dependencies so that when an input changes, all related outputs are automatically updated. The server logic is defined using reactive programming. --- # The server funciton .left-column[ The server function is invoked every time a new session starts, this why two different users are able to see and do different things. ] .right-column[ There are three parameters in the server function: + input<br><br> + output<br><br> + session <br> <br> We will look into input and output ] --- # Input .left-column[ "List-like" object<br><br> Contains user fed information<br><br> Can be accessed in the server function using `input$ID`<br><br> These are read only objects that can be read within reactive context only<br><br> ] .right-column[ ```r server <- function(input, output, session) { input$count <- 10 } shinyApp(ui, server) #> Error: Can't modify read-only reactive value 'count' ``` <br> ```r server <- function(input, output, session) { message("The value of input$count is ", input$count) } shinyApp(ui, server) #> Error: Can't access reactive value 'count' outside of reactive consumer. #> ℹ Do you need to wrap inside reactive() or observer()? ``` ] --- # Output .left-column[ .lilsmall[ "List-like" object<br><br> Named and accessed according to OutputID<br><br> Used for sending output to web browser<br><br> `output object` needs `render` functions <br><br> Render functions create reactive context to track which inputs are used by output<br><br> Render functions convert R code into HTML suitable for display on webpage ] ] .yscroll[ .right-column[ ```r ui <- fluidPage( textOutput("greeting") ) server <- function(input, output, session) { output$greeting <- renderText("Hello human!") } ``` ```r server <- function(input, output, session) { output$greeting <- "Hello human" } shinyApp(ui, server) #> Error: Unexpected character object for output$greeting #> ℹ Did you forget to use a render function? ``` ```r server <- function(input, output, session) { message("The greeting is ", output$greeting) } shinyApp(ui, server) #> Error: Reading from shinyoutput object is not allowed. ``` ] ] --- # Reactive Programming > No need to tell Shiny when to update, because Shiny automatically figures out for you. .big[Let us think about this code - Imperative vs Declarative] ```r output$greeting <- renderText({ paste0("Hello ", input$name, "!") }) ``` .big[`greeting` has a reactive dependency on `name`] --- # Activity Given this UI: ```r ui <- fluidPage( textInput("name", "What's your name?"), textOutput("greeting") ) ``` Fix the simple errors found in each of the three server functions below. First try spotting the problem just by reading the code; then run the code to make sure you’ve fixed it. ```r server1 <- function(input, output, server) { input$greeting <- renderText(paste0("Hello ", name)) } server2 <- function(input, output, server) { greeting <- paste0("Hello ", input$name) output$greeting <- renderText(greeting) } server3 <- function(input, output, server) { output$greting <- paste0("Hello", input$name) } ``` --- # Reactive expressions .pull-left[ ```r server <- function(input, output, session) { output$hist <- renderPlot({ x1 <- rnorm(input$n1, input$mean1, input$sd1) x2 <- rnorm(input$n2, input$mean2, input$sd2) freqpoly(x1, x2, binwidth = input$binwidth, xlim = input$range) }, res = 96) output$ttest <- renderText({ x1 <- rnorm(input$n1, input$mean1, input$sd1) x2 <- rnorm(input$n2, input$mean2, input$sd2) t_test(x1, x2) }) } ``` ] .pull-right[ ```r server <- function(input, output, session) { x1 <- reactive(rnorm(input$n1, input$mean1, input$sd1)) x2 <- reactive(rnorm(input$n2, input$mean2, input$sd2)) output$hist <- renderPlot({ freqpoly(x1(), x2(), binwidth = input$binwidth, xlim = input$range) }, res = 96) output$ttest <- renderText({ t_test(x1(), x2()) }) } ``` ] ??? Reduce code duplication reactive expressions are somewhere in between inputs and outputs --- # Tidyevaluation .big[Should the following function work??!] ```r library(tidyverse) library(palmerpenguins) plot_scatter <- function(df_data,var1,var2){ df_data %>% ggplot(aes(var1,var2))+ geom_jitter() } plot_scatter(penguins,"body_mass_g","bill_length_mm") ``` ??? An env-variable (environment variable) is a “programming” variables that you create with <-. input$var is a env-variable. A data-variable (data frame variables) is “statistical” variable that lives inside a data frame. carat is a data-variable. --- # Data masking ```r library(tidyverse) library(palmerpenguins) plot_scatter <- function(df_data,var1,var2){ df_data %>% ggplot(aes(.data[[var1]],.data[[var2]]))+ geom_jitter() } plot_scatter(penguins,"body_mass_g","bill_length_mm") ``` .big[check if this works] ??? .data vars .env vars --- # Activity .left-column[ Make a small shiny app that produces a plot(ggplot) by allowing a user to choose variables.<br><br> use the Iris data.<br><br> Try `?iris` in your console to get a sense of the data. ] .right-column[ Feel free to use this structure. ```r library(shiny) library(tidyverse) ui <- fluidPage( selectInput(), # Choice for var1 selectInput(), # Choice for var2 plotOutput("plot") ) server <- function(input, output, session) { output$plot <- renderPlot({ # ggplot code to generate chart }, res = 96) } ``` ] --- class: center, middle background-image: url("images/background2.jpg") background-size: cover