ggrepel
ggrepel copied to clipboard
Multiple geom_text_repel give overlapping labels
geom_text_repel
and geom_label_repel
do not "see" labels plotted by other calls of geom_*_repel resulting in overlapping labels.
Here I plot the same text twice using geom_text_repel
, once in blue and one in red (this is pointless of course, just for sake of example)
dat <- data.frame(x= 1:10, y= 1:10, text= LETTERS[1:10])
gg <- ggplot(data= dat, aes(x= x, y= y, label= text)) +
geom_point() +
geom_text_repel(colour= 'blue') +
geom_text_repel(colour= 'red') +
xlim(-10, 30) +
ylim(-10, 30)
The blue and red labels overlap each other:
In real life I'm using both geom_text_repel
and geom_label_repel
on the same plot and I get overlapping texts when the text from the two calls are close to each other.
Any fix much appreciated! (Great package anyway!) Dario
Session info:
R version 3.5.1 (2018-07-02)
Platform: x86_64-conda_cos6-linux-gnu (64-bit)
Running under: Ubuntu 18.04.4 LTS
Matrix products: default
BLAS/LAPACK: /home/dario/miniconda3/envs/tritume/lib/R/lib/libRlapack.so
locale:
[1] LC_CTYPE=en_GB.UTF-8 LC_NUMERIC=C LC_TIME=en_GB.UTF-8 LC_COLLATE=en_GB.UTF-8 LC_MONETARY=en_GB.UTF-8
[6] LC_MESSAGES=en_GB.UTF-8 LC_PAPER=en_GB.UTF-8 LC_NAME=C LC_ADDRESS=C LC_TELEPHONE=C
[11] LC_MEASUREMENT=en_GB.UTF-8 LC_IDENTIFICATION=C
attached base packages:
[1] stats graphics grDevices utils datasets methods base
other attached packages:
[1] ggrepel_0.8.2 ggplot2_3.3.0
loaded via a namespace (and not attached):
[1] Rcpp_1.0.4 digest_0.6.25 withr_2.1.2 crayon_1.3.4 dplyr_0.8.5 assertthat_0.2.1 grid_3.5.1 R6_2.4.1
[9] lifecycle_0.2.0 gtable_0.3.0 magrittr_1.5 scales_1.1.0 pillar_1.4.3 rlang_0.4.5 farver_2.0.3 labeling_0.3
[17] tools_3.5.1 glue_1.3.2 purrr_0.3.3 munsell_0.5.0 compiler_3.5.1 pkgconfig_2.0.3 colorspace_1.4-1 tidyselect_1.0.0
[25] tibble_2.1.3
Thanks for opening the issue, sharing code, and sharing a figure.
I would like for ggplot2 to allow each layer to depend on previous layers, but this is not currently supported.
Right now I can't think of a way to work around this issue, but maybe someone else has an idea?
+1 for this functionality.
+1 eagerly waiting as well...
A half-baked idea:
What if there's an option geom_text_repel(cascade = TRUE)
that tells ggrepel to write the x,y coordinates to a temporary file?
How it might work, step by step:
- The first call to
geom_text_repel()
creates a temporary file. ggrepel sees it is empty. It writes x,y coordinates to it. - The second call to
geom_text_repel()
reads the temporary file, and sees that it is not empty. So, it adds those x,y coordinates as additional points to repel away from. Then it appends its own x,y coordinates to the file. - The third call to
geom_text_repel()
reads the temporary file, and sees the x,y coords from the first two calls... etc.
That way, each call to geom_text_repel()
cumulatively appends more coordinates to repel away from.
What do you think?
I think it's a bit weird to use a temporary file for this, but I don't know how else to get the ggplot2 layers to "see" each other in a cascading way (kinda like CSS for HTML).
I don't think that ggrepel would have any way to know when the cascading is completed, so the temporary files would not be deleted at the end of the ggrepel calls. Instead, they would continue accumulating indefinitely with additional calls to ggrepel.
I'm not knowledgeable enough to know whether the cascade idea would work. But are there any new solutions to this? (+1) Right now I'm just cycling through seeds until randomness provides a decent result.
Not sure how much you're in contact with / part of the ggplot2 team, but I could imagine "layers being aware of each other" might be something they were or are thinking about themselfes.
It seems that the ggplot2 Google Group is not the best place to discuss this feature: https://groups.google.com/g/ggplot2/c/OAw_SijqzVg
I created a new post for discussion here: https://community.rstudio.com/t/feature-discussion-layers-aware-of-each-other-in-ggplot2/108156
Feel free to share your ideas, here or on the RStudio Community page.
Pull requests that implement a new feature to address this issue are welcome!
atusy from RStudio Community shared an excellent code example that we could use to address this issue:
https://community.rstudio.com/t/feature-discussion-layers-aware-of-each-other-in-ggplot2/108156/2
@aphalo I thought you might find this example interesting for your own work — I was not aware of ggplot_add()
.
library(ggrepel)
library(patchwork)
geom_text_repel2 <- function(...) {
layer <- ggrepel::geom_text_repel(...)
layer$ggrepel <- TRUE
class(layer) <- c("ggrepel", class(layer))
return(layer)
}
ggplot_add.ggrepel <- function(object, plot, object_name) {
if (any(do.call(c, lapply(plot$layer, function(x) x$ggrepel)))) {
warning(
"There is more than one ggrepel layers. ",
"This may cause overlap of labels"
)
}
# Optionally, one may modify `object` here.
NextMethod("ggplot_add")
}
dat <- data.frame(x= 1:10, y= 1:10, text= LETTERS[1:10])
p1 <- ggplot(data= dat, aes(x= x, y= y, label= text)) +
geom_point() +
geom_text_repel2(colour = 'blue', seed = 1) +
labs(title = "one layer")
p2 <- p1 + geom_text_repel2(colour = 'red', seed = 2) +
labs(title = "two layers")
#> Warning in ggplot_add.ggrepel(object, p, objectname): There is more than one
#> ggrepel layers. This may cause overlap of labels.
p1 + p2
This might be related, though I don't remember the details...
https://github.com/tidyverse/ggplot2/issues/3175