blob: 84df1dccdab3167e71928a1fd06f37fcc325b800 [file] [log] [blame]
JJ Allaire2ec40242014-09-15 09:18:39 -04001#' Convert to a reveal.js presentation
JJ Allaire4c178052016-01-30 19:35:39 -05002#'
JJ Allaire2ec40242014-09-15 09:18:39 -04003#' Format for converting from R Markdown to a reveal.js presentation.
JJ Allaire4c178052016-01-30 19:35:39 -05004#'
JJ Allaire2ec40242014-09-15 09:18:39 -04005#' @inheritParams rmarkdown::beamer_presentation
6#' @inheritParams rmarkdown::pdf_document
7#' @inheritParams rmarkdown::html_document
JJ Allaire4c178052016-01-30 19:35:39 -05008#'
Christophe Dervieux21239cf2021-09-15 15:34:01 +02009#' @param smart Produce typographically correct output, converting straight
10#' quotes to curly quotes, \code{---} to em-dashes, \code{--} to en-dashes, and
11#' \code{...} to ellipses.
JJ Allaire2ec40242014-09-15 09:18:39 -040012#' @param center \code{TRUE} to vertically center content on slides
JJ Allaire35c0b492017-02-10 09:30:24 -050013#' @param slide_level Level of heading to denote individual slides. If
14#' \code{slide_level} is 2 (the default), a two-dimensional layout will be
15#' produced, with level 1 headers building horizontally and level 2 headers
16#' building vertically. It is not recommended that you use deeper nesting of
JJ Allaire4c178052016-01-30 19:35:39 -050017#' section levels with reveal.js.
JJ Allaire35c0b492017-02-10 09:30:24 -050018#' @param theme Visual theme ("simple", "sky", "beige", "serif", "solarized",
JJ Allaire4c178052016-01-30 19:35:39 -050019#' "blood", "moon", "night", "black", "league" or "white").
junkkad4b3a162015-03-16 07:49:11 +010020#' @param transition Slide transition ("default", "none", "fade", "slide",
21#' "convex", "concave" or "zoom")
JJ Allaire35c0b492017-02-10 09:30:24 -050022#' @param background_transition Slide background-transition ("default", "none",
JJ Allaire4c178052016-01-30 19:35:39 -050023#' "fade", "slide", "convex", "concave" or "zoom")
24#' @param reveal_options Additional options to specify for reveal.js (see
Atsushi Yasumoto66c55852020-08-10 21:50:04 +090025#' \href{https://revealjs.com/config/}{https://revealjs.com/config/}
JJ Allaire35c0b492017-02-10 09:30:24 -050026#' for details).
27#' @param reveal_plugins Reveal plugins to include. Available plugins include
JJ Allaire064552c2017-02-10 10:28:41 -050028#' "notes", "search", "zoom", "chalkboard", and "menu". Note that
JJ Allaire35c0b492017-02-10 09:30:24 -050029#' \code{self_contained} must be set to \code{FALSE} in order to use Reveal
30#' plugins.
31#' @param template Pandoc template to use for rendering. Pass "default" to use
32#' the rmarkdown package default template; pass \code{NULL} to use pandoc's
33#' built-in template; pass a path to use a custom template that you've
34#' created. Note that if you don't use the "default" template then some
35#' features of \code{revealjs_presentation} won't be available (see the
JJ Allaire4c178052016-01-30 19:35:39 -050036#' Templates section below for more details).
JJ Allaire35c0b492017-02-10 09:30:24 -050037#' @param extra_dependencies Additional function arguments to pass to the base R
38#' Markdown HTML output formatter [rmarkdown::html_document_base()].
JJ Allaire8d1c2f42016-01-30 14:56:45 -050039#' @param ... Ignored
JJ Allaire4c178052016-01-30 19:35:39 -050040#'
JJ Allaire2ec40242014-09-15 09:18:39 -040041#' @return R Markdown output format to pass to \code{\link{render}}
JJ Allaire4c178052016-01-30 19:35:39 -050042#'
JJ Allaire2ec40242014-09-15 09:18:39 -040043#' @details
JJ Allaire309bc7d2016-01-30 19:17:54 -050044#'
JJ Allaire4c178052016-01-30 19:35:39 -050045#' In reveal.js presentations you can use level 1 or level 2 headers for slides.
46#' If you use a mix of level 1 and level 2 headers then a two-dimensional layout
JJ Allaire35c0b492017-02-10 09:30:24 -050047#' will be produced, with level 1 headers building horizontally and level 2
JJ Allaire4c178052016-01-30 19:35:39 -050048#' headers building vertically.
49#'
JJ Allaire35c0b492017-02-10 09:30:24 -050050#' For additional documentation on using revealjs presentations see
JJ Allaire4c178052016-01-30 19:35:39 -050051#' \href{https://github.com/rstudio/revealjs}{https://github.com/rstudio/revealjs}.
JJ Allaire35c0b492017-02-10 09:30:24 -050052#'
JJ Allaire2ec40242014-09-15 09:18:39 -040053#' @examples
54#' \dontrun{
JJ Allaire4c178052016-01-30 19:35:39 -050055#'
JJ Allaire2ec40242014-09-15 09:18:39 -040056#' library(rmarkdown)
57#' library(revealjs)
JJ Allaire4c178052016-01-30 19:35:39 -050058#'
JJ Allaire2ec40242014-09-15 09:18:39 -040059#' # simple invocation
60#' render("pres.Rmd", revealjs_presentation())
JJ Allaire4c178052016-01-30 19:35:39 -050061#'
JJ Allaire2ec40242014-09-15 09:18:39 -040062#' # specify an option for incremental rendering
63#' render("pres.Rmd", revealjs_presentation(incremental = TRUE))
64#' }
JJ Allaire4c178052016-01-30 19:35:39 -050065#'
66#'
JJ Allaire2ec40242014-09-15 09:18:39 -040067#' @export
68revealjs_presentation <- function(incremental = FALSE,
69 center = FALSE,
JJ Allaire4c178052016-01-30 19:35:39 -050070 slide_level = 2,
JJ Allaire2ec40242014-09-15 09:18:39 -040071 fig_width = 8,
72 fig_height = 6,
73 fig_retina = if (!fig_caption) 2,
74 fig_caption = FALSE,
75 smart = TRUE,
76 self_contained = TRUE,
JJ Allaire6da1bb62016-01-30 14:28:39 -050077 theme = "simple",
JJ Allaire2ec40242014-09-15 09:18:39 -040078 transition = "default",
junkkad55cff02015-03-15 22:27:03 +010079 background_transition = "default",
JJ Allaire37f45b72016-01-30 18:17:45 -050080 reveal_options = NULL,
JJ Allaire82a8dee2016-07-12 10:25:36 -040081 reveal_plugins = NULL,
JJ Allaire2ec40242014-09-15 09:18:39 -040082 highlight = "default",
83 mathjax = "default",
84 template = "default",
JJ Allairefad55232015-10-19 07:47:26 -040085 css = NULL,
JJ Allaire2ec40242014-09-15 09:18:39 -040086 includes = NULL,
87 keep_md = FALSE,
88 lib_dir = NULL,
89 pandoc_args = NULL,
JJ Allaire375805c2016-11-15 08:56:43 -050090 extra_dependencies = NULL,
Atsushi Yasumoto0bf44442020-02-15 00:08:30 +090091 md_extensions = NULL,
JJ Allaire2ec40242014-09-15 09:18:39 -040092 ...) {
93
94 # function to lookup reveal resource
95 reveal_resources <- function() {
JJ Allaire2d8f3f22016-01-30 13:08:52 -050096 system.file("rmarkdown/templates/revealjs_presentation/resources",
JJ Allaire2ec40242014-09-15 09:18:39 -040097 package = "revealjs")
98 }
99
100 # base pandoc options for all reveal.js output
101 args <- c()
102
103 # template path and assets
JJ Allairea8c414b2016-01-30 14:36:53 -0500104 if (identical(template, "default")) {
105 default_template <- system.file(
106 "rmarkdown/templates/revealjs_presentation/resources/default.html",
107 package = "revealjs"
108 )
109 args <- c(args, "--template", pandoc_path_arg(default_template))
110 } else if (!is.null(template)) {
JJ Allaire2ec40242014-09-15 09:18:39 -0400111 args <- c(args, "--template", pandoc_path_arg(template))
JJ Allairea8c414b2016-01-30 14:36:53 -0500112 }
JJ Allaire2ec40242014-09-15 09:18:39 -0400113
114 # incremental
115 if (incremental)
116 args <- c(args, "--incremental")
117
118 # centering
JJ Allaire40fec332016-01-30 16:54:51 -0500119 jsbool <- function(value) ifelse(value, "true", "false")
120 args <- c(args, pandoc_variable_arg("center", jsbool(center)))
JJ Allaire2ec40242014-09-15 09:18:39 -0400121
JJ Allaire4c178052016-01-30 19:35:39 -0500122 # slide level
JJ Allairec58a6c42016-03-30 23:45:21 -0400123 args <- c(args, "--slide-level", as.character(slide_level))
JJ Allaire4c178052016-01-30 19:35:39 -0500124
JJ Allaire2ec40242014-09-15 09:18:39 -0400125 # theme
126 theme <- match.arg(theme, revealjs_themes())
127 if (identical(theme, "default"))
128 theme <- "simple"
129 else if (identical(theme, "dark"))
JJ Allaire6da1bb62016-01-30 14:28:39 -0500130 theme <- "black"
131 if (theme %in% c("blood", "moon", "night", "black"))
JJ Allaire2ec40242014-09-15 09:18:39 -0400132 args <- c(args, "--variable", "theme-dark")
133 args <- c(args, "--variable", paste("theme=", theme, sep=""))
134
135
136 # transition
137 transition <- match.arg(transition, revealjs_transitions())
138 args <- c(args, "--variable", paste("transition=", transition, sep=""))
139
junkkad55cff02015-03-15 22:27:03 +0100140 # background_transition
141 background_transition <- match.arg(background_transition, revealjs_transitions())
JJ Allaire40fec332016-01-30 16:54:51 -0500142 args <- c(args, "--variable", paste("backgroundTransition=", background_transition, sep=""))
143
JJ Allaire4ae486f2016-03-23 06:08:42 -0400144 # use history
145 args <- c(args, pandoc_variable_arg("history", "true"))
JJ Allaire8bfd3642016-03-21 13:43:13 -0400146
JJ Allaire37f45b72016-01-30 18:17:45 -0500147 # additional reveal options
148 if (is.list(reveal_options)) {
JJ Allaire375805c2016-11-15 08:56:43 -0500149
150 add_reveal_option <- function(option, value) {
JJ Allaire37f45b72016-01-30 18:17:45 -0500151 if (is.logical(value))
152 value <- jsbool(value)
Christophe Dervieuxd5dc8e92021-07-26 14:50:53 +0200153 else if (is.character(value)) {
154 # Add quotes around some config that can be several type
155 # e.g slideNumber = true or slideNumber = 'c/t'
156 if (option %in% c("slideNumber")) {
157 value <- paste0("'", value, "'")
158 }
159 }
JJ Allaire375805c2016-11-15 08:56:43 -0500160 args <<- c(args, pandoc_variable_arg(option, value))
161 }
162
163 for (option in names(reveal_options)) {
JJ Allaire064552c2017-02-10 10:28:41 -0500164 # special handling for nested options
165 if (option %in% c("chalkboard", "menu")) {
166 nested_options <- reveal_options[[option]]
167 for (nested_option in names(nested_options)) {
168 add_reveal_option(paste0(option, "-", nested_option),
169 nested_options[[nested_option]])
JJ Allaire375805c2016-11-15 08:56:43 -0500170 }
171 }
172 # standard top-level options
173 else {
174 add_reveal_option(option, reveal_options[[option]])
175 }
JJ Allaire37f45b72016-01-30 18:17:45 -0500176 }
177 }
178
JJ Allaire82a8dee2016-07-12 10:25:36 -0400179 # reveal plugins
180 if (is.character(reveal_plugins)) {
181
182 # validate that we need to use self_contained for plugins
183 if (self_contained)
184 stop("Using reveal_plugins requires self_contained: false")
185
186 # validate specified plugins are supported
JJ Allaire064552c2017-02-10 10:28:41 -0500187 supported_plugins <- c("notes", "search", "zoom", "chalkboard", "menu")
JJ Allaire82a8dee2016-07-12 10:25:36 -0400188 invalid_plugins <- setdiff(reveal_plugins, supported_plugins)
189 if (length(invalid_plugins) > 0)
190 stop("The following plugin(s) are not supported: ",
191 paste(invalid_plugins, collapse = ", "), call. = FALSE)
192
193 # add plugins
194 sapply(reveal_plugins, function(plugin) {
195 args <<- c(args, pandoc_variable_arg(paste0("plugin-", plugin), "1"))
JJ Allaire064552c2017-02-10 10:28:41 -0500196 if (plugin %in% c("chalkboard", "menu")) {
JJ Allaire375805c2016-11-15 08:56:43 -0500197 extra_dependencies <<- append(extra_dependencies,
198 list(rmarkdown::html_dependency_font_awesome()))
199
200 }
JJ Allaire82a8dee2016-07-12 10:25:36 -0400201 })
202 }
203
JJ Allaire2ec40242014-09-15 09:18:39 -0400204 # content includes
205 args <- c(args, includes_to_pandoc_args(includes))
206
JJ Allairefad55232015-10-19 07:47:26 -0400207 # additional css
208 for (css_file in css)
209 args <- c(args, "--css", pandoc_path_arg(css_file))
210
JJ Allaire2ec40242014-09-15 09:18:39 -0400211 # pre-processor for arguments that may depend on the name of the
212 # the input file (e.g. ones that need to copy supporting files)
213 pre_processor <- function(metadata, input_file, runtime, knit_meta, files_dir,
214 output_dir) {
215
JJ Allairee60feb22016-01-30 18:28:47 -0500216 # we don't work with runtime shiny
217 if (identical(runtime, "shiny")) {
218 stop("revealjs_presentation is not compatible with runtime 'shiny'",
219 call. = FALSE)
220 }
221
JJ Allaire2ec40242014-09-15 09:18:39 -0400222 # use files_dir as lib_dir if not explicitly specified
223 if (is.null(lib_dir))
224 lib_dir <- files_dir
225
226 # extra args
227 args <- c()
228
229 # reveal.js
JJ Allaire01541a22017-02-10 10:32:28 -0500230 revealjs_path <- system.file("reveal.js-3.3.0.1", package = "revealjs")
JJ Allaire2ec40242014-09-15 09:18:39 -0400231 if (!self_contained || identical(.Platform$OS.type, "windows"))
232 revealjs_path <- relative_to(
233 output_dir, render_supporting_files(revealjs_path, lib_dir))
JJ Allaireeea3bca2016-07-13 12:42:01 -0400234 else
235 revealjs_path <- pandoc_path_arg(revealjs_path)
BruceZhao30f9bc02016-07-13 22:33:54 +0800236 args <- c(args, "--variable", paste0("revealjs-url=", revealjs_path))
JJ Allaire2ec40242014-09-15 09:18:39 -0400237
238 # highlight
239 args <- c(args, pandoc_highlight_args(highlight, default = "pygments"))
240
241 # return additional args
242 args
243 }
244
245 # return format
246 output_format(
247 knitr = knitr_options_html(fig_width, fig_height, fig_retina, keep_md),
248 pandoc = pandoc_options(to = "revealjs",
Atsushi Yasumoto0bf44442020-02-15 00:08:30 +0900249 from = rmarkdown_format(paste0(
250 md_extensions,
251 ifelse(fig_caption, "", "-implicit_figures")
252 )),
JJ Allaire2ec40242014-09-15 09:18:39 -0400253 args = args),
254 keep_md = keep_md,
255 clean_supporting = self_contained,
256 pre_processor = pre_processor,
257 base_format = html_document_base(smart = smart, lib_dir = lib_dir,
258 self_contained = self_contained,
259 mathjax = mathjax,
JJ Allaire375805c2016-11-15 08:56:43 -0500260 pandoc_args = pandoc_args,
261 extra_dependencies = extra_dependencies,
262 ...))
JJ Allaire2ec40242014-09-15 09:18:39 -0400263}
264
265revealjs_themes <- function() {
266 c("default",
JJ Allaire6da1bb62016-01-30 14:28:39 -0500267 "dark",
JJ Allaire2ec40242014-09-15 09:18:39 -0400268 "simple",
269 "sky",
270 "beige",
271 "serif",
272 "solarized",
JJ Allaire2ec40242014-09-15 09:18:39 -0400273 "blood",
274 "moon",
junkka77fbf082015-03-15 22:25:47 +0100275 "night",
276 "black",
277 "league",
278 "white")
JJ Allaire2ec40242014-09-15 09:18:39 -0400279}
280
281
282revealjs_transitions <- function() {
junkka77fbf082015-03-15 22:25:47 +0100283 c(
284 "default",
285 "none",
JJ Allaire2ec40242014-09-15 09:18:39 -0400286 "fade",
junkka77fbf082015-03-15 22:25:47 +0100287 "slide",
288 "convex",
289 "concave",
290 "zoom"
291 )
JJ Allaire2ec40242014-09-15 09:18:39 -0400292}
293
294