-
Notifications
You must be signed in to change notification settings - Fork 3.4k
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
Reflecting current set of CSS selectors : context.selectors not being carried over in eval contexts #2619
Comments
You lost me at:
I'm not sure why it should generate something other than what you've written. If you call the mixin from within To me, this seems like a situation of designing yourself into a corner. You designed your mixins and classes in a particular way, but then want to change the compiler from expected output? I dunno, it just seems like the proposed solution is a product of overthinking. There are a number of other ways to get to the proposed output other than adding a new feature. |
Let me rephrase: Ideally I would like to write a mixin that generates There's no built-in way to capture what the
While there are other ways to accomplish this than reflecting on the current selector and extracting the base of the CSS class that way, those methods are all leaky abstractions that place additional burden on the consumers of the mixin library. That's something I specifically want to try and avoid, because none of it scales. (Keep in mind; this presentation of my use-case is a severe reduction of what the actual thing looks like!) As for this being a request to 'add new features': i'm not asking for new core features at all. I'm only asking whether the non-propagation of the selector set through mixin calls or ruleset calls is a conscious choice, or whether it is in fact an oversight or bug. (And if it's a bug: whether fixing it would break other things that are implicitly relying on the 'broken' behavior.) I'm not asking for the custom functions that do the actual reflecting to become part of Less core. Those are supposed to stay as plugins.
Reflecting or capturing the current selector has been a part of Less's main competitor Sass for a long, long time. And it's most commonly used for exactly the kind of pattern I'm explaining here. |
I agree with @matthew-dean, the use-case example is too close to an XY problem. In other words, the code (using the proposed feature) in your example only tries to fight/cancel So not diminishing the feature itself in general, I also think the example just prays for another solutions. And though I probably would sympathize some extra "selector manipulation facility" in general, if it's only about "sniffing a ruleset the mixin is invoked in and change the output depending on that" - ouch... If you need to play with selectors in a mixin, pass them explicitly, e.g.:
Obviously those are simplified examples just for illustration. Indeed, there're many ways and exact code may vastly differ depending on the actual needs. Especially considering those "stripping default button styles, the quirks of So in summary: |
It's not. Please go back and re-read. The fact that I change output (i.e. either render with or without child combinator) is a nice optimization extra that makes the CSS cheaper to process by a browser during the style matching phase. It's not the core motivation behind the use-case. The core motivation is being able to generate a child class selector based on a common prefix in the parent class selector without having to explicitly hand-type the child class selector and pass it in as a parameter, which is nothing but extranuous noise.
The full use-case involves some 150 odd lines of CSS with intricate border/margin/padding computations to correctly align a font-icon on the left-hand/right-hand/etc. side of a text-label within the button while both text and icon can be individually sized with either px or em units and the button itself either follows a content-width (stretch-to-fit) model or fills out the available continer width. This is not trivial stuff that you want an inexperienced end-consumer to end up having to call by hand. Moreover, it's already cut up into componentized logic for setting button dimensions, icon dimensions, icon alignments, text-label alignment, spacing between icon and text label, etc. This is not a 'non-modular approach' at all.
Method 1 is the first I tried, actually: it breaks when you want to add additional properties to the ruleset itself. E.g.
Methods 2 and 3 break in the same way. Method 4 actually works though, and it's what I'm currently using. However, it's an ugly wart to have to pass in the name of the base class that way and to have to use two separate mixins (or an overloaded signature). Being able to capture the current ruleset selector would mean you could pull out that base class name automatically and automatically determine which version (main class or subclass) of the logic to run.
Again, as I have already mentioned in my last reply: there is NO NEW LANGUAGE FEATURE INVOLVED. The actual reflecting on the selector is A PLUGIN FUNCTION and is designed to REMAIN AS SUCH. I only need to know if it's a compiler bug or oversight that the actual running set of selectors is not copy-constructed into new eval contexts, because that's the only thing which is breaking the plugin I already wrote. (Geez... it would've been quicker to just hack in the change and run the unit test suite to find out for myself.) |
With "changing output" I directly mean "child class name extracted as a substring from a parent class name". Actually I'm a bit surprised that the idea of a technique like this comes from you (considering critiques like this and similar).
I probably missed this statement earlier, sorry (I was more focused on those parts you've stresses with BOO :). Actually, current context including parent selectors is available in functions ( |
Detached rulesets in Less query the full call-site scope before the declaration scope when evaluating variables and yes; I still believe that's a stupid thing to do. Reading back the current selector is not the same. The critical differences are:
|
Then you do have a bug; because
Ehm.. Yeah. Sorry about that; had a few long stressful days... |
@rjgotten Sorry, for false closing (I was editing my post, for a third time today, doh!, and accidentaly pushed the wrong button).
Most likely you've accidentally tried P.S. I updated above gist with "mixin selectors" fix, now it works as expected (test). Notice however it does not handle |
If there's a legitimate bug, yes, by all means let's fix it. My comment was only that there a variety of other supported patterns with no additional complexity that would achieve the same output. But if that wasn't what you were asking about, my mistake. |
Btw. I did more tests with the my prototype above and so far found the only bug related to detached rulesets. If a DR is defined before it's called - everything is fine. However, if it's defined after use, its P.S. I have added some labels just to have this categorised in some way (mostly just for others to have a clue w/o reading it all (+ easy query)). |
@seven-phases-max |
This issue has been automatically marked as stale because it has not had recent activity. It will be closed if no further activity occurs. Thank you for your contributions. |
In working on some complex framework components written in Less, I have a few times already come across situations where it would have been beneficial to be able to reflect the current set of CSS selectors into a variable for processing.
I've succesfully built a custom function that does just this, but it runs afoul of an issue deeper in the Less compiler.
My usecase
My current case and the one I'll treat here as an example for the reflection feature and how it leads up to the problem with the compiler, is a stylized button that contains both a left/right positioned font-icon and a text span that has horizontal ellipsis applied for overflow and can have its own text left/right/center aligned within the remaining space on the button. E.g.
Currently, I have a mixin which renders the base of the button's CSS; handling things like stripping default button styles, setting proper padding, line-height, etc. and handling the quirks of
::moz-focus-inner
on older Firefox builds.As part of this mixin, I have to set both CSS for the button itself and for the text span inside it and both class names have a shared root in the name. (OOCSS principles. This pattern is also re-used across multiple of our sites.)
No problem right?
Call via:
And done:
Where my usecase breaks
This is a nice pattern, but it still breaks when we subclass the button, e.g. make a compact variation:
It should generate
.imp-button--compact > .imp-button-text
instead, so that we don't have to double up on all the nested classes on subcomponents for different variations.Arriving at a solution
I've come up with a few complex builder patterns to work around this, but none of them really fit the bill 100% and really, they all feel like just stacking on needless complexity, when I could solve this tout-de-suite if I could have a little look at the right-hand key-selector and chop off the part after the double dash that we use in our convention and thus simply hide all the complexity from my libraries' consumers.
E.g.
Output:
I've already implemented a generic
selectors()
function to get the full set of active selectors (taking into account&
and variable-name token replacement) and akey-selectors()
function that does the same, but returns the key-selector (i.e. the most right-hand element) only.It took some digging to figure out how to work with the data structures that the compiler has in place, but it was very doable as a pluggable extension function that walks back over the stack of selectors maintained in the Eval context of the function call node and assembles them into the full set of complete selectors. If that sounds as a surprise: yes they are accessible via
this.context.selectors
, as functions are executed withthis
bound to the node instance.(Friendly Note: The fact that you can acccess the current eval context that way is also a god-sent that empowers plugin functions to fullfill several very complex scenarios. Please, please, please; don't ever change that!)
The actual problem
The problem is that this works fine for regular nested rulesets, but currently BREAKS when performed directly or indirectly inside a mixin or detached ruleset.
I traced down the root cause, which is the fact that when
callEval
is executed for a DetachedRuleset or MixinDefinition node, a new Eval context is created and the selectors are not part of the copy-constructed properties. Ergo, inside a mixin or ruleset callthis.context.selectors
resets to the empty set.Would there be any harm in adding
"selectors"
to the array of cloned properties for Eval contexts? (I could imagine it breaking the join-selector visitor in some way, but I'm not sure.)The text was updated successfully, but these errors were encountered: