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的主要内容,如果未能解决你的问题,请参考以下文章

compile.js源码分析

compile.js源码分析

VSCode自定义代码片段1——vue主模板

VSCode自定义代码片段——.vue文件的模板

VSCode自定义代码片段(vue主模板)

VSCode自定义代码片段11——vue路由的配置