-
Notifications
You must be signed in to change notification settings - Fork 165
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
Designing mixins #363
Comments
Can you do partial mixins then? You need to be able to extend |
Yes, good point, you want partial mixins. But hopefully you are not mixing the same stuff into two different mixins, so for mixins at least I hope we can say partial mixins is the extensibility avenue. |
Yeah, I feel like partials is a well understood and clearly scoped solution to the need to split things up across multiple specs. Mixing mixins into each other, on the other hand, complexifies a bunch of things. |
Some feedback from researching over the gecko code base:
|
It makes sense for mixins to not support things that change the "sort" of object you're dealing with (legacycaller, getter, setter, deleter, iterable, setlike, maplike). But supporting static bits probably does makes sense. |
I don't disagree, but given no current mixin actually declares any, I thought it was easier to leave it out for now and add it later if necessary. |
Would we ever add static bits to multiple objects though? Since static operations/attributes don't have context, it seems you'd almost always use partial for them. |
That's a good question. I guess there's not much call for adding the same statics to multiple things at once... |
As I mentioned on IRC, I feel like the mental model I'm using for mixins is that they're named interface partials that can extend more than one interface with a more limited set of members. When a partial interface doesn't have a [Exposed] extended attribute, its exposure set is "the exposure set of the original interface or namespace definition." When it does, the partial interface's exposure set must be a subset of the interface's exposure set. I feel like this is the behavior that we want for mixins (if my mental model is the right one). But while this works well with partials, it completely breaks down with mixins as @bzbarsky points out in #423 (comment), as that implies that a mixin and its members would have multiple exposure sets. (As a side note, this might also make the transition from [NoInterfaceObject] to mixins not as straightforward in some cases.) So if we want to go with this solution, it sounds like we need to make members' exposure sets function of the interface they're specified or included on, and thus no longer constant. Does that seem reasonable? |
I'm not sure how related the following is to your question, @tobie, but it seems somewhat related. I came here to state it after doing a skim of #433. In general I think everything would be greatly simplified if we could just treat mixins as some kind of "macro" that "ahead of time" modifies the given interface. That is, instead of constantly talking about the mixin and the places it's mixed in to, or talking about an interface's members and its mixins' members, we could just talk about the interface's members, and by definition that will also include those of any mixins. Do you think that direction is feasible to go in? I see some changes in #433 that just go from "A and A's consequential interfaces" to "A and A's mixins" and that seems like a bit of a shame. I guess this does impact your question, because in this framework, you don't have to worry about the mixin's members having different exposure sets depending on where they're mixed in to. Because there are no members of the mixin; there are just N separate "copies" of those members, living inside each of the interfaces they're mixed in to. That seems nice. |
That seems like a reasonable approach. The one drawback with it is that if you have an entire mixin that should all have on exposure set, which doesn't match the exposure set of the whole interface, you have to annotate each thing in the mixin individually. |
Thinking out loud, here, but what if instead of adding mixing we modified partial interfaces so they could do: partial interface Foo, Bar, Baz { }; to mean roughly the same as: mixin M { };
Foo includes M;
Bar includes M;
Baz includes M; We'd lose the indirection provided by the ability to add members to the mixin itself, but would that really be an issue? Either way, it seems that what we want is something conceptually close to partial interfaces. |
That would require centralizing the partial interface definition. I think we should separate the concept from the syntax/usage. The syntax for mixins seems good. It's just a refined version of the syntax we're already using, with |
This only ever matters if the mixin is exposed on a subset of the interface, right? I'm not sure if that is common, but if it is we might want to allow (And yeah, I don't think we want centralized definitions. Pretty sure we use all flexibility provided by the current system.) |
@domenic wrote:
Well, not necessarily. You could have multiple partial interfaces targeting the same set of interfaces.
But yes, I agree with the second part of your comment. The key is to shift the thinking into considering mixins as spec editing devices just like partials and not language constructs such as interfaces. |
@annevk wrote:
Well, they wouldn't need to be centralized, they just wouldn't have their own identifier. So instead of having partial mixin WindowOrWorkerGlobalScope { }; You wouldn't have the partial interface Window, WorkerGlobalScope { }; It's conceptually simpler than learning a new IDL construct, and makes things a lot more explicit. But it implies a bit more churn to move to and you loose the ability to suddenly expend |
@annevk wrote:
Yes, the idea was not to prevent mixins from being annontated with [Exposed]. It was to make the default behavior of missing an [Exposed] annotation the same as it is for partial interfaces. That is, to have mixin members exposed the same way as the (non-annotated) interface members of the interface the mixin was included in. Similarly, annotating a mixin with the [Exposed] extended attribute would be shorthand for annotating all of its members. Again, just like it works for partial interfaces. |
Right. You centralize control over who mixes-in the mixin with the mixin's author, taking it away from the consumers. This is not great; we want to be able to define mixins in one place and have other specs use them. Looking through your gist that flexibility seems to be used rarely; most mixins are used within the same spec. But it seems important to me... |
My understanding from previous conversations was that we used implements statements (instead of In #195 we need to add dictionary mixins. We already need interface mixin partials. We'll probably end up with dictionary mixin partials too. All in all we'll be end up with the following extension mechanisms in the near future:
Where we could have:
Which seems a much lighter cognitive load. Of course both are tradeoffs. I just want to make sure we're choosing the right tradeoffs. (I'm just exposing those tradeoffs, btw, not pushing towards a specific direction myself.) |
I guess it would be good to study precedent. Perhaps both is what we want. I do think there's value to having a mixin named |
Okay, so I let this rest quite a bit. I looked at some of the specs using mixins today, and it seems clear that there's a strong appetite for having mixin constructs in WebIDL. I'd like to account for #195 (dictionary mixins) right away, so I suggest adding support for the following syntax to the spec: interface mixin identifier { /* ... */ };
partial interface mixin identifier { /* ... */ };
// we'd only add those when handling #195
dictionary mixin identifier { /* ... */ };
partial dictionary mixin identifier { /* ... */ }; The alternative is to do, instead: mixin identifier { /* ... */ };
partial mixin identifier { /* ... */ };
// we'd only add those when handling #195
dictionary mixin identifier { /* ... */ };
partial dictionary mixin identifier { /* ... */ }; which seems less consistent, but simpler to write. Any preference or alternative suggestions? |
I'd prefer just mixin over "interface mixin". Do we keep using "implements" to couple them to interfaces? |
I'm slightly in favor of "interface mixin" for symmetry and because it helps ensure that we're not ambiguous with any future ECMAScript-native concept of mixin. |
WFM.
I moved to "includes" instead as I thought it would make it easier to figure out what was upgraded and what wasn't. |
That argument ways a little more than aesthetics, imho. |
I'm fine with that, I mostly argued for mixins since that's how I refer to them in prose. |
Defining them as |
* Obsolete use of [NoInterfaceObject] extended attribute as mixins. * Add new interface mixin and partial interface mixin constructs. * Replace implements statement by includes statement which only accepts mixins on its rhs. * Remove supplemental interface and related concepts altogether. * Add generic members dfn. * Add table to clarify which members each construct accepts. * Refactor default toJSON operation, and [Exposed] and [SecureContext] algorithms accordingly. Closes whatwg#118. * Prevent operation overloading across mixins and interfaces. Closes whatwg#261. Closes whatwg#363. Closes whatwg#164. Closes https://www.w3.org/Bugs/Public/show_bug.cgi?id=26452. Closes https://www.w3.org/Bugs/Public/show_bug.cgi?id=25495.
* Obsolete use of [NoInterfaceObject] extended attribute as mixins. * Add new interface mixin and partial interface mixin constructs. * Replace implements statement by includes statement which only accepts mixins on its rhs. * Remove supplemental interface and related concepts altogether. * Add generic members dfn. * Add table to clarify which members each construct accepts. * Refactor default toJSON operation, and [Exposed] and [SecureContext] algorithms accordingly. Closes whatwg#118. * Prevent operation overloading across mixins and interfaces. Closes whatwg#261. Closes whatwg#363. Closes whatwg#164. Closes https://www.w3.org/Bugs/Public/show_bug.cgi?id=26452. Closes https://www.w3.org/Bugs/Public/show_bug.cgi?id=25495.
* Obsolete use of [NoInterfaceObject] extended attribute as mixins. * Add new interface mixin and partial interface mixin constructs. * Replace implements statement by includes statement which only accepts mixins on its rhs. * Remove supplemental interface and related concepts altogether. * Add generic members dfn. * Add table to clarify which members each construct accepts. * Refactor default toJSON operation, and [Exposed] and [SecureContext] algorithms accordingly. Closes whatwg#118. * Prevent operation overloading across mixins and interfaces. Closes whatwg#261. Closes whatwg#363. Closes whatwg#164. Closes https://www.w3.org/Bugs/Public/show_bug.cgi?id=26452. Closes https://www.w3.org/Bugs/Public/show_bug.cgi?id=25495.
* Obsolete use of [NoInterfaceObject] extended attribute as mixins. * Add new interface mixin and partial interface mixin constructs. * Replace implements statement by includes statement which only accepts mixins on its rhs. * Remove supplemental interface and related concepts altogether. * Add generic members dfn. * Add table to clarify which members each construct accepts. * Refactor default toJSON operation, and [Exposed] and [SecureContext] algorithms accordingly. Closes whatwg#118. * Prevent operation overloading across mixins and interfaces. Closes whatwg#261. Closes whatwg#363. Closes whatwg#164. Closes https://www.w3.org/Bugs/Public/show_bug.cgi?id=26452. Closes https://www.w3.org/Bugs/Public/show_bug.cgi?id=25495.
* Obsolete use of [NoInterfaceObject] extended attribute as mixins. * Add new interface mixin and partial interface mixin constructs. * Replace implements statement by includes statement which only accepts mixins on its rhs. * Remove supplemental interface and related concepts altogether. * Add generic members dfn. * Add table to clarify which members each construct accepts. * Refactor default toJSON operation, and [Exposed] and [SecureContext] algorithms accordingly. Closes whatwg#118. * Prevent operation overloading across mixins and interfaces. Closes whatwg#261. Closes whatwg#363. Closes whatwg#164. Closes https://www.w3.org/Bugs/Public/show_bug.cgi?id=26452. Closes https://www.w3.org/Bugs/Public/show_bug.cgi?id=25495.
We've talked a lot about mixins, but I don't think we've written down our plans in one central location...
The basic idea is to stop using
[NoInterfaceObject]
interfaces +implements
to do mixins, and give them dedicated syntax. Along the way we can drastically simplify the model; currently, since interfaces are complicated, mixins are complicated. This leads to situations like the one in the example in @tobie's latest pull request, where you have mixins that inherit from things that have mixins and so on.We will either eliminate or repurpose
implements
. The only thing you can mix in (either via theimplements
keyword, or via a new one likemixes
that takes its place) its mixins; you cannot mix in interfaces.Once we have a clear way to talk about mixins, and a simpler model, we should be able to fix the following bugs that largely derive from confusion about how mixins interact with the rest of the system:
Concretely, the biggest simplifications we have lined up are:
The text was updated successfully, but these errors were encountered: