Vue 3.0 进阶之双向绑定探秘

开发 前端
本文阿宝哥主要介绍了双向绑定的概念和 Vue 3 中双向绑定背后的原理。为了让大家能够更深入地掌握 v-model 的相关知识,阿宝哥从源码的角度分析了 vModelText 指令的内部实现。

[[382290]]

本文转载自微信公众号「全栈修仙之路」,作者阿宝哥。转载本文请联系全栈修仙之路公众号。  

本文是 Vue 3.0 进阶系列 的第三篇文章,在阅读本文前,建议你先阅读 Vue 3.0 进阶之指令探秘 和 Vue 3.0 进阶之自定义事件探秘 这两篇文章。在看具体示例前,阿宝哥先来简单介绍一下双向绑定,它由两个单向绑定组成:

  • 模型 —> 视图数据绑定;
  • 视图 —> 模型事件绑定。

在 Vue 中 :value 实现了 模型到视图 的数据绑定,@event 实现了 视图到模型 的事件绑定:

  1. <input :value="searchText" @input="searchText = $event.target.value" /> 

而在表单中,通过使用内置的 v-model 指令,我们可以轻松地实现双向绑定,比如 。介绍完上面的内容,接下来阿宝哥将以一个简单的示例为切入点,带大家一起一步步揭开双向绑定背后的秘密。

  1. <div id="app"
  2.    <input v-model="searchText" /> 
  3.    <p>搜索的内容:{{searchText}}</p> 
  4. </div> 
  5. <script> 
  6.    const { createApp } = Vue 
  7.    const app = Vue.createApp({ 
  8.      data() { 
  9.        return { 
  10.          searchText: "阿宝哥" 
  11.        } 
  12.      } 
  13.    }) 
  14.    app.mount('#app'
  15. </script> 

在以上示例中,我们在 input 搜索输入框中应用了 v-model 指令,当输入框的内容发生变化时,p 标签中内容会同步更新。

要揭开 v-model 指令背后的秘密,我们可以利用 Vue 3 Template Explorer 在线工具,来看一下模板编译后的结果:

  1. <input v-model="searchText" /> 
  2.  
  3. const _Vue = Vue 
  4. return function render(_ctx, _cache, $props, $setup, $data, $options) { 
  5.   with (_ctx) { 
  6.     const { vModelText: _vModelText, createVNode: _createVNode,  
  7.       withDirectives: _withDirectives, openBlock: _openBlock, createBlock: _createBlock } = _Vue 
  8.  
  9.     return _withDirectives((_openBlock(), _createBlock("input", { 
  10.       "onUpdate:modelValue": $event => (searchText = $event) 
  11.     }, null, 8 /* PROPS */, ["onUpdate:modelValue"])),  
  12.     [  
  13.       [_vModelText, searchText]  
  14.     ]) 
  15.   } 

在 模板生成的渲染函数中,我们看到了 Vue 3.0 进阶之指令探秘 文章中介绍的 withDirectives 函数,该函数用于把指令信息添加到 VNode 对象上,它被定义在 runtime-core/src/directives.ts 文件中:

  1. // packages/runtime-core/src/directives.ts 
  2. export function withDirectives<T extends VNode>( 
  3.   vnode: T, 
  4.   directives: DirectiveArguments 
  5. ): T { 
  6.   const internalInstance = currentRenderingInstance 
  7.   // 省略部分代码 
  8.   const instance = internalInstance.proxy 
  9.   const bindings: DirectiveBinding[] = vnode.dirs || (vnode.dirs = []) 
  10.   for (let i = 0; i < directives.length; i++) { 
  11.     let [dir, value, arg, modifiers = EMPTY_OBJ] = directives[i] 
  12.     // 在 mounted 和 updated 时,触发相同行为,而不关系其他的钩子函数 
  13.     if (isFunction(dir)) { // 处理函数类型指令 
  14.       dir = { 
  15.         mounted: dir, 
  16.         updated: dir 
  17.       } as ObjectDirective 
  18.     } 
  19.     bindings.push({ // 把指令信息保存到vnode.dirs数组中 
  20.       dir, instance, value,  
  21.       oldValue: void 0, arg, modifiers 
  22.     }) 
  23.   } 
  24.   return vnode 

除此之外,在模板生成的渲染函数中,我们看到了 vModelText 指令,通过它的名称,我们猜测该指令与模型相关,所以我们先来分析 vModelText 指令。

一、vModelText 指令

vModelText 指令是 ObjectDirective 类型的指令,该指令中定义了 3 个钩子函数:

  • created:在绑定元素的属性或事件监听器被应用之前调用。
  • mounted:在绑定元素的父组件被挂载后调用。
  • beforeUpdate:在更新包含组件的 VNode 之前调用。
  1. // packages/runtime-dom/src/directives/vModel.ts 
  2. type ModelDirective<T> = ObjectDirective<T & { _assign: AssignerFn }> 
  3.  
  4. export const vModelText: ModelDirective< 
  5.   HTMLInputElement | HTMLTextAreaElement 
  6. > = { 
  7.   created(el, { modifiers: { lazy, trim, number } }, vnode) { 
  8.     // ... 
  9.   }, 
  10.   mounted(el, { value }) { 
  11.     // .. 
  12.   }, 
  13.   beforeUpdate(el, { value, modifiers: { trim, number } }, vnode) { 
  14.     // .. 
  15.   } 

接下来,阿宝哥将逐一分析每个钩子函数,这里先从 created 钩子函数开始。

1.1 created 钩子

  1. // packages/runtime-dom/src/directives/vModel.ts 
  2. export const vModelText: ModelDirective< 
  3.   HTMLInputElement | HTMLTextAreaElement 
  4. > = { 
  5.   created(el, { modifiers: { lazy, trim, number } }, vnode) { 
  6.     el._assign = getModelAssigner(vnode) 
  7.     const castToNumber = number || el.type === 'number' // 是否转为数值类型 
  8.     // 若使用 lazy 修饰符,则在 change 事件触发后将输入框的值与数据进行同步 
  9.     addEventListener(el, lazy ? 'change' : 'input', e => {  
  10.       if ((e.target as any).composing) return // 组合输入进行中 
  11.       let domValue: string | number = el.value 
  12.       if (trim) { // 自动过滤用户输入的首尾空白字符 
  13.         domValue = domValue.trim() 
  14.       } else if (castToNumber) { // 自动将用户的输入值转为数值类型 
  15.         domValue = toNumber(domValue) 
  16.       } 
  17.       el._assign(domValue) // 更新模型 
  18.     }) 
  19.     if (trim) { 
  20.       addEventListener(el, 'change', () => { 
  21.         el.value = el.value.trim() 
  22.       }) 
  23.     } 
  24.     if (!lazy) { 
  25.       addEventListener(el, 'compositionstart', onCompositionStart) 
  26.       addEventListener(el, 'compositionend', onCompositionEnd) 
  27.       // Safari < 10.2 & UIWebView doesn't fire compositionend when 
  28.       // switching focus before confirming composition choice 
  29.       // this also fixes the issue where some browsers e.g. iOS Chrome 
  30.       // fires "change" instead of "input" on autocomplete. 
  31.       addEventListener(el, 'change', onCompositionEnd) 
  32.     } 
  33.   }, 

对于 created 方法来说,它会通过解构的方式获取 v-model 指令上添加的修饰符,在 v-model 上可以添加 .lazy、.number 和 .trim 修饰符。这里我们简单介绍一下 3 种修饰符的作用:

  • .lazy 修饰符:在默认情况下,v-model 在每次 input 事件触发后将输入框的值与数据进行同步。你可以添加 lazy 修饰符,从而转为在 change 事件之后进行同步。
    1. <!-- 在 change 时而非 input 时更新 --> 
    2. <input v-model.lazy="msg" /> 
  • .number 修饰符:如果想自动将用户的输入值转为数值类型,可以给 v-model 添加 number 修饰符。这通常很有用,因为即使在 type="number" 时,HTML 输入元素的值也总会返回字符串。如果这个值无法被 parseFloat() 解析,则会返回原始的值。
  1. <input v-model.number="age" type="number" /> 
  • .trim 修饰符:如果要自动过滤用户输入的首尾空白字符,可以给 v-model 添加 trim 修饰符。
  1. <input v-model.trim="msg" /> 

而在 created 方法内部,会通过 getModelAssigner 函数获取 ModelAssigner,从而用于更新模型对象。

  1. // packages/runtime-dom/src/directives/vModel.ts 
  2. const getModelAssigner = (vnode: VNode): AssignerFn => { 
  3.   const fn = vnode.props!['onUpdate:modelValue'
  4.   return isArray(fn) ? value => invokeArrayFns(fn, value) : fn 

对于我们的示例来说,通过 getModelAssigner 函数获取的 ModelAssigner 对象是 $event => (searchText = $event) 函数。在获取 ModelAssigner 对象之后,我们就可以更新模型的值了。created 方法中的其他代码相对比较简单,阿宝哥就不详细介绍了。这里我们来介绍一下 compositionstart 和 compositionend 事件。

中文、日文、韩文等需要借助输入法组合输入,即使是英文,也可以利用组合输入进行选词等操作。在一些实际场景中,我们希望等用户组合输入完的一段文字才进行对应操作,而不是每输入一个字母,就执行相关操作。

比如,在关键字搜索场景中,等用户完整输入 阿宝哥 之后再执行搜索操作,而不是输入字母 a 之后就开始搜索。要实现这个功能,我们就需要借助 compositionstart 和 compositionend 事件。另外,需要注意的是,compositionstart 事件发生在 input 事件之前,因此利用它可以优化中文输入的体验。

了解完 compositionstart(组合输入开始) 和 compositionend (组合输入结束)事件,我们再来看一下 onCompositionStart 和 onCompositionEnd 这两个事件处理器:

  1. function onCompositionStart(e: Event) { 
  2.   ;(e.target as any).composing = true 
  3.  
  4. function onCompositionEnd(e: Event) { 
  5.   const target = e.target as any 
  6.   if (target.composing) {  
  7.     target.composing = false 
  8.     trigger(target, 'input'
  9.   } 
  10.  
  11. // 触发元素上的指定事件 
  12. function trigger(el: HTMLElement, type: string) { 
  13.   const e = document.createEvent('HTMLEvents'
  14.   e.initEvent(type, truetrue
  15.   el.dispatchEvent(e) 

当组合输入时,在 onCompositionStart 事件处理器中,会 e.target 对象上添加 composing 属性并设置该属性的值为 true。而在 change 事件或 input 事件回调函数中,如果发现 e.target 对象的 composing 属性为 true 则会直接返回。当组合输入完成后,在 onCompositionEnd 事件处理器中,会把 target.composing 的值设置为 false 并手动触发 input 事件:

  1. // packages/runtime-dom/src/directives/vModel.ts 
  2. export const vModelText: ModelDirective< 
  3.   HTMLInputElement | HTMLTextAreaElement 
  4. > = { 
  5.   created(el, { modifiers: { lazy, trim, number } }, vnode) { 
  6.     // 省略部分代码 
  7.     addEventListener(el, lazy ? 'change' : 'input', e => { 
  8.       if ((e.target as any).composing) return 
  9.      // ... 
  10.     }) 
  11.   }, 

好的,created 钩子函数就分析到这里,接下来我们来分析 mounted 钩子。

1.2 mounted 钩子

  1. // packages/runtime-dom/src/directives/vModel.ts 
  2. export const vModelText: ModelDirective< 
  3.   HTMLInputElement | HTMLTextAreaElement 
  4. > = { 
  5.   // set value on mounted so it's after min/max for type="range" 
  6.   mounted(el, { value }) { 
  7.     el.value = value == null ? '' : value 
  8.   }, 

mounted 钩子的逻辑很简单,如果 value 值为 null 时,把元素的值设置为空字符串,否则直接使用 value 的值。

1.3 beforeUpdate 钩子

  1. // packages/runtime-dom/src/directives/vModel.ts 
  2. export const vModelText: ModelDirective< 
  3.   HTMLInputElement | HTMLTextAreaElement 
  4. > = { 
  5.   beforeUpdate(el, { value, modifiers: { trim, number } }, vnode) { 
  6.     el._assign = getModelAssigner(vnode) 
  7.     // avoid clearing unresolved text. #2302 
  8.     if ((el as any).composing) return 
  9.     if (document.activeElement === el) { 
  10.       if (trim && el.value.trim() === value) { 
  11.         return 
  12.       } 
  13.       if ((number || el.type === 'number') && toNumber(el.value) === value) { 
  14.         return 
  15.       } 
  16.     } 
  17.     const newValue = value == null ? '' : value 
  18.     if (el.value !== newValue) { // 新旧值不相等时,执行更新操作 
  19.       el.value = newValue 
  20.     } 
  21.   } 

相信使用过 Vue 的小伙伴都知道,v-model 指令不仅可以应用在 input 和 textarea 元素上,在复选框(Checkbox)、单选框(Radio)和选择框(Select)上也可以使用 v-model 指令。不过需要注意的是,虽然这些元素上都是使用 v-model 指令,但实际上对于复选框、单选框和选择框来说,它们是由不同的指令来完成对应的功能。这里我们以单选框为例,来看一下应用 v-model 指令后,模板编译的结果:

  1. <input type="radio" value="One" v-model="picked" /> 
  2.  
  3. const _Vue = Vue 
  4. return function render(_ctx, _cache, $props, $setup, $data, $options) { 
  5.   with (_ctx) { 
  6.     const { vModelRadio: _vModelRadio, createVNode: _createVNode,  
  7.       withDirectives: _withDirectives, openBlock: _openBlock, createBlock: _createBlock } = _Vue 
  8.  
  9.     return _withDirectives((_openBlock(), _createBlock("input", { 
  10.       type: "radio"
  11.       value: "One"
  12.       "onUpdate:modelValue": $event => (picked = $event) 
  13.     }, null, 8 /* PROPS */, ["onUpdate:modelValue"])), [ 
  14.       [_vModelRadio, picked] 
  15.     ]) 
  16.   } 

由以上代码可知,在单选框应用 v-model 指令后,双向绑定的功能会交给 vModelRadio 指令来实现。除了 vModelRadio 之外,还有 vModelSelect 和 vModelCheckbox 指令,它们被定义在 runtime-dom/src/directives/vModel.ts 文件中,感兴趣的小伙伴可以自行研究一下。

其实 v-model 本质上是语法糖。它负责监听用户的输入事件来更新数据,并在某些场景下进行一些特殊处理。需要注意的是 v-model 会忽略所有表单元素的 value、checked、selected attribute 的初始值而总是将当前活动实例的数据作为数据来源。你应该通过在组件的 data 选项中声明初始值。

此外,v-model 在内部为不同的输入元素使用不同的 property 并抛出不同的事件:

  • text 和 textarea 元素使用 value property 和 input 事件;
  • checkbox 和 radio 元素使用 check property 和 change 事件;
  • select 元素将 value 作为 prop 并将 change 作为事件。

这里你已经知道,可以用 v-model 指令在表单 <input>、<textarea> 及 <select> 元素上创建双向数据绑定。但如果你也想在组件上使用 v-model 指令来创建双向数据绑定,那应该如何实现呢?

二、在组件上使用 v-model

假设你想定义一个 custom-input 组件并在该组件上使用 v-model 指令来实现双向绑定,在实现该功能前,我们先利用 Vue 3 Template Explorer 在线工具,看一下模板编译后的结果:

  1. <custom-input v-model="searchText"></custom-input> 
  2.  
  3. const _Vue = Vue 
  4. return function render(_ctx, _cache, $props, $setup, $data, $options) { 
  5.   with (_ctx) { 
  6.     const { resolveComponent: _resolveComponent, createVNode: _createVNode,  
  7.       openBlock: _openBlock, createBlock: _createBlock } = _Vue 
  8.  
  9.     const _component_custom_input = _resolveComponent("custom-input"
  10.     return (_openBlock(), _createBlock(_component_custom_input, { 
  11.       modelValue: searchText, 
  12.       "onUpdate:modelValue": $event => (searchText = $event) 
  13.     }, null, 8 /* PROPS */, ["modelValue""onUpdate:modelValue"])) 
  14.   } 

通过观察以上的渲染函数,我们可知在 custom-input 组件上应用了 v-model 指令,经过编译器编译之后,会生成一个名为 modelValue 的输入属性和一个名为 update:modelValue 的自定义事件名。如果你对自定义事件内部原理还不清楚的话,可以阅读 Vue 3.0 进阶之自定义事件探秘 这篇文章。了解完这些内容之后,我们就可以开始实现 custom-input 组件了:

  1. <div id="app"
  2.    <custom-input v-model="searchText"></custom-input> 
  3.    <p>搜索的内容:{{searchText}}</p> 
  4. </div> 
  5. <script> 
  6.    const { createApp } = Vue 
  7.    const app = Vue.createApp({ 
  8.      data() { 
  9.        return { 
  10.          searchText: "阿宝哥" 
  11.        } 
  12.      } 
  13.     }) 
  14.    app.component('custom-input', { 
  15.      props: ['modelValue'], 
  16.      emits: ['update:modelValue'], 
  17.      template: ` 
  18.        <input type="text"  
  19.           :value="modelValue" 
  20.           @input="$emit('update:modelValue', $event.target.value)" 
  21.        >` 
  22.    }) 
  23.    app.mount('#app'
  24. </script> 

在自定义组件中实现双向绑定的功能,除了使用自定义事件之外,还可以使用计算属性的功能来定义 getter 和 setter。这里阿宝哥就不展开介绍了,感兴趣的小伙伴可以阅读 Vue 3 官网 - 组件基础 的相关内容。

三、阿宝哥有话说

3.1 如何修改 v-model 默认的 prop 名和事件名?

默认情况下,组件上的 v-model 使用 modelValue 作为 prop 和 update:modelValue 作为事件。我们可以通过向 v-model 指令传递参数来修改这些名称:

  1. <custom-input v-model:name="searchText"></custom-input> 

以上的模板,经过编译器编译后的结果如下:

  1. const _Vue = Vue 
  2. return function render(_ctx, _cache, $props, $setup, $data, $options) { 
  3.   with (_ctx) { 
  4.     const { resolveComponent: _resolveComponent, createVNode: _createVNode,  
  5.       openBlock: _openBlock, createBlock: _createBlock } = _Vue 
  6.  
  7.     const _component_custom_input = _resolveComponent("custom-input"
  8.     return (_openBlock(), _createBlock(_component_custom_input, { 
  9.       name: searchText, 
  10.       "onUpdate:name": $event => (searchText = $event) 
  11.     }, null, 8 /* PROPS */, ["name""onUpdate:name"])) 
  12.   } 

通过观察生成的渲染函数,我们可知自定义 custom-input 组件接收一个 name 输入属性并含有一个名为 update:name 的自定义事件:

  1. app.component('custom-input', { 
  2.   props: { 
  3.     name: String 
  4.   }, 
  5.   emits: ['update:name'], 
  6.   template: ` 
  7.     <input type="text" 
  8.       :value="name" 
  9.       @input="$emit('update:name', $event.target.value)"
  10.   ` 
  11. }) 

至于自定义的事件名为什么是 "onUpdate:name" 这种形式,你可以从 Vue 3.0 进阶之自定义事件探秘 这篇文章中介绍的 emit 函数中找到对应的答案。

3.2 能否在组件上使用多个 v-model 指令?

在某些场景下,我们是希望在组件上使用多个 v-model 指令,每个指令与不同的数据做绑定。比如一个 user-name 组件,该组件允许用户输入 firstName 和 lastName。该组件期望的使用方式如下:

  1. <user-name 
  2.   v-model:first-name="firstName" 
  3.   v-model:last-name="lastName" 
  4. ></user-name

同样,我们使用 Vue 3 Template Explorer 在线工具,先来看一下以上模板编译后的结果:

  1. const _Vue = Vue 
  2. return function render(_ctx, _cache, $props, $setup, $data, $options) { 
  3.   with (_ctx) { 
  4.     const { resolveComponent: _resolveComponent, createVNode: _createVNode,  
  5.       openBlock: _openBlock, createBlock: _createBlock } = _Vue 
  6.  
  7.     const _component_user_name = _resolveComponent("user-name"
  8.     return (_openBlock(), _createBlock(_component_user_name, { 
  9.       "first-name": firstName, 
  10.       "onUpdate:first-name": $event => (firstName = $event), 
  11.       "last-name": lastName, 
  12.       "onUpdate:last-name": $event => (lastName = $event) 
  13.     }, null, 8 /* PROPS */, ["first-name""onUpdate:first-name""last-name""onUpdate:last-name"])) 
  14.   } 

通过观察以上的输出结果,我们可知 v-model:first-name 和 v-model:last-name 都会生成对应的 prop 属性和自定义事件。HTML 中的 attribute 名是大小写不敏感的,所以浏览器会把所有大写字符解释为小写字符。这意味着当你使用 DOM 中的模板时,camelCase (驼峰命名法)的 prop 名需要使用其等价的 kebab-case(短横线分隔命名)命名。比如:

  1. <!-- kebab-case in HTML --> 
  2. <blog-post post-title="hello!"></blog-post> 
  3.  
  4. app.component('blog-post', { 
  5.   props: ['postTitle'], 
  6.   template: '<h3>{{ postTitle }}</h3>' 
  7. }) 

反之,对于 first-name 和 last-name 属性名来说,在定义 user-name 组件时,我们将使用 firstName 和 lastName 驼峰命名方式。

  1.  <div id="app"
  2.     <user-name 
  3.        v-model:first-name="firstName" 
  4.        v-model:last-name="lastName"
  5.     </user-name
  6.     Your name: {{firstName}} {{lastName}} 
  7. </div> 
  8. <script> 
  9.    const { createApp } = Vue 
  10.    const app = Vue.createApp({ 
  11.      data() { 
  12.        return { 
  13.          firstName: ""
  14.          lastName: "" 
  15.        } 
  16.      } 
  17.    }) 
  18.    app.component('user-name', { 
  19.      props: { 
  20.        firstName: String, 
  21.        lastName: String 
  22.      }, 
  23.      emits: ['update:firstName''update:lastName'], 
  24.      template: ` 
  25.        <input 
  26.           type="text" 
  27.           :value="firstName" 
  28.           @input="$emit('update:firstName', $event.target.value)"
  29.        <input 
  30.           type="text" 
  31.           :value="lastName" 
  32.           @input="$emit('update:lastName', $event.target.value)"
  33.       ` 
  34.    }) 
  35.    app.mount('#app'
  36. </script> 

在以上的代码中,user-name 组件使用的自定义属性和事件名都是驼峰的形式。很明显与模板编译后生成的命名格式不一致,那么以上的 user-name 组件可以正常工作么?答案是可以的,这是因为对于自定义事件来说,在 emit 函数内部会通过 hyphenate 函数,把事件名从 camelCase(驼峰命名法)的形式转换为 kebab-case(短横线分隔命名)的形式,即 hyphenate(event):

  1. // packages/runtime-core/src/componentEmits.ts 
  2. export function emit( 
  3.   instance: ComponentInternalInstance, 
  4.   event: string, 
  5.   ...rawArgs: any[] 
  6. ) { 
  7.   // 省略部分代码 
  8.   // for v-model update:xxx events, also trigger kebab-case equivalent 
  9.   // for props passed via kebab-case 
  10.   if (!handler && isModelListener) { 
  11.     handlerName = toHandlerKey(hyphenate(event)) 
  12.     handler = props[handlerName] 
  13.   } 
  14.  
  15.   if (handler) { 
  16.     callWithAsyncErrorHandling( 
  17.       handler, 
  18.       instance, 
  19.       ErrorCodes.COMPONENT_EVENT_HANDLER, 
  20.       args 
  21.     ) 
  22.   } 

而 hyphenate 函数的实现也很简单,具体如下所示:

  1. // packages/shared/src/index.ts 
  2. const hyphenateRE = /\B([A-Z])/g 
  3.  
  4. // cacheStringFunction 函数提供了缓存功能 
  5. export const hyphenate = cacheStringFunction((str: string) => 
  6.   str.replace(hyphenateRE, '-$1').toLowerCase() 

3.3 如何为 v-model 添加自定义修饰符?

在前面阿宝哥已经介绍了 v-model 的内置修饰符:.trim、.number 和 .lazy。但在某些场景下,你可能希望自定义修饰符。在介绍如何自定义修饰符前,我们再次利用 Vue 3 Template Explorer 在线工具,看一下 v-model 使用内置修饰符后,模板编译的结果:

  1. <custom-input v-model.lazy.number="searchText"></custom-input> 
  2.  
  3. const _Vue = Vue 
  4. return function render(_ctx, _cache, $props, $setup, $data, $options) { 
  5.   with (_ctx) { 
  6.     const { resolveComponent: _resolveComponent, createVNode: _createVNode,  
  7.       openBlock: _openBlock, createBlock: _createBlock } = _Vue 
  8.  
  9.     const _component_custom_input = _resolveComponent("custom-input"
  10.     return (_openBlock(), _createBlock(_component_custom_input, { 
  11.       modelValue: searchText, 
  12.       "onUpdate:modelValue": $event => (searchText = $event), 
  13.       modelModifiers: { lazy: true, number: true } 
  14.     }, null, 8 /* PROPS */, ["modelValue""onUpdate:modelValue"])) 
  15.   } 

通过观察生成的渲染函数,我们可以看到 v-model 上添加的 .lazy 和 .number 修饰符,被编译到 modelModifiers prop 属性中。假设我们要为自定义一个 capitalize 修饰符 ,该修饰符的作用是将 v-model 绑定字符串的第一个字母转成大写:

  1. <custom-input v-model.capitalize="searchText"></custom-input> 
  2.  
  3. const _Vue = Vue 
  4. return function render(_ctx, _cache, $props, $setup, $data, $options) { 
  5.   with (_ctx) { 
  6.     const { resolveComponent: _resolveComponent, createVNode: _createVNode,  
  7.       openBlock: _openBlock, createBlock: _createBlock } = _Vue 
  8.  
  9.     const _component_custom_input = _resolveComponent("custom-input"
  10.     return (_openBlock(), _createBlock(_component_custom_input, { 
  11.       modelValue: searchText, 
  12.       "onUpdate:modelValue": $event => (searchText = $event), 
  13.       modelModifiers: { capitalize: true } 
  14.     }, null, 8 /* PROPS */, ["modelValue""onUpdate:modelValue"])) 
  15.   } 

很明显 v-model 上的 .capitalize 修饰符,也被编译到 modelModifiers prop 属性中。了解完这些,我们就可以实现上述的修饰符,具体如下所示:

  1. <div id="app"
  2.    <custom-input v-model.capitalize="searchText"></custom-input> 
  3.    <p>搜索的内容:{{searchText}}</p> 
  4. </div> 
  5. <script> 
  6.    const { createApp } = Vue 
  7.    const app = Vue.createApp({ 
  8.      data() { 
  9.        return { 
  10.          searchText: "" 
  11.        } 
  12.      } 
  13.    }) 
  14.    app.component('custom-input', { 
  15.      props: { 
  16.        modelValue: String, 
  17.        modelModifiers: { 
  18.          default: () => ({}) 
  19.        } 
  20.      }, 
  21.      emits: ['update:modelValue'], 
  22.      methods: { 
  23.        emitValue(e) { 
  24.          let value = e.target.value 
  25.          if (this.modelModifiers.capitalize) { 
  26.            value = value.charAt(0).toUpperCase() + value.slice(1) 
  27.          } 
  28.          this.$emit('update:modelValue', value) 
  29.        } 
  30.      }, 
  31.      template: `<input 
  32.        type="text" 
  33.        :value="modelValue" 
  34.        @input="emitValue">` 
  35.    }) 
  36.   app.mount('#app'
  37. </script> 

本文阿宝哥主要介绍了双向绑定的概念和 Vue 3 中双向绑定背后的原理。为了让大家能够更深入地掌握 v-model 的相关知识,阿宝哥从源码的角度分析了 vModelText 指令的内部实现。

此外,阿宝哥还介绍了在组件中如何使用多个 v-model 指令及如何为 v-model 添加自定义修饰符。Vue 3.0 进阶系列的文章还在持续更新,感兴趣的小伙伴请持续关注哟。

四、参考资源

Vue 3 官网 - 自定义指令

Vue 3 官网 - 自定义事件

 

责任编辑:武晓燕 来源: 全栈修仙之路
相关推荐

2021-02-16 16:41:45

Vue项目指令

2021-02-26 05:19:20

Vue 3.0 VNode虚拟

2021-02-28 20:41:18

Vue注入Angular

2021-02-22 21:49:33

Vue动态组件

2021-02-18 08:19:21

Vue自定义Vue 3.0

2021-03-04 22:31:02

Vue进阶函数

2017-08-08 09:15:41

前端JavaScript页面渲染

2021-03-09 22:29:46

Vue 响应式API

2021-03-08 00:08:29

Vue应用挂载

2022-08-22 09:01:24

Vue响应式原则双向数据绑定

2021-09-13 09:20:20

前端框架VUE

2021-04-02 11:24:22

Vue2.x双向绑定前端

2016-12-27 15:23:56

vue.js双向绑定操作

2022-09-02 10:34:23

数据Vue

2010-05-11 16:22:40

2017-10-27 22:03:35

javascrip

2021-09-01 14:36:14

鸿蒙HarmonyOS应用

2020-10-13 08:24:31

Vue3.0系列

2021-05-07 08:20:52

前端开发技术热点

2020-04-22 14:15:32

Vue 3.0语法前端
点赞
收藏

51CTO技术栈公众号