R/launch_shiny.R

Defines functions launch_shiny

Documented in launch_shiny

#' Launch Shiny application by clicking on its screenshot.
#'
#' Shiny applications can be embedded in certain R Markdown documents. However,
#' the application is automatically loaded at the same time as the main page,
#' and that may not be the desired behavior. With [launch_shiny()], you display
#' just a screenshot of the Shiny application in the page. The user has to click
#' on it to actually launch the application (that replaces the screenshot on the
#' page).
#'
#' @param url The URL of the Shiny application. If both `app =` and `baseurl =`
#' are provided, you don't need to specify it.
#' @param app Name of the shiny application (cannot be duplicated on a page).
#' @param imgdir The directory without trailing "/" where images relative
#' to Shiny applications are stored. By default, it is relative to current
#' directory, in `images/shinyapps` subdirectories.
#' @param img The relative or absolute path to the image with a screenshot of
#' the Shiny application, as produced by [webshot_shiny()].
#' @param createimg If the app image (`img`) is not found, and there is no
#' default image in `imgdir =`, do we put it there (yes be default)?
#' @param width The width of the image and iframe for the app.
#' @param height The height of image and iframe.
#' @param fun The function to run as alternative to start the Shiny application
#' locally. It is better to fully specify it (`package::function`), and it
#' should take one argument which is the application name in `app =`.
#' @param alt1 Alternate text to display at the bottom of the screenshot when
#' nothing is provided for `fun =`. If
#' `NULL`, nothing is displayed below the screenshot.
#' @param alt2 Alternate text to display at the bottom of the screenshot in case
#' `fun =` is provided.
#' @param toc Entry to use in the exercises table of content (`NULL` if no
#' entry, `""` for a default entry based on `toc.def =`).
#' @param toc.def Text for a default toc entry using [glue()] syntax for
#' replacement, e.g., `{app}`.
#' @param run.url The URL to use to start the Shiny application in RStudio
#' server in the SciViews Box. It should generally end with `?runrcode=`, and
#' the R code to execute will be appended to it from `run.arg =`.
#' @param run.cmd The command to use to launch the Shiny application in RStudio.
#' @param run.arg The URL encoded version of `run.cmd =`.
#' @param app.img The image to display in front of the toc entry
#' @param app.link The link when the image is clicked (sends to an help page
#' about Shiny applications).
#' @param ... Not used here, but it allows to add more arguments used by the
#' screenshot addin, like `delay =`, `offsetx =` or `offsety =`, see
#' [webshot_shiny()].
#'
#' @return The HTML content that creates the image and the iframe. The function
#' must be called from within an R inline expression or from an R chunk with
#' `results='asis'` in an HTML-rendered version of the R Markdown document to
#' get the correct result.
#'
#' @export
#'
#' @seealso [webshot_shiny()]
#' @examples
#' # TODO...
launch_shiny <- function(url, app = sub("\\?.+$", "", basename(url)),
imgdir = "images/shinyapps", img = paste0(imgdir, "/", app, ".png"),
createimg = TRUE, width = 790, height = 500, fun = NULL,
alt1 = "*Click to start the Shiny application.*",
alt2 = paste0("*Click to start",
  "or [run `{run.cmd}`]({run.url}{run.arg}){{target=\"_blank\"}}.*"),
toc = "", toc.def = "Shiny application {app}",
run.url = "start_rstudio.html?runrcode=", run.cmd = glue("{fun}(\"{app}\")"),
run.arg = URLencode(run.cmd, reserved = TRUE),
app.img = "images/list-app.png", app.link = "shiny_app", ...) {
  if (!is.null(toc)) {
    # Add an entry in the ex_toc
    ex_toc <- getOption("learnitdown_ex_toc", "")
    if (toc == "") {
      # Use default text
      toc <- glue::glue(toc.def)
    }
    ex_toc <- paste0(ex_toc, "\n",
      glue::glue("- [![app]({app.img})]({app.link}) [{toc}](#{app})"))
    options(learnitdown_ex_toc = ex_toc)
  }

  # Use alt2 if fun is provided, otherwise, use alt1
  if (!is.null(fun) && fun != "") {
    alt <- glue(alt2)
  } else {
    alt <- glue(alt1)
  }
  if (!length(alt))
    alt <- NULL

  # Check if the img exists, or fall back to a default image instead
  if (!file.exists(img)) {
    # If the default image does not exists, copy it from the package now
    img <- paste0(imgdir, "/shinyapp_default.png")
    if (!file.exists(img) & isTRUE(createimg)) {
      # Make sure imgdir directory exists
      dir.create(imgdir, showWarnings = FALSE, recursive = TRUE)
      # Copy the default image from the learnitdown package
      file.copy(from = system.file("images", "shinyapp_default.png",
        package = "learnitdown"), to = img)
    }
  }

  # If width and height are not explicitly provided, guess them from img size
  if (missing(width) || missing(height)) {
    info <- image_info(image_read(img))
    if (missing(width) && !is.null(info$width))
      width <- info$width
    if (missing(height) && !is.null(info$height))
      height <- info$height
  }

  # Construct the HTML code to be included
  res <- glue("\n[<img onclick=\"launchApp('{app}shiny', '{url}');\" src=\"{img}\" width=\"{width}\" height=\"{height}\" class=\"shiny-img\" id=\"img{app}shiny\"/>
<iframe width=\"{width}\" height=\"{height}\" frameborder=\"0\" scrolling=\"auto\" style=\"display:none\" class=\"shiny-app\" id=\"{app}shiny\"></iframe>]{{#{app} }}\n")
  if (!is.null(alt))
    res <- paste0(res, "\n", alt, "\n")
  res
}
SciViews/learndown documentation built on Sept. 17, 2024, 8:06 a.m.