-
-
Notifications
You must be signed in to change notification settings - Fork 4.5k
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
Support classes on nested components #2888
Conversation
539c0a8
to
04d6604
Compare
ba3677a
to
5214c75
Compare
5214c75
to
cd43340
Compare
We do want to support something providing this sort of functionality, but giving special treatment to the |
Thanks for the feedback. I'll keep this fully functional PR open for you as an inspiration then and will publish and use my fork in the mean time. |
After looking through the open issues and trying out the various workarounds, I see this PR as the only possibility to override css in a child from a parent in a safe way: If I try to pass a custom css class from a parent without using I think it's best to let the child decide what kind of things you should be able to override. The issue we face is that the optimizer removes classes from the css, when it's not used. So we need to provide a way to mark the class as used. I can't really think of good alternatives, but I'm not using Svelte for long. @Conduitry do you have alternative approaches you might want to share for discussion? :) The only alternatives I can think of:
I think No. 2 would be quite a bit more involved and I do not really see great benefits to it. If someone already uses |
How about passing all undeclared props down as attributes?
I see two problems here:
Let's paint that picture.. MyComponent.svelte:
Component.svelte:
Output:
Something like that. Writing this example feels a little funny, but it's a start. To make it a bit sweeter simple string concatenation would suffice as a default merge strategy since attributes may only contain string values anyway. DOM props are a bit more complicated I guess.. Most of this is inspired by the way VueJS handles inheritance of attributes. I'd love to see a feature like this, it would also make it a lot easier to extend existing components. Semantically speaking the components that we're writing are still simply wrappers of DOM nodes, which make me feel this is an essential feature. |
Finally weighing in, because I think we now have a good solution to this problem. For me the key issue is this:
Giving I think the right way to solve this issue is with CSS custom properties. It can easily be done today (just have a wrapper element that applies the custom properties using a class or an inline style), and this RFC proposes a quality-of-life improvement that would let you pass custom properties directly to components (by injecting the wrapper element for you). The RFC is still open for the time being but I anticipate something close to it being what we end up with, so I'll close this PR — thank you |
Joining the conversation late - but I've recently gotten to try out svelte (and love the idea! ❤️) and have run into this problem as well. The proposed RFC is great, but does not solve where I see the problem lies. There is a good amount of properties that should mostly be applied from a parent's point of view. We're talking stuff like It would be tiresome - and bloated - to include a class pass-through for every component or assigning custom properties (from the RFC linked) for all potential properties on every component, just in case it's gonna be used in layouts that requires it. Wrapping them in a wrapper div is certainly an option, but potentially creates 100s or 1000s (long lists, several lists etc.) of new elements in the DOM slowing down low-end devices. Ideally it'd be as easy as: <button class="button">
Like
<Icon icon="heart" class="button__icon" />
</button>
<style>
.button {
display: flex;
align-items: center;
}
.button__icon {
flex: none;
margin-left: .5rem;
}
</style> Problems I see:
I'd love to hear thoughts on this. Sorry in advance if I've missed something - still new to svelte. 🙏 Edit: For some reason I missed sheijne's comment, but it addresses some of my concerns as well. #2888 (comment) |
This is the probably wrong place for this discussion but to answer one of your questions:
The main reason using classes isn't a great solution is that it completely breaks encapsulation in a confusing way, the paren't shouldn't be dictating anything, the component itself should. The parent can pass things and the child can choose to use them or not but that is different: control is still in the hands of the component itself, not an arbitrary parent. This isn't a solution bvecuse svelte doesn't need a single root node in a component, should they be applied to all top level children? I don't think this would work. |
The point we're probably missing here: The main rationale for this PR is that, in my hones opinion, Svelte needs a way to support style overrides in an intuitive and close to plain HTML/CSS way. What I regard as intuitive is: Looking at how customizing of styles is being done when applying a typical CSS component framework, and making that possible with Svelte. One way to customize styles in literally any CSS framework works by overriding default styles via contextual (higher binding) selectors. Those by nature interfere with what the child defined, whether the child wants or not and also in ways the child component author did not anticipate. It is not the question whether you should do that but rather whether you can. Closing this PR essentially means: Svelte will not offer a generic way to support style customizing via contextual class overrides (as we'd do it in plain HTML). Instead we'll invent something new that is entirely different. If a child component is provided and does not anticipate some contextual usage scenario (style wise) you'd need to copy it or hack around that via That is a valid choice. |
Again, this breaks the model, I argue we do not want that. This isn't behaviour we want to encourage so the current situation where it is possible but annoying/ difficult is ideal: it is there as an escape hatch but not encouraged.
This is a framework and it comes with certain opinions about how things should be done, this isn't unique to Svelte. And before we can decide whether or not we will allow certain behaviour or encourage it with better ergonomics, we have to have a conversation about whether or not we should be doing things that way. You can't separate the can from the should in an opinionated framework. We want to make building UIs simpler, for sure, but also safer we don't want to add ease of use at the expense of component encapsulation, there has to be a balance For my point of view, and I've been annoyingly consistent in this for as long as people have been asking for this feature or something like it, style encapsulation is one of the core principles of Svelte's component model and this feature fundamentally breaks that. It would be too easy for people to use this feature and it would definitely get abused removing the style safety that Svelte previously provided. The RFC is more appropriate because it does not allow a parent to abritrarily control anything below it, that responsibility still relies on the component itself. Just because people have been passing classes round and overriding child styles for years doesn't mean it is a good choice and isn't something we wnat to encourage. Explicit interfaces are preferable, even if it places greater demand on library authors to design both their components and their style interfaces with these things in mind. |
Because they affect the outside of the component, this should purely be controlled by the parent hosting them. These are my opinions, but I stand strongly by them and it's by far the most efficient, performant and simple method of dealing with CSS I've found. Unfortunately CSS is built the way it is, and wasn't necessarily built for what it has grown to, but that doesn't mean we can start ignoring it. Ideally: Only let a parent control those specific CSS properties, and never let a child use them on the root element. |
I disagree with this for the simple reason is that a component isn't a visual concept. The component doesn't end where its box-model ends. A component doesn't need any DOM at all, in fact, and what DOM it does have definitely shouldn't be styled from outside. If you want this control then wrap them in a DOM node that the parent controls. If you want to pass in values then use props and if you want to pass in values from higher up the tree, the new style RFC may be able to help. |
This is a great concept for javascript, but I don't think this will work for css. There are many things you can style using css. It will result in the following two cases:
In most established component libraries from other frameworks (e.g. Material UI) you can also pass certain css properties using component properties (similar to RFC 13), however this is almost never enough. It happens just so often that you need to do some slight modifications in the css so the result fits your expectations. Svelte has already added a lot of innovation to web development, but I think in this case we should not reinvent the wheel. |
I've encountered a few scenarios where passing a class down from a parent component feels like the cleanest way of doing it as well. I think this is being rejected on grounds that are too arbitrary, and detract from what to me are the best things about Svelte -- it's fun and easy to use, and lets you write components in a way that's natural and expressive. The problems I see with the
I disagree with the idea of keeping users "safe" from this feature. Web developers are well aware of the mess you can get into with global CSS, and the action of writing The fears of breaking one of Svelte's core tenets seem overblown to me. Style encapsulation by default is great, but that doesn't mean we should contort the framework around it. I do see the problem of treating "class" specially, but it doesn't seem like a deal-breaking issue -- we could put it behind a compiler flag like |
Want to use this now?Existing projectsA more robust and maintainable way to incorporate this patch into your workflow using patch-package:
UpdatingWhen svelte is updated by using If it fails due to substantial svelte updates you have two options: 1: Download new patch
2: Make new patch yourself
New projectsDownload patched
InfoThis allows passing classes to child components with Child component should define a place where classes will be recieved: <button class="button {$$restProps.class || ''}"><slot /></button>
|
Too many bones were broken about this issue... @nikku really thank you man for your time and providing great PR! Unfortunately, with the current style hashing approach, your solution will be the cause of many confusing edge-cases. Even with the simplest uses. ;( Let's leave behind brackets the esoterical question - is it right or wrong. Actually, the only way to make your solution safe is to change the principle of style isolation and at the same time restrict which selectors allowed. For example, CSS modules can't be applied to tag names, only to classes, and hashing mechanism quite different. In Svelte, you'll be continually faced with things you completely break 3rd-party components, for no reason whatsoever. The simplest example I could imagine: REPL This breaks the component even without actual class passing (parent class removed from the output as usual) and your patch will only make it worse, because result styles will be something like:
So, this part - svelte-child-hash bar - will break the nested component's behavior in various, unpredictable manners. The landscape of potential edge-cases is too big to implement this in Svelte and thereby approve its use. As I said, the only way to make this possible is to completely re-write the css isolation approach and restrict the rules for writing selectors. I'm not sure we should do this. Thanks a lot to everyone who joined this and similar discussions! Good luck! |
(sorry last link was wrong) I like this workaround but it has downsides |
Workaround by using scope-nested global classnames: |
Solution! Is it visible enough?We can just end this suffering at the cost of scoping for non-class selectors and meet the consequences of class-passing, whatever it might be. There's a cssModules scoping preprocessor, which allows to:
Try itInstall it: npm i -D svelte-preprocess-cssmodules svelte-as-markup-preprocessor Adjust preprocess config in the bundler: const cssModules = require('svelte-preprocess-cssmodules');
const sveltePreprocess = require('svelte-preprocess');
const { asMarkupPreprocessor } = require('svelte-as-markup-preprocessor');
// ...
preprocess: [
asMarkupPreprocessor([
sveltePreprocess()
]),
cssModules()
],
// ... Use it: <button class="btn {$$restProps.class || ''}"><slot /></button>
<!-- something else below --> <div>
<p class="future-is-bright">The future is bright</p>
<Button class="margin-top">Click me!</Button>
</div>
<style lang="scss" module>
.future-is-bright {
color: bright;
}
.margin-top {
margin-top: 6px;
// whatever you want
}
</style> |
This adds class handling logic to
InlineComponent
:See my comment for a concise description of the behavior addressed.
Closes #2870