Skip to content

Commit

Permalink
Merge pull request #867 from KJByron/fn_every_page
Browse files Browse the repository at this point in the history
capturing show footnote on every page in a clean pull
  • Loading branch information
dmurdoch authored Nov 23, 2024
2 parents 6e1d6a7 + 8b8c9f6 commit eb5e3c8
Show file tree
Hide file tree
Showing 6 changed files with 175 additions and 16 deletions.
2 changes: 1 addition & 1 deletion DESCRIPTION
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
Package: kableExtra
Type: Package
Title: Construct Complex Table with 'kable' and Pipe Syntax
Version: 1.4.0.6
Version: 1.4.0.7
Authors@R: c(
person('Hao', 'Zhu', email = '[email protected]', role = c('aut', 'cre'),
comment = c(ORCID = '0000-0002-3386-6076')),
Expand Down
84 changes: 70 additions & 14 deletions R/footnote.R
Original file line number Diff line number Diff line change
Expand Up @@ -27,6 +27,8 @@
#' @param fixed_small_size T/F When you want to keep the footnote small after
#' specifying large font size with the kable_styling() (e.g. ideal font for headers
#' and table content with small font in footnotes).
#' @param show_every_page T/F To make footnotes print at bottom of all pages for
#' long tables. Only applies to pdf longtables.
#' @param general_title Section header for general footnotes. Default is
#' "Note: ".
#' @param number_title Section header for number footnotes. Default is "".
Expand Down Expand Up @@ -64,6 +66,7 @@ footnote <- function(kable_input,
escape = TRUE,
threeparttable = FALSE,
fixed_small_size = FALSE,
show_every_page = FALSE,
general_title = "Note: ",
number_title = "",
alphabet_title = "",
Expand Down Expand Up @@ -127,7 +130,7 @@ footnote <- function(kable_input,
}
if (kable_format == "latex") {
return(footnote_latex(kable_input, footnote_table, footnote_as_chunk,
threeparttable, fixed_small_size))
threeparttable, fixed_small_size, show_every_page))
}
}

Expand Down Expand Up @@ -265,9 +268,12 @@ html_tfoot_maker_ <- function(ft_contents, ft_title, ft_type, ft_chunk) {

# LaTeX
footnote_latex <- function(kable_input, footnote_table, footnote_as_chunk,
threeparttable, fixed_small_size) {
threeparttable, fixed_small_size, show_every_page) {
fn_regexp <- fn_text <- longtable_start <- longtable_text <- NULL

table_info <- magic_mirror(kable_input)
out <- solve_enc(kable_input)
fn_regexp <- fn_text <- longtable_start <- longtable_text <- NULL

footnote_text <- latex_tfoot_maker(footnote_table, footnote_as_chunk,
table_info$ncol, threeparttable)
Expand All @@ -284,12 +290,16 @@ footnote_latex <- function(kable_input, footnote_table, footnote_as_chunk,
paste0("\\\\end{", table_info$tabular,
"}\n\\\\end{ThreePartTable}"),
out)
if (table_info$booktabs) {
out <- sub(bottomrule_regexp, "\\1\n\\\\insertTableNotes", out)
} else {
out <- sub("\\\\hline\n\\\\end\\{longtable\\}",
"\\\\hline\n\\\\insertTableNotes\n\\\\end\\{longtable\\}",
out)
if (!show_every_page) {
if (table_info$booktabs) {
out <- sub(bottomrule_regexp,
"\\1\n\\\\insertTableNotes",
out)
} else if (!show_every_page) {
out <- sub("\\\\hline\n\\\\end\\{longtable\\}",
"\\\\hline\n\\\\insertTableNotes\n\\\\end\\{longtable\\}",
out)
}
}
} else {
if (table_info$tabular == "tabu") {
Expand All @@ -309,13 +319,59 @@ footnote_latex <- function(kable_input, footnote_table, footnote_as_chunk,
out)
}
} else {
if (table_info$booktabs) {
out <- sub(bottomrule_regexp,
paste0("\\1\n", footnote_text), out)
if(!show_every_page) {
if (table_info$booktabs) {
out <- sub(bottomrule_regexp,
paste0("\\1\n", footnote_text),
out)
} else {
out <- sub(table_info$end_tabular,
paste0(footnote_text, "\n\\\\end{", table_info$tabular, "}"),
out)
}
}
}

if (table_info$tabular == "longtable" & show_every_page) {
fn_regexp <- ifelse(threeparttable, "\\\\insertTableNotes",
footnote_text)
fn_text <- gsub("\\\\", "\\", fn_regexp, fixed = TRUE)
if(is.null(table_info$repeat_header_latex)) {
# need full \begin{longtable} command
# table_info valign2 ok but align missing vertical lines
longtable_start <- sub(".*\\\\begin\\{longtable\\}",
"\\\\begin\\{longtable\\}", out)
longtable_text <- sub("\n.*", "", longtable_start)
out <- sub(longtable_text,
paste(longtable_text, fn_text, "\n\\endfoot\n"),
out, fixed = TRUE)
} else {
out <- sub(table_info$end_tabular,
paste0(footnote_text, "\n\\\\end{", table_info$tabular, "}"),
out)
if(!table_info$booktabs){
out <- sub(
"\\\\endhead\\n",
paste0("\\\\endhead\n",
fn_regexp, "\n\\\\endfoot\n",
fn_regexp, "\n\\\\endlastfoot\n"),
out)
} else {
if (grepl( # no repeat_header_continued in kable_styling
"\\\\endhead\\n\\n\\\\endfoot",
out)) {
out <- sub(
"\\\\endhead\\n\\n\\\\endfoot",
paste0("\\\\endhead\n\\\\midrule\n", fn_regexp, "\n\\\\endfoot"),
out)
} else { # repeat_header_continued in kable_styling
out <- sub(
"\\\\endfoot",
paste0(fn_regexp, "\n\\\\endfoot"),
out)
}
out <- sub(
"\\\\endlastfoot",
paste0(fn_regexp, "\n\\\\endlastfoot"),
out)
}
}
}

Expand Down
7 changes: 6 additions & 1 deletion inst/NEWS.md
Original file line number Diff line number Diff line change
@@ -1,6 +1,11 @@
kableExtra 1.4.0.6

kableExtra 1.4.0.7
--------------------------------------------------------------------------------

New Features:

* Added `show_every_page` argument to `footnote()` (#867).

Bug Fixes:

* Fixed a bug in `collapse_rows()`, which failed on tables
Expand Down
4 changes: 4 additions & 0 deletions man/footnote.Rd

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

58 changes: 58 additions & 0 deletions tests/visual_tests/footnote_latex_perm.Rmd
Original file line number Diff line number Diff line change
@@ -0,0 +1,58 @@
---
title: "footnote_perm"
author: "Byron"
date: "9/11/2024"
output: pdf_document
---

```{r setup, include=FALSE}
knitr::opts_chunk$set(echo = !TRUE)
devtools::load_all()
# library(kableExtra)
dat <- data.frame(
Col1 = LETTERS[rep(1:26, each = 2)],
Col2 = letters[rep(1:26, 2)],
n = 0:51
)
# "\\tnote{1} is TPT only
names(dat)[3] <- paste0(names(dat)[3], footnote_marker_number(1))
```


```{r permute, results='asis'}
# Define new environment
latex_opt <- c("basic", "repeat_header") # test basic to show doesn't cause trouble
is_bt <- c(FALSE, TRUE)
is_tpt <- c(FALSE, TRUE)
is_cont_at_btm <- c(FALSE, TRUE) # only shows for booktabs
is_every <- c(FALSE, TRUE)
child_env <- new.env()
for(i_latex_opt in seq_along(latex_opt)){
for(i_bt in seq_along(is_bt)){
for(i_tpt in seq_along(is_tpt)){
for(i_is_cont_at_btm in seq_along(is_cont_at_btm)){
for(i_every in seq_along(is_every)){
child_env$i_latex_opt <- i_latex_opt
child_env$i_bt <- i_bt
child_env$i_tpt <- i_tpt
child_env$i_is_cont_at_btm <- i_is_cont_at_btm
child_env$i_every <- i_every
# knit the document and save the character output to an object
res <- knitr::knit_child(
"footnote_latex_perm_child.Rmd",
envir = child_env,
quiet = TRUE
)
# Cat the object to make it render in your main document
cat(res, sep = "\n")
}
}
}
}
}
```

36 changes: 36 additions & 0 deletions tests/visual_tests/footnote_latex_perm_child.Rmd
Original file line number Diff line number Diff line change
@@ -0,0 +1,36 @@
---
output:
pdf_document
---
```{r results='asis'}
# i_latex_opt <- i_bt <- i_tpt <- i_is_cont_at_btm <- i_every <- 2
use_lab <- paste("lab", i_latex_opt, i_bt, i_tpt,
i_is_cont_at_btm, i_every, sep = "-")
```

### `r use_lab`
Permutations
\vspace{-10pt}

* latex option: `r latex_opt[i_latex_opt]`
* booktab: `r is_bt[i_bt]`
* TPT: `r is_tpt[i_tpt]`
* continue on bottom: `r is_cont_at_btm[i_is_cont_at_btm]`
* every page: `r is_every[i_every]`


```{r results='asis'}
dat |>
kbl(format = "latex", caption = "Table caption",
label = use_lab, escape = FALSE,
booktabs = is_bt[i_bt], longtable = TRUE) |>
kable_styling(
latex_options = latex_opt[i_latex_opt],
repeat_header_continued = is_cont_at_btm[i_is_cont_at_btm]) |>
footnote(
number = c("footnote on colname"),
threeparttable = is_tpt[i_tpt],
show_every_page = is_every[i_every])
```

\clearpage

0 comments on commit eb5e3c8

Please sign in to comment.