blob: abcf0588a0ca19127c1656776529c4a85078387e [file] [log] [blame]
Hao Zhu8f417202017-05-20 16:37:14 -04001#' Add indentations to row headers
Hao Zhubd95bb22017-05-22 16:08:49 -04002#'
3#' @param kable_input Output of `knitr::kable()` with `format` specified
4#' @param positions A vector of numeric row numbers for the rows that need to
5#' be indented.
6#'
Hao Zhu78e61222017-05-24 20:53:35 -04007#' @examples x <- knitr::kable(head(mtcars), "html")
8#' # Add indentations to the 2nd & 4th row
9#' add_indent(x, c(2, 4))
10#'
Hao Zhu8f417202017-05-20 16:37:14 -040011#' @export
12add_indent <- function(kable_input, positions) {
Hao Zhud972e7f2017-05-22 13:27:15 -040013 if (!is.numeric(positions)) {
14 stop("Positions can only take numeric row numbers (excluding header rows).")
15 }
Hao Zhu8f417202017-05-20 16:37:14 -040016 kable_format <- attr(kable_input, "format")
17 if (!kable_format %in% c("html", "latex")) {
18 message("Currently generic markdown table using pandoc is not supported.")
19 return(kable_input)
20 }
21 if (kable_format == "html") {
Hao Zhu62cdde52017-05-20 22:16:03 -040022 return(add_indent_html(kable_input, positions))
Hao Zhu8f417202017-05-20 16:37:14 -040023 }
24 if (kable_format == "latex") {
25 return(add_indent_latex(kable_input, positions))
26 }
27}
28
Hao Zhu62cdde52017-05-20 22:16:03 -040029# Add indentation for LaTeX
Hao Zhu8f417202017-05-20 16:37:14 -040030add_indent_latex <- function(kable_input, positions) {
Hao Zhud972e7f2017-05-22 13:27:15 -040031 table_info <- magic_mirror(kable_input)
Hao Zhu8f417202017-05-20 16:37:14 -040032
Hao Zhu8f417202017-05-20 16:37:14 -040033 if (max(positions) > table_info$nrow - 1) {
34 stop("There aren't that many rows in the table. Check positions in ",
35 "add_indent_latex.")
36 }
37
38 out <- kable_input
39 for (i in positions) {
40 rowtext <- table_info$contents[i + 1]
Hao Zhue5bfbf12017-06-06 00:24:49 -040041 out <- sub(rowtext, latex_indent_unit(rowtext), out, perl = TRUE)
Hao Zhu2ce42b92017-06-15 17:15:33 -040042 table_info$contents[i + 1] <- latex_indent_unit(rowtext)
Hao Zhu8f417202017-05-20 16:37:14 -040043 }
Hao Zhu2ce42b92017-06-15 17:15:33 -040044 out <- structure(out, format = "latex", class = "knitr_kable")
Hao Zhu32f43f72017-06-20 18:24:54 -040045 attr(out, "kable_meta") <- table_info
Hao Zhu8f417202017-05-20 16:37:14 -040046 return(out)
47}
48
49latex_indent_unit <- function(rowtext) {
Hao Zhu2ce42b92017-06-15 17:15:33 -040050 paste0("\\\\hspace\\{1em\\}", rowtext)
Hao Zhu8f417202017-05-20 16:37:14 -040051}
Hao Zhu62cdde52017-05-20 22:16:03 -040052
53# Add indentation for HTML
54add_indent_html <- function(kable_input, positions) {
55 kable_attrs <- attributes(kable_input)
56
57 kable_xml <- read_xml(as.character(kable_input), options = "COMPACT")
58 kable_tbody <- xml_tpart(kable_xml, "tbody")
59
60 group_header_rows <- attr(kable_input, "group_header_rows")
61 if (!is.null(group_header_rows)) {
62 positions <- positions_corrector(positions, group_header_rows,
63 length(xml_children(kable_tbody)))
64 }
65 for (i in positions) {
66 node_to_edit <- xml_child(xml_children(kable_tbody)[[i]], 1)
67 if (!xml_has_attr(node_to_edit, "indentLevel")) {
68 xml_attr(node_to_edit, "style") <- paste(
69 xml_attr(node_to_edit, "style"), "padding-left: 2em;"
70 )
71 xml_attr(node_to_edit, "indentLevel") <- 1
72 } else {
73 indentLevel <- as.numeric(xml_attr(node_to_edit, "indentLevel"))
74 xml_attr(node_to_edit, "style") <- sub(
75 paste0("padding-left: ", indentLevel * 2, "em;"),
76 paste0("padding-left: ", (indentLevel + 1) * 2, "em;"),
77 xml_attr(node_to_edit, "style")
78 )
79 xml_attr(node_to_edit, "indentLevel") <- indentLevel + 1
80 }
81 }
82 out <- structure(as.character(kable_xml), format = "html",
83 class = "knitr_kable")
84 attributes(out) <- kable_attrs
85 return(out)
86}