Shiny Source Code Explained: Rendering the UI

Ever wondered what goes on behind the shinyApp function from Shiny? There’s actually a lot happening to turn your UI and server into something that displays and works in the web browser! In this series, you will dive into the Shiny source code to better understand what happens when you click on the “Run App” button. Let’s start with the UI, or HTML, side of your Shiny application. 

The Shiny App Lifecycle

To explain what exactly happens when running your Shiny application, we take the following simple (and classic) Shiny application:

				
					library(shiny)

ui <- fluidPage(
  titlePanel("Basic Shiny App"),
  sidebarLayout(
    sidebarPanel(
      sliderInput("slider", "Slider", min = 1, max = 100, value = 50)
    ),
    mainPanel(
      plotOutput("plot")
    )
  )
)

server <- function(input, output) {
  output$plot <- renderPlot({
    hist(rnorm(input$slider))
  })
}

shinyApp(ui = ui, server = server)
				
			

Saving a file with the above code will give you the beloved and magic “Run App” button in RStudio. When you access a Shiny app in your browser, a series of events occurs. This series of event we call the Shiny app lifecycle.

Roughly, this lifecycle consists of the following steps (if you run your app locally):

1. You call the runApp() function by either hitting the “Run App” button in RStudio or calling runApp("path/to/app.R") directly in your console.

2. runApp() finds a Shiny app object in your file, generated by shinyApp(). This shinyApp() function contains ui and server objects, which are called appParts.

3. The appParts are given to a function called startApp(), which creates HTTP and WebSocket handlers. These handlers are responsible for controlling the WebSocket behaviour.

4. startApp() calls startServer() from the httpuv package, which starts the HTTP server and opens the server WebSocket connection.

5. If the R code is error-free, the HTTP server returns the given UI in HTML code.

6. The client, or web browser, receives that HTML code and interprets it.

7. The HTML is rendered and displayed in the web browser.

8. The web browser makes sure that the WebSocket connection is opened with the necessary JavaScript.

Since we’re focussing on the UI part of the application in this article, let’s take a closer look at this step:

5. If the R code is error-free, the HTTP server returns the given UI in HTML code.

Looking at the ui of our simple demo application, it returns the following HTML when printing it right on the R console:

				
					<div class="container-fluid">
  <h2>Basic Shiny App</h2>
  <div class="row">
    <div class="col-sm-4">
      <form class="well" role="complementary">
        <div class="form-group shiny-input-container">
          <label class="control-label" id="slider-label" for="slider">Slider</label>
          <input class="js-range-slider" id="slider" data-skin="shiny" data-min="1" data-max="100" data-from="50" data-step="1" data-grid="true" data-grid-num="9.9" data-grid-snap="false" data-prettify-separator="," data-prettify-enabled="true" data-keyboard="true" data-data-type="number"/>
        </div>
      </form>
    </div>
    <div class="col-sm-8" role="main">
      <div class="shiny-plot-output html-fill-item" id="plot" style="width:100%;height:400px;"></div>
    </div>
  </div>
</div>
				
			

However, this is not a valid HTML template yet. A valid HTML template contains tags like <!DOCTYPE html>, <html>, <head>, and <body>. The above only provides some simple <div> tags with a title and some inputs. 

If we would provide the above HTML as is to a browser, the browser will not understand. So the question is: what happens to create that valid HTML template?

Shiny and htmltools

Long story short: Shiny handles R to HTML conversion with the htmltools package. More specifically, via this call stack:

				
					shinyApp()
└── shiny:::uiHttpHandler()
    └── shiny:::renderPage()
        ├── tags$body(ui)
        ├── htmltools::htmlTemplate()
        └── htmltools::renderDocument()
				
			

Where tags$body(ui) wraps the ui in already one important HTML tag: <body>. Two functions of htmltools take care of the rest: htmltools::htmlTemplate() and htmltools::renderDocument()

The first one, htmlTemplate() fills a boilerplate that looks like this:

				
					<!DOCTYPE html>
<html{{ if (isTRUE(nzchar(lang))) paste0(" lang=\"", lang, "\"") }}>
<head>
{{ headContent() }}
</head>
{{ body }}
</html>
				
			

Applying that to our demo app we can run the following:

				
					htmlTemplate(
  system.file("template", "default.html", package = "shiny"), 
  lang = "en", 
  body = tags$body(ui), 
  document_ = TRUE
)
				
			

Which leads to the following result:

				
					<!DOCTYPE html>
<html lang="en">
<head>
<!-- HEAD_CONTENT -->
</head>
<body>
  <div class="container-fluid">
    <h2>Basic Shiny App</h2>
    <div class="row">
      <div class="col-sm-4">
        <form class="well" role="complementary">
          <div class="form-group shiny-input-container">
            <label class="control-label" id="slider-label" for="slider">Slider</label>
            <input class="js-range-slider" id="slider" data-skin="shiny" data-min="1" data-max="100" data-from="50" data-step="1" data-grid="true" data-grid-num="9.9" data-grid-snap="false" data-prettify-separator="," data-prettify-enabled="true" data-keyboard="true" data-data-type="number"/>
          </div>
        </form>
      </div>
      <div class="col-sm-8" role="main">
        <div class="shiny-plot-output html-fill-item" id="plot" style="width:100%;height:400px;"></div>
      </div>
    </div>
  </div>
</body>
</html>
				
			

As you can see, there’s a placeholder for the head content. After htmltools::htmlTemplate(), htmltools::renderDocument() gets called. This one takes care of replacing the head content. 

Shiny dependencies

Looking at the Shiny source code, we see that htmltools::renderDocument() gets called as the final step of shiny:::renderPage()

				
					html <- renderDocument(ui, shiny_deps, processDep = createWebDependency)
				
			

And it is here, that the Shiny dependencies get attached to the HTML. These dependencies are: 

				
					  shiny_deps <- c(
    list(jqueryDependency()),
    shinyDependencies()
  )
				
			

The first one, jqueryDependency(), speaks for itself: it takes care of the jQuery dependency. The second one, shinyDependencies(), attaches something called “shiny-busy-indicators”, which is a CSS file containing styles for outputs that are recalculating, and main JavaScript that is needed for Shiny: shiny.min.js.

If you put your app in test mode or in dev mode, additional dependencies will be attached as well.

The result of htmltools::renderDocument() is an object called html, and it gets encoded to UTF-8. This is eventually returned by shiny:::renderPage().

If all went well, Shiny sends of a 200 HTTP response (meaning success) to the web browser with the full and valid HTML document. In the browser, that eventually looks like this: 

				
					<!DOCTYPE html>
<html class="">
<head>
  <meta http-equiv="Content-Type" content="text/html; charset=utf-8">
  <script type="application/shiny-singletons"></script>
  <script type="application/html-dependencies">jquery[3.6.0];shiny-css[1.8.1.9001];shiny-busy-indicators[1.8.1.9001];shiny-javascript[1.8.1.9001];ionrangeslider-javascript[2.3.1];strftime[0.9.2];ionrangeslider-css[2.3.1];htmltools-fill[0.5.8.1];bootstrap[3.4.1]</script>
  <script src="jquery-3.6.0/jquery.min.js"></script>
  <link href="shiny-css-1.8.1.9001/shiny.min.css" rel="stylesheet">
  <link href="shiny-busy-indicators-1.8.1.9001/busy-indicators.css" rel="stylesheet">
  <script src="shiny-javascript-1.8.1.9001/shiny.min.js"></script>
  <script src="ionrangeslider-javascript-2.3.1/js/ion.rangeSlider.min.js"></script>
  <script src="strftime-0.9.2/strftime-min.js"></script>
  <link href="ionrangeslider-css-2.3.1/css/ion.rangeSlider.css" rel="stylesheet">
  <link href="htmltools-fill-0.5.8.1/fill.css" rel="stylesheet">
  <meta name="viewport" content="width=device-width, initial-scale=1">
  <link href="bootstrap-3.4.1/css/bootstrap.min.css" rel="stylesheet">
  <link href="bootstrap-3.4.1/accessibility/css/bootstrap-accessibility.min.css" rel="stylesheet">
  <script src="bootstrap-3.4.1/js/bootstrap.min.js"></script>
  <script src="bootstrap-3.4.1/accessibility/js/bootstrap-accessibility.min.js"></script>  <title>Basic Shiny App</title>
</head>
<body>
  <div class="container-fluid">
    <h2>Basic Shiny App</h2>
    <div class="row">
      <div class="col-sm-4">
        <form class="well" role="complementary">
          <div class="form-group shiny-input-container">
            <label class="control-label" id="slider-label" for="slider">Slider</label>
            <span class="irs irs--shiny js-irs-0 irs-with-grid"><span class="irs"><span class="irs-line" tabindex="0"></span><span class="irs-min" style="visibility: visible;">1</span><span class="irs-max" style="visibility: visible;">100</span><span class="irs-from" style="visibility: hidden;">0</span><span class="irs-to" style="visibility: hidden;">0</span><span class="irs-single" style="left: 47.304794%;">50</span></span><span class="irs-grid" style="width: 94.653793%; left: 2.573104%;"><span class="irs-grid-pol" style="left: 0%"></span><span class="irs-grid-text js-grid-text-0" style="left: 0%; margin-left: -1.446069%;">1</span><span class="irs-grid-pol small" style="left: 6.7340067340067336%"></span><span class="irs-grid-pol small" style="left: 3.3670033670033668%"></span><span class="irs-grid-pol" style="left: 10.1010101010101%"></span><span class="irs-grid-text js-grid-text-1" style="left: 10.10101%; visibility: visible; margin-left: -2.16128%;">11</span><span class="irs-grid-pol small" style="left: 16.835016835016834%"></span><span class="irs-grid-pol small" style="left: 13.468013468013467%"></span><span class="irs-grid-pol" style="left: 20.2020202020202%"></span><span class="irs-grid-text js-grid-text-2" style="left: 20.20202%; visibility: visible; margin-left: -2.16128%;">21</span><span class="irs-grid-pol small" style="left: 26.936026936026934%"></span><span class="irs-grid-pol small" style="left: 23.569023569023567%"></span><span class="irs-grid-pol" style="left: 30.3030303030303%"></span><span class="irs-grid-text js-grid-text-3" style="left: 30.30303%; visibility: visible; margin-left: -2.16128%;">31</span><span class="irs-grid-pol small" style="left: 37.03703703703704%"></span><span class="irs-grid-pol small" style="left: 33.67003367003367%"></span><span class="irs-grid-pol" style="left: 40.4040404040404%"></span><span class="irs-grid-text js-grid-text-4" style="left: 40.40404%; margin-left: -2.16128%;">41</span><span class="irs-grid-pol small" style="left: 47.138047138047135%"></span><span class="irs-grid-pol small" style="left: 43.77104377104377%"></span><span class="irs-grid-pol" style="left: 50.505050505050505%"></span><span class="irs-grid-text js-grid-text-5" style="left: 50.505051%; visibility: visible; margin-left: -2.16128%;">51</span><span class="irs-grid-pol small" style="left: 57.23905723905724%"></span><span class="irs-grid-pol small" style="left: 53.87205387205387%"></span><span class="irs-grid-pol" style="left: 60.6060606060606%"></span><span class="irs-grid-text js-grid-text-6" style="left: 60.606061%; visibility: visible; margin-left: -2.16128%;">61</span><span class="irs-grid-pol small" style="left: 67.34006734006734%"></span><span class="irs-grid-pol small" style="left: 63.973063973063965%"></span><span class="irs-grid-pol" style="left: 70.7070707070707%"></span><span class="irs-grid-text js-grid-text-7" style="left: 70.707071%; visibility: visible; margin-left: -2.16128%;">71</span><span class="irs-grid-pol small" style="left: 77.44107744107744%"></span><span class="irs-grid-pol small" style="left: 74.07407407407406%"></span><span class="irs-grid-pol" style="left: 80.8080808080808%"></span><span class="irs-grid-text js-grid-text-8" style="left: 80.808081%; margin-left: -2.16128%;">81</span><span class="irs-grid-pol small" style="left: 87.54208754208754%"></span><span class="irs-grid-pol small" style="left: 84.17508417508417%"></span><span class="irs-grid-pol" style="left: 90.9090909090909%"></span><span class="irs-grid-text js-grid-text-9" style="left: 90.909091%; visibility: visible; margin-left: -2.16128%;">91</span><span class="irs-grid-pol small" style="left: 96.96969696969697%"></span><span class="irs-grid-pol small" style="left: 93.93939393939394%"></span><span class="irs-grid-pol" style="left: 100%"></span><span class="irs-grid-text js-grid-text-10" style="left: 100%; visibility: visible; margin-left: -2.876492%;">100</span></span><span class="irs-bar irs-bar--single" style="left: 0px; width: 49.521951%;"></span><span class="irs-shadow shadow-single" style="display: none;"></span><span class="irs-handle single" style="left: 46.848847%;"><i></i><i></i><i></i></span></span><input class="js-range-slider irs-hidden-input shiny-bound-input" id="slider" data-skin="shiny" data-min="1" data-max="100" data-from="50" data-step="1" data-grid="true" data-grid-num="9.9" data-grid-snap="false" data-prettify-separator="," data-prettify-enabled="true" data-keyboard="true" data-data-type="number" tabindex="-1" readonly="">
          </div>
        </form>
      </div>
      <div class="col-sm-8" role="main">
        <div class="shiny-plot-output html-fill-item shiny-bound-output" id="plot" style="width:100%;height:400px;" aria-live="polite"><img decoding="async" src="" width="933.1249389648438" height="399.9999694824219" alt="Plot object"></div>
      </div>
    </div>
  </div>
</body></html>
				
			

Bootstrap and other dependencies

But wait a minute… Where are the Bootstrap dependencies coming from? Or things like “ionrangeslider”? These dependencies are actually attached by the functions in the ui itself. For example, taking a look at fluidPage(), we see that this function itself also contains some dependencies: 

				
					htmltools::renderTags(fluidPage())$dependencies 
				
			

Again jQuery, and Bootstrap:

				
					[[1]]
List of 9
 $ name      : chr "jquery"
 $ version   : chr "3.6.0"
 $ src       :List of 1
  ..$ file: chr "/shiny/inst/www/shared"
 $ meta      : NULL
 $ script    : chr "jquery.min.js"
 $ stylesheet: NULL
 $ head      : NULL
 $ attachment: NULL
 $ all_files : logi FALSE
 - attr(*, "class")= chr "html_dependency"

[[2]]
List of 9
 $ name      : chr "bootstrap"
 $ version   : chr "3.4.1"
 $ src       :List of 1
  ..$ file: chr "/shiny/inst/www/shared/bootstrap"
 $ meta      :List of 1
  ..$ viewport: chr "width=device-width, initial-scale=1"
 $ script    : chr [1:2] "js/bootstrap.min.js" "accessibility/js/bootstrap-accessibility.min.js"
 $ stylesheet: chr [1:2] "css/bootstrap.min.css" "accessibility/css/bootstrap-accessibility.min.css"
 $ head      : NULL
 $ attachment: NULL
 $ all_files : logi TRUE
 - attr(*, "class")= chr "html_dependency"

				
			

The fact that these dependencies also end up in the <head> tag is because htmltools::renderDocument() also processes the dependencies that are already present in the ui. The shiny_deps object are extra dependencies added on top of the ones found in any of your ui elements. 

And what about “ionrangeslider”? That is coming from the sliderInput():

				
					htmltools::renderTags(sliderInput("slider", "Slider", min = 1, max = 100, value = 50))$dependencies 
				
			
				
					[[1]]
List of 9
 $ name      : chr "ionrangeslider-javascript"
 $ version   : chr "2.3.1"
 $ src       :List of 1
  ..$ file: chr "/shiny/inst/www/shared/ionrangeslider"
 $ meta      : NULL
 $ script    : chr "js/ion.rangeSlider.min.js"
 $ stylesheet: NULL
 $ head      : NULL
 $ attachment: NULL
 $ all_files : logi TRUE
 - attr(*, "class")= chr "html_dependency"

[[2]]
List of 9
 $ name      : chr "strftime"
 $ version   : chr "0.9.2"
 $ src       :List of 1
  ..$ file: chr "/shiny/inst/www/shared/strftime"
 $ meta      : NULL
 $ script    : chr "strftime-min.js"
 $ stylesheet: NULL
 $ head      : NULL
 $ attachment: NULL
 $ all_files : logi TRUE
 - attr(*, "class")= chr "html_dependency"

[[3]]
List of 9
 $ name      : chr "ionrangeslider-css"
 $ version   : chr "2.3.1"
 $ src       :List of 1
  ..$ file: chr "/shiny/inst/www/shared/ionrangeslider"
 $ meta      : NULL
 $ script    : NULL
 $ stylesheet: chr "css/ion.rangeSlider.css"
 $ head      : NULL
 $ attachment: NULL
 $ all_files : logi TRUE
 - attr(*, "class")= chr "html_dependency"
				
			

And that’s how it’s done!

Of course this doesn’t cover the server part- we’ll have to leave that to the next part of this series.

Resources

I cover this topic extensively in one of my courses: CustomiZing WidgetZ. It’s an advanced Shiny course where you will learn to build your own Shiny packages. And to be able to do that, you need to understand Shiny on a deeper level! 

The course is based on the brilliant book of David Granjon: Outstanding User Interfaces with Shiny. Specifically, the chapter about web application concepts was used to explain the Shiny App Lifecycle. I definitely recommend diving into this chapter (and whole book really!) to learn more about Shiny. The book is available online for free, but if you like a hard copy and want to support David in the process, you can purchase it on (amongst others) Amazon.

Wrap up

Hopefully this article uncovered some of the Shiny magic!

☕️ Was this useful to you and would you like to support me? You can buy me a coffee!

I provide R and Shiny consultancy. Do you need help with a project? Reach out and let’s have a chat!

Leave a Reply

Your email address will not be published. Required fields are marked *