-
Notifications
You must be signed in to change notification settings - Fork 25
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
[rfc] support for signals in lwc templates #82
base: master
Are you sure you want to change the base?
Conversation
Thanks for the contribution! Before we can merge this, we need @caridy to sign the Salesforce Inc. Contributor License Agreement. |
It is on this branch: https://github.com/salesforce/lwc/tree/jodarove/lwc-stores-v1 |
text/0000-lwc-signals.md
Outdated
} | ||
``` | ||
|
||
In this design, `this.$api` is a reactive wrapper around the component's API properties, enabling stores to react to changes in properties like recordId. This reensambles the internal mechanism used today in LWC via `vm.cmpProps` that is not exposed to developers. |
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.
Is this reactive wrapper also a signal-like thing? That is, would RecordStore
in this case be expected to do .value
and .subscribe
?
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.
yes! the argument would be a signal-like object created by LWC.
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.
Would RecordStore
be expected to do any .unsubscribe
cleanup when the component is disconnected/disposed?
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 would update the API to remove the unsubscribe all together to match what other frameworks are doing. As for the development experience, they should try to unsubscribe, but LWC would not track any of that. When LWC uses (internally) an store, it would unsubscribe accordingly
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.
reensambles
?
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.
new RecordStore(this.$api.recordId)
How does this allow the RecordStore
to react to changes in recordId
? It's receiving the value, not the signal, correct? Or is $api
an object where the props are strings and the values are signals?
text/0000-lwc-signals.md
Outdated
### Potential for Misuse | ||
|
||
* __Overuse of Signals__: There's a risk that developers might overuse signals for state management, leading to unnecessarily reactive code, which can be less performant and harder to maintain. This is specially relevant because of the runtime detection of `.value` to maintain backward compatibility with existing components. Any misused on the signals can affect the runtime performance of the component and therefore the overall performance of the application. | ||
* __Subscription Management__: Incorrect handling of subscriptions could introduce memory leaks or lead to unexpected behaviors if not managed correctly. This is specially important because the store implementation doesn't have access anymore to the component's lifecycle hooks, instead it would just stop receiving updates of the stores piped thru `this.$api`, meaning that we are at the mercy of the garbage collector. |
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.
This is the one that concerns me the most. I can see lots of component authors messing up subscriptions. @wire
has issues, but one of the nice things about it is that there's no way for a component author to mess up the subscription.
text/0000-lwc-signals.md
Outdated
|
||
### Integration with Components | ||
|
||
Components should treat signals as internal reactive state mechanisms. While a signal can technically be passed down to child components, it's recommended to pass only the necessary data, typically the `.value` of the signal, to keep the child components decoupled from the parent's state management strategy. |
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 & agree with the sentiment here, but I think injecting a signal into a component is going to be the key to making low-code scenarios work?
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.
@caridy I think I remember a demo when signals were introduced for either Solid or React, where signals were explicitly used to skip levels inside component hierarchies (kind of like a poor-man's state management), but where this was key to some major performance improvements observed. Using the following technique there lead to signals "magically" updating their values inside the HTML elements directly, without relying/executing a full re-render of the involved components. I know that this would scare the hell out of you, but that approach was like what brought "lightning" to the demo back then.
// bundle `acme/signals`
import { signal } from 'signals';
export const count = signal(0);
// bundle `acme/parent`
import { count } from 'acme/signals';
class Parent extends LightningElement {
handleClick() {
count.value++;
}
}
// bundle `acme/grandGrandChild`
import { count } from 'acme/signals';
class GrandGrandChild extends LightningElement {
get count() {
return count;
}
}
<!-- bundle `acme/grandGrandChild` -->
<div>{count.value}</div>
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.
The only thing to really try to figure out is whether or not we should support {count}
vs {count.value}
in the templates, a la-vue. The hesitation here is the signal down to other components, the ambiguity he being:
<template>
<input value={count} />
<x-foo count={count}></x-foo>
</template>
Should count
property on foo receive a signal or its value? That's the problem.
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.
it's recommended to pass only the necessary data
Let's assume for a second that our users won't always do the "recommended" thing. (A very good assumption!)
- What kinds of guardrails can we put in place? Warnings/errors/etc.?
- What happens if they mess it up? What are the consequences?
- Knowing all this, should we make it possible to do the wrong thing?
Using the following technique there lead to signals "magically" updating their values inside the HTML elements directly, without relying/executing a full re-render of the involved components.
This is called fine-grained reactivity. It is definitely a perf boost, but I'm not convinced that signals are required to achieve it: salesforce/lwc#3624
text/0000-lwc-signals.md
Outdated
} | ||
|
||
connectedCallback() { | ||
this.record.subscribe(record => this.updateRecord(record)); |
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.
Can this happen before connectedCallback
, eg in constructor
?
I don't think so, but does anything here prevent signals from provisioning values to the component before its initial render? That is, using a signal does not automatically force my component to be rendered twice? |
I expect it would take someone a couple days to figure out they can do something like this:
Should we just preemptively provide this? |
text/0000-lwc-signals.md
Outdated
|
||
## Alternatives | ||
|
||
### 1. Continued Use of `@track` and `@wire` |
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.
@mburr-salesforce @caridy It would be an interesting option to have wires that pass signals instead of deeply frozen objects. That would open things up for a lot of simplifications when for instance passing wired values into e.g. forms without having to create deep clones. But it would of course also open things up for all the mentioned misuse. Still something to assess further in terms of perf pros and cons maybe. Definitely would have an impact on perf, but also on development models as it would allow e.g. mutations without following established approaches like data down and events up.
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.
That would be a choice to make for the owner of the adapter, not LWC.
@mburr-salesforce that's correct. Since a signal can be initialized, plus its callback can be called at any given time, a component with access to a signal can gain access to a value for the first render. |
We are providing a mechanism to interact, reactively, with LWC engine, that's all. If people want to do these things... sure. Not sure I understand the point though. |
@caridy The point is that it's possible to use |
text/0000-lwc-signals.md
Outdated
```javascript | ||
export default class ExampleComponent extends LightningElement { | ||
@api recordId; | ||
record = new RecordStore(this.$api.recordId); |
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.
Why the dollar sign on $api
? Historically we have not put dollar signs for component-specific props like this.refs
and this.template
.
If the concern is backwards compatibility, then we can gate this behind an API version.
text/0000-lwc-signals.md
Outdated
|
||
1. __Framework Agnosticism__: With Signals, LWC components can interact with any external data store that follows the protocol. This change would make LWC more agnostic and flexible, allowing for easier integration with various state management patterns and libraries. | ||
1. __Simplified State Management__: Signals provide a straightforward way to manage reactive data. By deprecating the @wire and @track decorators, we can reduce complexity, making it easier for developers to write and maintain component logic. | ||
1. __Performance Optimizations__: The proposed model opens up possibilities for performance optimizations by allowing selective rehydration of the UI. Components could update the DOM directly in response to state changes without a full re-render, leading to faster updates and reduced resource consumption. |
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.
Nit: it's not clear to me that signals are required to achieve fine-grained reactivity. E.g. the design described in salesforce/lwc#3624 calls for adding fine-grained reactivity to template loops, without any user-facing API changes.
Also, it is entirely possible to implement signals while still re-running the entire template. It would be useful to see examples where signals unlock perf optimizations that we couldn't get otherwise.
text/0000-lwc-signals.md
Outdated
|
||
It is important to highlight that this is the less common method of using a wire adapter. With the new approach, it becomes a little bit more cumbersome, but it is still possible to do it, and it offers a lot more flexibility since you can now use any store that follows the protocol, not just Salesforce-specific services. | ||
|
||
## Backward Compatibility |
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.
The easiest way to achieve backwards compatibility would be to use API versioning. As a bonus, we could disallow @wire
and @track
based on an API version. (May need to be loosened for first-party components.)
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 don't think we need to remove either of those to have a fully functional signal integration. They are just two separate features, and eventually we can start the deprecation strategy.
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.
Although it'd be really attractive to retire @wire
, I think it shouldn't be particularly emphasized in the signals discussion. Initially, signals (or whatever alternative we may come up with) should be purely additive. Long term, we can look at retiring @wire
or reimplementing its internals with signals.
text/0000-lwc-signals.md
Outdated
## Open Discussion Topics | ||
|
||
* The signal/store protocol. Except for the `.value`, which seems to be a requirement, the rest of the protocol is open for discussion. | ||
* The `this.$api` as a way to access internal signal objects created by LWC, that seems to be a good compromise, but we can discuss alternatives. |
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.
It would be great to have examples from other frameworks in this document. I know nearly every framework is moving to signals these days, but I'm not familiar with the exact API patterns used in each one.
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.
Agreed 100%. I'm going to attempt to put together a survey of signals implementations & alternate approaches across web frameworks ahead of our dev sync discussion.
text/0000-lwc-signals.md
Outdated
position = signal({ x: 0, y: 0 }); | ||
|
||
move(x, y) { | ||
this.position.value = { ...this.position.value, x, y }; |
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.
This does become a challenge for nested objects from these two perspectives:
- Error prone trying to recreate a new object for every combination of change
- Requires boiler plate code to recreate the object.
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.
It's also potentially a perf tax due to cloning the object.
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.
sure. but I don't think this RFC is about that particular aspect of it, but the enablement of stores.
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.
This does become a challenge for nested objects from these two perspectives:
- Error prone trying to recreate a new object for every combination of change
- Requires boiler plate code to recreate the object.
Because this RFC deals with the low-level primitives that we'd provide, this isn't a huge concern to me. I'm hoping/expecting that this won't be heavily used by individual components and, instead, abstractions (state manager, data provider, etc) would be built on top of signals that provide better ergonomics.
For example, in a state manager, we could use/replicate Immer to provide a friendlier facade.
@caridy if I have N child components on the same level (siblings), all subscribed to the same signal, would there exist a way to determine the order components are updated and re-rendered? |
This is a good question. Without deliberately implementing something else, the default behavior would rely on the order that components had subscribed to the signal. Is there a specific scenario you're thinking through where you'd want some other, deterministic behavior? |
I was thinking about a page with lots of siblings components, most of which are outside of the viewable bounds, all subscribed to the same signal. If a signal is updated, I would want to prioritize the ones that are on the view to provide a better ux. So I was thinking that maybe there could exist a reactive/dynamic index that would allow one to control the order components get updated. Does it make any sense? |
I could not find a doc where solid describes its heuristics so I asked Bing gpt hehe obs: I did not check any info showed below "When a signal changes, Solid uses a heuristic called topological sorting to determine the order in which components should be re-rendered. Topological sorting is a graph algorithm that sorts nodes in a directed acyclic graph (DAG) such that for every directed edge The topological sorting algorithm works by first finding all nodes with no incoming edges (i.e., no dependencies), adding them to the ordering, and then removing them from the graph. This process is repeated until all nodes have been added to the ordering. If there are cycles in the graph (i.e., circular dependencies), then the algorithm will detect them and raise an error. In summary, Solid uses topological sorting to determine the order in which components should be re-rendered when a signal changes. This ensures that components are re-rendered in the correct order and that circular dependencies are detected and handled appropriately. Source: Conversation with Bing, 11/16/2023 "When there are multiple components on the same level in the tree, Solid uses the order in which the components are defined to determine the order in which they should be re-rendered. For example, consider the following code snippet: function App() {
const [count1, setCount1] = createSignal(0);
const [count2, setCount2] = createSignal(0);
return (
<>
<Counter count={count1} setCount={setCount1} />
<Counter count={count2} setCount={setCount2} />
</>
);
}
function Counter({ count, setCount }) {
return (
<div>
<p>Count: {count()}</p>
<button onClick={() => setCount(count() + 1)}>Increment</button>
</div>
);
} In this example, I hope that helps! Let me know if you have any other questions. Source: Conversation with Bing, 11/16/2023 |
I thought about a different heuristic to determine the order components react to signal changes. What if you think that the signal source is like an antenna broadcasting a change in a 3D space? Then you could order updates with these 2 rules:
You could even create the definition of single signal broadcasting multiple changes in a single event, as if each information was encoded using different frequencies, in order to avoid emitting n separate events at different times. Is this stupid? I just made an analogy based on how telecommunications systems work |
@AllanOricil The order in which a signal notifies its subscribers and the order in which LWC chooses to re-render components seem like orthogonal concerns? Trying to orchestrate rendering order within the scope of a single signal is going to miss many factors that could cause a component to re-render. |
I thought the order an event is received would change the order components are re-rendered. Doesn't the render engine re-render all components that changed in a time frame? |
I'm not sure how LWC determines in what order components should be re-rendered. A few things that make me reluctant to overlay rendering concerns on a signal:
|
text/0000-lwc-signals.md
Outdated
} | ||
``` | ||
|
||
It is important to highlight that this is the less common method of using a wire adapter. With the new approach, it becomes a little bit more cumbersome, but it is still possible to do it, and it offers a lot more flexibility since you can now use any store that follows the protocol, not just Salesforce-specific services. |
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.
Most of the internal code I see uses wire methods, not properties. Do we have numbers on the breakdown?
text/0000-lwc-signals.md
Outdated
@api recordId; | ||
record = new RecordStore(this.$api.recordId); | ||
|
||
updateRecord(record) { |
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.
It would be great to flesh out the error handling some more.
- Can creating a store cause an error?
- Do methods have an error object or is there a different mechanism for communicating errors to components? Error handling frequently breaks static analysis, so if there's a way to handle them separately from the data, that would be nice.
For low-code, we are moving towards completely presentational components that do not make data calls. The view metadata binds the data to the components, so the framework is responsible for handling errors. You can envision an admin saying "Show this error component if a data failure occurs".
text/0000-lwc-signals.md
Outdated
get value(): any { /* return current value */ } | ||
subscribe((newValue: any) => void): () => void { /* subscribe to changes */ } | ||
} | ||
``` |
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.
How can ISVs package a store? In the past, we've asked for headless components for things like headless actions but ended up having to package it as an LWC even though it's very awkward.
We have actual use cases where we need the store to be on the side (vs in the DOM) to solve problems. Use Case: With Islands architecture, I have a mostly static page (no app layer JS) and two small interactive regions that rely on the same state.
@AllanOricil @nolanlawson @mburr-salesforce I have to work on splitting this into two, bare with me as I recover, and catch up with pending work. Great questions. All I can say at the moment is that it depends on how LWC uses the signal. One of my goals is to get to the point where signals DO NOT trigger re-rendering, at least for simple interpolation, like it is the case today. @nolanlawson has been working on this as well for a while. But clearly we haven't think through all the implications here. Today, LWC does implement an algorithm to determine the proper order to rehydrate, but if there is no re-rendering, then what? |
Folks, I have updated the PR to split the RFC into two. This now only contains the details of how to use signals in LWC templates, without adding any new API to LWC. |
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.
Overall this looks reasonable to me, but I think this RFC is still missing a competitive analysis of what other frameworks are doing and whether we want to align or diverge from them.
Signals are a huge topic right now among JS frameworks – everyone is either adopting them or looking to adopt them. Outlining what other frameworks are doing would provide a lot of useful context for this proposal, rather than trying to design in a vacuum.
|
||
1. __Limited External Store Interaction__: LWC components cannot directly interact with external state management systems without going through LWC-specific APIs. This restricts the framework's flexibility and developer freedom to integrate with the broader JavaScript ecosystem. | ||
1. __Complexity and Overhead__: The use of a membrane based on proxies introduces additional complexity and overhead in component design and state management, from read-and-write to read-only objects, it can be challenging for developers to debug and reason about data flow and updates. | ||
1. __Performance Constraints__: The current model relies on components to re-render fully to reflect state changes, which can be inefficient, especially for large and complex component trees. |
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 see this as orthogonal, especially because for a v1, we will probably just re-render the whole template anyway (similar to the current system). We'd have to implement fine-grained reactivity to change that.
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.
well, I think the hint of a .value
in the template, it is certainly the difference. If we were to continue doing what we are doing, we could get far, but because we don't know what is reactive and what's not reactive, I suspect we might have perf challenges.
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.
You know what's reactive, but that doesn't mean you necessarily want fine-grained reactivity at the level of every observable property. Even Solid batches them together in some cases.
Without Signals, I think we can batch at a very coarse level, e.g. for individual items in a for:each
iteration, and get a lot of bang for our buck.
My main point is that I wouldn't sell Signals based on perf. I'd sell it based on DX.
|
||
### Integration with Components | ||
|
||
Components should treat signals as internal reactive state mechanisms. While a signal can technically be passed down to child components, it's recommended to pass only the necessary data, typically the `.value` of the signal, to keep the child components decoupled from the parent's state management strategy. |
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.
What do you think about formally restricting this, at least in a v1? This could be relaxed later if there are valid use cases for passing down the signal itself.
(I.e. the runtime engine could detect that an object looks like a signal and was passed down wholesale. This might be a breaking change due to false positives on objects that merely look like signals, but it could be done with API versioning.)
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.
How? today you do allow passing objects, and non-plain objects as well. Do you plan to just throw if the object has a .value
and .subscribe
? that seems expensive, for a very little value.
There are two main concerns with backward compatibility: | ||
|
||
1. A component receiving an object from a parent component that contains a `value` getter and a `subscribe` method. If the object is considered a POJO by our current test, then LWC would incorrectly avoid wrapping the object with a read-only reactive process. This is a minor concern, but it is something to consider. | ||
2. A template using `x.value` as part of an interpolation where the value of `x` is not a signal, but contains a `value` getter and a `subscribe` method. In this case, the compiled code would attempt to call `subscribe`, if it fails, we can swallow the error and show a warning. In terms of functionality, everything remains the same, the `.value` is used to render the value into the DOM, and if it is reactive it will be updated automatically by triggering the re-rendering, which would call the `.value` getter again. Technically, this is not a non-backward compabitibility issue since we can swallow the error. But it is something to consider. |
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.
These are both solvable with API versioning IMO – if you don't bump your <apiVersion>
, you don't get signal reactivity magic.
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 don't think we need to. If the library implementing the signals can take care of the first point, then we should be fine. Since this RFC doesn't cover that part, we can discuss the details on the next RFC.
|
||
### 1. Adoption of Other Reactivity Models | ||
|
||
Other frameworks' reactivity models, such as starbeam, Vue's reactivity system or Svelte's store contracts, could be considered. Each comes with its own set of trade-offs and would require significant adaptation to align with LWC's architecture and principles. |
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 would love to see a rundown of other frameworks' reactivity systems, and if there is anything we can learn from them, or if there is an emerging standard we can rely on, etc.
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.
@nolanlawson certainly. I also want to get @wycats involved here.
Objects can be wrapped in Signals to make them reactive. However, changes to nested properties must be handled by replacing the entire object: | ||
|
||
```javascript | ||
import { signal } from 'signals'; | ||
|
||
export default class ExampleComponent extends LightningElement { | ||
position = signal({ x: 0, y: 0 }); | ||
|
||
move(x, y) { | ||
this.position.value = { ...this.position.value, x, y }; | ||
} |
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.
What if this behavior is configurable?
position = signal({x:0, y:0}) //has to change the whole object
position = signal({x:0,y:0}, { deep: true }) //deep comparison will be made which enables signaling based on prop changes
It is like vue's watch.prop.deep
, but for signals
I think you have to describe what happens when binding object props to templates. Can I bind the whole object like using <div>{{position.value}}</div>
? or do I need to use <div>{{position.value.x}}</div>
or would it be <div>{{position.x.value}}</div>
?
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.
@AllanOricil all that is possible, but not part of this RFC. Remember, this is about the template, and how to use signals in the template, how to create and update signals, that's a different RFC.
As for the specific syntax in the template, you will probably need to do: <div>{{position.value.x}}</div>
. The .value
is the signal to listen for changes at runtime.
In the template, we can bind directly to the `.value` property: | ||
|
||
```html | ||
<template> | ||
<button onclick={increment}>Increment</button> | ||
<p>{count.value}</p> | ||
</template> | ||
``` |
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.
What happens if I bind with count
?
I really did not like to be forced to use .value
all the time. I recall you have already answered this question but I could not find your answer.
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.
What if lwc provides ref
and unref
like vue 3?
https://markus.oberlehner.net/blog/vue-3-composition-api-ref-vs-reactive
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.
the problem of not having a hint (e.g.: .value
), then you have two options:
- introduce a new syntax, to hint when to use it as a signal.
- introduce a big overhead to the template runtime execution to test every binding to see if the binding is a signal (I don't think we can afford this).
|
||
This ensures that child components remain agnostic of the parent's state management implementation, promoting better component encapsulation and reuse. | ||
|
||
In terms of optimizations, the compiler can detect when a signal is passed down to a child component because of the `.value` in the template interpolation, and generate code to subscribe to the signal's changes. This would allow the parent component to update the child's public property `count` without a full re-render, leading to better performance. On the receiving end, because the child component's `count` property is not a signal, but a value, it will react to changes as usual, without any special handling. |
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.
What is the use case for being able to pass the signal ref instead of .value
to a child component? Again, as a dev I really did not like to use count.value
. If there is no use case, I think we could just use count
. Then the framework can identify that it is a signal based on its type (maybe you could officially make lwc ts first to get errors at build time) and that the child has to react based on .value
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.
there are 3 ways to provide stores to a component:
- create or import a signal object
- use a global signal (not recommended)
- get it via a prop
So take your poison. If you have two side by side components that are using the same store provisioned by a parent component, what else can you do other than 3?
There are two main concerns with backward compatibility: | ||
|
||
1. A component receiving an object from a parent component that contains a `value` getter and a `subscribe` method. If the object is considered a POJO by our current test, then LWC would incorrectly avoid wrapping the object with a read-only reactive process. This is a minor concern, but it is something to consider. | ||
2. A template using `x.value` as part of an interpolation where the value of `x` is not a signal, but contains a `value` getter and a `subscribe` method. In this case, the compiled code would attempt to call `subscribe`, if it fails, we can swallow the error and show a warning. In terms of functionality, everything remains the same, the `.value` is used to render the value into the DOM, and if it is reactive it will be updated automatically by triggering the re-rendering, which would call the `.value` getter again. Technically, this is not a non-backward compabitibility issue since we can swallow the error. But it is something to consider. |
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.
Couldn't you be sure that it is a signal at build time using ts? This would help the engine to do less checks at runtime
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.
@AllanOricil No, the compiler compiles one file at a time in isolation.
Co-authored-by: Nolan Lawson <[email protected]>
Rendered