-
Notifications
You must be signed in to change notification settings - Fork 5
/
Copy pathlab09.qmd
685 lines (522 loc) · 26.2 KB
/
lab09.qmd
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
# Lab 9 Creating a Simple Shiny Application {.unnumbered}
## Package(s)
- [shiny](https://github.com/rstudio/shiny)
- [golem](https://engineering-shiny.org/index.html)
## Schedule
- 08.00 - 08.30: [Recap of Lab 8](https://raw.githack.com/r4bds/r4bds.github.io/main/recap_lab09.html)
- 08.30 - 09.00: [Lecture](https://raw.githack.com/r4bds/r4bds.github.io/main/lecture_lab09.html)
- 09.00 - 09.15: Break
- 09.00 - 12.00: [Exercises](#sec-exercises)
## Learning Materials
_Please prepare the following materials_
- Book: [Mastering Shiny](https://mastering-shiny.org/) by Hadley Wickham -- Read Chapter 1 (Chapter 2 and 3 are good to read as well, if you want).
- Book: [Engineering Production-Grade Shiny Apps](https://engineering-shiny.org/index.html) -- Read Chapter 2 - 5 (they are fairly short, but if you don't find Shiny Apps super cool, feel free to skip Chapter 3 and 5 and Sections 2.2.2 and 4.2.3 - 4.2.4), the rest are quite important for the exercises.
- Cheatsheet: [Shiny](https://raw.githubusercontent.com/rstudio/cheatsheets/main/shiny.pdf) -- This cheatsheet is a bit cluttered, but useful
- Cheatsheet: [Golem](https://raw.githubusercontent.com/rstudio/cheatsheets/main/golem.pdf) -- Look through this after reading the chapters in "Engineering Production-Grade Shiny Apps" - the exercises will remind you to look at the cheatsheet as well.
_Note: The following are suggested learning materials, i.e., do not go over everything, but poke around. You will use these materials as a point of reference for the group exercises_
- [Shiny Input Gallery](https://shiny.rstudio.com/gallery/widget-gallery.html)
- Web: [Shiny from RStudio](https://shiny.rstudio.com/)
- Web: [RStudio tutorials on Shiny](https://shiny.rstudio.com/tutorial/)
- Video: [Playlist: Web Apps in R: Building your First Web Application in R | Shiny Tutorial](https://www.youtube.com/watch?v=tfN10IUX9Lo&list=PLtqF5YXg7GLkxx_GGXDI_EiAvkhY9olbe)
- Example: [nnvizRt](https://leonjessen.shinyapps.io/nnvizRt/)
- More inspiration: [Shiny Gallery](https://shiny.rstudio.com/gallery/)
_Unless explicitly stated, do not do the per-chapter exercises in the R4DS2e book_
## Learning Objectives
_A student who has met the objectives of the session will be able to:_
- Prepare a simple shiny application
- Using relevant online resources to autonomously identify and obtain new and expand on existing knowledge of R
## Exercises {#sec-exercises}
### Getting Started
_First, make sure to read and discuss the feedback you got from last week's assignment!_
Then, for this particular session, we will make use of Posit's free Cloud infrastructure. Setting it up is quite easy:
1. Go to [posit Cloud](https://posit.cloud)
1. Click the blue `GET STARTED` button
1. Under `Cloud Free`, click the blue `Learn more` button
1. Click the long blue `Sign Up` button
1. Enter your information for setting up an account
1. Make sure to `Verify Your Email` as instructed
1. Then return to where you set up the account and click the blue `Continue` button
Congratulations! You have now created your personal Posit cloud account. You will land in `Your Workspace`. Here you will have to create a new project, which you will use to work on the exercises today, so please:
1. In the upper right corner, click the blue `New Project` button
1. Choose `New RStudio Project`
1. Your project will be prepared and then deployed and you will land in the familiar RStudio IDE. Next to `Your Workspace` in the upper left corner, click where it says `Untitled Project` and name your project, e.g. `R4BDS Shiny Exercises`
Now, let us do some quick setup:
1. Click `Tools` and then `Global Options...`
1. Under **Workspace**, untick `Restore .RData into workspace on startup`
1. Next to `Save workspace to .RData on exit:`, choose `Never`
1. In the lower right corner, click `Apply`
1. Now on the left, find and click `Pane Layout` and click `Environment, History, Conne` and choose `Console`
1. In the lower right corner, click `Apply`
Optionally you can adjust the visual appearance:
1. Now on the left, find and click `Appearance`
1. Under `Editor theme:`, click a few options and see if you can find something you like
1. If you want to return to the default, simply find and click `Textmate (default)`
Optionally, you can add a third column enabling you to e.g. have two Quarto Documents / Scripts / Text files open simultaneously:
1. On the left, find and click `Pane Layout`
1. Click `Add Column`
1. Note, this is best suited for wide desktop screens, so if you change your mind, simply click `Remove Column`
_Once you are done, click `OK` to close the `Options`_
### Initial "Hello Shiny World"
_IMPORTANT: If the Posit Cloud "hangs" for some reason, look in the upper right corner, there you will find a circle with three dots, click it and choose `Relaunch Project`. Also, you have limited ram, so in case you clutter your work environment, you may have to run a `rm(list = ls())` to clear your ram_
Now, let us get started on `Shiny`! First, find the console and enter:
```{r}
#| echo: true
#| eval: false
library("shiny")
```
You are going to get an error message, make sure to read it, so you can fix it.
<details><p><summary>Click here for hint</summary></p>
_R does not know what you are asking for here. You are missing "something", how can we install "something" we are missing? Note, here on the **posit cloud** you are allowed to install tools you need_
</details>
Once you have fixed the error, in the console, run:
```{r}
#| echo: true
#| eval: false
library("shiny")
runExample("01_hello")
```
You may get a `Popup Blocked` message, if so, simply click **Try Again**. The you should see:
![](images/shiny_hello_shiny_example.png)
Now, this is the frontend of a real life Shiny app. You can try to alter the `Number of bins:` and see what happens.
There are a multitude of such built in examples.
- **Task:** Try to play around with a few of the following to get an initial feel for what you can do with Shiny and discuss what you see in your group:
```{r}
#| echo: true
#| eval: false
runExample("01_hello") # a histogram
runExample("02_text") # tables and data frames
runExample("03_reactivity") # a reactive expression
runExample("04_mpg") # global variables
runExample("05_sliders") # slider bars
runExample("06_tabsets") # tabbed panels
runExample("07_widgets") # help text and submit buttons
runExample("08_html") # Shiny app built from HTML
runExample("09_upload") # file upload wizard
runExample("10_download") # file download wizard
runExample("11_timer") # an automated timer
```
### Your First Shiny App
Now, let us build the `01_hello`-app from scratch!
1. Go to your Posit Cloud session and click `File` $\rightarrow$ `New File` $\rightarrow$ `R Script`
1. Now, click `File` $\rightarrow$ `Save`
1. Name the file `01_hello.R`
1. Copy/paste the code below here into the empty file and again remember to save
1. Then at the menu just above your open file, click `Run App` or use the console: `runApp("PATH_TO_MY_APP")`
1. Check if it works and compare with the `01_hello`-example
_Important: This is an R-script file NOT a Quarto document! Think of it as if you take all the code boxes and **only** include those in your file._
<details><p><summary>Code for your first app</summary></p>
```{r}
#| echo: true
#| eval: false
# Load the Shiny library
library("shiny")
# Define the User Interface (Frontend)
ui <- fluidPage(
titlePanel("Hello Shiny!"),
sidebarLayout(
sidebarPanel(
sliderInput(inputId = "bins",
label = "Number of bins:",
min = 1,
max = 50,
value = 30)
),
mainPanel(
plotOutput(outputId = "distPlot")
)
)
)
# Define the Server (Backend)
server <- function(input, output) {
output$distPlot <- renderPlot({
x <- faithful$waiting
bins <- seq(min(x), max(x), length.out = input$bins + 1)
hist(x, breaks = bins, col = "#75AADB", border = "white",
xlab = "Waiting time to next eruption (in mins)",
main = "Histogram of waiting times")
})
}
# Launch the shiny app
shinyApp(ui = ui, server = server)
```
</details>
- **Task:** In your group, compare the code with the below illustration of the anatomy of your first shiny app. Pay specific attention to the `input`- and `output`-variables and how the `inputId = "bins"` end up in `input$bins`. This front-to-back-end communication is central!
![](images/shiny_01_hello_app_anatomy.png)
- **Task:** To further understand what is going on, we can ask constructively for input. Let us use your favourite LLM-technology as a sparring partner, e.g. you can cope/paste the following into a chatGPT prompt:
````{verbatim}
#| echo: true
#| eval: false
I am following the university course "R for Bio Data Science".
Today we are learning about Shiny, which is completely new to me.
The first thing we have done is to re-create a small shiny app,
which is part of the examples in the Shiny-package. Moreover,
it is the example which can be seen by running the code:
`runExample("01_hello")`
Now, we have manually implemented the app, but I am not certain
of how it works and the details. In the following, I will
paste the code from the app. Please in details, minding that I
am a begginer, explain the details of the code:
PASTE THE APP CODE HERE
````
- **Task:** Furthermore, let us also get some input on how the illustration above maps to the code. Right-click the "Shiny" image above and save it to disk, then copy/paste the following into the same prompt:
````{verbatim}
#| echo: true
#| eval: false
The lecturer has given me this graphical illustration,
but I am not sure I understand how it maps to the code,
can you please explain?
REMEMBER TO UPLOAD THE IMAGE
````
- **Task:** Now, look at the answers you got, if something is unclear, ask for an elaboration, e.g.:
````{verbatim}
#| echo: true
#| eval: false
I have not seen the terms "user interface" or "frontend"
or "backend" before. Please briefly explain, what these
mean and how they are related to building shiny apps
````
_Note here, how we using the LLM-technology productively as a sparring-partner. Instead of asking it to produce code, we ask it to explain existing code and to elaborate on any unclear specifics_
### Your Second Shiny App
We will continue where we left of last week. To make sure, that everyone is working with the same functions, we will use the functions outlined below here. Remember the task here is to build an interface, which allow people without coding skills to interact with data.
_Let's get started!_
<details><summary>Click here for functions</summary><p>
_Please note, that as an example of reducing dependencies, these functions are implemented using base R. In case you are not familiar, here again is a prime example of using LLM-technology such as e.g. chatGPT as a sparring partner. Try to submit the following prompt:_
````{verbatim}
#| echo: true
#| eval: false
I am not familiar with base R, I only know tidyverse R.
For the following function and R-code, please explain
in detail how the used base R functions work and what
the code in its entirety does:
PASTE THE BASE R CODE OF A FUNCTION HERE
````
<details><summary>Virtual Gene</summary><p>
```{r}
#| echo: true
#| eval: true
# Virtual gene
gene_dna <- function(length, base_probs = c(0.25, 0.25, 0.25, 0.25)){
if( length %% 3 != 0 ){
stop("The argument to the parameter 'l' has to be divisible by 3")
}
dna_vector <- sample(
x = c("A", "T", "C", "G"),
size = length,
replace = TRUE,
prob = base_probs)
dna_string <- paste0(
x = dna_vector,
collapse = "")
return(dna_string)
}
```
</p></details>
<details><summary>Virtual RNA Polymerase</summary><p>
```{r}
#| echo: true
#| eval: true
# Virtual RNA polymerase
transcribe_dna <- function(dna){
rna <- gsub(
pattern = "T",
replacement = "U",
x = dna)
return(rna)
}
```
</p></details>
<details><summary>Virtual Ribosome</summary><p>
```{r}
#| echo: true
#| eval: true
# Virtual Ribosome
translate_rna <- function(rna){
if( is.null(rna) || rna == "" ){ return("") }
l <- nchar(x = rna)
firsts <- seq(
from = 1,
to = l,
by = 3)
lasts <- seq(
from = 3,
to = l,
by = 3)
codons <- substring(
text = rna,
first = firsts,
last = lasts)
codon_table <- c(
"UUU" = "F", "UCU" = "S", "UAU" = "Y", "UGU" = "C",
"UUC" = "F", "UCC" = "S", "UAC" = "Y", "UGC" = "C",
"UUA" = "L", "UCA" = "S", "UAA" = "_", "UGA" = "_",
"UUG" = "L", "UCG" = "S", "UAG" = "_", "UGG" = "W",
"CUU" = "L", "CCU" = "P", "CAU" = "H", "CGU" = "R",
"CUC" = "L", "CCC" = "P", "CAC" = "H", "CGC" = "R",
"CUA" = "L", "CCA" = "P", "CAA" = "Q", "CGA" = "R",
"CUG" = "L", "CCG" = "P", "CAG" = "Q", "CGG" = "R",
"AUU" = "I", "ACU" = "T", "AAU" = "N", "AGU" = "S",
"AUC" = "I", "ACC" = "T", "AAC" = "N", "AGC" = "S",
"AUA" = "I", "ACA" = "T", "AAA" = "K", "AGA" = "R",
"AUG" = "M", "ACG" = "T", "AAG" = "K", "AGG" = "R",
"GUU" = "V", "GCU" = "A", "GAU" = "D", "GGU" = "G",
"GUC" = "V", "GCC" = "A", "GAC" = "D", "GGC" = "G",
"GUA" = "V", "GCA" = "A", "GAA" = "E", "GGA" = "G",
"GUG" = "V", "GCG" = "A", "GAG" = "E", "GGG" = "G")
protein <- paste0(
x = codon_table[codons],
collapse = "")
return(protein)
}
```
</p></details>
<details><summary>Simple base counts</summary><p>
```{r}
#| echo: true
#| eval: true
# Simple base counts
base_freqs <- function(dna){
if (is.null(dna) || dna == "" ){
return( data.frame(dna_vec = factor(c("A", "C", "G", "T")),
Freq = c(0, 0, 0, 0)) ) }
dna_vec <- strsplit(x = dna,
split = "")
base_counts <- table(dna_vec)
return( as.data.frame.table(base_counts) )
}
```
</p></details>
</p></details>
#### Building a Virtual Gene Generator
#### Version 1, the basics
Below here, you will find some boilerplate code, which is based on what you saw in the `01_hello`-example.
- **Task:** Look at this code and compare it to your first shiny app, discuss the changes in your group, what do you see?
- **Task:** Create your second Shiny App in a new file `virtual_gene_generator.R`, copy/paste making sure to insert the code needed for the function `gene_dna()` (See code for functions above)
- **Task:** Run the App and in your group, discuss how it works
```{r}
#| echo: true
#| eval: false
library("shiny")
# Define the "Virtual Gene"-function (See code for functions above)
gene_dna <- ...
# Define the User Interface (Frontend)
ui <- fluidPage(
titlePanel("Virtual Gene Generator"),
sidebarLayout(
sidebarPanel(
sliderInput(inputId = "n_bases",
label = "Number of bases:",
min = 1,
max = 60,
value = 30)
),
mainPanel(
textOutput(outputId = "dna")
)
)
)
# Define the Server (Backend)
server <- function(input, output) {
output$dna <- renderText({
gene_dna(length = input$n_bases)
})
}
# Launch the shiny app
shinyApp(ui = ui, server = server)
```
#### Version 2, expanding
Now, if you look at the code for the `gene_dna`-function, it has a second parameter `base_probs` with the default argument `c(0.25, 0.25, 0.25, 0.25)`. Let us now expand, so your app allows the user to define the individual base probabilities.
- **Task:** Expand the `ui` with one `numericInput()` for each of the four bases defining the probability of drawing that base and pass those numbers onto the `server`
<details><p><summary>Need a hint?</summary></p>
Recall, that you in the console can type `?numericInput` and get help on the function. Try to do so and compare with the `sliderInput()` you already have defined in your app.
_Tip: Start with just one probability and then hardcode the other 3 to e.g. 0.25_
</details>
<details><p><summary>Need a hint more?</summary></p>
The `inputId` defines the "name" of what is passed to the server, e.g. if you look at the `sliderInput()` it uses `inputId = "n_bases"`, which is then passed to the server as `input$n_bases`
</details>
<details><p><summary>Need just one hint more?</summary></p>
In your `ui`, you have to find room for:
```{r}
#| echo: true
#| eval: false
numericInput(inputId = "prob_A",
label = "Probability of A",
value = 0.25,
min = 0,
max = 1)
```
In your `server` you have to find room for:
```{r}
#| echo: true
#| eval: false
base_probs = c(input$prob_A, 0.25, 0.25, 0.25)
```
and then figure out how to expand that to also include `prob_T`, `prob_C` and `prob_G`
</details>
Once you are up and running, your app should look something like this:
![](images/shiny_virtual_gene_generator.png)
- **Task:** Update the `gene_dna()`-function in your app to check if the probabilities sum to 1 and make sure output a meaningful error message, if they do not
<details><p><summary>Need a hint?</summary></p>
The `gene_dna()`-function already checks it the length of the DNA is divisible by 3, use this as inspiration to add your own check. If you are uncertain on how to compute the `sum()` of a vector in `R`, try to google a bit...
</details>
<details><p><summary>A bit more advanced hint</summary></p>
Actually, you should be careful when comparing numbers when you create apps or any analysis code. If you sum a large vector, you might experience, that even when the sum should be equal to 1, it will fail a direct comparison, such as `sum(p) == 1`. This has to do with internal representation of numbers in a computer resulting in small rounding errors. Therefore, it is safer to use e.g.: `abs(sum(p) - 1) < 1e-10`, i.e. is the difference between my sum and the number 1 smaller than some very small number.
</details>
### Controlling the Layout
By now your base app is up and running, but you may want to tinker with the layout of the user interface. We can do so using the `card()`-function from the `bslib`-package:
<details><summary>Click to expand app code</summary><p>
```{r}
#| echo: true
#| eval: false
# Define the User Interface (Frontend)
ui <- page_fluid(
layout_columns(
col_widths = 12,
card(
titlePanel("Virtual Central Dogma"),
style = "background-color: #f0f0f0; padding: 15px;"
)),
layout_columns(
col_widths = 12,
card(
titlePanel("About"),
helpText("Describe what your app does...")
)),
layout_columns(
col_widths = 12,
card(
card_header("Virtual Gene Generator"),
sliderInput(inputId = "n_bases",
label = "Number of bases:",
min = 1,
max = 60,
value = 30,
width = "100%"),
layout_columns(
col_widths = c(3, 3, 3, 3),
numericInput(inputId = "prob_A",
label = "Probability of A",
value = 0.25,
min = 0,
max = 1,
step = 0.1),
numericInput(inputId = "prob_T",
label = "Probability of T",
value = 0.25,
min = 0,
max = 1,
step = 0.1),
numericInput(inputId = "prob_C",
label = "Probability of C",
value = 0.25,
min = 0,
max = 1,
step = 0.1),
numericInput(inputId = "prob_G",
label = "Probability of G",
value = 0.25,
min = 0,
max = 1,
step = 0.1)
))),
layout_columns(
col_widths = 12,
card(
card_header("Virtual Gene output"),
mainPanel(
verbatimTextOutput(outputId = "dna")
)
))
)
```
</details>
- **Task:** Update your app using the above `card()`-function layout. Remember to load the `bslib`-package.
### It's Complicated...
As you have probably realised by now, the final app can be rather many lines of code defining the user interface and likewise for the server and also associated functions. To address this, it is recommendable to divide your app into seperate files mimicking the illustration of the anatomy of a Shiny app. To do so, we will need four files:
1. The functions: `app_functions.R`
1. The user interface: `ui.R`
1. The server: `server.R`
1. The app: `virtual_central_dogma.R`
The latter will look something like this:
```{r}
#| echo: true
#| eval: false
# Load needed libraries
library("PACKAGE_NAME")
# Load needed functions
source(file = "app_functions.R")
# Run the frontend user interface ui.R
source(file = "ui.R")
# Run the backend server server.R
source(file = "server.R")
# Run the Shiny app
shinyApp(ui = ui, server = server)
```
- **Task:** First make sure, that your app runs as is and then create these four files and copy/paste the approproiate code into each of the files
### Completing the Central Dogma
So far you have just been working with the `gene_dna()`-function, but we naturally have to also include:
- Transcription
- Translation
- A bit of analysis
If you scroll a bit back, you were given some pre-built code for the following functions:
- `gene_dna()`
- `transcribe_dna()`
- `translate_rna()`
- `base_freqs()`
Right then:
- **Task:** Expand your app to include the central dogma functions `gene_dna()`, `transcribe_dna()` and `translate_rna()`. Note that any further analysis, such as e.g. `base_freqs()` or similar is optional
A bit of help to get you started:
- You can find an overview of the different [input controls here](https://shiny.posit.co/r/getstarted/shiny-basics/lesson3/)
- Use manual copy/paste when transferring DNA to RNA to Protein (Naturally, this can be done automatically, but that's a completely different can of worms)
- Perhaps you could look at the `01_hello`-example to get some inspiration on how you can include a `ggplot` of your nucleotide frequency analysis
### Getting Your Shiny App LIVE!
_Note, when we publish our app, each file will be "build" and this will create errors if some functions are not found. Therefore, make sure to include `library("PACKAGE_NAME")` in each files for all needed libraries_ and source the app_functions.R file where it's needed too.
1. Go to [shinyapps.io](https://www.shinyapps.io)
2. Log in or create an account
3. Go back to your RStudio session and launch your Shiny App
4. In the upper right corner, it'll say "Publish"
5. You may get prompted to install some packages, so go ahead and do so
6. You will then get prompted to "Connect Publishing Account", click "Next"
7. Choose your "ShinyApps.io" account
8. Follow the outlined connect procedure
9. Choose a "Title" for your app
10. Click "Publish"
_After some automated build procedures, your Shiny App is LIVE - CONGRATULATIONS!_
NB! If you an error saying:
- _"An error has occurred! An error has occurred. Check your logs or contact the app author for clarification."_
Then likely, there is a problem withs paths or libraries, did you remember to start this last part as a separate project? Also, did you remember to include library definitions and source the `app_functions.R`-file in all the necessary files?
## <span style="color: red;">GROUP ASSIGNMENT</span> (Important, see: [how to](assignments.qmd))
For this week's assignment you have to:
1. Create a new Github repository, name it `group_XX_shiny`, where `XX` is your group name, e.g. `06`
1. Finish the Central Dogma shiny app by Github collaborating (Psst... Perhaps branching and delegate specific cards to group members would be prudent)
1. Publish your finished shiny app to shinyapps.io (See How to Publish from Posit Cloud)
1. Place the link to your published app in the README of the repository
1. For your assignment hand in, simply create a text file containing the link to your group Github repository containing your shiny app
_You app should contain the central dogma functions `gene_dna()`, `transcribe_dna()` and `translate_rna()`, but any further analysis, such as e.g. `base_freqs()` or similar is optional. Please feel free to style your app as you please. Also, when creating your app, try to use your favourite LLM-technology, **BUT** use it wisely, use it to **EXPLAIN** code, **NOT** to **GENERATE** code!_
## How to Publish from Posit Cloud
Since October 2024 publishing on Posit Cloud has been deprecated. Therefore, we will have to setup publishing via shinyapps.io:
1. Go to your finished app on [Posit Cloud](http://posit.cloud)
1. Click `Run App` above your `central_dogma_app.R`-file and check that everything works as expected
1. Now, click `Tools` $\Rightarrow$ `Global Options...` $\Rightarrow$ `Publishing` (in the left panel) $\Rightarrow$ `Connect...` in the upper right corner
1. You may get prompted for some packages, which needs installing, click `Yes`
1. Once they are installed, you will be prompted to `Connect Account`, click `ShinyApps.io`
1. Now you will have to click on the `your account on ShinyApps`-link
1. Once on shinyapps.io, click `Log In` in the upper right corner (It's the same login as to Posit Cloud)
1. If this is your first time to shinyapps.io, you will see something like _"Hi! You must be new here..."_, nevermind this
1. In the upper right corner, click your name and choose `Tokens`
1. Then click the green `+ Add Tokens`-button
1. It's okay if you have en existing token, make sure to check the date-of-creation, which for the relevant token should be today and then click `Show`
1. Now you will see a new box and you should be in the `With R`-tab, find the green `Show secret`-button and click it
1. Now, click `Copy to clipboard`
1. Then go back to your Posit Cloud tab and paste the token and code, i.e. all of it, it should start with `rsconnect::setAccountInfo(...)`
1. Then still on the Posit Cloud, click `Connect Account`
1. Under `Publishing Accounts`, make sure you see `id-name: shinyapps.io`
1. Click `OK`
1. Then still on Posit Cloud, make sure you are in your `central_dogma_app.R`-file and then run the app
1. Click the small triangle in the upper right corner of the your app and choose `Other Destination...`
1. Click your shinyapps.io accout name
1. Click `Publish`
1. Your will get a message `Deployment Started...`, click `OK` - _Your app will be deployed by `rsconnect`, this might take a few minutes, so be patient_
1. Now, back in your shinyapps.io session, on the left click `Applications` and then click `All`
1. Find your app, likely named `project` and it will likely have status `Running`, and then click on the name of your app
1. In the `OVERVIEW`, find the URL and click it, this will be the public version of your app!
Phheeww! Well done!