Vue?組件上的v-model雙向綁定原理解析
之前我們分析了Vue中v-model指令在普通表單元素上的使用原理(點擊這里跳轉(zhuǎn)),這一節(jié)我們繼續(xù)分析v-model指令在組件上的原理。
組件上的v-model原理
v-model指令在組件上的編譯過程的parse階段與在表單元素上一樣(可以參考),與普通表單元素不同之處在于genCode的階段,在執(zhí)行model函數(shù)生成代碼的時候,會執(zhí)行genComponentModel函數(shù):
v-model編譯階段
export default function model (
el: ASTElement,
dir: ASTDirective,
_warn: Function
): ?boolean {
warn = _warn
// 解析指令對象的值、修飾符、標簽、類型
const value = dir.value
const modifiers = dir.modifiers
const tag = el.tag
const type = el.attrsMap.type
......
} else if (tag === 'input' || tag === 'textarea') {
// 本案例進入這個邏輯,我們分析一下
genDefaultModel(el, value, modifiers)
} else if (!config.isReservedTag(tag)) {
// 非保留標簽,說明是組件節(jié)點,執(zhí)行genComponentModel
genComponentModel(el, value, modifiers)
// component v-model doesn't need extra runtime
return false
}
......
return true
}export function genComponentModel (
el: ASTElement,
value: string,
modifiers: ?ASTModifiers
): ?boolean {
// 解析修飾符
const { number, trim } = modifiers || {}
const baseValueExpression = '$$v'
let valueExpression = baseValueExpression
// 有trim修飾符,進入下面邏輯,生成value表達式
if (trim) {
valueExpression =
`(typeof ${baseValueExpression} === 'string'` +
`? ${baseValueExpression}.trim()` +
`: ${baseValueExpression})`
}
// 有number修飾符,生成下面表達式
if (number) {
valueExpression = `_n(${valueExpression})`
}
// 解析 value,生成解析規(guī)范后的表達式
const assignment = genAssignmentCode(value, valueExpression)
// AST element上掛載model對象
el.model = {
value: `(${value})`,
expression: `"${value}"`,
callback: `function (${baseValueExpression}) {${assignment}}`
}
}可以看到組件執(zhí)行完genDirectives解析model指令后,會在AST element節(jié)點上生成model對象,這是與普通表單元素不同的地方。組件的v-model在genCode過程中,執(zhí)行完genDirectives后還有有一段邏輯,如下:
export function genData (el: ASTElement, state: CodegenState): string {
let data = '{'
// directives may mutate the el's other properties before they are generated.
// 解析model指令
const dirs = genDirectives(el, state)
if (dirs) data += dirs + ','
......
// component v-model
// 組件上的v-model,進入該邏輯,拼接生成下面代碼字符串
if (el.model) {
data += `model:{value:${
el.model.value
},callback:${
el.model.callback
},expression:${
el.model.expression
}},`
}到這個時候才生成了最終的代碼字符串。
組件生成階段
export function createComponent (
Ctor: Class<Component> | Function | Object | void,
data: ?VNodeData,
context: Component,
children: ?Array<VNode>,
tag?: string
): VNode | Array<VNode> | void {
// ...
// transform component v-model data into props & events
// 如果data上有model屬性
if (isDef(data.model)) {
// 調(diào)用transformModel,傳入的參數(shù)為組件構(gòu)造器的options配置項
transformModel(Ctor.options, data)
}
// ...
const vnode = new VNode(
`vue-component-${Ctor.cid}${name ? `-${name}` : ''}`,
data, undefined, undefined, undefined, context,
{ Ctor, propsData, listeners, tag, children },
asyncFactory
)
return vnode
}在創(chuàng)建組件的時候,有上面這樣一段邏輯,當分析到節(jié)點上有model對象的時候,會調(diào)用transformModel函數(shù),對v-model對象做下轉(zhuǎn)化:
// transform component v-model info (value and callback) into
// prop and event handler respectively.
function transformModel (options, data: any) {
// 找到model上的prop屬性
const prop = (options.model && options.model.prop) || 'value'
// 找到model上的event事件
const event = (options.model && options.model.event) || 'input'
// 在data的props屬性對象上添加prop屬性值
;(data.props || (data.props = {}))[prop] = data.model.value
// 在data的on屬性對象上添加event事件
const on = data.on || (data.on = {})
if (isDef(on[event])) {
on[event] = [data.model.callback].concat(on[event])
} else {
on[event] = data.model.callback
}
}可以看到最終是將編譯過程中生成的model對象,解析成為value屬性和input事件,擴展到組件構(gòu)造器的options配置項中。
以上可以得知,組件上v-model指令的本質(zhì)也是生成了value屬性和input事件。
到此這篇關于Vue 組件上的v-model雙向綁定原理的文章就介紹到這了,更多相關Vue v-model雙向綁定內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!
相關文章
vue項目中v-model父子組件通信的實現(xiàn)詳解
vue.js,是一個構(gòu)建數(shù)據(jù)驅(qū)動的 web 界面的庫。Vue.js 的目標是通過盡可能簡單的 API 實現(xiàn)響應的數(shù)據(jù)綁定和組合的視圖組件。下面這篇文章主要給大家介紹了關于vue項目中v-model父子組件通信實現(xiàn)的相關資料,需要的朋友可以參考下。2017-12-12
el-table 選擇框根據(jù)條件設置某項不可選中的操作代碼
這篇文章主要介紹了el-table 選擇框根據(jù)條件設置某項不可選中的操作代碼,本文給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友參考下吧2024-03-03
詳解Vue3.0 + TypeScript + Vite初體驗
這篇文章主要介紹了詳解Vue3.0 + TypeScript + Vite初體驗,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2021-02-02
element-UI?el-table樹形數(shù)據(jù)?修改小三角圖標方式
這篇文章主要介紹了element-UI?el-table樹形數(shù)據(jù)?修改小三角圖標方式,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2022-07-07

