Chapter 3 Dynamic visualization
As for leaflet for mapping, there exists many R packages for dynamic or interactive visualization. We present some of them in this part.
3.1 Basic charts with rAmCharts
and plotly
rAmCharts is user-friendly for standard graphs (scatterplot, times series, histogram…). We just have to use classical R functions with the prefix am. For instance
library(rAmCharts)
amHist(iris$Petal.Length)
amPlot(iris, col = colnames(iris)[1:2], type = c("l", "st"),
zoom = TRUE, legend = TRUE)
amBoxplot(iris)
plotly produces similar things but with a specific syntax. plotly commands are expanded into 3 parts:
- dataset and variables (plot_ly}) ;
- additional representaions (add_trace, add_markers…) ;
- options (axis, titles…) (layout).
We can find a description for each part at https://plot.ly/r/reference/. As a first chart, we propose to represent a scatterplot with its linear smoother. We start by generating the data and computing the linear model:
library(plotly)
<- 100
n <- runif(n,-5,5)
X <- 2+3*X+rnorm(n,0,1)
Y <- data.frame(X,Y)
D <- lm(Y~X,data=D) model
We obtain the required graph with
|> plot_ly(x=~X,y=~Y) |>
D add_markers(type="scatter",mode="markers",
marker=list(color="red"),name="Nuage") |>
add_trace(y=fitted(model),type="scatter",mode='lines',
name="Régression",line=list(color="blue")) |>
layout(title="Régression",xaxis=list(title="abscisse"),
yaxis=list(title="ordonnées"))
Unlike ggplot, we can make 3D with plotly. For instance
plot_ly(z = volcano, type = "surface")
We can also convert ggplot graph into plotly graph with ggplotly
:
<- ggplot(iris)+aes(x=Species,y=Sepal.Length)+geom_boxplot()+theme_classic()
p ggplotly(p)
You can find more informations in this book.
Exercise 3.1 (Basic charts with `rAmCharts` and `plotly`) We consider the iris dataset. Build the following graph with rAmCharts
and plotly
.
Scatterplot
Sepal.Length
in term ofSepal.Width
. Use different colors for each species.amPlot(Sepal.Length~Sepal.Width,data=iris,col=iris$Species)
|> plot_ly(x=~Sepal.Width,y=~Sepal.Length,color=~Species) |> iris add_markers(type="scatter",mode="markers")
Boxplot to visualize the distribution of
Petal.Length
for each species.amBoxplot(Petal.Length~Species,data=iris)
|> plot_ly(x=~Species,y=~Petal.Length) |> add_boxplot() iris
3.2 Graphs to visualize networks with visNetwork
Many datasets can be visualized with graphs, especially when one has to study connections between individuals. In this case, each individual is represented by a node and we use edges for the connections. igraph package proposes static representations for graph. For dynamic graphs, we can use visNetwork. To obtain dynamic graphs, we first have to specify nodes and edges, for instance
<- data.frame(id = 1:15, label = paste("Id", 1:15),
nodes group=sample(LETTERS[1:3], 15, replace = TRUE))
<- data.frame(from = trunc(runif(15)*(15-1))+1,to = trunc(runif(15)*(15-1))+1)
edges library(visNetwork)
visNetwork(nodes,edges)
visNetwork(nodes, edges) |> visOptions(highlightNearest = TRUE)
visNetwork(nodes, edges) |> visOptions(highlightNearest = TRUE, nodesIdSelection = TRUE)
visNetwork(nodes, edges) |> visOptions(selectedBy = "group")
Exercise 3.2 (Connections between medias) We consider a graph which represents connections between medias. Data are available here. We can import them with
<- read.csv("data/Dataset1-Media-Example-NODES.csv", header=T, as.is=T)
nodes <- read.csv("data/Dataset1-Media-Example-EDGES.csv", header=T, as.is=T)
links head(nodes)
id media media.type type.label1 s01 NY Times 1 Newspaper
2 s02 Washington Post 1 Newspaper
3 s03 Wall Street Journal 1 Newspaper
4 s04 USA Today 1 Newspaper
5 s05 LA Times 1 Newspaper
6 s06 New York Post 1 Newspaper
audience.size1 20
2 25
3 30
4 32
5 20
6 50
head(links)
from to weight type1 s01 s02 10 hyperlink
2 s01 s02 12 hyperlink
3 s01 s03 22 hyperlink
4 s01 s04 21 hyperlink
5 s04 s11 22 mention
6 s05 s15 21 mention
nodes
object represents the nodes (normal) while links
is for the edges. We can obtain a graph object with
library(igraph)
<- graph_from_data_frame(d=links, vertices=nodes, directed=T)
media V(media)$name <- nodes$media
and we can visualize the (static) graph with a simple plot
:
plot(media)
Visualize this graph with
VisNetwork
package. Hint: use toVisNetworkData.<- toVisNetworkData(media) media.VN visNetwork(nodes=media.VN$nodes,edges=media.VN$edges)
Add an option which allows to select the type of media (Newspaper, TV or Online).
names(media.VN$nodes)[4] <- "labels" visNetwork(nodes=media.VN$nodes,edges=media.VN$edges) |> visOptions(selectedBy = "labels")
Use different colors for each media.
We just have to give the name
group
to the variable type.label.<- media.VN media.VN1 names(media.VN1$nodes)[3] <- "group" visNetwork(nodes=media.VN1$nodes,edges=media.VN1$edges) |> visOptions(selectedBy = "labels")
Use arrows with different widths according to the variable weight. We can also add the option visOptions(highlightNearest = TRUE).
We just have to use the name
value
instead ofweight
.names(media.VN1$edges)[3] <- "value" visNetwork(nodes=media.VN1$nodes,edges=media.VN1$edges) |> visOptions(selectedBy = "labels",highlightNearest = TRUE)
3.3 Dashboard
Dashboards are very important tools in datascience. They allow to gather important messages on datasets and/or a models. We can build dashboard in R with the package flexdashboard
. The syntax is based on Rmarkdown, we don’t have to learn new tools. We can find a very nice tutorial on this package at https://rmarkdown.rstudio.com/flexdashboard/. You can use this tutorial to make the following exercise.
Exercise 3.3 (A Dashboard for linear models) We consider the dataset ozone.txt
. The goal is to explain the maximum daily ozone concentration (variable maxO3
) by the other variables (information about temperatures, nebulosity, wind…). We want to make a dashboard to
- visualize the data : the database and two or three graphs about the output variables (
maxO3
); - visualize simple linear models: we choose one input and we obtain the scatterplot and the linear smoother;
- visualize the full linear model: a summary of the models with some graphs about the residuals;
- select the inputs in the linear models;
- …
- As a first step, we propose to write some simple functions for the dashboard.
We only consider numeric variables. Visualize correlations between the variables with the
corrplot
function of thecorrplot
package.<- read.table("data/ozone.txt") df <- cor(df[,1:11]) cc <- corrplot::corrplot(cc) mat.cor
Draw the histogram of
maxO3
with ggplot, rAmCharts and plotly (use ggplotly).<- ggplot(df)+aes(x=maxO3)+geom_histogram(bins = 10) gg.H <- amHist(df$maxO3) am.H <- ggplotly(gg.H) pl.H
Fit the linear model with output
maxO3
(all the other variables as input). Calculate the Studentized residuals (rstudent) and visualize these residuals in term ofmaxO3
. You can also add a linear smoother on the graph.<- lm(maxO3~.,data=df) mod <- rstudent(mod) res <- data.frame(maxO3=df$maxO3,r.student=res) df1 <- ggplot(df1)+aes(x=maxO3,y=res)+geom_point()+geom_smooth() Ggg <- ggplotly(Ggg) Gggp
- We can now start the dashboard. Use File -> Rmarkdown -> From Template -> Flex Dashboard dialog to open a script.
Build a first dashboard which allows to visualize
- the dataset on a column (use datatable function from DT package) ;
- the histogram of
maxO3
and the correlation matrix on a second column.
Add a second tab to visualize the summary of the full linear model. You can use datatable function of
DT
package. Hint: a new tab could be added withName of the tab=====================================
Add another tab to visualize a simple linear model with one input of your choice. You can print in this tab both the summary of the model and the scatter plot with the linear smoother.
Taking things further: add a last tab where the user can select an input for the linear model. Hint: use the following Shiny commands:
- Input choice
radioButtons("variable1", label="Choisir la variable explicative", choices=names(df)[-1], selected=list("T9"))
- Interactive summary
<- reactive({ mod1 <- paste(input$variable1,collapse="+") XX <- paste("maxO3~",XX,sep="") |> formula() form lm(form,data=df) })#Df corresponds to the dataset renderDataTable({ <- summary(mod1())$coefficients |> round(3) |> as.data.frame() mod.sum1 ::datatable(mod.sum1,options = list(dom = 't')) DT })
- Interactive graph
renderPlotly({ ggplot(df)+aes(x=!!as.name(input$variable1),y=maxO3)+ (geom_point()+geom_smooth(method="lm")) |> ggplotly() })
Don’t forget to add
runtime: shiny
in the header.
The final dashboard may look like
It is available at https://lrouviere.shinyapps.io/dashboard/.