forked from MathiasHarrer/Doing-Meta-Analysis-in-R
-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy path17-risk-of-bias-plots.Rmd
615 lines (366 loc) · 23.6 KB
/
17-risk-of-bias-plots.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
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
# Risk of Bias Plots {#risk-of-bias-plots}
**by Luke A. McGuinness**
---
<img src="_figs/traffic_light.jpg" />
```{block2, type='boxempty'}
**Please cite this chapter as:**
McGuinness, L. A. (2021). Risk of Bias Plots. In Harrer, M., Cuijpers, P., Furukawa, T.A., & Ebert, D.D., _Doing Meta-Analysis with R: A Hands-On Guide_ (online version). https://bookdown.org/MathiasHarrer/Doing_Meta_Analysis_in_R/rob-plots.html.
```
<br></br>
<span class="firstcharacter">I</span>
n this chapter, we will describe how to create risk of bias plots in _R_, using the **{robvis}** package.
<br></br>
## Introduction
---
As part of a systematic review and meta-analysis, you may also want to examine the internal validity (risk of bias) of included studies using the relevant [domain-based risk of bias assessment tool](https://handbook-5-1.cochrane.org/chapter_8/8_3_1_types_of_tools.htm), and present the results of this assessment in a graphical format.
The Cochrane Handbook recommends two types of figure: a summary barplot figure showing the proportion of studies with a given risk of bias judgement within each domain, and a traffic light plot which presents the domain level judgments for each study.
However, the options available to researchers when creating these figures are limited. While RevMan has the functionality to create the plots, many researchers do not use it to conduct their systematic review and so copying the relevant data into the system is an inefficient solution.
Similarly, producing the graphs by hand, using software such as MS PowerPoint, is time consuming and means the figures have to manually updated if changes are needed. Additionally, journals usually require figures to be of publication quality (above ~300-400dpi), which can be hard to achieve when exporting the risk of bias figures from RevMan or creating them by hand.
```{r rob-revman, fig.cap="Example RevMan output.", out.width='75%', message = F, echo = F, fig.align='center'}
library(OpenImageR)
knitr::include_graphics('images/robsummaryrevman.jpg')
```
To avoid all of this, you can now easily plot the risk of bias figures yourself within R Studio, using the **{robvis}** package [@mcguinness2020risk; @robvis] which provides functions to convert a risk of bias assessment summary table into a summary plot or a traffic-light plot.
<br></br>
### Load **{robvis}**
---
Assuming that you have already installed the **{dmetar}** package (see Chapter \@ref(dmetar)), load the **{robvis}** package using:
```{r, message=F}
library(robvis)
```
<br></br>
### Importing Your Risk of Bias Summary Table Data
---
To produce our plots, we first have to import the results of our risk of bias assessment from **Excel** into _R_. Please note that **{robvis}** expects certain facts about the data you provide it, so be sure to follow the guidance below when setting up your table in Excel:
1. The first column is labelled “Study” and contains the study identifier (e.g. **Anthony et al, 2019**)
2. The second-to-last column is labelled “Overall” and contains the overall risk-of-bias judgments
3. The last column is labelled “Weight” and contains some measure of study precision e.g. the weight assigned to each study in the meta-analysis, or if no meta-analysis was performed, the sample size of each study). See Chapter \@ref(fem) for more details.
4. All other columns contain the results of the risk-of bias assessment for a specific domain.
To elaborate on the above guidance, consider as an example the ROB2 tool which has 5 domains. The resulting data set that **{robvis}** would expect for this tool would have 8 columns:
* **Column 1**: Study identifier
* **Column 2-6**: One RoB2 domain per column
* **Column 7**: Overall risk-of-bias judgments
* **Column 8**: Weight.
In Excel, this risk of bias summary table would look like this:
```{r rob-example-data, out.width='75%', message = F, echo = F, fig.align='center'}
library(OpenImageR)
knitr::include_graphics('images/rob_excel.png')
```
```{block, type='boximportant'}
**Column Names**
For three of the four tool templates (ROB2, ROBINS-I, QUADAS-2), what you name the columns containing the domain-level judgments is not important, as the templates within robvis will relabel each domain with the correct tool-specific heading.
```
Once you have saved the table you created in Excel to the working directory as a comma-separated-file (e.g. “robdata.csv”), you can either read the file into _R_ programmatically using the command below or via the “import assistant” method as described in Chapter \@ref(data-prep-R).
```{r, eval=F}
my_rob_data <- read.csv("robdata.csv", header = TRUE)
```
<br></br>
### Templates
---
**{robvis}** produces the risk of bias figures by using the data you provide to populate a template figure specific to the risk of bias assessment tool you used. At present, **{robvis}** contains templates for the following three tools:
* **ROB2**, the new Cochrane risk of bias tool for randomized controlled trials;
* **ROBINS-I**, the Risk of Bias In Non-randomized Studies - of Interventions tool;
* **QUADAS-2**, the Quality and Applicability of Diagnostic Accuracy Studies, Version 2.
**{robvis}** also contains a special generic template, labeled as ROB1. Designed for use with the original Cochrane risk of bias tool for randomized controlled trials, it can also be used to visualize the results of assessments performed with other domain-based tools not included in the list above. See Chapter \@ref(rob1-template) for more information on the additional steps required when using this template.
<br></br>
### Example Data Sets
---
The **{robvis}** package contains an example data set for each template outlined above. These are stored in the following objects:
* `data_rob2`: Example data for the ROB2 tool
* `data_robins`: Example data for the ROBINS-I tool
* `data_quadas`: Example data for the QUADAS-2 tool
* `data_rob1`: Example data for the RoB-1 tool.
You can explore these data sets using the `glimpse` function (see Chapter \@ref(class-conversion)). For example, once you have loaded the package using `library(robvis)`, viewing the ROBINS-I example data set can be achieved by running the following command:
```{r, message=F, eval=F}
glimpse(data_robins)
```
```{r, message=F, echo=F}
dplyr::glimpse(data_robins)
```
These example data sets are used to create the plots presented through the remainder of this guide.
<br></br>
## Summary Plots
---
### Basics
---
Once we have successfully imported the risk of bias summary table into _R_, creating the risk of bias figures is quite straightforward.
To get started, a simple weighted summary bar plot using the ROB2 example data set (`data_rob2`) is created by running the following code:
```{r, fig.width=9, fig.height=2.5, fig.align='center', out.width='90%'}
rob_summary(data = data_rob2,
tool = "ROB2")
```
<br></br>
### Modifying the Plot
---
The `rob_summary` function has the following parameters:
* `data`. A data frame containing summary (domain) level risk-of-bias assessments, with the first column containing the study details, the second column containing the first domain of your assessments, and the final column containing a weight to assign to each study. The function assumes that the data includes a column for overall risk-of-bias. For example, a ROB2.0 dataset would have 8 columns (1 for study details, 5 for domain level judgments, 1 for overall judgments, and 1 for weights, in that order).
* `tool`. The risk of bias assessment tool used. RoB2.0 (`"ROB2"`), `"ROBINS-I"`, and `"QUADAS-2"` are currently supported.
* `overall`. An option to include an additional bar for overall risk-of-bias in the figure. Default is `FALSE`.
* `weighted`. An option to specify whether weights should be used in the bar plot. Default is `TRUE`, in line with current Cochrane Collaboration guidance.
* `colour`. An argument to specify the colour scheme for the plot. Default is `"cochrane"`, which used the ubiquitous Cochrane colours, while a preset option for a colour-blind friendly palette is also available (`colour = "colourblind"`).
* `quiet`. A logical option to quietly produce the plot without displaying it. Default is `FALSE`.
Examples of the functionality of each argument are described below.
<br></br>
#### Tool
---
An argument to define the tool template you wish to use. In the example above, the ROB2 template is used. The two other primary templates - the ROBINS-I and QUADAS-2 templates - are demonstrated below:
```{r, fig.width=9, fig.height=3, fig.align='center', out.width='90%'}
rob_summary(data = data_robins,
tool = "ROBINS-I")
```
```{r, fig.width=9, fig.height=2.5, fig.align='center', out.width='90%'}
rob_summary(data = data_quadas,
tool = "QUADAS-2")
```
<br></br>
#### Overall
---
By default, an additional bar representing the overall risk of bias judgments is not included in the plot. If you would like to include this, set `overall = TRUE`. For example:
```{r, fig.width=9, fig.height=2.5, fig.align='center', out.width='90%'}
rob_summary(data = data_rob2,
tool = "ROB2",
overall = TRUE)
```
<br></br>
#### Weighted or Unweighted Bar Plots
---
By default, the bar plot is weighted by some measure of study precision, so that the bar plot shows the proportion of information rather than the proportion of studies that is at a particular risk of bias. This approach is in line with the [Cochrane Handbook](https://training.cochrane.org/handbook/current/chapter-07#section-7-4).
You can turn off this option by setting `weighted = FALSE` to create an unweighted bar plot. For example, compare the following two plots:
```{r, fig.width=9, fig.height=2.5, fig.align='center', out.width='90%'}
rob_summary(data = data_rob2,
tool = "ROB2")
```
```{r, fig.width=9, fig.height=2.5, fig.align='center', out.width='90%'}
rob_summary(data = data_rob2,
tool = "ROB2",
weighted = FALSE)
```
<br></br>
#### Colour Scheme
---
```{block2, type='boximportant'}
**British English Spelling**
Please note the non-US English spelling of **colour**!
```
The `colour` argument of both plotting functions allows users to select from two predefined colour schemes, `"cochrane"` (default) or `"colourblind"`, or to define their own palette by providing a vector of **hex codes**. For example, to use the predefined `"colourblind"` palette:
```{r, fig.width=9, fig.height=2.5, fig.align='center', out.width='90%'}
rob_summary(data = data_rob2,
tool = "ROB2",
colour = "colourblind")
```
And to define your own colour scheme:
```{r, fig.width=9, fig.height=2.5, fig.align='center', out.width='90%'}
rob_summary(data = data_rob2,
tool = "ROB2",
colour = c("#f442c8","#bef441","#000000"))
```
When defining your own colour scheme, you must ensure that the number of discrete judgments (e.g. “Low”, “Moderate”, “High”, “Critical”) and the number of colours specified are the same. Additionally, colours must be specified in order of ascending risk-of-bias (e.g. “Low” to “Critical”), with the first hex corresponding to “Low” risk of bias.
<br></br>
## Traffic Light Plots
---
Frequently, researchers will want to present the risk of bias in each domain for each study assessed. The resulting plots are commonly called traffic light plots, and can be produced with **{robvis}** via the `rob_traffic_light` function.
<br></br>
### Basics
---
To get started, a traffic light plot using the ROB2 example dataset (`data_rob2`) is created by running the following code:
```{r, fig.width=8, fig.height=10, fig.align='center', out.width='65%'}
rob_traffic_light(data = data_rob2,
tool = "ROB2")
```
<br></br>
### Modifying the Plot
---
The `rob_summary` function has the following parameters:
* `data`. A data frame containing summary (domain) level risk-of-bias assessments, with the first column containing the study details, the second column containing the first domain of your assessments, and the final column containing a weight to assign to each study. The function assumes that the data includes a column for overall risk-of-bias. For example, a ROB2.0 data set would have 8 columns (1 for study details, 5 for domain level judgments, 1 for overall judgments, and 1 for weights, in that order).
* `tool`. The risk of bias assessment tool used. RoB2.0 (`"ROB2"`), `"ROBINS-I"`, and `"QUADAS-2"` are currently supported.
* `colour`. An argument to specify the colour scheme for the plot. Default is `"cochrane"` which used the ubiquitous Cochrane colours, while a preset option for a colour-blind friendly palette is also available (`"colourblind"`).
* `psize`. An option to change the size of the "traffic light" points. Default is `20`.
* `quiet`. A logical option to quietly produce the plot without displaying it. Default is `FALSE`.
<br></br>
#### Tool
---
An argument to define the tool template you wish to use. The ROB2 template is demonstrated and the two other primary templates - the ROBINS-I and QUADAS-2 templates - are displayed below:
```{r, fig.width=10, fig.height=11, fig.align='center', out.width='65%'}
rob_traffic_light(data = data_robins,
tool = "ROBINS-I")
```
```{r, fig.width=8, fig.height=11, fig.align='center', out.width='65%'}
rob_traffic_light(data = data_quadas,
tool = "QUADAS-2")
```
<br></br>
#### Colour Scheme
---
```{block2, type='boximportant'}
**British English Spelling**
Please note the non-US English spelling of **colour**!
```
The `colour` argument of both plotting functions allows users to select from two predefined colour schemes, `"cochrane"` (default) or `"colourblind"`, or to define their own palette by providing a vector of hex codes.
For example, to use the predefined `"colourblind"` palette:
```{r, fig.width=8, fig.height=9, fig.align='center', out.width='65%'}
rob_traffic_light(data = data_rob2,
tool = "ROB2",
colour = "colourblind")
```
And to define your own colour scheme:
```{r, fig.width=8, fig.height=9, fig.align='center', out.width='65%'}
rob_traffic_light(data = data_rob2,
tool = "ROB2",
colour = c("#f442c8","#bef441","#000000"))
```
When defining your own colour scheme, you must ensure that the number of discrete judgments (e.g. “Low”, “Moderate”, “High”, “Critical”) and the number of colours specified are the same. Additionally, colours must be specified in order of ascending risk-of-bias (e.g. “Low” to “Critical”), with the first hex corresponding to “Low” risk of bias.
<br></br>
#### Point Size
---
Occasionally, when a large number of risk of bias assessment have been performed, the resulting traffic light plot may be too long to be useful. Users can address this by modifying the `psize` argument of the `rob_traffic_light` function to a smaller number (default is `20`). For example:
```{r, fig.width=5.5, fig.height=13, fig.align='center', out.width='45%'}
# Create bigger dataset (18 studies)
new_rob2_data <- rbind(data_rob2, data_rob2)
new_rob2_data$Study <- paste("Study", seq(1:length(new_rob2_data$Study)))
# Plot bigger dataset, reducing the psize argument from 20 to 8
rob_traffic_light(data = new_rob2_data,
tool = "ROB2",
psize = 8)
```
<br></br>
## "ROB1" Generic Template {#rob1-template}
---
### Motivation
---
This template offers increased flexibility in the domains that are included in the plot. It can handle any number of domains (cf. the other tool templates that have a set number of domains) and uses the user-defined column headings as domain titles in the resulting figures.
<br></br>
### Varying Numbers of Domains
---
The "ROB1" template (`tool = "ROB1"`) can handle varying numbers of columns. This was originally designed for use with the ROB1 assessment tool, to which frequently added or removed domains. **While this template could be used to present the result of assessments performed using adjusted versions of the other tools (ROB2, QUADAS-2, ROBINS-I), we would strongly discourage authors from doing so**. Authors using other published tools should use the stricter templates presented in the previous chapters to ensure they conform with the guidance.
<br></br>
### Domain Names
---
For the other tools listed in the previous sections, the names of the columns containing the domain-level risk of bias judgments are not important. For example, they are commonly named _D1_, _D2_, _D3_, etc. However, this is not the case when using the `"ROB1"` template.
Compare the column headings of the data_rob2 and the data_rob1 (presented horizontally here for ease of comparison):
```{r, echo=F, message=FALSE}
library(kableExtra)
k<-c(colnames(robvis::data_rob2),".",".")
kk<-colnames(robvis::data_rob1)
kkk<-seq(1:10)
ms3 <- data.frame(colnum = kkk, data_rob2 = k, data_rob1 = kk)
kableExtra::kable(
list(
ms3[,c(1,2)],
ms3[,c(1,3)]
), col.names = c("No.", "Column name"),
longtable = T,
booktabs = T,
caption = 'Comparison of column names in the `data_rob2` (left) and `data_rob1` (right) datasets.'
) %>%
kable_styling()
```
The domain columns (Columns 2-6) in the ROB2 example dataset have been given arbitrary names of _D1_ - _D5_, as they will be overwritten by the tool to correspond to the correct domain titles given by the ROB2 guidance.
In contrast, the domain columns (Columns 2-8) in the ROB1 example dataset are labelled correctly, as these will be used in the figures produced by `rob_summary` and `rob_traffic_light`.
As an example, suppose we change the name of the “Random.sequence.generation” column to “This is a test”. In the `rob_summary` figure, the title of the first bar is changed, while in the `rob_traffic_light` figure, the caption is updated to reflect this change.
```{r, fig.width=9, fig.height=2.5, fig.align='center', out.width='90%'}
# Create copy of the data_rob1 dataset
new_rob1_data <- data_rob1
# Change the column heading for the first domain
colnames(new_rob1_data)[2] <- "This is a test"
# Create the summary barplot
rob_summary(data = new_rob1_data, tool = "ROB1")
```
```{r, fig.width=10, fig.height=11, fig.align='center', out.width='65%'}
# Create the traffic light plot
rob_traffic_light(data = new_rob1_data,
tool = "ROB1")
```
<br></br>
## Costumizing and Saving
---
### The **{ggplot2}** Package
---
Both **{robvis}** functions (`rob_summary` and `rob_traffic_light`) produce a `ggplot` object, and so can be customized and saved using functions from the **{ggplot2}** package. Use the following code to load this package:
```{r, message=F}
library(ggplot2)
```
<br></br>
### Modifying Your Plots
---
There are a range of post-production modifications you can make to you plots using **{ggplot2}** functions. A useful example is adding a title to the plot:
```{r, fig.width=9, fig.height=2.5, fig.align='center', out.width='90%'}
# Make sure you have the ggplot2 package installed and loaded
rob_summary(data_rob2, "ROB2") +
ggtitle("Your custom title")
```
<br></br>
### Saving the Plot
---
In order to save a risk of bias plot, we first assign it to an object using the <- operator and then save it using the `ggsave` function of the **{ggplot2}** package.
When saving the summary bar plot, we recommend using the following code, with the default height and width values.
```{r, eval=F}
# Create your plot, and assign it to an object
rob_barplot <- rob_summary(data_rob2, "ROB2")
# Save your plot
ggsave(plot = rob_barplot, # Plot object to save
filename = "robplot2.png", # Destination file
width = 8, # Width of image (recommended)
height = 2.41, # Height of image (recommended)
dpi = 1000) # Resolution of image
```
When saving the traffic light plots, the approach is the same. However, there are no recommended values for the `width` and `height` parameters, as the best values for these parameters will vary from plot to plot as the number and names of included studies change.
<br></br>
### Saving in a Different Format
---
The plots can be saved in a range of formats using the function outlined above, simply by changing the extension of the file-name (e.g. from ".png" to ".pdf"). Acceptable formats include .png, .pdf, .tiff and .svg^[This format requires you to install and load the **{svglite}** package: `install.packages("svglite")`; `library(svglite)`.].
For example, to save the bar plot created above (`rob_barplot`) as a PDF:
```{r, eval=F}
# Save your plot
ggsave(plot = rob_barplot,
filename = "robplot2.pdf", # File extension now ".pdf"
width = 8,
height = 2.41,
dpi = 1000)
```
<br></br>
## Web App
---
In an effort to allow users to quickly explore the functionality of robvis, a web application was created which provides a graphically interface to the **{robvis}** package.
The web-app is available [here](https://mcguinlu.shinyapps.io/robvis). A brief guided walk-through is presented below.
<br></br>
### Landing Page
---
```{r, out.width='90%', message = F, echo = F, fig.align='center'}
library(OpenImageR)
knitr::include_graphics('images/robvis-app-landingpage.png')
```
The page presents a concise version of the guidance found in the previous chapters, specifically relating to setting up your dataset. More importantly, users can download the example datasets for each tool as a CSV file and use these to interact with the app and explore its functionality.
<br></br>
### Traffic Light Plot Page
---
Clicking on the second tab will bring you to the screen displayed below.
```{r, out.width='45%', message = F, echo = F, fig.align='center'}
library(OpenImageR)
knitr::include_graphics('images/robvis-app-traffic-light.png')
```
This menu acts as a graphical interface for the `rob_traffic_light` function:
* Upload your risk of bias summary table by clicking “Browse…” and navigating to where you stored your CSV file.
* Use the drop-down box to select the tool used to perform your risk of bias assessments.
The basic traffic light plot should now appear on the right hand side of the window. You can customize the plot using the following options:
* Select the colour scheme you wish to use (either “Cochrane” or “Colour-blind friendly”)
* Modify the point size (useful when you wish to plot a large number of studies on a single traffic light plot)
* Modify the text size.
Once you are happy with the plot, you can download it by selecting the required format (.png, .jpg,.tiff, .eps) and clicking the “Download plot” button. Note, if you do not first select a format, you will get a download error.
<br></br>
### Summary Plot Page
---
Clicking on the third tab will bring you to the screen displayed below.
```{r, out.width='45%', message = F, echo = F, fig.align='center'}
library(OpenImageR)
knitr::include_graphics('images/robvis-app-summary-plot.png')
```
**This menu acts as a graphical interface for the `rob_summary` function:**
* Upload your risk of bias summary table by clicking “Browse…” and navigating to where you stored your CSV file.
* Use the drop-down box to select the tool used to perform your risk of bias assessments.
The basic weighted summary bar plot should now appear on the right hand side of the window.
**You can customize the plot using the following options:**
* Choose whether or not to use weights when creating the figure
* Include an additional bar representing distribution of overall risk of bias judgments
* Select the colour scheme you wish to use (either “Cochrane” or “Colour-blind friendly”)
As with the traffic light plot tab, you can download your plot by selecting the required format and clicking the “Download plot” button.
$$\tag*{$\blacksquare$}$$