-
Notifications
You must be signed in to change notification settings - Fork 131
/
extending.Rmd
104 lines (78 loc) · 2.88 KB
/
extending.Rmd
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
---
title: "Extending tibble"
output: rmarkdown::html_vignette
vignette: >
%\VignetteIndexEntry{Extending tibble}
%\VignetteEngine{knitr::rmarkdown}
%\VignetteEncoding{UTF-8}
---
```{r setup, include = FALSE}
knitr::opts_chunk$set(collapse = TRUE, comment = "#>")
options(tibble.print_min = 4L, tibble.print_max = 4L)
library(tibble)
```
To extend the tibble package for new types of columnar data, you need to understand how printing works. The presentation of a column in a tibble is powered by two S3 generics:
* `type_sum()` determines what goes into the column header.
* `obj_sum()` is used when rendering list columns.
If you have written an S3 or S4 class that can be used as a column, you can override these generics to make sure your data prints well in a tibble. To start, you must import the `tibble` package. Either add `tibble` to the `Imports:` section of your `DESCRIPTION`, or simply call:
```{r, eval = FALSE}
devtools::use_package("tibble")
```
This vignette assumes a package that implements an S3 class `"foo"` and uses
`roxygen2` to create documentation and the `NAMESPACE` file:
```{r}
#' @export
foo <- function(x) {
stopifnot(is.numeric(x))
structure(x, class = "foo")
}
```
## `type_sum()`
This method should return a length-1 character vector that can be used in a column header. Strive for an evocative abbreviation that's under 6 characters.
```{r}
type_sum(1)
type_sum(1:10)
type_sum(Sys.time())
```
The default implementation works reasonably well for any kind of object,
but the generated output may be too wide and waste precious space when displaying the tibble:
```{r}
type_sum(foo(1:10))
```
To avoid this for provide a method for `type_sum()`:
```{r}
#' @export
type_sum.foo <- function(x, ...) {
"foo"
}
type_sum(foo(1:10))
```
## `obj_sum()`
This method is primarily used for displaying list columns. A list column is a powerful way to attach hierarchical or unstructured data to an observation in a data frame. Implementations of `obj_sum()` are expected to return a character vector as long as the input, with brief description of the contents of each input element.
Examples:
```{r}
obj_sum(1)
obj_sum(1:10)
obj_sum(Sys.time())
obj_sum(list(1:5))
obj_sum(list("a", "b", "c"))
```
The default implementation calls `type_sum()` and appends the size of the object in brackets. If your object is built on top of an atomic vector the default will be adequate. You, will, however, need to provide a method if your object is vectorised and built on top of a list.
An example of an object of this type in base R `POSIXlt`: it is a list with 9 components.
```{r}
x <- as.POSIXlt(Sys.time() + c(0, 60, 3600))
str(unclass(x))
```
But it pretends to be a vector with 3 elements:
```{r}
x
length(x)
str(x)
```
So we need to define a method that returns a character vector the same length as `x`:
```{r}
#' @export
obj_sum.POSIXlt <- function(x) {
rep("POSIXlt", length(x))
}
```