-
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
Minimal PR for unit support in positional aesthetics #5690
base: main
Are you sure you want to change the base?
Conversation
@teunbrand I wanted to circle back around here at some point and see if there's still potential for this. I think a principled inclusion of unit support would be very cool and help improve a lot of stuff at the annotation level. This PR isn't all the way there, but is much less hacky than the previous version, and I'd be happy to iterate on it if there's interest. |
I'm happy to help think about the code and I still like the concept a lot. It is great that this PR is much less invasive than the previous one. However I feel like the sheer scope of this PR is outside my jurisdiction and the decision on whether to forge ahead with this is in @thomasp85's hands. If he doesn't reply here after invoking his name, I'll be sure to bring it up the next time Thomas and I have a chat (in ~2 weeks time). |
I fully embrace the need for what this PR wants to solve. However, I'm very hesitant to retrofit this into ggplot2 in the way that this PR proposes. I do think that perhaps we can have the best of both worlds with some changes to |
Cool, thanks for entertaining the suggestion! Let me know if you want help for the next iteration. Two other less invasive approaches also just occurred to me: Move coord_mapping into a layer parameter or wrapper functionOne less invasive (but still general) solution is to add a parameter to Such an approach could replace this: geom_text(aes(
label = name,
x = stage(var1, after_coord = unit(x, "native") + unit(10, "pt"))
)) With something like this: geom_text(
aes(label = name, x = var1),
coord_mapping = aes(x = unit(x, "native") + unit(10, "pt"))
) or maybe this: geom_text(aes(label = name, x = var1)) |> after_coord(aes(x = unit(x, "native") + unit(10, "pt"))) This could be a more general solution than Add
|
Following discussion on #5609 and helpful feedback from @teunbrand, @pmur002, and @thomasp85 (thanks!), this is a second proof-of-concept PR for unit support in positional aesthetics in ggplot2. This PR focuses on being a minimal implementation without hacking on the unit type system.
The basic idea here is to instead add another aesthetic evaluation stage,
after_coord
, which is akin toafter_stat
orafter_scale
. It is implemented by passing aesthetic mappings withstage()
orafter_coord()
through toCoord$transform()
via thepanel_params
argument, which is already passed through to that function.Coord$transform()
can then apply theafter_coord
mappings after it does its ownCoord
-specific transformations.Some demos:
With polar coordinates:
Some notes:
The main workhorse here is
compute_staged_aes()
, which is based on a chunk of code I factored out ofGeom$use_defaults()
for computingafter_scale
aesthetics.after_coord()
is a bit different fromafter_scale()
in that it provides the placeholder valueInf
to earlier parts of the pipeline, because it needed a non-unit()
value that would not affect scales (see the comment on that function). There should probably be a better way to do this.the implementation moves current
Coord$transform()
implementations intoCoord$transform_numeric()
. This was done so thatGeom
s (which already callCoord$transform()
) would not have to opt in to supportingunit()
s; instead,Coord
s in extension packages can opt-in simply by renaming theirtransform()
methods totransform_numeric()
. An alternative might be to leaveCoord$transform()
as-is but add aCoord$transform_unit()
method, and then haveGeom
s opt-in to unit support instead of havingCoord
s opt-in to it. I think havingCoord
s opt-in is better since it requires fewer changes in extension packages and propagates support more easily, but I could be wrong.No hacking on
unit()
is required for this implementation, nor any upstream changes to {grid}. A small set of compatibility functions would need to be added to {vctrs}, akin to how {vctrs} supplies compatibility functions for other base-R types, like Dates. These are confined toR/utilities-unit.R
, which could be turned into a PR on {vctrs}.In exchange for less "magic", this approach is a little more verbose than my original proposal and requires a little more understanding of the ggplot2 pipeline for users. Something like this in the current proposal:
could be written as this in the original proposal:
However, this new proposal has more consistent semantics and no heuristic hacking of the
unit()
datatype :). Aggunit()
subtype ofunit()
with consistent coercion rules for numerics could still be created to improve the syntax a bit (also without heuristic hacking ofunit()
); allowing something like this to work:Adding shortcuts like
as_pt()
could shorten this still, to:The
ggunit()
subtype could be added to ggplot2, or if that is not desired, I'd be happy to create a small extension package with that capability.