From 50749c326ca579ab145d667cf43a2db4927858cf Mon Sep 17 00:00:00 2001 From: atusy <30277794+atusy@users.noreply.github.com> Date: Thu, 11 Mar 2021 20:10:45 +0900 Subject: [PATCH] Automatically add citations to reference (#48) --- NEWS.md | 3 ++- R/colformat.R | 10 ++++++++-- R/collect-citations.R | 14 ++++++++++++++ R/knit-print.R | 27 +++++++++++++++++++++++++++ R/zzz.R | 3 +++ docs/news/index.html | 4 +++- inst/lua/cite.lua | 12 ++++++++++++ 7 files changed, 69 insertions(+), 4 deletions(-) create mode 100644 R/collect-citations.R create mode 100644 R/knit-print.R create mode 100644 R/zzz.R create mode 100644 inst/lua/cite.lua diff --git a/NEWS.md b/NEWS.md index 998c933..09ebd79 100644 --- a/NEWS.md +++ b/NEWS.md @@ -2,7 +2,8 @@ ## New features -* `colformat_md` supports multiple paragprahs by collapsing them with a separator given to the `.sep` argument (default: `"\n\n"). +* `colformat_md` supports multiple paragprahs by collapsing them with a separator given to the `.sep` argument (default: `"\n\n") (#43). +* `colformat_md` can now automatically add citations to reference on R Markdown (#48). ## Bug fixes diff --git a/R/colformat.R b/R/colformat.R index 4c84e74..8176d60 100644 --- a/R/colformat.R +++ b/R/colformat.R @@ -54,11 +54,13 @@ colformat_md <- function(x, return(x) } + texts <- unlist(dataset[col], use.names = FALSE) + # Must evaluate outside add_footnotes due to lazy evaluation of arguments ft <- flextable::compose(x, i = seq(nrow(dataset)), j = col, part = part, value = as_paragraph_md( - unlist(dataset[col], use.names = FALSE), + texts, auto_color_link = auto_color_link, .from = .from, md_extensions = md_extensions, @@ -67,7 +69,11 @@ colformat_md <- function(x, .sep = .sep )) - add_footnotes(ft, part, .footnote_options) + structure( + add_footnotes(ft, part, .footnote_options), + class = c("ftExtra", class(ft)), + citations = collect_citations(paste(texts, collapse = "\n\n")) + ) } where <- function(...) { diff --git a/R/collect-citations.R b/R/collect-citations.R new file mode 100644 index 0000000..175cf9a --- /dev/null +++ b/R/collect-citations.R @@ -0,0 +1,14 @@ +collect_citations <- function(x, .from = 'markdown') { + tf <- tempfile() + xfun::write_utf8(x, tf) + rmarkdown::pandoc_convert( + input = tf, + to = "markdown", + from = .from, + output = tf, + citeproc = FALSE, + options = lua("cite.lua"), + wd = getwd() + ) + paste(readLines(tf), collapse = "") +} diff --git a/R/knit-print.R b/R/knit-print.R new file mode 100644 index 0000000..86e08ce --- /dev/null +++ b/R/knit-print.R @@ -0,0 +1,27 @@ +#' knit_print for extended flextable object +#' +#' Adds a YAML meta data block to cite materials found by `colformat_md()` in +#' the `flextable` object. +#' +#' @inheritParams knitr::knit_print +#' @param label +#' A key to be set in the YAML meta data block to cite materials. +#' This must be unique in the document. +#' Default value is the chunk label prepended by "ftExtra-cite-". +#' +#' @return The `knit_asis` class object. +#' @noRd +knit_print.ftExtra <- function(x, + options, + ..., + key = paste0("ftExtra-cite-", options$label)) { + ft <- NextMethod("knit_print", x, options, ...) + + cite <- attr(x, "citations", exact = TRUE) + + if (cite == "") return(ft) + + res <- sprintf('---\n%s: "%s"\n---\n\n%s', key, cite, ft) + attributes(res) <- attributes(ft) + res +} diff --git a/R/zzz.R b/R/zzz.R new file mode 100644 index 0000000..92c6dcd --- /dev/null +++ b/R/zzz.R @@ -0,0 +1,3 @@ +.onLoad <- function(libname, pkgname) { + registerS3method("knit_print", "ftExtra", knit_print.ftExtra, asNamespace("knitr")) +} diff --git a/docs/news/index.html b/docs/news/index.html index d282709..16dfa26 100644 --- a/docs/news/index.html +++ b/docs/news/index.html @@ -141,7 +141,9 @@

New features

diff --git a/inst/lua/cite.lua b/inst/lua/cite.lua new file mode 100644 index 0000000..82a9435 --- /dev/null +++ b/inst/lua/cite.lua @@ -0,0 +1,12 @@ +cite = {} + +function Cite(elem) + table.insert(cite, elem) + table.insert(cite, pandoc.Space()) +end + +function Pandoc(doc) + table.remove(cite) + doc.blocks = {pandoc.Para(cite)} + return doc +end