Three dimensional
objects are very popular but negatively affect the accuracy
and speed
at which one can interpret a graphic in most cases. Here is a review of the main graphic types involving 3d, with a discussion of whether or not they are considered bad practice.
Here is a 3d barplot. You are probably familiar with this kind of figure since it is created with Excel, and thus widely used. Compared with a classic grouped barplot, this representation has several drawbacks:
Workaround: Just stick to the normal barplot.
Nothing is worse than a pie chart in dataviz, except a 3D pie chart. Data to viz offers a whole post on pie chart issues.
Adding 3D makes it even worse since it distorts reality. Indeed, the part at the back looks smaller than the one at the front, which is not the case.
Workaround: Don’t make pie charts.
The 3d scatterplot is another common use of 3d in data visualization. This kind of chart takes 3 numeric variables as inputs, and represents them on the X, Y and Z axes.
The accuracy of 3D scatterplots is widely discussed. For instance, try to determine the X, Y and Z coordinates of a specific point. You will realize that it is quite impossible since the human eye reads 3d poorly. Thus, this graphic lacks accuracy.
Workaround: Use a bubble chart instead, or a scatterplot with color gradients.
In my opinion, the 3d scatterplot can make sense only if you can interact with the chart. On the following, you can use your mouse to change the orientation of the plot and zoom on it. It gives a real feeling of 3D that does not exist otherwise. It is still hard to get the accurate coordinates of points, but allows one to efficiently explore the group structure.
# library
library(rgl)
# This is to output a rgl plot in a rmarkdown document. Note that you must add webgl=TRUE, results='hide' in the chunck header
library(knitr)
knit_hooks$set(webgl = hook_webgl)
# Data: the iris data is provided by R
data <- iris
# Add a new column with color
mycolors <- c('royalblue1', 'darkcyan', 'oldlace')
data$color <- mycolors[ as.numeric(data$Species) ]
# Plot
par(mar=c(0,0,0,0))
plot3d(
x=data$`Sepal.Length`, y=data$`Sepal.Width`, z=data$`Petal.Length`,
col = data$color,
type = 's',
radius = .1,
xlab="Sepal Length", ylab="Sepal Width", zlab="Petal Length")
You must enable Javascript to view this page properly.
If you have grid coordinates with a numeric variable for each position, you can represent your data using a surface plot. This kind of representation makes sense especially if the numeric value represents an altitude. In this case, we are actually building a 3D element that is a real 3d object, which makes sense in my opinion.
# Library
library(plotly)
# Plot
plot_ly(z = volcano, type = "surface")
Any thoughts on this? Found any mistake? Disagree? Please drop me a word on twitter or in the comment section below:
A work by Yan Holtz for data-to-viz.com