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
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