suggested_dependent_pkgs <- c("dplyr", "tibble") knitr::opts_chunk$set( collapse = TRUE, comment = "#>", eval = all(vapply( suggested_dependent_pkgs, requireNamespace, logical(1), quietly = TRUE )) )
knitr::opts_chunk$set(comment = "#")
```{css, echo=FALSE} .reveal .r code { white-space: pre; }
## Introduction `TableTree` objects are based on a tree data structure as the name indicates. The package is written such that the user does not need to walk trees for many basic table manipulations. Walking trees will still be necessary for certain manipulation and will be the subject of a different vignette. In this vignette we show some methods to subset tables and to extract cell values. We will use the following table for illustrative purposes: ```r library(rtables) library(dplyr) lyt <- basic_table() %>% split_cols_by("ARM") %>% split_rows_by("SEX", split_fun = drop_split_levels) %>% analyze(c("AGE", "STRATA1")) tbl <- build_table(lyt, ex_adsl %>% filter(SEX %in% c("M", "F"))) tbl
[
The [
and [<-
accessor functions operate largely the same as their
data.frame
cousins:
[
and [<-
both treat Tables as rectangular objects (rather than
trees)[
accepts both column and row absolute position, and missing
arguments mean "all indexes in that dimension"[.data.frame
they cannot be mixed with positive ones[
always returns the same class as the object being subset unless
drop = TRUE
[ , drop = TRUE
returns the raw (possibly multi-element) value
associated with the cell.Known Differences from [.data.frame
- absolute position cannot currently be used to reorder columns or
rows. Note in general the result of such an ordering is unlikely to
be structurally valid. To change the order of values, please read
sorting and
pruning
vignette or relevant function (sort_at_path()
).
- character
indices are treated as paths, not vectors of names in
both [
and [<-
The [
accessor function always returns an TableTree
object if
drop=TRUE
is not set. The first argument are the row indices and the
second argument the column indices. Alternatively logical subsetting
can be used. The indices are based on visible rows and not on the tree
structure. So:
tbl[1, 1]
is a table with an empty cell because the first row is a label row. We need to access a cell with actual cell data:
tbl[3, 1]
To retrieve the value, we use drop = TRUE
:
tbl[3, 1, drop = TRUE]
One can access multiple rows and columns:
tbl[1:3, 1:2]
Note that we do not repeat label rows for descending children, e.g.
tbl[2:4, ]
does not show that the first row is derived from AGE
. In order to
repeat content/label information, one should use the pagination
feature. Please read the related vignette.
Character indices are interpreted as paths (see below), NOT elements
to be matched against names(tbl)
:
tbl[, c("ARM", "A: Drug X")]
As standard no additional information is kept after subsetting. Here, we show with a more complete table how it is still possible to keep the (possibly) relevant information.
top_left(tbl) <- "SEX" main_title(tbl) <- "Table 1" subtitles(tbl) <- c("Authors:", " - Abcd Zabcd", " - Cde Zbcd") main_footer(tbl) <- "Please regard this table as an example of smart subsetting" prov_footer(tbl) <- "Do remember where you read this though" fnotes_at_path(tbl, rowpath = c("M", "AGE", "Mean"), colpath = c("ARM", "A: Drug X")) <- "Very important mean"
Normal subsetting loses all the information showed above.
tbl[3, 3]
If all the rows are kept, top left information is also kept. This can be also
imposed by adding keep_topleft = TRUE
to the subsetting as follows:
tbl[, 2:3] tbl[1:3, 3, keep_topleft = TRUE]
If the referenced entry is present in the subsetting, also the referential footnote will appear. Please consider reading relevant vignette about referential footnotes. In case of subsetting, the referential footnotes are by default indexed again, as if the produced table is a new one.
tbl[10, 1] col_paths_summary(tbl) # Use these to find the right path to value or label row_paths_summary(tbl) # # To select column value, use `NULL` for `rowpath` fnotes_at_path(tbl, rowpath = NULL, colpath = c("ARM", "A: Drug X")) <- "Interesting" tbl[3, 1] # reindexing of {2} as {1} fnotes_at_path(tbl, rowpath = c("M", "AGE", "Mean"), colpath = NULL) <- "THIS mean" tbl # {1}, {2}, and {3} are present tbl[10, 2] # only {1} which was previously {2}
Similar to what we have used to keep top left information, we can specify to keep more information from the original table. As a standard the foot notes are always present if the titles are kept.
tbl[1:3, 2:3, keep_titles = TRUE] tbl[1:3, 2:3, keep_titles = FALSE, keep_footers = TRUE] # Referential footnotes are not influenced by `keep_footers = FALSE` tbl[1:3, keep_titles = TRUE, keep_footers = FALSE]
Tables can be subset or modified in a structurally aware manner via pathing.
Paths define semantically meaningful positions within a constructed table that correspond to the logic of the layout used to create it.
A path is an ordered set of split names, the names of subgroups
generated by the split, and the @content
directive, which steps into
a position's content (or row group summary) table.
We can see the row and column paths of an existing table via the
row_paths()
, col_paths()
, row_paths_summary()
, and
col_paths_summary()
, functions, or as a portion of the more general
make_row_df()
function output.
lyt2 <- basic_table() %>% split_cols_by("ARM") %>% split_cols_by("SEX", split_fun = drop_split_levels) %>% split_rows_by("RACE", split_fun = drop_split_levels) %>% summarize_row_groups() %>% analyze(c("AGE", "STRATA1")) tbl2 <- build_table(lyt2, ex_adsl %>% filter(SEX %in% c("M", "F") & RACE %in% (levels(RACE)[1:3]))) tbl2
So the column paths are as follows:
col_paths_summary(tbl2)
and the row paths are as follows:
row_paths_summary(tbl2)
To get a semantically meaningful subset of our table, then, we can use
[
(or tt_at_path()
which underlies it)
tbl2[c("RACE", "ASIAN"), c("ARM", "C: Combination")]
We can also retrieve individual cell-values via the value_at()
convenience function, which takes a pair of row and column paths which
resolve together to an individual cell, e.g. average age for Asian
female patients in arm A:
value_at(tbl2, c("RACE", "ASIAN", "AGE", "Mean"), c("ARM", "A: Drug X", "SEX", "F"))
You can also request information from non-cell specific paths with the
cell_values()
function:
cell_values(tbl2, c("RACE", "ASIAN", "AGE", "Mean"), c("ARM", "A: Drug X"))
Note the return value of cell_values()
is always a list even if you
specify a path to a cell:
cell_values(tbl2, c("RACE", "ASIAN", "AGE", "Mean"), c("ARM", "A: Drug X", "SEX", "F"))
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.