skip_on_cran()
my_tbl_summary <- trial |>
select(trt, age, death) |>
tbl_summary()
my_tbl_regression <- lm(marker ~ age, trial) |> tbl_regression()
my_spanning_tbl <- trial |>
tbl_summary(by = grade, include = age) |>
modify_spanning_header(c(stat_1, stat_3) ~ "**Testing**")
gt_tbl_summary <- my_tbl_summary |> as_gt()
gt_tbl_regression <- my_tbl_regression |> as_gt()
gt_spanning_tbl <- my_spanning_tbl |> as_gt()
test_that("as_gt works with standard use", {
# return_calls argument does not produce warnings
expect_silent(my_tbl_summary |> as_gt(return_calls = TRUE))
# include argument does not produce warnings
expect_silent(my_tbl_summary |> as_gt(include = gt))
# correct elements are returned
expect_equal(
names(gt_tbl_summary),
c("_data", "_boxhead", "_stub_df", "_row_groups", "_heading", "_spanners", "_stubhead", "_footnotes",
"_source_notes", "_formats", "_substitutions", "_styles", "_summary", "_options", "_transforms",
"_locale", "_has_built")
)
})
test_that("as_gt passes table body correctly", {
# tbl_summary
expect_equal(
my_tbl_summary$table_body,
gt_tbl_summary$`_data`
)
# tbl_cross
my_tbl_cross <- tbl_cross(trial, trt, grade)
expect_silent(gt_tbl_cross <- my_tbl_cross |> as_gt())
expect_equal(
my_tbl_cross$table_body |> as_tibble(),
gt_tbl_cross$`_data`
)
# tbl_regression
expect_equal(
my_tbl_regression$table_body,
gt_tbl_regression$`_data`,
ignore_attr = "class"
)
# tbl_uvregression
my_tbl_uvregression <- trial |> tbl_uvregression(method = lm, y = age)
expect_silent(gt_tbl_uvregression <- my_tbl_uvregression |> as_gt())
expect_equal(
my_tbl_uvregression$table_body,
gt_tbl_uvregression$`_data`,
ignore_attr = "class"
)
# spanning_tbl
expect_equal(
my_spanning_tbl$table_body,
gt_spanning_tbl$`_data`
)
})
test_that("as_gt works with bold/italics", {
tbl <- my_tbl_summary |>
bold_labels() |>
italicize_levels()
gt_tbl <- tbl |> as_gt()
gt_styles <- gt_tbl$`_styles` |>
dplyr::arrange(rownum) |>
dplyr::pull(styles) |>
unlist(use.names = FALSE)
# labels correctly bolded
expect_equal(
eval_tidy(tbl$table_styling$text_format$rows[[1]], data = tbl$table_body),
gt_styles == "bold"
)
# labels correctly italicized
expect_equal(
eval_tidy(tbl$table_styling$text_format$rows[[2]], data = tbl$table_body),
gt_styles == "italic"
)
})
test_that("as_gt passes table header labels correctly", {
# tbl_summary
expect_equal(
my_tbl_summary$table_styling$header$column,
gt_tbl_summary$`_boxhead`$var
)
expect_equal(
my_tbl_summary$table_styling$header$label,
gt_tbl_summary$`_boxhead`$column_label |> unlist()
)
# spanning_tbl
expect_equal(
my_spanning_tbl$table_styling$header$column,
gt_spanning_tbl$`_boxhead`$var
)
expect_equal(
my_spanning_tbl$table_styling$header$label,
gt_spanning_tbl$`_boxhead`$column_label |> unlist()
)
# spanning_tbl - spanning header
expect_equal(
my_spanning_tbl$table_styling$header |>
dplyr::filter(!is.na(spanning_header)) |>
dplyr::pull(column),
gt_spanning_tbl$`_spanners`$vars[[1]]
)
})
test_that("as_gt passes table column visibility correctly", {
expect_equal(
my_tbl_regression$table_styling$header |>
dplyr::filter(!hide) |>
dplyr::pull(column),
gt_tbl_regression$`_boxhead` |>
dplyr::filter(type != "hidden") |>
dplyr::pull(var)
)
# customize visibility
tbl <- my_tbl_regression |>
modify_table_styling(columns = "estimate", hide = TRUE) |>
modify_table_styling(columns = "N_obs", hide = FALSE)
gt_tbl <- tbl |> as_gt()
expect_equal(
tbl$table_styling$header |>
dplyr::filter(!hide) |>
dplyr::pull(column),
gt_tbl$`_boxhead` |>
dplyr::filter(type != "hidden") |>
dplyr::pull(var)
)
})
test_that("as_gt passes table column alignment correctly", {
expect_equal(
my_tbl_regression$table_styling$header |>
dplyr::filter(!hide) |>
dplyr::pull(align),
gt_tbl_regression$`_boxhead` |>
dplyr::filter(type != "hidden") |>
dplyr::pull(column_align)
)
# customize alignment
tbl <- my_tbl_regression |>
modify_table_styling(columns = "estimate", align = "right")
gt_tbl <- tbl |> as_gt()
expect_equal(
tbl$table_styling$header |>
dplyr::filter(!hide) |>
dplyr::pull(align),
gt_tbl$`_boxhead` |>
dplyr::filter(type != "hidden") |>
dplyr::pull(column_align)
)
})
test_that("as_gt passes table text interpreters correctly", {
# header
expect_equal(
lapply(
my_spanning_tbl$table_styling$header$interpret_label,
\(x) do.call(eval(parse(text = x)), list("")) |> class()
),
lapply(gt_spanning_tbl$`_boxhead`$column_label, class)
)
# spanning header
expect_equal(
sapply(
my_spanning_tbl$table_styling$header |>
dplyr::filter(!is.na(spanning_header)) |>
dplyr::pull(interpret_spanning_header),
\(x) do.call(eval(parse(text = x)), list("")) |> class()
),
gt_spanning_tbl$`_spanners`$spanner_label[[1]] |> class() |>
rep(length(gt_spanning_tbl$`_spanners`$vars[[1]])),
ignore_attr = "names"
)
# customize interpreter
tbl <- my_spanning_tbl |>
modify_table_styling(columns = "stat_1", label = "Stat I", spanning_header = "MyTest", text_interpret = "html")
gt_tbl <- tbl |> as_gt()
# header
expect_equal(
lapply(
tbl$table_styling$header$interpret_label,
\(x) do.call(eval(parse(text = x)), list("")) |> class()
),
lapply(gt_tbl$`_boxhead`$column_label, class)
)
# spanning header
expect_true(attr(gt_tbl$`_spanners`$spanner_label[[1]], "html"))
})
test_that("as_gt passes table footnotes & footnote abbreviations correctly", {
tbl_fn <- my_tbl_summary |>
modify_table_styling(columns = label, footnote = "test footnote", rows = variable == "age")
gt_tbl_fn <- tbl_fn |> as_gt()
# footnote
expect_equal(
tbl_fn$table_styling$footnote$column,
gt_tbl_fn$`_footnotes`$colname |> unique()
)
expect_equal(
tbl_fn$table_styling$footnote$footnote,
gt_tbl_fn$`_footnotes`$footnotes |> unlist() |> unique()
)
tbl_fa <- tbl_fn |>
modify_footnote(stat_0 = "N = number of observations", abbreviation = TRUE)
gt_tbl_fa <- tbl_fa |> as_gt()
# footnote_abbrev
expect_equal(
gt_tbl_fa$`_footnotes` |>
dplyr::distinct(pick(!any_of("rownum"))) |>
dplyr::arrange(locnum) |>
dplyr::pull(colname),
c("stat_0", "stat_0", "label")
)
expect_equal(
gt_tbl_fa$`_footnotes` |>
dplyr::distinct(pick(!any_of("rownum"))) |>
dplyr::arrange(locnum) |>
dplyr::pull(footnotes) |>
unlist(),
c("n (%); Median (Q1, Q3)", "N = number of observations", "test footnote")
)
# customized footnotes
tbl <- my_tbl_summary |>
modify_footnote(
all_stat_cols() ~ "replace old footnote",
label = "another new footnote"
)
gt_tbl <- tbl |> as_gt()
expect_equal(
gt_tbl$`_footnotes`$colname,
c("stat_0", "label")
)
expect_equal(
gt_tbl$`_footnotes`$footnotes |> unlist(),
c("replace old footnote", "another new footnote")
)
# footnotes in the body of the table
expect_equal(
tbl_summary(trial, include = "age") |>
modify_table_styling(columns = label, rows = TRUE, footnote = "my footnote") |>
modify_table_styling(columns = stat_0, rows = row_type == "label", footnote = "my footnote") |>
as_gt() |>
getElement("_footnotes") |>
dplyr::filter(footnotes == "my footnote") |>
dplyr::select(colname, rownum),
data.frame(
colname = c("label", "label", "stat_0"),
rownum = c(1, 2, 1)
)
)
})
test_that("as_gt passes table indentation correctly", {
expect_equal(
my_tbl_summary$table_styling$indent$n_spaces[2],
gt_tbl_summary$`_transforms`[[1]]$fn("") |> nchar()
)
# indentation removed
tbl <- my_tbl_summary |>
modify_column_indent(columns = label, indent = 0)
gt_tbl <- tbl |> as_gt()
expect_equal(
gt_tbl$`_transforms` |> length(),
0
)
})
test_that("as_gt passes appended glance statistics correctly", {
tbl <- my_tbl_regression |>
add_glance_source_note(c("r.squared", "BIC")) |>
add_glance_table(c("r.squared", "BIC"))
gt_tbl <- tbl |> as_gt()
loc_hline <- tbl$table_body |>
rownames_to_column() |>
dplyr::filter(!!tbl$table_styling$horizontal_line_above) |>
dplyr::pull(rowname) |>
as.numeric()
expect_equal(
tbl$table_body |> dplyr::select(-conf.low),
gt_tbl$`_data` |> dplyr::select(-conf.low),
ignore_attr = "class"
)
expect_equal(
tbl$table_styling$source_note$source_note,
gt_tbl$`_source_notes`[[1]],
ignore_attr = "class"
)
expect_equal(
loc_hline,
gt_tbl$`_styles`$rownum |> unique()
)
})
test_that("as_gt passes captions correctly", {
tbl <- my_tbl_regression |>
modify_caption("My table caption")
gt_tbl <- tbl |> as_gt()
expect_equal(
tbl$table_styling$caption,
(gt_tbl$`_options` |> dplyr::filter(parameter == "table_caption"))$value[[1]],
ignore_attr = "class"
)
})
test_that("as_gt passes missing symbols correctly", {
tbl <- my_tbl_summary |>
modify_table_body(~ .x |> mutate(stat_0 = NA_character_))
gt_tbl <- tbl |> as_gt()
# no substitution for missing values
expect_equal(
length(gt_tbl$`_substitutions`),
1
)
# specify missing symbol
tbl <- tbl |>
modify_table_styling(stat_0, rows = !is.na(label), missing_symbol = "n / a")
gt_tbl <- tbl |> as_gt()
# correct substitution for missing values
expect_equal(
tbl |>
as_tibble(col_labels = FALSE, fmt_missing = TRUE) |>
dplyr::pull(stat_0),
gt_tbl$`_substitutions`[[2]]$func$default(gt_tbl$`_data`$stat_0)
)
})
test_that("as_gt applies formatting functions correctly", {
tbl <- glm(response ~ age + grade, trial, family = binomial(link = "logit")) |>
tbl_regression(exponentiate = TRUE) |>
modify_fmt_fun(
p.value ~ function(x) style_pvalue(x, digits = 3),
rows = variable == "grade"
) |>
modify_fmt_fun(
estimate ~ function(x) style_ratio(x, digits = 4, decimal.mark = ",")
)
gt_tbl <- tbl |> as_gt()
# formatted cells
expect_equal(
gt_tbl$`_formats`[[12]]$func$default(gt_tbl$`_data`$p.value),
c("0.096", NA, NA, "0.688", "0.972")
)
# formatted column
expect_equal(
gt_tbl$`_formats`[[14]]$func$default(gt_tbl$`_data`$estimate),
c("1,0191", NA, NA, "0,8535", "1,0136")
)
tbl2 <- tbl_uvregression(
trial |> dplyr::select(response, age, grade),
method = glm,
y = response,
method.args = list(family = binomial),
exponentiate = TRUE
) |>
modify_fmt_fun(
stat_n ~ function(x) style_number(x, digits = 2),
rows = variable == "age"
) |>
modify_fmt_fun(
stat_n ~ label_style_number(digits = 4),
rows = variable == "grade"
) |>
modify_fmt_fun(
c(conf.low, conf.high) ~ label_style_sigfig(digits = 3)
)
gt_tbl2 <- tbl2 |> as_gt()
# formatted cell
expect_equal(
gt_tbl2$`_formats`[[22]]$func$default(gt_tbl2$`_data`$stat_n),
c("183.0000", "193.0000", NA, NA, NA)
)
# formatted column
expect_equal(
gt_tbl2$`_formats`[[23]]$func$default(gt_tbl2$`_data`$conf.low),
c("0.997", NA, NA, "0.446", "0.524")
)
expect_equal(
gt_tbl2$`_formats`[[24]]$func$default(gt_tbl2$`_data`$conf.high),
c("1.04", NA, NA, "2.00", "2.29")
)
})
test_that("as_gt passes column merging correctly", {
tbl <- lm(marker ~ age + grade, trial) |>
tbl_regression() |>
modify_column_merge(
pattern = "{estimate} (pval {p.value})",
rows = !is.na(estimate) & estimate < 0
)
gt_tbl <- tbl |> as_gt()
# conf.low (default column merging)
expect_equal(
eval_tidy(tbl$table_styling$cols_merge$rows[[1]], data = tbl$table_body) |> which(),
gt_tbl$`_col_merge`[[1]]$rows
)
expect_equal(
gt_tbl$`_col_merge`[[1]]$vars,
c("conf.low", "conf.high")
)
expect_equal(
gt_tbl$`_col_merge`[[1]]$pattern,
c("{1}, {2}")
)
# estimate (added custom column merging)
expect_equal(
eval_tidy(tbl$table_styling$cols_merge$rows[[2]], data = tbl$table_body) |> which(),
gt_tbl$`_col_merge`[[2]]$rows
)
expect_equal(
gt_tbl$`_col_merge`[[2]]$vars,
c("estimate", "p.value")
)
expect_equal(
gt_tbl$`_col_merge`[[2]]$pattern,
c("{1} (pval {2})")
)
expect_equal(
as.data.frame(gt_tbl)$estimate,
c("0.00 (pval >0.9)", "
", "—", "-0.38 (pval 0.015)", "-0.12 (pval 0.5)")
)
# modify column merging pattern
tbl <- tbl |>
modify_table_styling(
columns = estimate,
rows = !is.na(estimate) & estimate < 0,
cols_merge_pattern = "{estimate} (p is {p.value})"
)
gt_tbl <- tbl |> as_gt()
expect_equal(
gt_tbl$`_col_merge`[[2]]$pattern,
c("{1} (p is {2})")
)
expect_equal(
as.data.frame(gt_tbl)$estimate,
c("0.00 (p is >0.9)", "
", "—", "-0.38 (p is 0.015)", "-0.12 (p is 0.5)")
)
# remove column merging
tbl <- tbl |>
modify_table_styling(
columns = estimate,
cols_merge_pattern = NA
)
gt_tbl <- tbl |> as_gt()
expect_equal(length(gt_tbl$`_col_merge`), 1)
expect_equal(
as.data.frame(gt_tbl)$estimate,
c("0.00", "
", "—", "-0.38", "-0.12")
)
})