/*================================================== Copyright (c) 2013-2016 司徒正美 and other contributors http://www.cnblogs.com/rubylouvre/ https://github.com/RubyLouvre http://weibo.com/jslouvre/ Released under the MIT license avalon.js 1.5.9 built in 2016.11.27 support IE6+ and other browsers ==================================================*/ (function(global, factory) { if (typeof module === "object" && typeof module.exports === "object") { // For CommonJS and CommonJS-like environments where a proper `window` // is present, execute the factory and get avalon. // For environments that do not have a `window` with a `document` // (such as Node.js), expose a factory as module.exports. // This accentuates the need for the creation of a real `window`. // e.g. var avalon = require("avalon")(window); module.exports = global.document ? factory(global, true) : function(w) { if (!w.document) { throw new Error("Avalon requires a window with a document") } return factory(w) } } else { factory(global) } // Pass this if window is not defined yet }(typeof window !== "undefined" ? window : this, function(window, noGlobal){ /********************************************************************* * 全局变量及方法 * **********************************************************************/ var expose = new Date() - 0 //http://stackoverflow.com/questions/7290086/javascript-use-strict-and-nicks-find-global-function var DOC = window.document var head = DOC.getElementsByTagName("head")[0] //HEAD元素 var ifGroup = head.insertBefore(document.createElement("avalon"), head.firstChild) //避免IE6 base标签BUG ifGroup.innerHTML = "X" ifGroup.setAttribute("ms-skip", "1") ifGroup.className = "avalonHide" var rnative = /\[native code\]/ //判定是否原生函数 function log() { if (window.console && kernel.debug) { // http://stackoverflow.com/questions/8785624/how-to-safely-wrap-console-log Function.apply.call(console.log, console, arguments) } } var subscribers = "$" + expose var nullObject = {} //作用类似于noop,只用于代码防御,千万不要在它上面添加属性 var rword = /[^, ]+/g //切割字符串为一个个小块,以空格或豆号分开它们,结合replace实现字符串的forEach var rw20g = /\w+/g var rsvg = /^\[object SVG\w*Element\]$/ var rwindow = /^\[object (?:Window|DOMWindow|global)\]$/ var oproto = Object.prototype var ohasOwn = oproto.hasOwnProperty var serialize = oproto.toString var ap = Array.prototype var aslice = ap.slice var W3C = window.dispatchEvent var root = DOC.documentElement var avalonFragment = DOC.createDocumentFragment() var cinerator = DOC.createElement("div") var class2type = {} "Boolean Number String Function Array Date RegExp Object Error".replace(rword, function (name) { class2type["[object " + name + "]"] = name.toLowerCase() }) var bindingID = 1024 var IEVersion = NaN if (window.VBArray) { IEVersion = document.documentMode || (window.XMLHttpRequest ? 7 : 6) } function noop(){} function scpCompile(array){ return Function.apply(noop, array) } function oneObject(array, val) { if (typeof array === "string") { array = array.match(rword) || [] } var result = {}, value = val !== void 0 ? val : 1 for (var i = 0, n = array.length; i < n; i++) { result[array[i]] = value } return result } //生成UUID http://stackoverflow.com/questions/105034/how-to-create-a-guid-uuid-in-javascript var generateID = function (prefix) { prefix = prefix || "avalon" return String(Math.random() + Math.random()).replace(/\d\.\d{4}/, prefix) } avalon = function (el) { //创建jQuery式的无new 实例化结构 return new avalon.init(el) } /*视浏览器情况采用最快的异步回调*/ avalon.nextTick = new function () {// jshint ignore:line var tickImmediate = window.setImmediate var tickObserver = window.MutationObserver if (tickImmediate) { return tickImmediate.bind(window) } var queue = [] function callback() { var n = queue.length for (var i = 0; i < n; i++) { queue[i]() } queue = queue.slice(n) } if (tickObserver) { var node = document.createTextNode("avalon") new tickObserver(callback).observe(node, {characterData: true})// jshint ignore:line var bool = false return function (fn) { queue.push(fn) bool = !bool node.data = bool } } return function (fn) { setTimeout(fn, 4) } }// jshint ignore:line /********************************************************************* * javascript 底层补丁 * **********************************************************************/ if (!"司徒正美".trim) { var rtrim = /^[\s\uFEFF\xA0]+|[\s\uFEFF\xA0]+$/g String.prototype.trim = function () { return this.replace(rtrim, "") } } var hasDontEnumBug = !({ 'toString': null }).propertyIsEnumerable('toString'), hasProtoEnumBug = (function () { }).propertyIsEnumerable('prototype'), dontEnums = [ "toString", "toLocaleString", "valueOf", "hasOwnProperty", "isPrototypeOf", "propertyIsEnumerable", "constructor" ], dontEnumsLength = dontEnums.length; if (!Object.keys) { Object.keys = function (object) { //ecma262v5 15.2.3.14 var theKeys = [] var skipProto = hasProtoEnumBug && typeof object === "function" if (typeof object === "string" || (object && object.callee)) { for (var i = 0; i < object.length; ++i) { theKeys.push(String(i)) } } else { for (var name in object) { if (!(skipProto && name === "prototype") && ohasOwn.call(object, name)) { theKeys.push(String(name)) } } } if (hasDontEnumBug) { var ctor = object.constructor, skipConstructor = ctor && ctor.prototype === object for (var j = 0; j < dontEnumsLength; j++) { var dontEnum = dontEnums[j] if (!(skipConstructor && dontEnum === "constructor") && ohasOwn.call(object, dontEnum)) { theKeys.push(dontEnum) } } } return theKeys } } if (!Array.isArray) { Array.isArray = function (a) { return serialize.call(a) === "[object Array]" } } if (!noop.bind) { Function.prototype.bind = function (scope) { if (arguments.length < 2 && scope === void 0) return this var fn = this, argv = arguments return function () { var args = [], i for (i = 1; i < argv.length; i++) args.push(argv[i]) for (i = 0; i < arguments.length; i++) args.push(arguments[i]) return fn.apply(scope, args) } } } var __slice = ap.slice try { // Can't be used with DOM elements in IE < 9 __slice.call(document.documentElement) } catch (e) { // Fails in IE < 9 // This will work for genuine arrays, array-like objects, // NamedNodeMap (attributes, entities, notations), // NodeList (e.g., getElementsByTagName), HTMLCollection (e.g., childNodes), // and will not fail on other DOM objects (as do DOM elements in IE < 9) aslice = ap.slice = function (begin, end) { // IE < 9 gets unhappy with an undefined end argument end = (typeof end !== 'undefined') ? end : this.length // For native Array objects, we use the native slice function if (Array.isArray(this)) { return __slice.call(this, begin, end) } // For array like object we handle it ourselves. var i, cloned = [], size, len = this.length // Handle negative value for "begin" var start = begin || 0 start = (start >= 0) ? start : len + start // Handle negative value for "end" var upTo = (end) ? end : len if (end < 0) { upTo = len + end } // Actual expected size of the slice size = upTo - start if (size > 0) { cloned = new Array(size) if (this.charAt) { for (i = 0; i < size; i++) { cloned[i] = this.charAt(start + i) } } else { for (i = 0; i < size; i++) { cloned[i] = this[start + i] } } } return cloned } } function iterator(vars, body, ret) { var fun = 'for(var ' + vars + 'i=0,n = this.length; i < n; i++){' + body.replace('_', '((i in this) && fn.call(scope,this[i],i,this))') + '}' + ret /* jshint ignore:start */ return Function('fn,scope', fun) /* jshint ignore:end */ } if (!/\[native code\]/.test(ap.map)) { var shim = { //定位操作,返回数组中第一个等于给定参数的元素的索引值。 indexOf: function (item, index) { var n = this.length, i = ~~index if (i < 0) i += n for (; i < n; i++) if (this[i] === item) return i return -1 }, //定位操作,同上,不过是从后遍历。 lastIndexOf: function (item, index) { var n = this.length, i = index == null ? n - 1 : index if (i < 0) i = Math.max(0, n + i) for (; i >= 0; i--) if (this[i] === item) return i return -1 }, //迭代操作,将数组的元素挨个儿传入一个函数中执行。Prototype.js的对应名字为each。 forEach: iterator('', '_', ''), //迭代类 在数组中的每个项上运行一个函数,如果此函数的值为真,则此元素作为新数组的元素收集起来,并返回新数组 filter: iterator('r=[],j=0,', 'if(_)r[j++]=this[i]', 'return r'), //收集操作,将数组的元素挨个儿传入一个函数中执行,然后把它们的返回值组成一个新数组返回。Prototype.js的对应名字为collect。 map: iterator('r=[],', 'r[i]=_', 'return r'), //只要数组中有一个元素满足条件(放进给定函数返回true),那么它就返回true。Prototype.js的对应名字为any。 some: iterator('', 'if(_)return true', 'return false'), //只有数组中的元素都满足条件(放进给定函数返回true),它才返回true。Prototype.js的对应名字为all。 every: iterator('', 'if(!_)return false', 'return true') } for (var i in shim) { ap[i] = shim[i] } } /********************************************************************* * avalon的静态方法定义区 * **********************************************************************/ avalon.init = function (el) { this[0] = this.element = el } avalon.fn = avalon.prototype = avalon.init.prototype avalon.type = function (obj) { //取得目标的类型 if (obj == null) { return String(obj) } // 早期的webkit内核浏览器实现了已废弃的ecma262v4标准,可以将正则字面量当作函数使用,因此typeof在判定正则时会返回function return typeof obj === "object" || typeof obj === "function" ? class2type[serialize.call(obj)] || "object" : typeof obj } avalon.isFunction = typeof alert === "object" ? function (fn) { try { return /^\s*\bfunction\b/.test(fn + "") } catch (e) { return false } } : function (fn) { return serialize.call(fn) === "[object Function]" } avalon.isWindow = function (obj) { if (!obj) return false // 利用IE678 window == document为true,document == window竟然为false的神奇特性 // 标准浏览器及IE9,IE10等使用 正则检测 return obj == obj.document && obj.document != obj //jshint ignore:line } function isWindow(obj) { return rwindow.test(serialize.call(obj)) } if (isWindow(window)) { avalon.isWindow = isWindow } var enu, enumerateBUG for (enu in avalon({})) { break } enumerateBUG = enu !== "0" //IE6下为true, 其他为false /*判定是否是一个朴素的javascript对象(Object),不是DOM对象,不是BOM对象,不是自定义类的实例*/ avalon.isPlainObject = function (obj, key) { if (!obj || avalon.type(obj) !== "object" || obj.nodeType || avalon.isWindow(obj)) { return false; } try { //IE内置对象没有constructor if (obj.constructor && !ohasOwn.call(obj, "constructor") && !ohasOwn.call(obj.constructor.prototype, "isPrototypeOf")) { return false; } } catch (e) { //IE8 9会在这里抛错 return false; } if (enumerateBUG) { for (key in obj) { return ohasOwn.call(obj, key) } } for (key in obj) { } return key === void 0 || ohasOwn.call(obj, key) } if (rnative.test(Object.getPrototypeOf)) { avalon.isPlainObject = function (obj) { // 简单的 typeof obj === "object"检测,会致使用isPlainObject(window)在opera下通不过 return serialize.call(obj) === "[object Object]" && Object.getPrototypeOf(obj) === oproto } } //与jQuery.extend方法,可用于浅拷贝,深拷贝 avalon.mix = avalon.fn.mix = function () { var options, name, src, copy, copyIsArray, clone, target = arguments[0] || {}, i = 1, length = arguments.length, deep = false // 如果第一个参数为布尔,判定是否深拷贝 if (typeof target === "boolean") { deep = target target = arguments[1] || {} i++ } //确保接受方为一个复杂的数据类型 if (typeof target !== "object" && !avalon.isFunction(target)) { target = {} } //如果只有一个参数,那么新成员添加于mix所在的对象上 if (i === length) { target = this i-- } for (; i < length; i++) { //只处理非空参数 if ((options = arguments[i]) != null) { for (name in options) { src = target[name] try { copy = options[name] //当options为VBS对象时报错 } catch (e) { continue } // 防止环引用 if (target === copy) { continue } if (deep && copy && (avalon.isPlainObject(copy) || (copyIsArray = Array.isArray(copy)))) { if (copyIsArray) { copyIsArray = false clone = src && Array.isArray(src) ? src : [] } else { clone = src && avalon.isPlainObject(src) ? src : {} } target[name] = avalon.mix(deep, clone, copy) } else if (copy !== void 0) { target[name] = copy } } } } return target } function _number(a, len) { //用于模拟slice, splice的效果 a = Math.floor(a) || 0 return a < 0 ? Math.max(len + a, 0) : Math.min(a, len); } avalon.mix({ rword: rword, subscribers: subscribers, version: 1.59, ui: {}, log: log, slice: function (nodes, start, end) { return aslice.call(nodes, start, end) } , noop: noop, /*如果不用Error对象封装一下,str在控制台下可能会乱码*/ error: function (str, e) { throw (e || Error)(str) }, /*将一个以空格或逗号隔开的字符串或数组,转换成一个键值都为1的对象*/ oneObject: oneObject, /* avalon.range(10) => [0, 1, 2, 3, 4, 5, 6, 7, 8, 9] avalon.range(1, 11) => [1, 2, 3, 4, 5, 6, 7, 8, 9, 10] avalon.range(0, 30, 5) => [0, 5, 10, 15, 20, 25] avalon.range(0, -10, -1) => [0, -1, -2, -3, -4, -5, -6, -7, -8, -9] avalon.range(0) => []*/ range: function (start, end, step) { // 用于生成整数数组 step || (step = 1) if (end == null) { end = start || 0 start = 0 } var index = -1, length = Math.max(0, Math.ceil((end - start) / step)), result = new Array(length) while (++index < length) { result[index] = start start += step } return result }, eventHooks: {}, /*绑定事件*/ bind: function (el, type, fn, phase) { var hooks = avalon.eventHooks var hook = hooks[type] if (typeof hook === "object") { type = hook.type || type phase = hook.phase || !!phase fn = hook.fix ? hook.fix(el, fn) : fn } var callback = W3C ? fn : function (e) { fn.call(el, fixEvent(e)); } if (W3C) { el.addEventListener(type, callback, phase) } else { el.attachEvent("on" + type, callback) } return callback }, /*卸载事件*/ unbind: function (el, type, fn, phase) { var hooks = avalon.eventHooks var hook = hooks[type] var callback = fn || noop if (typeof hook === "object") { type = hook.type || type phase = hook.phase || !!phase } if (W3C) { el.removeEventListener(type, callback, phase) } else { el.detachEvent("on" + type, callback) } }, /*读写删除元素节点的样式*/ css: function (node, name, value) { if (node instanceof avalon) { node = node[0] } var prop = /[_-]/.test(name) ? camelize(name) : name, fn name = avalon.cssName(prop) || prop if (value === void 0 || typeof value === "boolean") { //获取样式 fn = cssHooks[prop + ":get"] || cssHooks["@:get"] if (name === "background") { name = "backgroundColor" } var val = fn(node, name) return value === true ? parseFloat(val) || 0 : val } else if (value === "") { //请除样式 node.style[name] = "" } else { //设置样式 if (value == null || value !== value) { return } if (isFinite(value) && !avalon.cssNumber[prop]) { value += "px" } fn = cssHooks[prop + ":set"] || cssHooks["@:set"] fn(node, name, value) } }, /*遍历数组与对象,回调的第一个参数为索引或键名,第二个或元素或键值*/ each: function (obj, fn) { if (obj) { //排除null, undefined var i = 0 if (isArrayLike(obj)) { for (var n = obj.length; i < n; i++) { if (fn(i, obj[i]) === false) break } } else { for (i in obj) { if (obj.hasOwnProperty(i) && fn(i, obj[i]) === false) { break } } } } }, //收集元素的data-{{prefix}}-*属性,并转换为对象 getWidgetData: function (elem, prefix) { var raw = avalon(elem).data() var result = {} for (var i in raw) { if (i.indexOf(prefix) === 0) { result[i.replace(prefix, "").replace(/\w/, function (a) { return a.toLowerCase() })] = raw[i] } } return result }, Array: { /*只有当前数组不存在此元素时只添加它*/ ensure: function (target, item) { if (target.indexOf(item) === -1) { return target.push(item) } }, /*移除数组中指定位置的元素,返回布尔表示成功与否*/ removeAt: function (target, index) { return !!target.splice(index, 1).length }, /*移除数组中第一个匹配传参的那个元素,返回布尔表示成功与否*/ remove: function (target, item) { var index = target.indexOf(item) if (~index) return avalon.Array.removeAt(target, index) return false } } }) var bindingHandlers = avalon.bindingHandlers = {} var bindingExecutors = avalon.bindingExecutors = {} var directives = avalon.directives = {} avalon.directive = function (name, obj) { bindingHandlers[name] = obj.init = (obj.init || noop) bindingExecutors[name] = obj.update = (obj.update || noop) return directives[name] = obj } /*判定是否类数组,如节点集合,纯数组,arguments与拥有非负整数的length属性的纯JS对象*/ function isArrayLike(obj) { if (!obj) return false var n = obj.length if (n === (n >>> 0)) { //检测length属性是否为非负整数 var type = serialize.call(obj).slice(8, -1) if (/(?:regexp|string|function|window|global)$/i.test(type)) return false if (type === "Array") return true try { if ({}.propertyIsEnumerable.call(obj, "length") === false) { //如果是原生对象 return /^\s?function/.test(obj.item || obj.callee) } return true } catch (e) { //IE的NodeList直接抛错 return !obj.window //IE6-8 window } } return false } // https://github.com/rsms/js-lru var Cache = new function() {// jshint ignore:line function LRU(maxLength) { this.size = 0 this.limit = maxLength this.head = this.tail = void 0 this._keymap = {} } var p = LRU.prototype p.put = function(key, value) { var entry = { key: key, value: value } this._keymap[key] = entry if (this.tail) { this.tail.newer = entry entry.older = this.tail } else { this.head = entry } this.tail = entry if (this.size === this.limit) { this.shift() } else { this.size++ } return value } p.shift = function() { var entry = this.head if (entry) { this.head = this.head.newer this.head.older = entry.newer = entry.older = this._keymap[entry.key] = void 0 delete this._keymap[entry.key] //#1029 } } p.get = function(key) { var entry = this._keymap[key] if (entry === void 0) return if (entry === this.tail) { return entry.value } // HEAD--------------TAIL // <.older .newer> // <--- add direction -- // A B C E if (entry.newer) { if (entry === this.head) { this.head = entry.newer } entry.newer.older = entry.older // C <-- E. } if (entry.older) { entry.older.newer = entry.newer // C. --> E } entry.newer = void 0 // D --x entry.older = this.tail // D. --> E if (this.tail) { this.tail.newer = entry // E. <-- D } this.tail = entry return entry.value } return LRU }// jshint ignore:line /********************************************************************* * DOM 底层补丁 * **********************************************************************/ avalon.contains = function(root, el) { try { //IE6-8,游离于DOM树外的文本节点,访问parentNode有时会抛错 while ((el = el.parentNode)) if (el === root) return true return false } catch (e) { return false } } //IE6-11的文档对象没有contains if (!DOC.contains) { DOC.contains = function (b) { return avalon.contains(DOC, b) } } function outerHTML() { return new XMLSerializer().serializeToString(this) } if (window.SVGElement) { //safari5+是把contains方法放在Element.prototype上而不是Node.prototype if (!DOC.createTextNode("x").contains) { Node.prototype.contains = function (arg) {//IE6-8没有Node对象 return !!(this.compareDocumentPosition(arg) & 16) } } var svgns = "http://www.w3.org/2000/svg" var svg = DOC.createElementNS(svgns, "svg") svg.innerHTML = '' if (!rsvg.test(svg.firstChild)) { // #409 function enumerateNode(node, targetNode) {// jshint ignore:line if (node && node.childNodes) { var nodes = node.childNodes for (var i = 0, el; el = nodes[i++]; ) { if (el.tagName) { var svg = DOC.createElementNS(svgns, el.tagName.toLowerCase()) ap.forEach.call(el.attributes, function (attr) { svg.setAttribute(attr.name, attr.value) //复制属性 })// jshint ignore:line // 递归处理子节点 enumerateNode(el, svg) targetNode.appendChild(svg) } } } } Object.defineProperties(SVGElement.prototype, { "outerHTML": {//IE9-11,firefox不支持SVG元素的innerHTML,outerHTML属性 enumerable: true, configurable: true, get: outerHTML, set: function (html) { var tagName = this.tagName.toLowerCase(), par = this.parentNode, frag = avalon.parseHTML(html) // 操作的svg,直接插入 if (tagName === "svg") { par.insertBefore(frag, this) // svg节点的子节点类似 } else { var newFrag = DOC.createDocumentFragment() enumerateNode(frag, newFrag) par.insertBefore(newFrag, this) } par.removeChild(this) } }, "innerHTML": { enumerable: true, configurable: true, get: function () { var s = this.outerHTML var ropen = new RegExp("<" + this.nodeName + '\\b(?:(["\'])[^"]*?(\\1)|[^>])*>', "i") var rclose = new RegExp("<\/" + this.nodeName + ">$", "i") return s.replace(ropen, "").replace(rclose, "") }, set: function (html) { if (avalon.clearHTML) { avalon.clearHTML(this) var frag = avalon.parseHTML(html) enumerateNode(frag, this) } } } }) } } if (!root.outerHTML && window.HTMLElement) { //firefox 到11时才有outerHTML HTMLElement.prototype.__defineGetter__("outerHTML", outerHTML); } //============================= event binding ======================= var rmouseEvent = /^(?:mouse|contextmenu|drag)|click/ function fixEvent(event) { var ret = {} for (var i in event) { ret[i] = event[i] } var target = ret.target = event.srcElement if (event.type.indexOf("key") === 0) { ret.which = event.charCode != null ? event.charCode : event.keyCode } else if (rmouseEvent.test(event.type)) { var doc = target.ownerDocument || DOC var box = doc.compatMode === "BackCompat" ? doc.body : doc.documentElement ret.pageX = event.clientX + (box.scrollLeft >> 0) - (box.clientLeft >> 0) ret.pageY = event.clientY + (box.scrollTop >> 0) - (box.clientTop >> 0) ret.wheelDeltaY = ret.wheelDelta ret.wheelDeltaX = 0 } ret.timeStamp = new Date() - 0 ret.originalEvent = event ret.preventDefault = function () { //阻止默认行为 event.returnValue = false } ret.stopPropagation = function () { //阻止事件在DOM树中的传播 event.cancelBubble = true } return ret } var eventHooks = avalon.eventHooks //针对firefox, chrome修正mouseenter, mouseleave if (!("onmouseenter" in root)) { avalon.each({ mouseenter: "mouseover", mouseleave: "mouseout" }, function (origType, fixType) { eventHooks[origType] = { type: fixType, fix: function (elem, fn) { return function (e) { var t = e.relatedTarget if (!t || (t !== elem && !(elem.compareDocumentPosition(t) & 16))) { delete e.type e.type = origType return fn.call(elem, e) } } } } }) } //针对IE9+, w3c修正animationend avalon.each({ AnimationEvent: "animationend", WebKitAnimationEvent: "webkitAnimationEnd" }, function (construct, fixType) { if (window[construct] && !eventHooks.animationend) { eventHooks.animationend = { type: fixType } } }) //针对IE6-8修正input if (!("oninput" in DOC.createElement("input"))) { eventHooks.input = { type: "propertychange", fix: function (elem, fn) { return function (e) { if (e.propertyName === "value") { e.type = "input" return fn.call(elem, e) } } } } } if (DOC.onmousewheel === void 0) { /* IE6-11 chrome mousewheel wheelDetla 下 -120 上 120 firefox DOMMouseScroll detail 下3 上-3 firefox wheel detlaY 下3 上-3 IE9-11 wheel deltaY 下40 上-40 chrome wheel deltaY 下100 上-100 */ var fixWheelType = DOC.onwheel !== void 0 ? "wheel" : "DOMMouseScroll" var fixWheelDelta = fixWheelType === "wheel" ? "deltaY" : "detail" eventHooks.mousewheel = { type: fixWheelType, fix: function (elem, fn) { return function (e) { e.wheelDeltaY = e.wheelDelta = e[fixWheelDelta] > 0 ? -120 : 120 e.wheelDeltaX = 0 if (Object.defineProperty) { Object.defineProperty(e, "type", { value: "mousewheel" }) } fn.call(elem, e) } } } } /********************************************************************* * 配置系统 * **********************************************************************/ function kernel(settings) { for (var p in settings) { if (!ohasOwn.call(settings, p)) continue var val = settings[p] if (typeof kernel.plugins[p] === "function") { kernel.plugins[p](val) } else if (typeof kernel[p] === "object") { avalon.mix(kernel[p], val) } else { kernel[p] = val } } return this } avalon.config = kernel var openTag, closeTag, rexpr, rexprg, rbind, rregexp = /[-.*+?^${}()|[\]\/\\]/g function escapeRegExp(target) { //http://stevenlevithan.com/regex/xregexp/ //将字符串安全格式化为正则表达式的源码 return (target + "").replace(rregexp, "\\$&") } var plugins = { interpolate: function (array) { openTag = array[0] closeTag = array[1] if (openTag === closeTag) { throw new SyntaxError("openTag!==closeTag") var test = openTag + "test" + closeTag cinerator.innerHTML = test if (cinerator.innerHTML !== test && cinerator.innerHTML.indexOf("<") > -1) { throw new SyntaxError("此定界符不合法") } cinerator.innerHTML = "" } kernel.openTag = openTag kernel.closeTag = closeTag var o = escapeRegExp(openTag), c = escapeRegExp(closeTag) rexpr = new RegExp(o + "([\\s\\S]*)" + c) rexprg = new RegExp(o + "([\\s\\S]*)" + c, "g") rbind = new RegExp(o + "[\\s\\S]*" + c + "|\\sms-") } } kernel.plugins = plugins kernel.plugins['interpolate'](["{{", "}}"]) kernel.async = true kernel.debug = true kernel.paths = {} kernel.shim = {} kernel.maxRepeatSize = 100 function $watch(expr, binding) { var $events = this.$events || (this.$events = {}), queue = $events[expr] || ($events[expr] = []) if (typeof binding === "function") { var backup = binding backup.uuid = "_"+ (++bindingID) binding = { element: root, type: "user-watcher", handler: noop, vmodels: [this], expr: expr, uuid: backup.uuid } binding.wildcard = /\*/.test(expr) } if (!binding.update) { if (/\w\.*\B/.test(expr) || expr === "*") { binding.getter = noop var host = this binding.update = function () { var args = this.fireArgs || [] if (args[2]) binding.handler.apply(host, args) delete this.fireArgs } queue.sync = true avalon.Array.ensure(queue, binding) } else { avalon.injectBinding(binding) } if (backup) { binding.handler = backup } } else if (!binding.oneTime) { avalon.Array.ensure(queue, binding) } return function () { binding.update = binding.getter = binding.handler = noop binding.element = DOC.createElement("a") } } function $emit(key, args) { var event = this.$events if (event && event[key]) { if (args) { args[2] = key } var arr = event[key] notifySubscribers(arr, args) if (args && event["*"] && !/\./.test(key)) { for (var sub, k = 0; sub = event["*"][k++]; ) { try { sub.handler.apply(this, args) } catch (e) { } } } var parent = this.$up if (parent) { if (this.$pathname) { $emit.call(parent, this.$pathname + "." + key, args)//以确切的值往上冒泡 } $emit.call(parent, "*." + key, args)//以模糊的值往上冒泡 } } else { parent = this.$up if (this.$ups) { for (var i in this.$ups) { $emit.call(this.$ups[i], i + "." + key, args)//以确切的值往上冒泡 } return } if (parent) { var p = this.$pathname if (p === "") p = "*" var path = p + "." + key arr = path.split(".") if (arr.indexOf("*") === -1) { $emit.call(parent, path, args)//以确切的值往上冒泡 arr[1] = "*" $emit.call(parent, arr.join("."), args)//以模糊的值往上冒泡 } else { $emit.call(parent, path, args)//以确切的值往上冒泡 } } } } function collectDependency(el, key) { do { if (el.$watch) { var e = el.$events || (el.$events = {}) var array = e[key] || (e[key] = []) dependencyDetection.collectDependency(array) return } el = el.$up if (el) { key = el.$pathname + "." + key } else { break } } while (true) } function notifySubscribers(subs, args) { if (!subs) return if (new Date() - beginTime > 444 && typeof subs[0] === "object") { rejectDisposeQueue() } var users = [], renders = [] for (var i = 0, sub; sub = subs[i++]; ) { if (sub.type === "user-watcher") { users.push(sub) } else { renders.push(sub) } } if (kernel.async) { buffer.render()//1 for (i = 0; sub = renders[i++]; ) { if (sub.update) { sub.uuid = sub.uuid || "_"+(++bindingID) var uuid = sub.uuid if (!buffer.queue[uuid]) { buffer.queue[uuid] = "__" buffer.queue.push(sub) } } } } else { for (i = 0; sub = renders[i++]; ) { if (sub.update) { sub.update()//最小化刷新DOM树 } } } for (i = 0; sub = users[i++]; ) { if (args && args[2] === sub.expr || sub.wildcard) { sub.fireArgs = args } sub.update() } } //avalon最核心的方法的两个方法之一(另一个是avalon.scan),返回一个ViewModel(VM) var VMODELS = avalon.vmodels = {} //所有vmodel都储存在这里 avalon.define = function (source) { var $id = source.$id if (!$id) { log("warning: vm必须指定$id") } var vmodel = modelFactory(source) vmodel.$id = $id return VMODELS[$id] = vmodel } //一些不需要被监听的属性 var $$skipArray = oneObject("$id,$watch,$fire,$events,$model,$skipArray,$active,$pathname,$up,$track,$accessors,$ups") var defineProperty = Object.defineProperty var canHideOwn = true //如果浏览器不支持ecma262v5的Object.defineProperties或者存在BUG,比如IE8 //标准浏览器使用__defineGetter__, __defineSetter__实现 try { defineProperty({}, "_", { value: "x" }) var defineProperties = Object.defineProperties } catch (e) { canHideOwn = false } function modelFactory(source, options) { options = options || {} options.watch = true return observeObject(source, options) } //监听对象属性值的变化(注意,数组元素不是数组的属性),通过对劫持当前对象的访问器实现 //监听对象或数组的结构变化, 对对象的键值对进行增删重排, 或对数组的进行增删重排,都属于这范畴 // 通过比较前后代理VM顺序实现 function Component() { } function observeObject(source, options) { if (!source || (source.$id && source.$accessors) || (source.nodeName && source.nodeType > 0)) { return source } //source为原对象,不能是元素节点或null //options,可选,配置对象,里面有old, force, watch这三个属性 options = options || nullObject var force = options.force || nullObject var old = options.old var oldAccessors = old && old.$accessors || nullObject var $vmodel = new Component() //要返回的对象, 它在IE6-8下可能被偷龙转凤 var accessors = {} //监控属性 var hasOwn = {} var skip = [] var simple = [] var $skipArray = {} if (source.$skipArray) { $skipArray = oneObject(source.$skipArray) delete source.$skipArray } //处理计算属性 var computed = source.$computed if (computed) { delete source.$computed for (var name in computed) { hasOwn[name] = true; (function (key, value) { var old accessors[key] = { get: function () { return old = value.get.call(this) }, set: function (x) { if (typeof value.set === "function") { var older = old value.set.call(this, x) var newer = this[key] if (this.$fire && (newer !== older)) { this.$fire(key, newer, older) } } }, enumerable: true, configurable: true } })(name, computed[name])// jshint ignore:line } } for (name in source) { var value = source[name] if (!$$skipArray[name]) hasOwn[name] = true if (typeof value === "function" || (value && value.nodeName && value.nodeType > 0) || (!force[name] && (name.charAt(0) === "$" || $$skipArray[name] || $skipArray[name]))) { skip.push(name) } else if (isComputed(value)) { log("warning:计算属性建议放在$computed对象中统一定义"); (function (key, value) { var old accessors[key] = { get: function () { return old = value.get.call(this) }, set: function (x) { if (typeof value.set === "function") { var older = old value.set.call(this, x) var newer = this[key] if (this.$fire && (newer !== older)) { this.$fire(key, newer, older) } } }, enumerable: true, configurable: true } })(name, value)// jshint ignore:line } else { simple.push(name) if (oldAccessors[name]) { accessors[name] = oldAccessors[name] } else { accessors[name] = makeGetSet(name, value) } } } accessors["$model"] = $modelDescriptor $vmodel = defineProperties($vmodel, accessors, source) function trackBy(name) { return hasOwn[name] === true } skip.forEach(function (name) { $vmodel[name] = source[name] }) /* jshint ignore:start */ hideProperty($vmodel, "$ups", null) hideProperty($vmodel, "$id", "anonymous") hideProperty($vmodel, "$up", old ? old.$up : null) hideProperty($vmodel, "$track", Object.keys(hasOwn)) hideProperty($vmodel, "$active", false) hideProperty($vmodel, "$pathname", old ? old.$pathname : "") hideProperty($vmodel, "$accessors", accessors) hideProperty($vmodel, "hasOwnProperty", trackBy) if (options.watch) { hideProperty($vmodel, "$watch", function () { return $watch.apply($vmodel, arguments) }) hideProperty($vmodel, "$fire", function (path, a) { if(path.indexOf("all!") === 0 ){ var ee = path.slice(4) for(var i in avalon.vmodels){ var v = avalon.vmodels[i] v.$fire && v.$fire.apply(v, [ee, a]) } }else{ $emit.call($vmodel, path, [a]) } }) } /* jshint ignore:end */ //必须设置了$active,$events simple.forEach(function (name) { var oldVal = old && old[name] var val = $vmodel[name] = source[name] if (val && typeof val === "object") { val.$up = $vmodel val.$pathname = name } $emit.call($vmodel, name, [val,oldVal]) }) for (name in computed) { value = $vmodel[name] } $vmodel.$active = true return $vmodel } /* 新的VM拥有如下私有属性 $id: vm.id $events: 放置$watch回调与绑定对象 $watch: 增强版$watch $fire: 触发$watch回调 $track:一个数组,里面包含用户定义的所有键名 $active:boolean,false时防止依赖收集 $model:返回一个纯净的JS对象 $accessors:放置所有读写器的数据描述对象 $up:返回其上级对象 $pathname:返回此对象在上级对象的名字,注意,数组元素的$pathname为空字符串 ============================= $skipArray:用于指定不可监听的属性,但VM生成是没有此属性的 */ function isComputed(val) {//speed up! if (val && typeof val === "object") { for (var i in val) { if (i !== "get" && i !== "set") { return false } } return typeof val.get === "function" } } function makeGetSet(key, value) { var childVm value = NaN return { get: function () { if (this.$active) { collectDependency(this, key) } return value }, set: function (newVal) { if (value === newVal) return var oldValue = value childVm = observe(newVal, value) if (childVm) { value = childVm } else { childVm = void 0 value = newVal } if (Object(childVm) === childVm) { childVm.$pathname = key childVm.$up = this } if (this.$active) { $emit.call(this, key, [value, oldValue]) } }, enumerable: true, configurable: true } } function observe(obj, old, hasReturn, watch) { if (Array.isArray(obj)) { return observeArray(obj, old, watch) } else if (avalon.isPlainObject(obj)) { if (old && typeof old === 'object') { var keys = getKeys(obj) var keys2 = getKeys(old) if (keys.join(";") === keys2.join(";")) { for (var i in obj) { if (obj.hasOwnProperty(i)) { old[i] = obj[i] } } return old } old.$active = false } return observeObject(obj, { old: old, watch: watch }) } if (hasReturn) { return obj } } var getKeys = rnative.test(Object.key) ? Object.key : function (a) { var ret = [] for (var i in a) { if (a.hasOwnProperty(i) && !$$skipArray[i]) { ret.push(i) } } return ret } function observeArray(array, old, watch) { if (old && old.splice) { var args = [0, old.length].concat(array) old.splice.apply(old, args) return old } else { for (var i in newProto) { array[i] = newProto[i] } hideProperty(array, "$up", null) hideProperty(array, "$pathname", "") hideProperty(array, "$track", createTrack(array.length)) array._ = observeObject({ length: NaN }, { watch: true }) array._.length = array.length array._.$watch("length", function (a, b) { $emit.call(array.$up, array.$pathname + ".length", [a, b]) }) if (watch) { hideProperty(array, "$watch", function () { return $watch.apply(array, arguments) }) } if (W3C) { Object.defineProperty(array, "$model", $modelDescriptor) } else { array.$model = toJson(array) } for (var j = 0, n = array.length; j < n; j++) { var el = array[j] = observe(array[j], 0, 1, 1) if (Object(el) === el) {//#1077 el.$up = array } } return array } } function hideProperty(host, name, value) { if (canHideOwn) { Object.defineProperty(host, name, { value: value, writable: true, enumerable: false, configurable: true }) } else { host[name] = value } } function toJson(val) { var xtype = avalon.type(val) if (xtype === "array") { var array = [] for (var i = 0; i < val.length; i++) { array[i] = toJson(val[i]) } return array } else if (xtype === "object") { var obj = {} for (i in val) { if(i === "__proxy__" || i === "__data__" || i === "__const__") continue if (val.hasOwnProperty(i)) { var value = val[i] obj[i] = value && value.nodeType ? value :toJson(value) } } return obj } return val } var $modelDescriptor = { get: function () { return toJson(this) }, set: noop, enumerable: false, configurable: true } //===================修复浏览器对Object.defineProperties的支持================= if (!canHideOwn) { if ("__defineGetter__" in avalon) { defineProperty = function (obj, prop, desc) { if ('value' in desc) { obj[prop] = desc.value } if ("get" in desc) { obj.__defineGetter__(prop, desc.get) } if ('set' in desc) { obj.__defineSetter__(prop, desc.set) } return obj } defineProperties = function (obj, descs) { for (var prop in descs) { if (descs.hasOwnProperty(prop)) { defineProperty(obj, prop, descs[prop]) } } return obj } } if (IEVersion) { var VBClassPool = {} window.execScript([// jshint ignore:line "Function parseVB(code)", "\tExecuteGlobal(code)", "End Function" //转换一段文本为VB代码 ].join("\n"), "VBScript") function VBMediator(instance, accessors, name, value) {// jshint ignore:line var accessor = accessors[name] if (arguments.length === 4) { accessor.set.call(instance, value) } else { return accessor.get.call(instance) } } defineProperties = function (name, accessors, properties) { // jshint ignore:line var buffer = [] buffer.push( "\r\n\tPrivate [__data__], [__proxy__]", "\tPublic Default Function [__const__](d" + expose + ", p" + expose + ")", "\t\tSet [__data__] = d" + expose + ": set [__proxy__] = p" + expose, "\t\tSet [__const__] = Me", //链式调用 "\tEnd Function") //添加普通属性,因为VBScript对象不能像JS那样随意增删属性,必须在这里预先定义好 var uniq = {} //添加访问器属性 for (name in accessors) { uniq[name] = true buffer.push( //由于不知对方会传入什么,因此set, let都用上 "\tPublic Property Let [" + name + "](val" + expose + ")", //setter "\t\tCall [__proxy__](Me,[__data__], \"" + name + "\", val" + expose + ")", "\tEnd Property", "\tPublic Property Set [" + name + "](val" + expose + ")", //setter "\t\tCall [__proxy__](Me,[__data__], \"" + name + "\", val" + expose + ")", "\tEnd Property", "\tPublic Property Get [" + name + "]", //getter "\tOn Error Resume Next", //必须优先使用set语句,否则它会误将数组当字符串返回 "\t\tSet[" + name + "] = [__proxy__](Me,[__data__],\"" + name + "\")", "\tIf Err.Number <> 0 Then", "\t\t[" + name + "] = [__proxy__](Me,[__data__],\"" + name + "\")", "\tEnd If", "\tOn Error Goto 0", "\tEnd Property") } for (name in properties) { if (uniq[name] !== true) { uniq[name] = true buffer.push("\tPublic [" + name + "]") } } for (name in $$skipArray) { if (uniq[name] !== true) { uniq[name] = true buffer.push("\tPublic [" + name + "]") } } buffer.push("\tPublic [" + 'hasOwnProperty' + "]") buffer.push("End Class") var body = buffer.join("\r\n") var className = VBClassPool[body] if (!className) { className = generateID("VBClass") window.parseVB("Class " + className + body) window.parseVB([ "Function " + className + "Factory(a, b)", //创建实例并传入两个关键的参数 "\tDim o", "\tSet o = (New " + className + ")(a, b)", "\tSet " + className + "Factory = o", "End Function" ].join("\r\n")) VBClassPool[body] = className } var ret = window[className + "Factory"](accessors, VBMediator) //得到其产品 return ret //得到其产品 } } } /********************************************************************* * 监控数组(与ms-each, ms-repeat配合使用) * **********************************************************************/ var arrayMethods = ['push', 'pop', 'shift', 'unshift', 'splice'] var arrayProto = Array.prototype var newProto = { notify: function () { $emit.call(this.$up, this.$pathname) }, set: function (index, val) { if (((index >>> 0) === index) && this[index] !== val) { if (index > this.length) { throw Error(index + "set方法的第一个参数不能大于原数组长度") } $emit.call(this.$up, this.$pathname + ".*", [val, this[index]]) this.splice(index, 1, val) } }, contains: function (el) { //判定是否包含 return this.indexOf(el) !== -1 }, ensure: function (el) { if (!this.contains(el)) { //只有不存在才push this.push(el) } return this }, pushArray: function (arr) { return this.push.apply(this, arr) }, remove: function (el) { //移除第一个等于给定值的元素 return this.removeAt(this.indexOf(el)) }, removeAt: function (index) { //移除指定索引上的元素 if ((index >>> 0) === index) { return this.splice(index, 1) } return [] }, size: function () { //取得数组长度,这个函数可以同步视图,length不能 return this._.length }, removeAll: function (all) { //移除N个元素 var eliminate = Array.isArray(all) ? function(el) { return all.indexOf(el) !== -1 } : typeof all === 'function' ? all : false if (eliminate) { for (var i = this.length - 1; i >= 0; i--) { if (eliminate(this[i], i)) { _splice.call(this.$track, i, 1) _splice.call(this, i, 1) } } } else { _splice.call(this.$track, 0, this.length) _splice.call(this, 0, this.length) } if (!W3C) { this.$model = toJson(this) } this.notify() this._.length = this.length }, clear: function () { this.removeAll() return this } } var _splice = arrayProto.splice arrayMethods.forEach(function (method) { var original = arrayProto[method] newProto[method] = function () { // 继续尝试劫持数组元素的属性 var args = [] for (var i = 0, n = arguments.length; i < n; i++) { args[i] = observe(arguments[i], 0, 1, 1) } var result = original.apply(this, args) addTrack(this.$track, method, args) if (!W3C) { this.$model = toJson(this) } this.notify() this._.length = this.length return result } }) "sort,reverse".replace(rword, function (method) { newProto[method] = function () { var oldArray = this.concat() //保持原来状态的旧数组 var newArray = this var mask = Math.random() var indexes = [] var hasSort = false arrayProto[method].apply(newArray, arguments) //排序 for (var i = 0, n = oldArray.length; i < n; i++) { var neo = newArray[i] var old = oldArray[i] if (neo === old) { indexes.push(i) } else { var index = oldArray.indexOf(neo) indexes.push(index)//得到新数组的每个元素在旧数组对应的位置 oldArray[index] = mask //屏蔽已经找过的元素 hasSort = true } } if (hasSort) { sortByIndex(this.$track, indexes) if (!W3C) { this.$model = toJson(this) } this.notify() } return this } }) function sortByIndex(array, indexes) { var map = {}; for (var i = 0, n = indexes.length; i < n; i++) { map[i] = array[i] var j = indexes[i] if (j in map) { array[i] = map[j] delete map[j] } else { array[i] = array[j] } } } function createTrack(n) { var ret = [] for (var i = 0; i < n; i++) { ret[i] = generateID("$proxy$each") } return ret } function addTrack(track, method, args) { switch (method) { case 'push': case 'unshift': args = createTrack(args.length) break case 'splice': if (args.length > 2) { // 0, 5, a, b, c --> 0, 2, 0 // 0, 5, a, b, c, d, e, f, g--> 0, 0, 3 var del = args[1] var add = args.length - 2 // args = [args[0], Math.max(del - add, 0)].concat(createTrack(Math.max(add - del, 0))) args = [args[0], args[1]].concat(createTrack(args.length - 2)) } break } Array.prototype[method].apply(track, args) } /********************************************************************* * 依赖调度系统 * **********************************************************************/ //检测两个对象间的依赖关系 var dependencyDetection = (function () { var outerFrames = [] var currentFrame return { begin: function (binding) { //accessorObject为一个拥有callback的对象 outerFrames.push(currentFrame) currentFrame = binding }, end: function () { currentFrame = outerFrames.pop() }, collectDependency: function (array) { if (currentFrame) { //被dependencyDetection.begin调用 currentFrame.callback(array) } } }; })() //将绑定对象注入到其依赖项的订阅数组中 var roneval = /^on$/ function returnRandom() { return new Date() - 0 } avalon.injectBinding = function (binding) { binding.handler = binding.handler || directives[binding.type].update || noop binding.update = function () { var begin = false if (!binding.getter) { begin = true dependencyDetection.begin({ callback: function (array) { injectDependency(array, binding) } }) binding.getter = parseExpr(binding.expr, binding.vmodels, binding) binding.observers.forEach(function (a) { a.v.$watch(a.p, binding) }) delete binding.observers } try { var args = binding.fireArgs, a, b delete binding.fireArgs if (!args) { if (binding.type === "on") { a = binding.getter + "" } else { try { a = binding.getter.apply(0, binding.args) } catch(e) { a = null avalon.log('execute expr : ' + binding.expr + ' Error. ' + e) } } } else { a = args[0] b = args[1] } b = typeof b === "undefined" ? binding.oldValue : b if (binding._filters) { a = filters.$filter.apply(0, [a].concat(binding._filters)) } if (binding.signature) { var xtype = avalon.type(a) if (xtype !== "array" && xtype !== "object") { throw Error("warning:" + binding.expr + "只能是对象或数组") } binding.xtype = xtype var vtrack = getProxyIds(binding.proxies || [], xtype) var mtrack = a.$track || (xtype === "array" ? createTrack(a.length) : Object.keys(a)) binding.track = mtrack if (vtrack !== mtrack.join(";")) { binding.handler(a, b) binding.oldValue = 1 } } else if (Array.isArray(a) ? a.length !== (b && b.length) : false) { binding.handler(a, b) binding.oldValue = a.concat() } else if (!("oldValue" in binding) || a !== b) { binding.handler(a, b) binding.oldValue = Array.isArray(a) ? a.concat() : a } } catch (e) { delete binding.getter log("warning:exception throwed in [avalon.injectBinding] ", e) var node = binding.element if (node && node.nodeType === 3) { node.nodeValue = openTag + (binding.oneTime ? "::" : "") + binding.expr + closeTag } } finally { begin && dependencyDetection.end() } } binding.update() } //将依赖项(比它高层的访问器或构建视图刷新函数的绑定对象)注入到订阅者数组 function injectDependency(list, binding) { if (binding.oneTime) return if (list && avalon.Array.ensure(list, binding) && binding.element) { injectDisposeQueue(binding, list) if (new Date() - beginTime > 444) { rejectDisposeQueue() } } } function getProxyIds(a, isArray) { var ret = [] for (var i = 0, el; el = a[i++]; ) { ret.push(isArray ? el.$id : el.$key) } return ret.join(";") } /********************************************************************* * 定时GC回收机制 (基于1.6基于频率的GC) * **********************************************************************/ var disposeQueue = avalon.$$subscribers = [] var beginTime = new Date() //添加到回收列队中 function injectDisposeQueue(data, list) { data.list = list data.i = ~~data.i if (!data.uuid) { data.uuid = "_" + (++bindingID) } if (!disposeQueue[data.uuid]) { disposeQueue[data.uuid] = "__" disposeQueue.push(data) } } var lastGCIndex = 0 function rejectDisposeQueue(data) { var i = lastGCIndex || disposeQueue.length var threshold = 0 while (data = disposeQueue[--i]) { if (data.i < 7) { if (data.element === null) { disposeQueue.splice(i, 1) if (data.list) { avalon.Array.remove(data.list, data) delete disposeQueue[data.uuid] } continue } if (shouldDispose(data.element)) { //如果它的虚拟DOM不在VTree上或其属性不在VM上 disposeQueue.splice(i, 1) avalon.Array.remove(data.list, data) disposeData(data) //avalon会在每次全量更新时,比较上次执行时间, //假若距离上次有半秒,就会发起一次GC,并且只检测当中的500个绑定 //而一个正常的页面不会超过2000个绑定(500即取其4分之一) //用户频繁操作页面,那么2,3秒内就把所有绑定检测一遍,将无效的绑定移除 if (threshold++ > 500) { lastGCIndex = i break } continue } data.i++ //基于检测频率,如果检测过7次,可以认为其是长久存在的节点,那么以后每7次才检测一次 if (data.i === 7) { data.i = 14 } } else { data.i-- } } beginTime = new Date() } function disposeData(data) { delete disposeQueue[data.uuid] // 先清除,不然无法回收了 data.element = null data.rollback && data.rollback() for (var key in data) { data[key] = null } } function shouldDispose(el) { try {//IE下,如果文本节点脱离DOM树,访问parentNode会报错 var fireError = el.parentNode.nodeType } catch (e) { return true } if (el.ifRemove) { // 如果节点被放到ifGroup,才移除 if (!root.contains(el.ifRemove) && (ifGroup === el.parentNode)) { el.parentNode && el.parentNode.removeChild(el) return true } } return el.msRetain ? 0 : (el.nodeType === 1 ? !root.contains(el) : !avalon.contains(root, el)) } /************************************************************************ * HTML处理(parseHTML, innerHTML, clearHTML) * *************************************************************************/ // We have to close these tags to support XHTML var tagHooks = { area: [1, "", ""], param: [1, "", ""], col: [2, "", "
"], legend: [1, "
", "
"], option: [1, ""], thead: [1, "", "
"], tr: [2, "", "
"], td: [3, "", "
"], g: [1, '', ''], //IE6-8在用innerHTML生成节点时,不能直接创建no-scope元素与HTML5的新标签 _default: W3C ? [0, "", ""] : [1, "X
", "
"] //div可以不用闭合 } tagHooks.th = tagHooks.td tagHooks.optgroup = tagHooks.option tagHooks.tbody = tagHooks.tfoot = tagHooks.colgroup = tagHooks.caption = tagHooks.thead String("circle,defs,ellipse,image,line,path,polygon,polyline,rect,symbol,text,use").replace(rword, function (tag) { tagHooks[tag] = tagHooks.g //处理SVG }) var rtagName = /<([\w:]+)/ //取得其tagName var rxhtml = /<(?!area|br|col|embed|hr|img|input|link|meta|param)(([\w:]+)[^>]*)\/>/ig var rcreate = W3C ? /[^\d\D]/ : /(<(?:script|link|style|meta|noscript))/ig var scriptTypes = oneObject(["", "text/javascript", "text/ecmascript", "application/ecmascript", "application/javascript"]) var rnest = /<(?:tb|td|tf|th|tr|col|opt|leg|cap|area)/ //需要处理套嵌关系的标签 var script = DOC.createElement("script") var rhtml = /<|&#?\w+;/ avalon.parseHTML = function (html) { var fragment = avalonFragment.cloneNode(false) if (typeof html !== "string") { return fragment } if (!rhtml.test(html)) { fragment.appendChild(DOC.createTextNode(html)) return fragment } html = html.replace(rxhtml, "<$1>").trim() var tag = (rtagName.exec(html) || ["", ""])[1].toLowerCase(), //取得其标签名 wrap = tagHooks[tag] || tagHooks._default, wrapper = cinerator, firstChild, neo if (!W3C) { //fix IE html = html.replace(rcreate, "
$1") //在link style script等标签之前添加一个补丁 } wrapper.innerHTML = wrap[1] + html + wrap[2] var els = wrapper.getElementsByTagName("script") if (els.length) { //使用innerHTML生成的script节点不会发出请求与执行text属性 for (var i = 0, el; el = els[i++];) { if (scriptTypes[el.type]) { //以偷龙转凤方式恢复执行脚本功能 neo = script.cloneNode(false) //FF不能省略参数 ap.forEach.call(el.attributes, function (attr) { if (attr && attr.specified) { neo[attr.name] = attr.value //复制其属性 neo.setAttribute(attr.name, attr.value) } }) // jshint ignore:line neo.text = el.text el.parentNode.replaceChild(neo, el) //替换节点 } } } if (!W3C) { //fix IE var target = wrap[1] === "X
" ? wrapper.lastChild.firstChild : wrapper.lastChild if (target && target.tagName === "TABLE" && tag !== "tbody") { //IE6-7处理 --> , // --> , // -->
for (els = target.childNodes, i = 0; el = els[i++];) { if (el.tagName === "TBODY" && !el.innerHTML) { target.removeChild(el) break } } } els = wrapper.getElementsByTagName("br") var n = els.length while (el = els[--n]) { if (el.className === "msNoScope") { el.parentNode.removeChild(el) } } for (els = wrapper.all, i = 0; el = els[i++];) { //fix VML if (isVML(el)) { fixVML(el) } } } //移除我们为了符合套嵌关系而添加的标签 for (i = wrap[0]; i--; wrapper = wrapper.lastChild) {} while (firstChild = wrapper.firstChild) { // 将wrapper上的节点转移到文档碎片上! fragment.appendChild(firstChild) } return fragment } function isVML(src) { var nodeName = src.nodeName return nodeName.toLowerCase() === nodeName && src.scopeName && src.outerText === "" } function fixVML(node) { if (node.currentStyle.behavior !== "url(#default#VML)") { node.style.behavior = "url(#default#VML)" node.style.display = "inline-block" node.style.zoom = 1 //hasLayout } } avalon.innerHTML = function (node, html) { if (!W3C && (!rcreate.test(html) && !rnest.test(html))) { try { node.innerHTML = html return } catch (e) {} } var a = this.parseHTML(html) this.clearHTML(node).appendChild(a) } avalon.clearHTML = function (node) { node.textContent = "" while (node.firstChild) { node.removeChild(node.firstChild) } return node } /********************************************************************* * avalon的原型方法定义区 * **********************************************************************/ function hyphen(target) { //转换为连字符线风格 return target.replace(/([a-z\d])([A-Z]+)/g, "$1-$2").toLowerCase() } function camelize(target) { //提前判断,提高getStyle等的效率 if (!target || target.indexOf("-") < 0 && target.indexOf("_") < 0) { return target } //转换为驼峰风格 return target.replace(/[-_][^-_]/g, function (match) { return match.charAt(1).toUpperCase() }) } var fakeClassListMethods = { _toString: function () { var node = this.node var cls = node.className var str = typeof cls === "string" ? cls : cls.baseVal return str.split(/\s+/).join(" ") }, _contains: function (cls) { return (" " + this + " ").indexOf(" " + cls + " ") > -1 }, _add: function (cls) { if (!this.contains(cls)) { this._set(this + " " + cls) } }, _remove: function (cls) { this._set((" " + this + " ").replace(" " + cls + " ", " ")) }, __set: function (cls) { cls = cls.trim() var node = this.node if (rsvg.test(node)) { //SVG元素的className是一个对象 SVGAnimatedString { baseVal="", animVal=""},只能通过set/getAttribute操作 node.setAttribute("class", cls) } else { node.className = cls } } //toggle存在版本差异,因此不使用它 } function fakeClassList(node) { if (!("classList" in node)) { node.classList = { node: node } for (var k in fakeClassListMethods) { node.classList[k.slice(1)] = fakeClassListMethods[k] } } return node.classList } "add,remove".replace(rword, function (method) { avalon.fn[method + "Class"] = function (cls) { var el = this[0] //https://developer.mozilla.org/zh-CN/docs/Mozilla/Firefox/Releases/26 if (cls && typeof cls === "string" && el && el.nodeType === 1) { cls.replace(/\S+/g, function (c) { fakeClassList(el)[method](c) }) } return this } }) avalon.fn.mix({ hasClass: function (cls) { var el = this[0] || {} return el.nodeType === 1 && fakeClassList(el).contains(cls) }, toggleClass: function (value, stateVal) { var className, i = 0 var classNames = String(value).match(/\S+/g) || [] var isBool = typeof stateVal === "boolean" while ((className = classNames[i++])) { var state = isBool ? stateVal : !this.hasClass(className) this[state ? "addClass" : "removeClass"](className) } return this }, attr: function (name, value) { if (arguments.length === 2) { this[0].setAttribute(name, value) return this } else { return this[0].getAttribute(name) } }, data: function (name, value) { name = "data-" + hyphen(name || "") switch (arguments.length) { case 2: this.attr(name, value) return this case 1: var val = this.attr(name) return parseData(val) case 0: var ret = {} ap.forEach.call(this[0].attributes, function (attr) { if (attr) { name = attr.name if (!name.indexOf("data-")) { name = camelize(name.slice(5)) ret[name] = parseData(attr.value) } } }) return ret } }, removeData: function (name) { name = "data-" + hyphen(name) this[0].removeAttribute(name) return this }, css: function (name, value) { if (avalon.isPlainObject(name)) { for (var i in name) { avalon.css(this, i, name[i]) } } else { var ret = avalon.css(this, name, value) } return ret !== void 0 ? ret : this }, position: function () { var offsetParent, offset, elem = this[0], parentOffset = { top: 0, left: 0 } if (!elem) { return } if (this.css("position") === "fixed") { offset = elem.getBoundingClientRect() } else { offsetParent = this.offsetParent() //得到真正的offsetParent offset = this.offset() // 得到正确的offsetParent if (offsetParent[0].tagName !== "HTML") { parentOffset = offsetParent.offset() } parentOffset.top += avalon.css(offsetParent[0], "borderTopWidth", true) parentOffset.left += avalon.css(offsetParent[0], "borderLeftWidth", true) // Subtract offsetParent scroll positions parentOffset.top -= offsetParent.scrollTop() parentOffset.left -= offsetParent.scrollLeft() } return { top: offset.top - parentOffset.top - avalon.css(elem, "marginTop", true), left: offset.left - parentOffset.left - avalon.css(elem, "marginLeft", true) } }, offsetParent: function () { var offsetParent = this[0].offsetParent while (offsetParent && avalon.css(offsetParent, "position") === "static") { offsetParent = offsetParent.offsetParent; } return avalon(offsetParent || root) }, bind: function (type, fn, phase) { if (this[0]) { //此方法不会链 return avalon.bind(this[0], type, fn, phase) } }, unbind: function (type, fn, phase) { if (this[0]) { avalon.unbind(this[0], type, fn, phase) } return this }, val: function (value) { var node = this[0] if (node && node.nodeType === 1) { var get = arguments.length === 0 var access = get ? ":get" : ":set" var fn = valHooks[getValType(node) + access] if (fn) { var val = fn(node, value) } else if (get) { return (node.value || "").replace(/\r/g, "") } else { node.value = value } } return get ? val : this } }) function parseData(data) { try { if (typeof data === "object") return data data = data === "true" ? true : data === "false" ? false : data === "null" ? null : +data + "" === data ? +data : rbrace.test(data) ? avalon.parseJSON(data) : data } catch (e) {} return data } var rbrace = /(?:\{[\s\S]*\}|\[[\s\S]*\])$/, rvalidchars = /^[\],:{}\s]*$/, rvalidbraces = /(?:^|:|,)(?:\s*\[)+/g, rvalidescape = /\\(?:["\\\/bfnrt]|u[\da-fA-F]{4})/g, rvalidtokens = /"[^"\\\r\n]*"|true|false|null|-?(?:\d+\.|)\d+(?:[eE][+-]?\d+|)/g avalon.parseJSON = window.JSON ? JSON.parse : function (data) { if (typeof data === "string") { data = data.trim(); if (data) { if (rvalidchars.test(data.replace(rvalidescape, "@") .replace(rvalidtokens, "]") .replace(rvalidbraces, ""))) { return (new Function("return " + data))() // jshint ignore:line } } avalon.error("Invalid JSON: " + data) } return data } avalon.fireDom = function (elem, type, opts) { if (DOC.createEvent) { var hackEvent = DOC.createEvent("Events"); hackEvent.initEvent(type, true, true, opts) avalon.mix(hackEvent, opts) elem.dispatchEvent(hackEvent) } else if (root.contains(elem)) {//IE6-8触发事件必须保证在DOM树中,否则报"SCRIPT16389: 未指明的错误" hackEvent = DOC.createEventObject() avalon.mix(hackEvent, opts) elem.fireEvent("on" + type, hackEvent) } } //生成avalon.fn.scrollLeft, avalon.fn.scrollTop方法 avalon.each({ scrollLeft: "pageXOffset", scrollTop: "pageYOffset" }, function (method, prop) { avalon.fn[method] = function (val) { var node = this[0] || {}, win = getWindow(node), top = method === "scrollTop" if (!arguments.length) { return win ? (prop in win) ? win[prop] : root[method] : node[method] } else { if (win) { win.scrollTo(!top ? val : avalon(win).scrollLeft(), top ? val : avalon(win).scrollTop()) } else { node[method] = val } } } }) function getWindow(node) { return node.window && node.document ? node : node.nodeType === 9 ? node.defaultView || node.parentWindow : false; } //=============================css相关======================= var cssHooks = avalon.cssHooks = {} var prefixes = ["", "-webkit-", "-o-", "-moz-", "-ms-"] var cssMap = { "float": W3C ? "cssFloat" : "styleFloat" } avalon.cssNumber = oneObject("animationIterationCount,columnCount,order,flex,flexGrow,flexShrink,fillOpacity,fontWeight,lineHeight,opacity,orphans,widows,zIndex,zoom") avalon.cssName = function (name, host, camelCase) { if (cssMap[name]) { return cssMap[name] } host = host || root.style for (var i = 0, n = prefixes.length; i < n; i++) { camelCase = camelize(prefixes[i] + name) if (camelCase in host) { return (cssMap[name] = camelCase) } } return null } cssHooks["@:set"] = function (node, name, value) { try { //node.style.width = NaN;node.style.width = "xxxxxxx";node.style.width = undefine 在旧式IE下会抛异常 node.style[name] = value } catch (e) {} } if (window.getComputedStyle) { cssHooks["@:get"] = function (node, name) { if (!node || !node.style) { throw new Error("getComputedStyle要求传入一个节点 " + node) } var ret, styles = getComputedStyle(node, null) if (styles) { ret = name === "filter" ? styles.getPropertyValue(name) : styles[name] if (ret === "") { ret = node.style[name] //其他浏览器需要我们手动取内联样式 } } return ret } cssHooks["opacity:get"] = function (node) { var ret = cssHooks["@:get"](node, "opacity") return ret === "" ? "1" : ret } } else { var rnumnonpx = /^-?(?:\d*\.)?\d+(?!px)[^\d\s]+$/i var rposition = /^(top|right|bottom|left)$/ var ralpha = /alpha\([^)]+\)/i var ropactiy = /(opacity|\d(\d|\.)*)/g var ie8 = !!window.XDomainRequest var border = { thin: ie8 ? '1px' : '2px', medium: ie8 ? '3px' : '4px', thick: ie8 ? '5px' : '6px' } cssHooks["@:get"] = function (node, name) { //取得精确值,不过它有可能是带em,pc,mm,pt,%等单位 var currentStyle = node.currentStyle var ret = currentStyle[name] if ((rnumnonpx.test(ret) && !rposition.test(ret))) { //①,保存原有的style.left, runtimeStyle.left, var style = node.style, left = style.left, rsLeft = node.runtimeStyle.left //②由于③处的style.left = xxx会影响到currentStyle.left, //因此把它currentStyle.left放到runtimeStyle.left, //runtimeStyle.left拥有最高优先级,不会style.left影响 node.runtimeStyle.left = currentStyle.left //③将精确值赋给到style.left,然后通过IE的另一个私有属性 style.pixelLeft //得到单位为px的结果;fontSize的分支见http://bugs.jquery.com/ticket/760 style.left = name === 'fontSize' ? '1em' : (ret || 0) ret = style.pixelLeft + "px" //④还原 style.left,runtimeStyle.left style.left = left node.runtimeStyle.left = rsLeft } if (ret === "medium") { name = name.replace("Width", "Style") //border width 默认值为medium,即使其为0" if (currentStyle[name] === "none") { ret = "0px" } } return ret === "" ? "auto" : border[ret] || ret } cssHooks["opacity:set"] = function (node, name, value) { var style = node.style var opacity = Number(value) <= 1 ? 'alpha(opacity=' + value * 100 + ')' : '' var filter = style.filter || '' style.zoom = 1 //不能使用以下方式设置透明度 //node.filters.alpha.opacity = value * 100 style.filter = (ralpha.test(filter) ? filter.replace(ralpha, opacity) : filter + ' ' + opacity).trim() if (!style.filter) { style.removeAttribute('filter') } } cssHooks["opacity:get"] = function (node) { var match = node.style.filter.match(ropactiy) || [] var ret = false for (var i = 0, el; el = match[i++];) { if (el === 'opacity') { ret = true } else if (ret) { return (el / 100) + '' } } return '1' } } "top,left".replace(rword, function (name) { cssHooks[name + ":get"] = function (node) { var computed = cssHooks["@:get"](node, name) return /px$/.test(computed) ? computed : avalon(node).position()[name] + "px" } }) var cssShow = { position: "absolute", visibility: "hidden", display: "block" } var rdisplayswap = /^(none|table(?!-c[ea]).+)/ function showHidden(node, array) { //http://www.cnblogs.com/rubylouvre/archive/2012/10/27/2742529.html if (node.offsetWidth <= 0) { //opera.offsetWidth可能小于0 if (rdisplayswap.test(cssHooks["@:get"](node, "display"))) { var obj = { node: node } for (var name in cssShow) { obj[name] = node.style[name] node.style[name] = cssShow[name] } array.push(obj) } var parent = node.parentNode if (parent && parent.nodeType === 1) { showHidden(parent, array) } } } "Width,Height".replace(rword, function (name) { //fix 481 var method = name.toLowerCase(), clientProp = "client" + name, scrollProp = "scroll" + name, offsetProp = "offset" + name cssHooks[method + ":get"] = function (node, which, override) { var boxSizing = -4 if (typeof override === "number") { boxSizing = override } which = name === "Width" ? ["Left", "Right"] : ["Top", "Bottom"] var ret = node[offsetProp] // border-box 0 if (boxSizing === 2) { // margin-box 2 return ret + avalon.css(node, "margin" + which[0], true) + avalon.css(node, "margin" + which[1], true) } if (boxSizing < 0) { // padding-box -2 ret = ret - avalon.css(node, "border" + which[0] + "Width", true) - avalon.css(node, "border" + which[1] + "Width", true) } if (boxSizing === -4) { // content-box -4 ret = ret - avalon.css(node, "padding" + which[0], true) - avalon.css(node, "padding" + which[1], true) } return ret } cssHooks[method + "&get"] = function (node) { var hidden = []; showHidden(node, hidden); var val = cssHooks[method + ":get"](node) for (var i = 0, obj; obj = hidden[i++];) { node = obj.node for (var n in obj) { if (typeof obj[n] === "string") { node.style[n] = obj[n] } } } return val; } avalon.fn[method] = function (value) { //会忽视其display var node = this[0] if (arguments.length === 0) { if (node.setTimeout) { //取得窗口尺寸 return node["inner" + name] || node.document.documentElement[clientProp] || node.document.body[clientProp] //IE6下前两个分别为undefined,0 } if (node.nodeType === 9) { //取得页面尺寸 var doc = node.documentElement //FF chrome html.scrollHeight< body.scrollHeight //IE 标准模式 : html.scrollHeight> body.scrollHeight //IE 怪异模式 : html.scrollHeight 最大等于可视窗口多一点? return Math.max(node.body[scrollProp], doc[scrollProp], node.body[offsetProp], doc[offsetProp], doc[clientProp]) } return cssHooks[method + "&get"](node) } else { return this.css(method, value) } } avalon.fn["inner" + name] = function () { return cssHooks[method + ":get"](this[0], void 0, -2) } avalon.fn["outer" + name] = function (includeMargin) { return cssHooks[method + ":get"](this[0], void 0, includeMargin === true ? 2 : 0) } }) avalon.fn.offset = function () { //取得距离页面左右角的坐标 var node = this[0], box = { left: 0, top: 0 } if (!node || !node.tagName || !node.ownerDocument) { return box } var doc = node.ownerDocument, body = doc.body, root = doc.documentElement, win = doc.defaultView || doc.parentWindow if (!avalon.contains(root, node)) { return box } //http://hkom.blog1.fc2.com/?mode=m&no=750 body的偏移量是不包含margin的 //我们可以通过getBoundingClientRect来获得元素相对于client的rect. //http://msdn.microsoft.com/en-us/library/ms536433.aspx if (node.getBoundingClientRect) { box = node.getBoundingClientRect() // BlackBerry 5, iOS 3 (original iPhone) } //chrome/IE6: body.scrollTop, firefox/other: root.scrollTop var clientTop = root.clientTop || body.clientTop, clientLeft = root.clientLeft || body.clientLeft, scrollTop = Math.max(win.pageYOffset || 0, root.scrollTop, body.scrollTop), scrollLeft = Math.max(win.pageXOffset || 0, root.scrollLeft, body.scrollLeft) // 把滚动距离加到left,top中去。 // IE一些版本中会自动为HTML元素加上2px的border,我们需要去掉它 // http://msdn.microsoft.com/en-us/library/ms533564(VS.85).aspx return { top: box.top + scrollTop - clientTop, left: box.left + scrollLeft - clientLeft } } //==================================val相关============================ function getValType(elem) { var ret = elem.tagName.toLowerCase() return ret === "input" && /checkbox|radio/.test(elem.type) ? "checked" : ret } var roption = /^]+))?)*\s+value[\s=]/i var valHooks = { "option:get": IEVersion ? function (node) { //在IE11及W3C,如果没有指定value,那么node.value默认为node.text(存在trim作),但IE9-10则是取innerHTML(没trim操作) //specified并不可靠,因此通过分析outerHTML判定用户有没有显示定义value return roption.test(node.outerHTML) ? node.value : node.text.trim() } : function (node) { return node.value }, "select:get": function (node, value) { var option, options = node.options, index = node.selectedIndex, getter = valHooks["option:get"], one = node.type === "select-one" || index < 0, values = one ? null : [], max = one ? index + 1 : options.length, i = index < 0 ? max : one ? index : 0 for (; i < max; i++) { option = options[i] //旧式IE在reset后不会改变selected,需要改用i === index判定 //我们过滤所有disabled的option元素,但在safari5下,如果设置select为disable,那么其所有孩子都disable //因此当一个元素为disable,需要检测其是否显式设置了disable及其父节点的disable情况 if ((option.selected || i === index) && !option.disabled) { value = getter(option) if (one) { return value } //收集所有selected值组成数组返回 values.push(value) } } return values }, "select:set": function (node, values, optionSet) { values = [].concat(values) //强制转换为数组 var getter = valHooks["option:get"] for (var i = 0, el; el = node.options[i++];) { if ((el.selected = values.indexOf(getter(el)) > -1)) { optionSet = true } } if (!optionSet) { node.selectedIndex = -1 } } } var keyMap = {} var keys = ["break,case,catch,continue,debugger,default,delete,do,else,false", "finally,for,function,if,in,instanceof,new,null,return,switch,this", "throw,true,try,typeof,var,void,while,with", /* 关键字*/ "abstract,boolean,byte,char,class,const,double,enum,export,extends", "final,float,goto,implements,import,int,interface,long,native", "package,private,protected,public,short,static,super,synchronized", "throws,transient,volatile", /*保留字*/ "arguments,let,yield,undefined"].join(",") keys.replace(/\w+/g, function (a) { keyMap[a] = true }) var ridentStart = /[a-z_$]/i var rwhiteSpace = /[\s\uFEFF\xA0]/ function getIdent(input, lastIndex) { var result = [] var subroutine = !!lastIndex lastIndex = lastIndex || 0 //将表达式中的标识符抽取出来 var state = "unknown" var variable = "" for (var i = 0; i < input.length; i++) { var c = input.charAt(i) if (c === "'" || c === '"') {//字符串开始 if (state === "unknown") { state = c } else if (state === c) {//字符串结束 state = "unknown" } } else if (c === "\\") { if (state === "'" || state === '"') { i++ } } else if (ridentStart.test(c)) {//碰到标识符 if (state === "unknown") { state = "variable" variable = c } else if (state === "maybePath") { variable = result.pop() variable += "." + c state = "variable" } else if (state === "variable") { variable += c } } else if (/\w/.test(c)) { if (state === "variable") { variable += c } } else if (c === ".") { if (state === "variable") { if (variable) { result.push(variable) variable = "" state = "maybePath" } } } else if (c === "[") { if (state === "variable" || state === "maybePath") { if (variable) {//如果前面存在变量,收集它 result.push(variable) variable = "" } var lastLength = result.length var last = result[lastLength - 1] var innerResult = getIdent(input.slice(i), i) if (innerResult.length) {//如果括号中存在变量,那么这里添加通配符 result[lastLength - 1] = last + ".*" result = innerResult.concat(result) } else { //如果括号中的东西是确定的,直接转换为其子属性 var content = input.slice(i + 1, innerResult.i) try { var text = (scpCompile(["return " + content]))() result[lastLength - 1] = last + "[" + text + "]" } catch (e) { } } state = "maybePath"//]后面可能还接东西 i = innerResult.i } } else if (c === "]") { if (subroutine) { result.i = i + lastIndex addVar(result, variable) return result } } else if (rwhiteSpace.test(c) && c !== "\r" && c !== "\n") { if (state === "variable") { if (addVar(result, variable)) { state = "maybePath" // aaa . bbb 这样的情况 } variable = "" } } else { addVar(result, variable) state = "unknown" variable = "" } } addVar(result, variable) return result } function addVar(array, element) { if (element && !keyMap[element]) { array.push(element) return true } } function addAssign(vars, vmodel, name, binding) { var ret = [], prefix = " = " + name + "." for (var i = vars.length, prop; prop = vars[--i]; ) { //修改prop格式: arr[0].pro1 ==> arr.0.pro1, var arr, a ,oldprop = prop if( prop.indexOf('[') >= 0 ){ prop = prop.replace('[','.').replace(']','') } arr = prop.split(".") var first = arr[0] while (a = arr.shift()) { if (vmodel.hasOwnProperty(a)) { ret.push(first + prefix + first) //bugfix:https://github.com/RubyLouvre/avalon/issues/1682 //对于由于expr中可能存在不可达的语句如:j5son[0].cn1 ==='' || j5son[0].cn2 ===''造成当cn2改变时dom未修改 //解决办法:对于vars数组里的所有变量,都取一次值,保证可以触发依赖收集。但不影响表达式的结果 //bugfix:https://github.com/RubyLouvre/avalon/issues/1742 //考虑如果绑定表达式中包含变量,会生成'obj.*.property'这样的表达式 //会造成var x = obj.*.property;语句编译失败。 //表达式中包含变量的不生成附值语句 if(oldprop.indexOf('.*')<0) ret.push(generateID('_nousevar_' + i) + ' = '+ oldprop) binding.observers.push({ v: vmodel, p: prop }) vars.splice(i, 1) } else { break } } } return ret } var rproxy = /(\$proxy\$[a-z]+)\d+$/ var variablePool = new Cache(218) //缓存求值函数,以便多次利用 var evaluatorPool = new Cache(128) function getVars(expr) { expr = expr.trim() var ret = variablePool.get(expr) if (ret) { return ret.concat() } var array = getIdent(expr) var uniq = {} var result = [] for (var i = 0, el; el = array[i++]; ) { if (!uniq[el]) { uniq[el] = 1 result.push(el) } } return variablePool.put(expr, result).concat() } function parseExpr(expr, vmodels, binding) { var filters = binding.filters if (typeof filters === "string" && filters.trim() && !binding._filters) { binding._filters = parseFilter(filters.trim()) } var vars = getVars(expr) var expose = new Date() - 0 var assigns = [] var names = [] var args = [] binding.observers = [] for (var i = 0, sn = vmodels.length; i < sn; i++) { if (vars.length) { var name = "vm" + expose + "_" + i names.push(name) args.push(vmodels[i]) assigns.push.apply(assigns, addAssign(vars, vmodels[i], name, binding)) } } binding.args = args var dataType = binding.type var exprId = vmodels.map(function (el) { return String(el.$id).replace(rproxy, "$1") }) + expr + dataType var getter = evaluatorPool.get(exprId) //直接从缓存,免得重复生成 if (getter) { binding.getter = getter // https://github.com/RubyLouvre/avalon/issues/1833 if (dataType === "duplex") { var setter = evaluatorPool.get(exprId + "setter") if(setter){ binding.setter = setter.apply(setter, binding.args) return getter } }else{ return getter } } if (!assigns.length) { assigns.push("fix" + expose) } if (dataType === "duplex") { var nameOne = {} assigns.forEach(function (a) { var arr = a.split("=") nameOne[arr[0].trim()] = arr[1].trim() }) expr = expr.replace(/[\$\w]+/, function (a) { return nameOne[a] ? nameOne[a] : a }) /* jshint ignore:start */ var fn2 = scpCompile(names.concat("'use strict';" + "return function(vvv){" + expr + " = vvv\n}\n")) /* jshint ignore:end */ evaluatorPool.put(exprId + "setter", fn2) binding.setter = fn2.apply(fn2, binding.args) } if (dataType === "on") { //事件绑定 if (expr.indexOf("(") === -1) { expr += ".call(this, $event)" } else { expr = expr.replace("(", ".call(this,") } names.push("$event") expr = "\nreturn " + expr + ";" //IE全家 Function("return ")出错,需要Function("return ;") var lastIndex = expr.lastIndexOf("\nreturn") var header = expr.slice(0, lastIndex) var footer = expr.slice(lastIndex) expr = header + "\n" + footer } else { expr = "\nreturn " + expr + ";" //IE全家 Function("return ")出错,需要Function("return ;") } var assignstr = [] avalon.each(assigns,function(idx,el){ // 这里跟上面 //bugfix:https://github.com/RubyLouvre/avalon/issues/1682 是相对应的。 if(el.indexOf('_nousevar_')>-1){ //子属性还没有创建,这里避免报错 assignstr.push("try{var " + el + "}catch(e){}") } else{ assignstr.push("var " + el ) } }); /* jshint ignore:start */ getter = scpCompile(names.concat("'use strict';\n" + assignstr.join(";\n") + expr)) /* jshint ignore:end */ return evaluatorPool.put(exprId, getter) } function normalizeExpr(code) { var hasExpr = rexpr.test(code) //比如ms-class="width{{w}}"的情况 if (hasExpr) { var array = scanExpr(code) if (array.length === 1) { return array[0].expr } return array.map(function (el) { return el.type ? "(" + el.expr + ")" : quote(el.expr) }).join(" + ") } else { return code } } avalon.normalizeExpr = normalizeExpr avalon.parseExprProxy = parseExpr var rthimRightParentheses = /\)\s*$/ var rthimOtherParentheses = /\)\s*\|/g var rquoteFilterName = /\|\s*([$\w]+)/g var rpatchBracket = /"\s*\["/g var rthimLeftParentheses = /"\s*\(/g function parseFilter(filters) { filters = filters .replace(rthimRightParentheses, "")//处理最后的小括号 .replace(rthimOtherParentheses, function () {//处理其他小括号 return "],|" }) .replace(rquoteFilterName, function (a, b) { //处理|及它后面的过滤器的名字 return "[" + quote(b) }) .replace(rpatchBracket, function () { return '"],["' }) .replace(rthimLeftParentheses, function () { return '",' }) + "]" /* jshint ignore:start */ return scpCompile(["return [" + filters + "]"])() /* jshint ignore:end */ } /********************************************************************* * 编译系统 * **********************************************************************/ var Escapes = { 92: "\\\\", 34: '\\"', 8: "\\b", 12: "\\f", 10: "\\n", 13: "\\r", 9: "\\t" } // Internal: Converts `value` into a zero-padded string such that its // length is at least equal to `width`. The `width` must be <= 6. var leadingZeroes = "000000" var toPaddedString = function (width, value) { // The `|| 0` expression is necessary to work around a bug in // Opera <= 7.54u2 where `0 == -0`, but `String(-0) !== "0"`. return (leadingZeroes + (value || 0)).slice(-width) }; var unicodePrefix = "\\u00" var escapeChar = function (character) { var charCode = character.charCodeAt(0), escaped = Escapes[charCode] if (escaped) { return escaped } return unicodePrefix + toPaddedString(2, charCode.toString(16)) }; var reEscape = /[\x00-\x1f\x22\x5c]/g function _quote(value) { reEscape.lastIndex = 0 return '"' + ( reEscape.test(value)? String(value).replace(reEscape, escapeChar) : value ) + '"' } var quote = typeof JSON !== 'undefined' ? JSON.stringify : _quote /********************************************************************* * 扫描系统 * **********************************************************************/ avalon.scan = function (elem, vmodel) { elem = elem || root var vmodels = vmodel ? [].concat(vmodel) : [] scanTag(elem, vmodels) } //http://www.w3.org/TR/html5/syntax.html#void-elements var stopScan = oneObject("area,base,basefont,br,col,command,embed,hr,img,input,link,meta,param,source,track,wbr,noscript,script,style,textarea".toUpperCase()) function checkScan(elem, callback, innerHTML) { var id = setTimeout(function () { var currHTML = elem.innerHTML clearTimeout(id) if (currHTML === innerHTML) { callback() } else { checkScan(elem, callback, currHTML) } }) } function createSignalTower(elem, vmodel) { var id = elem.getAttribute("avalonctrl") || vmodel.$id elem.setAttribute("avalonctrl", id) if (vmodel.$events) { vmodel.$events.expr = elem.tagName + '[avalonctrl="' + id + '"]' } } function getBindingCallback(elem, name, vmodels) { var callback = elem.getAttribute(name) if (callback) { for (var i = 0, vm; vm = vmodels[i++]; ) { if (vm.hasOwnProperty(callback) && typeof vm[callback] === "function") { return vm[callback] } } } } function executeBindings(bindings, vmodels) { for (var i = 0, binding; binding = bindings[i++]; ) { binding.vmodels = vmodels directives[binding.type].init(binding) avalon.injectBinding(binding) if (binding.getter && binding.element.nodeType === 1) { //移除数据绑定,防止被二次解析 //chrome使用removeAttributeNode移除不存在的特性节点时会报错 https://github.com/RubyLouvre/avalon/issues/99 binding.element.removeAttribute(binding.name) } } bindings.length = 0 } //https://github.com/RubyLouvre/avalon/issues/636 var mergeTextNodes = IEVersion && window.MutationObserver ? function (elem) { var node = elem.firstChild, text while (node) { var aaa = node.nextSibling if (node.nodeType === 3) { if (text) { text.nodeValue += node.nodeValue elem.removeChild(node) } else { text = node } } else { text = null } node = aaa } } : 0 var roneTime = /^\s*::/ var rmsAttr = /ms-(\w+)-?(.*)/ var events = oneObject("animationend,blur,change,input,click,dblclick,focus,keydown,keypress,keyup,mousedown,mouseenter,mouseleave,mousemove,mouseout,mouseover,mouseup,scan,scroll,submit") var obsoleteAttrs = oneObject("value,title,alt,checked,selected,disabled,readonly,enabled,href,src") function bindingSorter(a, b) { return a.priority - b.priority } var rnoCollect = /^(ms-\S+|data-\S+|on[a-z]+|id|style|class)$/ var ronattr = /^on\-[\w-]+$/ function getOptionsFromTag(elem, vmodels) { var attributes = elem.attributes var ret = {} for (var i = 0, attr; attr = attributes[i++]; ) { var name = attr.name if (attr.specified && !rnoCollect.test(name)) { var camelizeName = camelize(attr.name) if (/^on\-[\w-]+$/.test(name)) { ret[camelizeName] = getBindingCallback(elem, name, vmodels) } else { ret[camelizeName] = parseData(attr.value) } } } return ret } function scanAttr(elem, vmodels, match) { var scanNode = true if (vmodels.length) { var attributes = getAttributes ? getAttributes(elem) : elem.attributes var bindings = [] var uniq = {} for (var i = 0, attr; attr = attributes[i++]; ) { var name = attr.name if (uniq[name]) {//IE8下ms-repeat,ms-with BUG continue } uniq[name] = 1 if (attr.specified) { if (match = name.match(rmsAttr)) { //如果是以指定前缀命名的 var type = match[1] var param = match[2] || "" var value = attr.value if (events[type]) { param = type type = "on" } else if (obsoleteAttrs[type]) { param = type type = "attr" name = "ms-" + type + "-" + param log("warning!请改用" + name + "代替" + attr.name + "!") } if (directives[type]) { var newValue = value.replace(roneTime, "") var oneTime = value !== newValue var binding = { type: type, param: param, element: elem, name: name, expr: newValue, oneTime: oneTime, uuid: "_" + (++bindingID), //chrome与firefox下Number(param)得到的值不一样 #855 priority: (directives[type].priority || type.charCodeAt(0) * 10) + (Number(param.replace(/\D/g, "")) || 0) } if (type === "html" || type === "text") { var filters = getToken(value).filters binding.expr = binding.expr.replace(filters, "") binding.filters = filters.replace(rhasHtml, function () { binding.type = "html" binding.group = 1 return "" }).trim() // jshint ignore:line } else if (type === "duplex") { var hasDuplex = name } else if (name === "ms-if-loop") { binding.priority += 100 } else if (name === "ms-attr-value") { var hasAttrValue = name } bindings.push(binding) } } } } if (bindings.length) { bindings.sort(bindingSorter) //http://bugs.jquery.com/ticket/7071 //在IE下对VML读取type属性,会让此元素所有属性都变成 if (hasDuplex && hasAttrValue && elem.nodeName === "INPUT" && elem.type === "text") { log("warning!一个控件不能同时定义ms-attr-value与" + hasDuplex) } for (i = 0; binding = bindings[i]; i++) { type = binding.type if (rnoscanAttrBinding.test(type)) { return executeBindings(bindings.slice(0, i + 1), vmodels) } else if (scanNode) { scanNode = !rnoscanNodeBinding.test(type) } } executeBindings(bindings, vmodels) } } if (scanNode && !stopScan[elem.tagName] && (isWidget(elem) ? elem.msResolved : 1)) { mergeTextNodes && mergeTextNodes(elem) scanNodeList(elem, vmodels) //扫描子孙元素 } } var rnoscanAttrBinding = /^if|widget|repeat$/ var rnoscanNodeBinding = /^each|with|html|include$/ //IE67下,在循环绑定中,一个节点如果是通过cloneNode得到,自定义属性的specified为false,无法进入里面的分支, //但如果我们去掉scanAttr中的attr.specified检测,一个元素会有80+个特性节点(因为它不区分固有属性与自定义属性),很容易卡死页面 if (!W3C) { var attrPool = new Cache(512) var rattrs = /\s+([^=\s]+)(?:=("[^"]*"|'[^']*'|[^\s>]+))?/g, rquote = /^['"]/, rtag = /<\w+\b(?:(["'])[^"]*?(\1)|[^>])*>/i, ramp = /&/g //IE6-8解析HTML5新标签,会将它分解两个元素节点与一个文本节点 //
ddd
// window.onload = function() { // var body = document.body // for (var i = 0, el; el = body.children[i++]; ) { // avalon.log(el.outerHTML) // } // } //依次输出
,
var getAttributes = function (elem) { var html = elem.outerHTML //处理IE6-8解析HTML5新标签的情况,及
等半闭合标签outerHTML为空的情况 if (html.slice(0, 2) === "") str = str.slice(0,-1) var attributes = [], k, v var ret = attrPool.get(str) if (ret) { return ret } while (k = rattrs.exec(str)) { v = k[2] if (v) { v = (rquote.test(v) ? v.slice(1, -1) : v).replace(ramp, "&") } var name = k[1].toLowerCase() var binding = { name: name, specified: true, value: v || "" } attributes.push(binding) } return attrPool.put(str, attributes) } } function scanNodeList(parent, vmodels) { var nodes = avalon.slice(parent.childNodes) scanNodeArray(nodes, vmodels) } function scanNodeArray(nodes, vmodels) { function _delay_component(name) { setTimeout(function () { avalon.component(name) }) } for (var i = 0, node; node = nodes[i++]; ) { switch (node.nodeType) { case 1: var elem = node if (!elem.msResolved && elem.parentNode && elem.parentNode.nodeType === 1) { var library = isWidget(elem) if (library) { var widget = elem.localName ? elem.localName.replace(library + ":", "") : elem.nodeName var fullName = library + ":" + camelize(widget) componentQueue.push({ library: library, element: elem, fullName: fullName, widget: widget, vmodels: vmodels, name: "widget" }) if (avalon.components[fullName]) { //确保所有ms-attr-name扫描完再处理 _delay_component(fullName) } } } scanTag(node, vmodels) //扫描元素节点 if (node.msHasEvent) { avalon.fireDom(node, "datasetchanged", { bubble: node.msHasEvent }) } break case 3: if (rexpr.test(node.nodeValue)) { scanText(node, vmodels, i) //扫描文本节点 } break } } } function scanTag(elem, vmodels, node) { //扫描顺序 ms-skip(0) --> ms-important(1) --> ms-controller(2) --> ms-if(10) --> ms-repeat(100) //--> ms-if-loop(110) --> ms-attr(970) ...--> ms-each(1400)-->ms-with(1500)--〉ms-duplex(2000)垫后 var a = elem.getAttribute("ms-skip") //#360 在旧式IE中 Object标签在引入Flash等资源时,可能出现没有getAttributeNode,innerHTML的情形 if (!elem.getAttributeNode) { return log("warning " + elem.tagName + " no getAttributeNode method") } var b = elem.getAttributeNode("ms-important") var c = elem.getAttributeNode("ms-controller") if (typeof a === "string") { return } else if (node = b || c) { var newVmodel = avalon.vmodels[node.value] if (!newVmodel) { return } //ms-important不包含父VM,ms-controller相反 vmodels = node === b ? [newVmodel] : [newVmodel].concat(vmodels) var name = node.name elem.removeAttribute(name) //removeAttributeNode不会刷新[ms-controller]样式规则 avalon(elem).removeClass(name) createSignalTower(elem, newVmodel) } scanAttr(elem, vmodels) //扫描特性节点 if (newVmodel) { setTimeout(function () { newVmodel.$fire("ms-scan-end", elem) }) } } var rhasHtml = /\|\s*html(?:\b|$)/, r11a = /\|\|/g, rlt = /</g, rgt = />/g, rstringLiteral = /(['"])(\\\1|.)+?\1/g, rline = /\r?\n/g function getToken(value) { if (value.indexOf("|") > 0) { var scapegoat = value.replace(rstringLiteral, function (_) { return Array(_.length + 1).join("1") // jshint ignore:line }) var index = scapegoat.replace(r11a, "\u1122\u3344").indexOf("|") //干掉所有短路或 if (index > -1) { return { type: "text", filters: value.slice(index).trim(), expr: value.slice(0, index) } } } return { type: "text", expr: value, filters: "" } } function scanExpr(str) { var tokens = [], value, start = 0, stop do { stop = str.indexOf(openTag, start) if (stop === -1) { break } value = str.slice(start, stop) if (value) { // {{ 左边的文本 tokens.push({ expr: value }) } start = stop + openTag.length stop = str.indexOf(closeTag, start) if (stop === -1) { break } value = str.slice(start, stop) if (value) { //处理{{ }}插值表达式 tokens.push(getToken(value.replace(rline,""))) } start = stop + closeTag.length } while (1) value = str.slice(start) if (value) { //}} 右边的文本 tokens.push({ expr: value }) } return tokens } function scanText(textNode, vmodels, index) { var bindings = [], tokens = scanExpr(textNode.data) if (tokens.length) { for (var i = 0, token; token = tokens[i++];) { var node = DOC.createTextNode(token.expr) //将文本转换为文本节点,并替换原来的文本节点 if (token.type) { token.expr = token.expr.replace(roneTime, function () { token.oneTime = true return "" }) // jshint ignore:line token.element = node token.filters = token.filters.replace(rhasHtml, function () { token.type = "html" return "" }) // jshint ignore:line token.pos = index * 1000 + i bindings.push(token) //收集带有插值表达式的文本 } avalonFragment.appendChild(node) } textNode.parentNode.replaceChild(avalonFragment, textNode) if (bindings.length) executeBindings(bindings, vmodels) } } //使用来自游戏界的双缓冲技术,减少对视图的冗余刷新 var Buffer = function () { this.queue = [] } Buffer.prototype = { render: function (isAnimate) { if (!this.locked) { this.locked = isAnimate ? root.offsetHeight + 10 : 1 var me = this avalon.nextTick(function () { me.flush() }) } }, flush: function () { for (var i = 0, sub; sub = this.queue[i++]; ) { sub.update && sub.update() } this.locked = 0 this.queue = [] } } var buffer = new Buffer() var componentQueue = [] var widgetList = [] var componentHooks = { $construct: function () { return avalon.mix.apply(null, arguments) }, $ready: noop, $init: noop, $dispose: noop, $container: null, $childReady: noop, $replace: false, $extend: null, $$template: function (str) { return str } } avalon.components = {} avalon.component = function (name, opts) { if (opts) { avalon.components[name] = avalon.mix({}, componentHooks, opts) } for (var i = 0, obj; obj = componentQueue[i]; i++) { if (name === obj.fullName) { componentQueue.splice(i, 1) i--; (function (host, hooks, elem, widget) { //如果elem已从Document里移除,直接返回 //issuse : https://github.com/RubyLouvre/avalon2/issues/40 if (!avalon.contains(DOC, elem) || elem.msResolved) { avalon.Array.remove(componentQueue, host) return } var dependencies = 1 var library = host.library var global = avalon.libraries[library] || componentHooks //===========收集各种配置======= if (elem.getAttribute("ms-attr-identifier")) { //如果还没有解析完,就延迟一下 #1155 return } var elemOpts = getOptionsFromTag(elem, host.vmodels) var vmOpts = getOptionsFromVM(host.vmodels, elemOpts.config || host.fullName) var $id = elemOpts.$id || elemOpts.identifier || generateID(widget) delete elemOpts.config delete elemOpts.$id delete elemOpts.identifier var componentDefinition = {} var parentHooks = avalon.components[hooks.$extend] if (parentHooks) { avalon.mix(true, componentDefinition, parentHooks) componentDefinition = parentHooks.$construct.call(elem, componentDefinition, {}, {}) } else { avalon.mix(true, componentDefinition, hooks) } componentDefinition = avalon.components[name].$construct.call(elem, componentDefinition, vmOpts, elemOpts) componentDefinition.$refs = {} componentDefinition.$id = $id //==========构建VM========= var keepSlot = componentDefinition.$slot var keepReplace = componentDefinition.$replace var keepContainer = componentDefinition.$container var keepTemplate = componentDefinition.$template delete componentDefinition.$slot delete componentDefinition.$replace delete componentDefinition.$container delete componentDefinition.$construct var vmodel = avalon.define(componentDefinition) || {} elem.msResolved = 1 //防止二进扫描此元素 vmodel.$init(vmodel, elem) global.$init(vmodel, elem) var nodes = elem.childNodes //收集插入点 var slots = {}, snode for (var s = 0, el; el = nodes[s++]; ) { var type = el.nodeType === 1 && el.getAttribute("slot") || keepSlot if (type) { if (slots[type]) { slots[type].push(el) } else { slots[type] = [el] } } } if (vmodel.$$template) { avalon.clearHTML(elem) elem.innerHTML = vmodel.$$template(keepTemplate) } for (s in slots) { if (vmodel.hasOwnProperty(s)) { var ss = slots[s] if (ss.length) { var fragment = avalonFragment.cloneNode(true) for (var ns = 0; snode = ss[ns++]; ) { fragment.appendChild(snode) } vmodel[s] = fragment } slots[s] = null } } slots = null var child = elem.children[0] || elem.firstChild if (keepReplace) { elem.parentNode.replaceChild(child, elem) child.msResolved = 1 var cssText = elem.style.cssText var className = elem.className elem = host.element = child elem.style.cssText += ";"+ cssText if (className) { avalon(elem).addClass(className) } } if (keepContainer) { keepContainer.appendChild(elem) } avalon.fireDom(elem, "datasetchanged", {library: library, vm: vmodel, childReady: 1}) var children = 0 var removeFn = avalon.bind(elem, "datasetchanged", function (e) { if (e.childReady && e.library === library) { dependencies += e.childReady if (vmodel !== e.vm) { vmodel.$refs[e.vm.$id] = e.vm if (e.childReady === -1) { children++ vmodel.$childReady(vmodel, elem, e) } e.stopPropagation() } } if (dependencies === 0) { var id1 = setTimeout(function () { clearTimeout(id1) vmodel.$ready(vmodel, elem, host.vmodels) global.$ready(vmodel, elem, host.vmodels) }, children ? Math.max(children * 17, 100) : 17) avalon.unbind(elem, "datasetchanged", removeFn) //================== host.rollback = function () { try { vmodel.$dispose(vmodel, elem) global.$dispose(vmodel, elem) } catch (e) { } delete avalon.vmodels[vmodel.$id] } injectDisposeQueue(host, widgetList) if (window.chrome) { elem.addEventListener("DOMNodeRemovedFromDocument", function () { setTimeout(rejectDisposeQueue) }) } } }) scanTag(elem, [vmodel].concat(host.vmodels)) avalon.vmodels[vmodel.$id] = vmodel if (!elem.childNodes.length) { avalon.fireDom(elem, "datasetchanged", {library: library, vm: vmodel, childReady: -1}) } else { var id2 = setTimeout(function () { clearTimeout(id2) avalon.fireDom(elem, "datasetchanged", {library: library, vm: vmodel, childReady: -1}) }, 17) } })(obj, avalon.components[name], obj.element, obj.widget)// jshint ignore:line } } } function getOptionsFromVM(vmodels, pre) { if (pre) { for (var i = 0, v; v = vmodels[i++]; ) { if (v.hasOwnProperty(pre) && typeof v[pre] === "object") { var vmOptions = v[pre] return vmOptions.$model || vmOptions break } } } return {} } avalon.libraries = [] avalon.library = function (name, opts) { if (DOC.namespaces) { DOC.namespaces.add(name, 'http://www.w3.org/1999/xhtml'); } avalon.libraries[name] = avalon.mix({ $init: noop, $ready: noop, $dispose: noop }, opts || {}) } avalon.library("ms") /* broswer nodeName scopeName localName IE9 ONI:BUTTON oni button IE10 ONI:BUTTON undefined oni:button IE8 button oni undefined chrome ONI:BUTTON undefined oni:button */ function isWidget(el) { //如果为自定义标签,返回UI库的名字 if (el.scopeName && el.scopeName !== "HTML") { return el.scopeName } var fullName = el.nodeName.toLowerCase() var index = fullName.indexOf(":") if (index > 0) { return fullName.slice(0, index) } } //各种MVVM框架在大型表格下的性能测试 // https://github.com/RubyLouvre/avalon/issues/859 var bools = ["autofocus,autoplay,async,allowTransparency,checked,controls", "declare,disabled,defer,defaultChecked,defaultSelected", "contentEditable,isMap,loop,multiple,noHref,noResize,noShade", "open,readOnly,selected" ].join(",") var boolMap = {} bools.replace(rword, function (name) { boolMap[name.toLowerCase()] = name }) var propMap = {//属性名映射 "accept-charset": "acceptCharset", "char": "ch", "charoff": "chOff", "class": "className", "for": "htmlFor", "http-equiv": "httpEquiv" } var anomaly = ["accessKey,bgColor,cellPadding,cellSpacing,codeBase,codeType,colSpan", "dateTime,defaultValue,frameBorder,longDesc,maxLength,marginWidth,marginHeight", "rowSpan,tabIndex,useMap,vSpace,valueType,vAlign" ].join(",") anomaly.replace(rword, function (name) { propMap[name.toLowerCase()] = name }) var attrDir = avalon.directive("attr", { init: function (binding) { //{{aaa}} --> aaa //{{aaa}}/bbb.html --> (aaa) + "/bbb.html" binding.expr = normalizeExpr(binding.expr.trim()) if (binding.type === "include") { var elem = binding.element effectBinding(elem, binding) binding.includeRendered = getBindingCallback(elem, "data-include-rendered", binding.vmodels) binding.includeLoaded = getBindingCallback(elem, "data-include-loaded", binding.vmodels) var outer = binding.includeReplace = !!avalon(elem).data("includeReplace") if (avalon(elem).data("includeCache")) { binding.templateCache = {} } binding.start = DOC.createComment("ms-include") binding.end = DOC.createComment("ms-include-end") if (outer) { binding.element = binding.end binding._element = elem elem.parentNode.insertBefore(binding.start, elem) elem.parentNode.insertBefore(binding.end, elem.nextSibling) } else { elem.insertBefore(binding.start, elem.firstChild) elem.appendChild(binding.end) } } }, update: function (val) { var elem = this.element var attrName = this.param //这模块在1.5.9被重构了 if (attrName.indexOf('data-') === 0 || rsvg.test(elem)) { elem.setAttribute(attrName, val) } else { var propName = propMap[attrName] || attrName if (typeof elem[propName] === 'boolean') { elem[propName] = !!val //布尔属性必须使用el.xxx = true|false方式设值 //如果为false, IE全系列下相当于setAttribute(xxx,''), //会影响到样式,需要进一步处理 } if (val === false) {//移除属性 elem.removeAttribute(propName) return } //IE6中classNamme, htmlFor等无法检测它们为内建属性  if(!W3C && /[A-Z]/.test(propName)){ elem[propName] = val + '' return } //SVG只能使用setAttribute(xxx, yyy), VML只能使用node.xxx = yyy , //HTML的固有属性必须node.xxx = yyy var isInnate = (!W3C && isVML(elem)) ? true : isInnateProps(elem.nodeName, attrName) /* istanbul ignore next */ if (isInnate) { if (attrName === 'href' || attrName === 'src') { val = String(val).replace(/&/g, '&') //处理IE67自动转义的问题 } elem[propName] = val + '' } else { elem.setAttribute(attrName, val) } } } }) var innateMap = {} function isInnateProps(nodeName, attrName) { var key = nodeName + ":" + attrName if (key in innateMap) { return innateMap[key] } return innateMap[key] = (attrName in document.createElement(nodeName)) } //这几个指令都可以使用插值表达式,如ms-src="aaa/{{b}}/{{c}}.html" "title,alt,src,value,css,include,href".replace(rword, function (name) { directives[name] = attrDir }) //根据VM的属性值或表达式的值切换类名,ms-class="xxx yyy zzz:flag" //http://www.cnblogs.com/rubylouvre/archive/2012/12/17/2818540.html avalon.directive("class", { init: function (binding) { var oldStyle = binding.param var method = binding.type if (!oldStyle || isFinite(oldStyle)) { binding.param = "" //去掉数字 directives.effect.init(binding) } else { log('ms-' + method + '-xxx="yyy"这种用法已经过时,请使用ms-' + method + '="xxx:yyy"') binding.expr = '[' + quote(oldStyle) + "," + binding.expr + "]" binding.oldStyle = oldStyle } if (method === "hover" || method === "active") { //确保只绑定一次 if (!binding.hasBindEvent) { var elem = binding.element var $elem = avalon(elem) var activate = "mouseenter" //在移出移入时切换类名 var abandon = "mouseleave" if (method === "active") { //在聚焦失焦中切换类名 elem.tabIndex = elem.tabIndex || -1 activate = "mousedown" abandon = "mouseup" var fn0 = $elem.bind("mouseleave", function () { binding.toggleClass && $elem.removeClass(binding.newClass) }) } } var fn1 = $elem.bind(activate, function () { binding.toggleClass && $elem.addClass(binding.newClass) }) var fn2 = $elem.bind(abandon, function () { binding.toggleClass && $elem.removeClass(binding.newClass) }) binding.rollback = function () { $elem.unbind("mouseleave", fn0) $elem.unbind(activate, fn1) $elem.unbind(abandon, fn2) } binding.hasBindEvent = true } }, update: function (arr) { var binding = this var $elem = avalon(this.element) binding.newClass = arr[0] binding.toggleClass = !!arr[1] if (binding.oldClass && binding.newClass !== binding.oldClass) { $elem.removeClass(binding.oldClass) } binding.oldClass = binding.newClass if (binding.type === "class") { if (binding.oldStyle) { $elem.toggleClass(binding.oldStyle, !!arr[1]) } else { $elem.toggleClass(binding.newClass, binding.toggleClass) } } } }) "hover,active".replace(rword, function (name) { directives[name] = directives["class"] }) //ms-controller绑定已经在scanTag 方法中实现 avalon.directive("css", { init: directives.attr.init, update: function (val) { avalon(this.element).css(this.param, val) } }) avalon.directive("data", { priority: 100, update: function (val) { var elem = this.element var key = "data-" + this.param if (val && typeof val === "object") { elem[key] = val } else { elem.setAttribute(key, String(val)) } } }) //双工绑定 var rduplexType = /^(?:checkbox|radio)$/ var rduplexParam = /^(?:radio|checked)$/ var rnoduplexInput = /^(file|button|reset|submit|checkbox|radio|range)$/ var duplexBinding = avalon.directive("duplex", { priority: 2000, init: function (binding, hasCast) { var elem = binding.element var vmodels = binding.vmodels binding.changed = getBindingCallback(elem, "data-duplex-changed", vmodels) || noop var params = [] var casting = oneObject("string,number,boolean,checked") if (elem.type === "radio" && binding.param === "") { binding.param = "checked" } binding.param.replace(rw20g, function (name) { if (rduplexType.test(elem.type) && rduplexParam.test(name)) { if (name === "radio") log("ms-duplex-radio已经更名为ms-duplex-checked") name = "checked" binding.isChecked = true binding.xtype = "radio" } if (name === "bool") { name = "boolean" log("ms-duplex-bool已经更名为ms-duplex-boolean") } else if (name === "text") { name = "string" log("ms-duplex-text已经更名为ms-duplex-string") } if (casting[name]) { hasCast = true } avalon.Array.ensure(params, name) }) if (!hasCast) { params.push("string") } binding.param = params.join("-") if (!binding.xtype) { binding.xtype = elem.tagName === "SELECT" ? "select" : elem.type === "checkbox" ? "checkbox" : elem.type === "radio" ? "radio" : /^change/.test(elem.getAttribute("data-duplex-event")) ? "change" : "input" } //===================绑定事件====================== var bound = binding.bound = function (type, callback) { if (elem.addEventListener) { elem.addEventListener(type, callback, false) } else { elem.attachEvent("on" + type, callback) } var old = binding.rollback binding.rollback = function () { elem.avalonSetter = null avalon.unbind(elem, type, callback) old && old() } } function callback(value) { binding.changed.call(this, value, binding) } var composing = false function compositionStart() { composing = true } function compositionEnd() { composing = false setTimeout(updateVModel) } var updateVModel = function (e) { var val = elem.value //防止递归调用形成死循环 if (composing || val === binding.oldValue || binding.pipe === null) //处理中文输入法在minlengh下引发的BUG return var lastValue = binding.pipe(val, binding, "get") try { binding.oldValue = val binding.setter(lastValue) callback.call(elem, lastValue) } catch (ex) { log(ex) } } switch (binding.xtype) { case "radio": binding.bound("click", function () { var lastValue = binding.pipe(elem.value, binding, "get") try { binding.setter(lastValue) callback.call(elem, lastValue) } catch (ex) { log(ex) } }) break case "checkbox": bound(W3C ? "change" : "click", function () { var method = elem.checked ? "ensure" : "remove" var array = binding.getter.apply(0, binding.vmodels) if (!Array.isArray(array)) { log("ms-duplex应用于checkbox上要对应一个数组") array = [array] } var val = binding.pipe(elem.value, binding, "get") avalon.Array[method](array, val) callback.call(elem, array) }) break case "change": bound("change", updateVModel) break case "input": if (!IEVersion) { // W3C bound("input", updateVModel) //非IE浏览器才用这个 bound("compositionstart", compositionStart) bound("compositionend", compositionEnd) bound("DOMAutoComplete", updateVModel) } else { // IE下通过selectionchange事件监听IE9+点击input右边的X的清空行为,及粘贴,剪切,删除行为 if (IEVersion > 8) { if (IEVersion === 9) { //IE9删除字符后再失去焦点不会同步 #1167 bound("keyup", updateVModel) } bound("input", updateVModel) //IE9使用propertychange无法监听中文输入改动 } else { //onpropertychange事件无法区分是程序触发还是用户触发 //IE6-8下第一次修改时不会触发,需要使用keydown或selectionchange修正 bound("propertychange", function (e) { if (e.propertyName === "value") { updateVModel() } }) } bound("dragend", function () { setTimeout(function () { updateVModel() }, 17) }) //http://www.cnblogs.com/rubylouvre/archive/2013/02/17/2914604.html //http://www.matts411.com/post/internet-explorer-9-oninput/ } break case "select": bound("change", function () { var val = avalon(elem).val() //字符串或字符串数组 if (Array.isArray(val)) { val = val.map(function (v) { return binding.pipe(v, binding, "get") }) } else { val = binding.pipe(val, binding, "get") } if (val + "" !== binding.oldValue) { try { binding.setter(val) } catch (ex) { log(ex) } } }) bound("datasetchanged", function (e) { if (e.bubble === "selectDuplex") { var value = binding._value var curValue = Array.isArray(value) ? value.map(String) : value + "" avalon(elem).val(curValue) elem.oldValue = curValue + "" callback.call(elem, curValue) } }) break } if (binding.xtype === "input" && !rnoduplexInput.test(elem.type)) { if (elem.type !== "hidden") { bound("focus", function () { elem.msFocus = true }) bound("blur", function () { elem.msFocus = false }) } elem.avalonSetter = updateVModel //#765 watchValueInTimer(function () { if (avalon.contains(root, elem)) { if (!this.msFocus) { updateVModel() } } else if (!elem.msRetain) { return false } }) } }, update: function (value) { var elem = this.element, binding = this, curValue if (!this.init) { for (var i in avalon.vmodels) { var v = avalon.vmodels[i] v.$fire("avalon-ms-duplex-init", binding) } var cpipe = binding.pipe || (binding.pipe = pipe) cpipe(null, binding, "init") this.init = 1 } switch (this.xtype) { case "input": case "change": curValue = this.pipe(value, this, "set") //fix #673 if (curValue !== this.oldValue) { var fixCaret = false if (elem.msFocus) { try { var pos = getCaret(elem) if (pos.start === pos.end) { pos = pos.start fixCaret = true } } catch (e) { } } elem.value = binding.oldValue = curValue if (fixCaret) { setCaret(elem, pos, pos) } } break case "radio": curValue = binding.isChecked ? !!value : value + "" === elem.value if (IEVersion === 6) { setTimeout(function () { //IE8 checkbox, radio是使用defaultChecked控制选中状态, //并且要先设置defaultChecked后设置checked //并且必须设置延迟 elem.defaultChecked = curValue elem.checked = curValue }, 31) } else { elem.checked = curValue } break case "checkbox": var array = [].concat(value) //强制转换为数组 curValue = this.pipe(elem.value, this, "get") elem.checked = array.indexOf(curValue) > -1 break case "select": //必须变成字符串后才能比较 binding._value = value if (!elem.msHasEvent) { elem.msHasEvent = "selectDuplex" //必须等到其孩子准备好才触发 } else { avalon.fireDom(elem, "datasetchanged", { bubble: elem.msHasEvent }) } break } } }) if (IEVersion) { avalon.bind(DOC, "selectionchange", function (e) { var el = DOC.activeElement || {} if (!el.msFocus && el.avalonSetter) { el.avalonSetter() } }) } function fixNull(val) { return val == null ? "" : val } avalon.duplexHooks = { checked: { get: function (val, binding) { return !binding.oldValue } }, string: { get: function (val) { //同步到VM return val }, set: fixNull }, "boolean": { get: function (val) { return val === "true" }, set: fixNull }, number: { get: function (val, binding) { var number = parseFloat(val + "") if (-val === -number) { return number } var arr = /strong|medium|weak/.exec(binding.element.getAttribute("data-duplex-number")) || ["medium"] switch (arr[0]) { case "strong": return 0 case "medium": return val === "" ? "" : 0 case "weak": return val } }, set: fixNull } } function pipe(val, binding, action) { binding.param.replace(rw20g, function (name) { var hook = avalon.duplexHooks[name] if (hook && typeof hook[action] === "function") { val = hook[action](val, binding) } }) return val } var TimerID, ribbon = [] avalon.tick = function (fn) { if (ribbon.push(fn) === 1) { TimerID = setInterval(ticker, 60) } } function ticker() { for (var n = ribbon.length - 1; n >= 0; n--) { var el = ribbon[n] if (el() === false) { ribbon.splice(n, 1) } } if (!ribbon.length) { clearInterval(TimerID) } } var watchValueInTimer = noop new function () { // jshint ignore:line try { //#272 IE9-IE11, firefox var setters = {} var aproto = HTMLInputElement.prototype var bproto = HTMLTextAreaElement.prototype function newSetter(value) { // jshint ignore:line setters[this.tagName].call(this, value) if (!this.msFocus && this.avalonSetter) { this.avalonSetter() } } var inputProto = HTMLInputElement.prototype Object.getOwnPropertyNames(inputProto) //故意引发IE6-8等浏览器报错 setters["INPUT"] = Object.getOwnPropertyDescriptor(aproto, "value").set Object.defineProperty(aproto, "value", { set: newSetter }) setters["TEXTAREA"] = Object.getOwnPropertyDescriptor(bproto, "value").set Object.defineProperty(bproto, "value", { set: newSetter }) } catch (e) { //在chrome 43中 ms-duplex终于不需要使用定时器实现双向绑定了 // http://updates.html5rocks.com/2015/04/DOM-attributes-now-on-the-prototype // https://docs.google.com/document/d/1jwA8mtClwxI-QJuHT7872Z0pxpZz8PBkf2bGAbsUtqs/edit?pli=1 watchValueInTimer = avalon.tick } } // jshint ignore:line function getCaret(ctrl) { var start = NaN, end = NaN if (ctrl.setSelectionRange) { start = ctrl.selectionStart end = ctrl.selectionEnd } else if (document.selection && document.selection.createRange) { var range = document.selection.createRange() start = 0 - range.duplicate().moveStart('character', -100000) end = start + range.text.length } return { start: start, end: end } } function setCaret(ctrl, begin, end) { if (!ctrl.value || ctrl.readOnly) return if (ctrl.createTextRange) {//IE6-8 var range = ctrl.createTextRange() range.collapse(true) range.moveStart("character", begin) range.select() } else { ctrl.selectionStart = begin ctrl.selectionEnd = end } } avalon.directive("effect", { priority: 5, init: function (binding) { var text = binding.expr, className, rightExpr var colonIndex = text.replace(rexprg, function (a) { return a.replace(/./g, "0") }).indexOf(":") //取得第一个冒号的位置 if (colonIndex === -1) { // 比如 ms-class/effect="aaa bbb ccc" 的情况 className = text rightExpr = true } else { // 比如 ms-class/effect-1="ui-state-active:checked" 的情况 className = text.slice(0, colonIndex) rightExpr = text.slice(colonIndex + 1) } if (!rexpr.test(text)) { className = quote(className) } else { className = normalizeExpr(className) } binding.expr = "[" + className + "," + rightExpr + "]" }, update: function (arr) { var name = arr[0] var elem = this.element if (elem.getAttribute("data-effect-name") === name) { return } else { elem.removeAttribute("data-effect-driver") } var inlineStyles = elem.style var computedStyles = window.getComputedStyle ? window.getComputedStyle(elem) : null var useAni = false if (computedStyles && (supportTransition || supportAnimation)) { //如果支持CSS动画 var duration = inlineStyles[transitionDuration] || computedStyles[transitionDuration] if (duration && duration !== '0s') { elem.setAttribute("data-effect-driver", "t") useAni = true } if (!useAni) { duration = inlineStyles[animationDuration] || computedStyles[animationDuration] if (duration && duration !== '0s') { elem.setAttribute("data-effect-driver", "a") useAni = true } } } if (!useAni) { if (avalon.effects[name]) { elem.setAttribute("data-effect-driver", "j") useAni = true } } if (useAni) { elem.setAttribute("data-effect-name", name) } } }) avalon.effects = {} avalon.effect = function (name, callbacks) { avalon.effects[name] = callbacks } var supportTransition = false var supportAnimation = false var transitionEndEvent var animationEndEvent var transitionDuration = avalon.cssName("transition-duration") var animationDuration = avalon.cssName("animation-duration") new function () {// jshint ignore:line var checker = { 'TransitionEvent': 'transitionend', 'WebKitTransitionEvent': 'webkitTransitionEnd', 'OTransitionEvent': 'oTransitionEnd', 'otransitionEvent': 'otransitionEnd' } var tran //有的浏览器同时支持私有实现与标准写法,比如webkit支持前两种,Opera支持1、3、4 for (var name in checker) { if (window[name]) { tran = checker[name] break; } try { var a = document.createEvent(name); tran = checker[name] break; } catch (e) { } } if (typeof tran === "string") { supportTransition = true transitionEndEvent = tran } //大致上有两种选择 //IE10+, Firefox 16+ & Opera 12.1+: animationend //Chrome/Safari: webkitAnimationEnd //http://blogs.msdn.com/b/davrous/archive/2011/12/06/introduction-to-css3-animat ions.aspx //IE10也可以使用MSAnimationEnd监听,但是回调里的事件 type依然为animationend // el.addEventListener("MSAnimationEnd", function(e) { // alert(e.type)// animationend!!! // }) checker = { 'AnimationEvent': 'animationend', 'WebKitAnimationEvent': 'webkitAnimationEnd' } var ani; for (name in checker) { if (window[name]) { ani = checker[name]; break; } } if (typeof ani === "string") { supportTransition = true animationEndEvent = ani } }() var effectPool = []//重复利用动画实例 function effectFactory(el, opts) { if (!el || el.nodeType !== 1) { return null } if (opts) { var name = opts.effectName var driver = opts.effectDriver } else { name = el.getAttribute("data-effect-name") driver = el.getAttribute("data-effect-driver") } if (!name || !driver) { return null } var instance = effectPool.pop() || new Effect() instance.el = el instance.driver = driver instance.useCss = driver !== "j" if (instance.useCss) { opts && avalon(el).addClass(opts.effectClass) instance.cssEvent = driver === "t" ? transitionEndEvent : animationEndEvent } instance.name = name instance.callbacks = avalon.effects[name] || {} return instance } function effectBinding(elem, binding) { var name = elem.getAttribute("data-effect-name") if (name) { binding.effectName = name binding.effectDriver = elem.getAttribute("data-effect-driver") var stagger = +elem.getAttribute("data-effect-stagger") binding.effectLeaveStagger = +elem.getAttribute("data-effect-leave-stagger") || stagger binding.effectEnterStagger = +elem.getAttribute("data-effect-enter-stagger") || stagger binding.effectClass = elem.className || NaN } } function upperFirstChar(str) { return str.replace(/^[\S]/g, function (m) { return m.toUpperCase() }) } var effectBuffer = new Buffer() function Effect() { }//动画实例,做成类的形式,是为了共用所有原型方法 Effect.prototype = { contrustor: Effect, enterClass: function () { return getEffectClass(this, "enter") }, leaveClass: function () { return getEffectClass(this, "leave") }, // 共享一个函数 actionFun: function (name, before, after) { if (document.hidden) { return } var me = this var el = me.el var isLeave = name === "leave" name = isLeave ? "leave" : "enter" var oppositeName = isLeave ? "enter" : "leave" callEffectHook(me, "abort" + upperFirstChar(oppositeName)) callEffectHook(me, "before" + upperFirstChar(name)) if (!isLeave) before(el) //这里可能做插入DOM树的操作,因此必须在修改类名前执行 var cssCallback = function (cancel) { el.removeEventListener(me.cssEvent, me.cssCallback) if (isLeave) { before(el) //这里可能做移出DOM树操作,因此必须位于动画之后 avalon(el).removeClass(me.cssClass) } else { if (me.driver === "a") { avalon(el).removeClass(me.cssClass) } } if (cancel !== true) { callEffectHook(me, "after" + upperFirstChar(name)) after && after(el) } me.dispose() } if (me.useCss) { if (me.cssCallback) { //如果leave动画还没有完成,立即完成 me.cssCallback(true) } me.cssClass = getEffectClass(me, name) me.cssCallback = cssCallback me.update = function () { el.addEventListener(me.cssEvent, me.cssCallback) if (!isLeave && me.driver === "t") {//transtion延迟触发 avalon(el).removeClass(me.cssClass) } } avalon(el).addClass(me.cssClass)//animation会立即触发 effectBuffer.render(true) effectBuffer.queue.push(me) } else { callEffectHook(me, name, cssCallback) } }, enter: function (before, after) { this.actionFun.apply(this, ["enter"].concat(avalon.slice(arguments))) }, leave: function (before, after) { this.actionFun.apply(this, ["leave"].concat(avalon.slice(arguments))) }, dispose: function () {//销毁与回收到池子中 this.update = this.cssCallback = null if (effectPool.unshift(this) > 100) { effectPool.pop() } } } function getEffectClass(instance, type) { var a = instance.callbacks[type + "Class"] if (typeof a === "string") return a if (typeof a === "function") return a() return instance.name + "-" + type } function callEffectHook(effect, name, cb) { var hook = effect.callbacks[name] if (hook) { hook.call(effect, effect.el, cb) } } var applyEffect = function (el, dir/*[before, [after, [opts]]]*/) { var args = aslice.call(arguments, 0) if (typeof args[2] !== "function") { args.splice(2, 0, noop) } if (typeof args[3] !== "function") { args.splice(3, 0, noop) } var before = args[2] var after = args[3] var opts = args[4] var effect = effectFactory(el, opts) if (!effect) { before() after() return false } else { var method = dir ? 'enter' : 'leave' effect[method](before, after) } } avalon.mix(avalon.effect, { apply: applyEffect, append: function (el, parent, after, opts) { return applyEffect(el, 1, function () { parent.appendChild(el) }, after, opts) }, before: function (el, target, after, opts) { return applyEffect(el, 1, function () { target.parentNode.insertBefore(el, target) }, after, opts) }, remove: function (el, parent, after, opts) { return applyEffect(el, 0, function () { if (el.parentNode === parent) parent.removeChild(el) }, after, opts) } }) avalon.directive("html", { update: function (val) { var binding = this var elem = this.element var isHtmlFilter = elem.nodeType !== 1 var parent = isHtmlFilter ? elem.parentNode : elem if (!parent) return val = val == null ? "" : val if (elem.nodeType === 3) { var signature = generateID("html") parent.insertBefore(DOC.createComment(signature), elem) binding.element = DOC.createComment(signature + ":end") parent.replaceChild(binding.element, elem) elem = binding.element } if (typeof val !== "object") {//string, number, boolean var fragment = avalon.parseHTML(String(val)) } else if (val.nodeType === 11) { //将val转换为文档碎片 fragment = val } else if (val.nodeType === 1 || val.item) { var nodes = val.nodeType === 1 ? val.childNodes : val.item fragment = avalonFragment.cloneNode(true) while (nodes[0]) { fragment.appendChild(nodes[0]) } } nodes = avalon.slice(fragment.childNodes) //插入占位符, 如果是过滤器,需要有节制地移除指定的数量,如果是html指令,直接清空 if (isHtmlFilter) { var endValue = elem.nodeValue.slice(0, -4) while (true) { var node = elem.previousSibling if (!node || node.nodeType === 8 && node.nodeValue === endValue) { break } else { parent.removeChild(node) } } parent.insertBefore(fragment, elem) } else { avalon.clearHTML(elem).appendChild(fragment) } scanNodeArray(nodes, binding.vmodels) } }) avalon.directive("if", { priority: 10, update: function (val) { var binding = this var elem = this.element var stamp = binding.stamp = +new Date() var par var after = function () { if (stamp !== binding.stamp) return binding.recoverNode = null } if (binding.recoverNode) binding.recoverNode() // 还原现场,有移动节点的都需要还原现场 try { if (!elem.parentNode) return par = elem.parentNode } catch (e) { return } if (val) { //插回DOM树 function alway() {// jshint ignore:line if (elem.getAttribute(binding.name)) { elem.removeAttribute(binding.name) scanAttr(elem, binding.vmodels) } binding.rollback = null } if (elem.nodeType === 8) { var keep = binding.keep var hasEffect = avalon.effect.apply(keep, 1, function () { if (stamp !== binding.stamp) return elem.parentNode.replaceChild(keep, elem) elem = binding.element = keep //这时可能为null if (keep.getAttribute("_required")) {//#1044 elem.required = true elem.removeAttribute("_required") } if (elem.querySelectorAll) { avalon.each(elem.querySelectorAll("[_required=true]"), function (el) { el.required = true el.removeAttribute("_required") }) } alway() }, after) hasEffect = hasEffect === false } if (!hasEffect) alway() } else { //移出DOM树,并用注释节点占据原位置 if (elem.nodeType === 1) { if (elem.required === true) { elem.required = false elem.setAttribute("_required", "true") } try {//如果不支持querySelectorAll或:required,可以直接无视 avalon.each(elem.querySelectorAll(":required"), function (el) { elem.required = false el.setAttribute("_required", "true") }) } catch (e) { } var node = binding.element = DOC.createComment("ms-if"), pos = elem.nextSibling binding.recoverNode = function () { binding.recoverNode = null if (node.parentNode !== par) { par.insertBefore(node, pos) binding.keep = elem } } avalon.effect.apply(elem, 0, function () { binding.recoverNode = null if (stamp !== binding.stamp) return elem.parentNode.replaceChild(node, elem) //解决 当使用ms-if条件为常量表达式时 if节点泄漏 //https://github.com/RubyLouvre/avalon2/issues/78 //通过判断binding.getter()调用是否抛异常,来判断ms-if的表达式是否为常量。 try{ if(!binding.getter()){ //如果binding.expr是常量,binding.getter调用会返回false,不需要将elem移动到ifGroup binding.keep = elem //元素节点 binding.rollback = noop }else{ //never been here log('directive:ms-if: never been here.') } }catch(e){ //如果是vm属性绑定,binding.getter调用会抛异常. 把节点移到ifGroup binding.keep = elem //元素节点 ifGroup.appendChild(elem) binding.rollback = function () { if (elem.parentNode === ifGroup) { ifGroup.removeChild(elem) } } } }, after) } } } }) //ms-important绑定已经在scanTag 方法中实现 var rnoscripts = /(?:[\s\S]+?)<\/noscript>/img var rnoscriptText = /([\s\S]+?)<\/noscript>/im var getXHR = function () { return new (window.XMLHttpRequest || ActiveXObject)("Microsoft.XMLHTTP") // jshint ignore:line } //将所有远程加载的模板,以字符串形式存放到这里 var templatePool = avalon.templateCache = {} function getTemplateContainer(binding, id, text) { var div = binding.templateCache && binding.templateCache[id] if (div) { var dom = DOC.createDocumentFragment(), firstChild while (firstChild = div.firstChild) { dom.appendChild(firstChild) } return dom } return avalon.parseHTML(text) } function nodesToFrag(nodes) { var frag = DOC.createDocumentFragment() for (var i = 0, len = nodes.length; i < len; i++) { frag.appendChild(nodes[i]) } return frag } avalon.directive("include", { init: directives.attr.init, update: function (val) { var binding = this var elem = this.element var vmodels = binding.vmodels var rendered = binding.includeRendered var effectClass = binding.effectName && binding.effectClass // 是否开启动画 var templateCache = binding.templateCache // 是否data-include-cache var outer = binding.includeReplace // 是否data-include-replace var loaded = binding.includeLoaded var target = outer ? elem.parentNode : elem var _ele = binding._element // data-include-replace binding.element === binding.end binding.recoverNodes = binding.recoverNodes || avalon.noop var scanTemplate = function (text) { var _stamp = binding._stamp = +(new Date()) // 过滤掉频繁操作 if (loaded) { var newText = loaded.apply(target, [text].concat(vmodels)) if (typeof newText === "string") text = newText } if (rendered) { checkScan(target, function () { rendered.call(target) }, NaN) } var lastID = binding.includeLastID || "_default" // 默认 binding.includeLastID = val var leaveEl = templateCache && templateCache[lastID] || DOC.createElement(elem.tagName || binding._element.tagName) // 创建一个离场元素 if (effectClass) { leaveEl.className = effectClass target.insertBefore(leaveEl, binding.start) // 插入到start之前,防止被错误的移动 } // cache or animate,移动节点 (templateCache || {})[lastID] = leaveEl var fragOnDom = binding.recoverNodes() // 恢复动画中的节点 if (fragOnDom) { target.insertBefore(fragOnDom, binding.end) } while (true) { var node = binding.start.nextSibling if (node && node !== leaveEl && node !== binding.end) { leaveEl.appendChild(node) } else { break } } // 元素退场 avalon.effect.remove(leaveEl, target, function () { if (templateCache) { // write cache if (_stamp === binding._stamp) ifGroup.appendChild(leaveEl) } }, binding) var enterEl = target, before = avalon.noop, after = avalon.noop var fragment = getTemplateContainer(binding, val, text) var nodes = avalon.slice(fragment.childNodes) if (outer && effectClass) { enterEl = _ele enterEl.innerHTML = "" // 清空 enterEl.setAttribute("ms-skip", "true") target.insertBefore(enterEl, binding.end.nextSibling) // 插入到bingding.end之后避免被错误的移动 before = function () { enterEl.insertBefore(fragment, null) // 插入节点 } after = function () { binding.recoverNodes = avalon.noop if (_stamp === binding._stamp) { fragment = nodesToFrag(nodes) target.insertBefore(fragment, binding.end) // 插入真实element scanNodeArray(nodes, vmodels) } if (enterEl.parentNode === target) target.removeChild(enterEl) // 移除入场动画元素 } binding.recoverNodes = function () { binding.recoverNodes = avalon.noop return nodesToFrag(nodes) } } else { before = function () {//新添加元素的动画 target.insertBefore(fragment, binding.end) scanNodeArray(nodes, vmodels) } } avalon.effect.apply(enterEl, "enter", before, after) } if (binding.param === "src") { if (typeof templatePool[val] === "string") { avalon.nextTick(function () { scanTemplate(templatePool[val]) }) } else if (Array.isArray(templatePool[val])) { //#805 防止在循环绑定中发出许多相同的请求 templatePool[val].push(scanTemplate) } else { var xhr = getXHR() xhr.onreadystatechange = function () { if (xhr.readyState === 4) { var s = xhr.status if (s >= 200 && s < 300 || s === 304 || s === 1223) { var text = xhr.responseText for (var f = 0, fn; fn = templatePool[val][f++]; ) { fn(text) } templatePool[val] = text }else{ log("ms-include load ["+ val +"] error") } } } templatePool[val] = [scanTemplate] xhr.open("GET", val, true) if ("withCredentials" in xhr) { xhr.withCredentials = true } xhr.setRequestHeader("X-Requested-With", "XMLHttpRequest") xhr.send(null) } } else { //IE系列与够新的标准浏览器支持通过ID取得元素(firefox14+) //http://tjvantoll.com/2012/07/19/dom-element-references-as-global-variables/ var el = val && val.nodeType === 1 ? val : DOC.getElementById(val) if (el) { if (el.tagName === "NOSCRIPT" && !(el.innerHTML || el.fixIE78)) { //IE7-8 innerText,innerHTML都无法取得其内容,IE6能取得其innerHTML xhr = getXHR() //IE9-11与chrome的innerHTML会得到转义的内容,它们的innerText可以 xhr.open("GET", location, false) xhr.send(null) //http://bbs.csdn.net/topics/390349046?page=1#post-393492653 var noscripts = DOC.getElementsByTagName("noscript") var array = (xhr.responseText || "").match(rnoscripts) || [] var n = array.length for (var i = 0; i < n; i++) { var tag = noscripts[i] if (tag) { //IE6-8中noscript标签的innerHTML,innerText是只读的 tag.style.display = "none" //http://haslayout.net/css/noscript-Ghost-Bug tag.fixIE78 = (array[i].match(rnoscriptText) || ["", " "])[1] } } } avalon.nextTick(function () { scanTemplate(el.fixIE78 || el.value || el.innerText || el.innerHTML) }) } } } }) var rdash = /\(([^)]*)\)/ var onDir = avalon.directive("on", { priority: 3000, init: function (binding) { var value = binding.expr binding.type = "on" var eventType = binding.param.replace(/-\d+$/, "") // ms-on-mousemove-10 if (typeof onDir[eventType + "Hook"] === "function") { onDir[eventType + "Hook"](binding) } if (value.indexOf("(") > 0 && value.indexOf(")") > -1) { var matched = (value.match(rdash) || ["", ""])[1].trim() if (matched === "" || matched === "$event") { // aaa() aaa($event)当成aaa处理 value = value.replace(rdash, "") } } binding.expr = value }, update: function (callback) { var binding = this var elem = this.element callback = function (e) { var fn = binding.getter || noop return fn.apply(this, binding.args.concat(e)) } var eventType = binding.param.replace(/-\d+$/, "") // ms-on-mousemove-10 if (eventType === "scan") { callback.call(elem, { type: eventType }) } else if (typeof binding.specialBind === "function") { binding.specialBind(elem, callback) } else { var removeFn = avalon.bind(elem, eventType, callback) } binding.rollback = function () { if (typeof binding.specialUnbind === "function") { binding.specialUnbind() } else { avalon.unbind(elem, eventType, removeFn) } } } }) avalon.directive("repeat", { priority: 90, init: function (binding) { var type = binding.type binding.cache = {} //用于存放代理VM binding.enterCount = 0 var elem = binding.element if (elem.nodeType === 1) { elem.removeAttribute(binding.name) effectBinding(elem, binding) binding.param = binding.param || "el" binding.sortedCallback = getBindingCallback(elem, "data-with-sorted", binding.vmodels) var rendered = getBindingCallback(elem, "data-" + type + "-rendered", binding.vmodels) var signature = generateID(type) var start = DOC.createComment(signature + ":start") var end = binding.element = DOC.createComment(signature + ":end") binding.signature = signature binding.start = start binding.template = avalonFragment.cloneNode(false) if (type === "repeat") { var parent = elem.parentNode parent.replaceChild(end, elem) parent.insertBefore(start, end) binding.template.appendChild(elem) } else { while (elem.firstChild) { binding.template.appendChild(elem.firstChild) } elem.appendChild(start) elem.appendChild(end) parent = elem } binding.element = end if (rendered) { var removeFn = avalon.bind(parent, "datasetchanged", function () { rendered.apply(parent, parent.args) avalon.unbind(parent, "datasetchanged", removeFn) parent.msRendered = rendered }) } } }, update: function (value, oldValue) { var binding = this var xtype = this.xtype this.enterCount += 1 var init = !oldValue if (init) { binding.$outer = {} var check0 = "$key" var check1 = "$val" if (xtype === "array") { check0 = "$first" check1 = "$last" } for (var i = 0, v; v = binding.vmodels[i++]; ) { if (v.hasOwnProperty(check0) && v.hasOwnProperty(check1)) { binding.$outer = v break } } } var track = this.track if (binding.sortedCallback) { //如果有回调,则让它们排序 var keys2 = binding.sortedCallback.call(parent, track) if (keys2 && Array.isArray(keys2)) { track = keys2 } } var action = "move" binding.$repeat = value var fragments = [] var transation = init && avalonFragment.cloneNode(false) var proxies = [] var param = this.param var retain = avalon.mix({}, this.cache) var elem = this.element var length = track.length var parent = elem.parentNode //检查新元素数量 var newCount = 0 for (i = 0; i < length; i++) { var keyOrId = track[i] if (!retain[keyOrId]) newCount++ } var oldCount = 0 for (i in retain){ oldCount++ } var clear = (!length || newCount === length) && oldCount > 10 //当全部是新元素,且移除元素较多(10)时使用clear if (clear){ var kill = elem.previousSibling var start = binding.start while(kill !== start) { parent.removeChild(kill) kill = elem.previousSibling } } for (i = 0; i < length; i++) { keyOrId = track[i] //array为随机数, object 为keyName var proxy = retain[keyOrId] if (!proxy) { proxy = getProxyVM(this) proxy.$up = null if (xtype === "array") { action = "add" proxy.$id = keyOrId var valueItem = value[i] proxy[param] = valueItem //index if (Object(valueItem) === valueItem) { valueItem.$ups = valueItem.$ups || {} valueItem.$ups[param] = proxy } } else { action = "append" proxy.$key = keyOrId proxy.$val = value[keyOrId] //key proxy[param] = { $key: proxy.$key, $val: proxy.$val } } this.cache[keyOrId] = proxy var node = proxy.$anchor || (proxy.$anchor = elem.cloneNode(false)) node.nodeValue = this.signature shimController(binding, transation, proxy, fragments, init && !binding.effectDriver) decorateProxy(proxy, binding, xtype) } else { // if (xtype === "array") { // proxy[param] = value[i] // } fragments.push({}) retain[keyOrId] = true } //重写proxy if (this.enterCount === 1) {//防止多次进入,导致位置不对 proxy.$active = false proxy.$oldIndex = proxy.$index proxy.$active = true proxy.$index = i } if (xtype === "array") { proxy.$first = i === 0 proxy.$last = i === length - 1 // proxy[param] = value[i] } else { proxy.$val = toJson(value[keyOrId]) //这里是处理vm.object = newObject的情况 } proxies.push(proxy) } this.proxies = proxies if (init && !binding.effectDriver) { parent.insertBefore(transation, elem) fragments.forEach(function (fragment) { scanNodeArray(fragment.nodes || [], fragment.vmodels) //if(fragment.vmodels.length > 2) fragment.nodes = fragment.vmodels = null })// jshint ignore:line } else { var staggerIndex = binding.staggerIndex = 0 for (keyOrId in retain) { if (retain[keyOrId] !== true) { action = "del" !clear && removeItem(retain[keyOrId].$anchor, binding,true) // 相当于delete binding.cache[key] proxyRecycler(this.cache, keyOrId, param) retain[keyOrId] = null } } for (i = 0; i < length; i++) { proxy = proxies[i] keyOrId = xtype === "array" ? proxy.$id : proxy.$key var pre = proxies[i - 1] var preEl = pre ? pre.$anchor : binding.start if (!retain[keyOrId]) {//如果还没有插入到DOM树,进行插入动画 (function (fragment, preElement) { var nodes = fragment.nodes var vmodels = fragment.vmodels if (nodes) { staggerIndex = mayStaggerAnimate(binding.effectEnterStagger, function () { parent.insertBefore(fragment.content, preElement.nextSibling) scanNodeArray(nodes, vmodels) !init && animateRepeat(nodes, 1, binding) }, staggerIndex) } fragment.nodes = fragment.vmodels = null })(fragments[i], preEl)// jshint ignore:line } else if (proxy.$index !== proxy.$oldIndex) {//进行移动动画 (function (proxy2, preElement) { staggerIndex = mayStaggerAnimate(binding.effectEnterStagger, function () { var curNode = removeItem(proxy2.$anchor) var inserted = avalon.slice(curNode.childNodes) parent.insertBefore(curNode, preElement.nextSibling) animateRepeat(inserted, 1, binding) }, staggerIndex) })(proxy, preEl)// jshint ignore:line } } } if (!value.$track) {//如果是非监控对象,那么就将其$events清空,阻止其持续监听 for (keyOrId in this.cache) { proxyRecycler(this.cache, keyOrId, param) } } //repeat --> duplex (function (args) { parent.args = args if (parent.msRendered) {//第一次事件触发,以后直接调用 parent.msRendered.apply(parent, args) } })(kernel.newWatch ? arguments : [action]); var id = setTimeout(function () { clearTimeout(id) //触发上层的select回调及自己的rendered回调 avalon.fireDom(parent, "datasetchanged", { bubble: parent.msHasEvent }) }) this.enterCount -= 1 } }) "with,each".replace(rword, function (name) { directives[name] = avalon.mix({}, directives.repeat, { priority: 1400 }) }) function animateRepeat(nodes, isEnter, binding) { for (var i = 0, node; node = nodes[i++]; ) { if (node.className === binding.effectClass) { avalon.effect.apply(node, isEnter, noop, noop, binding) } } } function mayStaggerAnimate(staggerTime, callback, index) { if (staggerTime) { setTimeout(callback, (++index) * staggerTime) } else { callback() } return index } function removeItem(node, binding, flagRemove) { var fragment = avalonFragment.cloneNode(false) var last = node var breakText = last.nodeValue var staggerIndex = binding && Math.max(+binding.staggerIndex, 0) var nodes = avalon.slice(last.parentNode.childNodes) var index = nodes.indexOf(last) while (true) { var pre = nodes[--index] //node.previousSibling if (!pre || String(pre.nodeValue).indexOf(breakText) === 0) { break } if (!flagRemove && binding && (pre.className === binding.effectClass)) { node = pre; (function (cur) { binding.staggerIndex = mayStaggerAnimate(binding.effectLeaveStagger, function () { avalon.effect.apply(cur, 0, noop, function () { fragment.appendChild(cur) }, binding) }, staggerIndex) })(pre);// jshint ignore:line } else { fragment.insertBefore(pre, fragment.firstChild) } } fragment.appendChild(last) return fragment } function shimController(data, transation, proxy, fragments, init) { var content = data.template.cloneNode(true) var nodes = avalon.slice(content.childNodes) content.appendChild(proxy.$anchor) init && transation.appendChild(content) var itemName = data.param || "el" var valueItem = proxy[itemName], nv nv = [proxy].concat(data.vmodels) var fragment = { nodes: nodes, vmodels: nv, content: content } fragments.push(fragment) } // {} --> {xx: 0, yy: 1, zz: 2} add // {xx: 0, yy: 1, zz: 2} --> {xx: 0, yy: 1, zz: 2, uu: 3} // [xx: 0, yy: 1, zz: 2} --> {xx: 0, zz: 1, yy: 2} function getProxyVM(binding) { var agent = binding.xtype === "object" ? withProxyAgent : eachProxyAgent var proxy = agent(binding) var node = proxy.$anchor || (proxy.$anchor = binding.element.cloneNode(false)) node.nodeValue = binding.signature proxy.$outer = binding.$outer return proxy } function decorateProxy(proxy, binding, type) { if (type === "array") { proxy.$remove = function () { binding.$repeat.removeAt(proxy.$index) } var param = binding.param proxy.$watch(param, function (a) { var index = proxy.$index binding.$repeat[index] = a }) } else { proxy.$watch("$val", function fn(a) { binding.$repeat[proxy.$key] = a }) } } var eachProxyPool = [] function eachProxyAgent(data, proxy) { var itemName = data.param || "el" for (var i = 0, n = eachProxyPool.length; i < n; i++) { var candidate = eachProxyPool[i] if (candidate && candidate.hasOwnProperty(itemName)) { eachProxyPool.splice(i, 1) proxy = candidate break } } if (!proxy) { proxy = eachProxyFactory(itemName) } return proxy } function eachProxyFactory(itemName) { var source = { $outer: {}, $index: 0, $oldIndex: 0, $anchor: null, //----- $first: false, $last: false, $remove: avalon.noop } source[itemName] = NaN var force = { $last: 1, $first: 1, $index: 1 } force[itemName] = 1 var proxy = modelFactory(source, { force: force }) proxy.$id = generateID("$proxy$each") return proxy } var withProxyPool = [] function withProxyAgent(data) { var itemName = data.param || "el" return withProxyPool.pop() || withProxyFactory(itemName) } function withProxyFactory(itemName) { var source = { $key: "", $val: NaN, $index: 0, $oldIndex: 0, $outer: {}, $anchor: null } source[itemName] = NaN var force = { $key: 1, $val: 1, $index: 1 } force[itemName] = 1 var proxy = modelFactory(source, { force: force }) proxy.$id = generateID("$proxy$with") return proxy } function proxyRecycler(cache, key, param) { var proxy = cache[key] if (proxy) { var proxyPool = proxy.$id.indexOf("$proxy$each") === 0 ? eachProxyPool : withProxyPool proxy.$outer = {} for (var i in proxy.$events) { var a = proxy.$events[i] if (Array.isArray(a)) { a.length = 0 if (i === param) { proxy[param] = NaN } else if (i === "$val") { proxy.$val = NaN } } } if (proxyPool.unshift(proxy) > kernel.maxRepeatSize) { proxyPool.pop() } delete cache[key] } } /********************************************************************* * 各种指令 * **********************************************************************/ //ms-skip绑定已经在scanTag 方法中实现 avalon.directive("text", { update: function (value) { var elem = this.element value = value == null ? "" : value //不在页面上显示undefined null if (elem.nodeType === 3) { //绑定在文本节点上 try { //IE对游离于DOM树外的节点赋值会报错 elem.data = value } catch (e) { } } else { //绑定在特性节点上 if ("textContent" in elem) { elem.textContent = value } else { elem.innerText = value } } } }) function parseDisplay(nodeName, val) { //用于取得此类标签的默认display值 var key = "_" + nodeName if (!parseDisplay[key]) { var node = DOC.createElement(nodeName) root.appendChild(node) if (W3C) { val = getComputedStyle(node, null).display } else { val = node.currentStyle.display } root.removeChild(node) parseDisplay[key] = val } return parseDisplay[key] } avalon.parseDisplay = parseDisplay avalon.directive("visible", { init: function (binding) { effectBinding(binding.element, binding) }, update: function (val) { var binding = this, elem = this.element, stamp var noEffect = !this.effectName if (!this.stamp) { stamp = this.stamp = +new Date() if (val) { elem.style.display = binding.display || "" if (avalon(elem).css("display") === "none") { elem.style.display = binding.display = parseDisplay(elem.nodeName) } } else { elem.style.display = "none" } return } stamp = this.stamp = +new Date() if (val) { avalon.effect.apply(elem, 1, function () { if (stamp !== binding.stamp) return var driver = elem.getAttribute("data-effect-driver") || "a" if (noEffect) {//不用动画时走这里 elem.style.display = binding.display || "" } // "a", "t" if (driver === "a" || driver === "t") { if (avalon(elem).css("display") === "none") { elem.style.display = binding.display || parseDisplay(elem.nodeName) } } }) } else { avalon.effect.apply(elem, 0, function () { if (stamp !== binding.stamp) return elem.style.display = "none" }) } } }) /********************************************************************* * 自带过滤器 * **********************************************************************/ var rscripts = /]*>([\S\s]*?)<\/script\s*>/gim var ron = /\s+(on[^=\s]+)(?:=("[^"]*"|'[^']*'|[^\s>]+))?/g var ropen = /<\w+\b(?:(["'])[^"]*?(\1)|[^>])*>/ig var rsanitize = { a: /\b(href)\=("javascript[^"]*"|'javascript[^']*')/ig, img: /\b(src)\=("javascript[^"]*"|'javascript[^']*')/ig, form: /\b(action)\=("javascript[^"]*"|'javascript[^']*')/ig } var rsurrogate = /[\uD800-\uDBFF][\uDC00-\uDFFF]/g var rnoalphanumeric = /([^\#-~| |!])/g; function numberFormat(number, decimals, point, thousands) { //form http://phpjs.org/functions/number_format/ //number 必需,要格式化的数字 //decimals 可选,规定多少个小数位。 //point 可选,规定用作小数点的字符串(默认为 . )。 //thousands 可选,规定用作千位分隔符的字符串(默认为 , ),如果设置了该参数,那么所有其他参数都是必需的。 number = (number + '') .replace(/[^0-9+\-Ee.]/g, '') var n = !isFinite(+number) ? 0 : +number, prec = !isFinite(+decimals) ? 3 : Math.abs(decimals), sep = thousands || ",", dec = point || ".", s = '', toFixedFix = function(n, prec) { var k = Math.pow(10, prec) return '' + (Math.round(n * k) / k) .toFixed(prec) } // Fix for IE parseFloat(0.55).toFixed(0) = 0; s = (prec ? toFixedFix(n, prec) : '' + Math.round(n)) .split('.') if (s[0].length > 3) { s[0] = s[0].replace(/\B(?=(?:\d{3})+(?!\d))/g, sep) } if ((s[1] || '') .length < prec) { s[1] = s[1] || '' s[1] += new Array(prec - s[1].length + 1) .join('0') } return s.join(dec) } var filters = avalon.filters = { uppercase: function(str) { return str.toUpperCase() }, lowercase: function(str) { return str.toLowerCase() }, truncate: function(str, length, truncation) { //length,新字符串长度,truncation,新字符串的结尾的字段,返回新字符串 length = length || 30 truncation = typeof truncation === "string" ? truncation : "..." return str.length > length ? str.slice(0, length - truncation.length) + truncation : String(str) }, $filter: function(val) { for (var i = 1, n = arguments.length; i < n; i++) { var array = arguments[i] var fn = avalon.filters[array[0]] if (typeof fn === "function") { var arr = [val].concat(array.slice(1)) val = fn.apply(null, arr) } } return val }, camelize: camelize, //https://www.owasp.org/index.php/XSS_Filter_Evasion_Cheat_Sheet // chrome // chrome // IE67chrome // IE67chrome // IE67chrome sanitize: function(str) { return str.replace(rscripts, "").replace(ropen, function(a, b) { var match = a.toLowerCase().match(/<(\w+)\s/) if (match) { //处理a标签的href属性,img标签的src属性,form标签的action属性 var reg = rsanitize[match[1]] if (reg) { a = a.replace(reg, function(s, name, value) { var quote = value.charAt(0) return name + "=" + quote + "javascript:void(0)" + quote// jshint ignore:line }) } } return a.replace(ron, " ").replace(/\s+/g, " ") //移除onXXX事件 }) }, escape: function(str) { //将字符串经过 str 转义得到适合在页面中显示的内容, 例如替换 < 为 < return String(str). replace(/&/g, '&'). replace(rsurrogate, function(value) { var hi = value.charCodeAt(0) var low = value.charCodeAt(1) return '&#' + (((hi - 0xD800) * 0x400) + (low - 0xDC00) + 0x10000) + ';' }). replace(rnoalphanumeric, function(value) { return '&#' + value.charCodeAt(0) + ';' }). replace(//g, '>') }, currency: function(amount, symbol, fractionSize) { return (symbol || "\u00a5") + numberFormat(amount, isFinite(fractionSize) ? fractionSize : 2) }, number: numberFormat } /* 'yyyy': 4 digit representation of year (e.g. AD 1 => 0001, AD 2010 => 2010) 'yy': 2 digit representation of year, padded (00-99). (e.g. AD 2001 => 01, AD 2010 => 10) 'y': 1 digit representation of year, e.g. (AD 1 => 1, AD 199 => 199) 'MMMM': Month in year (January-December) 'MMM': Month in year (Jan-Dec) 'MM': Month in year, padded (01-12) 'M': Month in year (1-12) 'dd': Day in month, padded (01-31) 'd': Day in month (1-31) 'EEEE': Day in Week,(Sunday-Saturday) 'EEE': Day in Week, (Sun-Sat) 'HH': Hour in day, padded (00-23) 'H': Hour in day (0-23) 'hh': Hour in am/pm, padded (01-12) 'h': Hour in am/pm, (1-12) 'mm': Minute in hour, padded (00-59) 'm': Minute in hour (0-59) 'ss': Second in minute, padded (00-59) 's': Second in minute (0-59) 'a': am/pm marker 'Z': 4 digit (+sign) representation of the timezone offset (-1200-+1200) format string can also be one of the following predefined localizable formats: 'medium': equivalent to 'MMM d, y h:mm:ss a' for en_US locale (e.g. Sep 3, 2010 12:05:08 pm) 'short': equivalent to 'M/d/yy h:mm a' for en_US locale (e.g. 9/3/10 12:05 pm) 'fullDate': equivalent to 'EEEE, MMMM d,y' for en_US locale (e.g. Friday, September 3, 2010) 'longDate': equivalent to 'MMMM d, y' for en_US locale (e.g. September 3, 2010 'mediumDate': equivalent to 'MMM d, y' for en_US locale (e.g. Sep 3, 2010) 'shortDate': equivalent to 'M/d/yy' for en_US locale (e.g. 9/3/10) 'mediumTime': equivalent to 'h:mm:ss a' for en_US locale (e.g. 12:05:08 pm) 'shortTime': equivalent to 'h:mm a' for en_US locale (e.g. 12:05 pm) */ new function() {// jshint ignore:line function toInt(str) { return parseInt(str, 10) || 0 } function padNumber(num, digits, trim) { var neg = "" if (num < 0) { neg = '-' num = -num } num = "" + num while (num.length < digits) num = "0" + num if (trim) num = num.substr(num.length - digits) return neg + num } function dateGetter(name, size, offset, trim) { return function(date) { var value = date["get" + name]() if (offset > 0 || value > -offset) value += offset if (value === 0 && offset === -12) { value = 12 } return padNumber(value, size, trim) } } function dateStrGetter(name, shortForm) { return function(date, formats) { var value = date["get" + name]() var get = (shortForm ? ("SHORT" + name) : name).toUpperCase() return formats[get][value] } } function timeZoneGetter(date) { var zone = -1 * date.getTimezoneOffset() var paddedZone = (zone >= 0) ? "+" : "" paddedZone += padNumber(Math[zone > 0 ? "floor" : "ceil"](zone / 60), 2) + padNumber(Math.abs(zone % 60), 2) return paddedZone } //取得上午下午 function ampmGetter(date, formats) { return date.getHours() < 12 ? formats.AMPMS[0] : formats.AMPMS[1] } var DATE_FORMATS = { yyyy: dateGetter("FullYear", 4), yy: dateGetter("FullYear", 2, 0, true), y: dateGetter("FullYear", 1), MMMM: dateStrGetter("Month"), MMM: dateStrGetter("Month", true), MM: dateGetter("Month", 2, 1), M: dateGetter("Month", 1, 1), dd: dateGetter("Date", 2), d: dateGetter("Date", 1), HH: dateGetter("Hours", 2), H: dateGetter("Hours", 1), hh: dateGetter("Hours", 2, -12), h: dateGetter("Hours", 1, -12), mm: dateGetter("Minutes", 2), m: dateGetter("Minutes", 1), ss: dateGetter("Seconds", 2), s: dateGetter("Seconds", 1), sss: dateGetter("Milliseconds", 3), EEEE: dateStrGetter("Day"), EEE: dateStrGetter("Day", true), a: ampmGetter, Z: timeZoneGetter } var rdateFormat = /((?:[^yMdHhmsaZE']+)|(?:'(?:[^']|'')*')|(?:E+|y+|M+|d+|H+|h+|m+|s+|a|Z))(.*)/ var raspnetjson = /^\/Date\((\d+)\)\/$/ filters.date = function(date, format) { var locate = filters.date.locate, text = "", parts = [], fn, match format = format || "mediumDate" format = locate[format] || format if (typeof date === "string") { if (/^\d+$/.test(date)) { date = toInt(date) } else if (raspnetjson.test(date)) { date = +RegExp.$1 } else { var trimDate = date.trim() var dateArray = [0, 0, 0, 0, 0, 0, 0] var oDate = new Date(0) //取得年月日 trimDate = trimDate.replace(/^(\d+)\D(\d+)\D(\d+)/, function(_, a, b, c) { var array = c.length === 4 ? [c, a, b] : [a, b, c] dateArray[0] = toInt(array[0]) //年 dateArray[1] = toInt(array[1]) - 1 //月 dateArray[2] = toInt(array[2]) //日 return "" }) var dateSetter = oDate.setFullYear var timeSetter = oDate.setHours trimDate = trimDate.replace(/[T\s](\d+):(\d+):?(\d+)?\.?(\d)?/, function(_, a, b, c, d) { dateArray[3] = toInt(a) //小时 dateArray[4] = toInt(b) //分钟 dateArray[5] = toInt(c) //秒 if (d) { //毫秒 dateArray[6] = Math.round(parseFloat("0." + d) * 1000) } return "" }) var tzHour = 0 var tzMin = 0 trimDate = trimDate.replace(/Z|([+-])(\d\d):?(\d\d)/, function(z, symbol, c, d) { dateSetter = oDate.setUTCFullYear timeSetter = oDate.setUTCHours if (symbol) { tzHour = toInt(symbol + c) tzMin = toInt(symbol + d) } return "" }) dateArray[3] -= tzHour dateArray[4] -= tzMin dateSetter.apply(oDate, dateArray.slice(0, 3)) timeSetter.apply(oDate, dateArray.slice(3)) date = oDate } } if (typeof date === "number") { date = new Date(date) } if (avalon.type(date) !== "date") { return } while (format) { match = rdateFormat.exec(format) if (match) { parts = parts.concat(match.slice(1)) format = parts.pop() } else { parts.push(format) format = null } } parts.forEach(function(value) { fn = DATE_FORMATS[value] text += fn ? fn(date, locate) : value.replace(/(^'|'$)/g, "").replace(/''/g, "'") }) return text } var locate = { AMPMS: { 0: "上午", 1: "下午" }, DAY: { 0: "星期日", 1: "星期一", 2: "星期二", 3: "星期三", 4: "星期四", 5: "星期五", 6: "星期六" }, MONTH: { 0: "1月", 1: "2月", 2: "3月", 3: "4月", 4: "5月", 5: "6月", 6: "7月", 7: "8月", 8: "9月", 9: "10月", 10: "11月", 11: "12月" }, SHORTDAY: { "0": "周日", "1": "周一", "2": "周二", "3": "周三", "4": "周四", "5": "周五", "6": "周六" }, fullDate: "y年M月d日EEEE", longDate: "y年M月d日", medium: "yyyy-M-d H:mm:ss", mediumDate: "yyyy-M-d", mediumTime: "H:mm:ss", "short": "yy-M-d ah:mm", shortDate: "yy-M-d", shortTime: "ah:mm" } locate.SHORTMONTH = locate.MONTH filters.date.locate = locate }// jshint ignore:line /********************************************************************* * AMD加载器 * **********************************************************************/ //https://www.devbridge.com/articles/understanding-amd-requirejs/ //http://maxogden.com/nested-dependencies.html var modules = avalon.modules = { "domReady!": { exports: avalon, state: 3 }, "avalon": { exports: avalon, state: 4 } } //Object(modules[id]).state拥有如下值 // undefined 没有定义 // 1(send) 已经发出请求 // 2(loading) 已经被执行但还没有执行完成,在这个阶段define方法会被执行 // 3(loaded) 执行完毕,通过onload/onreadystatechange回调判定,在这个阶段checkDeps方法会执行 // 4(execute) 其依赖也执行完毕, 值放到exports对象上,在这个阶段fireFactory方法会执行 modules.exports = modules.avalon var otherRequire = window.require var otherDefine = window.define var innerRequire plugins.loader = function (builtin) { var flag = innerRequire && builtin window.require = flag ? innerRequire : otherRequire window.define = flag ? innerRequire.define : otherDefine } new function () { // jshint ignore:line var loadings = [] //正在加载中的模块列表 var factorys = [] //放置define方法的factory函数 var rjsext = /\.js$/i var rquery = /(\?[^#]*)$/ function makeRequest(name, config) { //1. 去掉资源前缀 var res = "js" name = name.replace(/^(\w+)\!/, function (a, b) { res = b return "" }) if (res === "ready") { log("debug: ready!已经被废弃,请使用domReady!") res = "domReady" } //2. 去掉querystring, hash var query = "" name = name.replace(rquery, function (a) { query = a return "" }) //3. 去掉扩展名 var suffix = "." + res var ext = /js|css/.test(suffix) ? suffix : "" name = name.replace(/\.[a-z0-9]+$/g, function (a) { if (a === suffix) { ext = a return "" } else { return a } }) var req = avalon.mix({ query: query, ext: ext, res: res, name: name, toUrl: toUrl }, config) req.toUrl(name) return req } function fireRequest(req) { var name = req.name var res = req.res //1. 如果该模块已经发出请求,直接返回 var module = modules[name] var urlNoQuery = name && req.urlNoQuery if (module && module.state >= 1) { return name } module = modules[urlNoQuery] if (module && module.state >= 3) { innerRequire(module.deps || [], module.factory, urlNoQuery) return urlNoQuery } if (name && !module) { module = modules[urlNoQuery] = { id: urlNoQuery, state: 1 //send } var wrap = function (obj) { resources[res] = obj//标识该插件已注册 obj.load(name, req, function (a) { if (arguments.length && a !== void 0) { module.exports = a } module.state = 4 checkDeps() }) } if (!resources[res]) {//如果资源插件不存在,先加载插件 innerRequire([res], wrap) } else { wrap(resources[res])//使用资源插件的load方法加载我们的模块 } } return name ? urlNoQuery : res + "!" } //核心API之一 require var requireQueue = [] var isUserFirstRequire = false innerRequire = avalon.require = function (array, factory, parentUrl, defineConfig) { if (!isUserFirstRequire) { requireQueue.push(avalon.slice(arguments)) if (arguments.length <= 2) { isUserFirstRequire = true var queue = requireQueue.splice(0, requireQueue.length), args while (args = queue.shift()) { innerRequire.apply(null, args) } } return } if (!Array.isArray(array)) { avalon.error("require方法的第一个参数应为数组 " + array) } var deps = [] // 放置所有依赖项的完整路径 var uniq = {} var id = parentUrl || "callback" + setTimeout("1") // jshint ignore:line defineConfig = defineConfig || {} defineConfig.baseUrl = kernel.baseUrl var isBuilt = !!defineConfig.built if (parentUrl) { defineConfig.parentUrl = parentUrl.substr(0, parentUrl.lastIndexOf("/")) defineConfig.mapUrl = parentUrl.replace(rjsext, "") } if (isBuilt) { var req = makeRequest(defineConfig.defineName, defineConfig) id = req.urlNoQuery } else { array.forEach(function (name) { var req = makeRequest(name, defineConfig) var url = fireRequest(req) //加载资源,并返回该资源的完整地址 if (url) { if (!uniq[url]) { deps.push(url) uniq[url] = "司徒正美" //去重 } } }) } var module = modules[id] if (!module || module.state !== 4) { modules[id] = { id: id, deps: isBuilt ? array.concat() : deps, factory: factory || noop, state: 3 } } if (!module) { //如果此模块是定义在另一个JS文件中, 那必须等该文件加载完毕, 才能放到检测列队中 loadings.push(id) } checkDeps() } //核心API之二 require innerRequire.define = function (name, deps, factory) { //模块名,依赖列表,模块本身 if (typeof name !== "string") { factory = deps deps = name name = "anonymous" } if (!Array.isArray(deps)) { factory = deps deps = [] } var config = { built: !isUserFirstRequire, //用r.js打包后,所有define方法会放到require方法之前() defineName: name } var args = [deps, factory, config] factory.require = function (url) { args.splice(2, 0, url) if (modules[url]) { modules[url].state = 3 //loaded var isCycle = false try { isCycle = checkCycle(modules[url].deps, url) } catch (e) { } if (isCycle) { avalon.error(url + "模块与之前的模块存在循环依赖,请不要直接用script标签引入" + url + "模块") } } delete factory.require //释放内存 innerRequire.apply(null, args) //0,1,2 --> 1,2,0 } //根据标准,所有遵循W3C标准的浏览器,script标签会按标签的出现顺序执行。 //老的浏览器中,加载也是按顺序的:一个文件下载完成后,才开始下载下一个文件。 //较新的浏览器中(IE8+ 、FireFox3.5+ 、Chrome4+ 、Safari4+),为了减小请求时间以优化体验, //下载可以是并行的,但是执行顺序还是按照标签出现的顺序。 //但如果script标签是动态插入的, 就未必按照先请求先执行的原则了,目测只有firefox遵守 //唯一比较一致的是,IE10+及其他标准浏览器,一旦开始解析脚本, 就会一直堵在那里,直接脚本解析完毕 //亦即,先进入loading阶段的script标签(模块)必然会先进入loaded阶段 var url = config.built ? "unknown" : getCurrentScript() if (url) { var module = modules[url] if (module) { module.state = 2 } factory.require(url) } else { //合并前后的safari,合并后的IE6-9走此分支 factorys.push(factory) } } //核心API之三 require.config(settings) innerRequire.config = kernel //核心API之四 define.amd 标识其符合AMD规范 innerRequire.define.amd = modules //==========================对用户配置项进行再加工========================== var allpaths = kernel["orig.paths"] = {} var allmaps = kernel["orig.map"] = {} var allpackages = kernel["packages"] = [] var allargs = kernel["orig.args"] = {} avalon.mix(plugins, { paths: function (hash) { avalon.mix(allpaths, hash) kernel.paths = makeIndexArray(allpaths) }, map: function (hash) { avalon.mix(allmaps, hash) var list = makeIndexArray(allmaps, 1, 1) avalon.each(list, function (_, item) { item.val = makeIndexArray(item.val) }) kernel.map = list }, packages: function (array) { array = array.concat(allpackages) var uniq = {} var ret = [] for (var i = 0, pkg; pkg = array[i++]; ) { pkg = typeof pkg === "string" ? { name: pkg } : pkg var name = pkg.name if (!uniq[name]) { var url = joinPath(pkg.location || name, pkg.main || "main") url = url.replace(rjsext, "") ret.push(pkg) uniq[name] = pkg.location = url pkg.reg = makeMatcher(name) } } kernel.packages = ret.sort() }, urlArgs: function (hash) { if (typeof hash === "string") { hash = { "*": hash } } avalon.mix(allargs, hash) kernel.urlArgs = makeIndexArray(allargs, 1) }, baseUrl: function (url) { if (!isAbsUrl(url)) { var baseElement = head.getElementsByTagName("base")[0] if (baseElement) { head.removeChild(baseElement) } var node = DOC.createElement("a") node.href = url url = getFullUrl(node, "href") if (baseElement) { head.insertBefore(baseElement, head.firstChild) } } if (url.length > 3) kernel.baseUrl = url }, shim: function (obj) { for (var i in obj) { var value = obj[i] if (Array.isArray(value)) { value = obj[i] = { deps: value } } if (!value.exportsFn && (value.exports || value.init)) { value.exportsFn = makeExports(value) } } kernel.shim = obj } }) //==============================内部方法================================= function checkCycle(deps, nick) { //检测是否存在循环依赖 for (var i = 0, id; id = deps[i++]; ) { if (modules[id].state !== 4 && (id === nick || checkCycle(modules[id].deps, nick))) { return true } } } function checkFail(node, onError, fuckIE) { var id = trimQuery(node.src) //检测是否死链 node.onload = node.onreadystatechange = node.onerror = null if (onError || (fuckIE && modules[id] && !modules[id].state)) { setTimeout(function () { head.removeChild(node) node = null // 处理旧式IE下的循环引用问题 }) log("debug: 加载 " + id + " 失败" + onError + " " + (!modules[id].state)) } else { return true } } function checkDeps() { //检测此模块的依赖是否都执行完毕,是则执行自身 loop: for (var i = loadings.length, id; id = loadings[--i]; ) { var obj = modules[id], deps = obj.deps if (!deps) continue for (var j = 0, key; key = deps[j]; j++) { if (Object(modules[key]).state !== 4) { continue loop } } //如果deps是空对象或者其依赖的模块的状态都是4 if (obj.state !== 4) { loadings.splice(i, 1) //必须先移除再安装,防止在IE下DOM树建完后手动刷新页面,会多次执行它 fireFactory(obj.id, obj.deps, obj.factory) checkDeps() //如果成功,则再执行一次,以防有些模块就差本模块没有执行 } } } var rreadyState = /complete|loaded/ function loadJS(url, id, callback) { //通过script节点加载目标模块 var node = DOC.createElement("script") var supportLoad = "onload" in node var onEvent = supportLoad ? "onload" : "onreadystatechange" function onload() { var factory = factorys.pop()//处理safari早期版本 factory && factory.require(id) if (callback) { callback() } if (checkFail(node, false, !supportLoad)) { log("debug: 已成功加载 " + url) id && loadings.push(id) checkDeps() } } node[onEvent] = supportLoad ? onload : function () { if (rreadyState.test(node.readyState)) { onload() } } node.onerror = function () { checkFail(node, true) } node.className = subscribers //让getCurrentScript只处理类名为subscribers的script节点 node.src = url //插入到head的第一个节点前,防止IE6下head标签没闭合前使用appendChild抛错 head.insertBefore(node, head.firstChild) //chrome下第二个参数不能为null log("debug: 正准备加载 " + url) //更重要的是IE6下可以收窄getCurrentScript的寻找范围 } var resources = innerRequire.plugins = { //三大常用资源插件 js!, css!, text!, domReady! domReady: { load: noop }, js: { load: function (name, req, onLoad) { var url = req.url var id = req.urlNoQuery var shim = kernel.shim[name.replace(rjsext, "")] if (shim) { //shim机制 innerRequire(shim.deps || [], function () { var args = avalon.slice(arguments) loadJS(url, id, function () { onLoad(shim.exportsFn ? shim.exportsFn.apply(0, args) : void 0) }) }) } else { loadJS(url, id) } } }, css: { load: function (name, req, onLoad) { var url = req.url var node = DOC.createElement("link") node.rel = "stylesheet" node.href = url head.insertBefore(node, head.firstChild) log("debug: 已成功加载 " + url) onLoad() } }, text: { load: function (name, req, onLoad) { var url = req.url var xhr = getXHR() xhr.onreadystatechange = function () { if (xhr.readyState === 4) { var status = xhr.status; if (status > 399 && status < 600) { avalon.error(url + " 对应资源不存在或没有开启 CORS") } else { log("debug: 已成功加载 " + url) onLoad(xhr.responseText) } } } var time = "_=" + (new Date() - 0) var _url = url.indexOf("?") === -1 ? url + "?" + time : url + "&" + time xhr.open("GET", _url, true) if ("withCredentials" in xhr) { //这是处理跨域 xhr.withCredentials = true } xhr.setRequestHeader("X-Requested-With", "XMLHttpRequest") //告诉后端这是AJAX请求 xhr.send() log("debug: 正准备加载 " + url) } } } innerRequire.checkDeps = checkDeps function trimQuery(url) { return (url || "").replace(rquery, "") } function isAbsUrl(path) { //http://stackoverflow.com/questions/10687099/how-to-test-if-a-url-string-is-absolute-or-relative return /^(?:[a-z]+:)?\/\//i.test(String(path)) } function getFullUrl(node, src) { return "1" [0] ? node[src] : node.getAttribute(src, 4) } function getCurrentScript() { // inspireb by https://github.com/samyk/jiagra/blob/master/jiagra.js var stack try { a.b.c() //强制报错,以便捕获e.stack } catch (e) { //safari5的sourceURL,firefox的fileName,它们的效果与e.stack不一样 stack = e.stack if (!stack && window.opera) { //opera 9没有e.stack,但有e.Backtrace,但不能直接取得,需要对e对象转字符串进行抽取 stack = (String(e).match(/of linked script \S+/g) || []).join(" ") } } if (stack) { /**e.stack最后一行在所有支持的浏览器大致如下: *chrome23: * at http://113.93.50.63/data.js:4:1 *firefox17: *@http://113.93.50.63/query.js:4 *opera12:http://www.oldapps.com/opera.php?system=Windows_XP *@http://113.93.50.63/data.js:4 *IE10: * at Global code (http://113.93.50.63/data.js:4:1) * //firefox4+ 可以用document.currentScript */ stack = stack.split(/[@ ]/g).pop() //取得最后一行,最后一个空格或@之后的部分 stack = stack[0] === "(" ? stack.slice(1, -1) : stack.replace(/\s/, "") //去掉换行符 return trimQuery(stack.replace(/(:\d+)?:\d+$/i, "")) //去掉行号与或许存在的出错字符起始位置 } var nodes = head.getElementsByTagName("script") //只在head标签中寻找 for (var i = nodes.length, node; node = nodes[--i]; ) { if (node.className === subscribers && node.readyState === "interactive") { var url = getFullUrl(node, "src") return node.className = trimQuery(url) } } } var rcallback = /^callback\d+$/ function fireFactory(id, deps, factory) { var module = Object(modules[id]) module.state = 4 for (var i = 0, array = [], d; d = deps[i++]; ) { if (d === "exports") { var obj = module.exports || (module.exports = {}) array.push(obj) } else { array.push(modules[d].exports) } } try { var ret = factory.apply(window, array) } catch (e) { log("执行[" + id + "]模块的factory抛错: ", e) } if (ret !== void 0) { modules[id].exports = ret } modules[id].state = 4 if (rcallback.test(id)) { delete modules[id] } delete module.factory return ret } function toUrl(id) { if (id.indexOf(this.res + "!") === 0) { id = id.slice(this.res.length + 1) //处理define("css!style",[], function(){})的情况 } var url = id //1. 是否命中paths配置项 var usePath = 0 var baseUrl = this.baseUrl var rootUrl = this.parentUrl || baseUrl eachIndexArray(id, kernel.paths, function (value, key) { url = url.replace(key, value) usePath = 1 }) //2. 是否命中packages配置项 if (!usePath) { eachIndexArray(id, kernel.packages, function (value, key, item) { url = url.replace(item.name, item.location) }) } //3. 是否命中map配置项 if (this.mapUrl) { eachIndexArray(this.mapUrl, kernel.map, function (array) { eachIndexArray(url, array, function (mdValue, mdKey) { url = url.replace(mdKey, mdValue) rootUrl = baseUrl }) }) } var ext = this.ext if (ext && usePath && url.slice(-ext.length) === ext) { url = url.slice(0, -ext.length) } //4. 转换为绝对路径 if (!isAbsUrl(url)) { rootUrl = this.built || /^\w/.test(url) ? baseUrl : rootUrl url = joinPath(rootUrl, url) } //5. 还原扩展名,query var urlNoQuery = url + ext url = urlNoQuery + this.query urlNoQuery = url.replace(rquery, function (a) { this.query = a return "" }) //6. 处理urlArgs eachIndexArray(id, kernel.urlArgs, function (value) { url += (url.indexOf("?") === -1 ? "?" : "&") + value; }) this.url = url return this.urlNoQuery = urlNoQuery } function makeIndexArray(hash, useStar, part) { //创建一个经过特殊算法排好序的数组 var index = hash2array(hash, useStar, part) index.sort(descSorterByName) return index } function makeMatcher(prefix) { return new RegExp('^' + prefix + '(/|$)') } function makeExports(value) { return function () { var ret if (value.init) { ret = value.init.apply(window, arguments) } return ret || (value.exports && getGlobal(value.exports)) } } function hash2array(hash, useStar, part) { var array = []; for (var key in hash) { if (ohasOwn.call(hash, key)) { var item = { name: key, val: hash[key] } array.push(item) item.reg = key === "*" && useStar ? /^/ : makeMatcher(key) if (part && key !== "*") { item.reg = new RegExp('\/' + key.replace(/^\//, "") + '(/|$)') } } } return array } function eachIndexArray(moduleID, array, matcher) { array = array || [] for (var i = 0, el; el = array[i++]; ) { if (el.reg.test(moduleID)) { matcher(el.val, el.name, el) return false } } } // 根据元素的name项进行数组字符数逆序的排序函数 function descSorterByName(a, b) { var aaa = a.name var bbb = b.name if (bbb === "*") { return -1 } if (aaa === "*") { return 1 } return bbb.length - aaa.length } var rdeuce = /\/\w+\/\.\./ function joinPath(a, b) { if (a.charAt(a.length - 1) !== "/") { a += "/" } if (b.slice(0, 2) === "./") { //相对于兄弟路径 return a + b.slice(2) } if (b.slice(0, 2) === "..") { //相对于父路径 a += b while (rdeuce.test(a)) { a = a.replace(rdeuce, "") } return a } if (b.slice(0, 1) === "/") { return a + b.slice(1) } return a + b } function getGlobal(value) { if (!value) { return value } var g = window value.split(".").forEach(function (part) { g = g[part] }) return g } var mainNode = DOC.scripts[DOC.scripts.length - 1] var dataMain = mainNode.getAttribute("data-main") if (dataMain) { plugins.baseUrl(dataMain) var href = kernel.baseUrl kernel.baseUrl = href.slice(0, href.lastIndexOf("/") + 1) loadJS(href.replace(rjsext, "") + ".js") } else { var loaderUrl = trimQuery(getFullUrl(mainNode, "src")) kernel.baseUrl = loaderUrl.slice(0, loaderUrl.lastIndexOf("/") + 1) } } // jshint ignore:line /********************************************************************* * DOMReady * **********************************************************************/ var readyList = [], isReady var fireReady = function (fn) { isReady = true var require = avalon.require if (require && require.checkDeps) { modules["domReady!"].state = 4 require.checkDeps() } while (fn = readyList.shift()) { fn(avalon) } } function doScrollCheck() { try { //IE下通过doScrollCheck检测DOM树是否建完 root.doScroll("left") fireReady() } catch (e) { setTimeout(doScrollCheck) } } if (DOC.readyState === "complete") { setTimeout(fireReady) //如果在domReady之外加载 } else if (W3C) { DOC.addEventListener("DOMContentLoaded", fireReady) } else { DOC.attachEvent("onreadystatechange", function () { if (DOC.readyState === "complete") { fireReady() } }) try { var isTop = window.frameElement === null } catch (e) {} if (root.doScroll && isTop && window.external) { //fix IE iframe BUG doScrollCheck() } } avalon.bind(window, "load", fireReady) avalon.ready = function (fn) { if (!isReady) { readyList.push(fn) } else { fn(avalon) } } avalon.config({ loader: true }) avalon.ready(function () { avalon.scan(DOC.body) }) // Register as a named AMD module, since avalon can be concatenated with other // files that may use define, but not via a proper concatenation script that // understands anonymous AMD modules. A named AMD is safest and most robust // way to register. Lowercase avalon is used because AMD module names are // derived from file names, and Avalon is normally delivered in a lowercase // file name. Do this after creating the global so that if an AMD module wants // to call noConflict to hide this version of avalon, it will work. // Note that for maximum portability, libraries that are not avalon should // declare themselves as anonymous modules, and avoid setting a global if an // AMD loader is present. avalon is a special case. For more information, see // https://github.com/jrburke/requirejs/wiki/Updating-existing-libraries#wiki-anon if (typeof define === "function" && define.amd) { define("avalon", [], function() { return avalon }) } // Map over avalon in case of overwrite var _avalon = window.avalon avalon.noConflict = function(deep) { if (deep && window.avalon === avalon) { window.avalon = _avalon } return avalon } // Expose avalon identifiers, even in AMD // and CommonJS for browser emulators if (noGlobal === void 0) { window.avalon = avalon } return avalon }));