-
Notifications
You must be signed in to change notification settings - Fork 565
/
Copy pathaction-feedback.Rmd
879 lines (688 loc) · 34 KB
/
action-feedback.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
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
# User feedback {#action-feedback}
```{r, include = FALSE}
source("common.R")
source("demo.R")
```
You can often make your app more usable by giving the user more insight into what is happening.
This might take the form of better messages when inputs don't make sense, or progress bars for operations that take a long time.
Some feedback occurs naturally through outputs, which you already know how to use, but you'll often need something else.
The goal of this chapter is to show you some of your other options.
We'll start with techniques for **validation**, informing the user when an input (or combination of inputs) is in an invalid state.
We'll then continue on to **notification**, sending general messages to the user, and **progress bars**, which give details for time consuming operations made up of many small steps.
We'll finish up by discussing dangerous actions, and how you give your users peace of mind with **confirmation** dialogs or the ability to **undo** an action.
In this chapter we'll use [shinyFeedback](https://github.com/merlinoa/shinyFeedback), by Andy Merlino, and [waiter](http://waiter.john-coene.com/), by John Coene.
You should also keep your eyes open for [shinyvalidate](https://rstudio.github.io/shinyvalidate/), a package by Joe Cheng, that is currently under development.
```{r setup}
library(shiny)
```
## Validation {#validate}
The first and most important feedback you can give to the user is that they've given you bad input.
This is analogous to writing good functions in R: user-friendly functions give clear error messages describing what the expected input is and how you have violated those expectations.
Thinking through how the user might misuse your app allows you to provide informative messages in the UI, rather than allowing errors to trickle through into the R code and generate uninformative errors.
### Validating input
A great way to give additional feedback to the user is via the [shinyFeedback](https://github.com/merlinoa/shinyFeedback) package.
Using it is a two step process.
First, you add `useShinyFeedback()` to the `ui`.
This sets up the needed HTML and JavaScript for attractive error message display:
```{r}
ui <- fluidPage(
shinyFeedback::useShinyFeedback(),
numericInput("n", "n", value = 10),
textOutput("half")
)
```
Then in your `server()` function, you call one of the feedback functions: `feedback()`, `feedbackWarning()`, `feedbackDanger()`, and `feedbackSuccess()`.
They all have three key arguments:
- `inputId`, the id of the input where the feedback should be placed.
- `show`, a logical determining whether or not to show the feedback.
- `text`, the text to display.
They also have `color` and `icon` arguments that you can use to further customise the appearance.
See the documentation for more details.
Lets see how this comes together in a real example, pretending that we only want to allow even numbers.
Figure \@ref(fig:feedback) shows the results.
```{r}
server <- function(input, output, session) {
half <- reactive({
even <- input$n %% 2 == 0
shinyFeedback::feedbackWarning("n", !even, "Please select an even number")
input$n / 2
})
output$half <- renderText(half())
}
```
```{r feedback, fig.cap = demo$caption("Using `feedbackWarning()` to display warning for invalid inputs. The app on the left shows a valid input, the app on the right shows an invalid (odd) input with warning message."), echo = FALSE, message = FALSE, out.width = "50%", }
demo <- demoApp$new("action-feedback/feedback", ui, server, packages = "shinyFeedback", before = ~ shinyFeedback::useShinyFeedback())
demo$resize(width = 250)
demo$takeScreenshot("even")
demo$setInputs(n = 9)
demo$takeScreenshot("odd")
demo$deploy()
```
Notice that the error message is displayed, but the output is still updated.
Typically you don't want that because invalid inputs are likely to cause uninformative R errors that you don't want to show to the user.
To stop inputs from triggering reactive changes, you need a new tool: `req()`, short for "required".
It looks like this:
```{r}
server <- function(input, output, session) {
half <- reactive({
even <- input$n %% 2 == 0
shinyFeedback::feedbackWarning("n", !even, "Please select an even number")
req(even)
input$n / 2
})
output$half <- renderText(half())
}
```
When the input to `req()` is not true, it sends a special signal to tell Shiny that the reactive does not have all the inputs that it requires, so it should be "paused".
We'll take a brief digression to talk about this before we come back to using it in concert with `validate()`.
### Cancelling execution with `req()`
It's easiest to understand `req()` by starting outside of validation.
You may have noticed that when you start an app, the complete reactive graph is computed even before the user does anything.
This works well when you can choose meaningful default `value`s for your inputs.
But that's not always possible, and sometimes you want to wait until the user actually does something.
This tends to crop up with three controls:
- In `textInput()`, you've used `value = ""` and you don't want to do anything until the user types something.
- In `selectInput()`, you've provide an empty choice, `""`, and you don't want to do anything until the user makes a selection.
- In `fileInput()`, which has an empty result before the user has uploaded anything.
We'll come back to this in Section \@ref(upload).
We need some way to "pause" reactives so that nothing happens until some condition is true.
That's the job of `req()`, which checks for required values before allowing a reactive producer to continue.
For example, consider the following app which will generate a greeting in English or Maori.
If you run this app, you'll see an error, as in Figure \@ref(fig:require-simple), because there's no entry in the `greetings` vector that corresponds to the default choice of `""`.
```{r}
ui <- fluidPage(
selectInput("language", "Language", choices = c("", "English", "Maori")),
textInput("name", "Name"),
textOutput("greeting")
)
server <- function(input, output, session) {
greetings <- c(
English = "Hello",
Maori = "Kia ora"
)
output$greeting <- renderText({
paste0(greetings[[input$language]], " ", input$name, "!")
})
}
```
```{r require-simple, echo = FALSE, message = FALSE, cache = FALSE, fig.cap = "The app displays an uninformative error when it is loaded because a language hasn't been selected yet", out.width = "75%"}
demo <- demoApp$new("action-feedback/require-simple", ui, server)
demo$resize(400)
demo$takeScreenshot()
```
We can fix this problem by using `req()`, as below.
Now nothing will be displayed until the user has supplied values for both language and name, as shown in Figure \@ref(fig:require-simple2).
```{r}
server <- function(input, output, session) {
greetings <- c(
English = "Hello",
Maori = "Kia ora"
)
output$greeting <- renderText({
req(input$language, input$name)
paste0(greetings[[input$language]], " ", input$name, "!")
})
}
```
```{r require-simple2, fig.cap = demo$caption("By using `req()`, the output is only shown once both language and name have been supplied"), out.width = "33%", fig.show = "hold", fig.align = "default", echo = FALSE, message = FALSE, cache = FALSE}
demo <- demoApp$new("action-feedback/require-simple2", ui, server)
demo$resize(400)
s1 <- demo$takeScreenshot("on-load")
demo$setInputs(language = "English")
s2 <- demo$takeScreenshot("langauge")
demo$setInputs(name = "Hadley")
s3 <- demo$takeScreenshot("name")
demo$deploy()
knitr::include_graphics(c(s1, s2, s3))
```
`req()` works by signalling a special **condition**[^action-feedback-1].
This special condition causes all downstream reactives and outputs to stop executing.
Technically, it leaves any downstream reactive consumers in an invalidated state.
We'll come back to this terminology in Chapter \@ref(reactivity-components).
[^action-feedback-1]: "Condition" is a technical term that includes errors, warnings, and messages.
If you're interested, you can learn more of the details of R's condition system in <https://adv-r.hadley.nz/conditions.html>.
`req()` is designed so that `req(input$x)` will only proceed if the user has supplied a value, regardless of the type of the input control[^action-feedback-2].
You can also use `req()` with your own logical statement if needed.
For example, `req(input$a > 0)` will permit computation to proceed when `a` is greater than 0; this is typically the form you'll use when performing validation, as we'll see next.
[^action-feedback-2]: More precisely, `req()` proceeds only if its inputs are **truthy**, i.e. any value apart from `FALSE`, `NULL` , `""`, or a handful of other special cases described in `?isTruthy`.
### `req()` and validation
Let's combine `req()` and shinyFeedback to solve a more challenging problem.
I'm going to return to the simple app we made in Chapter \@ref(basic-app) which allowed you to select a built-in dataset and see its contents.
I'm going to make it more general and more complex by using `textInput()` instead of `selectInput()`.
The UI changes very little:
```{r}
ui <- fluidPage(
shinyFeedback::useShinyFeedback(),
textInput("dataset", "Dataset name"),
tableOutput("data")
)
```
But the server function needs to get a little more complex.
We're going to use `req()` in two ways:
- We only want to proceed with computation if the user has entered a value so we do `req(input$dataset)`.
- Then we check to see if the supplied name actually exists.
If it doesn't, we display an error message, and then use `req()` to cancel computation.
Note the use of `cancelOutput = TRUE`: normally cancelling a reactive will reset all downstream outputs; using `cancelOutput = TRUE` leaves them displaying the last good value.
This is important for `textInput()` which may trigger an update while you're in the middle of typing a name.
The results are shown in Figure \@ref(fig:require-cancel).
```{r}
server <- function(input, output, session) {
data <- reactive({
req(input$dataset)
exists <- exists(input$dataset, "package:datasets")
shinyFeedback::feedbackDanger("dataset", !exists, "Unknown dataset")
req(exists, cancelOutput = TRUE)
get(input$dataset, "package:datasets")
})
output$data <- renderTable({
head(data())
})
}
```
```{r require-cancel, fig.cap = demo$caption("On load, the table is empty because the dataset name is empty. The data is shown after we type a valid dataset name (iris), and continues to be shown when we press backspace in order to type a new dataset name."), echo = FALSE, message = FALSE, out.width = "33%"}
demo <- demoApp$new("action-feedback/require-cancel", ui, server,
packages = "shinyFeedback",
before = ~ shinyFeedback::useShinyFeedback()
)
demo$resize(400)
demo$takeScreenshot("empty")
demo$setInputs(dataset = "iris")
demo$takeScreenshot("ok")
demo$setInputs(dataset = "iri")
demo$takeScreenshot("error")
demo$deploy()
```
### Validate output
shinyFeedback is great when the problem is related to a single input.
But sometimes the invalid state is a result of a combination of inputs.
In this case it doesn't really make sense to put the error next to an input (which one would you put it beside?) and instead it makes more sense to put it in the output.
You can do so with a tool built into shiny: `validate()`.
When called inside a reactive or an output, `validate(message)` stops execution of the rest of the code and instead displays `message` in any downstream outputs.
The following code shows a simple example where we don't want to log or square-root negative values.
You can see the results in Figure \@ref(fig:validate).
```{r}
ui <- fluidPage(
numericInput("x", "x", value = 0),
selectInput("trans", "transformation",
choices = c("square", "log", "square-root")
),
textOutput("out")
)
server <- function(input, output, session) {
output$out <- renderText({
if (input$x < 0 && input$trans %in% c("log", "square-root")) {
validate("x can not be negative for this transformation")
}
switch(input$trans,
square = input$x ^ 2,
"square-root" = sqrt(input$x),
log = log(input$x)
)
})
}
```
```{r validate, fig.cap = "If the inputs are valid, the output shows the transformation. If the combination of inputs is invalid, then the output is replaced with an informative message.", echo = FALSE, message = FALSE, out.width = "50%"}
demo <- demoApp$new("action-feedback/validate", ui, server, packages = "shinyFeedback", before = ~ shinyFeedback::useShinyFeedback())
demo$resize(300)
demo$takeScreenshot("init")
demo$setInputs(x = -1, trans = "log")
demo$takeScreenshot("log")
```
```{=html}
<!--
### Exercises
1. How do you use a single `req()` to verify that multiple inputs have been successfully filled in?
2. A common alternative approach to using `if` + `validate()` above is to use `validate()` + `need()`:
```{r, eval = FALSE}
validate(
need(
input$x < 0 && input$trans %in% c("log", "square-root"),
"x can not be negative for this transformation"
)
)
```
Read `?need` and then discuss the pros and cons of the two approaches.
-->
```
## Notifications {#notifications}
If there isn't a problem and you just want to let the user know what's happening, then you want a **notification**.
In Shiny, notifications are created with `showNotification()`, and stack in the bottom right of the page.
There are three basic ways to use `showNotification()`:
- To show a transient notification that automatically disappears after a fixed amount of time.
- To show a notification when a process starts and remove it when the process ends.
- To update a single notification with progressive updates.
These three techniques are discussed below.
### Transient notification
The simplest way to use `showNotification()` is to call it with a single argument: the message that you want to display.
It's very hard to capture this behaviour with a screenshot, so go to <https://hadley.shinyapps.io/ms-notification-transient> if you want to see it in action.
```{r}
ui <- fluidPage(
actionButton("goodnight", "Good night")
)
server <- function(input, output, session) {
observeEvent(input$goodnight, {
showNotification("So long")
Sys.sleep(1)
showNotification("Farewell")
Sys.sleep(1)
showNotification("Auf Wiedersehen")
Sys.sleep(1)
showNotification("Adieu")
})
}
```
```{r, echo = FALSE, message = FALSE}
demo <- demoApp$new("action-feedback/notification-transient", ui, server)
demo$deploy()
```
By default, the message will disappear after 5 seconds, which you can override by setting `duration`, or the user can dismiss it earlier by clicking the close button.
If you want to make the notification more prominent, you can set the `type` argument to one of "message", "warning", or "error".
Figure \@ref(fig:notify) gives a sense of what this looks like.
```{r}
server <- function(input, output, session) {
observeEvent(input$goodnight, {
showNotification("So long")
Sys.sleep(1)
showNotification("Farewell", type = "message")
Sys.sleep(1)
showNotification("Auf Wiedersehen", type = "warning")
Sys.sleep(1)
showNotification("Adieu", type = "error")
})
}
```
```{r echo = FALSE, message = FALSE}
demo <- demoApp$new("action-feedback/notify-persistent", ui, server)
demo$deploy()
```
```{r notify, echo = FALSE, fig.cap = demo$caption("The progression of notifications after clicking 'Good night': the first notification appears, after three more seconds all notifications are shown, then the notifications start to fade away."), fig.align = "center"}
knitr::include_graphics(c(
"images/action-feedback/notify-1.png",
"images/action-feedback/notify-2.png",
"images/action-feedback/notify-3.png"
))
```
### Removing on completion
It's often useful to tie the presence of a notification to a long-running task.
In this case, you want to show the notification when the task starts, and remove the notification when the task completes.
To do this, you'll need to:
- Set `duration = NULL` and `closeButton = FALSE` so that the notification stays visible until the task is complete.
- Store the `id` returned by `showNotification()`, and then pass this value to `removeNotification()`.
The most reliable way to do so is to use `on.exit()`, which ensures that the notification is removed regardless of how the task completes (either successfully or with an error).
You can learn more about `on.exit()` in [Changing and restoring state](https://withr.r-lib.org/articles/changing-and-restoring-state.html).
The following example puts the pieces together to show how you might keep the user up to date when reading in a large csv file[^action-feedback-3]:
[^action-feedback-3]: If reading csv files is a bottleneck in your application should consider using `data.table::fread()` and `vroom::vroom()` instead; they can be orders of magnitude faster than `read.csv()`.
```{r}
server <- function(input, output, session) {
data <- reactive({
id <- showNotification("Reading data...", duration = NULL, closeButton = FALSE)
on.exit(removeNotification(id), add = TRUE)
read.csv(input$file$datapath)
})
}
```
Generally, these sort of notifications will live in a reactive, because that ensures that the long-running computation is only re-run when needed.
### Progressive updates
As you saw in the first example, multiple calls to `showNotification()` usually create multiple notifications.
You can instead update a single notification by capturing the `id` from the first call and using it in subsequent calls.
This is useful if your long-running task has multiple subcomponents.
You can see the results in <https://hadley.shinyapps.io/ms-notification-updates>.
```{r}
ui <- fluidPage(
tableOutput("data")
)
server <- function(input, output, session) {
notify <- function(msg, id = NULL) {
showNotification(msg, id = id, duration = NULL, closeButton = FALSE)
}
data <- reactive({
id <- notify("Reading data...")
on.exit(removeNotification(id), add = TRUE)
Sys.sleep(1)
notify("Reticulating splines...", id = id)
Sys.sleep(1)
notify("Herding llamas...", id = id)
Sys.sleep(1)
notify("Orthogonalizing matrices...", id = id)
Sys.sleep(1)
mtcars
})
output$data <- renderTable(head(data()))
}
```
```{r, echo = FALSE, message = FALSE}
demo <- demoApp$new("action-feedback/notification-updates", ui, server)
demo$deploy()
```
## Progress bars
For long-running tasks, the best type of feedback is a progress bar.
As well as telling you where you are in the process, it also helps you estimate how much longer it's going to be: should you take a deep breath, go get a coffee, or come back tomorrow?
In this section, I'll show two techniques for displaying progress bars, one built into Shiny, and one from the [waiter](https://waiter.john-coene.com/) package developed by John Coene.
Unfortunately both techniques suffer from the same major drawback: to use a progress bar you need to be able to divide the big task into a known number of small pieces that each take roughly the same amount of time.
This is often hard, particularly since the underlying code is often written in C and it has no way to communicate progress updates to you.
We are working on tools in the [progress package](https://github.com/r-lib/progress) so that packages like dplyr, readr, and vroom will one day generate progress bars that you can easily forward to Shiny.
### Shiny
To create a progress bar with Shiny, you need to use `withProgress()` and `incProgress()`.
Imagine you have some slow running code that looks like this[^action-feedback-4]:
[^action-feedback-4]: If your code doesn't involve a for loop or a apply/map function, it's going to be very difficult to make a progress bar.
```{r, eval = FALSE}
for (i in seq_len(step)) {
x <- function_that_takes_a_long_time(x)
}
```
You start by wrapping it in `withProgress()`.
This shows the progress bar when the code starts, and automatically removes it when it's done:
```{r, eval = FALSE}
withProgress({
for (i in seq_len(step)) {
x <- function_that_takes_a_long_time(x)
}
})
```
Then call `incProgress()` after each step:
```{r, eval = FALSE}
withProgress({
for (i in seq_len(step)) {
x <- function_that_takes_a_long_time(x)
incProgress(1 / length(step))
}
})
```
The first argument of `incProgress()` is the amount to increment the progress bar.
By default, the progress bar starts at 0 and ends at 1, so the incrementing by 1 divided by the number of steps will ensure that the progress bar is complete at the end of the loop.
Here's how that might look in a complete Shiny app, as shown in Figure \@ref(fig:progress).
```{r}
ui <- fluidPage(
numericInput("steps", "How many steps?", 10),
actionButton("go", "go"),
textOutput("result")
)
server <- function(input, output, session) {
data <- eventReactive(input$go, {
withProgress(message = "Computing random number", {
for (i in seq_len(input$steps)) {
Sys.sleep(0.5)
incProgress(1 / input$steps)
}
runif(1)
})
})
output$result <- renderText(round(data(), 2))
}
```
```{r, echo = FALSE, message = FALSE}
demo <- demoApp$new("action-feedback/progress", ui, server)
demo$deploy()
```
```{r progress, fig.cap=demo$caption("A progress bar helps indicate how much longer a calculation has to run."), echo = FALSE, out.width = NULL}
knitr::include_graphics("images/action-feedback/progress-2.png", dpi = 300)
```
A few things to note:
- I used the optional `message` argument to add some explanatory text to the progress bar.
- I used `Sys.sleep()` to simulate a long running operation; in your code this would be a slow function.
- I allowed the user to control when the event starts by combining a button with `eventReactive()`.
This is good practice for any task that requires a progress bar.
### Waiter
The built-in progress bar is great for the basics, but if you want something that provides more visual options, you might try the [waiter](https://waiter.john-coene.com/ "⌘+Click to follow link") package.
Adapting the above code to work with Waiter is straightforward.
In the UI, we add `use_waitress()`:
```{r}
ui <- fluidPage(
waiter::use_waitress(),
numericInput("steps", "How many steps?", 10),
actionButton("go", "go"),
textOutput("result")
)
```
The interface for waiter's progress bars are a little different.
The waiter package uses an R6 object to bundle all progress related functions into a single object.
If you've never used an R6 object before, don't worry too much about the details; you can just copy and paste this template.
The basic lifecycle looks like this:
```{r, eval = FALSE}
# Create a new progress bar
waitress <- waiter::Waitress$new(max = input$steps)
# Automatically close it when done
on.exit(waitress$close())
for (i in seq_len(input$steps)) {
Sys.sleep(0.5)
# increment one step
waitress$inc(1)
}
```
And we can use it in a Shiny app as follows:
```{r}
server <- function(input, output, session) {
data <- eventReactive(input$go, {
waitress <- waiter::Waitress$new(max = input$steps)
on.exit(waitress$close())
for (i in seq_len(input$steps)) {
Sys.sleep(0.5)
waitress$inc(1)
}
runif(1)
})
output$result <- renderText(round(data(), 2))
}
```
```{r, echo = FALSE, message = FALSE}
demo <- demoApp$new("action-feedback/waiter", ui, server, packages = "waiter", before = ~ waiter::use_waitress())
demo$deploy()
```
The default display is a thin progress bar at the top of the page (which you can see `r demo$link()`), but there are a number of ways to customise the output:
- You can override the default `theme` to use one of:
- `overlay`: an opaque progress bar that hides the whole page
- `overlay-opacity`: a translucent progress bar that covers the whole page
- `overlay-percent`: an opaque progress bar that also displays a numeric percentage.
- Instead of showing a progress bar for the entire page, you can overlay it on an existing input or output by setting the `selector` parameter, e.g.:
```{r, eval = FALSE}
waitress <- Waitress$new(selector = "#steps", theme = "overlay")
```
### Spinners
Sometimes you don't know exactly how long an operation will take, and you just want to display an animated spinner that reassures the user that something is happening.
You can also use the waiter package for this task; just switch from using a `Waitress` to using a `Waiter`:
```{r}
ui <- fluidPage(
waiter::use_waiter(),
actionButton("go", "go"),
textOutput("result")
)
server <- function(input, output, session) {
data <- eventReactive(input$go, {
waiter <- waiter::Waiter$new()
waiter$show()
on.exit(waiter$hide())
Sys.sleep(sample(5, 1))
runif(1)
})
output$result <- renderText(round(data(), 2))
}
```
```{r, echo = FALSE, message = FALSE}
demo <- demoApp$new("action-feedback/spinner-1", ui, server, packages = "waiter", before = ~ waiter::use_waiter())
demo$deploy()
```
```{r spinner-1, fig.cap = demo$caption("A 'Waiter' displays a whole app spinner while something is happen."), echo = FALSE, message = FALSE, out.width = NULL}
knitr::include_graphics("images/action-feedback/spinner-1.png", dpi = 300 * 1.25)
```
Like `Waitress`, you can also use `Waiter`s for specific outputs.
These waiters can automatically remove the spinner when the output updates, so the code is even simpler:
```{r}
ui <- fluidPage(
waiter::use_waiter(),
actionButton("go", "go"),
plotOutput("plot"),
)
server <- function(input, output, session) {
data <- eventReactive(input$go, {
waiter::Waiter$new(id = "plot")$show()
Sys.sleep(3)
data.frame(x = runif(50), y = runif(50))
})
output$plot <- renderPlot(plot(data()), res = 96)
}
```
```{r, echo = FALSE, message = FALSE}
demo <- demoApp$new("action-feedback/spinner-2", ui, server, packages = "waiter", before = ~ waiter::use_waiter())
demo$deploy()
```
```{r spinner-2, fig.cap = demo$caption("You can instead display a spinner for single output."), echo = FALSE, message = FALSE, out.width = NULL}
knitr::include_graphics("images/action-feedback/spinner-2.png", dpi = 300 * 1.25)
```
The waiter package provides a large variety of spinners to choose from; see your options at `?waiter::spinners` and then choose one with (e.g.) `Waiter$new(html = spin_ripple())`.
An even simpler alternative is to use the [shinycssloaders](https://github.com/daattali/shinycssloaders) package by Dean Attali.
It uses JavaScript to listen to Shiny events, so it doesn't even need any code on the server side.
Instead, you just use `shinycssloaders::withSpinner()` to wrap outputs that you want to automatically get a spinner when they have been invalidated.
```{r}
library(shinycssloaders)
ui <- fluidPage(
actionButton("go", "go"),
withSpinner(plotOutput("plot")),
)
server <- function(input, output, session) {
data <- eventReactive(input$go, {
Sys.sleep(3)
data.frame(x = runif(50), y = runif(50))
})
output$plot <- renderPlot(plot(data()), res = 96)
}
```
## Confirming and undoing
Sometimes an action is potentially dangerous, and you either want to make sure that the user *really* wants to do it, or you want to give them the ability to back out before it's too late.
The three techniques in this section lay out your basic options and give you some tips for how you might implement them in your app.
### Explicit confirmation {#feedback-modal}
The simplest approach to protecting the user from accidentally performing a dangerous action is to require an explicit confirmation.
The easiest way is to use a dialog box which forces the user to pick from one of a small set of actions.
In Shiny, you create a dialog box with `modalDialog()`.
This is called a "modal" dialog because it creates a new "mode" of interaction; you can't interact with the main application until you have dealt with the dialog.
Imagine you have a Shiny app that deletes some files from a directory (or rows in a database etc).
This is hard to undo so you want to make sure that the user is really sure.
You could create a dialog box, as shown in Figure \@ref(fig:dialog), that requires an explicit confirmation as follows:
```{r}
modal_confirm <- modalDialog(
"Are you sure you want to continue?",
title = "Deleting files",
footer = tagList(
actionButton("cancel", "Cancel"),
actionButton("ok", "Delete", class = "btn btn-danger")
)
)
```
```{r dialog, fig.cap = "A dialog box checking whether or not you want to delete some files.", echo = FALSE, message = FALSE, out.width=NULL}
server <- function(input, output, session) {
showModal(modal_confirm)
}
demo <- demoApp$new("action-feedback/dialog", fluidPage(), server)
demo$resize(400, 200)
demo$takeScreenshot()
```
There are a few small, but important, details to consider when creating a dialog box:
- What should you call the buttons?
It's best to be descriptive, so avoid yes/no or continue/cancel in favour of recapitulating the key verb.
- How should you order the buttons?
Do you put cancel first (like the Mac), or continue first (like Windows)?
Your best option is to mirror the platform that you think most people will be using.
- Can you make the dangerous option more obvious?
Here I've used `class = "btn btn-danger"` to style the button prominently.
Jakob Nielsen has more good advice at <https://www.nngroup.com/articles/ok-cancel-or-cancel-ok/>.
Let's use this dialog in a real (if very simple) app.
Our UI exposes a single button to "delete all the files":
```{r}
ui <- fluidPage(
actionButton("delete", "Delete all files?")
)
```
There are two new ideas in the `server()`:
- We use `showModal()` and `removeModal()` to show and hide the dialog.
- We observe events generated by the UI from `modal_confirm`.
These objects aren't created statically in the `ui`, but are instead dynamically added in the `server()` by `showModal()`.
You'll see that idea in much more detail in Chapter \@ref(action-dynamic).
```{r}
server <- function(input, output, session) {
observeEvent(input$delete, {
showModal(modal_confirm)
})
observeEvent(input$ok, {
showNotification("Files deleted")
removeModal()
})
observeEvent(input$cancel, {
removeModal()
})
}
```
### Undoing an action
Explicit confirmation is most useful for destructive actions that are only performed infrequently.
You should avoid it if you want to reduce the errors made by frequent actions.
For example, this technique would not work for twitter --- if there was a dialog box that said "Are you sure you want to tweet this?" you would soon learn to automatically click yes, and still feel the same feeling of regret when you notice a typo 10s after tweeting.
In this situation a better approach is to wait few seconds before actually performing the action, giving the user a chance to notice any problems and undo them.
This isn't really an undo (since you're not actually doing anything), but it's an evocative word that users will understand.
I illustrate the technique with a website that I personally wish had an undo button: Twitter.
The essence of the Twitter UI is very simple: there's a text area to compose your tweet and a button to send it:
```{r}
ui <- fluidPage(
textAreaInput("message",
label = NULL,
placeholder = "What's happening?",
rows = 3
),
actionButton("tweet", "Tweet")
)
```
The server function is quite complex and requires some techniques that we haven't talked about.
Don't worry too much about understanding the code, focus on the basic idea: we use some special arguments to `observeEvent()` to run some code after a few seconds.
The big new idea is that we capture the result of `observeEvent()` and save it to a variable; this allows us to destroy the observer so the code that would really send the tweet is never run.
You can try out the live app at <https://hadley.shinyapps.io/ms-undo>.
```{r}
runLater <- function(action, seconds = 3) {
observeEvent(
invalidateLater(seconds * 1000), action,
ignoreInit = TRUE,
once = TRUE,
ignoreNULL = FALSE,
autoDestroy = FALSE
)
}
server <- function(input, output, session) {
waiting <- NULL
last_message <- NULL
observeEvent(input$tweet, {
notification <- glue::glue("Tweeted '{input$message}'")
last_message <<- input$message
updateTextAreaInput(session, "message", value = "")
showNotification(
notification,
action = actionButton("undo", "Undo?"),
duration = NULL,
closeButton = FALSE,
id = "tweeted",
type = "warning"
)
waiting <<- runLater({
cat("Actually sending tweet...\n")
removeNotification("tweeted")
})
})
observeEvent(input$undo, {
waiting$destroy()
showNotification("Tweet retracted", id = "tweeted")
updateTextAreaInput(session, "message", value = last_message)
})
}
```
```{r, message = FALSE, echo = FALSE}
demo <- demoApp$new("action-feedback/undo", ui, server)
demo$deploy()
```
### Trash
For actions that you might regret days later, a more sophisticated pattern is to implement something like the trash or recycling bin on your computer.
When you delete a file, it isn't permanently deleted but instead is moved to a holding cell, which requires a separate action to empty.
This is like the "undo" option on steroids; you have a lot of time to regret your action.
It's also a bit like the confirmation; you have to do two separate actions to make deletion permanent.
The primary downside of this technique is that it is substantially more complicated to implement (you have to have a separate "holding cell" that stores the information needed to undo the action), and requires regular intervention from the user to avoid accumulating.
For that reason, I think it's beyond the scope of all but the most complicated Shiny apps, so I'm not going to show an implementation here.
## Summary
This chapter has given you a number of tools to help communicate to the user what's happening with your app.
In some sense, these techniques are mostly optional.
But while your app will work without them, their thoughtful application can have a huge impact on the quality of the user experience.
You can often omit feedback when you're the only user of an app, but the more people use it, the more that thoughtful notification will pay off.
In the next chapter, you'll learn how to transfer files to and from the user.