/*==================================================
*
* @authors yutent (yutent@doui.cc)
* @date 2017-03-21 21:05:57
* support IE10+ and other browsers
*
==================================================*/
/*********************************************************************
* 全局变量及方法 *
**********************************************************************/
var expose = generateID()
//http://stackoverflow.com/questions/7290086/javascript-use-strict-and-nicks-find-global-function
var DOC = window.document
var head = DOC.head //HEAD元素
head.insertAdjacentHTML(
'afterBegin',
''
)
var ifGroup = head.firstChild
function log() {
// http://stackoverflow.com/questions/8785624/how-to-safely-wrap-console-log
console.log.apply(console, arguments)
}
/**
* Creates a new object without a prototype. This object is useful for lookup without having to
* guard against prototypically inherited properties via hasOwnProperty.
*
* Related micro-benchmarks:
* - http://jsperf.com/object-create2
* - http://jsperf.com/proto-map-lookup/2
* - http://jsperf.com/for-in-vs-object-keys2
*/
function createMap() {
return Object.create(null)
}
var subscribers = '$' + expose
var nullObject = {} //作用类似于noop,只用于代码防御,千万不要在它上面添加属性
var rword = /[^, ]+/g //切割字符串为一个个小块,以空格或豆号分开它们,结合replace实现字符串的forEach
var rw20g = /\w+/g
var rsvg = /^\[object SVG\w*Element\]$/
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 yuaFragment = DOC.createDocumentFragment()
var cinerator = DOC.createElement('div')
var class2type = {
'[object Boolean]': 'boolean',
'[object Number]': 'number',
'[object String]': 'string',
'[object Function]': 'function',
'[object Array]': 'array',
'[object Date]': 'date',
'[object RegExp]': 'regexp',
'[object Object]': 'object',
'[object Error]': 'error',
'[object AsyncFunction]': 'asyncfunction',
'[object Promise]': 'promise',
'[object Generator]': 'generator',
'[object GeneratorFunction]': 'generatorfunction'
}
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
}
function generateID(mark) {
mark = (mark && mark + '-') || 'yua-'
return (
mark +
Date.now().toString(16) +
'-' +
Math.random()
.toString(16)
.slice(2, 6)
)
}
yua = function(el) {
//创建jQuery式的无new 实例化结构
return new yua.init(el)
}
/*视浏览器情况采用最快的异步回调*/
yua.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('yua')
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
/*********************************************************************
* yua的静态方法定义区 *
**********************************************************************/
yua.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
}
/*判定是否是一个朴素的javascript对象(Object),不是DOM对象,不是BOM对象,不是自定义类的实例*/
yua.isPlainObject = function(obj) {
// 简单的 typeof obj === "object"检测,会致使用isPlainObject(window)在opera下通不过
return (
serialize.call(obj) === '[object Object]' &&
Object.getPrototypeOf(obj) === oproto
)
}
var VMODELS = (yua.vmodels = {}) //所有vmodel都储存在这里
yua.init = function(source) {
if (yua.isPlainObject(source)) {
var $id = source.$id,
vm
if (!$id) {
log('warning: vm必须指定$id')
}
vm = modelFactory(source)
vm.$id = $id
return (VMODELS[$id] = vm)
} else {
this[0] = this.element = source
}
}
yua.fn = yua.prototype = yua.init.prototype
//与jQuery.extend方法,可用于浅拷贝,深拷贝
yua.mix = yua.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' && yua.type(target) !== 'function') {
target = {}
}
//如果只有一个参数,那么新成员添加于mix所在的对象上
if (i === length) {
target = this
i--
}
for (; i < length; i++) {
//只处理非空参数
if ((options = arguments[i]) != null) {
for (name in options) {
src = target[name]
copy = options[name]
// 防止环引用
if (target === copy) {
continue
}
if (
deep &&
copy &&
(yua.isPlainObject(copy) ||
(copyIsArray = Array.isArray(copy)))
) {
if (copyIsArray) {
copyIsArray = false
clone = src && Array.isArray(src) ? src : []
} else {
clone = src && yua.isPlainObject(src) ? src : {}
}
target[name] = yua.mix(deep, clone, copy)
} else if (copy !== void 0) {
target[name] = copy
}
}
}
}
return target
}
/*-----------------部分ES6的JS实现 start---------------*/
if (!Object.assign) {
Object.defineProperty(Object, 'assign', {
enumerable: false,
value: function(target, first) {
'use strict'
if (target === undefined || target === null)
throw new TypeError('Can not convert first argument to object')
var to = Object(target)
for (var i = 0, len = arguments.length; i < len; i++) {
var next = arguments[i]
if (next === undefined || next === null) continue
var keys = Object.keys(Object(next))
for (var j = 0, n = keys.length; j < n; j++) {
var key = keys[j]
var desc = Object.getOwnPropertyDescriptor(next, key)
if (desc !== undefined && desc.enumerable)
to[key] = next[key]
}
}
return to
}
})
}
if (!Array.from) {
Object.defineProperty(Array, 'from', {
enumerable: false,
value: (function() {
var toStr = Object.prototype.toString
var isCallable = function(fn) {
return (
typeof fn === 'function' ||
toStr.call(fn) === '[object Function]'
)
}
var toInt = function(val) {
var num = val - 0
if (isNaN(num)) return 0
if (num === 0 || isFinite(num)) return num
return (num > 0 ? 1 : -1) * Math.floor(Math.abs(num))
}
var maxInt = Math.pow(2, 53) - 1
var toLen = function(val) {
var len = toInt(val)
return Math.min(Math.max(len, 0), maxInt)
}
return function(arrLike) {
var _this = this
var items = Object(arrLike)
if (arrLike === null)
throw new TypeError(
'Array.from requires an array-like object - not null or undefined'
)
var mapFn = arguments.length > 1 ? arguments[1] : undefined
var other
if (mapFn !== undefined) {
if (!isCallable(mapFn))
throw new TypeError(
'Array.from: when provided, the second argument must be a function'
)
if (arguments.length > 2) other = arguments[2]
}
var len = toLen(items.length)
var arr = isCallable(_this)
? Object(new _this(len))
: new Array(len)
var k = 0
var kVal
while (k < len) {
kVal = items[k]
if (mapFn)
arr[k] =
other === 'undefined'
? mapFn(kVal, k)
: mapFn.call(other, kVal, k)
else arr[k] = kVal
k++
}
arr.length = len
return arr
}
})()
})
}
// 判断数组是否包含指定元素
if (!Array.prototype.includes) {
Object.defineProperty(Array.prototype, 'includes', {
value: function(val) {
for (var i in this) {
if (this[i] === val) return true
}
return false
},
enumerable: false
})
}
//类似于Array 的splice方法
if (!String.prototype.splice) {
Object.defineProperty(String.prototype, 'splice', {
value: function(start, len, fill) {
var length = this.length,
argLen = arguments.length
fill = fill === undefined ? '' : fill
if (argLen < 1) {
return this
}
//处理负数
if (start < 0) {
if (Math.abs(start) >= length) start = 0
else start = length + start
}
if (argLen === 1) {
return this.slice(0, start)
} else {
len -= 0
var strl = this.slice(0, start),
strr = this.slice(start + len)
return strl + fill + strr
}
},
enumerable: false
})
}
if (!Date.prototype.getFullWeek) {
//获取当天是本年度第几周
Object.defineProperty(Date.prototype, 'getFullWeek', {
value: function() {
var thisYear = this.getFullYear(),
that = new Date(thisYear, 0, 1),
firstDay = that.getDay(),
numsOfToday = (this - that) / 86400000
return Math.ceil((numsOfToday + firstDay) / 7)
},
enumerable: false
})
//获取当天是本月第几周
Object.defineProperty(Date.prototype, 'getWeek', {
value: function() {
var today = this.getDate(),
thisMonth = this.getMonth(),
thisYear = this.getFullYear(),
firstDay = new Date(thisYear, thisMonth, 1).getDay()
return Math.ceil((today + firstDay) / 7)
},
enumerable: false
})
}
if (!Date.isDate) {
Object.defineProperty(Date, 'isDate', {
value: function(obj) {
return typeof obj === 'object' && obj.getTime ? true : false
},
enumerable: false
})
}
//时间格式化
if (!Date.prototype.format) {
Object.defineProperty(Date.prototype, 'format', {
value: function(str) {
str = str || 'Y-m-d H:i:s'
var week = ['Mon', 'Tue', 'Wed', 'Thu', 'Fri', 'Sat', 'Sun'],
dt = {
fullyear: this.getFullYear(),
year: this.getYear(),
fullweek: this.getFullWeek(),
week: this.getWeek(),
month: this.getMonth() + 1,
date: this.getDate(),
day: week[this.getDay()],
hours: this.getHours(),
minutes: this.getMinutes(),
seconds: this.getSeconds()
},
re
dt.g = dt.hours > 12 ? dt.hours - 12 : dt.hours
re = {
Y: dt.fullyear,
y: dt.year,
m: dt.month < 10 ? '0' + dt.month : dt.month,
n: dt.month,
d: dt.date < 10 ? '0' + dt.date : dt.date,
j: dt.date,
H: dt.hours < 10 ? '0' + dt.hours : dt.hours,
h: dt.g < 10 ? '0' + dt.g : dt.g,
G: dt.hours,
g: dt.g,
i: dt.minutes < 10 ? '0' + dt.minutes : dt.minutes,
s: dt.seconds < 10 ? '0' + dt.seconds : dt.seconds,
W: dt.fullweek,
w: dt.week,
D: dt.day
}
for (var i in re) {
str = str.replace(new RegExp(i, 'g'), re[i])
}
return str
},
enumerable: false
})
}
/*-----------------部分ES6的JS实现 ending---------------*/
yua.mix({
rword: rword,
subscribers: subscribers,
version: '1.0.0',
log: log,
ui: {}, //仅用于存放组件版本信息等
slice: function(nodes, start, end) {
return aslice.call(nodes, start, end)
},
noop: noop,
/*如果不用Error对象封装一下,str在控制台下可能会乱码*/
error: function(str, e) {
throw new (e || Error)(str) // jshint ignore:line
},
/* yua.range(10)
=> [0, 1, 2, 3, 4, 5, 6, 7, 8, 9]
yua.range(1, 11)
=> [1, 2, 3, 4, 5, 6, 7, 8, 9, 10]
yua.range(0, 30, 5)
=> [0, 5, 10, 15, 20, 25]
yua.range(0, -10, -1)
=> [0, -1, -2, -3, -4, -5, -6, -7, -8, -9]
yua.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
},
deepCopy: toJson,
eventHooks: {},
/*绑定事件*/
bind: function(el, type, fn, phase) {
var hooks = yua.eventHooks
type = type.split(',')
yua.each(type, function(i, t) {
t = t.trim()
var hook = hooks[t]
if (typeof hook === 'object') {
type = hook.type || type
phase = hook.phase || !!phase
fn = hook.fix ? hook.fix(el, fn) : fn
}
el.addEventListener(t, fn, phase)
})
return fn
},
/*卸载事件*/
unbind: function(el, type, fn, phase) {
var hooks = yua.eventHooks
type = type.split(',')
fn = fn || noop
yua.each(type, function(i, t) {
t = t.trim()
var hook = hooks[t]
if (typeof hook === 'object') {
type = hook.type || type
phase = hook.phase || !!phase
}
el.removeEventListener(t, fn, phase)
})
},
/*读写删除元素节点的样式*/
css: function(node, name, value) {
if (node instanceof yua) {
node = node[0]
}
var prop = /[_-]/.test(name) ? camelize(name) : name,
fn
name = yua.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) && !yua.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
}
}
}
}
},
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 yua.Array.removeAt(target, index)
return false
}
},
/**
* [ls localStorage操作]
* @param {[type]} name [键名]
* @param {[type]} val [键值,为空时删除]
* @return
*/
ls: function(name, val) {
if (!window.localStorage)
return log('该浏览器不支持本地储存localStorage')
if (this.type(name) === 'object') {
for (var i in name) {
localStorage.setItem(i, name[i])
}
return
}
switch (arguments.length) {
case 1:
return localStorage.getItem(name)
default:
if (
(this.type(val) == 'string' && val.trim() === '') ||
val === null
) {
localStorage.removeItem(name)
return
}
if (this.type(val) !== 'object' && this.type(val) !== 'array') {
localStorage.setItem(name, val.toString())
} else {
localStorage.setItem(name, JSON.stringify(val))
}
}
},
/**
* [cookie cookie 操作 ]
* @param name [cookie名]
* @param value [cookie值]
* @param {[json]} opt [有效期,域名,路径等]
* @return {[boolean]} [读取时返回对应的值,写入时返回true]
*/
cookie: function(name, value, opt) {
if (arguments.length > 1) {
if (!name) return
//设置默认的参数
opt = opt || {}
opt = this.mix(
{ expires: '', path: '/', domain: document.domain, secure: '' },
opt
)
if (!value) {
document.cookie =
encodeURIComponent(name) +
'=; expires=Thu, 01 Jan 1970 00:00:00 GMT; domain=' +
opt.domain +
'; path=' +
opt.path
return true
}
if (opt.expires) {
switch (opt.expires.constructor) {
case Number:
opt.expires =
opt.expires === Infinity
? '; expires=Fri, 31 Dec 9999 23:59:59 GMT'
: '; max-age=' + opt.expires
break
case String:
opt.expires = '; expires=' + opt.expires
break
case Date:
opt.expires = '; expires=' + opt.expires.toUTCString()
break
}
}
document.cookie =
encodeURIComponent(name) +
'=' +
encodeURIComponent(value) +
opt.expires +
'; domain=' +
opt.domain +
'; path=' +
opt.path +
'; ' +
opt.secure
return true
} else {
if (!name) {
var keys = document.cookie
.replace(
/((?:^|\s*;)[^\=]+)(?=;|$)|^\s*|\s*(?:\=[^;]*)?(?:\1|$)/g,
''
)
.split(/\s*(?:\=[^;]*)?;\s*/)
for (var i = 0, len = keys.length; i < len; i++) {
keys[i] = decodeURIComponent(keys[i])
}
return keys
}
return (
decodeURIComponent(
document.cookie.replace(
new RegExp(
'(?:(?:^|.*;)\\s*' +
encodeURIComponent(name).replace(
/[\-\.\+\*]/g,
'\\$&'
) +
'\\s*\\=\\s*([^;]*).*$)|^.*$'
),
'$1'
)
) || null
)
}
},
//获取url的参数
search: function(key) {
key += ''
var uri = location.search
if (!key || !uri) return null
uri = uri.slice(1)
uri = uri.split('&')
var obj = {}
for (var i in uri) {
var item = uri[i]
var tmp = item.split('=')
tmp[1] = tmp.length < 2 ? null : tmp[1]
tmp[1] = decodeURIComponent(tmp[1])
if (obj.hasOwnProperty(tmp[0])) {
if (typeof obj[tmp[0]] === 'object') {
obj[tmp[0]].push(tmp[1])
} else {
obj[tmp[0]] = [obj[tmp[0]]]
obj[tmp[0]].push(tmp[1])
}
} else {
obj[tmp[0]] = tmp[1]
}
}
return obj.hasOwnProperty(key) ? obj[key] : null
},
//复制文本到粘贴板
copy: function(txt) {
if (!DOC.queryCommandSupported || !DOC.queryCommandSupported('copy'))
return log('该浏览器不支持复制到粘贴板')
var ta = DOC.createElement('textarea')
ta.textContent = txt
ta.style.position = 'fixed'
ta.style.bottom = '-1000px'
DOC.body.appendChild(ta)
ta.select()
try {
DOC.execCommand('copy')
} catch (err) {
log('复制到粘贴板失败')
}
DOC.body.removeChild(ta)
}
})
var bindingHandlers = (yua.bindingHandlers = {})
var bindingExecutors = (yua.bindingExecutors = {})
var directives = (yua.directives = {})
yua.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 && typeof obj === 'object') {
var n = obj.length,
str = serialize.call(obj)
if (/(Array|List|Collection|Map|Arguments)\]$/.test(str)) {
return true
} else if (str === '[object Object]' && n === n >>> 0) {
return true //由于ecma262v5能修改对象属性的enumerable,因此不能用propertyIsEnumerable来判定了
}
}
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 底层补丁 *
**********************************************************************/
//safari5+是把contains方法放在Element.prototype上而不是Node.prototype
if (!DOC.contains) {
Node.prototype.contains = function(arg) {
return !!(this.compareDocumentPosition(arg) & 16)
}
}
yua.contains = function(root, el) {
try {
while ((el = el.parentNode)) if (el === root) return true
return false
} catch (e) {
return false
}
}
if (window.SVGElement) {
var svgns = 'http://www.w3.org/2000/svg'
var svg = DOC.createElementNS(svgns, 'svg')
svg.innerHTML = ''
if (!rsvg.test(svg.firstChild)) {
// #409
/* jshint ignore:start */
function enumerateNode(node, targetNode) {
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()
)
// copy attrs
ap.forEach.call(el.attributes, function(attr) {
svg.setAttribute(attr.name, attr.value)
})
// 递归处理子节点
enumerateNode(el, svg)
targetNode.appendChild(svg)
}
}
}
}
/* jshint ignore:end */
Object.defineProperties(SVGElement.prototype, {
outerHTML: {
//IE9-11,firefox不支持SVG元素的innerHTML,outerHTML属性
enumerable: true,
configurable: true,
get: function() {
return new XMLSerializer().serializeToString(this)
},
set: function(html) {
var tagName = this.tagName.toLowerCase(),
par = this.parentNode,
frag = yua.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 (yua.clearHTML) {
yua.clearHTML(this)
var frag = yua.parseHTML(html)
enumerateNode(frag, this)
}
}
}
})
}
}
//========================= event binding ====================
var eventHooks = yua.eventHooks
//针对firefox, chrome修正mouseenter, mouseleave(chrome30+)
if (!('onmouseenter' in root)) {
yua.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
yua.each(
{
AnimationEvent: 'animationend',
WebKitAnimationEvent: 'webkitAnimationEnd'
},
function(construct, fixType) {
if (window[construct] && !eventHooks.animationend) {
eventHooks.animationend = {
type: fixType
}
}
}
)
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 */
eventHooks.mousewheel = {
type: 'wheel',
fix: function(elem, fn) {
return function(e) {
e.wheelDeltaY = e.wheelDelta = e.deltaY > 0 ? -120 : 120
e.wheelDeltaX = 0
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') {
yua.mix(kernel[p], val)
} else {
kernel[p] = val
}
}
return this
}
yua.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 + '|\\s:') //此处有疑问
}
}
kernel.plugins = plugins
kernel.plugins['interpolate'](['{{', '}}'])
kernel.async = 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
yua.Array.ensure(queue, binding)
} else {
yua.injectBinding(binding)
}
if (backup) {
binding.handler = backup
}
} else if (!binding.oneTime) {
yua.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('.')
args = (args && args.concat([path, key])) || [path, key]
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()
}
}
//一些不需要被监听的属性
var $$skipArray = oneObject(
'$id,$watch,$fire,$events,$model,$skipArray,$active,$pathname,$up,$ups,$track,$accessors'
)
//如果浏览器不支持ecma262v5的Object.defineProperties或者存在BUG,比如IE8
//标准浏览器使用__defineGetter__, __defineSetter__实现
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
if (typeof value === 'function') {
value = { get: value, set: noop }
}
if (typeof value.set !== 'function') {
value.set = noop
}
accessors[key] = {
get: function() {
return (old = value.get.call(this))
},
set: function(x) {
var older = old,
newer
value.set.call(this, x)
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
if (typeof value === 'function') {
value = { get: value, set: noop }
}
if (typeof value.set !== 'function') {
value.set = noop
}
accessors[key] = {
get: function() {
return (old = value.get.call(this))
},
set: function(x) {
var older = old,
newer
value.set.call(this, x)
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 = Object.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 yua.vmodels) {
var v = yua.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 (yua.isPlainObject(obj)) {
if (old && typeof old === 'object') {
var keys = Object.keys(obj)
var keys2 = Object.keys(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
}
}
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)
})
}
Object.defineProperty(array, '$model', $modelDescriptor)
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) {
Object.defineProperty(host, name, {
value: value,
writable: true,
enumerable: false,
configurable: true
})
}
function toJson(val) {
var xtype = yua.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 (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
}
/*********************************************************************
* 监控数组(: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个元素
if (Array.isArray(all)) {
for (var i = this.length - 1; i >= 0; i--) {
if (all.indexOf(this[i]) !== -1) {
_splice.call(this.$track, i, 1)
_splice.call(this, i, 1)
}
}
} else if (typeof all === 'function') {
for (i = this.length - 1; i >= 0; i--) {
var el = this[i]
if (all(el, 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
}
yua.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
}
}
} 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 = yua.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 [yua.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 && yua.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 = (yua.$$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) {
yua.Array.remove(data.list, data)
delete disposeQueue[data.uuid]
}
continue
}
if (shouldDispose(data.element)) {
//如果它的虚拟DOM不在VTree上或其属性不在VM上
disposeQueue.splice(i, 1)
yua.Array.remove(data.list, data)
disposeData(data)
//yua会在每次全量更新时,比较上次执行时间,
//假若距离上次有半秒,就会发起一次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) : !yua.contains(root, el)
}
/************************************************************************
* HTML处理(parseHTML, innerHTML, clearHTML) *
*************************************************************************/
//parseHTML的辅助变量
var tagHooks = new function() {
// jshint ignore:line
yua.mix(this, {
option: DOC.createElement('select'),
thead: DOC.createElement('table'),
td: DOC.createElement('tr'),
area: DOC.createElement('map'),
tr: DOC.createElement('tbody'),
col: DOC.createElement('colgroup'),
legend: DOC.createElement('fieldset'),
_default: DOC.createElement('div'),
g: DOC.createElementNS('http://www.w3.org/2000/svg', 'svg')
})
this.optgroup = this.option
this.tbody = this.tfoot = this.colgroup = this.caption = this.thead
this.th = this.td
}() // jshint ignore:line
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:]+)/
var rxhtml = /<(?!area|br|col|embed|hr|img|input|link|meta|param)(([\w:]+)[^>]*)\/>/gi
var scriptTypes = oneObject([
'',
'text/javascript',
'text/ecmascript',
'application/ecmascript',
'application/javascript'
])
var script = DOC.createElement('script')
var rhtml = /<|?\w+;/
yua.parseHTML = function(html) {
var fragment = yuaFragment.cloneNode(false)
if (typeof html !== 'string') {
return fragment
}
if (!rhtml.test(html)) {
fragment.appendChild(DOC.createTextNode(html))
return fragment
}
html = html.replace(rxhtml, '<$1>$2>').trim()
var tag = (rtagName.exec(html) || ['', ''])[1].toLowerCase(),
//取得其标签名
wrapper = tagHooks[tag] || tagHooks._default,
firstChild
wrapper.innerHTML = html
var els = wrapper.getElementsByTagName('script')
if (els.length) {
//使用innerHTML生成的script节点不会发出请求与执行text属性
for (var i = 0, el; (el = els[i++]); ) {
if (scriptTypes[el.type]) {
var neo = script.cloneNode(false) //FF不能省略参数
ap.forEach.call(el.attributes, function(attr) {
neo.setAttribute(attr.name, attr.value)
}) // jshint ignore:line
neo.text = el.text
el.parentNode.replaceChild(neo, el)
}
}
}
while ((firstChild = wrapper.firstChild)) {
// 将wrapper上的节点转移到文档碎片上!
fragment.appendChild(firstChild)
}
return fragment
}
yua.innerHTML = function(node, html) {
var a = this.parseHTML(html)
this.clearHTML(node).appendChild(a)
}
yua.clearHTML = function(node) {
node.textContent = ''
while (node.firstChild) {
node.removeChild(node.firstChild)
}
return node
}
/*********************************************************************
* yua的原型方法定义区 *
**********************************************************************/
function hyphen(target) {
//转换为连字符线风格
return target.replace(/([a-z\d])([A-Z]+)/g, '$1-$2').toLowerCase()
}
function camelize(target) {
//转换为驼峰风格
if (target.indexOf('-') < 0 && target.indexOf('_') < 0) {
return target //提前判断,提高getStyle等的效率
}
return target.replace(/[-_][^-_]/g, function(match) {
return match.charAt(1).toUpperCase()
})
}
'add,remove'.replace(rword, function(method) {
yua.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) {
el.classList[method](c)
})
}
return this
}
})
yua.fn.mix({
hasClass: function(cls) {
var el = this[0] || {} //IE10+, chrome8+, firefox3.6+, safari5.1+,opera11.5+支持classList,chrome24+,firefox26+支持classList2.0
return el.nodeType === 1 && el.classList.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 (yua.isPlainObject(name)) {
for (var i in name) {
yua.css(this, i, name[i])
}
} else {
var ret = yua.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 += yua.css(offsetParent[0], 'borderTopWidth', true)
parentOffset.left += yua.css(
offsetParent[0],
'borderLeftWidth',
true
)
// Subtract offsetParent scroll positions
parentOffset.top -= offsetParent.scrollTop()
parentOffset.left -= offsetParent.scrollLeft()
}
return {
top:
offset.top -
parentOffset.top -
yua.css(elem, 'marginTop', true),
left:
offset.left -
parentOffset.left -
yua.css(elem, 'marginLeft', true)
}
},
offsetParent: function() {
var offsetParent = this[0].offsetParent
while (offsetParent && yua.css(offsetParent, 'position') === 'static') {
offsetParent = offsetParent.offsetParent
}
return yua(offsetParent || root)
},
bind: function(type, fn, phase) {
if (this[0]) {
//此方法不会链
return yua.bind(this[0], type, fn, phase)
}
},
unbind: function(type, fn, phase) {
if (this[0]) {
yua.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
}
})
if (root.dataset) {
yua.fn.data = function(name, val) {
name = name && camelize(name)
var dataset = this[0].dataset
switch (arguments.length) {
case 2:
dataset[name] = val
return this
case 1:
val = dataset[name]
return parseData(val)
case 0:
var ret = createMap()
for (name in dataset) {
ret[name] = parseData(dataset[name])
}
return ret
}
}
}
yua.parseJSON = JSON.parse
var rbrace = /(?:\{[\s\S]*\}|\[[\s\S]*\])$/
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) ? JSON.parse(data) : data
} catch (e) {}
return data
}
yua.fireDom = function(elem, type, opts) {
var hackEvent = DOC.createEvent('Events')
hackEvent.initEvent(type, true, true)
yua.mix(hackEvent, opts)
elem.dispatchEvent(hackEvent)
}
yua.each(
{
scrollLeft: 'pageXOffset',
scrollTop: 'pageYOffset'
},
function(method, prop) {
yua.fn[method] = function(val) {
var node = this[0] || {},
win = getWindow(node),
top = method === 'scrollTop'
if (!arguments.length) {
return win ? win[prop] : node[method]
} else {
if (win) {
win.scrollTo(!top ? val : win[prop], top ? val : win[prop])
} else {
node[method] = val
}
}
}
}
)
function getWindow(node) {
return node.window && node.document
? node
: node.nodeType === 9 ? node.defaultView : false
}
//=============================css相关==================================
var cssHooks = (yua.cssHooks = createMap())
var prefixes = ['', '-webkit-', '-moz-', '-ms-'] //去掉opera-15的支持
var cssMap = {
float: 'cssFloat'
}
yua.cssNumber = oneObject(
'animationIterationCount,animationIterationCount,columnCount,order,flex,flexGrow,flexShrink,fillOpacity,fontWeight,lineHeight,opacity,orphans,widows,zIndex,zoom'
)
yua.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) {
node.style[name] = value
}
cssHooks['@:get'] = function(node, name) {
if (!node || !node.style) {
throw new Error('getComputedStyle要求传入一个节点 ' + node)
}
var ret,
computed = getComputedStyle(node)
if (computed) {
ret =
name === 'filter' ? computed.getPropertyValue(name) : computed[name]
if (ret === '') {
ret = node.style[name] //其他浏览器需要我们手动取内联样式
}
}
return ret
}
cssHooks['opacity:get'] = function(node) {
var ret = cssHooks['@:get'](node, 'opacity')
return ret === '' ? '1' : ret
}
'top,left'.replace(rword, function(name) {
cssHooks[name + ':get'] = function(node) {
var computed = cssHooks['@:get'](node, name)
return /px$/.test(computed)
? computed
: yua(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
var styles = getComputedStyle(node, null)
if (rdisplayswap.test(styles['display'])) {
var obj = {
node: node
}
for (var name in cssShow) {
obj[name] = styles[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 +
yua.css(node, 'margin' + which[0], true) +
yua.css(node, 'margin' + which[1], true)
)
}
if (boxSizing < 0) {
// padding-box -2
ret =
ret -
yua.css(node, 'border' + which[0] + 'Width', true) -
yua.css(node, 'border' + which[1] + 'Width', true)
}
if (boxSizing === -4) {
// content-box -4
ret =
ret -
yua.css(node, 'padding' + which[0], true) -
yua.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
}
yua.fn[method] = function(value) {
//会忽视其display
var node = this[0]
if (arguments.length === 0) {
if (node.setTimeout) {
//取得窗口尺寸,IE9后可以用node.innerWidth /innerHeight代替
return node['inner' + name]
}
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)
}
}
yua.fn['inner' + name] = function() {
return cssHooks[method + ':get'](this[0], void 0, -2)
}
yua.fn['outer' + name] = function(includeMargin) {
return cssHooks[method + ':get'](
this[0],
void 0,
includeMargin === true ? 2 : 0
)
}
})
yua.fn.offset = function() {
//取得距离页面左右角的坐标
var node = this[0]
try {
var rect = node.getBoundingClientRect()
// Make sure element is not hidden (display: none) or disconnected
// https://github.com/jquery/jquery/pull/2043/files#r23981494
if (rect.width || rect.height || node.getClientRects().length) {
var doc = node.ownerDocument
var root = doc.documentElement
var win = doc.defaultView
return {
top: rect.top + win.pageYOffset - root.clientTop,
left: rect.left + win.pageXOffset - root.clientLeft
}
}
} catch (e) {
return {
left: 0,
top: 0
}
}
}
//=============================val相关=======================
function getValType(elem) {
var ret = elem.tagName.toLowerCase()
return ret === 'input' && /checkbox|radio/.test(elem.type) ? 'checked' : ret
}
var valHooks = {
'select:get': function(node, value) {
var option,
options = node.options,
index = node.selectedIndex,
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 = option.value
if (one) {
return value
}
//收集所有selected值组成数组返回
values.push(value)
}
}
return values
},
'select:set': function(node, values, optionSet) {
values = [].concat(values) //强制转换为数组
for (var i = 0, el; (el = node.options[i++]); ) {
if ((el.selected = values.indexOf(el.value) > -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]); ) {
var arr = prop.split('.'),
a
var first = arr[0]
while ((a = arr.shift())) {
if (vmodel.hasOwnProperty(a)) {
ret.push(first + prefix + first)
binding.observers.push({
v: vmodel,
p: prop
})
vars.splice(i, 1)
} else {
break
}
}
}
return ret
}
var rproxy = /(\$proxy\$[a-z]+)\-[\-0-9a-f]+$/
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) {
if (dataType === 'duplex') {
var setter = evaluatorPool.get(exprId + 'setter')
binding.setter = setter.apply(setter, binding.args)
}
return (binding.getter = 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 ;")
}
/* jshint ignore:start */
getter = scpCompile(
names.concat("'use strict';\nvar " + assigns.join(',\n') + expr)
)
/* jshint ignore:end */
return evaluatorPool.put(exprId, getter)
}
function normalizeExpr(code) {
var hasExpr = rexpr.test(code) //比如: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
}
}
yua.normalizeExpr = normalizeExpr
yua.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 quote = JSON.stringify
/*********************************************************************
* 扫描系统 *
**********************************************************************/
yua.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('yuactrl') || vmodel.$id
elem.setAttribute('yuactrl', id)
if (vmodel.$events) {
vmodel.$events.expr = elem.tagName + '[yuactrl="' + 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)
yua.injectBinding(binding)
if (binding.getter && binding.element.nodeType === 1) {
//移除数据绑定,防止被二次解析
//chrome使用removeAttributeNode移除不存在的特性节点时会报错 https://github.com/RubyLouvre/yua/issues/99
binding.element.removeAttribute(binding.name)
}
}
bindings.length = 0
}
//https://github.com/RubyLouvre/yua/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 = /:(\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 = /^(:\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 = elem.attributes
var bindings = []
var uniq = {}
for (var i = 0, attr; (attr = attributes[i++]); ) {
var name = attr.name
if (uniq[name]) {
//IE8下:repeat,: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 = ':' + 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,
priority:
(directives[type].priority ||
type.charCodeAt(0) * 10) +
(Number(param.replace(/\D/g, '')) || 0)
}
if (
type === 'html' ||
type === 'text' ||
type === 'attr'
) {
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 === ':if-loop') {
binding.priority += 100
} else if (name === ':attr-value') {
var hasAttrValue = name
}
bindings.push(binding)
}
}
}
}
if (bindings.length) {
bindings.sort(bindingSorter)
if (hasDuplex && hasAttrValue && elem.type === 'text') {
log('warning!一个控件不能同时定义: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 = /^html|include$/
function scanNodeList(elem, vmodels) {
var nodes = yua.slice(elem.childNodes)
scanNodeArray(nodes, vmodels)
}
function scanNodeArray(nodes, vmodels) {
function _delay_component(name) {
setTimeout(function() {
yua.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 widget = isWidget(elem)
if (widget) {
componentQueue.push({
element: elem,
vmodels: vmodels,
name: widget
})
if (yua.components[widget]) {
// log(widget, yua.components)
//确保所有:attr-name扫描完再处理
_delay_component(widget)
}
}
}
scanTag(node, vmodels) //扫描元素节点
if (node.msHasEvent) {
yua.fireDom(node, 'datasetchanged', {
bubble: node.msHasEvent
})
}
break
case 3:
if (rexpr.test(node.nodeValue)) {
scanText(node, vmodels, i) //扫描文本节点
}
break
}
}
}
function scanTag(elem, vmodels, node) {
//扫描顺序 :skip(0) --> :important(1) --> :controller(2) --> :if(10) --> :repeat(100)
//--> :if-loop(110) --> :attr(970) ...--> :each(1400)-->:with(1500)--〉:duplex(2000)垫后
var a = elem.getAttribute(':skip')
var b = elem.getAttributeNode(':important')
var c = elem.getAttributeNode(':controller')
if (typeof a === 'string') {
return
} else if ((node = b || c)) {
var newVmodel = yua.vmodels[node.value]
if (!newVmodel) {
return
}
//把父级VM补上
newVmodel.$up = vmodels[0]
//:important不包含父VM,:controller相反
vmodels = node === b ? [newVmodel] : [newVmodel].concat(vmodels)
elem.removeAttribute(node.name) //removeAttributeNode不会刷新[:controller]样式规则
elem.classList.remove(node.name)
createSignalTower(elem, newVmodel)
}
scanAttr(elem, vmodels) //扫描特性节点
if (newVmodel) {
setTimeout(function() {
newVmodel.$fire(':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) //收集带有插值表达式的文本
}
yuaFragment.appendChild(node)
}
textNode.parentNode.replaceChild(yuaFragment, 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
yua.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 yua.mix.apply(null, arguments)
},
$ready: noop,
$init: noop,
$dispose: noop,
$container: null,
$childReady: noop,
$$template: function(str) {
return str
}
}
yua.components = {}
yua.component = function(name, opts) {
if (opts) {
yua.components[name] = yua.mix({}, componentHooks, opts)
}
for (var i = 0, obj; (obj = componentQueue[i]); i++) {
if (name === obj.name) {
componentQueue.splice(i, 1)
i--
;(function(host, hooks, elem, widget) {
//如果elem已从Document里移除,直接返回
if (!yua.contains(DOC, elem) || elem.msResolved) {
yua.Array.remove(componentQueue, host)
return
}
var dependencies = 1
var global = componentHooks
//===========收集各种配置=======
if (elem.getAttribute(':attr-identifier')) {
//如果还没有解析完,就延迟一下 #1155
return
}
var elemOpts = getOptionsFromTag(elem, host.vmodels)
var vmOpts = getOptionsFromVM(
host.vmodels,
elemOpts.config || host.name
)
var $id =
elemOpts.$id || elemOpts.identifier || generateID(widget)
delete elemOpts.config
delete elemOpts.$id
delete elemOpts.identifier
var componentDefinition = {
$up: host.vmodels[0],
$ups: host.vmodels
}
yua.mix(true, componentDefinition, hooks)
componentDefinition = yua.components[name].$construct.call(
elem,
componentDefinition,
vmOpts,
elemOpts
)
componentDefinition.$refs = {}
componentDefinition.$id = $id
//==========构建VM=========
var keepContainer = componentDefinition.$container
var keepTemplate = componentDefinition.$template
delete componentDefinition.$up
delete componentDefinition.$ups
delete componentDefinition.$slot
delete componentDefinition.$replace
delete componentDefinition.$container
delete componentDefinition.$construct
var vmodel = yua(componentDefinition) || {}
vmodel.$ups = host.vmodels
vmodel.$up = host.vmodels[0]
elem.msResolved = 1 //防止二进扫描此元素
vmodel.$init(vmodel, elem)
global.$init(vmodel, elem)
var nodes = elem.childNodes
if (vmodel.$$template) {
yua.clearHTML(elem)
elem.innerHTML = vmodel.$$template(keepTemplate)
}
// 组件所使用的标签是temlate,所以必须要要用子元素替换掉
var child = elem.content.firstChild
if (!child || serialize.call(child) === '[object Text]') {
var tmpDom = document.createElement('div')
if (child) {
tmpDom.appendChild(child)
}
child = tmpDom
tmpDom = null
}
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) {
yua(elem).addClass(className)
}
//指定了组件的容器的话,则把组件节点转过去
if (keepContainer) {
keepContainer.appendChild(elem)
}
yua.fireDom(elem, 'datasetchanged', {
vm: vmodel,
childReady: 1
})
var children = 0
var removeFn = yua.bind(elem, 'datasetchanged', function(e) {
if (e.childReady) {
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)
yua.unbind(elem, 'datasetchanged', removeFn)
//==================
host.rollback = function() {
try {
vmodel.$dispose(vmodel, elem)
global.$dispose(vmodel, elem)
} catch (e) {}
delete yua.vmodels[vmodel.$id]
}
injectDisposeQueue(host, widgetList)
if (window.chrome) {
elem.addEventListener(
'DOMNodeRemovedFromDocument',
function() {
setTimeout(rejectDisposeQueue)
}
)
}
}
})
scanTag(elem, [vmodel].concat(host.vmodels))
yua.vmodels[vmodel.$id] = vmodel
if (!elem.childNodes.length) {
yua.fireDom(elem, 'datasetchanged', {
vm: vmodel,
childReady: -1
})
} else {
var id2 = setTimeout(function() {
clearTimeout(id2)
yua.fireDom(elem, 'datasetchanged', {
vm: vmodel,
childReady: -1
})
}, 17)
}
})(obj, yua.components[name], obj.element, obj.name) // 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 {}
}
function isWidget(el) {
//如果是组件,则返回组件的名字
var name = el.nodeName.toLowerCase()
if (name === 'template' && el.getAttribute('name')) {
return el.getAttribute('name')
}
return null
}
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 = yua.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 = !!yua(elem).data(
'includeReplace'
))
if (yua(elem).data('includeCache')) {
binding.templateCache = {}
}
binding.start = DOC.createComment(':include')
binding.end = DOC.createComment(':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 obj = val
if (typeof obj === 'object' && obj !== null) {
if (!yua.isPlainObject(obj)) obj = obj.$model
} else {
if (!this.param) return
obj = {}
obj[this.param] = val
}
for (var i in obj) {
if (i === 'style') {
console.error('设置style样式, 请改用 :css指令')
continue
}
if (i === 'href' || i === 'src') {
//处理IE67自动转义的问题
if (!root.hasAttribute) obj[i] = obj[i].replace(/&/g, '&')
elem[i] = obj[i]
//chrome v37- 下embed标签动态设置的src,无法发起请求
if (window.chrome && elem.tagName === 'EMBED') {
var parent = elem.parentNode
var com = document.createComment(':src')
parent.replaceChild(com, elem)
parent.replaceChild(elem, com)
}
} else {
var k = i
//古董IE下,部分属性名字要进行映射
if (!W3C && propMap[k]) k = propMap[k]
if (typeof elem[boolMap[k]] === 'boolean') {
//布尔属性必须使用el.xxx = true|false方式设值
elem[boolMap[k]] = !!obj[i]
//如果为false, IE全系列下相当于setAttribute(xxx, ''),会影响到样式,需要进一步处理
if (!obj[i]) obj[i] = !!obj[i]
}
if (obj[i] === false || obj[i] === null || obj[i] === undefined)
return elem.removeAttribute(k)
//SVG只能使用setAttribute(xxx, yyy), VML只能使用elem.xxx = yyy ,HTML的固有属性必须elem.xxx = yyy
var isInnate = rsvg.test(elem)
? false
: DOC.namespaces && isVML(elem)
? true
: k in elem.cloneNode(false)
if (isInnate) {
elem[k] = obj[i]
} else {
elem.setAttribute(k, obj[i])
}
}
}
}
})
//这几个指令都可以使用插值表达式,如:src="aaa/{{b}}/{{c}}.html"
'title,alt,src,value,css,include,href,data'.replace(rword, function(name) {
directives[name] = attrDir
})
//类名定义, :class="xx:yy" :class="{xx: yy}" :class="xx" :class="{{xx}}"
yua.directive('class', {
init: function(binding) {
var expr = []
if (!/^\{.*\}$/.test(binding.expr)) {
expr = binding.expr.split(':')
expr[1] = (expr[1] && expr[1].trim()) || 'true'
var arr = expr[0].split(/\s+/)
binding.expr =
'{' +
arr
.map(function(it) {
return it + ': ' + expr[1]
})
.join(', ') +
'}'
} else if (/^\{\{.*\}\}$/.test(binding.expr)) {
binding.expr = binding.expr.slice(2, -2)
}
if (binding.type === 'hover' || binding.type === 'active') {
//确保只绑定一次
if (!binding.hasBindEvent) {
var elem = binding.element
var $elem = yua(elem)
var activate = 'mouseenter' //在移出移入时切换类名
var abandon = 'mouseleave'
if (binding.type === 'active') {
//在聚焦失焦中切换类名
elem.tabIndex = elem.tabIndex || -1
activate = 'mousedown'
abandon = 'mouseup'
var fn0 = $elem.bind('mouseleave', function() {
$elem.removeClass(expr[0])
})
}
}
var fn1 = $elem.bind(activate, function() {
$elem.addClass(expr[0])
})
var fn2 = $elem.bind(abandon, function() {
$elem.removeClass(expr[0])
})
binding.rollback = function() {
$elem.unbind('mouseleave', fn0)
$elem.unbind(activate, fn1)
$elem.unbind(abandon, fn2)
}
binding.hasBindEvent = true
}
},
update: function(val) {
if (this.type !== 'class') {
return
}
var obj = val
if (!obj || this.param)
return log(
'class指令语法错误 %c %s="%s"',
'color:#f00',
this.name,
this.expr
)
if (typeof obj === 'string') {
obj = {}
obj[val] = true
}
if (!yua.isPlainObject(obj)) {
obj = obj.$model
}
var $elem = yua(this.element)
for (var i in obj) {
$elem.toggleClass(i, !!obj[i])
}
}
})
'hover,active'.replace(rword, function(name) {
directives[name] = directives['class']
})
//样式定义 :css-width="200"
//:css="{width: 200}"
yua.directive('css', {
init: directives.attr.init,
update: function(val) {
var $elem = yua(this.element)
if (!this.param) {
var obj = val
try {
if (typeof val === 'object') {
if (!yua.isPlainObject(val)) obj = val.$model
} else {
obj = new Function('return ' + val)()
}
for (var i in obj) {
$elem.css(i, obj[i])
}
} catch (err) {
log(
'样式格式错误 %c %s="%s"',
'color:#f00',
this.name,
this.expr
)
}
} else {
$elem.css(this.param, val)
}
}
})
//兼容2种写法 :data-xx="yy", :data="{xx: yy}"
yua.directive('data', {
priority: 100,
update: function(val) {
var obj = val
if (typeof obj === 'object' && obj !== null) {
if (!yua.isPlainObject(obj)) obj = val.$model
for (var i in obj) {
this.element.setAttribute('data-' + i, obj[i])
}
} else {
if (!this.param) return
this.element.setAttribute('data-' + this.param, obj)
}
}
})
/*------ 表单验证 -------*/
var __rules = {}
yua.validate = function(key) {
return (
!__rules[key] ||
__rules[key].every(function(it) {
return it.checked
})
)
}
yua.directive('rule', {
priority: 2010,
init: function(binding) {
if (binding.param && !__rules[binding.param]) {
__rules[binding.param] = []
}
binding.target = __rules[binding.param]
},
update: function(obj) {
var _this = this,
elem = this.element,
ruleID = -1
if (!['INPUT', 'TEXTAREA'].includes(elem.nodeName)) return
if (this.target) {
ruleID = this.target.length
this.target.push({ checked: true })
}
//如果父级元素没有定位属性,则加上相对定位
if (getComputedStyle(elem.parentNode).position === 'static') {
elem.parentNode.style.position = 'relative'
}
var $elem = yua(elem),
ol = elem.offsetLeft + elem.offsetWidth - 50,
ot = elem.offsetTop + elem.offsetHeight + 8,
tips = document.createElement('div')
tips.className = 'do-rule-tips'
tips.style.left = ol + 'px'
tips.style.bottom = ot + 'px'
function checked(ev) {
var txt = '',
val = elem.value
if (obj.require && (val === '' || val === null)) txt = '必填项'
if (!txt && obj.isNumeric)
txt = !isFinite(val) ? '必须为合法数字' : ''
if (!txt && obj.isEmail)
txt = !/^[\w\.\-]+@\w+([\.\-]\w+)*\.\w+$/.test(val)
? 'Email格式错误'
: ''
if (!txt && obj.isPhone)
txt = !/^1[34578]\d{9}$/.test(val) ? '手机格式错误' : ''
if (!txt && obj.isCN)
txt = !/^[\u4e00-\u9fa5]+$/.test(val) ? '必须为纯中文' : ''
if (!txt && obj.exp)
txt = !obj.exp.test(val) ? obj.msg || '格式错误' : ''
if (!txt && obj.maxLen)
txt =
val.length > obj.maxLen
? '长度不得超过' + obj.maxLen + '位'
: ''
if (!txt && obj.minLen)
txt =
val.length < obj.minLen
? '长度不得小于' + obj.minLen + '位'
: ''
if (!txt && obj.hasOwnProperty('max'))
txt = val > obj.max ? '输入值不能大于' + obj.max : ''
if (!txt && obj.hasOwnProperty('min'))
txt = val < obj.min ? '输入值不能小于' + obj.min : ''
if (!txt && obj.eq) {
var eqEl = document.querySelector('#' + obj.eq)
txt = val !== eqEl.value ? obj.msg || '2次值不一致' : ''
}
if (txt) {
if (ev) {
tips.textContent = txt
elem.parentNode.appendChild(tips)
}
//必须是"必填项"才会更新验证状态
if (_this.target && obj.require) {
_this.target[ruleID].checked = false
}
} else {
if (_this.target) {
_this.target[ruleID].checked = true
}
try {
elem.parentNode.removeChild(tips)
} catch (err) {}
}
}
$elem.bind('change,blur', checked)
$elem.bind('focus', function(ev) {
try {
elem.parentNode.removeChild(tips)
} catch (err) {}
})
checked()
}
})
//双工绑定
var rduplexType = /^(?:checkbox|radio)$/
var rduplexParam = /^(?:radio|checked)$/
var rnoduplexInput = /^(file|button|reset|submit|checkbox|radio|range)$/
var duplexBinding = yua.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(':duplex-radio已经更名为:duplex-checked')
name = 'checked'
binding.isChecked = true
binding.xtype = 'radio'
}
if (name === 'bool') {
name = 'boolean'
log(':duplex-bool已经更名为:duplex-boolean')
} else if (name === 'text') {
name = 'string'
log(':duplex-text已经更名为:duplex-string')
}
if (casting[name]) {
hasCast = true
}
yua.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) {
elem.addEventListener(type, callback, false)
var old = binding.rollback
binding.rollback = function() {
elem.yuaSetter = null
yua.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
//防止递归调用形成死循环
//处理中文输入法在minlengh下引发的BUG
if (
composing ||
val === binding.oldValue ||
binding.pipe === null
) {
return
}
var lastValue = binding.pipe(val, binding, 'get')
binding.oldValue = val
binding.setter(lastValue)
callback.call(elem, lastValue)
yua.fireDom(elem, 'change')
}
switch (binding.xtype) {
case 'radio':
bound('click', function() {
var lastValue = binding.pipe(elem.value, binding, 'get')
binding.setter(lastValue)
callback.call(elem, lastValue)
})
break
case 'checkbox':
bound('change', function() {
var method = elem.checked ? 'ensure' : 'remove'
var array = binding.getter.apply(0, binding.vmodels)
if (!Array.isArray(array)) {
log(':duplex应用于checkbox上要对应一个数组')
array = [array]
}
var val = binding.pipe(elem.value, binding, 'get')
yua.Array[method](array, val)
callback.call(elem, array)
})
break
case 'change':
bound('change', updateVModel)
break
case 'input':
bound('input', updateVModel)
bound('keyup', updateVModel)
if (!IEVersion) {
bound('compositionstart', compositionStart)
bound('compositionend', compositionEnd)
bound('DOMAutoComplete', updateVModel)
}
break
case 'select':
bound('change', function() {
var val = yua(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 + ''
yua(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.yuaSetter = updateVModel //#765
watchValueInTimer(function() {
if (root.contains(elem)) {
if (!elem.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 yua.vmodels) {
var v = yua.vmodels[i]
v.$fire('yua-duplex-init', binding)
}
var cpipe = binding.pipe || (binding.pipe = pipe)
cpipe(null, binding, 'init')
this.init = 1
}
switch (this.xtype) {
case 'input':
elem.value = value
break
case 'change':
curValue = this.pipe(value, this, 'set') //fix #673
if (curValue !== this.oldValue) {
var fixCaret = false
if (elem.msFocus) {
try {
var start = elem.selectionStart
var end = elem.selectionEnd
if (start === end) {
var pos = start
fixCaret = true
}
} catch (e) {}
}
elem.value = this.oldValue = curValue
if (fixCaret && !elem.readOnly) {
elem.selectionStart = elem.selectionEnd = pos
}
}
break
case 'radio':
curValue = binding.isChecked
? !!value
: value + '' === elem.value
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 {
yua.fireDom(elem, 'datasetchanged', {
bubble: elem.msHasEvent
})
}
break
}
}
})
function fixNull(val) {
return val == null ? '' : val
}
yua.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 = val - 0
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, e) {
binding.param.replace(rw20g, function(name) {
var hook = yua.duplexHooks[name]
if (hook && typeof hook[action] === 'function') {
val = hook[action](val, binding)
}
})
return val
}
var TimerID,
ribbon = []
yua.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.yuaSetter) {
this.yuaSetter()
}
}
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中 :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 = yua.tick
}
}() // jshint ignore:line
/*-------------动画------------*/
yua.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) {
// 比如 :class/effect="aaa bbb ccc" 的情况
className = text
rightExpr = true
} else {
// 比如 :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 (yua.effects[name]) {
elem.setAttribute('data-effect-driver', 'j')
useAni = true
}
}
if (useAni) {
elem.setAttribute('data-effect-name', name)
}
}
})
yua.effects = {}
yua.effect = function(name, callbacks) {
yua.effects[name] = callbacks
}
var supportTransition = false
var supportAnimation = false
var transitionEndEvent
var animationEndEvent
var transitionDuration = yua.cssName('transition-duration')
var animationDuration = yua.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 && yua(el).addClass(opts.effectClass)
instance.cssEvent =
driver === 't' ? transitionEndEvent : animationEndEvent
}
instance.name = name
instance.callbacks = yua.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树操作,因此必须位于动画之后
yua(el).removeClass(me.cssClass)
} else {
if (me.driver === 'a') {
yua(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延迟触发
yua(el).removeClass(me.cssClass)
}
}
yua(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(yua.slice(arguments)))
},
leave: function(before, after) {
this.actionFun.apply(this, ['leave'].concat(yua.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)
}
}
yua.mix(yua.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
)
}
})
yua.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 = yua.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 = yuaFragment.cloneNode(true)
while (nodes[0]) {
fragment.appendChild(nodes[0])
}
}
nodes = yua.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 {
yua.clearHTML(elem).appendChild(fragment)
}
scanNodeArray(nodes, binding.vmodels)
}
})
yua.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 = yua.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) {
yua.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,可以直接无视
yua.each(elem.querySelectorAll(':required'), function(el) {
elem.required = false
el.setAttribute('_required', 'true')
})
} catch (e) {}
var node = (binding.element = DOC.createComment(':if')),
pos = elem.nextSibling
binding.recoverNode = function() {
binding.recoverNode = null
if (node.parentNode !== par) {
par.insertBefore(node, pos)
binding.keep = elem
}
}
yua.effect.apply(
elem,
0,
function() {
binding.recoverNode = null
if (stamp !== binding.stamp) return
elem.parentNode.replaceChild(node, elem)
binding.keep = elem //元素节点
ifGroup.appendChild(elem)
binding.rollback = function() {
if (elem.parentNode === ifGroup) {
ifGroup.removeChild(elem)
}
}
},
after
)
}
}
}
})
//:important绑定已经在scanTag 方法中实现
var rnoscripts = /(?:[\s\S]+?)<\/noscript>/gim
var rnoscriptText = /([\s\S]+?)<\/noscript>/im
var getXHR = function() {
return new window.XMLHttpRequest() // jshint ignore:line
}
//将所有远程加载的模板,以字符串形式存放到这里
var templatePool = (yua.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 yua.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
}
yua.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 || yua.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
}
}
// 元素退场
yua.effect.remove(
leaveEl,
target,
function() {
if (templateCache) {
// write cache
if (_stamp === binding._stamp)
ifGroup.appendChild(leaveEl)
}
},
binding
)
var enterEl = target,
before = yua.noop,
after = yua.noop
var fragment = getTemplateContainer(binding, val, text)
var nodes = yua.slice(fragment.childNodes)
if (outer && effectClass) {
enterEl = _ele
enterEl.innerHTML = '' // 清空
enterEl.setAttribute(':skip', 'true')
target.insertBefore(enterEl, binding.end.nextSibling) // 插入到bingding.end之后避免被错误的移动
before = function() {
enterEl.insertBefore(fragment, null) // 插入节点
}
after = function() {
binding.recoverNodes = yua.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 = yua.noop
return nodesToFrag(nodes)
}
} else {
before = function() {
//新添加元素的动画
target.insertBefore(fragment, binding.end)
scanNodeArray(nodes, vmodels)
}
}
yua.effect.apply(enterEl, 'enter', before, after)
}
if (!val) return
var el = val
if (typeof el === 'object') {
if (el.nodeType !== 1) return log('include 不支持非DOM对象')
} else {
el = DOC.getElementById(val)
if (!el) {
if (typeof templatePool[val] === 'string') {
yua.nextTick(function() {
scanTemplate(templatePool[val])
})
} else if (Array.isArray(templatePool[val])) {
//#805 防止在循环绑定中发出许多相同的请求
templatePool[val].push(scanTemplate)
} else {
var xhr = getXHR()
xhr.onload = function() {
if (xhr.status !== 200)
return log(
'获取网络资源出错, httpError[' +
xhr.status +
']'
)
var text = xhr.responseText
for (var f = 0, fn; (fn = templatePool[val][f++]); ) {
fn(text)
}
templatePool[val] = text
}
xhr.onerror = function() {
log(':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)
}
return
}
}
yua.nextTick(function() {
scanTemplate(el.value || el.innerText || el.innerHTML)
})
}
})
var rdash = /\(([^)]*)\)/
var onDir = yua.directive('on', {
priority: 3000,
init: function(binding) {
var value = binding.expr
binding.type = 'on'
var eventType = binding.param.replace(/-\d+$/, '') // :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+$/, '') // :on-mousemove-10
if (eventType === 'scan') {
callback.call(elem, {
type: eventType
})
} else if (typeof binding.specialBind === 'function') {
binding.specialBind(elem, callback)
} else {
var removeFn = yua.bind(elem, eventType, callback)
}
binding.rollback = function() {
if (typeof binding.specialUnbind === 'function') {
binding.specialUnbind()
} else {
yua.unbind(elem, eventType, removeFn)
}
}
}
})
yua.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-repeat-sortby',
binding.vmodels
)
var rendered = getBindingCallback(
elem,
'data-repeat-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 = yuaFragment.cloneNode(false)
var parent = elem.parentNode
parent.replaceChild(end, elem)
parent.insertBefore(start, end)
binding.template.appendChild(elem)
binding.element = end
if (rendered) {
var removeFn = yua.bind(parent, 'datasetchanged', function() {
rendered.apply(parent, parent.args)
yua.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 && yuaFragment.cloneNode(false)
var proxies = []
var param = this.param
var retain = yua.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
var kill = elem.previousSibling
var start = binding.start
/*log(kill === start, kill)
while(kill !== start && kill.nodeName !== '#comment'){
parent.removeChild(kill)
kill = elem.previousSibling
}*/
if (clear) {
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 = yua.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回调
yua.fireDom(parent, 'datasetchanged', {
bubble: parent.msHasEvent
})
})
this.enterCount -= 1
}
})
function animateRepeat(nodes, isEnter, binding) {
for (var i = 0, node; (node = nodes[i++]); ) {
if (node.className === binding.effectClass) {
yua.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 = yuaFragment.cloneNode(false)
var last = node
var breakText = last.nodeValue
var staggerIndex = binding && Math.max(+binding.staggerIndex, 0)
var nodes = yua.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() {
yua.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 = yua.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: yua.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]
}
}
/*********************************************************************
* 各种指令 *
**********************************************************************/
//:skip绑定已经在scanTag 方法中实现
yua.directive('text', {
update: function(val) {
var elem = this.element
val = val == null ? '' : val //不在页面上显示undefined null
if (elem.nodeType === 3) {
//绑定在文本节点上
try {
//IE对游离于DOM树外的节点赋值会报错
elem.data = val
} catch (e) {}
} else {
//绑定在特性节点上
elem.textContent = val
}
}
})
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]
}
yua.parseDisplay = parseDisplay
yua.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 (yua(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) {
yua.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 (yua(elem).css('display') === 'none') {
elem.style.display =
binding.display || parseDisplay(elem.nodeName)
}
}
})
} else {
yua.effect.apply(elem, 0, function() {
if (stamp !== binding.stamp) return
elem.style.display = 'none'
})
}
}
})
/*********************************************************************
* 自带过滤器 *
**********************************************************************/
var rscripts = /