From React to Vue3, when I saw the V-Model, I felt that it should be an outdated design, so LET me share my thoughts with you.
What is a V-Model?
V-model is a syntactic sugar that makes it easier to bind data in both directions.
Two-way binding
const Demo = defineComponent({
setup() {
const text = ref(' ');
return () = > (
<>
<input v-model={text.value} />
<div>{text.value}</div>
</>)}})Copy the code
The above code implements a bidirectional binding of input data to text. It is simply a shorthand for the following code:
<input value={text.value} onInput={event= > text.value = event.target.value} />
Copy the code
Apply to components
const Parent = defineComponent({
setup() {
const text = ref(' ');
return () = > (
<>
<Child v-model={text.value} />
<div>{text.value}</div>
</>)}})const Child = defineComponent({
props: {
modelValue: String.'onUpdate:modelValue': Function
},
setup(props, { emit }) {
const handleInput = event= > {
emit('update:modelValue', event.target.value);
// or
// props['onUpdate:modelValue'](event.target.value);
}
return () = > <input value={props.modelValue} onInput={handleInput} />}})Copy the code
With v-model, the parent component binds text. Value to the props. ModelValue of the child component. Text. value releases changes, props. ModelValue changes synchronously, and the child component can make the text.value change via events.
Here the V-model is also a shorthand, which is equivalent to:
<Child modelValue={text.value} onUpdate:modelValue={event= > text.value=event} />
Copy the code
Maybe you’re wondering what modelValue is? ModelValue is the name of the default v-Model attribute in VUe3. You can also use other names: for example, V-model ={[text.value, ‘name’]}, indicating that the child component will receive a prop of name and a prop of onUpdate:name.
V-model ={text.value} ={[text.value, ‘modelValue’]}
Let’s look at the child component, which receives two properties: props. ModelValue and props[‘onUpdate:modelValue’]. The latter is a function that you can call to assign a new value to modelValue. However, we usually use the emit function to send an event, which is equivalent.
The V-Model has efficiency issues
Why do I feel the V-Model has become obsolete? Because what time?
- The V-Model has a component efficiency problem
- The V-model is cumbersome to write
- Vue3 has a better approach: share refs, Reactive between components
There’s nothing to be said for the fact that it’s hard to write, that to change something in a child component, you need to throw an event instead of assigning it directly,
So what is the efficiency problem?
As mentioned earlier,
An example:
const Parent = defineComponent({
setup() {
const text = ref(' ');
const count = ref(0);
return () = > (
<>
<Child v-model={text.value} />
<button onClick={()= >Count.value+ +}> I was hit {count.value} times</button>
</>)}})const Child = defineComponent({
setup() {
onRenderTracked((. args) = > {
console.log(args);
})
onUpdated(() = > {
console.log('Child component updated');
})
return () = > {
console.log('Child component start to render');
return <div>Child</div>}}})Copy the code
When the button is clicked, the parent component is re-rendered. We also see in the browser’s Console that the child component has been updated as well, although it doesn’t need to be updated at all.
The subcomponent doesn’t even use any properties, but it’s updated anyway, and in this case onRenderTracked doesn’t work either, there’s no output in it.
Components share ref/reactive variables
Ref/Reactive are two types of reactive data that, if used in multiple components, each responds to updates. I think it’s much simpler and much more efficient.
const Parent = defineComponent({
setup() {
const text = ref(' ');
return () = > (
<>
<Child text={text} />
<div>{text.value}</div>
</>)}})const Child = defineComponent({
props: {
text: { validator: isRef }
},
setup(props) {
return () = > <input v-model={props.text.value} />}})Copy the code
In this example, we share a ref: text between the parent and child components. When the text changes, both components update in response.
The pattern of the V-Model is to define a variable in each parent component and bind it in both directions.
A shared REF, Reactive is a single variable to which multiple components respond simultaneously.
Sometimes we need to restrict a prop to a ref type. We can use {validator: isRef}. Reactive, readonly you can also use isReactive, isReadonly
Some would argue that you have multiple components sharing a variable that make your code unreadable and maintainable. But is this really true?
When we see code passing a REF/Reactive to a child component, we are obviously aware that this variable may be modified by the component. When we see that prop has isRef/isReactive limits, we also realize that changes to this variable affect the upper components. In terms of readability, this is not fundamentally different from the V-Model. It’s not very different from the React value/onChange mode. But it’s much easier to write.
We also have a readOnly tool. If an object does not want the quilt component to change, it simply passes a ReadOnly.
In the example below, userReadonly proxies a reactive as a readonly, and both respond to data changes together. We pass the readOnly type to the child component, which can use it in response to data changes, but cannot modify it.
const Parent = defineComponent({
setup() {
const user = reactive({
name: 'name',})const userReadonly = readonly(user);
return () = > (
<>
<Child user={userReadonly} />
<div onClick={()= > user.name = 'another name'}>Parent: {user.name}</div>
</>)}})const Child = defineComponent({
props: {
user: { validator: isReadonly }
},
setup(props) {
return () = > <div>Child: {props.user.name}</div>}})Copy the code
In real code, a variable is actually passed at most 3 levels, which is basically easy to maintain.
If the business really needs a variable to be used in many, many components, then you should use provide/inject.
How to use a component library with V-Model?
In practice, we may use some third-party component libraries. Components may provide v-Model interfaces. To be efficient, we need to define the onUpdate: XXX function in setup. Because setup is executed only once, this function does not change and does not cause unnecessary rendering of child components.
const Parent = defineComponent({
const text = ref(' ');
const onUpdateModel = val= > text.value = val;
return () = > <Child modelValue={text.value} onUpdate:modelValue={onUpdateModel}} />
})
Copy the code