-
Notifications
You must be signed in to change notification settings - Fork 2k
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
Color scale reuse #5484
base: main
Are you sure you want to change the base?
Color scale reuse #5484
Conversation
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.
Hi Steve, thanks for the PR! I haven't looked in-depth yet, but here are some early thoughts.
I really like seeing that so many duplicated lines of code can be reduced, that's great!
I'm not entirely sure about reuse_all_colour_scales()
though, because of the following reasons.
- It invites users to ask for similar functions for other aesthetics that are harder to implement.
- The pattern of defining scales as
NULL
and only filling them in later is harder to reason about than necessary. - I'm uncertain that binding the produced scales to the parent environment is fool-proof enough. If a user just calls it in their interactive session, are you then just populating their global environment with new scales?
If we could find a way to pass additional arguments to the reused scales, we might also simplify all (or many of) the scale_{aes}_manual()
and scale_{aes}_identity()
functions. Lastly, I know that I'm the one who suggested the reuse_scale()
name, but perhaps there are better names out there that more clearly convey the intent of the function (I'm willing to accept there might not be).
R/scale-brewer.R
Outdated
scale_fill_brewer <- function(..., type = "seq", palette = 1, direction = 1, aesthetics = "fill") { | ||
discrete_scale(aesthetics, palette = brewer_pal(type, palette, direction), ...) | ||
} | ||
scale_fill_brewer <- NULL |
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 understand something needs to be declared for the documentation to work out, but I don't think it helps code readability to assign NULL
here. It might be clearer to just use reuse_scale()
directly here (and other places where scales are declared as NULL
), because then one doesn't have to lookup where and how these scales are instantiated.
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.
Yeah, I've waffled back and forth on how to initialize the fill scales.
On one hand, as you said, readability is best when it's defined in place. However, if each is defined individually, they'd be defined twice with the reuse_all function. Also, defining each one in place makes it easy to miss one or accidentally assign the wrong colour scale to a fill scale.
How about using comments like this?
scale_fill_brewer <- NULL # definition added in `reuse_all_colour_scales()`
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.
Also, an option which may make more sense now that fill scales are much simpler is moving all fill scales into a scale-fill.r
file. If there's a scale-colour.r
file, might as well be one for fill.
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 agree with Teun that having this delayed assignment is not a good direction for readability. Also, the point of the automatic creation kind of falls apart when you have to keep track of all the assignments anyway. So, the optimal approach IMO is to forego the whole reuse_all
pattern
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.
@thomasp85
Sounds good. That makes sense. Any thoughts on moving all the fill scales to a scale-fill.r
file? I think it'd help people making extensions if we can say "To make a new color scale, copy everything in the scale-fill.r
but replace 'fill' with your own."
See also #5484 (comment)
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.
Yeah, that could work
An overall goal here is to try to make the fill aesthetic as a built-in test of ggplot's extensibility. If ggplot implements the fill aesthetic just like an extension package would define it's own custom color aesthetic, then any issues or pain points regarding color aesthetics are clear within the ggplot build. Another step in that overarching goal is to remove "fill" from |
Also initialized the scales with `reuse_scale()` This commit touched a LOT of files, so I'm not committing the docs.
This PR takes a different approach to #5471 based on a suggestion by @teunbrand.
Changes
reuse_scale
can copy a colour scale for use with another color-based aesthetic. For examplescale_fill_grey <- reuse_scale(scale_colour_grey, "fill")
scale_fill*
functions were moved toscale-fill.R
.zxx.R
were moved toscale-colour-ordinal.R
andscale-color.R
.zxx.R
is no longer needed.Main benefit
Bonus benefit
scale-fill.R
file as a template. This update accomplishes most of that. The only aspect remaining is thatguide_colourbar
still has fill hardcoded.To do
Before finalizing PR:
type
is handled for US vs UK spellingMaybe address in another PR:
scale-hue.R
only have colour functions in them (no fill functions), rename toscale-colour-hue.R
.guide_colourbar
. Then modifyguide_colourbar
to allow fill to be added after the fact. Add fill toguide_colourbar
inscale-fill.R
.My tests