Nothing
usepackage_latex <- function(name, options = NULL, extra_lines = NULL) {
assert_dependency("rmarkdown")
invisible(
knitr::knit_meta_add(
list(rmarkdown::latex_dependency(name, options, extra_lines))
)
)
}
sanity_align <- function(align, i) {
if (any(grepl("d", align)) && !is.null(i)) {
msg <- "d column alignment can only be applied to entire columns. `i` must be `NULL`."
stop(msg, call. = FALSE)
}
}
sanitize_i <- function(
i,
x,
pre_group_i = FALSE,
lazy = TRUE,
calling_function = "other"
) {
if (is.character(i)) {
# format_tt() accepts certain character inputs
# if (calling_function %in% c("format_tt", "style_tt")) {
assert_choice(
i,
choice = c("caption", "colnames", "groupi", "~groupi", "groupj", "notes"),
null.ok = TRUE
)
}
if (identical(i, "groupi")) {
idx <- x@group_index_i
if (length(idx) == 0) {
msg <- "No grouping index found. Please use `group_tt()` first."
stop(msg, call. = FALSE)
}
i <- x@group_index_i
} else if (identical(i, "~groupi")) {
i <- setdiff(seq_len(nrow(x)), x@group_index_i)
} else if (identical(i, "groupj")) {
has_colnames <- length(names(x)) > 0
has_groupj <- nrow(x@group_data_j) > 0
if (!has_groupj) {
msg <- "No column grouping found. Please use `group_tt(j = ...)` first."
stop(msg, call. = FALSE)
}
if (has_colnames) {
i <- -(1:x@nhead)
} else {
i <- -(0:x@nhead)
}
} else if (identical(i, "colnames")) {
if (isTRUE(length(colnames(x)) == 0)) {
msg <- "No column names found. Please set column names first."
stop(msg, call. = FALSE)
}
i <- 0
}
if (is.matrix(i) && is.logical(i)) {
return(i)
}
out <- seq_len(nrow(x))
assert_numeric(i, null.ok = TRUE, name = "i")
if (is.null(i) && isTRUE(lazy)) {
out <- NA
attr(out, "null") <- TRUE
attr(out, "body") <- seq_len(nrow(x))
attr(out, "head") <- integer()
} else {
if (!is.null(i)) {
out <- i
} else if (inherits(x, "tinytable")) {
out <- seq_len(nrow(x))
}
attr(out, "null") <- FALSE
attr(out, "body") <- out[out > 0]
attr(out, "head") <- out[out < 1]
}
return(out)
}
sanitize_j <- function(j, x, skip_tabulator_types = FALSE) {
# regex
if (is.character(j) && length(j) == 1 && !is.null(colnames(x))) {
out <- grep(j, colnames(x), perl = TRUE)
# Check if regex pattern matched any columns
if (length(out) == 0) {
msg <- sprintf("No columns match the pattern: %s", j)
stop(msg, call. = FALSE)
}
# full names
} else if (is.character(j) && length(j) > 1 && !is.null(colnames(x))) {
bad <- setdiff(j, colnames(x))
if (length(bad) > 0) {
msg <- sprintf("Missing columns: %s", paste(bad, collapse = ", "))
stop(msg, call. = FALSE)
}
out <- which(colnames(x) %in% j)
} else {
assert_integerish(j, lower = 1, upper = ncol(x), null.ok = TRUE)
if (is.null(j)) {
out <- seq_len(ncol(x))
} else {
out <- j
}
}
# Filter out columns that should be skipped for tabulator formatting
if (skip_tabulator_types && inherits(x, "tinytable") && x@output == "tabulator") {
skip_cols <- c()
for (col_idx in out) {
col_data <- x@data[[col_idx]]
if (inherits(col_data, c("integer", "numeric", "double", "logical", "Date", "POSIXct", "POSIXlt"))) {
skip_cols <- c(skip_cols, col_idx)
}
}
out <- setdiff(out, skip_cols)
}
attr(out, "null") <- is.null(j)
return(out)
}
sanitize_output <- function(x, output) {
if (is.null(x)) {
stop("'x' cannot be NULL", call. = FALSE)
}
if (is.null(output)) {
return(x)
}
assert_choice(
output,
choice = c(
"latex",
"html",
"typst",
"dataframe",
"markdown"
)
)
x@output <- output
return(x)
}
infer_output <- function(x) {
output <- x@output
# If output is already set to a specific format, respect it
if (!is.null(output) && !identical(output, "tinytable")) {
return(output)
} else {
# Only do inference when output is NULL or "tinytable"
has_viewer <- interactive() && !is.null(getOption("viewer"))
out <- if (has_viewer) "html" else "markdown"
}
# Environmental overrides only apply when we're doing inference (not explicit output)
if (isTRUE(check_dependency("litedown"))) {
fmt <- tryCatch(litedown::get_context("format"), error = function(e) NULL)
if (identical(fmt, "latex")) {
return("latex")
} else if (identical(fmt, "markdown")) {
return("markdown")
} else if (identical(fmt, "commonmark")) {
return("markdown")
} else if (identical(fmt, "html")) {
return("html")
}
}
if (isTRUE(check_dependency("knitr"))) {
pandoc_to <- knitr::pandoc_to()
if (isTRUE(pandoc_to %in% c("latex", "beamer"))) {
if (isTRUE(x@latex_preamble)) {
usepackage_latex("float")
usepackage_latex(
"tabularray",
extra_lines = c(
"\\usepackage[normalem]{ulem}",
"\\usepackage{graphicx}",
"\\usepackage{rotating}",
"\\UseTblrLibrary{siunitx}",
"\\NewTableCommand{\\tinytableDefineColor}[3]{\\definecolor{#1}{#2}{#3}}",
"\\newcommand{\\tinytableTabularrayUnderline}[1]{\\underline{#1}}",
"\\newcommand{\\tinytableTabularrayStrikeout}[1]{\\sout{#1}}"
)
)
}
out <- "latex"
} else if (isTRUE(pandoc_to %in% c("html", "revealjs"))) {
out <- "html"
} else if (isTRUE(pandoc_to == "typst")) {
out <- "typst"
if (isTRUE(check_dependency("quarto"))) {
if (isTRUE(quarto::quarto_version() < "1.5.29")) {
msg <- "Typst tables require version 1.5.29 or later of Quarto and version 0.11.0 or later of Typst. This software may (or may not) only be available in pre-release builds: https://quarto.org/docs/download"
stop(msg, call. = FALSE)
}
}
# docx, markdown variants, and unknown formats
} else if (!is.null(pandoc_to)) {
out <- "markdown"
}
}
return(out)
}
#' check if dependency is installed
#'
#' @noRd
check_dependency <- function(library_name) {
flag <- requireNamespace(library_name, quietly = TRUE)
if (isFALSE(flag)) {
msg <- sprintf("Please install the `%s` package.", library_name)
return(msg)
} else {
return(TRUE)
}
}
assert_dependency <- function(library_name) {
flag <- check_dependency(library_name)
if (!isTRUE(flag)) {
stop(flag, call. = FALSE)
}
return(invisible())
}
assert_choice <- function(
x,
choice,
null.ok = FALSE,
name = as.character(substitute(x))
) {
if (is.null(x) && isTRUE(null.ok)) {
return(TRUE)
}
if (is.character(x) && length(x) == 1 && x %in% choice) {
return(TRUE)
}
msg <- sprintf(
"`%s` must be one of: %s",
name,
paste(choice, collapse = ", ")
)
stop(msg, call. = FALSE)
}
check_string <- function(x, null.ok = FALSE, len = NULL) {
if (is.null(x) && isTRUE(null.ok)) {
return(invisible(TRUE))
}
if (is.character(x) && length(x) == 1) {
if (is.null(len)) {
return(invisible(TRUE))
} else if (nchar(x) == len) {
return(invisible(TRUE))
}
}
return(FALSE)
}
assert_string <- function(
x,
null.ok = FALSE,
len = NULL,
name = as.character(substitute(x))) {
if (is.null(len)) {
msg <- sprintf("`%s` must be a string.", name)
} else {
msg <- sprintf("`%s` must be a string of length %s.", name, len)
}
if (!isTRUE(check_string(x, len = len, null.ok = null.ok))) {
stop(msg, call. = FALSE)
}
if (!is.null(x) && !is.null(len) && nchar(x) != len) {
msg <- sprintf("`%s` must be a string of length %s.", name, len)
stop(msg, call. = FALSE)
}
}
check_flag <- function(x, null.ok = FALSE) {
if (is.null(x) && isTRUE(null.ok)) {
return(TRUE)
}
if (is.logical(x) && length(x) == 1) {
return(TRUE)
}
return(FALSE)
}
assert_flag <- function(
x,
null.ok = FALSE,
name = as.character(substitute(x))
) {
msg <- sprintf("`%s` must be a logical flag.", name)
if (!isTRUE(check_flag(x, null.ok = null.ok))) {
stop(msg, call. = FALSE)
}
}
check_function <- function(x, null.ok = FALSE) {
if (is.null(x) && isTRUE(null.ok)) {
return(TRUE)
}
if (is.function(x)) {
return(TRUE)
}
return(FALSE)
}
assert_function <- function(
x,
null.ok = FALSE,
name = as.character(substitute(x))
) {
msg <- sprintf("`%s` must be a function.", name)
if (!isTRUE(check_function(x, null.ok = null.ok))) {
stop(msg, call. = FALSE)
}
}
assert_length <- function(
x,
len = 1,
null.ok = FALSE,
name = as.character(substitute(x))
) {
if (is.null(x) && isTRUE(null.ok)) {
return(invisible(TRUE))
}
msg <- sprintf(
"`%s` must be one of these lengths: %s",
name,
paste(unique(len), collapse = ", ")
)
if (!length(x) %in% len) {
stop(msg, call. = FALSE)
}
}
assert_logical <- function(
x,
null.ok = FALSE,
name = as.character(substitute(x))
) {
if (is.null(x) && isTRUE(null.ok)) {
return(invisible(TRUE))
}
msg <- sprintf("`%s` must be a logical vector", name)
if (!is.logical(x)) stop(msg, call. = FALSE)
}
check_integerish <- function(
x,
len = NULL,
lower = NULL,
upper = NULL,
null.ok = TRUE
) {
if (is.null(x) && isTRUE(null.ok)) {
return(TRUE)
}
if (!is.numeric(x)) {
return(FALSE)
}
x <- stats::na.omit(x)
if (!is.null(len) && length(x) != len) {
return(FALSE)
}
if (!is.null(lower) && any(x < lower)) {
return(FALSE)
}
if (!is.null(upper) && any(x > upper)) {
return(FALSE)
}
if (isTRUE(any(abs(x - round(x)) > (.Machine$double.eps)^0.5))) {
return(FALSE)
}
return(TRUE)
}
assert_integerish <- function(
x,
len = NULL,
lower = NULL,
upper = NULL,
null.ok = FALSE,
name = as.character(substitute(x))
) {
if (isTRUE(null.ok) && is.null(x)) {
return(invisible())
}
msg <- sprintf("`%s` must be integer-ish", name)
if (is.null(x) && !isTRUE(null.ok)) {
stop(sprintf("%s should not be NULL.", name), call. = FALSE)
}
if (
!isTRUE(
check_integerish(
x,
len = len,
lower = lower,
upper = upper,
null.ok = null.ok
)
)
) {
if (!is.numeric(x)) {
msg <- paste0(msg, "; it is not numeric")
}
if (!is.null(len) && length(x) != len) {
msg <- paste0(msg, sprintf("; its length must be %s", len))
}
if (!is.null(lower) && any(x < lower)) {
msg <- paste0(
msg,
sprintf("; all values must be greater than or equal to %s", lower)
)
}
if (!is.null(upper) && any(x > upper)) {
msg <- paste0(
msg,
sprintf("; all values must be less than or equal to %s", upper)
)
}
if (isTRUE(any(abs(x - round(x)) > (.Machine$double.eps)^0.5))) {
msg <- paste0(msg, "; all values must be close to integers")
}
stop(msg, call. = FALSE)
}
}
check_null <- function(x) {
is.null(x)
}
assert_null <- function(x, name = as.character(substitute(x))) {
if (!isTRUE(check_null(x))) {
stop(sprintf("%s should be NULL.", name), call. = FALSE)
}
}
check_numeric <- function(
x,
len = NULL,
lower = NULL,
upper = NULL,
null.ok = TRUE
) {
if (is.null(x) && isTRUE(null.ok)) {
return(TRUE)
}
if (!is.numeric(x)) {
return(FALSE)
}
if (!is.null(len) && length(x) != len) {
return(FALSE)
}
if (!is.null(lower) && any(x < lower)) {
return(FALSE)
}
if (!is.null(upper) && any(x > upper)) {
return(FALSE)
}
return(TRUE)
}
assert_numeric <- function(
x,
len = NULL,
lower = NULL,
upper = NULL,
null.ok = FALSE,
name = as.character(substitute(x))
) {
msg <- sprintf("`%s` must be numeric", name)
if (
!isTRUE(
check_numeric(
x,
len = len,
lower = lower,
upper = upper,
null.ok = null.ok
)
)
) {
if (!is.null(len) && length(x) != len) {
msg <- paste0(msg, sprintf("; its length must be %s", len))
}
if (!is.null(lower) && any(x < lower)) {
msg <- paste0(
msg,
sprintf("; all values must be greater than or equal to %s", lower)
)
}
if (!is.null(upper) && any(x > upper)) {
msg <- paste0(
msg,
sprintf("; all values must be less than or equal to %s", upper)
)
}
stop(msg, call. = FALSE)
}
}
assert_data_frame <- function(
x,
min_rows = 0,
min_cols = 0,
name = as.character(substitute(x))
) {
msg <- sprintf("`%s` must be a data.frame.", name)
if (!is.data.frame(x)) {
stop(msg, call. = FALSE)
}
msg <- sprintf("Number of rows in `%s` must be at least `%s`", name, min_rows)
if (nrow(x) < min_rows) {
stop(msg, call. = FALSE)
}
msg <- sprintf(
"Number of columns in `%s` must be at least `%s`",
name,
min_cols
)
if (ncol(x) < min_cols) stop(msg, call. = FALSE)
}
check_matrix <- function(
x,
min_rows = 0,
min_cols = 0,
null.ok = FALSE
) {
if (is.null(x) && isTRUE(null.ok)) {
return(TRUE)
}
if (!is.matrix(x)) {
return(FALSE)
}
if (nrow(x) < min_rows) {
return(FALSE)
}
if (ncol(x) < min_cols) {
return(FALSE)
}
return(TRUE)
}
assert_matrix <- function(
x,
min_rows = 0,
min_cols = 0,
null.ok = FALSE,
name = as.character(substitute(x))
) {
if (is.null(x) && isTRUE(null.ok)) {
return(invisible(TRUE))
}
msg <- sprintf("`%s` must be a matrix.", name)
if (!is.matrix(x)) {
stop(msg, call. = FALSE)
}
msg <- sprintf("Number of rows in `%s` must be at least `%s`", name, min_rows)
if (nrow(x) < min_rows) {
stop(msg, call. = FALSE)
}
msg <- sprintf(
"Number of columns in `%s` must be at least `%s`",
name,
min_cols
)
if (ncol(x) < min_cols) stop(msg, call. = FALSE)
}
check_character <- function(
x,
len = NULL,
null.ok = FALSE,
name = as.character(substitute(x))
) {
if (isTRUE(null.ok) && is.null(x)) {
return(TRUE)
} else if (!is.character(x)) {
msg <- sprintf("`%s` must be character.", name)
return(msg)
} else if (!is.null(len) && length(x) != len) {
msg <- sprintf("`%s` must have length %s.", name, len)
return(msg)
}
return(TRUE)
}
assert_character <- function(
x,
len = NULL,
null.ok = FALSE,
name = as.character(substitute(x))
) {
flag <- check_character(x, len = len, null.ok = null.ok, name = name)
if (!isTRUE(flag)) {
stop(flag, call. = FALSE)
} else {
return(invisible(TRUE))
}
}
assert_list <- function(
x,
named = FALSE,
len = NULL,
null.ok = FALSE,
name = as.character(substitute(x))
) {
if (isTRUE(null.ok) && is.null(x)) {
return(invisible(TRUE))
}
if (!is.list(x)) {
stop("Input is not a list.", call. = FALSE)
}
if (isTRUE(named)) {
if (is.null(names(x))) {
stop(sprintf("`%s` should be named list.", name), call. = FALSE)
}
}
if (!is.null(len)) {
if (length(x) != len) {
stop(sprintf("`%s` must be of length %s.", name, len), call. = FALSE)
}
}
}
assert_function <- function(
x,
null.ok = FALSE,
name = as.character(substitute(x))
) {
if (isTRUE(null.ok) && is.null(x)) {
return(invisible(TRUE))
}
if (!is.function(x)) {
msg <- sprintf("`%s` must be a function.", name)
stop(msg, call. = FALSE)
}
}
check_atomic_vector <- function(
x,
null.ok = FALSE,
name = as.character(substitute(x))
) {
if (isTRUE(null.ok) && is.null(x)) {
return(invisible(TRUE))
}
# doesn't work on glue::glue() output
# flag <- is.atomic(x) && is.vector(x) && !is.list(x)
flag <- is.atomic(x) && is.null(dim(x)) && length(x) > 0 && !is.list(x)
if (flag) {
out <- TRUE
} else if (is.factor(x) && is.null(dim(x))) {
out <- TRUE
} else {
out <- sprintf("`%s` must be an atomic vector.", name)
}
return(out)
}
assert_atomic_vector <- function(x,
null.ok = FALSE,
name = as.character(substitute(x))) {
flag <- check_atomic_vector(x, null.ok = null.ok, name = name)
if (!isTRUE(flag)) {
stop(flag, call. = FALSE)
} else {
return(invisible(TRUE))
}
}
assert_class <- function(x, classname) {
if (!inherits(x, classname)) {
msg <- sprintf("`x` must be of class `%s`.", classname)
stop(msg, call. = FALSE)
}
}
sanitize_notes <- function(notes) {
if (is.character(notes) && length(notes) > 0) {
notes <- as.list(notes)
}
assert_list(notes, null.ok = TRUE)
for (idx in seq_along(notes)) {
n <- notes[[idx]]
bad <- FALSE
if (is.list(n)) {
if (is.null(names(notes)[idx])) {
bad <- TRUE
}
if (!all(c("i", "j", "text") %in% names(n))) {
bad <- TRUE
}
} else if (!is.character(n) || length(n) != 1) {
bad <- TRUE
}
if (isTRUE(bad)) {
stop(
"`notes` includes invalid elements. Please refer to the documentation for details.",
call. = FALSE
)
}
}
return(notes)
}
sanitize_replace <- function(replace) {
if (isTRUE(replace)) {
replace <- list(" " = c("NA", "NaN"), " " = c(NA, NaN))
} else if (isFALSE(replace)) {
replace <- list(NULL)
} else if (isTRUE(check_string(replace))) {
if (identical(replace, "")) {
replace <- " "
}
replace <- stats::setNames(list(NA, NaN), c(replace, replace))
} else if (!is.list(replace) || is.null(names(replace))) {
stop(
"`replace` should be TRUE/FALSE, a single string, or a named list.",
call. = FALSE
)
}
return(replace)
}
sanity_num_mark <- function(digits, num_mark_big, num_mark_dec) {
# commented out because doesn't work with french decimals
if (identical(getOption("OutDec"), ".")) {
if (is.null(digits)) {
if (num_mark_big != "") {
stop("`num_mark_big` requires a `digits` value.", call. = FALSE)
}
if (num_mark_dec != ".") {
stop("`num_mark_dec` requires a `digits` value.", call. = FALSE)
}
}
}
}
check_true <- function(x, null.ok = FALSE) {
if (is.null(x) && isTRUE(null.ok)) {
return(TRUE)
}
if (isTRUE(x)) {
return(TRUE)
}
return(FALSE)
}
assert_true <- function(
x,
null.ok = FALSE,
name = as.character(substitute(x))
) {
msg <- sprintf("`%s` must be a logical true.", name)
if (!isTRUE(check_true(x, null.ok = null.ok))) {
stop(msg, call. = FALSE)
}
}
assert_number <- function(
x,
lower = NULL,
upper = NULL,
null.ok = FALSE,
name = as.character(substitute(x))
) {
# Use assert_numeric with len = 1 to ensure single number
assert_numeric(
x,
len = 1,
lower = lower,
upper = upper,
null.ok = null.ok,
name = name
)
}
Any scripts or data that you put into this service are public.
Add the following code to your website.
For more information on customizing the embed code, read Embedding Snippets.