Ggplot2 - аннотировать вне участка
Я хотел бы связать значения размера выборки с точками на графике. Я могу использовать geom_text
для размещения чисел рядом с точками, но это грязно. Было бы намного проще выровнять их по внешнему краю сюжета.
Например, у меня есть:
df=data.frame(y=c("cat1","cat2","cat3"),x=c(12,10,14),n=c(5,15,20))
ggplot(df,aes(x=x,y=y,label=n))+geom_point()+geom_text(size=8,hjust=-0.5)
Что производит этот сюжет:
![enter image description here]()
Я бы предпочел нечто большее:
![enter image description here]()
Я знаю, что могу создать второй сюжет и использовать grid.arrange
(a la этот пост), но было бы утомительно определить интервал textGrobs на строку с осью y. Есть ли более простой способ сделать это? Спасибо!
Ответы
Ответ 1
Вам не нужно рисовать второй сюжет. Вы можете использовать annotation_custom
для размещения глыбов в любом месте внутри или вне области графика. Позиционирование грыз в терминах координат данных. Предполагая, что "5", "10", "15" совпадают с "cat1", "cat2", "cat3", по вертикальному расположению textGrobs позаботятся - y-координаты ваших трех textGrobs даются y-координаты трех точек данных. По умолчанию ggplot2
зажимает grobs в области построения графика, но отсечение может быть отменено. Соответствующий запас должен быть расширен, чтобы освободить место для grob. Следующее (используя ggplot2 0.9.2) дает график, аналогичный вашему второму сюжету:
library (ggplot2)
library(grid)
df=data.frame(y=c("cat1","cat2","cat3"),x=c(12,10,14),n=c(5,15,20))
p <- ggplot(df, aes(x,y)) + geom_point() + # Base plot
theme(plot.margin = unit(c(1,3,1,1), "lines")) # Make room for the grob
for (i in 1:length(df$n)) {
p <- p + annotation_custom(
grob = textGrob(label = df$n[i], hjust = 0, gp = gpar(cex = 1.5)),
ymin = df$y[i], # Vertical position of the textGrob
ymax = df$y[i],
xmin = 14.3, # Note: The grobs are positioned outside the plot area
xmax = 14.3)
}
# Code to override clipping
gt <- ggplot_gtable(ggplot_build(p))
gt$layout$clip[gt$layout$name == "panel"] <- "off"
grid.draw(gt)
![enter image description here]()
Ответ 2
Теперь это легко сделать с ggplot2 3.0.0, так как теперь обрезание может быть отключено на графиках путем вызова coord_cartesian(clip = 'off')
, как в примере ниже.
# Generate data
df <- data.frame(y=c("cat1","cat2","cat3"),
x=c(12,10,14),
n=c(5,15,20))
# Create the plot
ggplot(df,aes(x=x,y=y,label=n)) +
geom_point()+
geom_text(x = 14.25, # Set the position of the text to always be at '14.25'
hjust = 0,
size = 8) +
coord_cartesian(xlim = c(10, 14), # This focuses the x-axis on the range of interest
clip = 'off') + # This keeps the labels from disappearing
theme(plot.margin = unit(c(1,3,1,1), "lines")) # This widens the right margin
![enter image description here]()
Ответ 3
Упрощенное решение на основе grid
require(grid)
df = data.frame(y = c("cat1", "cat2", "cat3"), x = c(12, 10, 14), n = c(5, 15, 20))
p <- ggplot(df, aes(x, y)) + geom_point() + # Base plot
theme(plot.margin = unit(c(1, 3, 1, 1), "lines"))
p
grid.text("20", x = unit(0.91, "npc"), y = unit(0.80, "npc"))
grid.text("15", x = unit(0.91, "npc"), y = unit(0.56, "npc"))
grid.text("5", x = unit(0.91, "npc"), y = unit(0.31, "npc"))