Frankenstein
Want to share your content on R-bloggers? click here if you have a blog, or here if you don't.
Remember me, remember me, but ah! forget my fate (Dido’s Lament, Henry Purcell)
A Voronoi diagram divides a plane based on a set of original points. Each polygon, or Voronoi cell, contains an original point and all that are closer to that point than any other.
This is a nice example of a Voronoi tesselation. You can find good explanations of Voronoi diagrams and Delaunay triangulations here (in English) or here (in Spanish).
A grayscale image is simply a matrix where darkness of pixel located in coordinates (i, j) is represented by the value of its corresponding element of the matrix: a grayscale image is a dataset. This is a Voronoi diagraman of Frankenstein:
To do it I followed the next steps:
- Read this image
- Convert it to gray scale
- Turn it into a pure black and white image
- Obtain a random sample of black pixels (previous image corresponds to a sample of 6.000 points)
- Computes the Voronoi tesselation
Steps 1 to 3 were done with imager, a very appealing package to proccess and analice images. Step 5 was done with deldir, also a convenient package which computes Delaunay triangulation and the Dirichlet or Voronoi tessellations.
The next grid shows tesselations for sample size from 500 to 12.000 points and step equal to 500:
data:image/s3,"s3://crabby-images/71154/7115471fee049e2e94d1efe29bac75410f8bade4" alt=""
data:image/s3,"s3://crabby-images/2c845/2c8452a5a9c8be4079d70f0d6d5cda3c2745a2e0" alt=""
data:image/s3,"s3://crabby-images/34713/34713d56aedad172c50559552eed6f426939403d" alt=""
data:image/s3,"s3://crabby-images/84ec9/84ec975f8690f2e3c10a6894acca960399524d05" alt=""
data:image/s3,"s3://crabby-images/ba7c4/ba7c4449f46c50265abcb16fbc2adf9bc5f7eb6e" alt=""
data:image/s3,"s3://crabby-images/31a1c/31a1c45e9e331310d940d5a5edb4b0d8cb5f173e" alt=""
data:image/s3,"s3://crabby-images/e7e1c/e7e1c21d60a8516c8f3ea34a7be810c4addaddbc" alt=""
data:image/s3,"s3://crabby-images/6cd71/6cd71a52823d50741b5a41f7506c552eae076ae7" alt=""
data:image/s3,"s3://crabby-images/852d5/852d53e5f85bfa393aed4906468a48be5acc94c8" alt=""
data:image/s3,"s3://crabby-images/1ba0c/1ba0cd53f233328abf1fdc60919459447020cdb5" alt=""
data:image/s3,"s3://crabby-images/d216b/d216bd220571205237c0f634695afa4b6b8f1dda" alt=""
data:image/s3,"s3://crabby-images/b9876/b98765f2e182b7eda0fcccca1e4c8a473ca3fa04" alt=""
data:image/s3,"s3://crabby-images/14194/141949ffb9d0bbeacf8203c8ed78031e17ae339d" alt=""
data:image/s3,"s3://crabby-images/23e6d/23e6dfe12c51686dec577bcf3fd0e0f34fc3d9be" alt=""
data:image/s3,"s3://crabby-images/ba8c0/ba8c0330102aa2a9c34fb8568d8d0096931f652f" alt=""
data:image/s3,"s3://crabby-images/1213c/1213c18cd9cf6cd6d48a8ae992bffae97e7325f9" alt=""
data:image/s3,"s3://crabby-images/4298e/4298ee65861caad6ad5308f802bbad569d54cdda" alt=""
data:image/s3,"s3://crabby-images/39d65/39d654d4348632962f4c859f1fe7d79926fa5d67" alt=""
data:image/s3,"s3://crabby-images/903a7/903a7d4f8ffc3c4d0cd4ece13add02037baf8fde" alt=""
data:image/s3,"s3://crabby-images/e7461/e746170cd00fdfe164f63c065b1d8a265663a614" alt=""
data:image/s3,"s3://crabby-images/130c4/130c48fe7976312f47f8d4710794f6102c4f8cf6" alt=""
data:image/s3,"s3://crabby-images/2f4e0/2f4e031909d3791c11f6c5379734c66616ac6e9b" alt=""
data:image/s3,"s3://crabby-images/69a1c/69a1c8cd050d937ab1b1e12041854df5067bafee" alt=""
data:image/s3,"s3://crabby-images/d0180/d0180be403ec8130cf3f765348e0a01bbb5f1b63" alt=""
I gathered all previous images in this gif created with magick, another amazing package of R I discovered recently:
This is the code:
library(imager) library(dplyr) library(deldir) library(ggplot2) library(scales) # Download the image file="http://ereaderbackgrounds.com/movies/bw/Frankenstein.jpg" download.file(file, destfile = "frankenstein.jpg", mode = 'wb') # Read and convert to grayscale load.image("frankenstein.jpg") %>% grayscale() -> x # This is just to define frame limits x %>% as.data.frame() %>% group_by() %>% summarize(xmin=min(x), xmax=max(x), ymin=min(y), ymax=max(y)) %>% as.vector()->rw # Filter image to convert it to bw x %>% threshold("45%") %>% as.data.frame() -> df # Function to compute and plot Voronoi tesselation depending on sample size doPlot = function(n) { #Voronoi tesselation df %>% sample_n(n, weight=(1-value)) %>% select(x,y) %>% deldir(rw=rw, sort=TRUE) %>% .$dirsgs -> data # This is just to add some alpha to lines depending on its longitude data %>% mutate(long=sqrt((x1-x2)^2+(y1-y2)^2), alpha=findInterval(long, quantile(long, probs = seq(0, 1, length.out = 20)))/21)-> data # A little bit of ggplot to plot results data %>% ggplot(aes(alpha=(1-alpha))) + geom_segment(aes(x = x1, y = y1, xend = x2, yend = y2), color="black", lwd=1) + scale_x_continuous(expand=c(0,0))+ scale_y_continuous(expand=c(0,0), trans=reverse_trans())+ theme(legend.position = "none", panel.background = element_rect(fill="white"), axis.ticks = element_blank(), panel.grid = element_blank(), axis.title = element_blank(), axis.text = element_blank())->plot return(plot) } # I call the previous function and store resulting plot in jpeg format i=5000 name=paste0("frankie",i,".jpeg") jpeg(name, width = 600, height = 800, units = "px", quality = 100) doPlot(i) dev.off() # Once all images are stored I can create gif library(magick) frames=c() images=list.files(pattern="jpeg") for (i in length(images):1) { x=image_read(images[i]) x=image_scale(x, "300") c(x, frames) -> frames } animation=image_animate(frames, fps = 2) image_write(animation, "Frankenstein.gif")
R-bloggers.com offers daily e-mail updates about R news and tutorials about learning R and many other topics. Click here if you're looking to post or find an R/data-science job.
Want to share your content on R-bloggers? click here if you have a blog, or here if you don't.