Interactive Documents


Adding interactivity to an article is a great way to help readers explore the concepts and data you are presenting more deeply. There are three ways to add interactive components to Quarto documents:

  1. Create custom JavaScript visualizations using Observable JS.

  2. Use the Shiny R package to add interactivity to Knitr engine documents.

  3. Incorporate Jupyter Widgets or htmlwidgets (for the Jupyter and Knitr engines, respectively) into your document.

Each of these techniques has distinct benefits and drawbacks in terms of expressiveness, ease of development, and deployment requirements. We’ll touch on these considerations briefly below, then provide links to more in depth documentation for learning more.

Observable JS

Quarto includes native support for Observable JS, a set of enhancements to vanilla JavaScript created by Mike Bostock (also the author of D3). Observable JS is distinguished by its reactive runtime, which is especially well suited for interactive data exploration and analysis.

Here’s an example that provides slider inputs to condition the behavior of a visualization:

Observable JS uses some special keywords and a custom runtime to make JavaScript reactive. For example, the “minimum fame” slider in the example above was created with the following code:

viewof minimum = Inputs.range([-2, 2], { 
  value: 1, step: 0.01, 
  label: "minimum fame"

It’s then referenced as a normal JavaScript variable in code that creates the plot:

sel.filter(d => d.fame <= minimum)

As the user interacts with the slider, the minimum value is updated and any code that references it is automatically re-executed.

One benefit of using JavaScript for interactive documents is that all the logic and computation is performed on the client (so no server is required for deployment).

To learn more see the articles on Observable JS.


The Shiny package provides a flexible, easy to use framework for creating interactive web applications with R. Quarto in turn includes support for embedding Shiny components and applets into documents created with the Knitr engine.

Here’s a live example of Shiny interactive components along with a brief explanation of the code required to create them:

Shiny comes with a variety of built in input widgets. With minimal syntax it is possible to include widgets like the ones shown on the left in your apps:

# Select type of trend to plot
selectInput(inputId = "type", 
            label = strong("Trend index"),
            choices = unique(trend_data$type),
            selected = "Travel")

Displaying outputs is equally hassle-free:

  plotOutput(outputId = "lineplot", 
             height = "300px"),

Build your plots or tables as you normally would in R, and make them reactive with a call to the appropriate render function:

output$lineplot <- renderPlot({
  plot(x = selected_trends()$date, 
       y = selected_trends()$close, 
       type = "l",
       xlab = "Date", 
       ylab = "Trend index")

Shiny makes it very straightforward to create interactive documents using only R. Unlike using JavaScript though, you will need to deploy documents that use Shiny to a server.

To learn more see the articles on Using Shiny with Quarto.


Jupyter Widgets and htmlwidgets are great ways to incorporate interactivity into your documents if you don’t know JavaScript and prefer to work exclusively in Python or R. They also run entirely client-side so can be deployed within normal static HTML documents.

For example, the following Python code is all that is required to embed a Leaflet map into a Jupyter engine document:

from ipyleaflet import Map, Marker, basemaps, basemap_to_tiles
m = Map(
    basemaps.NASAGIBS.ModisTerraTrueColorCR, "2017-04-08"
  center=(52.204793, 360.121558),
m.add_layer(Marker(location=(52.204793, 360.121558)))

To learn more see these articles on using widgets with Quarto:


Once you’ve gotten familiar with using various interactive components see the article on Component Layout to learn how to: