Как использовать изображение в качестве точки в ggplot? - PullRequest
34 голосов
/ 02 февраля 2010

Есть ли способ использовать определенное маленькое изображение в качестве точки на диаграмме рассеяния с помощью ggplot2. В идеале я хочу изменить размеры изображений на основе переменной.

Вот пример:

library(ggplot2)
p <- ggplot(mtcars, aes(wt, mpg))
p + geom_point(aes(size = qsec, shape = factor(cyl)))

Итак, я хочу узнать, есть ли способ предоставить конкретное изображение в виде фигуры?

Ответы [ 3 ]

22 голосов
/ 23 марта 2016

Вот минималистский geom для отображения растровых изображений вместо точек,

library(ggplot2)
library(grid)

## replace by a named list with matrices to be displayed
## by rasterGrob
.flaglist <- list("ar" = matrix(c("blue", "white", "blue"), 1), 
                  "fr" = matrix(c("blue", "white", "red"), 1))

flagGrob <- function(x, y, country, size=1, alpha=1){
  grob(x=x, y=y, country=country, size=size, cl = "flag")
}

drawDetails.flag <- function(x, recording=FALSE){

  for(ii in seq_along(x$country)){
    grid.raster(x$x[ii], x$y[ii], 
                width = x$size[ii]*unit(1,"mm"), height = x$size[ii]*unit(0.5,"mm"),
                image = .flaglist[[x$country[[ii]]]], interpolate=FALSE)
  }
}


scale_country <- function(..., guide = "legend") {
  sc <- discrete_scale("country", "identity", scales::identity_pal(), ..., guide = guide)

  sc$super <- ScaleDiscreteIdentity
  class(sc) <- class(ScaleDiscreteIdentity)
  sc
}

GeomFlag <- ggproto("GeomFlag", Geom,
                    required_aes = c("x", "y", "country"),
                    default_aes = aes(size = 5, country="fr"),

                    draw_key = function (data, params, size) 
                    {
                      flagGrob(0.5,0.5, country=data$country,  size=data$size)
                    },

                    draw_group = function(data, panel_scales, coord) {
                      coords <- coord$transform(data, panel_scales)     
                      flagGrob(coords$x, coords$y, coords$country, coords$size)
                    }
)

geom_flag <- function(mapping = NULL, data = NULL, stat = "identity",
                      position = "identity", na.rm = FALSE, show.legend = NA, 
                      inherit.aes = TRUE, ...) {
  layer(
    geom = GeomFlag, mapping = mapping,  data = data, stat = stat, 
    position = position, show.legend = show.legend, inherit.aes = inherit.aes,
    params = list(na.rm = na.rm, ...)
  )
}


set.seed(1234)
d <- data.frame(x=rnorm(10), y=rnorm(10), 
                country=sample(c("ar","fr"), 10, TRUE), 
                stringsAsFactors = FALSE)


ggplot(d, aes(x=x, y=y, country=country, size=x)) + 
  geom_flag() + 
  scale_country()

enter image description here

(вывод из пакета ggflags)

13 голосов
/ 13 апреля 2017

Для этого существует библиотека с именем ggimage. Смотрите вступительная виньетка здесь

Вам просто нужно добавить в ваш data.frame столбец с адресом изображений, который можно сохранить в Интернете или локально на вашем компьютере, а затем вы можете использовать geom_image():

library("ggplot2")
library("ggimage")

# create a df

set.seed(2017-02-21)
d <- data.frame(x = rnorm(10),
                y = rnorm(10),
                image = sample(c("https://www.r-project.org/logo/Rlogo.png",
                                 "https://jeroenooms.github.io/images/frink.png"),
                               size=10, replace = TRUE)
                )
# plot2
  ggplot(d, aes(x, y)) + geom_image(aes(image=image), size=.05)

enter image description here

пс. Обратите внимание, что ggimage зависит от EBImage . Так что для установки gginamge мне пришлось сделать это:

# install EBImage
  source("https://bioconductor.org/biocLite.R")
  biocLite("EBImage")
# install ggimage
  install.packages("ggimage")
4 голосов
/ 02 февраля 2010

Во-первых, вот ваш ответ:

Чтобы показать, как лучше использовать виджеты для представления дифференциации данных, я приведу пример черновых граней в галерее R-графа .:

alt text
(источник: free.fr )

Весь код для генерации этого примера доступен на сайте.

В качестве альтернативы, посмотрите ggplot stat_spoke для простого виджета: alt text
(источник: had.co.nz )

grImport предоставляет механизм для импорта простых PDF-изображений на график для использования в качестве точек.

Теперь следует критическому анализу вашего примера.


Это не диаграмма рассеяния. По сути, это потоковый список упорядоченных точек данных, в которых цвет используется для обозначения одной из текстовых переменных, а неинформативный и избыточный виджет используется для кадрирования данных, но в остальном не обеспечивает визуальной обратной связи с точки зрения размера или формы.

Это не очень хороший график, потому что он совершенно не отвечает на поставленный вопрос «Платит ли больше результатов для достижения лучших результатов», и оставляет читателя в борьбе за то, чтобы самим сделать этот вывод (и этот другой график, если необходимо) сами.

Кроме того, авторы потратили впустую оси x, y - которые могли бы быть использованы для позиционирования элементов по исходящим и результатам, чтобы обеспечить визуальное понимание соотношения цены и качества. Вместо этого они решили упорядочить значки по соотношению стоимости на душу населения к средней ставке выпускников, что является своего рода полезным, но не отвечает на поставленный вопрос и не позволяет провести прямое визуальное сравнение относительного соотношения между колледжами, или связь между стоимостью и стоимостью.

Как я уже сказал, по моему мнению, это плохой график, и вашим читателям было бы плохо, если бы вы его повторили.

...