-
Notifications
You must be signed in to change notification settings - Fork 11
/
Copy path11a-plots-examples-scatter.Rmd
185 lines (161 loc) · 7.88 KB
/
11a-plots-examples-scatter.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
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
### Scatter Plot
#### Overview
This article will demonstrate how to construct a scatter plot in ggplot2. Recently, the FDA has published a document for a more standardized presentation of safety data (see below). We will use some sample figures from this guide for this section and the section about [safety plots][Safety Plots].
```{r out.height = "460px", out.width='800px', echo=F}
knitr::include_graphics("./additional resources/FDA_safety_guide.pdf")
```
```{r out.height = "460px", out.width='700px', echo=F, fig.cap = "Sample figure from the SSTFIG (Figure 17)"}
knitr::include_graphics("./additional resources/f17.png")
```
#### Packages and Sample Data
Let's start with accessing ADVS from the PhUSE Test Data Factory repository.
```{r, message = FALSE, warning = FALSE, error = FALSE}
# Packages
library(ggplot2)
library(dplyr)
# Data
advs <- haven::read_xpt(
paste0("https://github.com/phuse-org/TestDataFactory/",
"raw/main/Updated/TDF_ADaM/advs.xpt"))
```
#### Data Wrangling
In order to make this example meaningful, data wrangling and processing steps will be covered here. In practice, you might have other considerations or processing steps that need to be considered.
The first step is to obtain baseline values for each USUBJID for the parameter and analysis time point of interest.
```{r message = FALSE, warning = FALSE, error = FALSE}
baseline <- advs %>%
filter(
SAFFL == "Y",
PARAMCD == "SYSBP",
ATPT == "AFTER LYING DOWN FOR 5 MINUTES",
TRTA != "Placebo"
) %>%
dplyr::select(USUBJID, TRTA, BASE) %>%
distinct(USUBJID, .keep_all = TRUE)
```
The next step is to obtain the maximum value for each USUBJID post-baseline for the same parameter and analysis time point of interest.
```{r message = FALSE, warning = FALSE, error = FALSE}
post <- advs %>%
filter(
SAFFL == "Y",
PARAMCD == "SYSBP",
ATPT == "AFTER LYING DOWN FOR 5 MINUTES",
TRTA != "Placebo",
AVISIT != "Baseline",
ANL01FL == "Y"
) %>%
dplyr::select(USUBJID, AVAL) %>%
group_by(USUBJID) %>%
arrange(desc(AVAL)) %>%
slice(1) %>%
ungroup()
```
The last step is to merge these two data frames together.
```{r message = FALSE, warning = FALSE, error = FALSE}
all <- baseline %>%
left_join(post)
```
The resulting data frame should look like this.
```{r echo = F}
all %>%
head(n=5) %>%
kableExtra::kable(align = 'c')
```
#### Basic Plot
Now that we have some data, we can start with a basic scatter plot.
```{r message = FALSE, warning = FALSE, error = FALSE}
ggplot(data = all, aes(x = BASE, y = AVAL)) +
geom_point()
```
#### Customized Plot
We have a few customization to be included in order to match the figure provided in the SSTFIG. Below we separate the code and annotations in two different steps for clarity.
Let's setup the plot annotations first. This includes headers, footers, and axis text.
```{r message = FALSE, warning = FALSE, error = FALSE}
yaxis_text <- "\nMaximum Post-baseline Systolic Blood Pressure (mmHg)\n"
xaxis_text <- "\nBaseline Systolic Blood Pressure (mmHg)"
header <- "Figure 17. Baseline vs. Maximum Systolic Blood Pressure by Treatment Arm, Safety Population¹, Pooled Analysis\n"
footer1 <- "Source: [include Applicant source, datasets and/or software tools used]."
footer2 <- "¹Gray dotted line = no increase; blue line = high-dose treatment linear regression; orange dashed line = low-dose linear regression."
```
Next, let's extend our basic plot to include customizations we desire. One notable customization involves the use of `stat_smooth()`. This function will automatically compute the linear regressions for each treatment arm and append the lines of best fit to the plot.
```{r message = FALSE, warning = FALSE, error = FALSE, fig.fullwidth=TRUE, fig.width=12, fig.height=10}
f17 <- ggplot(all, aes(x = BASE, y = AVAL, linetype = TRTA)) +
geom_point(colour = "black", shape = 21, size = 4, alpha = 0.6, aes(fill = TRTA)) +
stat_smooth(method = "lm", se = FALSE, aes(color = TRTA)) +
scale_color_manual(values = c("skyblue", "orange")) +
scale_fill_manual(values = c("skyblue", "orange")) +
geom_abline(intercept = 0, slope = 1, size = 1, lty = "dotted", color = "#5A5A5A") +
scale_x_continuous(breaks = seq(90, 210, 30), limits = c(90, 210)) +
scale_y_continuous(breaks = seq(90, 210, 30), limits = c(90, 210)) +
labs(
y = yaxis_text,
x = xaxis_text,
title = header,
caption = paste(footer1, footer2, sep = "\n")
) +
theme_light() +
theme(
panel.background = element_rect(fill = NA, color = "skyblue3", size = 2, linetype = "solid"),
plot.caption = element_text(hjust = 0),
legend.position = "bottom",
legend.title = element_blank(),
axis.text = element_text(face = "bold")
)
f17
```
#### Confirmation
Using `stat_smoooth()` is a handy shortcut to achieve what we wanted. We should take some care to verify the results of this function are inline with what we expect. We can achieve this by manually running an `lm()`.
```{r message = FALSE, warning = FALSE, error = FALSE}
coef <- all %>%
filter(TRTA == "Xanomeline High Dose") %>%
lm(data = ., AVAL ~ BASE) %>%
coefficients()
coef
```
The next step is to see if this is congruent with what `stat_smooth()` provided in our plot. We'll plot the coefficients and check whether they overlap, which would indicate they are similar.
```{r message = FALSE, warning = FALSE, error = FALSE}
all %>%
filter(TRTA == "Xanomeline High Dose") %>%
ggplot(., aes(x = BASE, y = AVAL)) +
geom_point() +
stat_smooth(method = "lm", se = FALSE, size = 5) +
geom_abline(intercept = coef[[1]], slope = coef[[2]], color = "red")
```
Looks okay!
#### More faithful aesthetics
While the above plot matched the technical specification, it did differ in a few aesthetics namely colors, and the border of the main plot. Below is code to achieve a closer aesthetic match with what's in the SSTIF for the interested.
Line spaces have been removed from the beginning and end of text.
```{r message = FALSE, warning = FALSE, error = FALSE}
# setup plot text
yaxis_text_alt <- "Maximum Post-baseline Systolic Blood Pressure (mmHg)"
xaxis_text_alt <- "Baseline Systolic Blood Pressure (mmHg)"
header_alt <- "Figure 17. Baseline vs. Maximum Systolic Blood Pressure by Treatment Arm, Safety Population¹, Pooled Analysis"
footer1_alt <- "Source: [include Applicant source, datasets and/or software tools used]."
footer2_alt <- "¹Gray dotted line = no increase; blue line = high-dose treatment linear regression; grey dashed line = low-dose linear regression."
```
Colors and border contexts have changed. Title/footers have been re-factored using `patchwork::plot_annotation()`
```{r message = FALSE, warning = FALSE, error = FALSE, fig.fullwidth=TRUE, fig.width=12, fig.height=10}
f17_alt <- ggplot(all, aes(x = BASE, y = AVAL, linetype = TRTA)) +
geom_point(colour = "black", shape = 21, size = 4, alpha = 0.6, aes(fill = TRTA)) +
stat_smooth(method = "lm", se = FALSE, aes(color = TRTA)) +
scale_color_manual(values = c("skyblue", "grey")) +
scale_fill_manual(values = c("skyblue", "white")) +
geom_abline(intercept = 0, slope = 1, size = 1, lty = "dotted", color = "#5A5A5A") +
scale_x_continuous(breaks = seq(90, 210, 30), limits = c(90, 210)) +
scale_y_continuous(breaks = seq(90, 210, 30), limits = c(90, 210)) +
labs(
y = yaxis_text_alt,
x = xaxis_text_alt
) +
theme_bw() +
theme(
plot.background = element_rect(fill = NA, color = "skyblue3", size = 2, linetype = "solid"),
legend.position = "bottom",
legend.title = element_blank(),
axis.text = element_text(face = "bold")
)
# Use the patchwork package to specify title, caption (e.g. footers) and footers position (i.e. left align) to get a closer aesthetic feel.
library(patchwork)
f17_alt + patchwork::plot_annotation(title = header_alt,
caption = paste(footer1_alt, footer2_alt, sep = "\n"),
theme = theme(plot.caption = element_text(hjust = 0)))
```