迷你MVVM框架 avalonjs 0.85发布
迷你MVVM框架 avalonjs 0.85发布
本版本对循环绑定做了巨大改进,感谢@soom, @limodou, @ztz, @Gaubee 提供的大量测试文件。
- fix scanNodes, 在循环绑定(ms-each)扫描元素节点时必须 nextTick,否则旧式IE会忙碌不过来。
- fix ms-css ,旧式IE style[name] = value, 当value为NaN ,不带单位或不是数值什么会抛异常,需要try catch。
- 旧式IE下有些元素的innerHTML是只读的, 因此不能一律使用innerHTML,并且有些元素的生成,如script标签是不会执行,为此我引入新的parseHTML模块来处理此事。
- fix AMD 加载因为手误进错分支的BUG
- fix scanExpr bug, 它在IE10有时会多生成一个绑定对象,异致不渲染错误。
- 重构Collection内部对象与ms-each绑定,引入“事务”的概念,让其插入节点时更加智能高效。
我们看最后一条,我们可以类似纯JS操作为内存操作,DOM操作为IO操作,执行一万次前者所需的时间可能还比不上一次后者的。DOM操作的开销就是这么大。有的DOM操作还会引起reflow,这危害更大。因此明智的做法就是将要操作的节点移出DOM树。更好的办法是,此多个DOM操作合成一个,全部在文档碎片中搞完才插入DOM树。
我们看下面的注解:
< div ms-controller = "box" > < div ms-each-el = "array" id = "aaa" > < p >{{$index}}----{{el}}</ p > </ div > </ div > avalon.define("box", function(vm) { vm.array = [1, 2, 3, 4, 5] }) 实现过程 当扫描到div#aaa 将div#aaa的所有子节点复制一份到文档碎片vTemplate 执行begin命令,将vTemplate复制一个空的文档碎片vTransation( cloneNode(false) ), 设置全局变量flagTransation = true; 开始循环数组 执行insert命令 将vTemplate复制一个文档碎片vEl( cloneNode(true) ), 将对应的子VM与它进行扫描 此时它的内容应为 < p >0 --- 1</ p > 将vEl appendChild到 vTemplate ..... 重复执行array.length次 执行commit命令,将vTemplate append到div#aaa节点中, 设置全局变量flagTransation = false 重新排列所有$index |
在数组有关添加元素的push, unshift, splice这三个方法中,都调用了add方法,它里面就默认使用事件进行处理。
array._splice = array.splice array.add = function (arr, insertPos) { insertPos = typeof insertPos === "number" ? insertPos : this .length; notifySubscribers( this , "begin" ) for ( var i = 0, n = arr.length; i < n; i++) { var el = convert(arr[i]) var pos = insertPos + i this ._splice(pos, 0, el) notifySubscribers( this , "insert" , pos, el) } notifySubscribers( this , "commit" , insertPos) if (! this .stopFireLength) { return dynamic.length = this .length } } |
notifySubscribers会向上通知updateListView方法,然后让它执行相关的DOM操作
case "begin" : list.vTransation = data.vTemplate.cloneNode( false ) flagTransation = true case "insert" : //将子视图插入到文档碎片中 var tmodel = createVModel(pos, el, list, data.args) var tview = data.vTemplate.cloneNode( true ) tmodel.$view = tview vmodels = [tmodel].concat(vmodels) tmodels.splice(pos, 0, tmodel) scanNodes(tview, vmodels); data.group = ~~tview.childNodes.length //记录每个模板一共有多少子节点 list.vTransation.appendChild(tview) break case "commit" : pos = ~~pos //得到插入位置 IE6-10要求insertBefore的第2个参数为节点或null,不能为undefined var insertNode = parent.childNodes[ data.group * pos] || null parent.insertBefore(list.vTransation, insertNode) flagTransation = false resetItemIndex(tmodels) break |
嘛,不过这次改动太大了,有关Collection与bindingHandlers["each"]的代码都几乎改清光。另一个值得一提的是VM数组在腓序时,与视图的同步。这里涉及如何让一个数组基于另一个数组进行排序,我的解决方式如下:
var aaa = [1, 2, 3, 4, 5, 1] var bbb = [{v: 2}, {v: 3}, {v: 1}, {v: 1}, {v: 5}, {v: 4}] var swapTime = 0 var isEqual = Object.is || function (x, y) { //主要用于处理NaN 与 NaN 比较 if (x === y) { return x !== 0 || 1 / x === 1 / y; } return x !== x && y !== y; }; for ( var i = 0, n = bbb.length; i < n; i++) { var a = aaa[i]; var b = bbb[i] var b = b && b.v ? b.v : b if (!isEqual(a, b)) { console.log(++swapTime) var index = getIndex(a, bbb, i); var el = bbb.splice(index, 1) bbb.splice(i, 0, el[0]) } } function getIndex(a, bbb, start) { for ( var i = start, n = bbb.length; i < n; i++) { var b = bbb[i]; var check = b && b.v ? b.v : b if (isEqual(a, check)) { return i } } } console.log(JSON.stringify(bbb)) //在框架中,aaa为数据模型M中的数组,bbb为视图模型VM中的数组 |
如果有更好的算法,请多多指教。
经过这次大重构后,avalon在API上基本没有变化了,未来的v0.9就是fix BUG然后发布正式版。
迷你MVVM框架在github的仓库https://github.com/RubyLouvre/avalon
官网地址http://rubylouvre.github.io/mvvm/
大家可以加入QQ群:79641290进行讨论,此群为技术群,禁水!
迷你MVVM框架 avalonjs 0.85发布的更多相关文章
- 迷你MVVM框架 avalonjs 0.95发布
迷你MVVM框架 avalonjs 0.95发布 本版本最主要的改进是ms-with 深层绑定的实现,至少,avalon1.0所有重要的feature已经开发完毕,之后就是小补小漏,性能优化了. ms ...
- 迷你MVVM框架 avalonjs 0.82发布
迷你MVVM框架 avalonjs 0.82发布 本版本最大的改进是启用全新的parser. parser是用于干什么的?在视图中,我们通过绑定属性实现双向绑定,比如ms-text="fir ...
- 迷你MVVM框架 avalonjs 0.8发布
本版本最重要的特性是引进了AMD规范的模块加载器,亦即原来mass Framework 的并行加载器, 不同之处,它引进了requirejs的xxx!风格的插件机制,比如要延迟到DOM树建完时触发,是 ...
- 迷你MVVM框架 avalonjs 0.99发布
在本版本主要是性能优化,添加一些有用的功能(如回调什么的),离成品阶段不远了. 修正 updateViewModel bug 修正监控数组的set方法 bug 添加data-each-rendered ...
- 迷你MVVM框架 avalonjs 0.9发布
本版本最大的改进是引进了ms-with绑定,现在可轻松遍历对象了. 改进列表如下: 重新使用082的scanNodes方法,因为有关旧式IE下UI渲染锁死的问题已经解决了. 优化each绑定与Coll ...
- 迷你MVVM框架 avalonjs 0.91发布
本版本修了一些BUG与不合理的地方,感谢感谢ztz, 民工精髓, 姚立, qiangtou等人指正. 处理AMD加载 旧式IE下移除script节点内存泄漏的问题 fix firefox 全系列vis ...
- 迷你MVVM框架 avalonjs 0.92发布
本版本最大的改进是引入ms-class的新风格支持,以前的不支持大写类名及多个类名同时操作,新风格支持了.还有对2维监控数组的支持.并着手修复UI框架. 重构 class, hover, active ...
- 迷你MVVM框架 avalonjs 0.93发布
这段时间吸取@limodou, @东灵等人的意见,做了以下改进 重构isArrayLike,提高avalon.each的性能,原来avalon.each是依赖于isArrayLike来判定是循环普通对 ...
- 迷你MVVM框架 avalonjs 0.94发布
本版本主要做了如下改进: 优化ms-if的逻辑,现在描述DOM的顺序是 ms-skip, ms-important, ms-controller, ms-if ... 只要元素存在ms-skip 这个 ...
随机推荐
- MVC EF 修改 封装类 通用泛型方法(二)
修改 这个 方法 如下. 排除 null 值. /// <summary> /// 修改 多数 数据, 个别数据除外, proNames 不写 则是 修改全部 /// </summa ...
- jQuery UI dialog 參数说明
前段时间碰到个问题 jquery UI dialog弹出层 弹出多个层是 比方弹出两个层A和B B层假设显示的数据表格太大,伸到了A层的外面,那伸到A层之外的部分就看不到了,由于B层是在A层上弹出的 ...
- java中HashSet详解
HashSet 的实现 对于 HashSet 而言,它是基于 HashMap 实现的,HashSet 底层采用 HashMap 来保存所有元素,因此 HashSet 的实现比较简单,查看 HashSe ...
- Backup and Recovery Strategies1
2.1.Data Recovery Strategy Determines Backup Strategy 在设计备份策略.如若数据恢复需求和数据恢复战略启动.每种类型的数据恢复需要你采取相应的备份类 ...
- SQL Server 2008 R2中,变表的右键弹出菜单中的“选择前1000行”为“选择所有行”
原文:SQL Server 2008 R2中,变表的右键弹出菜单中的"选择前1000行"为"选择所有行" 从SQL Server 2008开始,微软为了提高查询 ...
- 11g R2RAC Dynamic remastering
In this post, I will demonstrate dynamic remastering of the resources in RAC . In RAC, every data bl ...
- JQuery在Ajax的Post提交中国乱码的解决方案
介绍: 在JQuery的Ajax POST要求,一个要求.中国的背景之中,乱码,如何解决呢? 问题简介: var regid = $('#oregion').combobox('getValue'); ...
- LINUX SCP 远程 文件 复制
首先,以确保直接两个机器IP可以在每个ping通过 然后使用SCP命令从第一台主机向第二台主机复制文件 scp src chiwei@192.168.8.144:/home/chiwei/mydisk ...
- ACE_Message_Block 学习
本文参考http://www.cnblogs.com/TianFang/archive/2006/12/30/607960.html 1. ACE_Message_Block 的length() 返回 ...
- asp.net mvc4
select省市联动选择城市 asp.net mvc4 2014-05-24 16:48 by P.C ++, 159 阅读, 2 评论, 收藏, 编辑 本文在 http://www.cnblogs. ...