MVVM源码解析之模板解析篇
Posted 理小理...
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了MVVM源码解析之模板解析篇相关的知识,希望对你有一定的参考价值。
源码解析之模板解析
解析表达式
解析步骤:
- 从文本节点中取出表达式
- 从data中取出表达式对应的属性值
- 将属性值设置为文本节点的
textContent
const vm = new MVVM(
el: "#app",
data:
name: "aa"
);
在源码中,判断当前节点是否存在,从而进行解析。
通过正则进行匹配大括号表达式,匹配成功后,判断当前元素是否包含文本节点
compileElement: function(el)
//初始化数据,保存所有子节点 保存this
var childNodes = el.childNodes,
me = this;
//对所有子节点进行递归遍历
[].slice.call(childNodes).forEach(function(node)
//text节点的文本内容
var text = node.textContent;
//声明匹配大括号表达式的正则
var reg = /\\\\(.*)\\\\/; //name+age+phone //()非贪婪匹配 ->name+age+phone
// var reg = /(.*)/; //name
//判断当前节点是不是元素节点
if (me.isElementNode(node))
//解析指令
me.compile(node);
//判断当前元素是否为文本节点 并且 文本节点中是否拥有xxx
else if (me.isTextNode(node) && reg.test(text))
//解析文本(大括号表达式)并且赋值
me.compileText(node, RegExp.$1); //name
//如果当前节点还有子节点 那么就需要递归查找所有的子节点是否符合以上条件
if (node.childNodes && node.childNodes.length)
me.compileElement(node);
);
,
compileText: function(node, exp)
compileUtil.text(node, this.$vm, exp);
,
var compileUtil =
//解析v-text指令
text: function(node, vm, exp)
this.bind(node, vm, exp, 'text');
,
解析普通指令
解析步骤:
- 从元素节点中取出指令
- 从指令名中取出事件名
- 根据指令的值(表达式)从methods中得到对应的回调函数
- 给当前元素节点绑定指定的事件名和回调函数(指定this指向为vm)
- 移除元素的指令属性
<button v-on:click="show">提示</button>
<body>
<div id="app">
<h2>name</h2>
<button v-on:click="show">提示</button>
</div>
<script src="./mvvm-master/js/observer.js"></script>
<script src="./mvvm-master/js/compile.js"></script>
<script src="./mvvm-master/js/watcher.js"></script>
<script src="./mvvm-master/js/mvvm.js"></script>
<script>
const vm = new MVVM(
el: "#app",
data:
name: "aa"
,
methods:
show()
alert("这是一个提示" + this.name)
);
function show()
alert("这是一个提示" + this.name)
show();
var btn = document.getElementById("btn");
btn.addEventListener("click", show, false)
</script>
compileElement: function(el)
//初始化数据,保存所有子节点 保存this
var childNodes = el.childNodes,
me = this;
//对所有子节点进行递归遍历
[].slice.call(childNodes).forEach(function(node)
//text节点的文本内容
var text = node.textContent;
//声明匹配大括号表达式的正则
var reg = /\\\\(.*)\\\\/; //name+age+phone //()非贪婪匹配 ->name+age+phone
// var reg = /(.*)/; //name
//判断当前节点是不是元素节点
if (me.isElementNode(node))
//解析指令
me.compile(node);
//如果当前节点还有子节点 那么就需要递归查找所有的子节点是否符合以上条件
if (node.childNodes && node.childNodes.length)
me.compileElement(node);
);
,
//解析指令
compile: function(node) //button
//获取元素中的所有属性节点
var nodeAttrs = node.attributes,
me = this;
//遍历所有属性节点
[].slice.call(nodeAttrs).forEach(function(attr)
var attrName = attr.name;//取出属性名
if (me.isDirective(attrName)) //判断当前属性名是否为指令 (根据是否有v-)
var exp = attr.value;//show //获取指令值
var dir = attrName.substring(2);//on:click //去掉v- 取出指令名
// 判断当前指令是否为事件指令(是否有on)
if (me.isEventDirective(dir))
// node.addEventListener("dir",exp,false);
//为当前元素绑定事件
compileUtil.eventHandler(node, me.$vm, exp, dir);
// 普通指令
else
compileUtil[dir] && compileUtil[dir](node, me.$vm, exp);
//移除解析完成的指令
node.removeAttribute(attrName);
);
,
isDirective: function(attr)
return attr.indexOf('v-') == 0;
,
isEventDirective: function(dir)
return dir.indexOf('on') === 0;
,
//解析v-html指令
html: function(node, vm, exp)
this.bind(node, vm, exp, 'html');
,
//解析v-model指令
model: function(node, vm, exp)
this.bind(node, vm, exp, 'model');
var me = this,
val = this._getVMVal(vm, exp);
node.addEventListener('input', function(e)
var newValue = e.target.value;
if (val === newValue)
return;
me._setVMVal(vm, exp, newValue);
val = newValue;
);
,
//解析v-class指令
class: function(node, vm, exp)
this.bind(node, vm, exp, 'class');
,
//解析v-bind指令
bind: function(node, vm, exp, dir)
//根据指令名称获取对应的更新函数
var updaterFn = updater[dir + 'Updater'];
//如果更新函数存在 则执行更新
// updaterFn && updaterFn(node, this._getVMVal(vm, exp));
if(updaterFn)
updaterFn(node, this._getVMVal(vm, exp));
//Watcher监听者 vm实例 exp表达式a/v-text="a"
new Watcher(vm, exp, function(value, oldValue)
updaterFn && updaterFn(node, value, oldValue);
);
,
//获取vm中data里相对应的属性值
_getVMVal: function(vm, exp)
var val = vm._data;
exp = exp.split('.'); //[age, a1]
exp.forEach(function(k) //age
val = val[k];
);
return val;
,
//设置vm中data里相对应的属性值
_setVMVal: function(vm, exp, value)
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;
);
;
//更新器 操作原生DOM的方法
var updater =
//更新节点的textContent属性
textUpdater: function(node, value)
node.textContent = typeof value == 'undefined' ? '' : value;
,
//更新节点的innerHTML属性
htmlUpdater: function(node, value)
node.innerHTML = typeof value == 'undefined' ? '' : value;
,
//更新节点的className属性
classUpdater: function(node, value, oldValue)
var className = node.className; //className = > "bb"
node.className = className + (className?' ':'') + value; //bb aa
,
//更新节点的value属性
modelUpdater: function(node, value, oldValue)
node.value = typeof value == 'undefined' ? '' : value;
;
关于源码的解析大概就是这样,都是根据条件进行判断,通过柯力化
函数的思想,来对函数进行了封装,通过调用,来执行。
最后是所有的源码
function Compile(el, vm)
this.$vm = vm; //this Compile的实例 $vm 是MVVM的实例 (vm)
// el == "#app" 判断当前用户传递的el属性是元素节点还是选择器,如果是元素节点则直接保存到$el中通,
//如果不是 则根据选择器 去查找对应的元素 然后保存
this.$el = this.isElementNode(el) ? el : document.querySelector(el);
//确定元素是否真正存在
if (this.$el) //#app
this.$fragment = this.node2Fragment(this.$el);
this.$vm.$options.beforeMounted && this.$vm.$options.beforeMounted();
this.init();//初始化
this.$el.appendChild(this.$fragment);
this.$vm.$options.mounted && this.$vm.$options.mounted();
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)
//初始化数据,保存所有子节点 保存this
var childNodes = el.childNodes,
me = this;
//对所有子节点进行递归遍历
[].slice.call(childNodes).forEach(function(node)
//text节点的文本内容
var text = node.textContent;
//声明匹配大括号表达式的正则
var reg = /\\\\(.*)\\\\/; //name+age+phone //()非贪婪匹配 ->name+age+phone
// var reg = /(.*)/; //name
//判断当前节点是不是元素节点
if (me.isElementNode(node))
//解析指令
me.compile(node);
//判断当前元素是否为文本节点 并且 文本节点中是否拥有xxx
else if (me.isTextNode(node) && reg.test(text))
//解析文本(大括号表达式)并且赋值
me.compileText(node, RegExp.$1); //name
//如果当前节点还有子节点 那么就需要递归查找所有的子节点是否符合以上条件
if (node.childNodes && node.childNodes.length)
me.compileElement(node);
);
,
//解析指令
compile: function(node) //button
//获取元素中的所有属性节点
var nodeAttrs = node.attributes,
me = this;
//遍历所有属性节点
[].slice.call(nodeAttrs).forEach(function(attr)
var attrName = attr.name;//取出属性名
if (me.isDirective(attrName)) //判断当前属性名是否为指令 (根据是否有v-)
var exp = attr.value;//show //获取指令值
var dir = attrName.substring(2);//on:click //去掉v- 取出指令名
// 判断当前指令是否为事件指令(是否有on)
if (me.isEventDirective(dir))
// node.addEventListener("dir",exp,false);
//为当前元素绑定事件
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);
,
isDirective: function(attr)
return attr.indexOf('v-') == 0;
,
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 =
//解析v-text指令
text: function(node, vm, exp)
this.bind(node, vm, exp, 'text');
,
//解析v-html指令
html: function(node, vm, exp)
this.bind(node, vm, expMVVM源码解析之Watcher监听