将有图例的地块与没有图例的地块结合起来

Combine plots that have a legend with one that doesn't

我可以组合三个没有任何图例的地块,但是当我向其中两个地块添加图例时,我无法将它们组合起来。如何将一些带有图例的图与没有图例的图结合起来?

这是我如何组合没有图例的情节的示例:

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"))

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

# 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() 

这是它的样子:

# 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"))

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

使用包 cowplot 中的 plot_grid() 可能会有所帮助——至少它确实设法将有和没有图例的图结合起来,但要使图完美对齐似乎很棘手。我 post 无论如何,因为这可能会有帮助,也许其他人知道如何解决对齐问题

注意,还向线图中添加了 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')

您收到错误的原因是因为您的 3 个 gtable object 之一有 5 列,而另一个有 6 列(p.3 object 有 5 列,因为您排除了传奇)。您可以通过以下操作查看 gtable objects 的不同布局:

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

我不会 post 这里的图片,因为它们会影响答案,但我认为值得自己检查一下。

无论如何,我们可以做的就是在您的 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"))

最后一件事是 y-axis 最后一张图中的标题与标签略有重叠,这是因为我们采用了第一个图中的尺寸并且第一个图中的标签较小(因此y-axis 标签需要更少 space)。要解决此问题,您可以将 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))

重新运行得到:

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

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"))