diff算法實現

大體流程
var vnode = {
  tag: 'ul',
  attrs: {
    id: 'list'
  },
  children: [{
    tag: 'li',
    attrs: {
      className: 'item'
    },
    children: ['item 1']
  },{
    tag: 'li',
    attrs: {
      className: 'item'
    },
    children: ['item 2']
  }]
}
function createElement(vnode){
  var tag = vnode.tag;
  var attrs = vnode.attrs || {};
  var children = vnode.children || [];
  if (tag == null) {
    return null;
  }

  // 建立真實的 dom 元素
  var elem = document.createElement(tag);
  // 屬性
  var attrName;
  for (attrName in attrs) {
    if(attrs.hasOwnProperty(attrName)){
      elem.setAttribute(attrName, attrs[attrName]);
    }
  }
  // 子元素
  children.forEach(function(childVnode){
    elem.appendChild(childElem(childVnode)) // 遞歸
  });

  // 返回真實的 dom 元素
  return elem;
}

就是將對象一一轉化成domnode




上面是patch的第一個參數沒有內容的時候,若是有內容就會作對比。
var vnode = {
  tag: 'ul',
  attrs: {
    id: 'list'
  },
  children: [{
    tag: 'li',
    attrs: {
      className: 'item'
    },
    children: ['item 1']
  },{
    tag: 'li',
    attrs: {
      className: 'item'
    },
    children: ['item 2']
  }]
}

var newVnode = {
  tag: 'ul',
  attrs: {
    id: 'list'
  },
  children: [{
    tag: 'li',
    attrs: {
      className: 'item'
    },
    children: ['item 1']
  },{
    tag: 'li',
    attrs: {
      className: 'item'
    },
  children: ['item b']
  }]
}


function updateChildren(vnode ,newVnode){
  var children = vnode.children || [];
  var newChildren = newVnode.children || [];

  children.forEach(function(childrenVnode, index){
    var newChildrenVnode = newChildren[index];
    if(childrenVnode.tag === newChildrenVnode.tag){
      // 深層次對比,遞歸
      updateChildren(childrenVnode, newChildrenVnode);
    } else {
      // 替換
      replaceNode(childrenVnode, newChildrenVnode);
    }
  });
}

function replaceNode(vnode, newVnode){
  var elem = vnode.elem; // 真實的 dom
  var newElem = newVnode.elem;

  // 替換
}

無法知道有多少層,就必須用遞歸app

相關文章
相關標籤/搜索