如何以特定值为条件获取geom_line的多种颜色

时间:2012-03-03 04:17:02

标签: r ggplot2

我想要绘制以下数据框。我想知道是否有可能将连接我的结果变量(stackOne $ y)的部分线以不同颜色着色,具体取决于它是否小于某个值。例如,我希望部分线条在2.2以下变成红色。

id<-rep(c(1,2,3),each=3)
y<-rnorm(9,2,1)
x<-rep(c(1,2,3),3)
stackOne<-data.frame(cbind(id,y,x))
p<-ggplot(stackOne,aes(x=x,y=y)) +
geom_point()+
geom_line(aes(group=id))
p

谢谢!

2 个答案:

答案 0 :(得分:9)

这里至少有几个选项。第一个非常简单,一般(因为它不限于直线段)和精确,但使用基础plot而不是ggplot。第二个使用ggplot,但稍微复杂一点,颜色过渡不会100%精确(但足够接近,只要你指定一个合适的分辨率......阅读)。

<强>碱

如果您愿意使用base绘图功能而不是ggplot,则可以将绘图区域剪切到阈值(2.2)以上,然后以您喜欢的颜色绘制线段,然后剪辑到阈值以下的区域,并再次以红色绘制。虽然第一个剪辑是严格不必要的,但它可以防止过度绘制不同的颜色,这看起来有点哑。

threshold <- 2.2
set.seed(123)
stackOne=data.frame(id=rep(c(1,2,3),each=3),
                y=rnorm(9,2,1),
                x=rep(c(1,2,3),3))
# create a second df to hold segment data
d <- stackOne 
d$y2 <- c(d$y[-1], NA)
d$x2 <- c(d$x[-1], NA) 
d <- d[-findInterval(unique(d$id), d$id), ] # remove last row for each group

plot(stackOne[, 3:2], pch=20)
# clip to region above the threshold
clip(min(stackOne$x), max(stackOne$x), threshold, max(stackOne$y))
segments(d$x, d$y, d$x2, d$y2, lwd=2)
# clip to region below the threshold
clip(min(stackOne$x), max(stackOne$x), min(stackOne$y), threshold)
segments(d$x, d$y, d$x2, d$y2, lwd=2, col='red')
points(stackOne[, 3:2], pch=20) # plot points again so they lie over lines

base package, changing line colours

<强> ggplot:

如果您想要或需要使用ggplot,您可以考虑以下内容......

一种解决方案是使用geom_line(aes(group=id, color = y < 2.2)),但是这将根据每个段开头的点的y值来指定颜色。我相信你不仅要在节点上进行颜色变换,而且要想在一条线超过你给定的2.2的阈值时进行颜色变化。我并不熟悉ggplot,但实现此目的的一种方法是通过沿连接现有点的线创建新点来制作更高分辨率的数据版本,然后使用color = y < 2.2参数达到预期的效果。

例如:

threshold <- 2.2 # set colour-transition threshold
yres <- 0.01 # y-resolution (accuracy of colour change location)

d <- stackOne # for code simplification
# new cols for point coordinates of line end
d$y2 <- c(d$y[-1], NA)
d$x2 <- c(d$x[-1], NA) 
d <- d[-findInterval(unique(d$id), d$id), ] # remove last row for each group
# new high-resolution y coordinates between each pair within each group
y.new <- apply(d, 1, function(x) {
  seq(x['y'], x['y2'], yres*sign(x['y2'] - x['y']))
})
d$len <- sapply(y.new, length) # length of each series of points
# new high-resolution x coordinates corresponding with new y-coords
x.new <- apply(d, 1, function(x) {
  seq(x['x'], x['x2'], length.out=x['len'])
})
id <- rep(seq_along(y.new), d$len) # new group id vector
y.new <- unlist(y.new)
x.new <- unlist(x.new)
d.new <- data.frame(id=id, x=x.new, y=y.new)

p <- ggplot(d.new, aes(x=x,y=y)) +
  geom_line(aes(group=d.new$id, color=d.new$y < threshold))+
  geom_point(data=stackOne)+
  scale_color_discrete(sprintf('Below %s', threshold))
p

conditional line colour - ggplot

通过ggplot函数可能有一种方法可以做到这一点,但在此期间我希望这会有所帮助。我无法弄清楚如何将ggplotGrob绘制到剪切的viewport中(相反,它似乎只是缩放图)。如果你想让颜色以某个x值阈值为条件,那么显然需要进行一些调整。

答案 1 :(得分:3)

受到我的 answer 中的人的鼓励,他们提出了一个较新但相关的问题,我还将在这里分享一个更易于使用的近似问题。

可以使用 ggforce::geom_link2() 来插值线并使用 after_stat() 在插值后分配正确的颜色,而不是准确地插入正确的值。如果您想要更高的精度,您可以增加该函数的 n

library(ggplot2)
library(ggforce)
#> Warning: package 'ggforce' was built under R version 4.0.3

set.seed(123)
stackOne = data.frame(id = rep(c(1, 2, 3), each = 3),
                      y = rnorm(9, 2, 1),
                      x = rep(c(1, 2, 3), 3))

ggplot(stackOne, aes(x = x, y = y)) +
  geom_point() +
  geom_link2(
    aes(group = id,
        colour = after_stat(y < 2.2))
  ) +
  scale_colour_manual(
    values = c("black", "red")
  )

reprex package (v1.0.0) 于 2021 年 3 月 26 日创建