Vue雙向綁定的實現原理系列(四):補充指令解析器compile

補充指令解析器compile

github源碼vue

補充下HTML節點類型的知識:
    元素節點              Node.ELEMENT_NODE(1)
    屬性節點              Node.ATTRIBUTE_NODE(2)
    文本節點              Node.TEXT_NODE(3)
    CDATA節點             Node.CDATA_SECTION_NODE(4)
    實體引用名稱節點       Node.ENTRY_REFERENCE_NODE(5)
    實體名稱節點          Node.ENTITY_NODE(6)
    處理指令節點          Node.PROCESSING_INSTRUCTION_NODE(7)
    註釋節點              Node.COMMENT_NODE(8)
    文檔節點              Node.DOCUMENT_NODE(9)
    文檔類型節點          Node.DOCUMENT_TYPE_NODE(10)
    文檔片斷節點          Node.DOCUMENT_FRAGMENT_NODE(11)
    DTD聲明節點            Node.NOTATION_NODE(12)

Compile指令解析器,解析DOM節點,直接固定某個節點進行替換數據的node

解析模板指令,替換模板數據,初始化試圖

將模板指令對應的節點綁定對應的更新函數,初始化對應的訂閱器git

首先須要獲取到DOM元素,而後對含有DOM元素上含有指令的節點進行處理,
所以這個環節須要對DOM操做比較頻繁,全部能夠先建一個fragment片斷,
將須要解析的DOM節點存入fragment片斷裏再進行處理:github

//直接上代碼:(先判斷"{{}}")
    function Compile(elm){// el->"#name" ,vm->{el:;data:;}
        this.vm = elm;
        this.el = document.querySelector(elm.el);
        this.fragment = null;
        this.init();
    }
    Compile.prototype = {
        init:function(){
            if(this.el) {
                //將須要解析的DOM節點存入fragment片斷裏再進行處理
                this.fragment = this.nodeToFragment(this.el);
                
                //接下來遍歷各個節點,對含有指定的節點特殊處理,先處理指令「{{}}」:
                this.compileElement(this.fragment);
                
                //綁定到el上
                this.el.appendChild(this.fragment);
            }else{
                console.log('DOM元素不存在');
            }
        },
        //建立代碼片斷
        nodeToFragment:function(el){
            var fragment = document.createDocumentFragment();
            var child = el.firstChild;
            while(child){//將DOM元素移入fragment
                fragment.appendChild(child);
                child = el.firstChild;
            }
            return fragment;
        },
         //對全部子節點進行判斷,1.初始化視圖數據,2.綁定更新函數的訂閱器
        compileElement:function(el){
            var childNodes = el.childNodes;
            var self = this;
            [].slice.call(childNodes).forEach(function(node){
                var reg = /\{\{(.*)\}\}/;//匹配" {{}} "
                var text = node.textContent;
    
                if(self.isTextNode(node) && reg.test(text)) {//判斷" {{}} "
                    self.compileText(node,reg.exec(text)[1]);
                }
    
                if(node.childNodes && node.childNodes.length){
                    self.compileElement(node);//// 繼續遞歸遍歷子節點
                }
            });
        },
        //初始化視圖updateText和生成訂閱器:
        compileText:function(node,exp){
            var self = this;
            var initText = this.vm[exp];   //代理訪問self_vue.data.name1 -> self_vue.name1
            this.updateText(node,initText);//將初始化的數據初始化到視圖中
            new Watcher(this.vm,exp,function(value){//{},name, // 生成訂閱器並綁定更新函數
                self.updateText(node,value);
            })
        },
        updateText: function (node, value) {
            node.textContent = typeof value == 'undefined' ? '' : value;
        },
        isTextNode:function(node){
            return node.nodeType == 3;//文本節點
        }
    };

爲了將解析器Compile與監聽器Observer和訂閱者Watcher關聯起來,咱們須要再修改一下類SelfVue函數:segmentfault

// function SelfVue(data,el,exp){  //first
    function SelfVue(options){
        var self = this;
    
        // this.data = data;  //first
        this.data = options.data;
        this.el = options.el;
    
        this.vm = this; //second
        console.log(this)
    
        Object.keys(this.data).forEach(function (key) {
            self.proxyKeys(key);//綁定代理屬性
        });
    
        //監聽數據:
        observers(this.data);
    
        //first:
        /*el.innerHTML = this.data[exp];//初始化模板數據的值
        new Watcher(this,exp,function(value){//綁定訂閱器
            el.innerHTML = value;
        });*/
    
        //初始化視圖updateText和生成訂閱器
        new Compile(this);
    
        return this;
    }

到這裏,大括號"{{}}"類型的雙向數據綁定完成;設計模式

補充上v-model和事件指令:

在compileElement函數加上對其餘指令節點進行判斷,而後遍歷其全部屬性bash

添加事件指令

添加一個v-model指令app

Compile.prototype = {
      init:function(){
          if(this.el) {
              //將須要解析的DOM節點存入fragment片斷裏再進行處理
              this.fragment = this.nodeToFragment(this.el);
  
              //接下來遍歷各個節點,對含有指定的節點特殊處理,先處理指令「{{}}」:
              this.compileElement(this.fragment);
  
              //綁定到el上
              this.el.appendChild(this.fragment);
          }else{
              console.log('DOM元素不存在');
          }
      },
      //建立代碼片斷
      nodeToFragment:function(el){
          var fragment = document.createDocumentFragment();
          var child = el.firstChild;
          while(child){//將DOM元素移入fragment
              fragment.appendChild(child);
              child = el.firstChild;
          }
          return fragment;
      },
      //對全部子節點進行判斷,1.初始化視圖數據,2.綁定更新函數的訂閱器
      compileElement:function(el){
          var childNodes = el.childNodes;
          var self = this;
          [].slice.call(childNodes).forEach(function(node){
              var reg = /\{\{(.*)\}\}/;//匹配" {{}} "
              var text = node.textContent;
/*      補充判斷:     */
              if(self.isElementNode(node)){//元素節點判斷
                  self.compile(node);
              }else if(self.isTextNode(node) && reg.test(text)) {//文本節點判斷 ,判斷" {{}} "
                  self.compileText(node,reg.exec(text)[1]);
              }
  
              if(node.childNodes && node.childNodes.length){
                  self.compileElement(node);//// 繼續遞歸遍歷子節點
              }
          });
      },
      //初始化視圖updateText和生成訂閱器:
      compileText:function(node,exp){
          var self = this;
          var initText = this.vm[exp];   //代理訪問self_vue.data.name1 -> self_vue.name1
          this.updateText(node,initText);//將初始化的數據初始化到視圖中
          new Watcher(this.vm,exp,function(value){//{},name, // 生成訂閱器並綁定更新函數
              self.updateText(node,value);
          })
      },
      updateText: function (node, value) {
          node.textContent = typeof value == 'undefined' ? '' : value;
      },
      compile:function(node){
          var nodeAttrs = node.attributes;
          var self = this;
          Array.prototype.forEach.call(nodeAttrs,function(attr){
              var attrName = attr.name;
              if(self.isDirective(attrName)){//查到" v- "
                  var exp = attr.value;
                  var dir = attrName.substring(2);//" v-on/v-model "
  
                  if(self.isEventDirective(dir)){ // 事件指令
                      self.compileEvent(node,self.vm,exp,dir);
                  }else{
                      self.compileModel(node,self.vm,exp,dir);
                  }
                  node.removeAttribute(attrName);
              }
          })
      },
      compileEvent:function(node,vm,exp,dir) {//代碼片斷<><>,{data:;vm:;el:;},v-on="add",on:
          var eventType = dir.split(':')[1];//on
          var cb = vm.methods && vm.methods[exp];
  
          if(eventType && cb){
              node.addEventListener(eventType,cb.bind(vm),false);
          }
      },
      compileModel:function(node,vm,exp,dir){//代碼片斷<><>,{data:;vm:;el:;},v-on="addCounts",model:
          var self = this;
          var val = this.vm[exp];
          this.modelUpdater(node,val);
          new Watcher(this.vm,exp,function(value){
              self.modelUpdater(node,value);
          });
  
          node.addEventListener('input',function(e){
              var newValue = e.target.value;
              if(val === newValue){
                  return;
              }
              self.vm[exp] = newValue;
              val = newValue;
          })
      },
      modelUpdater:function(node,value){
          node.value = typeof value == 'undefined' ? '' : value;
      },
      isTextNode:function(node){
          return node.nodeType == 3;//文本節點
      },
      isElementNode:function(node){
          return node.nodeType == 1;//元素節點<p></p>
      },
      isDirective:function(attr){//查找自定義屬性爲:v- 的屬性
          return attr.indexOf('v-') == 0;
      },
      isEventDirective:function(dir){ // 事件指令
          return dir.indexOf('on:') === 0
      }
  };

再改造下類SelfVue,使它更像Vue的用法:函數

function SelfVue(options){
        var self = this;
        this.data = options.data;
        this.el = options.el;
        this.methods = options.methods;
    
        this.vm = this; //second
    
        Object.keys(this.data).forEach(function (key) {
            self.proxyKeys(key);//綁定代理屬性
        });
    
        //監聽數據:
        observers(this.data);
     
        //初始化視圖updateText和生成訂閱器
        new Compile(this);
        options.mounted.call(this);
    
        return this;
    }

測試:測試

<body>
<div id="app">
    <h2>{{name1}}</h2>
    <h2>{{name2}}</h2>


    <input type="text" v-model="title">
    <h3>{{title}}</h3>


    <button v-on:click="clickMe">v-on事件</button>
    <h3>{{event}}</h3>
</div>

    <script src="js-second/observer.js"></script>
    <script src="js-second/watcher.js"></script>
    <script src="js-second/compile.js"></script>
    <script src="js-second/selfVue.js"></script>
<script>

    var self_vue = new SelfVue({
        el:"#app",
        data:{
            name1:'我是name1',
            name2:'我是name2',
            event:'event',
            title:'title初始值'
        },
        methods:{
            clickMe:function(){
                this.event = '事件從新賦值'
            }
        },
        mounted:function(){
            console.log('mounted')
        }
    });

   /* window.setTimeout(function(){
        self_vue.name1 = 'name1再次賦值'
    },2000);
    window.setTimeout(function(){
        self_vue.name2 = 'name2再次賦值'
    },3000)*/
</script>
</body>

系列文章的目錄:

Vue雙向綁定的實現原理系列(一):Object.defineproperty
Vue雙向綁定的實現原理系列(二):設計模式
Vue雙向綁定的實現原理系列(三):監聽器Observer和訂閱者Watcher
Vue雙向綁定的實現原理系列(四):補充指令解析器compile

相關文章
相關標籤/搜索