我的MVVM框架 v0.1发布
如果經(jīng)常瀏覽我博客的人就發(fā)現(xiàn),我三個月前就搞鼓過一下什么MVVM與MVC的比較,knockout.js與ember.js。然后就沒動靜了,因為之后一個月,我的MVVM就孵化出來,開發(fā)代碼很逆天也很齷鹺,就叫avalon(WPF開發(fā)代號)。我覺得,都是好東西,相互借鑒是沒所謂,只要造福IT民工!
說說MVVM的重要性,它雖然是從MVC中衍生出來的,但其雙向綁定機制是特有的,完全是面向界面開發(fā)而生,這是傳統(tǒng)的MVC比不了。最近我拉了一伙人在搞我的mass UI就遇到這問題了。
//由mass UI開發(fā)團隊的Hodor 提供 define('panel',['$node','$event','$css','$flow','./ejs' ], function(){$.ui = $.ui||{}var defaults = {showHead : true,showFoot : true,closeAble : true,parent : 'body',content : {title : 'title',body : 'body',foot : ''},css : {width : 400,height : 200}};$.ui.Panel = $.factory({inherit: $.Flow,init: function(opts) {this.setOptions ("data", defaults, opts )var self = this;self.template = $.ejs.compile('<div class="panel_wrap">\<% if( data.showHead ){ %>\<div class="panel_header">\<div class="panel_title">\<%= data.content.title %>\</div>\<% if( data.closeAble ){ %>\<span class="panel_closer"></span>\<% } %>\</div>\<% } %>\<div class="panel_body">\<%= data.content.body %>\</div>\<% if( data.showFoot ){ %>\<div class="panel_foot">\<%= data.content.foot %>\</div>\<% } %>\</div>');self.show();},show : function() {this.fire ( 'beforeshow' )this.ui && this.ui.remove();this.ui = $(this.template( this.data )).appendTo( this.parent ).css ( this.css ).show ();this.fire ( 'show' )},hide : function() {this.ui && this.ui.hide().remove();this.ui = undefined;this.fire ( 'hide' );},set : function( keyChain, val ) {//每改一個屬性就重刷整個視圖,因此不能容納子控件,除非我們多做一些額外工作var keys = keyChain.split('.');var key;var ret = this;while( keys.length > 1){key = keys.shift();ret[key] = ret[key] != undefined ? ret[key] :{};ret = ret[key];}ret[keys.shift()] = val;this.show();return this;}}); })遇到的問題與其他UI團隊一樣,其實你看一下jquery UI就知了。一個控件,它肯定有視圖層,這里我們是用ejs v10來生成。生成HTML插入到DOM算是完成了一半,但當我們要修改這個控件的一些屬性,一些與視圖相關(guān)的屬性就遇到麻煩了。比如說title屬性,如果是jquery ui,它肯定先到這個控件的ID,然后再找包含這個title的元素節(jié)點,再替換掉它的文本。用jquery是這樣寫出:
setTilte: function(newTitle){$(this.UIid).find("jquery-panel-title").text(newTitle);//更新視圖this.title = newTitle;//同時同步對應(yīng)的屬性}getTitle: function(){return this.title;}如果一個控件涉及的視圖屬性越多,控件這樣的訪問器就越多!長此以往不是辦法!像日歷組件,你可以看到它是多么臃腫。
這時該到MVVM出馬了。它把這些與視圖顯示相關(guān)的屬性全部收集到一起,包括基于這些屬性的屬性,比如說fullName是基于lastName與firstName,它是通過函數(shù)生成的,這個在模型(M)里是不存在的,但它存于視圖模型(VM)中。由于雙向綁定的存在,我們修改了VM的一個值,它立即自動刷新視圖中對應(yīng)的位置的值,這過程完全不需要動用到選擇器!
基于選擇器的方式是最不可靠的,因為視圖的變更最頻繁,某一天PM說要多加一功能或不要一功能,于是就多幾個父節(jié)點或少幾個子節(jié)點,把HTML的層級關(guān)系搞得亂七八糟,我們的選擇器尋找就得重寫一次!
因此對于這些DOM操作,我們最好也封裝一層,這是比jquery的DOM操作函數(shù)更高層的封裝,目的是讓控件開發(fā)者遠離視圖層,他們只需要關(guān)注于模型層。如果我們把DOM操作看成增刪改查,那么原始的DOM API相當于直接用二進制手段暴力去改數(shù)據(jù)庫,而像jQuery等主流庫提供了強大的選擇器引擎與DOM操作函數(shù),相當于發(fā)明SQL,而像knockout.js這樣的MVVM框架擁有尖端的依賴鏈機制、雙向綁定,讓智能的集化操作,事務(wù)與鎖成為可能,是DOM級別的“ORM系統(tǒng)”!
順便一提,在VM中,所有屬性都是函數(shù),每個函數(shù)都是讀寫結(jié)合,像jquery的html, attr, text那樣便捷!這有出于兼容IE678的考量,因為它們不支持Object.defineProperty這樣的屬性描述符(或支持不良好,如IE8)。
我的MVVM v1完全是向knockout.js致敬的,用法與它的一模一樣,不過代碼量少了許多。具體教程與實可以看以下鏈接:
- MVVM模式的實踐——avalon模塊
- avalon模塊:依賴鏈
- avalon模塊的內(nèi)建適配器
最近看了許多MVVM的實現(xiàn),功力大增,估計v2秒換胎脫骨,更精簡更高效,敬請期待!
define("avalon",["data","attr","event","fx"], function(){/* JS UI Component 最終還是通過 HTML 來描述界面,當 js object 的數(shù)據(jù)發(fā)生變化或者執(zhí)行某個動作時,需要通知到對應(yīng)的html,使其發(fā)生相應(yīng)變化。于是js object 需要得到他在頁面上對應(yīng)的html的句柄,通常做法,是在創(chuàng)建html的時候?qū)reateElement返回的句柄保存在js object 內(nèi)部的某個變量中,或者賦值給html eLement一個唯一的ID,js object 根據(jù)這個ID來找到對應(yīng)的HTML Element。同樣,當htm elementl的事件(例如onclick)要通知到相對應(yīng)的 js object 或者回調(diào)js object的某個方法或?qū)傩詴r,也需要得到該js object的一個引用。我的意思是建立一種統(tǒng)一的規(guī)則,js object和他相對應(yīng)的 html 能通過這種規(guī)則互相訪問到對方。 建立這個關(guān)聯(lián)以后,實現(xiàn)js object和對應(yīng) html 的數(shù)據(jù)邦定和數(shù)據(jù)同步等問題就簡單多了*/var disposeObject = {}var cur, ID = 1;var registry = {}var dependent = {}var fieldFns = {ensure : function(d){if(this.list.indexOf(d) == -1){this.list.push(d);}},lock : function(){this.locked = true;},unlock : function(){delete this.locked;},notify : function(){//通知依賴于field的上層$.computed更新var list = this.list || [] ;if( list.length ){var safelist = list.concat(), dispose = falsefor(var i = 0, el; el = safelist[i++];){delete el.cache;//清除緩存if(el.locked === true)breakif(el.dispose === true || el() == disposeObject ){//通知頂層的computed更新自身el.dispose = dispose = true}}if( dispose == true ){//移除無意義的綁定for ( i = list.length; el = list[ --i ]; ) {if( el.dispose == true ){el.splice( i, 1 );}}}}}}$.avalon = {//為一個Binding Target(節(jié)點)綁定Binding Source(viewModel)setBindings: function( source, node ){node = node || document.body; //確保是綁定在元素節(jié)點上,沒有指定默認是綁在body上//開始在其自身與孩子中綁定return setBindingsToElementAndChildren( node, source, true );},//取得節(jié)點的數(shù)據(jù)隱藏hasBindings: function( node ){var str = node.getAttribute( "data-bind" );return typeof str === "string" && str.indexOf(":") > 1},//將字符串變成一個函數(shù)evalBindings: function(expression, level){var body = "return (" + expression + ")";for (var i = 0; i < level; i++) {body = "with(sc[" + i + "]) { " + body + " } ";}return Function( "sc", body );},//轉(zhuǎn)換數(shù)據(jù)隱藏為一個函數(shù)parseBindings : function( node, context ){var jsonstr = $.normalizeJSON( node.getAttribute("data-bind"), true, context );var fn = $.avalon.evalBindings( jsonstr, 2 );//限制為兩層,減少作用鏈的長度return fn;},//開始收集依賴detectBegin: function( field ){var uuid = $.avalon.register( field )if( cur ){cur[uuid] = field}//用于收集依賴var prev = cur;cur = dependent[uuid];cur.prev = prev},//添加依賴到鏈中detectAdd: function( field ){if(cur){var uuid = $.avalon.register( field )cur[ uuid ] = field;}},//結(jié)束依賴收集detectEnd: function( field ){var deps = dependent[ field.observableID ] || {};cur = deps.prev;for(var key in deps){if(deps.hasOwnProperty(key) && (key != "prev")){var low = registry[ key ];low.ensure(field)}}},//注冊依賴register: function( field ){var uuid = field.observableIDif(!uuid || !registry[uuid] ){field.observableID = uuid = "observable" +(++ID);registry[uuid] = field;//供發(fā)布者使用dependent[uuid] = {};//收集依賴field.list = []$.mix(field, fieldFns);}return uuid;}}$.observable = function( val ){var cur = val;function field( neo ){$.avalon.detectAdd(field)if( arguments.length ){//setterif(cur !== neo || Array.isArray(cur) ){cur = neo;field.notify()}}else{//getterreturn cur;}}field.toString = field.valueOf = function(){return cur;}field();return field;}$.computed = function( obj, scope ){var getter, setter, cur//構(gòu)建一個至少擁有g(shù)etter,scope屬性的對象if(typeof obj == "function"){//getter必然存在getter = obj}else if( typeof obj == "object" && obj ){getter = obj.gettersetter = obj.setter;scope = obj.scope;}function field( neo ){if( arguments.length ){if(typeof setter === "function"){field.lock()//setter會喚起其依賴的$.observable與$.computed重新計算自身,但它們也會觸發(fā)其上級更新自身//由于自身已經(jīng)先行更新了,沒有再計算一次neo = setter.apply( scope, arguments );field.unlock()}}else{if( "cache" in field ){//getterneo = field.cache;//從緩存中讀取,防止遞歸讀取}else{neo = getter.call( scope );field.cache = neo;//保存到緩存}}if(cur !== neo || Array.isArray(cur) && (JSON.stringify(cur) != JSON.stringify(neo)) ){cur = neofield.notify()}return cur;}field.toString = field.valueOf = function(){return cur;}$.avalon.detectBegin( field )field();$.avalon.detectEnd( field )return field;}$.observableArray = function(array){if(!arguments.length){array = []}else if(!Array.isArray){throw "$.observableArray arguments must be a array"}var field = $.observable(array);makeObservableArray(field);return field;}function makeObservableArray( field ){("pop,push,shift,unshift,slice,splice,sort,reverse,remove,removeAt").replace( $.rword, function( method ){field[method] = function(){var array = this(), n = array.lengthArray.prototype.unshift.call(arguments, array);$.Array[method].apply( $.Array, arguments );if( /sort|reverse|splice/.test(method) ){field.notify()}else if( array.length != n ){field.notify()}}});}//template - name//foreach - data//value - data//options - data//event - handler//MVVM三大入口函數(shù)之一$.applyBindings = $.setBindings = $.avalon.setBindings;var parseBindings = $.avalon.parseBindings;//dataFor與contextFor是為事件的無侵入綁定服務(wù)的$.contextFor = function(node) {switch (node.nodeType) {case 1:var context = $._data(node,"bindings-context");if (context) return context;if (node.parentNode) return $.contextFor(node.parentNode);break;case 9:return void 0}return void 0;};$.dataFor = function(node) {var context = $.contextFor(node);return context ? context['$data'] : void 0;};//在元素及其后代中將數(shù)據(jù)隱藏與viewModel關(guān)聯(lián)在一起function setBindingsToElementAndChildren( node, source, setData ){if ( node.nodeType === 1 ){var continueBindings = true;if( $.avalon.hasBindings( node ) ){continueBindings = setBindingsToElement(node, source, setData ) }if( continueBindings ){var elems = getChildren( node )elems.length && setBindingsToChildren( elems, source, setData )}}}//viewModel類$.viewModel = function(current, parent){$.mix( this,current );if ( parent) {$.mix( this, parent );this['$parentContext'] = parent;this['$parent'] = parent['$data'];this['$parents'] = (parent['$parents'] || []).slice(0);this['$parents'].unshift( this['$parent'] );} else {this['$parents'] = [];this['$root'] = current;}this['$data'] = current;}$.viewModel.prototype = {extend : function(source){return $.mix( this,source )},alias: function( neo, old){if(this[ neo ]){this[ this[neo] ] = this[old]}return this;}}//為當前元素把數(shù)據(jù)隱藏與視圖模塊綁定在一塊function setBindingsToElement( node, context, setData ){//如果bindings不存在,則通過getBindings獲取,getBindings會調(diào)用parseBindingsString,變成對象var callback = parseBindings( node, context )//保存到閉包中context = context instanceof $.viewModel ? context : new $.viewModel( context );if( setData ){$._data(node,"bindings-context",context)}var getBindings = function(){//用于取得數(shù)據(jù)隱藏try{return callback( [ node, context ] )}catch(e){$.log(e)}}var bindings = getBindings();var continueBindings = true;for(var key in bindings){var adapter = $.bindingAdapter[key];if( adapter ){if( adapter.stopBindings ){continueBindings = false;}associateDataAndUI( node, bindings[key], context, key, getBindings)}}return continueBindings;}//setBindingsToChildren的第三第四參數(shù)是為了實現(xiàn)事件的無侵入綁定function setBindingsToChildren(elems, context, setData, force){for(var i = 0, n = elems.length; i < n ; i++){var node = elems[i]setBindingsToElementAndChildren( node, context, setData && !force );if( setData && force ){//這是由foreach綁定觸發(fā)$._data(node,"bindings-context", context)}}}//有一些域的依賴在定義vireModel時已經(jīng)確認了//而對元素的操作的$.computed則要在bindings中執(zhí)行它們才知function associateDataAndUI(node, field, context, key, getBindings){var adapter = $.bindingAdapter[key], initPhase = 0, cur;function symptom(){//這是依賴鏈的末梢,通過process操作節(jié)點if(!node){return disposeObject;//解除綁定}if(typeof field !== "function"){var bindings = getBindings();//每次都取一次,因為viewModel的數(shù)據(jù)已經(jīng)發(fā)生改變field = bindings["@mass_fields"][key];}if(initPhase === 0){cur = field();adapter.init && adapter.init(node, cur, field, context, symptom);}var neo = field();if( key == "case"){//這個應(yīng)該如何處理更好呢?if(field === context.$switch){//$default;neo = !context.$switch.not;}else{//如果前面有一個通過,那么它將不會進入$default分支;neo = context.$switch() == neo;if( neo ){context.$switch.not = true;}}}if(initPhase === 0 || cur != neo || Array.isArray(cur) ){//只要是處理bool假值的比較cur = neo;adapter.update && adapter.update(node, cur, field, context, symptom);}initPhase = 1;}$.computed( symptom, context.$data );}var inputOne = $.oneObject("text,password,textarea,tel,url,search,number,month,email,datetime,week,datetime-local")//一個數(shù)據(jù)綁定,負責界面的展示,另一個是事件綁定,負責更高層次的交互,比如動畫,數(shù)據(jù)請求,//從現(xiàn)影響viewModel,導致界面的再渲染$.bindingAdapter = {text: {update: function( node, val ){val = val == null ? "" : val+""if(node.childNodes.length === 1 && node.firstChild.nodeType == 3){node.firstChild.data = val;}else{$( node ).text( val );}}},value:{init: function(node, val, field){node.value = val;if(/input|textarea/i.test(node.nodeName) && inputOne[node.type]){$(node).on("input",function(){field(node.value)});}}},html: {update: function( node, val ){$( node ).html( val )},stopBindings: true},visible: {update: function( node, val ){node.style.display = val ? "" : "none";}},enable: {update: function( node, val ){if (val && node.disabled)node.removeAttribute("disabled");else if ((!val) && (!node.disabled))node.disabled = true;}},"class": {update: function( node, val ){if (typeof val == "object") {for (var className in val) {var shouldHaveClass = val[className];toggleClass(node, className, shouldHaveClass);}} else {val = String(val || '');toggleClass(node, val, true);}}} ,// { text-decoration: someValue }// { color: currentProfit() < 0 ? 'red' : 'black' }style: {update: function( node, val ){var style = node.style, styleNamefor (var name in val) {styleName = $.cssName(name, style) || namestyle[styleName] = val[ name ] || "";}}},attr: {update: function( node, val ){for (var name in val) {$.attr(node, name, val[ name ] )}}},click: {init: function( node, val, field, context ){$(node).bind("click",function(e){field.call( context, e )});}},"switch":{init:function( node, val, field, context){context.$switch = field;context.$default = fieldsetBindingsToChildren( node.childNodes, context )},update:function(node, val, field, context){delete context.$switch.not;//每次都清空它},stopBindings: true},checked: {init: function( node, val, field, context ){if(context.$hoist && context.$hoist.nodeType == 1 ){var expr = node.tagName +"['data-bind'="+node.getAttribute("data-bind") +"]";context.$hoisting && $(context.$hoist).delegate( expr,"change", function(){field(node.checked);});}else{$(node).bind("change",function(){field(node.checked);});}},update:function(node, val ){if ( node.type == "checkbox" ) {if (Array.isArray( val )) {node.checked = val.indexOf(node.value) >= 0;} else {node.checked = val;}} else if (node.type == "radio") {node.checked = ( node.value == val );}}}}//if unless with foreach四種bindings都是使用template bindings"if,unless,with,foreach,case".replace($.rword, function( type ){$.bindingAdapter[ type ] = {update : function(node, val, field, context, symptom){if(type == "case" && (typeof context.$switch != "function" )){throw "Must define switch statement above all";}$.bindingAdapter['template']['update'](node, val, function(){switch(type){//返回結(jié)果可能為 -1 0 1 2case "case":case "if":return !!val - 0;//1case "unless":return !val - 0;//0case "with":return 2;//2default:return -1;}}, context, symptom);},stopBindings: true}});var Tmpl = function(t){this.template = tthis.nodes = $.slice(t.childNodes)}Tmpl.prototype.recovery = function(){this.nodes.forEach(function( el ){this.template.appendChild(el)},this);return this.template}$.bindingAdapter[ "template" ] = {update: function(node, data, field, context, symptom){var ganso = symptom.ganso//取得最初的那個節(jié)點的內(nèi)部作為模塊if( !symptom.ganso ){//緩存,省得每次都創(chuàng)建//合并文本節(jié)點數(shù)node.normalize();//保存模板ganso = node.ownerDocument.createDocumentFragment();while((el = node.firstChild)){ganso.appendChild(el)}symptom.ganso = ganso;//復(fù)制一份出來放回原位var first = ganso.cloneNode(true);symptom.references = [ new Tmpl( first ) ];//先取得nodes的引用再插入DOM樹node.appendChild( first );symptom.prevData = [{}];//這是偽數(shù)據(jù),目的讓其update}// console.log("===============")var code = field(), el;first = symptom.references[0];// console.log(code)if( code > 0 ){ //處理with if bindingstemplate = first.recovery();var elems = getChildren( template );node.appendChild( template ); //顯示出來if( elems.length ){if( code == 2 ){//處理with bindingscontext = new $.viewModel( data, context )}return setBindingsToChildren( elems, context, true )}}else if( code == 0){//處理unless bindingsfirst.recovery();}if( code < 0 && data && isFinite(data.length) ){//處理foreach bindingsvar scripts = getEditScripts( symptom.prevData, data, true ), hasDelete//obj必須有x,yfor(var i = 0, n = scripts.length; i < n ; i++){var obj = scripts[i], tmpl = false;switch(obj.action){case "update":tmpl = symptom.references[ obj.x ];//這里要增強break;case "add":tmpl = new Tmpl( ganso.cloneNode(true) );symptom.references.push( tmpl );break;case "retain"://如果發(fā)生刪除操作,那么位于刪除元素之后的元素的索引值會發(fā)生改變//則重置它們if(obj.x !== obj.y){tmpl = symptom.references[ obj.x ];tmpl.index(obj.y);tmpl = null;}break;case "delete":tmpl = symptom.references[ obj.y ];$(tmpl.nodes).remove();hasDelete = tmpl.destroy = true;tmpl = null;break;};if(tmpl){(function( k, tmpl ){var template = tmpl.templateif(!template.childNodes.length){tmpl.recovery();//update}tmpl.index = $.observable(k)var subclass = new $.viewModel( data[ k ], context);subclass.extend( {$index: tmpl.index// $item: data[ k ]} )// .alias("$itemName", "$data")// .alias("$indexName", "$index");elems = getChildren( template );node.appendChild( template );if(elems.length){setBindingsToChildren(elems, subclass, true, true );}})(obj.y || 0, tmpl);}}symptom.prevData = data.concat();if(hasDelete){symptom.references = symptom.references.filter(function(el){return !el.destroy})};}return void 0},stopBindings: true}$.bindingAdapter.disable = {update: function( node, val ){$.bindingAdapter.enable.update(node, !val);}}var getChildren = function(node){var elems = [] ,ri = 0;for (node = node.firstChild; node; node = node.nextSibling){if (node.nodeType === 1){elems[ri++] = node;}}return elems;}$.bindingAdapter["css"] = $.bindingAdapter["class"]var toggleClass = function (node, className, shouldHaveClass) {var classes = (node.className || "").split(/\s+/);var hasClass = classes.indexOf( className) >= 0;//原className是否有這東西if (shouldHaveClass && !hasClass) {node.className += (classes[0] ? " " : "") + className;} else if (hasClass && !shouldHaveClass) {var newClassName = "";for (var i = 0; i < classes.length; i++)if (classes[i] != className)newClassName += classes[i] + " ";node.className = newClassName.trim();}}var getEditScripts = (function () {// 一個簡單的Levenshtein distance算法//編輯距離就是用來計算從原串(s)轉(zhuǎn)換到目標串(t)所需要的最少的插入,刪除和替換的數(shù)目,//在NLP中應(yīng)用比較廣泛,如一些評測方法中就用到了(wer,mWer等),同時也常用來計算你對原文本所作的改動數(shù)。//http://www.cnblogs.com/pandora/archive/2009/12/20/levenshtein_distance.html//https://gist.github.com/982927//http://www.blogjava.net/phyeas/archive/2009/01/10/250807.html//通過levenshtein distance算法返回一個矩陣,matrix[y][x]為最短的編輯長度var getEditDistance = function(from, to, table){var matrix = [], fn = from.length, tn = to.length;// 初始化一個矩陣,行數(shù)為b,列數(shù)為avar i, j, td;for(i = 0; i <= tn; i++){matrix[i] = [i];//設(shè)置第一列的值table && table.insertRow(i)}for(j = 0; j <= fn; j++){matrix[0][j] = j;//設(shè)置第一行的值if(table){for(i = 0; i <= tn; i++){td = table.rows[i].insertCell(j);if(isFinite(matrix[i][j])){td.innerHTML = matrix[i][j];td.className = "zero";}}}}// 填空矩陣for(i = 1; i <= tn; i++){for(j = 1; j <= fn; j++){if( to[i-1] == from[j-1] ){matrix[i][j] = matrix[i-1][j-1];//保留} else {matrix[i][j] = Math.min(matrix[i-1][j-1] + 1, //更新matrix[i][j-1] + 1, // 插入matrix[i-1][j] + 1); //刪除}if(table){td = table.rows[i].cells[j];td.innerHTML = matrix[i][j];}}}$.log(matrix.join("\n"));return matrix;};//返回具體的編輯步驟var _getEditScripts = function(from, to, matrix, table){var x = from.length, y = to.length, scripts = [], _action;if(x == 0 || y == 0){//如果原數(shù)組為0,那么新數(shù)組的都是新增的,如果新數(shù)組為0,那么我們要刪除所有舊數(shù)組的元素var n = Math.max(x,y), action = x == 0 ? "add" : "delete";for( var i = 0; i < n; i++ ){scripts[scripts.length] = {action: action,x: i,y: i}}}else{while( 1 ){var cur = matrix[y][x];if( y == 0 && x == 0){break;}var left = matrix[y][x-1]var diagon = matrix[y-1][x-1];var top = matrix[y-1][x];action = "retain"//top == left && cur == diagonvar min = Math.min(top, diagon, left);var td = table && (table.rows[y].cells[x]);x--;y--;if( min < cur ){switch(min){case top:action = "add";x++;break;case left:action = "delete";y++;break;case diagon:action = "update";if(_action){action = _action;_action = false;}break;}} else{switch(min){case top:_action = "add";x++;break;case left:_action = "delete";y++;break;}}if(table){td.className = action;}scripts[scripts.length] = {action:action,x:x,y:y}}}scripts.reverse();return scripts}return function( old, neo, debug ){if(debug){debug = document.createElement("table");document.body.appendChild(debug);debug.className = "compare";}var matrix = getEditDistance( old, neo, debug );return _getEditScripts( old, neo, matrix, debug );}})();//normalizeJSON及其輔助方法與變量void function(){var restoreCapturedTokensRegex = /\@mass_token_(\d+)\@/g;function restoreTokens(string, tokens) {var prevValue = null;while (string != prevValue) { // Keep restoring tokens until it no longer makes a difference (they may be nested)prevValue = string;string = string.replace(restoreCapturedTokensRegex, function (match, tokenIndex) {return tokens[tokenIndex];});}return string;}//https://github.com/SteveSanderson/knockout/wiki/Asynchronous-Dependent-Observables 偉大的東西//https://github.com/rniemeyer/knockout-kendo 一個UI庫//https://github.com/mbest/js-object-literal-parse/blob/master/js-object-literal-parse.jsfunction parseObjectLiteral(objectLiteralString) {var str = objectLiteralString.trim();if (str.length < 3)return [];if (str.charAt(0) === "{")// 去掉最開始{與最后的}str = str.substring(1, str.length - 1);// 首先用占位符把字段中的字符串與正則處理掉var tokens = [];var tokenStart = null, tokenEndChar;for (var position = 0; position < str.length; position++) {var c = str.charAt(position);//IE6字符串不支持[],開始一個個字符分析if (tokenStart === null) {switch (c) {case '"':case "'":case "/":tokenStart = position;//索引tokenEndChar = c;//值break;}//如果再次找到一個與tokenEndChar相同的字符,并且此字符前面不是轉(zhuǎn)義符} else if ((c == tokenEndChar) && (str.charAt(position - 1) !== "\\")) {var token = str.substring(tokenStart, position + 1);tokens.push(token);var replacement = "@mass_token_" + (tokens.length - 1) + "@";//對應(yīng)的占位符str = str.substring(0, tokenStart) + replacement + str.substring(position + 1);position -= (token.length - replacement.length);tokenStart = null;}}// 將{},[],()等括起來的部分全部用占位符代替tokenEndChar = tokenStart = null;var tokenDepth = 0, tokenStartChar = null;for (position = 0; position < str.length; position++) {var c = str.charAt(position);if (tokenStart === null) {switch (c) {case "{": tokenStart = position; tokenStartChar = c;tokenEndChar = "}";break;case "(": tokenStart = position; tokenStartChar = c;tokenEndChar = ")";break;case "[": tokenStart = position; tokenStartChar = c;tokenEndChar = "]";break;}}if (c === tokenStartChar)tokenDepth++;else if (c === tokenEndChar) {tokenDepth--;if (tokenDepth === 0) {var token = str.substring(tokenStart, position + 1);tokens.push(token);replacement = "@mass_token_" + (tokens.length - 1) + "@";str = str.substring(0, tokenStart) + replacement + str.substring(position + 1);position -= (token.length - replacement.length);tokenStart = null;}}}//拆解字段,還原占位符的部分var result = [];var keyValuePairs = str.split(",");for (var i = 0, j = keyValuePairs.length; i < j; i++) {var pair = keyValuePairs[i];var colonPos = pair.indexOf(":");if ((colonPos > 0) && (colonPos < pair.length - 1)) {var key = pair.substring(0, colonPos);var value = pair.substring(colonPos + 1);result.push({'key': restoreTokens(key, tokens),'value': restoreTokens(value, tokens)});} else {//到這里應(yīng)該拋錯吧result.push({'unknown': restoreTokens(pair, tokens)});}}return result;}function ensureQuoted(key) {var trimmedKey = key.trim()switch (trimmedKey.length && trimmedKey.charAt(0)) {case "'":case '"':return key;default:return "'" + trimmedKey + "'";}}// var e = $.normalizeJSON("{aaa:111,bbb:{ccc:333, class:'xxx', eee:{ddd:444}}}");$.normalizeJSON = function (json, insertFields, extra) {//對鍵名添加引號,以便安全通過編譯var keyValueArray = parseObjectLiteral(json),resultStrings = [] ,keyValueEntry, propertyToHook = [];for (var i = 0; keyValueEntry = keyValueArray[i]; i++) {if (resultStrings.length > 0)resultStrings.push(",");if (keyValueEntry['key']) {var key = keyValueEntry['key'].trim();var quotedKey = ensureQuoted(key), val = keyValueEntry['value'].trim();resultStrings.push(quotedKey);resultStrings.push(":");// if(insertFields === true && key === "foreach"){//特殊處理foreach// var array = val.match($.rword);// val = array.shift();// if(array[0] === "as"){//如果用戶定義了多余參數(shù)// extra.$itemName = array[1];// extra.$indexName = array[2];// }// }if(val.charAt(0) == "{" && val.charAt(val.length - 1) == "}"){val = $.normalizeJSON( val );//逐層加引號}resultStrings.push(val);if(insertFields == true){//用函數(shù)延遲值部分的執(zhí)行if (propertyToHook.length > 0)propertyToHook.push(", ");propertyToHook.push(quotedKey + " : function() { return " + val + " }")}} else if (keyValueEntry['unknown']) {resultStrings.push(keyValueEntry['unknown']);//基于跑到這里就是出錯了}}resultStrings = resultStrings.join("");if(insertFields == true){resultStrings += ' , "@mass_fields": {'+ propertyToHook.join("") + '}'}return "{" +resultStrings +"}";}}();});總結(jié)
以上是生活随笔為你收集整理的我的MVVM框架 v0.1发布的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: sencha touch中list如何撑
- 下一篇: Hive Performance 学习