Vue源码Compile.js
Posted 龖龖龖
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Vue源码Compile.js相关的知识,希望对你有一定的参考价值。
compile.js
function Compile(el, vm) {
this.$vm = vm; //this是compile的实例 $vm是MVVM的实例即vm
//判断当前用户传递的el属性是元素节点还是选择器,如果是元素节点则直接保存到$el中通,
//如果不是 根据选择器 去查找对应的元素然后进行保存
this.$el = this.isElementNode(el) ? el : document.querySelector(el);
//确定元素是否真正存在
if (this.$el) {//#app
this.$fragment = this.node2Fragment(this.$el);
this.init();
this.$el.appendChild(this.$fragment);
}
}
Compile.prototype = {
/**
* node to fragment 把节点转换成文档碎片
* @param el
* @returns {DocumentFragment}
*/
node2Fragment: function(el) {
var fragment = document.createDocumentFragment(),//文档碎片
child;
// 将原生节点拷贝到fragment
while (child = el.firstChild) {
fragment.appendChild(child);
}
return fragment;
},
/**
* 初始化
*/
init: function() {
this.compileElement(this.$fragment);
},
/**
* 解析html元素
* @param el 元素
*/
compileElement: function(el) {
var childNodes = el.childNodes,
me = this;
[].slice.call(childNodes).forEach(function(node) {
var text = node.textContent;
var reg = /\\{\\{(.*)\\}\\}/; //{{name+{{age}}+phone}} //()非贪婪匹配 ->name+{{age}}+phone
// var reg = /(.*)/; //{{name}}
//判断当前节点是不是元素节点
if (me.isElementNode(node)) {
//解析指令
me.compile(node);
} else if (me.isTextNode(node) && reg.test(text)) {
//判断当前元素是否为文本节点 并且 文本节点中是否拥有{{xxx}}
me.compileText(node, RegExp.$1);
}
//如果当前节点还有子节点 那么就需要递归查找所有的子节点是否符合以上条件
if (node.childNodes && node.childNodes.length) {
me.compileElement(node);
}
});
},
compile: function(node) {
var nodeAttrs = node.attributes,
me = this;//改变this指向
// [].slice.call(nodeAttrs)转数组
[].slice.call(nodeAttrs).forEach(function(attr) {
var attrName = attr.name;
if (me.isDirective(attrName)) {
var exp = attr.value;
var dir = attrName.substring(2);//substring 截取字符串
// 事件指令
if (me.isEventDirective(dir)) {
compileUtil.eventHandler(node, me.$vm, exp, dir);
// 普通指令
} else {
compileUtil[dir] && compileUtil[dir](node, me.$vm, exp);
}
node.removeAttribute(attrName);//移除属性
}
});
},
compileText: function(node, exp) {
compileUtil.text(node, this.$vm, exp);
},
//检测v-开头的指令
isDirective: function(attr) {
return attr.indexOf('v-') == 0;
},
//检测on开头的事件
isEventDirective: function(dir) {
return dir.indexOf('on') === 0;
},
/**
* 判断当前的node是不是元节点节点
* @param node 节点
* @returns {boolean}
*/
isElementNode: function(node) {
// node = "#app"
//node.nodeType 1 element元素
return node.nodeType == 1;
},
/**
* 判断当前的node是不是文本节点
* @param node 节点
* @returns {boolean}
*/
isTextNode: function(node) {
return node.nodeType == 3;
}
};
// 指令处理集合
var compileUtil = {
text: function(node, vm, exp) {
this.bind(node, vm, exp, 'text');
},
html: function(node, vm, exp) {
this.bind(node, vm, exp, 'html');
},
model: function(node, vm, exp) {
this.bind(node, vm, exp, 'model');
var me = this,//改变this指向
val = this._getVMVal(vm, exp);
//监听节点的内容发生改变
node.addEventListener('input', function(e) {
//将节点改变后的内容赋值给newValue
var newValue = e.target.value;
//然后判断新旧内容
if (val === newValue) {
return;
}
me._setVMVal(vm, exp, newValue);
val = newValue;
});
},
class: function(node, vm, exp) {
this.bind(node, vm, exp, 'class');
},
bind: function(node, vm, exp, dir) {
var updaterFn = updater[dir + 'Updater'];
// updaterFn && updaterFn(node, this._getVMVal(vm, exp));
if(updaterFn){
//node 当前的文本节点, 值 name
//updaterFn ==> node #text {{name}} _data.name
// updaterFn(node, this._data.name);
updaterFn(node, this._getVMVal(vm, exp));
}
new Watcher(vm, exp, function(value, oldValue) {
updaterFn && updaterFn(node, value, oldValue);
});
},
// 事件处理
eventHandler: function(node, vm, exp, dir) {
var eventType = dir.split(':')[1],
fn = vm.$options.methods && vm.$options.methods[exp];
if (eventType && fn) {
node.addEventListener(eventType, fn.bind(vm), false);
}
},
_getVMVal: function(vm, exp) {
//vm => $vm exp==>"name" "age.a1"
var val = vm._data;
// {
// name: "aa",
// age: {
// a1: 18
// }
// }
exp = exp.split('.'); //[age, a1]
exp.forEach(function(k) {//age
//val = {
// a1: 18
// }
val = val[k];
});
return val;
},
_setVMVal: function(vm, exp, value) {
//将vm实例上的_data赋值给val
var val = vm._data;
exp = exp.split('.');//转字符串以点相互隔开
exp.forEach(function(k, i) {
// 非最后一个key,更新val的值
if (i < exp.length - 1) {
val = val[k];
} else {
val[k] = value;
}
});
}
};
// a-> b ->c ->d 函数嵌套调用
// a-> a -> a -> 递归 ->特殊的函数嵌套
//更新器
var updater = {
textUpdater: function(node, value) {
//与下同理 textContent是节点原型上的属性
node.textContent = typeof value == 'undefined' ? '' : value;
},
htmlUpdater: function(node, value) {
//如果节点的内容传递过来的参数是什么数据类型 如果等于nudefined 执行表达式二即为'',否则是它本身
node.innerHTML = typeof value == 'undefined' ? '' : value;
},
classUpdater: function(node, value, oldValue) {
var className = node.className;
className = className.replace(oldValue, '').replace(/\\s$/, '');
var space = className && String(value) ? ' ' : '';
node.className = className + space + value;
},
//node 节点 value 传递过来的参数
modelUpdater: function(node, value, oldValue) {
node.value = typeof value == 'undefined' ? '' : value;
}
};
以上是关于Vue源码Compile.js的主要内容,如果未能解决你的问题,请参考以下文章