将具有图例的图表与不具有图例的图表组合在一起

时间:2016-06-24 15:32:28

标签: r plot ggplot2 legend

我可以结合三个没有任何图例的情节,但是当我在两个情节中添加图例时,我无法将它们组合在一起。如何将一些情节与传说与没有图例的情节结合起来?

以下是我如何在没有图例的情况下合并情节的示例:

library(ggplot2)
map_data <- data.frame(lon = c(-177.5, -172.5, -167.5, -162.5, -157.5, -152.5),
                       lat = c(87.5, 87.5, 87.5, 87.5, 87.5, 87.5),
                       ssta = c(-2.78999996, 0.09999999,  0.19999999,  0.06000000,  1.65999997, -0.41000000))

# one map
p.1 <- ggplot() + 
    borders("world", 
          colour="black", 
          fill="white") +
  geom_tile(data = map_data,
            aes(lon,
                lat)) + # no fill value so no legend
  coord_fixed() + 
  scale_fill_viridis(na.value = "transparent") +
  theme_minimal() 

# a second map, the same general shape
p.2 <- p.1

# a line plot
library(lubridate)
time_sequence <-  seq(dmy("01-01-1800"), 
                      dmy("01-01-1960"), 
                      by = 'month')
long_plot_data <- data.frame(variable = seq(dmy("01-01-1800"), 
                                            dmy("01-01-1960"), 
                                            by = 'month'),
                             value = runif(length(time_sequence)),
                             col = runif(length(time_sequence)))

p.3 <-  ggplot(long_plot_data, 
              aes(variable, 
                  value)) +
  geom_line() +
  theme_minimal()


# combine plots
library(grid)
grid.newpage()
grid.draw(rbind(ggplotGrob(p.1), 
                ggplotGrob(p.2),
                ggplotGrob(p.3),
                size = "last"))

enter image description here

现在,如果我在地图中添加填充值,我会得到一个图例,当我尝试以相同的方式组合时出现错误:

# one map
p.1 <- ggplot() + 
    borders("world", 
          colour="black", 
          fill="white") +
  geom_tile(data = map_data,
            aes(lon,
                lat,
                fill = ssta)) + # gives a legend
  coord_fixed() + 
  scale_fill_viridis(na.value = "transparent") +
  theme_minimal() 

以下是这样的:

enter image description here

# a second map, the same general shape
p.2 <- p.1

# a line plot
library(lubridate)
time_sequence <-  seq(dmy("01-01-1800"), 
                      dmy("01-01-1960"), 
                      by = 'month')
long_plot_data <- data.frame(variable = seq(dmy("01-01-1800"), 
                                            dmy("01-01-1960"), 
                                            by = 'month'),
                             value = runif(length(time_sequence)),
                             col = runif(length(time_sequence)))

p.3 <-  ggplot(long_plot_data, 
              aes(variable, 
                  value)) +
  geom_line() +
  theme_minimal()

现在我尝试将两个地图与图例和没有图例的线图合并:

# combine plots
library(grid)
grid.newpage()
grid.draw(rbind(ggplotGrob(p.1), 
                ggplotGrob(p.2),
                ggplotGrob(p.3),
                size = "last"))

输出为Error: ncol(x) == ncol(y) is not TRUE

但是,如果我在线图中添加一个图例,以便所有三个图都有图例,我可以将它们组合在一起:

p.3 <-  ggplot(long_plot_data, 
              aes(variable, 
                  value, 
                  colour = col)) +
  geom_line() +
  theme_minimal()


# combine plots
library(grid)
grid.newpage()
grid.draw(rbind(ggplotGrob(p.1), 
                ggplotGrob(p.2),
                ggplotGrob(p.3),
                size = "last"))

enter image description here

如何将两个图表与一个没有图例的图表合并?

3 个答案:

答案 0 :(得分:7)

您收到错误的原因是因为您的3个gtable对象中有一个有5列而另一个有6个(p.3对象有5个因为您排除了图例)。您可以通过执行以下操作来查看gtable对象的不同布局:

library(gtable)
gtable_show_layout(ggplotGrob(p.1))
gtable_show_layout(ggplotGrob(p.2))
gtable_show_layout(ggplotGrob(p.3))

我不会在这里发布图片,因为它们会影响答案,但我认为值得自己查看。

无论如何,我们可以做些什么来解决这个问题,只需在p.3 gtable上添加一个额外的列,然后使用size="first"选项(因为我们希望图例位于正确的位置我们传说中的情节。)

这应该有效:

library(grid)
library(gtable)

grid.draw(rbind(ggplotGrob(p.1), 
                ggplotGrob(p.2),
                gtable_add_cols(ggplotGrob(p.3),unit(1,"lines")),
                size = "first"))

enter image description here

最后一个问题是最后一个绘图上的y轴标题与标签略有重叠,这是因为我们从第一个绘图中获取了大小并且第一个绘图上的标签更小(因此y轴标签需要更少的空间)。要解决此问题,您可以将p.3代码略微更改为:

p.3<-  ggplot(long_plot_data, 
               aes(variable, 
                   value)) +
  geom_line() +
  theme_minimal() %+replace% theme(axis.title.y=element_text(margin=margin(0,20,0,0),angle=90))

Rerunning给了我们:

grid.draw(rbind(ggplotGrob(p.1), 
                ggplotGrob(p.2),
                gtable_add_cols(ggplotGrob(p.3),unit(1,"lines")),
                size = "first"))

enter image description here

答案 1 :(得分:2)

使用程序包plot_grid()中的cowplot可能会有所帮助 - 至少它确实能够将图表与包含图例的图表结合起来,但是让图表完美对齐似乎很棘手。无论如何,我发布,因为这可能会有所帮助,也许其他人知道如何解决对齐问题

注意,也将coord_fixed()添加到折线图中,以使其大小与地图相同。

p.3 <-  ggplot(long_plot_data, aes(variable, value)) +
  geom_line() +
  coord_fixed() + 
  theme_minimal() 

library(cowplot)
plot_grid(p.1,p.2,p.3,ncol=1, align='h')

enter image description here

答案 2 :(得分:2)

gtable_add_cols 答案不再有效,因为 3 个 gtable 对象现在有 11、11 和 9 列(在编写上述内容时,它们有 6、6 和 5)。所以你必须在最后一个图(p.3)中添加两列,例如:

g1 <- ggplotGrob(p.1)
g2 <- ggplotGrob(p.2)
g3 <- ggplotGrob(p.3)
g3 <- gtable_add_cols(g3, g1$widths[5:6])
grid.newpage()
grid.draw(rbind(g1,g2,g3,
                size = "max"))