-
Notifications
You must be signed in to change notification settings - Fork 4
/
Copy pathfunctions.R
279 lines (267 loc) · 11.8 KB
/
functions.R
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
# user2019_scheduler - functions.R #############################################
# Author: Hallie Swan
# Date: 2019.06.11
# This script contains helper functions for the user2019_scheduler.
# scrape_process_data.R ########################################################
#' Scrape useR! 2019 Conference Schedule
#'
#' This function will scrape the useR! 2019 conference schedule and return a
#' formatted data.frame. For the majority of events, we assume that events end
#' 3 minutes before the next event in the same room. For the last event in each
#' room, we assume the event ends 3 minutes before the keynote starts. We assume
#' the afternoon keynotes last for 1 hour 15 minutes.
#'
#' @param url string of webpage containing conference schedule
#' @param days character vector of conference days in YYYYMMDD format
#'
#' @importFrom magrittr %>%
#' @importFrom rvest html html_node html_table
#' @importFrom dplyr rename bind_rows arrange rename_all mutate select filter group_by mutate_at vars case_when everything ungroup lead
#' @importFrom tidyselect ends_with
#'
#' @return data.frame of conference schedule information
#' @export
#'
#' @examples
#' scrape_user_talks("http://www.user2019.fr/talk_schedule/", paste0("201907", 10:12))
scrape_user_talks <- function(url, days) {
talks_page <- rvest::html(url)
lapply(days, function(day) {
talks_page %>%
rvest::html_node(paste0("#schedule", day)) %>%
rvest::html_table() %>%
dplyr::rename(description = "") %>%
dplyr::rename_all(tolower) %>%
dplyr::mutate(description = lead(description)) %>%
dplyr::filter(rep(c(TRUE, FALSE), times = nrow(.)/2)) %>%
dplyr::mutate(start_time = time,
day = as.POSIXct(day, format = "%Y%m%d")) %>%
dplyr::arrange(day, room, start_time) %>%
dplyr::group_by(day, room) %>%
dplyr::mutate(end_time = dplyr::lead(start_time)) %>%
dplyr::ungroup() %>%
dplyr::mutate_at(dplyr::vars(tidyselect::ends_with("_time")),
list(~as.POSIXct(paste0(day, .),
format = "%Y-%m-%d%H:%M"))) %>%
dplyr::mutate(
end_time = dplyr::case_when(
!is.na(end_time) ~ end_time,
is.na(end_time) & session == "Keynote" ~ start_time + 60*75,
is.na(end_time) ~ max(start_time)
)) %>%
dplyr::mutate(
# give some time for a passing period
# time needs to be in seconds
end_time = end_time - 0.5*60,
duration = difftime(end_time, start_time, "mins")) %>%
dplyr::select(day, start_time, end_time, duration, title, description, dplyr::everything()) %>%
dplyr::arrange(day, time) %>%
# remove blank events
dplyr::filter(title != "")
}) %>%
dplyr::bind_rows()
}
#' Prepare Schedule Data for Plotting Calendar
#'
#' This function will add additional columns to a data.frame containing schedule
#' for easier plotting.
#'
#' @param df data.frame containing schedule data, generated by scrape_user_talks
#'
#' @importFrom magrittr %>%
#' @importFrom dplyr rowwise mutate ungroup
#' @importFrom lubridate hour minute
#' @importFrom stringr str_trunc
#'
#' @return data.frame
#' @export
#'
#' @examples
#' scrape_user_talks("http://www.user2019.fr/talk_schedule/", paste0("201907", 10:12)) %>%
#' prep_data_for_plot()
prep_data_for_plot <- function(df) {
check <- df %>%
dplyr::rowwise() %>%
dplyr::mutate(
nchar_print = 25 * as.numeric(duration) / 10,
nchar_title = nchar(title),
nchar = ifelse(nchar_title <= nchar_print, nchar_title, nchar_print),
label = stringr::str_trunc(title, nchar),
start_hour = lubridate::hour(start_time) + lubridate::minute(start_time)/60,
end_hour = lubridate::hour(end_time) + lubridate::minute(end_time)/60) %>%
dplyr::ungroup() %>%
dplyr::mutate(event_id = 1:n(),
session_category = trimws(gsub("[[:digit:]]+", "", session)))
}
bold <- function(val) paste0("<b>", val, "</b>")
# https://support.google.com/calendar/answer/37118?hl=en
# cols:
# Subject: name of the event, required
# Start Date: first day of the event, required
# Start Time: time the event begins
# End Date: last day of the event
# End Time: time the event ends
# All Day Event: True, False
# Description: description or notes about the event
# Location: the location for the event
# Private: True, False
# how to handle timezone???
add_google_calendar_cols <- function(df) {
df %>%
ungroup() %>%
mutate(Subject = title,
`Start Date` = as.character(as.Date(day)),
`Start Time` = substr(start_time, 12, 16),
`End Date` = as.character(as.Date(day)),
`End Time` = substr(end_time, 12, 16),
`All Day Event` = "False",
Description = paste0(
bold("Session: "), session, "<br>",
bold("Speaker: "), speaker, "<br>",
bold("Chair: "), chair, "<br>",
bold("Slides: "), slides, "<br>",
description, "<br>"
),
Description = gsub("\n", "<br><br>", Description),
Location = paste0(room, ", Centre de Congrès Pierre Baudis, Toulouse, France"),
Private = "False") %>%
filter(!is.na(Subject) & Subject != "", !is.na(`Start Date`))
}
# server.R #####################################################################
# calendar formatting from: https://stackoverflow.com/a/52487803/8099834
create_calendar_plot <- function(df) {
df <- df %>%
mutate(session_category = ifelse(user_cal, "! Your Calendar !", session_category),
text_color = ifelse(user_cal, "white", "black"))
df %>%
# create plot
ggplot(aes(y = start_hour, yend = end_hour, x = 0, xend = 0,
text = Description)) +
facet_grid(.~room, drop = FALSE) +
geom_segment(size = 80, aes(color = session_category)) +
geom_text(aes(label = str_wrap(label, 25),
y = (end_hour - start_hour)/2 + start_hour),
color = df$text_color,
size = 4) +
theme_minimal() +
theme(axis.ticks.x = element_blank(),
axis.text.x = element_blank(),
axis.text.y = element_text(size = 15),
axis.title = element_text(size = 15),
legend.text = element_text(size = 13),
legend.title = element_blank(),
legend.position = "bottom",
strip.text = element_text(size = 15),
panel.grid = element_blank(),
panel.grid.major.y = element_line(color = "black")) +
labs(x = NULL, y = "Hour CEST") +
guides(color = guide_legend(override.aes = list(size=5),
nrow = 3, byrow = TRUE)) +
scale_y_reverse(breaks = hour_breaks,
labels = paste0(hour_breaks, ":00"),
limits = c(max(hour_breaks), min(hour_breaks))) +
scale_color_manual(values = bg_cols)
}
create_event_modal <- function(click_df) {
# get constants
search_url <- paste0("https://maps.google.com/?q=", gsub(" ", "+", click_df$Location))
event_toggle <- ifelse(!click_df$user_cal, "check-circle", "times-circle")
event_toggle_text <- ifelse(!click_df$user_cal, "add event to", "remove event from")
# build modal
modalDialog(
div(class = "right-align",
actionButton("close_modal", "", icon = icon("times"))),
h3(click_df$title),
div(icon("calendar"),
strftime(click_df$start_time, "%A, %B %d %H:%M"), " - ",
strftime(click_df$end_time, "%H:%M"), " CEST", br()),
div(icon("map-marker-alt"),
a(click_df$Location, href = search_url, target = "_blank")),
hr(),
HTML(click_df$Description),
div(align = "center",
actionButton("change_event", "", icon = icon(event_toggle, "fa-5x")),
helpText("Click to ", event_toggle_text, " your calendar.")),
easyClose = TRUE,
footer = NULL)
}
#' Get Alert Values
#'
#' This function will return a list of values that make up the content of a
#' sweet alert.
#'
#' @param add_rem boolean, if TRUE, will create values indicating successful
#' addition of event to calendar. if FALSE, will create values indicating
#' successful removal of event from calendar.
#'
#' @return list
#' @export
#'
#' @examples
#' get_alert_vals(TRUE)
#' get_alert_vals(FALSE)
get_alert_vals <- function(add_rem) {
alert_vals <- list()
alert_vals$title <- ifelse(!add_rem, "Saved!", "Removed!")
alert_vals$text <- ifelse(!add_rem, "Event saved to your calendar.", "Event removed from your calendar.")
alert_vals$type <- ifelse(!add_rem, "success", "error")
alert_vals
}
# ui.R #########################################################################
ui_create_header <- function() {
fluidRow(
column(width = 12,
h1("Build Calendar for useR! 2019 - Toulouse"),
helpText(
icon("github"),
"Built by ",
a("Hallie Swan",
href = "https://github.com/hallieswan/", target = "_blank"),
HTML(" • "),
icon("code-branch"),
a("Fork me",
href = "https://github.com/hallieswan/user_2019_scheduler",
target = "_blank"),
HTML(" • "),
icon("table"),
"Original data: ",
a("http://www.user2019.fr/talk_schedule/",
href = "http://www.user2019.fr/talk_schedule/",
target = "_blank")
)
)
)
}
ui_create_controls <- function() {
fluidRow(
column(width = 4,
helpText("Toggle between all events and events you have added ",
"to your calendar. Add events by clicking on the event, ",
"and then clicking the check on the bottom of the modal."),
switchInput("calendar_toggle", value = TRUE,
onLabel = "View all events", offLabel = "View my events",
width = 400)),
column(width = 4,
helpText("Select a conference date for which you would like to view events.",
"Note: you can add events from multiple days to your schedule. If you ",
"switch between days, the selected events will be saved."),
selectInput("date", "Conference Date", choices = conf_days)),
column(width = 4,
helpText("Add events to Google Calendar by downloading a csv of events",
"and following instructions ",
a("here. ",
href = "https://github.com/hallieswan/user_2019_scheduler#add-events-to-your-google-calendar", target = "_blank"),
"Note: if you do not create a new calendar with CEST timezone, the events will ",
"be added using your local timezone."),
downloadButton("dl_your_cal", "Your Events"),
downloadButton("dl_full_cal", "All Events"))
)
}
ui_create_calendar <- function() {
fluidRow(
column(width = 12,
plotOutput("calendar", width = 1200, height = 1000,
click = "calendar_click")
)
)
}