-
Notifications
You must be signed in to change notification settings - Fork 2.1k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
POC: double dispatch for ggplot_add()
#5537
base: main
Are you sure you want to change the base?
Conversation
I don't think it's a good idea to implement custom double dispatch in ggplot2, and it would instead be better to use S7 (which would require switching the ggplot2 base class to a S7, but that in theory should be easy). |
Thanks for the advice, Hadley! I'll experiment around with S7 for a bit then. |
Because a ggplot object as S3 class might be baked into many extension assumptions, I haven't converted it to an S7 class (yet). devtools::load_all("~/packages/ggplot2")
#> ℹ Loading ggplot2
library(S7)
# Making a ggplot subclass
p <- ggplot(mpg, aes(displ, cty))
class(p) <- union("foo", class(p))
# Adding custom method for adding layers
method(ggplot_add, list(object = new_S3_class("Layer"), plot = new_S3_class("foo"))) <-
function(object, plot, object_name) {
print("I'm adding a layer to foo")
# This is the awkward part:
method(ggplot_add, list(object = new_S3_class("Layer"), plot = class_ggplot))(
object, plot, object_name
)
}
# Should print text as well as the plot
p + geom_point()
#> [1] "I'm adding a layer to foo" Created on 2023-11-26 with reprex v2.0.2 |
You should be able to create an S7 class with exactly the same S3 class as previously (since S7 is designed to be backward compatible with S3); i.e. you should be able to make the without affecting extensions (unless they use |
R/plot-construction.R
Outdated
ggplot_add <- new_generic( | ||
"ggplot_add", | ||
dispatch_args = c("object", "plot"), | ||
fun = function(object, plot, object_name) S7_dispatch() |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I'd be tempted to leave object_name
as an optional argument for the methods (i.e. just rely on the default constructor which will also include ...
, which I'd now consider to be best practice in a generic. OTOH it might be better to make that change in a separate PR, as I'd probably now call this error_arg
, and I'd expect a matching error_call
argument in order to apply our current error best practices.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Fair point, I'll leave this comment unresolved to not forget about this for now.
Thanks for working on this! I don't know if S7 is ready for use in such a high profile package, but I think it's extremely worthwhile to try it out so we can make changes to S7 if needed. |
Thanks for pointing out all these improvements Hadley!
I had a shallow attempt at implementing a ggplot class, but the issue I kept running into is that the rest of the codebase always extracts information with
This would be a neat mechanism to have, or more broadly a 'dispatch on the next s3 class'-mechanism.
I think it might be good to start using S7 for this double dispatch issue, and leave converting S3 to S7 classes for another day when the need arises. |
Could you add a |
This comment was marked as resolved.
This comment was marked as resolved.
I've converted this PR to a draft to indicate that it is 'on hold' untill S7 is stable enough. |
This PR explores suggestions from #3818 to use double dispatch to
+
objects to ggplot subclasses. The ultimate goal is to fix #3986 and fix #5536.Double dispatch is implemented for layers only at the moment, but we can expand that once we agree on the approach, which I'm still happy to change.
It currently borrows from the {vctrs} style of S3 double dispatch, though we could consider using {S7} if we're willing to take on an extra dependency.
Open questions:
Small demo that a custom method is invoked:
Created on 2023-11-23 with reprex v2.0.2