-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy path07-Shiny.Rmd
309 lines (230 loc) · 12.1 KB
/
07-Shiny.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
# Shiny {#chap-shiny}
\toc{1}
Shiny allows to publish an interactive application using R code as a web site.
The site can run locally, on a user's workstation that launches it from RStudio, or online, on a dedicated server running Shiny Server[^701].
[^701]: https://rstudio.com/products/shiny/download-server/
Basically, a form allows to enter the arguments of a function and a visualization window to display the results of the calculation.
A Shiny application makes the execution of the code very simple, even for users not familiar with R, but obviously limits the possibilities.
## First application
In RStudio, create an application with the menu "File > New File > Shiny Web App...", enter the name of the application "MyShinyApp" and select the folder where to put it.
The name of the application has been used to create a folder that we now need to transform into a project (project menu in the top right of RStudio, "New Project > Existing Directory", select the application folder).
The application file named `app.R` contains two functions: `ui()` which defines the GUI and `server()` which contains the R code to be executed.
The application can be launched by clicking on `Run App` in the code window.
(ref:shiny-geiser) Shiny Application *Old Faithful Geyser Data*.
```{r}
#| label: shiny-geiser
#| echo: false
#| fig.cap: "(ref:shiny-geiser)"
knitr::include_graphics('images/shiny-geiser.png')
```
The correspondence between the displayed window (figure \@ref(fig:shiny-geiser)) and the `ui()` function code is easy to see:
* The title of the application is displayed by the `titlePanel()` function.
* The slider that sets the number of bars in the histogram is created by `sliderInput()`.
* The `sidebarLayout()` function sets the layout of the page elements, `sidebarPanel` for the input controls and `mainPanel()` for the result display.
The result is displayed by the `plotOutput()` function whose argument is the name of an element of `output`, the variable filled by the `server()` function.
Any modification of an element of the interface, precisely of an element displayed by a function whose name ends with `Input()` (there are some for all types of inputs, for example `textInput()`) of **Shiny** causes `server()` to be executed and the elements of `output` to be updated.
## More elaborate application
### Working method
An application is created by choosing:
* A window layout.
* The controls for entering parameters (*intput*).
* The controls for displaying the results (*output*).
The code to process the inputs and produce the outputs is then written to `server()`.
The RStudio tutorial[^702] is very detailed and should be used to go further.
[^702]: https://shiny.rstudio.com/tutorial/
### Example
This simple application uses the **scholar** package to query Google Scholar and get the bibliometric data of an author from his or her identifier.
The `app.R` file contains all the code and is built incrementally here.
The full application, with graphical output in addition to its simplified version presented here is available on GitHub[^703].
[^703]: https://github.com/EricMarcon/bibliometrics
The beginning of the code consists of preparing the application to run by loading the necessary packages:
```{r}
#| label: shiny_pkg
#| eval: false
# Prepare the application ####
# Load packages
library("shiny")
library("tidyverse")
```
The code of the complete application includes a function to install the missing packages, to be executed only when the application is executed on a workstation (on a server, the management of packages is not the responsibility of the application).
The user interface is as follows:
```{r}
#| label: UI
#| eval: false
# UI ####
ui <- fluidPage(
# Application title
titlePanel("Bibliometrics"),
sidebarLayout(
sidebarPanel(
helpText("Enter the Google Scholar ID of an author."),
textInput("AuthorID", "Google Scholar ID", "4iLBmbUAAAAJ"),
# End of input
br(),
# Display author's name and h
uiOutput("name"),
uiOutput("h")
),
# Show plots in the main panel
mainPanel(
plotOutput("network"),
plotOutput("citations")
)
)
)
```
The application window is fluid, i.e. it reorganizes itself when its size changes, and is composed of a side panel (for text input and display) and a main panel, for displaying graphics.
The elements of the side panel are:
* A help text: `helpText()`.
* A text input field, `textInput()`, whose arguments are the name, the displayed text, and the default value (an author ID).
* A line break: `br()`.
* HTML output controls: `uiOutput()`, whose single argument is the name.
The main panel contains two graphical output controls, `plotOutput()` whose argument is also the name.
The code to execute to process the inputs and produce the outputs is in the `server()` function.
```{r}
#| label: server
#| eval: false
# Server logic ####
server <- function(input, output) {
# Run the get_profile function only once ####
# Store the author profile
AuthorProfile <- reactiveVal()
# Update it when input$AuthorID is changed
observeEvent(
input$AuthorID,
AuthorProfile(get_profile(input$AuthorID))
)
# Output ####
output$name <- renderUI({
h2(AuthorProfile()$name)
})
output$h <- renderUI({
a(href = paste0(
"https://scholar.google.com/citations?user=",
input$AuthorID),
paste("h index:", AuthorProfile()$h_index),
target = "_blank"
)
})
output$citations <- renderPlot({
get_citation_history(input$AuthorID) %>%
ggplot(aes(year, cites)) +
geom_segment(aes(xend = year, yend = 0), size = 1, color = 'darkgrey') +
geom_point(size = 3, color = "firebrick") +
labs(
title = "Citations per year",
caption = "Source: Google Scholar"
)
})
output$network <- renderPlot({
ggplot() + geom_blank()
})
}
```
The information needed for the output fields `$name` and `$h` (author's name and h-index) is obtained by the `get_profile()` function of the **scholar** package.
This function queries the author's Google Scholar web page and extracts the values from the result: this is a heavy processing, which is better executed only once rather than twice, in the `renderUI()` functions in charge of computing the values of `output$h` and `output$name`.
The simplest code to do this would be as follows:
```{r}
#| label: AuthorProfile1
#| eval: false
# Run the get_profile function only once ####
# Store the author profile
AuthorProfile <- get_profile(input$AuthorID)
```
The difficulty with programming a Shiny application is that any computation referring to an input interface element must be *reactive*.
If the latter code were executed, the following error message would appear:
"Operation not allowed without an active reactive context. (You tried to do something that can only be done from inside a reactive expression or observer.)"
In practice, the execution of the code is started by modifying an input control (here: `intput$AuthorID`).
The code referring to one of these controls must be permanently waiting for a modification: it must therefore be placed in particular functions like `renderPlot()` in the *Old Faithful Geyser Data* application or `renderUI()` here.
The following code would run without error:
```{r}
#| label: AuthorProfile2
#| eval: false
# Output ####
output$name <- renderUI({
AuthorProfile <- get_profile(input$AuthorID)
h2(AuthorProfile$name)
})
```
The call to the value of the `input$AuthorID` control does occur in a reactive function (but `get_profile()` would have to be used a second time in the calculation of `output$h`, which we want to avoid).
The function `h2(AuthorProfile$name)` produces HTML code, a level 2 title paragraph whose value is passed as an argument.
All functions whose names begin with `render` in the **shiny** package are reactive, and each is intended to produce a different type of output, for example text (`renderText()`) or HTML code (`renderUI()`).
If code is needed to compute variables common to several output controls (`output$name` and `output$h`), it must itself be responsive.
Two functions are very useful:
* `observeEvent()` watches for changes in an input control and executes code when they occur.
* `reactiveVal()` allows you to define a reactive variable, which will be modified by the `observeEvent()` code and will in turn cause other reactive functions that use its value to execute.
So the optimal code creates a reactive variable to store the result of the Google Scholar query in:
```{r}
#| label: reactiveVal
#| eval: false
# Store the author profile
AuthorProfile <- reactiveVal()
```
The reactive variable is empty at this point.
Its use is then that of a function: `AuthorProfile(x)` assigns it the value `x` and `AuthorProfile()`, without arguments, returns its value.
The `observeEvent()` function is triggered when `input$AuthorID` is modified and executes the code passed as the second argument, in this case the update of `AuthorProfile`.
```{r}
#| label: observeEvent
#| eval: false
# Update it when input$AuthorID is changed
observeEvent(input$AuthorID, AuthorProfile(get_profile(input$AuthorID)))
```
Finally, the `renderUI()` functions that provide output control values use the value of `AuthorProfile`:
```{r}
#| label: renderUI
#| eval: false
# Output ####
output$name <- renderUI({
h2(AuthorProfile()$name)
})
```
Note the parentheses in `AuthorProfile()`, a reactive variable, as opposed to the `AuthorProfile$name` syntax for a classic variable.
The value of `output$h` is an internet link, `<a href=...` in HTML, written by the `a()` function of the **htmltools** package used by `renderUI()`.
```{r}
#| label: htmltools
#| eval: false
output$h <- renderUI({
a(href = paste0(
"https://scholar.google.com/citations?user=", input$AuthorID
),
paste("h index:", AuthorProfile()$h_index),
target = "_blank"
)
})
```
The link is to the author's Google Scholar page.
The value displayed is its h index.
The argument `target = "_blank"` indicates that the link should be opened in a new browser window.
The `output$citations` graph is created by the `renderPlot()` reactive function.
The data provided by the `get_citation_history()` function of **scholar** (which queries the Google Scholar API) is processed by `ggplot()`.
Finally, the `output$network` graph is an empty graph in this simplified version of the application.
The full application takes this code and adds error handling in case the author code does not exist on Google Scholar and the co-author network graph.
## Hosting {#sec:hebergement-shiny}
A Shiny application is not necessarily hosted by a web server: it can be run on users' workstations if they have R.
For a wider use, a dedicated server is necessary.
Shinyapps.io[^704] is a service from RStudio that allows to host 5 Shiny applications for free with a maximum uptime of 5 hours per month.
[^704]: https://www.shinyapps.io/
First of all, you have to open an account on the site, preferably with your GitHub identifiers.
To allow the management of online applications directly from RStudio, you must then install the **rsconnect** package and set it up:
```{r}
#| label: setAccountInfo
#| eval: false
rsconnect::setAccountInfo(
name='firstname.name',
token='xxx',
secret='<SECRET>'
)
```
The exact code, along with the username and token to use, are displayed on the Shinyapps.io homepage: click on "Show Secret", copy the code and paste it into the RStudio console to run it.
A "Publish" button is available just to the right of the "Run App" button.
Click on it and validate the publication (figure \@ref(fig:shiny-publish)).
(ref:shiny-publish) Publication of the Shiny application on Shinyapps.io.
```{r}
#| label: shiny-publish
#| echo: false
#| fig.cap: "(ref:shiny-publish)"
knitr::include_graphics('images/shiny-publish.png')
```
The application is now available at https://firstname-lastname.shinyapps.io/MyShinyApp/
The "Bibliometrics" application does not work on Shinyapps.io because the way the **Scholar** package queries Google Scholar is not supported.
Most Shiny applications work without difficulty, as long as they don't require complex networking features.