|
|
/**
|
|
|
* jQuery WeUI V1.2.1
|
|
|
* By 言川
|
|
|
* http://lihongxun945.github.io/jquery-weui/
|
|
|
*/
|
|
|
/* global $:true */
|
|
|
/* global WebKitCSSMatrix:true */
|
|
|
|
|
|
(function($) {
|
|
|
"use strict";
|
|
|
|
|
|
$.fn.transitionEnd = function(callback) {
|
|
|
var events = ['webkitTransitionEnd', 'transitionend', 'oTransitionEnd', 'MSTransitionEnd', 'msTransitionEnd'],
|
|
|
i, dom = this;
|
|
|
|
|
|
function fireCallBack(e) {
|
|
|
/*jshint validthis:true */
|
|
|
if (e.target !== this) return;
|
|
|
callback.call(this, e);
|
|
|
for (i = 0; i < events.length; i++) {
|
|
|
dom.off(events[i], fireCallBack);
|
|
|
}
|
|
|
}
|
|
|
if (callback) {
|
|
|
for (i = 0; i < events.length; i++) {
|
|
|
dom.on(events[i], fireCallBack);
|
|
|
}
|
|
|
}
|
|
|
return this;
|
|
|
};
|
|
|
|
|
|
$.support = (function() {
|
|
|
var support = {
|
|
|
touch: !!(('ontouchstart' in window) || window.DocumentTouch && document instanceof window.DocumentTouch)
|
|
|
};
|
|
|
return support;
|
|
|
})();
|
|
|
|
|
|
$.touchEvents = {
|
|
|
start: $.support.touch ? 'touchstart' : 'mousedown',
|
|
|
move: $.support.touch ? 'touchmove' : 'mousemove',
|
|
|
end: $.support.touch ? 'touchend' : 'mouseup'
|
|
|
};
|
|
|
|
|
|
$.getTouchPosition = function(e) {
|
|
|
e = e.originalEvent || e; //jquery wrap the originevent
|
|
|
if(e.type === 'touchstart' || e.type === 'touchmove' || e.type === 'touchend') {
|
|
|
return {
|
|
|
x: e.targetTouches[0].pageX,
|
|
|
y: e.targetTouches[0].pageY
|
|
|
};
|
|
|
} else {
|
|
|
return {
|
|
|
x: e.pageX,
|
|
|
y: e.pageY
|
|
|
};
|
|
|
}
|
|
|
};
|
|
|
|
|
|
$.fn.scrollHeight = function() {
|
|
|
return this[0].scrollHeight;
|
|
|
};
|
|
|
|
|
|
$.fn.transform = function(transform) {
|
|
|
for (var i = 0; i < this.length; i++) {
|
|
|
var elStyle = this[i].style;
|
|
|
elStyle.webkitTransform = elStyle.MsTransform = elStyle.msTransform = elStyle.MozTransform = elStyle.OTransform = elStyle.transform = transform;
|
|
|
}
|
|
|
return this;
|
|
|
};
|
|
|
$.fn.transition = function(duration) {
|
|
|
if (typeof duration !== 'string') {
|
|
|
duration = duration + 'ms';
|
|
|
}
|
|
|
for (var i = 0; i < this.length; i++) {
|
|
|
var elStyle = this[i].style;
|
|
|
elStyle.webkitTransitionDuration = elStyle.MsTransitionDuration = elStyle.msTransitionDuration = elStyle.MozTransitionDuration = elStyle.OTransitionDuration = elStyle.transitionDuration = duration;
|
|
|
}
|
|
|
return this;
|
|
|
};
|
|
|
|
|
|
$.getTranslate = function (el, axis) {
|
|
|
var matrix, curTransform, curStyle, transformMatrix;
|
|
|
|
|
|
// automatic axis detection
|
|
|
if (typeof axis === 'undefined') {
|
|
|
axis = 'x';
|
|
|
}
|
|
|
|
|
|
curStyle = window.getComputedStyle(el, null);
|
|
|
if (window.WebKitCSSMatrix) {
|
|
|
// Some old versions of Webkit choke when 'none' is passed; pass
|
|
|
// empty string instead in this case
|
|
|
transformMatrix = new WebKitCSSMatrix(curStyle.webkitTransform === 'none' ? '' : curStyle.webkitTransform);
|
|
|
}
|
|
|
else {
|
|
|
transformMatrix = curStyle.MozTransform || curStyle.OTransform || curStyle.MsTransform || curStyle.msTransform || curStyle.transform || curStyle.getPropertyValue('transform').replace('translate(', 'matrix(1, 0, 0, 1,');
|
|
|
matrix = transformMatrix.toString().split(',');
|
|
|
}
|
|
|
|
|
|
if (axis === 'x') {
|
|
|
//Latest Chrome and webkits Fix
|
|
|
if (window.WebKitCSSMatrix)
|
|
|
curTransform = transformMatrix.m41;
|
|
|
//Crazy IE10 Matrix
|
|
|
else if (matrix.length === 16)
|
|
|
curTransform = parseFloat(matrix[12]);
|
|
|
//Normal Browsers
|
|
|
else
|
|
|
curTransform = parseFloat(matrix[4]);
|
|
|
}
|
|
|
if (axis === 'y') {
|
|
|
//Latest Chrome and webkits Fix
|
|
|
if (window.WebKitCSSMatrix)
|
|
|
curTransform = transformMatrix.m42;
|
|
|
//Crazy IE10 Matrix
|
|
|
else if (matrix.length === 16)
|
|
|
curTransform = parseFloat(matrix[13]);
|
|
|
//Normal Browsers
|
|
|
else
|
|
|
curTransform = parseFloat(matrix[5]);
|
|
|
}
|
|
|
|
|
|
return curTransform || 0;
|
|
|
};
|
|
|
$.requestAnimationFrame = function (callback) {
|
|
|
if (window.requestAnimationFrame) return window.requestAnimationFrame(callback);
|
|
|
else if (window.webkitRequestAnimationFrame) return window.webkitRequestAnimationFrame(callback);
|
|
|
else if (window.mozRequestAnimationFrame) return window.mozRequestAnimationFrame(callback);
|
|
|
else {
|
|
|
return window.setTimeout(callback, 1000 / 60);
|
|
|
}
|
|
|
};
|
|
|
|
|
|
$.cancelAnimationFrame = function (id) {
|
|
|
if (window.cancelAnimationFrame) return window.cancelAnimationFrame(id);
|
|
|
else if (window.webkitCancelAnimationFrame) return window.webkitCancelAnimationFrame(id);
|
|
|
else if (window.mozCancelAnimationFrame) return window.mozCancelAnimationFrame(id);
|
|
|
else {
|
|
|
return window.clearTimeout(id);
|
|
|
}
|
|
|
};
|
|
|
|
|
|
$.fn.join = function(arg) {
|
|
|
return this.toArray().join(arg);
|
|
|
}
|
|
|
})($);
|
|
|
|
|
|
/*===========================
|
|
|
Template7 Template engine
|
|
|
===========================*/
|
|
|
/* global $:true */
|
|
|
/* jshint unused:false */
|
|
|
/* jshint forin:false */
|
|
|
+function ($) {
|
|
|
"use strict";
|
|
|
$.Template7 = $.t7 = (function () {
|
|
|
function isArray(arr) {
|
|
|
return Object.prototype.toString.apply(arr) === '[object Array]';
|
|
|
}
|
|
|
function isObject(obj) {
|
|
|
return obj instanceof Object;
|
|
|
}
|
|
|
function isFunction(func) {
|
|
|
return typeof func === 'function';
|
|
|
}
|
|
|
var cache = {};
|
|
|
function helperToSlices(string) {
|
|
|
var helperParts = string.replace(/[{}#}]/g, '').split(' ');
|
|
|
var slices = [];
|
|
|
var shiftIndex, i, j;
|
|
|
for (i = 0; i < helperParts.length; i++) {
|
|
|
var part = helperParts[i];
|
|
|
if (i === 0) slices.push(part);
|
|
|
else {
|
|
|
if (part.indexOf('"') === 0) {
|
|
|
// Plain String
|
|
|
if (part.match(/"/g).length === 2) {
|
|
|
// One word string
|
|
|
slices.push(part);
|
|
|
}
|
|
|
else {
|
|
|
// Find closed Index
|
|
|
shiftIndex = 0;
|
|
|
for (j = i + 1; j < helperParts.length; j++) {
|
|
|
part += ' ' + helperParts[j];
|
|
|
if (helperParts[j].indexOf('"') >= 0) {
|
|
|
shiftIndex = j;
|
|
|
slices.push(part);
|
|
|
break;
|
|
|
}
|
|
|
}
|
|
|
if (shiftIndex) i = shiftIndex;
|
|
|
}
|
|
|
}
|
|
|
else {
|
|
|
if (part.indexOf('=') > 0) {
|
|
|
// Hash
|
|
|
var hashParts = part.split('=');
|
|
|
var hashName = hashParts[0];
|
|
|
var hashContent = hashParts[1];
|
|
|
if (hashContent.match(/"/g).length !== 2) {
|
|
|
shiftIndex = 0;
|
|
|
for (j = i + 1; j < helperParts.length; j++) {
|
|
|
hashContent += ' ' + helperParts[j];
|
|
|
if (helperParts[j].indexOf('"') >= 0) {
|
|
|
shiftIndex = j;
|
|
|
break;
|
|
|
}
|
|
|
}
|
|
|
if (shiftIndex) i = shiftIndex;
|
|
|
}
|
|
|
var hash = [hashName, hashContent.replace(/"/g,'')];
|
|
|
slices.push(hash);
|
|
|
}
|
|
|
else {
|
|
|
// Plain variable
|
|
|
slices.push(part);
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
return slices;
|
|
|
}
|
|
|
function stringToBlocks(string) {
|
|
|
var blocks = [], i, j, k;
|
|
|
if (!string) return [];
|
|
|
var _blocks = string.split(/({{[^{^}]*}})/);
|
|
|
for (i = 0; i < _blocks.length; i++) {
|
|
|
var block = _blocks[i];
|
|
|
if (block === '') continue;
|
|
|
if (block.indexOf('{{') < 0) {
|
|
|
blocks.push({
|
|
|
type: 'plain',
|
|
|
content: block
|
|
|
});
|
|
|
}
|
|
|
else {
|
|
|
if (block.indexOf('{/') >= 0) {
|
|
|
continue;
|
|
|
}
|
|
|
if (block.indexOf('{#') < 0 && block.indexOf(' ') < 0 && block.indexOf('else') < 0) {
|
|
|
// Simple variable
|
|
|
blocks.push({
|
|
|
type: 'variable',
|
|
|
contextName: block.replace(/[{}]/g, '')
|
|
|
});
|
|
|
continue;
|
|
|
}
|
|
|
// Helpers
|
|
|
var helperSlices = helperToSlices(block);
|
|
|
var helperName = helperSlices[0];
|
|
|
var helperContext = [];
|
|
|
var helperHash = {};
|
|
|
for (j = 1; j < helperSlices.length; j++) {
|
|
|
var slice = helperSlices[j];
|
|
|
if (isArray(slice)) {
|
|
|
// Hash
|
|
|
helperHash[slice[0]] = slice[1] === 'false' ? false : slice[1];
|
|
|
}
|
|
|
else {
|
|
|
helperContext.push(slice);
|
|
|
}
|
|
|
}
|
|
|
|
|
|
if (block.indexOf('{#') >= 0) {
|
|
|
// Condition/Helper
|
|
|
var helperStartIndex = i;
|
|
|
var helperContent = '';
|
|
|
var elseContent = '';
|
|
|
var toSkip = 0;
|
|
|
var shiftIndex;
|
|
|
var foundClosed = false, foundElse = false, foundClosedElse = false, depth = 0;
|
|
|
for (j = i + 1; j < _blocks.length; j++) {
|
|
|
if (_blocks[j].indexOf('{{#') >= 0) {
|
|
|
depth ++;
|
|
|
}
|
|
|
if (_blocks[j].indexOf('{{/') >= 0) {
|
|
|
depth --;
|
|
|
}
|
|
|
if (_blocks[j].indexOf('{{#' + helperName) >= 0) {
|
|
|
helperContent += _blocks[j];
|
|
|
if (foundElse) elseContent += _blocks[j];
|
|
|
toSkip ++;
|
|
|
}
|
|
|
else if (_blocks[j].indexOf('{{/' + helperName) >= 0) {
|
|
|
if (toSkip > 0) {
|
|
|
toSkip--;
|
|
|
helperContent += _blocks[j];
|
|
|
if (foundElse) elseContent += _blocks[j];
|
|
|
}
|
|
|
else {
|
|
|
shiftIndex = j;
|
|
|
foundClosed = true;
|
|
|
break;
|
|
|
}
|
|
|
}
|
|
|
else if (_blocks[j].indexOf('else') >= 0 && depth === 0) {
|
|
|
foundElse = true;
|
|
|
}
|
|
|
else {
|
|
|
if (!foundElse) helperContent += _blocks[j];
|
|
|
if (foundElse) elseContent += _blocks[j];
|
|
|
}
|
|
|
|
|
|
}
|
|
|
if (foundClosed) {
|
|
|
if (shiftIndex) i = shiftIndex;
|
|
|
blocks.push({
|
|
|
type: 'helper',
|
|
|
helperName: helperName,
|
|
|
contextName: helperContext,
|
|
|
content: helperContent,
|
|
|
inverseContent: elseContent,
|
|
|
hash: helperHash
|
|
|
});
|
|
|
}
|
|
|
}
|
|
|
else if (block.indexOf(' ') > 0) {
|
|
|
blocks.push({
|
|
|
type: 'helper',
|
|
|
helperName: helperName,
|
|
|
contextName: helperContext,
|
|
|
hash: helperHash
|
|
|
});
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
return blocks;
|
|
|
}
|
|
|
var Template7 = function (template) {
|
|
|
var t = this;
|
|
|
t.template = template;
|
|
|
|
|
|
function getCompileFn(block, depth) {
|
|
|
if (block.content) return compile(block.content, depth);
|
|
|
else return function () {return ''; };
|
|
|
}
|
|
|
function getCompileInverse(block, depth) {
|
|
|
if (block.inverseContent) return compile(block.inverseContent, depth);
|
|
|
else return function () {return ''; };
|
|
|
}
|
|
|
function getCompileVar(name, ctx) {
|
|
|
var variable, parts, levelsUp = 0, initialCtx = ctx;
|
|
|
if (name.indexOf('../') === 0) {
|
|
|
levelsUp = name.split('../').length - 1;
|
|
|
var newDepth = ctx.split('_')[1] - levelsUp;
|
|
|
ctx = 'ctx_' + (newDepth >= 1 ? newDepth : 1);
|
|
|
parts = name.split('../')[levelsUp].split('.');
|
|
|
}
|
|
|
else if (name.indexOf('@global') === 0) {
|
|
|
ctx = '$.Template7.global';
|
|
|
parts = name.split('@global.')[1].split('.');
|
|
|
}
|
|
|
else if (name.indexOf('@root') === 0) {
|
|
|
ctx = 'ctx_1';
|
|
|
parts = name.split('@root.')[1].split('.');
|
|
|
}
|
|
|
else {
|
|
|
parts = name.split('.');
|
|
|
}
|
|
|
variable = ctx;
|
|
|
for (var i = 0; i < parts.length; i++) {
|
|
|
var part = parts[i];
|
|
|
if (part.indexOf('@') === 0) {
|
|
|
if (i > 0) {
|
|
|
variable += '[(data && data.' + part.replace('@', '') + ')]';
|
|
|
}
|
|
|
else {
|
|
|
variable = '(data && data.' + name.replace('@', '') + ')';
|
|
|
}
|
|
|
}
|
|
|
else {
|
|
|
if (isFinite(part)) {
|
|
|
variable += '[' + part + ']';
|
|
|
}
|
|
|
else {
|
|
|
if (part.indexOf('this') === 0) {
|
|
|
variable = part.replace('this', ctx);
|
|
|
}
|
|
|
else {
|
|
|
variable += '.' + part;
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
|
|
|
return variable;
|
|
|
}
|
|
|
function getCompiledArguments(contextArray, ctx) {
|
|
|
var arr = [];
|
|
|
for (var i = 0; i < contextArray.length; i++) {
|
|
|
if (contextArray[i].indexOf('"') === 0) arr.push(contextArray[i]);
|
|
|
else {
|
|
|
arr.push(getCompileVar(contextArray[i], ctx));
|
|
|
}
|
|
|
}
|
|
|
return arr.join(', ');
|
|
|
}
|
|
|
function compile(template, depth) {
|
|
|
depth = depth || 1;
|
|
|
template = template || t.template;
|
|
|
if (typeof template !== 'string') {
|
|
|
throw new Error('Template7: Template must be a string');
|
|
|
}
|
|
|
var blocks = stringToBlocks(template);
|
|
|
if (blocks.length === 0) {
|
|
|
return function () { return ''; };
|
|
|
}
|
|
|
var ctx = 'ctx_' + depth;
|
|
|
var resultString = '(function (' + ctx + ', data) {\n';
|
|
|
if (depth === 1) {
|
|
|
resultString += 'function isArray(arr){return Object.prototype.toString.apply(arr) === \'[object Array]\';}\n';
|
|
|
resultString += 'function isFunction(func){return (typeof func === \'function\');}\n';
|
|
|
resultString += 'function c(val, ctx) {if (typeof val !== "undefined") {if (isFunction(val)) {return val.call(ctx);} else return val;} else return "";}\n';
|
|
|
}
|
|
|
resultString += 'var r = \'\';\n';
|
|
|
var i, j, context;
|
|
|
for (i = 0; i < blocks.length; i++) {
|
|
|
var block = blocks[i];
|
|
|
// Plain block
|
|
|
if (block.type === 'plain') {
|
|
|
resultString += 'r +=\'' + (block.content).replace(/\r/g, '\\r').replace(/\n/g, '\\n').replace(/'/g, '\\' + '\'') + '\';';
|
|
|
continue;
|
|
|
}
|
|
|
var variable, compiledArguments;
|
|
|
// Variable block
|
|
|
if (block.type === 'variable') {
|
|
|
variable = getCompileVar(block.contextName, ctx);
|
|
|
resultString += 'r += c(' + variable + ', ' + ctx + ');';
|
|
|
}
|
|
|
// Helpers block
|
|
|
if (block.type === 'helper') {
|
|
|
if (block.helperName in t.helpers) {
|
|
|
compiledArguments = getCompiledArguments(block.contextName, ctx);
|
|
|
resultString += 'r += ($.Template7.helpers.' + block.helperName + ').call(' + ctx + ', ' + (compiledArguments && (compiledArguments + ', ')) +'{hash:' + JSON.stringify(block.hash) + ', data: data || {}, fn: ' + getCompileFn(block, depth+1) + ', inverse: ' + getCompileInverse(block, depth+1) + ', root: ctx_1});';
|
|
|
}
|
|
|
else {
|
|
|
if (block.contextName.length > 0) {
|
|
|
throw new Error('Template7: Missing helper: "' + block.helperName + '"');
|
|
|
}
|
|
|
else {
|
|
|
variable = getCompileVar(block.helperName, ctx);
|
|
|
resultString += 'if (' + variable + ') {';
|
|
|
resultString += 'if (isArray(' + variable + ')) {';
|
|
|
resultString += 'r += ($.Template7.helpers.each).call(' + ctx + ', ' + variable + ', {hash:' + JSON.stringify(block.hash) + ', data: data || {}, fn: ' + getCompileFn(block, depth+1) + ', inverse: ' + getCompileInverse(block, depth+1) + ', root: ctx_1});';
|
|
|
resultString += '}else {';
|
|
|
resultString += 'r += ($.Template7.helpers.with).call(' + ctx + ', ' + variable + ', {hash:' + JSON.stringify(block.hash) + ', data: data || {}, fn: ' + getCompileFn(block, depth+1) + ', inverse: ' + getCompileInverse(block, depth+1) + ', root: ctx_1});';
|
|
|
resultString += '}}';
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
resultString += '\nreturn r;})';
|
|
|
return eval.call(window, resultString);
|
|
|
}
|
|
|
t.compile = function (template) {
|
|
|
if (!t.compiled) {
|
|
|
t.compiled = compile(template);
|
|
|
}
|
|
|
return t.compiled;
|
|
|
};
|
|
|
};
|
|
|
Template7.prototype = {
|
|
|
options: {},
|
|
|
helpers: {
|
|
|
'if': function (context, options) {
|
|
|
if (isFunction(context)) { context = context.call(this); }
|
|
|
if (context) {
|
|
|
return options.fn(this, options.data);
|
|
|
}
|
|
|
else {
|
|
|
return options.inverse(this, options.data);
|
|
|
}
|
|
|
},
|
|
|
'unless': function (context, options) {
|
|
|
if (isFunction(context)) { context = context.call(this); }
|
|
|
if (!context) {
|
|
|
return options.fn(this, options.data);
|
|
|
}
|
|
|
else {
|
|
|
return options.inverse(this, options.data);
|
|
|
}
|
|
|
},
|
|
|
'each': function (context, options) {
|
|
|
var ret = '', i = 0;
|
|
|
if (isFunction(context)) { context = context.call(this); }
|
|
|
if (isArray(context)) {
|
|
|
if (options.hash.reverse) {
|
|
|
context = context.reverse();
|
|
|
}
|
|
|
for (i = 0; i < context.length; i++) {
|
|
|
ret += options.fn(context[i], {first: i === 0, last: i === context.length - 1, index: i});
|
|
|
}
|
|
|
if (options.hash.reverse) {
|
|
|
context = context.reverse();
|
|
|
}
|
|
|
}
|
|
|
else {
|
|
|
for (var key in context) {
|
|
|
i++;
|
|
|
ret += options.fn(context[key], {key: key});
|
|
|
}
|
|
|
}
|
|
|
if (i > 0) return ret;
|
|
|
else return options.inverse(this);
|
|
|
},
|
|
|
'with': function (context, options) {
|
|
|
if (isFunction(context)) { context = context.call(this); }
|
|
|
return options.fn(context);
|
|
|
},
|
|
|
'join': function (context, options) {
|
|
|
if (isFunction(context)) { context = context.call(this); }
|
|
|
return context.join(options.hash.delimiter || options.hash.delimeter);
|
|
|
},
|
|
|
'js': function (expression, options) {
|
|
|
var func;
|
|
|
if (expression.indexOf('return')>=0) {
|
|
|
func = '(function(){'+expression+'})';
|
|
|
}
|
|
|
else {
|
|
|
func = '(function(){return ('+expression+')})';
|
|
|
}
|
|
|
return eval.call(this, func).call(this);
|
|
|
},
|
|
|
'js_compare': function (expression, options) {
|
|
|
var func;
|
|
|
if (expression.indexOf('return')>=0) {
|
|
|
func = '(function(){'+expression+'})';
|
|
|
}
|
|
|
else {
|
|
|
func = '(function(){return ('+expression+')})';
|
|
|
}
|
|
|
var condition = eval.call(this, func).call(this);
|
|
|
if (condition) {
|
|
|
return options.fn(this, options.data);
|
|
|
}
|
|
|
else {
|
|
|
return options.inverse(this, options.data);
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
};
|
|
|
var t7 = function (template, data) {
|
|
|
if (arguments.length === 2) {
|
|
|
var instance = new Template7(template);
|
|
|
var rendered = instance.compile()(data);
|
|
|
instance = null;
|
|
|
return (rendered);
|
|
|
}
|
|
|
else return new Template7(template);
|
|
|
};
|
|
|
t7.registerHelper = function (name, fn) {
|
|
|
Template7.prototype.helpers[name] = fn;
|
|
|
};
|
|
|
t7.unregisterHelper = function (name) {
|
|
|
Template7.prototype.helpers[name] = undefined;
|
|
|
delete Template7.prototype.helpers[name];
|
|
|
};
|
|
|
|
|
|
t7.compile = function (template, options) {
|
|
|
var instance = new Template7(template, options);
|
|
|
return instance.compile();
|
|
|
};
|
|
|
|
|
|
t7.options = Template7.prototype.options;
|
|
|
t7.helpers = Template7.prototype.helpers;
|
|
|
return t7;
|
|
|
})();
|
|
|
}($);
|
|
|
|
|
|
/*! Hammer.JS - v2.0.8 - 2016-04-23
|
|
|
* http://hammerjs.github.io/
|
|
|
*
|
|
|
* Copyright (c) 2016 Jorik Tangelder;
|
|
|
* Licensed under the MIT license */
|
|
|
(function(window, document, exportName, undefined) {
|
|
|
'use strict';
|
|
|
|
|
|
var VENDOR_PREFIXES = ['', 'webkit', 'Moz', 'MS', 'ms', 'o'];
|
|
|
var TEST_ELEMENT = document.createElement('div');
|
|
|
|
|
|
var TYPE_FUNCTION = 'function';
|
|
|
|
|
|
var round = Math.round;
|
|
|
var abs = Math.abs;
|
|
|
var now = Date.now;
|
|
|
|
|
|
/**
|
|
|
* set a timeout with a given scope
|
|
|
* @param {Function} fn
|
|
|
* @param {Number} timeout
|
|
|
* @param {Object} context
|
|
|
* @returns {number}
|
|
|
*/
|
|
|
function setTimeoutContext(fn, timeout, context) {
|
|
|
return setTimeout(bindFn(fn, context), timeout);
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* if the argument is an array, we want to execute the fn on each entry
|
|
|
* if it aint an array we don't want to do a thing.
|
|
|
* this is used by all the methods that accept a single and array argument.
|
|
|
* @param {*|Array} arg
|
|
|
* @param {String} fn
|
|
|
* @param {Object} [context]
|
|
|
* @returns {Boolean}
|
|
|
*/
|
|
|
function invokeArrayArg(arg, fn, context) {
|
|
|
if (Array.isArray(arg)) {
|
|
|
each(arg, context[fn], context);
|
|
|
return true;
|
|
|
}
|
|
|
return false;
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* walk objects and arrays
|
|
|
* @param {Object} obj
|
|
|
* @param {Function} iterator
|
|
|
* @param {Object} context
|
|
|
*/
|
|
|
function each(obj, iterator, context) {
|
|
|
var i;
|
|
|
|
|
|
if (!obj) {
|
|
|
return;
|
|
|
}
|
|
|
|
|
|
if (obj.forEach) {
|
|
|
obj.forEach(iterator, context);
|
|
|
} else if (obj.length !== undefined) {
|
|
|
i = 0;
|
|
|
while (i < obj.length) {
|
|
|
iterator.call(context, obj[i], i, obj);
|
|
|
i++;
|
|
|
}
|
|
|
} else {
|
|
|
for (i in obj) {
|
|
|
obj.hasOwnProperty(i) && iterator.call(context, obj[i], i, obj);
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* wrap a method with a deprecation warning and stack trace
|
|
|
* @param {Function} method
|
|
|
* @param {String} name
|
|
|
* @param {String} message
|
|
|
* @returns {Function} A new function wrapping the supplied method.
|
|
|
*/
|
|
|
function deprecate(method, name, message) {
|
|
|
var deprecationMessage = 'DEPRECATED METHOD: ' + name + '\n' + message + ' AT \n';
|
|
|
return function() {
|
|
|
var e = new Error('get-stack-trace');
|
|
|
var stack = e && e.stack ? e.stack.replace(/^[^\(]+?[\n$]/gm, '')
|
|
|
.replace(/^\s+at\s+/gm, '')
|
|
|
.replace(/^Object.<anonymous>\s*\(/gm, '{anonymous}()@') : 'Unknown Stack Trace';
|
|
|
|
|
|
var log = window.console && (window.console.warn || window.console.log);
|
|
|
if (log) {
|
|
|
log.call(window.console, deprecationMessage, stack);
|
|
|
}
|
|
|
return method.apply(this, arguments);
|
|
|
};
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* extend object.
|
|
|
* means that properties in dest will be overwritten by the ones in src.
|
|
|
* @param {Object} target
|
|
|
* @param {...Object} objects_to_assign
|
|
|
* @returns {Object} target
|
|
|
*/
|
|
|
var assign;
|
|
|
if (typeof Object.assign !== 'function') {
|
|
|
assign = function assign(target) {
|
|
|
if (target === undefined || target === null) {
|
|
|
throw new TypeError('Cannot convert undefined or null to object');
|
|
|
}
|
|
|
|
|
|
var output = Object(target);
|
|
|
for (var index = 1; index < arguments.length; index++) {
|
|
|
var source = arguments[index];
|
|
|
if (source !== undefined && source !== null) {
|
|
|
for (var nextKey in source) {
|
|
|
if (source.hasOwnProperty(nextKey)) {
|
|
|
output[nextKey] = source[nextKey];
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
return output;
|
|
|
};
|
|
|
} else {
|
|
|
assign = Object.assign;
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* extend object.
|
|
|
* means that properties in dest will be overwritten by the ones in src.
|
|
|
* @param {Object} dest
|
|
|
* @param {Object} src
|
|
|
* @param {Boolean} [merge=false]
|
|
|
* @returns {Object} dest
|
|
|
*/
|
|
|
var extend = deprecate(function extend(dest, src, merge) {
|
|
|
var keys = Object.keys(src);
|
|
|
var i = 0;
|
|
|
while (i < keys.length) {
|
|
|
if (!merge || (merge && dest[keys[i]] === undefined)) {
|
|
|
dest[keys[i]] = src[keys[i]];
|
|
|
}
|
|
|
i++;
|
|
|
}
|
|
|
return dest;
|
|
|
}, 'extend', 'Use `assign`.');
|
|
|
|
|
|
/**
|
|
|
* merge the values from src in the dest.
|
|
|
* means that properties that exist in dest will not be overwritten by src
|
|
|
* @param {Object} dest
|
|
|
* @param {Object} src
|
|
|
* @returns {Object} dest
|
|
|
*/
|
|
|
var merge = deprecate(function merge(dest, src) {
|
|
|
return extend(dest, src, true);
|
|
|
}, 'merge', 'Use `assign`.');
|
|
|
|
|
|
/**
|
|
|
* simple class inheritance
|
|
|
* @param {Function} child
|
|
|
* @param {Function} base
|
|
|
* @param {Object} [properties]
|
|
|
*/
|
|
|
function inherit(child, base, properties) {
|
|
|
var baseP = base.prototype,
|
|
|
childP;
|
|
|
|
|
|
childP = child.prototype = Object.create(baseP);
|
|
|
childP.constructor = child;
|
|
|
childP._super = baseP;
|
|
|
|
|
|
if (properties) {
|
|
|
assign(childP, properties);
|
|
|
}
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* simple function bind
|
|
|
* @param {Function} fn
|
|
|
* @param {Object} context
|
|
|
* @returns {Function}
|
|
|
*/
|
|
|
function bindFn(fn, context) {
|
|
|
return function boundFn() {
|
|
|
return fn.apply(context, arguments);
|
|
|
};
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* let a boolean value also be a function that must return a boolean
|
|
|
* this first item in args will be used as the context
|
|
|
* @param {Boolean|Function} val
|
|
|
* @param {Array} [args]
|
|
|
* @returns {Boolean}
|
|
|
*/
|
|
|
function boolOrFn(val, args) {
|
|
|
if (typeof val == TYPE_FUNCTION) {
|
|
|
return val.apply(args ? args[0] || undefined : undefined, args);
|
|
|
}
|
|
|
return val;
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* use the val2 when val1 is undefined
|
|
|
* @param {*} val1
|
|
|
* @param {*} val2
|
|
|
* @returns {*}
|
|
|
*/
|
|
|
function ifUndefined(val1, val2) {
|
|
|
return (val1 === undefined) ? val2 : val1;
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* addEventListener with multiple events at once
|
|
|
* @param {EventTarget} target
|
|
|
* @param {String} types
|
|
|
* @param {Function} handler
|
|
|
*/
|
|
|
function addEventListeners(target, types, handler) {
|
|
|
each(splitStr(types), function(type) {
|
|
|
target.addEventListener(type, handler, false);
|
|
|
});
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* removeEventListener with multiple events at once
|
|
|
* @param {EventTarget} target
|
|
|
* @param {String} types
|
|
|
* @param {Function} handler
|
|
|
*/
|
|
|
function removeEventListeners(target, types, handler) {
|
|
|
each(splitStr(types), function(type) {
|
|
|
target.removeEventListener(type, handler, false);
|
|
|
});
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* find if a node is in the given parent
|
|
|
* @method hasParent
|
|
|
* @param {HTMLElement} node
|
|
|
* @param {HTMLElement} parent
|
|
|
* @return {Boolean} found
|
|
|
*/
|
|
|
function hasParent(node, parent) {
|
|
|
while (node) {
|
|
|
if (node == parent) {
|
|
|
return true;
|
|
|
}
|
|
|
node = node.parentNode;
|
|
|
}
|
|
|
return false;
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* small indexOf wrapper
|
|
|
* @param {String} str
|
|
|
* @param {String} find
|
|
|
* @returns {Boolean} found
|
|
|
*/
|
|
|
function inStr(str, find) {
|
|
|
return str.indexOf(find) > -1;
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* split string on whitespace
|
|
|
* @param {String} str
|
|
|
* @returns {Array} words
|
|
|
*/
|
|
|
function splitStr(str) {
|
|
|
return str.trim().split(/\s+/g);
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* find if a array contains the object using indexOf or a simple polyFill
|
|
|
* @param {Array} src
|
|
|
* @param {String} find
|
|
|
* @param {String} [findByKey]
|
|
|
* @return {Boolean|Number} false when not found, or the index
|
|
|
*/
|
|
|
function inArray(src, find, findByKey) {
|
|
|
if (src.indexOf && !findByKey) {
|
|
|
return src.indexOf(find);
|
|
|
} else {
|
|
|
var i = 0;
|
|
|
while (i < src.length) {
|
|
|
if ((findByKey && src[i][findByKey] == find) || (!findByKey && src[i] === find)) {
|
|
|
return i;
|
|
|
}
|
|
|
i++;
|
|
|
}
|
|
|
return -1;
|
|
|
}
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* convert array-like objects to real arrays
|
|
|
* @param {Object} obj
|
|
|
* @returns {Array}
|
|
|
*/
|
|
|
function toArray(obj) {
|
|
|
return Array.prototype.slice.call(obj, 0);
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* unique array with objects based on a key (like 'id') or just by the array's value
|
|
|
* @param {Array} src [{id:1},{id:2},{id:1}]
|
|
|
* @param {String} [key]
|
|
|
* @param {Boolean} [sort=False]
|
|
|
* @returns {Array} [{id:1},{id:2}]
|
|
|
*/
|
|
|
function uniqueArray(src, key, sort) {
|
|
|
var results = [];
|
|
|
var values = [];
|
|
|
var i = 0;
|
|
|
|
|
|
while (i < src.length) {
|
|
|
var val = key ? src[i][key] : src[i];
|
|
|
if (inArray(values, val) < 0) {
|
|
|
results.push(src[i]);
|
|
|
}
|
|
|
values[i] = val;
|
|
|
i++;
|
|
|
}
|
|
|
|
|
|
if (sort) {
|
|
|
if (!key) {
|
|
|
results = results.sort();
|
|
|
} else {
|
|
|
results = results.sort(function sortUniqueArray(a, b) {
|
|
|
return a[key] > b[key];
|
|
|
});
|
|
|
}
|
|
|
}
|
|
|
|
|
|
return results;
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* get the prefixed property
|
|
|
* @param {Object} obj
|
|
|
* @param {String} property
|
|
|
* @returns {String|Undefined} prefixed
|
|
|
*/
|
|
|
function prefixed(obj, property) {
|
|
|
var prefix, prop;
|
|
|
var camelProp = property[0].toUpperCase() + property.slice(1);
|
|
|
|
|
|
var i = 0;
|
|
|
while (i < VENDOR_PREFIXES.length) {
|
|
|
prefix = VENDOR_PREFIXES[i];
|
|
|
prop = (prefix) ? prefix + camelProp : property;
|
|
|
|
|
|
if (prop in obj) {
|
|
|
return prop;
|
|
|
}
|
|
|
i++;
|
|
|
}
|
|
|
return undefined;
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* get a unique id
|
|
|
* @returns {number} uniqueId
|
|
|
*/
|
|
|
var _uniqueId = 1;
|
|
|
function uniqueId() {
|
|
|
return _uniqueId++;
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* get the window object of an element
|
|
|
* @param {HTMLElement} element
|
|
|
* @returns {DocumentView|Window}
|
|
|
*/
|
|
|
function getWindowForElement(element) {
|
|
|
var doc = element.ownerDocument || element;
|
|
|
return (doc.defaultView || doc.parentWindow || window);
|
|
|
}
|
|
|
|
|
|
var MOBILE_REGEX = /mobile|tablet|ip(ad|hone|od)|android/i;
|
|
|
|
|
|
var SUPPORT_TOUCH = ('ontouchstart' in window);
|
|
|
var SUPPORT_POINTER_EVENTS = prefixed(window, 'PointerEvent') !== undefined;
|
|
|
var SUPPORT_ONLY_TOUCH = SUPPORT_TOUCH && MOBILE_REGEX.test(navigator.userAgent);
|
|
|
|
|
|
var INPUT_TYPE_TOUCH = 'touch';
|
|
|
var INPUT_TYPE_PEN = 'pen';
|
|
|
var INPUT_TYPE_MOUSE = 'mouse';
|
|
|
var INPUT_TYPE_KINECT = 'kinect';
|
|
|
|
|
|
var COMPUTE_INTERVAL = 25;
|
|
|
|
|
|
var INPUT_START = 1;
|
|
|
var INPUT_MOVE = 2;
|
|
|
var INPUT_END = 4;
|
|
|
var INPUT_CANCEL = 8;
|
|
|
|
|
|
var DIRECTION_NONE = 1;
|
|
|
var DIRECTION_LEFT = 2;
|
|
|
var DIRECTION_RIGHT = 4;
|
|
|
var DIRECTION_UP = 8;
|
|
|
var DIRECTION_DOWN = 16;
|
|
|
|
|
|
var DIRECTION_HORIZONTAL = DIRECTION_LEFT | DIRECTION_RIGHT;
|
|
|
var DIRECTION_VERTICAL = DIRECTION_UP | DIRECTION_DOWN;
|
|
|
var DIRECTION_ALL = DIRECTION_HORIZONTAL | DIRECTION_VERTICAL;
|
|
|
|
|
|
var PROPS_XY = ['x', 'y'];
|
|
|
var PROPS_CLIENT_XY = ['clientX', 'clientY'];
|
|
|
|
|
|
/**
|
|
|
* create new input type manager
|
|
|
* @param {Manager} manager
|
|
|
* @param {Function} callback
|
|
|
* @returns {Input}
|
|
|
* @constructor
|
|
|
*/
|
|
|
function Input(manager, callback) {
|
|
|
var self = this;
|
|
|
this.manager = manager;
|
|
|
this.callback = callback;
|
|
|
this.element = manager.element;
|
|
|
this.target = manager.options.inputTarget;
|
|
|
|
|
|
// smaller wrapper around the handler, for the scope and the enabled state of the manager,
|
|
|
// so when disabled the input events are completely bypassed.
|
|
|
this.domHandler = function(ev) {
|
|
|
if (boolOrFn(manager.options.enable, [manager])) {
|
|
|
self.handler(ev);
|
|
|
}
|
|
|
};
|
|
|
|
|
|
this.init();
|
|
|
|
|
|
}
|
|
|
|
|
|
Input.prototype = {
|
|
|
/**
|
|
|
* should handle the inputEvent data and trigger the callback
|
|
|
* @virtual
|
|
|
*/
|
|
|
handler: function() { },
|
|
|
|
|
|
/**
|
|
|
* bind the events
|
|
|
*/
|
|
|
init: function() {
|
|
|
this.evEl && addEventListeners(this.element, this.evEl, this.domHandler);
|
|
|
this.evTarget && addEventListeners(this.target, this.evTarget, this.domHandler);
|
|
|
this.evWin && addEventListeners(getWindowForElement(this.element), this.evWin, this.domHandler);
|
|
|
},
|
|
|
|
|
|
/**
|
|
|
* unbind the events
|
|
|
*/
|
|
|
destroy: function() {
|
|
|
this.evEl && removeEventListeners(this.element, this.evEl, this.domHandler);
|
|
|
this.evTarget && removeEventListeners(this.target, this.evTarget, this.domHandler);
|
|
|
this.evWin && removeEventListeners(getWindowForElement(this.element), this.evWin, this.domHandler);
|
|
|
}
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* create new input type manager
|
|
|
* called by the Manager constructor
|
|
|
* @param {Hammer} manager
|
|
|
* @returns {Input}
|
|
|
*/
|
|
|
function createInputInstance(manager) {
|
|
|
var Type;
|
|
|
var inputClass = manager.options.inputClass;
|
|
|
|
|
|
if (inputClass) {
|
|
|
Type = inputClass;
|
|
|
} else if (SUPPORT_POINTER_EVENTS) {
|
|
|
Type = PointerEventInput;
|
|
|
} else if (SUPPORT_ONLY_TOUCH) {
|
|
|
Type = TouchInput;
|
|
|
} else if (!SUPPORT_TOUCH) {
|
|
|
Type = MouseInput;
|
|
|
} else {
|
|
|
Type = TouchMouseInput;
|
|
|
}
|
|
|
return new (Type)(manager, inputHandler);
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* handle input events
|
|
|
* @param {Manager} manager
|
|
|
* @param {String} eventType
|
|
|
* @param {Object} input
|
|
|
*/
|
|
|
function inputHandler(manager, eventType, input) {
|
|
|
var pointersLen = input.pointers.length;
|
|
|
var changedPointersLen = input.changedPointers.length;
|
|
|
var isFirst = (eventType & INPUT_START && (pointersLen - changedPointersLen === 0));
|
|
|
var isFinal = (eventType & (INPUT_END | INPUT_CANCEL) && (pointersLen - changedPointersLen === 0));
|
|
|
|
|
|
input.isFirst = !!isFirst;
|
|
|
input.isFinal = !!isFinal;
|
|
|
|
|
|
if (isFirst) {
|
|
|
manager.session = {};
|
|
|
}
|
|
|
|
|
|
// source event is the normalized value of the domEvents
|
|
|
// like 'touchstart, mouseup, pointerdown'
|
|
|
input.eventType = eventType;
|
|
|
|
|
|
// compute scale, rotation etc
|
|
|
computeInputData(manager, input);
|
|
|
|
|
|
// emit secret event
|
|
|
manager.emit('hammer.input', input);
|
|
|
|
|
|
manager.recognize(input);
|
|
|
manager.session.prevInput = input;
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* extend the data with some usable properties like scale, rotate, velocity etc
|
|
|
* @param {Object} manager
|
|
|
* @param {Object} input
|
|
|
*/
|
|
|
function computeInputData(manager, input) {
|
|
|
var session = manager.session;
|
|
|
var pointers = input.pointers;
|
|
|
var pointersLength = pointers.length;
|
|
|
|
|
|
// store the first input to calculate the distance and direction
|
|
|
if (!session.firstInput) {
|
|
|
session.firstInput = simpleCloneInputData(input);
|
|
|
}
|
|
|
|
|
|
// to compute scale and rotation we need to store the multiple touches
|
|
|
if (pointersLength > 1 && !session.firstMultiple) {
|
|
|
session.firstMultiple = simpleCloneInputData(input);
|
|
|
} else if (pointersLength === 1) {
|
|
|
session.firstMultiple = false;
|
|
|
}
|
|
|
|
|
|
var firstInput = session.firstInput;
|
|
|
var firstMultiple = session.firstMultiple;
|
|
|
var offsetCenter = firstMultiple ? firstMultiple.center : firstInput.center;
|
|
|
|
|
|
var center = input.center = getCenter(pointers);
|
|
|
input.timeStamp = now();
|
|
|
input.deltaTime = input.timeStamp - firstInput.timeStamp;
|
|
|
|
|
|
input.angle = getAngle(offsetCenter, center);
|
|
|
input.distance = getDistance(offsetCenter, center);
|
|
|
|
|
|
computeDeltaXY(session, input);
|
|
|
input.offsetDirection = getDirection(input.deltaX, input.deltaY);
|
|
|
|
|
|
var overallVelocity = getVelocity(input.deltaTime, input.deltaX, input.deltaY);
|
|
|
input.overallVelocityX = overallVelocity.x;
|
|
|
input.overallVelocityY = overallVelocity.y;
|
|
|
input.overallVelocity = (abs(overallVelocity.x) > abs(overallVelocity.y)) ? overallVelocity.x : overallVelocity.y;
|
|
|
|
|
|
input.scale = firstMultiple ? getScale(firstMultiple.pointers, pointers) : 1;
|
|
|
input.rotation = firstMultiple ? getRotation(firstMultiple.pointers, pointers) : 0;
|
|
|
|
|
|
input.maxPointers = !session.prevInput ? input.pointers.length : ((input.pointers.length >
|
|
|
session.prevInput.maxPointers) ? input.pointers.length : session.prevInput.maxPointers);
|
|
|
|
|
|
computeIntervalInputData(session, input);
|
|
|
|
|
|
// find the correct target
|
|
|
var target = manager.element;
|
|
|
if (hasParent(input.srcEvent.target, target)) {
|
|
|
target = input.srcEvent.target;
|
|
|
}
|
|
|
input.target = target;
|
|
|
}
|
|
|
|
|
|
function computeDeltaXY(session, input) {
|
|
|
var center = input.center;
|
|
|
var offset = session.offsetDelta || {};
|
|
|
var prevDelta = session.prevDelta || {};
|
|
|
var prevInput = session.prevInput || {};
|
|
|
|
|
|
if (input.eventType === INPUT_START || prevInput.eventType === INPUT_END) {
|
|
|
prevDelta = session.prevDelta = {
|
|
|
x: prevInput.deltaX || 0,
|
|
|
y: prevInput.deltaY || 0
|
|
|
};
|
|
|
|
|
|
offset = session.offsetDelta = {
|
|
|
x: center.x,
|
|
|
y: center.y
|
|
|
};
|
|
|
}
|
|
|
|
|
|
input.deltaX = prevDelta.x + (center.x - offset.x);
|
|
|
input.deltaY = prevDelta.y + (center.y - offset.y);
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* velocity is calculated every x ms
|
|
|
* @param {Object} session
|
|
|
* @param {Object} input
|
|
|
*/
|
|
|
function computeIntervalInputData(session, input) {
|
|
|
var last = session.lastInterval || input,
|
|
|
deltaTime = input.timeStamp - last.timeStamp,
|
|
|
velocity, velocityX, velocityY, direction;
|
|
|
|
|
|
if (input.eventType != INPUT_CANCEL && (deltaTime > COMPUTE_INTERVAL || last.velocity === undefined)) {
|
|
|
var deltaX = input.deltaX - last.deltaX;
|
|
|
var deltaY = input.deltaY - last.deltaY;
|
|
|
|
|
|
var v = getVelocity(deltaTime, deltaX, deltaY);
|
|
|
velocityX = v.x;
|
|
|
velocityY = v.y;
|
|
|
velocity = (abs(v.x) > abs(v.y)) ? v.x : v.y;
|
|
|
direction = getDirection(deltaX, deltaY);
|
|
|
|
|
|
session.lastInterval = input;
|
|
|
} else {
|
|
|
// use latest velocity info if it doesn't overtake a minimum period
|
|
|
velocity = last.velocity;
|
|
|
velocityX = last.velocityX;
|
|
|
velocityY = last.velocityY;
|
|
|
direction = last.direction;
|
|
|
}
|
|
|
|
|
|
input.velocity = velocity;
|
|
|
input.velocityX = velocityX;
|
|
|
input.velocityY = velocityY;
|
|
|
input.direction = direction;
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* create a simple clone from the input used for storage of firstInput and firstMultiple
|
|
|
* @param {Object} input
|
|
|
* @returns {Object} clonedInputData
|
|
|
*/
|
|
|
function simpleCloneInputData(input) {
|
|
|
// make a simple copy of the pointers because we will get a reference if we don't
|
|
|
// we only need clientXY for the calculations
|
|
|
var pointers = [];
|
|
|
var i = 0;
|
|
|
while (i < input.pointers.length) {
|
|
|
pointers[i] = {
|
|
|
clientX: round(input.pointers[i].clientX),
|
|
|
clientY: round(input.pointers[i].clientY)
|
|
|
};
|
|
|
i++;
|
|
|
}
|
|
|
|
|
|
return {
|
|
|
timeStamp: now(),
|
|
|
pointers: pointers,
|
|
|
center: getCenter(pointers),
|
|
|
deltaX: input.deltaX,
|
|
|
deltaY: input.deltaY
|
|
|
};
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* get the center of all the pointers
|
|
|
* @param {Array} pointers
|
|
|
* @return {Object} center contains `x` and `y` properties
|
|
|
*/
|
|
|
function getCenter(pointers) {
|
|
|
var pointersLength = pointers.length;
|
|
|
|
|
|
// no need to loop when only one touch
|
|
|
if (pointersLength === 1) {
|
|
|
return {
|
|
|
x: round(pointers[0].clientX),
|
|
|
y: round(pointers[0].clientY)
|
|
|
};
|
|
|
}
|
|
|
|
|
|
var x = 0, y = 0, i = 0;
|
|
|
while (i < pointersLength) {
|
|
|
x += pointers[i].clientX;
|
|
|
y += pointers[i].clientY;
|
|
|
i++;
|
|
|
}
|
|
|
|
|
|
return {
|
|
|
x: round(x / pointersLength),
|
|
|
y: round(y / pointersLength)
|
|
|
};
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* calculate the velocity between two points. unit is in px per ms.
|
|
|
* @param {Number} deltaTime
|
|
|
* @param {Number} x
|
|
|
* @param {Number} y
|
|
|
* @return {Object} velocity `x` and `y`
|
|
|
*/
|
|
|
function getVelocity(deltaTime, x, y) {
|
|
|
return {
|
|
|
x: x / deltaTime || 0,
|
|
|
y: y / deltaTime || 0
|
|
|
};
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* get the direction between two points
|
|
|
* @param {Number} x
|
|
|
* @param {Number} y
|
|
|
* @return {Number} direction
|
|
|
*/
|
|
|
function getDirection(x, y) {
|
|
|
if (x === y) {
|
|
|
return DIRECTION_NONE;
|
|
|
}
|
|
|
|
|
|
if (abs(x) >= abs(y)) {
|
|
|
return x < 0 ? DIRECTION_LEFT : DIRECTION_RIGHT;
|
|
|
}
|
|
|
return y < 0 ? DIRECTION_UP : DIRECTION_DOWN;
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* calculate the absolute distance between two points
|
|
|
* @param {Object} p1 {x, y}
|
|
|
* @param {Object} p2 {x, y}
|
|
|
* @param {Array} [props] containing x and y keys
|
|
|
* @return {Number} distance
|
|
|
*/
|
|
|
function getDistance(p1, p2, props) {
|
|
|
if (!props) {
|
|
|
props = PROPS_XY;
|
|
|
}
|
|
|
var x = p2[props[0]] - p1[props[0]],
|
|
|
y = p2[props[1]] - p1[props[1]];
|
|
|
|
|
|
return Math.sqrt((x * x) + (y * y));
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* calculate the angle between two coordinates
|
|
|
* @param {Object} p1
|
|
|
* @param {Object} p2
|
|
|
* @param {Array} [props] containing x and y keys
|
|
|
* @return {Number} angle
|
|
|
*/
|
|
|
function getAngle(p1, p2, props) {
|
|
|
if (!props) {
|
|
|
props = PROPS_XY;
|
|
|
}
|
|
|
var x = p2[props[0]] - p1[props[0]],
|
|
|
y = p2[props[1]] - p1[props[1]];
|
|
|
return Math.atan2(y, x) * 180 / Math.PI;
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* calculate the rotation degrees between two pointersets
|
|
|
* @param {Array} start array of pointers
|
|
|
* @param {Array} end array of pointers
|
|
|
* @return {Number} rotation
|
|
|
*/
|
|
|
function getRotation(start, end) {
|
|
|
return getAngle(end[1], end[0], PROPS_CLIENT_XY) + getAngle(start[1], start[0], PROPS_CLIENT_XY);
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* calculate the scale factor between two pointersets
|
|
|
* no scale is 1, and goes down to 0 when pinched together, and bigger when pinched out
|
|
|
* @param {Array} start array of pointers
|
|
|
* @param {Array} end array of pointers
|
|
|
* @return {Number} scale
|
|
|
*/
|
|
|
function getScale(start, end) {
|
|
|
return getDistance(end[0], end[1], PROPS_CLIENT_XY) / getDistance(start[0], start[1], PROPS_CLIENT_XY);
|
|
|
}
|
|
|
|
|
|
var MOUSE_INPUT_MAP = {
|
|
|
mousedown: INPUT_START,
|
|
|
mousemove: INPUT_MOVE,
|
|
|
mouseup: INPUT_END
|
|
|
};
|
|
|
|
|
|
var MOUSE_ELEMENT_EVENTS = 'mousedown';
|
|
|
var MOUSE_WINDOW_EVENTS = 'mousemove mouseup';
|
|
|
|
|
|
/**
|
|
|
* Mouse events input
|
|
|
* @constructor
|
|
|
* @extends Input
|
|
|
*/
|
|
|
function MouseInput() {
|
|
|
this.evEl = MOUSE_ELEMENT_EVENTS;
|
|
|
this.evWin = MOUSE_WINDOW_EVENTS;
|
|
|
|
|
|
this.pressed = false; // mousedown state
|
|
|
|
|
|
Input.apply(this, arguments);
|
|
|
}
|
|
|
|
|
|
inherit(MouseInput, Input, {
|
|
|
/**
|
|
|
* handle mouse events
|
|
|
* @param {Object} ev
|
|
|
*/
|
|
|
handler: function MEhandler(ev) {
|
|
|
var eventType = MOUSE_INPUT_MAP[ev.type];
|
|
|
|
|
|
// on start we want to have the left mouse button down
|
|
|
if (eventType & INPUT_START && ev.button === 0) {
|
|
|
this.pressed = true;
|
|
|
}
|
|
|
|
|
|
if (eventType & INPUT_MOVE && ev.which !== 1) {
|
|
|
eventType = INPUT_END;
|
|
|
}
|
|
|
|
|
|
// mouse must be down
|
|
|
if (!this.pressed) {
|
|
|
return;
|
|
|
}
|
|
|
|
|
|
if (eventType & INPUT_END) {
|
|
|
this.pressed = false;
|
|
|
}
|
|
|
|
|
|
this.callback(this.manager, eventType, {
|
|
|
pointers: [ev],
|
|
|
changedPointers: [ev],
|
|
|
pointerType: INPUT_TYPE_MOUSE,
|
|
|
srcEvent: ev
|
|
|
});
|
|
|
}
|
|
|
});
|
|
|
|
|
|
var POINTER_INPUT_MAP = {
|
|
|
pointerdown: INPUT_START,
|
|
|
pointermove: INPUT_MOVE,
|
|
|
pointerup: INPUT_END,
|
|
|
pointercancel: INPUT_CANCEL,
|
|
|
pointerout: INPUT_CANCEL
|
|
|
};
|
|
|
|
|
|
// in IE10 the pointer types is defined as an enum
|
|
|
var IE10_POINTER_TYPE_ENUM = {
|
|
|
2: INPUT_TYPE_TOUCH,
|
|
|
3: INPUT_TYPE_PEN,
|
|
|
4: INPUT_TYPE_MOUSE,
|
|
|
5: INPUT_TYPE_KINECT // see https://twitter.com/jacobrossi/status/480596438489890816
|
|
|
};
|
|
|
|
|
|
var POINTER_ELEMENT_EVENTS = 'pointerdown';
|
|
|
var POINTER_WINDOW_EVENTS = 'pointermove pointerup pointercancel';
|
|
|
|
|
|
// IE10 has prefixed support, and case-sensitive
|
|
|
if (window.MSPointerEvent && !window.PointerEvent) {
|
|
|
POINTER_ELEMENT_EVENTS = 'MSPointerDown';
|
|
|
POINTER_WINDOW_EVENTS = 'MSPointerMove MSPointerUp MSPointerCancel';
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* Pointer events input
|
|
|
* @constructor
|
|
|
* @extends Input
|
|
|
*/
|
|
|
function PointerEventInput() {
|
|
|
this.evEl = POINTER_ELEMENT_EVENTS;
|
|
|
this.evWin = POINTER_WINDOW_EVENTS;
|
|
|
|
|
|
Input.apply(this, arguments);
|
|
|
|
|
|
this.store = (this.manager.session.pointerEvents = []);
|
|
|
}
|
|
|
|
|
|
inherit(PointerEventInput, Input, {
|
|
|
/**
|
|
|
* handle mouse events
|
|
|
* @param {Object} ev
|
|
|
*/
|
|
|
handler: function PEhandler(ev) {
|
|
|
var store = this.store;
|
|
|
var removePointer = false;
|
|
|
|
|
|
var eventTypeNormalized = ev.type.toLowerCase().replace('ms', '');
|
|
|
var eventType = POINTER_INPUT_MAP[eventTypeNormalized];
|
|
|
var pointerType = IE10_POINTER_TYPE_ENUM[ev.pointerType] || ev.pointerType;
|
|
|
|
|
|
var isTouch = (pointerType == INPUT_TYPE_TOUCH);
|
|
|
|
|
|
// get index of the event in the store
|
|
|
var storeIndex = inArray(store, ev.pointerId, 'pointerId');
|
|
|
|
|
|
// start and mouse must be down
|
|
|
if (eventType & INPUT_START && (ev.button === 0 || isTouch)) {
|
|
|
if (storeIndex < 0) {
|
|
|
store.push(ev);
|
|
|
storeIndex = store.length - 1;
|
|
|
}
|
|
|
} else if (eventType & (INPUT_END | INPUT_CANCEL)) {
|
|
|
removePointer = true;
|
|
|
}
|
|
|
|
|
|
// it not found, so the pointer hasn't been down (so it's probably a hover)
|
|
|
if (storeIndex < 0) {
|
|
|
return;
|
|
|
}
|
|
|
|
|
|
// update the event in the store
|
|
|
store[storeIndex] = ev;
|
|
|
|
|
|
this.callback(this.manager, eventType, {
|
|
|
pointers: store,
|
|
|
changedPointers: [ev],
|
|
|
pointerType: pointerType,
|
|
|
srcEvent: ev
|
|
|
});
|
|
|
|
|
|
if (removePointer) {
|
|
|
// remove from the store
|
|
|
store.splice(storeIndex, 1);
|
|
|
}
|
|
|
}
|
|
|
});
|
|
|
|
|
|
var SINGLE_TOUCH_INPUT_MAP = {
|
|
|
touchstart: INPUT_START,
|
|
|
touchmove: INPUT_MOVE,
|
|
|
touchend: INPUT_END,
|
|
|
touchcancel: INPUT_CANCEL
|
|
|
};
|
|
|
|
|
|
var SINGLE_TOUCH_TARGET_EVENTS = 'touchstart';
|
|
|
var SINGLE_TOUCH_WINDOW_EVENTS = 'touchstart touchmove touchend touchcancel';
|
|
|
|
|
|
/**
|
|
|
* Touch events input
|
|
|
* @constructor
|
|
|
* @extends Input
|
|
|
*/
|
|
|
function SingleTouchInput() {
|
|
|
this.evTarget = SINGLE_TOUCH_TARGET_EVENTS;
|
|
|
this.evWin = SINGLE_TOUCH_WINDOW_EVENTS;
|
|
|
this.started = false;
|
|
|
|
|
|
Input.apply(this, arguments);
|
|
|
}
|
|
|
|
|
|
inherit(SingleTouchInput, Input, {
|
|
|
handler: function TEhandler(ev) {
|
|
|
var type = SINGLE_TOUCH_INPUT_MAP[ev.type];
|
|
|
|
|
|
// should we handle the touch events?
|
|
|
if (type === INPUT_START) {
|
|
|
this.started = true;
|
|
|
}
|
|
|
|
|
|
if (!this.started) {
|
|
|
return;
|
|
|
}
|
|
|
|
|
|
var touches = normalizeSingleTouches.call(this, ev, type);
|
|
|
|
|
|
// when done, reset the started state
|
|
|
if (type & (INPUT_END | INPUT_CANCEL) && touches[0].length - touches[1].length === 0) {
|
|
|
this.started = false;
|
|
|
}
|
|
|
|
|
|
this.callback(this.manager, type, {
|
|
|
pointers: touches[0],
|
|
|
changedPointers: touches[1],
|
|
|
pointerType: INPUT_TYPE_TOUCH,
|
|
|
srcEvent: ev
|
|
|
});
|
|
|
}
|
|
|
});
|
|
|
|
|
|
/**
|
|
|
* @this {TouchInput}
|
|
|
* @param {Object} ev
|
|
|
* @param {Number} type flag
|
|
|
* @returns {undefined|Array} [all, changed]
|
|
|
*/
|
|
|
function normalizeSingleTouches(ev, type) {
|
|
|
var all = toArray(ev.touches);
|
|
|
var changed = toArray(ev.changedTouches);
|
|
|
|
|
|
if (type & (INPUT_END | INPUT_CANCEL)) {
|
|
|
all = uniqueArray(all.concat(changed), 'identifier', true);
|
|
|
}
|
|
|
|
|
|
return [all, changed];
|
|
|
}
|
|
|
|
|
|
var TOUCH_INPUT_MAP = {
|
|
|
touchstart: INPUT_START,
|
|
|
touchmove: INPUT_MOVE,
|
|
|
touchend: INPUT_END,
|
|
|
touchcancel: INPUT_CANCEL
|
|
|
};
|
|
|
|
|
|
var TOUCH_TARGET_EVENTS = 'touchstart touchmove touchend touchcancel';
|
|
|
|
|
|
/**
|
|
|
* Multi-user touch events input
|
|
|
* @constructor
|
|
|
* @extends Input
|
|
|
*/
|
|
|
function TouchInput() {
|
|
|
this.evTarget = TOUCH_TARGET_EVENTS;
|
|
|
this.targetIds = {};
|
|
|
|
|
|
Input.apply(this, arguments);
|
|
|
}
|
|
|
|
|
|
inherit(TouchInput, Input, {
|
|
|
handler: function MTEhandler(ev) {
|
|
|
var type = TOUCH_INPUT_MAP[ev.type];
|
|
|
var touches = getTouches.call(this, ev, type);
|
|
|
if (!touches) {
|
|
|
return;
|
|
|
}
|
|
|
|
|
|
this.callback(this.manager, type, {
|
|
|
pointers: touches[0],
|
|
|
changedPointers: touches[1],
|
|
|
pointerType: INPUT_TYPE_TOUCH,
|
|
|
srcEvent: ev
|
|
|
});
|
|
|
}
|
|
|
});
|
|
|
|
|
|
/**
|
|
|
* @this {TouchInput}
|
|
|
* @param {Object} ev
|
|
|
* @param {Number} type flag
|
|
|
* @returns {undefined|Array} [all, changed]
|
|
|
*/
|
|
|
function getTouches(ev, type) {
|
|
|
var allTouches = toArray(ev.touches);
|
|
|
var targetIds = this.targetIds;
|
|
|
|
|
|
// when there is only one touch, the process can be simplified
|
|
|
if (type & (INPUT_START | INPUT_MOVE) && allTouches.length === 1) {
|
|
|
targetIds[allTouches[0].identifier] = true;
|
|
|
return [allTouches, allTouches];
|
|
|
}
|
|
|
|
|
|
var i,
|
|
|
targetTouches,
|
|
|
changedTouches = toArray(ev.changedTouches),
|
|
|
changedTargetTouches = [],
|
|
|
target = this.target;
|
|
|
|
|
|
// get target touches from touches
|
|
|
targetTouches = allTouches.filter(function(touch) {
|
|
|
return hasParent(touch.target, target);
|
|
|
});
|
|
|
|
|
|
// collect touches
|
|
|
if (type === INPUT_START) {
|
|
|
i = 0;
|
|
|
while (i < targetTouches.length) {
|
|
|
targetIds[targetTouches[i].identifier] = true;
|
|
|
i++;
|
|
|
}
|
|
|
}
|
|
|
|
|
|
// filter changed touches to only contain touches that exist in the collected target ids
|
|
|
i = 0;
|
|
|
while (i < changedTouches.length) {
|
|
|
if (targetIds[changedTouches[i].identifier]) {
|
|
|
changedTargetTouches.push(changedTouches[i]);
|
|
|
}
|
|
|
|
|
|
// cleanup removed touches
|
|
|
if (type & (INPUT_END | INPUT_CANCEL)) {
|
|
|
delete targetIds[changedTouches[i].identifier];
|
|
|
}
|
|
|
i++;
|
|
|
}
|
|
|
|
|
|
if (!changedTargetTouches.length) {
|
|
|
return;
|
|
|
}
|
|
|
|
|
|
return [
|
|
|
// merge targetTouches with changedTargetTouches so it contains ALL touches, including 'end' and 'cancel'
|
|
|
uniqueArray(targetTouches.concat(changedTargetTouches), 'identifier', true),
|
|
|
changedTargetTouches
|
|
|
];
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* Combined touch and mouse input
|
|
|
*
|
|
|
* Touch has a higher priority then mouse, and while touching no mouse events are allowed.
|
|
|
* This because touch devices also emit mouse events while doing a touch.
|
|
|
*
|
|
|
* @constructor
|
|
|
* @extends Input
|
|
|
*/
|
|
|
|
|
|
var DEDUP_TIMEOUT = 2500;
|
|
|
var DEDUP_DISTANCE = 25;
|
|
|
|
|
|
function TouchMouseInput() {
|
|
|
Input.apply(this, arguments);
|
|
|
|
|
|
var handler = bindFn(this.handler, this);
|
|
|
this.touch = new TouchInput(this.manager, handler);
|
|
|
this.mouse = new MouseInput(this.manager, handler);
|
|
|
|
|
|
this.primaryTouch = null;
|
|
|
this.lastTouches = [];
|
|
|
}
|
|
|
|
|
|
inherit(TouchMouseInput, Input, {
|
|
|
/**
|
|
|
* handle mouse and touch events
|
|
|
* @param {Hammer} manager
|
|
|
* @param {String} inputEvent
|
|
|
* @param {Object} inputData
|
|
|
*/
|
|
|
handler: function TMEhandler(manager, inputEvent, inputData) {
|
|
|
var isTouch = (inputData.pointerType == INPUT_TYPE_TOUCH),
|
|
|
isMouse = (inputData.pointerType == INPUT_TYPE_MOUSE);
|
|
|
|
|
|
if (isMouse && inputData.sourceCapabilities && inputData.sourceCapabilities.firesTouchEvents) {
|
|
|
return;
|
|
|
}
|
|
|
|
|
|
// when we're in a touch event, record touches to de-dupe synthetic mouse event
|
|
|
if (isTouch) {
|
|
|
recordTouches.call(this, inputEvent, inputData);
|
|
|
} else if (isMouse && isSyntheticEvent.call(this, inputData)) {
|
|
|
return;
|
|
|
}
|
|
|
|
|
|
this.callback(manager, inputEvent, inputData);
|
|
|
},
|
|
|
|
|
|
/**
|
|
|
* remove the event listeners
|
|
|
*/
|
|
|
destroy: function destroy() {
|
|
|
this.touch.destroy();
|
|
|
this.mouse.destroy();
|
|
|
}
|
|
|
});
|
|
|
|
|
|
function recordTouches(eventType, eventData) {
|
|
|
if (eventType & INPUT_START) {
|
|
|
this.primaryTouch = eventData.changedPointers[0].identifier;
|
|
|
setLastTouch.call(this, eventData);
|
|
|
} else if (eventType & (INPUT_END | INPUT_CANCEL)) {
|
|
|
setLastTouch.call(this, eventData);
|
|
|
}
|
|
|
}
|
|
|
|
|
|
function setLastTouch(eventData) {
|
|
|
var touch = eventData.changedPointers[0];
|
|
|
|
|
|
if (touch.identifier === this.primaryTouch) {
|
|
|
var lastTouch = {x: touch.clientX, y: touch.clientY};
|
|
|
this.lastTouches.push(lastTouch);
|
|
|
var lts = this.lastTouches;
|
|
|
var removeLastTouch = function() {
|
|
|
var i = lts.indexOf(lastTouch);
|
|
|
if (i > -1) {
|
|
|
lts.splice(i, 1);
|
|
|
}
|
|
|
};
|
|
|
setTimeout(removeLastTouch, DEDUP_TIMEOUT);
|
|
|
}
|
|
|
}
|
|
|
|
|
|
function isSyntheticEvent(eventData) {
|
|
|
var x = eventData.srcEvent.clientX, y = eventData.srcEvent.clientY;
|
|
|
for (var i = 0; i < this.lastTouches.length; i++) {
|
|
|
var t = this.lastTouches[i];
|
|
|
var dx = Math.abs(x - t.x), dy = Math.abs(y - t.y);
|
|
|
if (dx <= DEDUP_DISTANCE && dy <= DEDUP_DISTANCE) {
|
|
|
return true;
|
|
|
}
|
|
|
}
|
|
|
return false;
|
|
|
}
|
|
|
|
|
|
var PREFIXED_TOUCH_ACTION = prefixed(TEST_ELEMENT.style, 'touchAction');
|
|
|
var NATIVE_TOUCH_ACTION = PREFIXED_TOUCH_ACTION !== undefined;
|
|
|
|
|
|
// magical touchAction value
|
|
|
var TOUCH_ACTION_COMPUTE = 'compute';
|
|
|
var TOUCH_ACTION_AUTO = 'auto';
|
|
|
var TOUCH_ACTION_MANIPULATION = 'manipulation'; // not implemented
|
|
|
var TOUCH_ACTION_NONE = 'none';
|
|
|
var TOUCH_ACTION_PAN_X = 'pan-x';
|
|
|
var TOUCH_ACTION_PAN_Y = 'pan-y';
|
|
|
var TOUCH_ACTION_MAP = getTouchActionProps();
|
|
|
|
|
|
/**
|
|
|
* Touch Action
|
|
|
* sets the touchAction property or uses the js alternative
|
|
|
* @param {Manager} manager
|
|
|
* @param {String} value
|
|
|
* @constructor
|
|
|
*/
|
|
|
function TouchAction(manager, value) {
|
|
|
this.manager = manager;
|
|
|
this.set(value);
|
|
|
}
|
|
|
|
|
|
TouchAction.prototype = {
|
|
|
/**
|
|
|
* set the touchAction value on the element or enable the polyfill
|
|
|
* @param {String} value
|
|
|
*/
|
|
|
set: function(value) {
|
|
|
// find out the touch-action by the event handlers
|
|
|
if (value == TOUCH_ACTION_COMPUTE) {
|
|
|
value = this.compute();
|
|
|
}
|
|
|
|
|
|
if (NATIVE_TOUCH_ACTION && this.manager.element.style && TOUCH_ACTION_MAP[value]) {
|
|
|
this.manager.element.style[PREFIXED_TOUCH_ACTION] = value;
|
|
|
}
|
|
|
this.actions = value.toLowerCase().trim();
|
|
|
},
|
|
|
|
|
|
/**
|
|
|
* just re-set the touchAction value
|
|
|
*/
|
|
|
update: function() {
|
|
|
this.set(this.manager.options.touchAction);
|
|
|
},
|
|
|
|
|
|
/**
|
|
|
* compute the value for the touchAction property based on the recognizer's settings
|
|
|
* @returns {String} value
|
|
|
*/
|
|
|
compute: function() {
|
|
|
var actions = [];
|
|
|
each(this.manager.recognizers, function(recognizer) {
|
|
|
if (boolOrFn(recognizer.options.enable, [recognizer])) {
|
|
|
actions = actions.concat(recognizer.getTouchAction());
|
|
|
}
|
|
|
});
|
|
|
return cleanTouchActions(actions.join(' '));
|
|
|
},
|
|
|
|
|
|
/**
|
|
|
* this method is called on each input cycle and provides the preventing of the browser behavior
|
|
|
* @param {Object} input
|
|
|
*/
|
|
|
preventDefaults: function(input) {
|
|
|
var srcEvent = input.srcEvent;
|
|
|
var direction = input.offsetDirection;
|
|
|
|
|
|
// if the touch action did prevented once this session
|
|
|
if (this.manager.session.prevented) {
|
|
|
srcEvent.preventDefault();
|
|
|
return;
|
|
|
}
|
|
|
|
|
|
var actions = this.actions;
|
|
|
var hasNone = inStr(actions, TOUCH_ACTION_NONE) && !TOUCH_ACTION_MAP[TOUCH_ACTION_NONE];
|
|
|
var hasPanY = inStr(actions, TOUCH_ACTION_PAN_Y) && !TOUCH_ACTION_MAP[TOUCH_ACTION_PAN_Y];
|
|
|
var hasPanX = inStr(actions, TOUCH_ACTION_PAN_X) && !TOUCH_ACTION_MAP[TOUCH_ACTION_PAN_X];
|
|
|
|
|
|
if (hasNone) {
|
|
|
//do not prevent defaults if this is a tap gesture
|
|
|
|
|
|
var isTapPointer = input.pointers.length === 1;
|
|
|
var isTapMovement = input.distance < 2;
|
|
|
var isTapTouchTime = input.deltaTime < 250;
|
|
|
|
|
|
if (isTapPointer && isTapMovement && isTapTouchTime) {
|
|
|
return;
|
|
|
}
|
|
|
}
|
|
|
|
|
|
if (hasPanX && hasPanY) {
|
|
|
// `pan-x pan-y` means browser handles all scrolling/panning, do not prevent
|
|
|
return;
|
|
|
}
|
|
|
|
|
|
if (hasNone ||
|
|
|
(hasPanY && direction & DIRECTION_HORIZONTAL) ||
|
|
|
(hasPanX && direction & DIRECTION_VERTICAL)) {
|
|
|
return this.preventSrc(srcEvent);
|
|
|
}
|
|
|
},
|
|
|
|
|
|
/**
|
|
|
* call preventDefault to prevent the browser's default behavior (scrolling in most cases)
|
|
|
* @param {Object} srcEvent
|
|
|
*/
|
|
|
preventSrc: function(srcEvent) {
|
|
|
this.manager.session.prevented = true;
|
|
|
srcEvent.preventDefault();
|
|
|
}
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* when the touchActions are collected they are not a valid value, so we need to clean things up. *
|
|
|
* @param {String} actions
|
|
|
* @returns {*}
|
|
|
*/
|
|
|
function cleanTouchActions(actions) {
|
|
|
// none
|
|
|
if (inStr(actions, TOUCH_ACTION_NONE)) {
|
|
|
return TOUCH_ACTION_NONE;
|
|
|
}
|
|
|
|
|
|
var hasPanX = inStr(actions, TOUCH_ACTION_PAN_X);
|
|
|
var hasPanY = inStr(actions, TOUCH_ACTION_PAN_Y);
|
|
|
|
|
|
// if both pan-x and pan-y are set (different recognizers
|
|
|
// for different directions, e.g. horizontal pan but vertical swipe?)
|
|
|
// we need none (as otherwise with pan-x pan-y combined none of these
|
|
|
// recognizers will work, since the browser would handle all panning
|
|
|
if (hasPanX && hasPanY) {
|
|
|
return TOUCH_ACTION_NONE;
|
|
|
}
|
|
|
|
|
|
// pan-x OR pan-y
|
|
|
if (hasPanX || hasPanY) {
|
|
|
return hasPanX ? TOUCH_ACTION_PAN_X : TOUCH_ACTION_PAN_Y;
|
|
|
}
|
|
|
|
|
|
// manipulation
|
|
|
if (inStr(actions, TOUCH_ACTION_MANIPULATION)) {
|
|
|
return TOUCH_ACTION_MANIPULATION;
|
|
|
}
|
|
|
|
|
|
return TOUCH_ACTION_AUTO;
|
|
|
}
|
|
|
|
|
|
function getTouchActionProps() {
|
|
|
if (!NATIVE_TOUCH_ACTION) {
|
|
|
return false;
|
|
|
}
|
|
|
var touchMap = {};
|
|
|
var cssSupports = window.CSS && window.CSS.supports;
|
|
|
['auto', 'manipulation', 'pan-y', 'pan-x', 'pan-x pan-y', 'none'].forEach(function(val) {
|
|
|
|
|
|
// If css.supports is not supported but there is native touch-action assume it supports
|
|
|
// all values. This is the case for IE 10 and 11.
|
|
|
touchMap[val] = cssSupports ? window.CSS.supports('touch-action', val) : true;
|
|
|
});
|
|
|
return touchMap;
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* Recognizer flow explained; *
|
|
|
* All recognizers have the initial state of POSSIBLE when a input session starts.
|
|
|
* The definition of a input session is from the first input until the last input, with all it's movement in it. *
|
|
|
* Example session for mouse-input: mousedown -> mousemove -> mouseup
|
|
|
*
|
|
|
* On each recognizing cycle (see Manager.recognize) the .recognize() method is executed
|
|
|
* which determines with state it should be.
|
|
|
*
|
|
|
* If the recognizer has the state FAILED, CANCELLED or RECOGNIZED (equals ENDED), it is reset to
|
|
|
* POSSIBLE to give it another change on the next cycle.
|
|
|
*
|
|
|
* Possible
|
|
|
* |
|
|
|
* +-----+---------------+
|
|
|
* | |
|
|
|
* +-----+-----+ |
|
|
|
* | | |
|
|
|
* Failed Cancelled |
|
|
|
* +-------+------+
|
|
|
* | |
|
|
|
* Recognized Began
|
|
|
* |
|
|
|
* Changed
|
|
|
* |
|
|
|
* Ended/Recognized
|
|
|
*/
|
|
|
var STATE_POSSIBLE = 1;
|
|
|
var STATE_BEGAN = 2;
|
|
|
var STATE_CHANGED = 4;
|
|
|
var STATE_ENDED = 8;
|
|
|
var STATE_RECOGNIZED = STATE_ENDED;
|
|
|
var STATE_CANCELLED = 16;
|
|
|
var STATE_FAILED = 32;
|
|
|
|
|
|
/**
|
|
|
* Recognizer
|
|
|
* Every recognizer needs to extend from this class.
|
|
|
* @constructor
|
|
|
* @param {Object} options
|
|
|
*/
|
|
|
function Recognizer(options) {
|
|
|
this.options = assign({}, this.defaults, options || {});
|
|
|
|
|
|
this.id = uniqueId();
|
|
|
|
|
|
this.manager = null;
|
|
|
|
|
|
// default is enable true
|
|
|
this.options.enable = ifUndefined(this.options.enable, true);
|
|
|
|
|
|
this.state = STATE_POSSIBLE;
|
|
|
|
|
|
this.simultaneous = {};
|
|
|
this.requireFail = [];
|
|
|
}
|
|
|
|
|
|
Recognizer.prototype = {
|
|
|
/**
|
|
|
* @virtual
|
|
|
* @type {Object}
|
|
|
*/
|
|
|
defaults: {},
|
|
|
|
|
|
/**
|
|
|
* set options
|
|
|
* @param {Object} options
|
|
|
* @return {Recognizer}
|
|
|
*/
|
|
|
set: function(options) {
|
|
|
assign(this.options, options);
|
|
|
|
|
|
// also update the touchAction, in case something changed about the directions/enabled state
|
|
|
this.manager && this.manager.touchAction.update();
|
|
|
return this;
|
|
|
},
|
|
|
|
|
|
/**
|
|
|
* recognize simultaneous with an other recognizer.
|
|
|
* @param {Recognizer} otherRecognizer
|
|
|
* @returns {Recognizer} this
|
|
|
*/
|
|
|
recognizeWith: function(otherRecognizer) {
|
|
|
if (invokeArrayArg(otherRecognizer, 'recognizeWith', this)) {
|
|
|
return this;
|
|
|
}
|
|
|
|
|
|
var simultaneous = this.simultaneous;
|
|
|
otherRecognizer = getRecognizerByNameIfManager(otherRecognizer, this);
|
|
|
if (!simultaneous[otherRecognizer.id]) {
|
|
|
simultaneous[otherRecognizer.id] = otherRecognizer;
|
|
|
otherRecognizer.recognizeWith(this);
|
|
|
}
|
|
|
return this;
|
|
|
},
|
|
|
|
|
|
/**
|
|
|
* drop the simultaneous link. it doesnt remove the link on the other recognizer.
|
|
|
* @param {Recognizer} otherRecognizer
|
|
|
* @returns {Recognizer} this
|
|
|
*/
|
|
|
dropRecognizeWith: function(otherRecognizer) {
|
|
|
if (invokeArrayArg(otherRecognizer, 'dropRecognizeWith', this)) {
|
|
|
return this;
|
|
|
}
|
|
|
|
|
|
otherRecognizer = getRecognizerByNameIfManager(otherRecognizer, this);
|
|
|
delete this.simultaneous[otherRecognizer.id];
|
|
|
return this;
|
|
|
},
|
|
|
|
|
|
/**
|
|
|
* recognizer can only run when an other is failing
|
|
|
* @param {Recognizer} otherRecognizer
|
|
|
* @returns {Recognizer} this
|
|
|
*/
|
|
|
requireFailure: function(otherRecognizer) {
|
|
|
if (invokeArrayArg(otherRecognizer, 'requireFailure', this)) {
|
|
|
return this;
|
|
|
}
|
|
|
|
|
|
var requireFail = this.requireFail;
|
|
|
otherRecognizer = getRecognizerByNameIfManager(otherRecognizer, this);
|
|
|
if (inArray(requireFail, otherRecognizer) === -1) {
|
|
|
requireFail.push(otherRecognizer);
|
|
|
otherRecognizer.requireFailure(this);
|
|
|
}
|
|
|
return this;
|
|
|
},
|
|
|
|
|
|
/**
|
|
|
* drop the requireFailure link. it does not remove the link on the other recognizer.
|
|
|
* @param {Recognizer} otherRecognizer
|
|
|
* @returns {Recognizer} this
|
|
|
*/
|
|
|
dropRequireFailure: function(otherRecognizer) {
|
|
|
if (invokeArrayArg(otherRecognizer, 'dropRequireFailure', this)) {
|
|
|
return this;
|
|
|
}
|
|
|
|
|
|
otherRecognizer = getRecognizerByNameIfManager(otherRecognizer, this);
|
|
|
var index = inArray(this.requireFail, otherRecognizer);
|
|
|
if (index > -1) {
|
|
|
this.requireFail.splice(index, 1);
|
|
|
}
|
|
|
return this;
|
|
|
},
|
|
|
|
|
|
/**
|
|
|
* has require failures boolean
|
|
|
* @returns {boolean}
|
|
|
*/
|
|
|
hasRequireFailures: function() {
|
|
|
return this.requireFail.length > 0;
|
|
|
},
|
|
|
|
|
|
/**
|
|
|
* if the recognizer can recognize simultaneous with an other recognizer
|
|
|
* @param {Recognizer} otherRecognizer
|
|
|
* @returns {Boolean}
|
|
|
*/
|
|
|
canRecognizeWith: function(otherRecognizer) {
|
|
|
return !!this.simultaneous[otherRecognizer.id];
|
|
|
},
|
|
|
|
|
|
/**
|
|
|
* You should use `tryEmit` instead of `emit` directly to check
|
|
|
* that all the needed recognizers has failed before emitting.
|
|
|
* @param {Object} input
|
|
|
*/
|
|
|
emit: function(input) {
|
|
|
var self = this;
|
|
|
var state = this.state;
|
|
|
|
|
|
function emit(event) {
|
|
|
self.manager.emit(event, input);
|
|
|
}
|
|
|
|
|
|
// 'panstart' and 'panmove'
|
|
|
if (state < STATE_ENDED) {
|
|
|
emit(self.options.event + stateStr(state));
|
|
|
}
|
|
|
|
|
|
emit(self.options.event); // simple 'eventName' events
|
|
|
|
|
|
if (input.additionalEvent) { // additional event(panleft, panright, pinchin, pinchout...)
|
|
|
emit(input.additionalEvent);
|
|
|
}
|
|
|
|
|
|
// panend and pancancel
|
|
|
if (state >= STATE_ENDED) {
|
|
|
emit(self.options.event + stateStr(state));
|
|
|
}
|
|
|
},
|
|
|
|
|
|
/**
|
|
|
* Check that all the require failure recognizers has failed,
|
|
|
* if true, it emits a gesture event,
|
|
|
* otherwise, setup the state to FAILED.
|
|
|
* @param {Object} input
|
|
|
*/
|
|
|
tryEmit: function(input) {
|
|
|
if (this.canEmit()) {
|
|
|
return this.emit(input);
|
|
|
}
|
|
|
// it's failing anyway
|
|
|
this.state = STATE_FAILED;
|
|
|
},
|
|
|
|
|
|
/**
|
|
|
* can we emit?
|
|
|
* @returns {boolean}
|
|
|
*/
|
|
|
canEmit: function() {
|
|
|
var i = 0;
|
|
|
while (i < this.requireFail.length) {
|
|
|
if (!(this.requireFail[i].state & (STATE_FAILED | STATE_POSSIBLE))) {
|
|
|
return false;
|
|
|
}
|
|
|
i++;
|
|
|
}
|
|
|
return true;
|
|
|
},
|
|
|
|
|
|
/**
|
|
|
* update the recognizer
|
|
|
* @param {Object} inputData
|
|
|
*/
|
|
|
recognize: function(inputData) {
|
|
|
// make a new copy of the inputData
|
|
|
// so we can change the inputData without messing up the other recognizers
|
|
|
var inputDataClone = assign({}, inputData);
|
|
|
|
|
|
// is is enabled and allow recognizing?
|
|
|
if (!boolOrFn(this.options.enable, [this, inputDataClone])) {
|
|
|
this.reset();
|
|
|
this.state = STATE_FAILED;
|
|
|
return;
|
|
|
}
|
|
|
|
|
|
// reset when we've reached the end
|
|
|
if (this.state & (STATE_RECOGNIZED | STATE_CANCELLED | STATE_FAILED)) {
|
|
|
this.state = STATE_POSSIBLE;
|
|
|
}
|
|
|
|
|
|
this.state = this.process(inputDataClone);
|
|
|
|
|
|
// the recognizer has recognized a gesture
|
|
|
// so trigger an event
|
|
|
if (this.state & (STATE_BEGAN | STATE_CHANGED | STATE_ENDED | STATE_CANCELLED)) {
|
|
|
this.tryEmit(inputDataClone);
|
|
|
}
|
|
|
},
|
|
|
|
|
|
/**
|
|
|
* return the state of the recognizer
|
|
|
* the actual recognizing happens in this method
|
|
|
* @virtual
|
|
|
* @param {Object} inputData
|
|
|
* @returns {Const} STATE
|
|
|
*/
|
|
|
process: function(inputData) { }, // jshint ignore:line
|
|
|
|
|
|
/**
|
|
|
* return the preferred touch-action
|
|
|
* @virtual
|
|
|
* @returns {Array}
|
|
|
*/
|
|
|
getTouchAction: function() { },
|
|
|
|
|
|
/**
|
|
|
* called when the gesture isn't allowed to recognize
|
|
|
* like when another is being recognized or it is disabled
|
|
|
* @virtual
|
|
|
*/
|
|
|
reset: function() { }
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* get a usable string, used as event postfix
|
|
|
* @param {Const} state
|
|
|
* @returns {String} state
|
|
|
*/
|
|
|
function stateStr(state) {
|
|
|
if (state & STATE_CANCELLED) {
|
|
|
return 'cancel';
|
|
|
} else if (state & STATE_ENDED) {
|
|
|
return 'end';
|
|
|
} else if (state & STATE_CHANGED) {
|
|
|
return 'move';
|
|
|
} else if (state & STATE_BEGAN) {
|
|
|
return 'start';
|
|
|
}
|
|
|
return '';
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* direction cons to string
|
|
|
* @param {Const} direction
|
|
|
* @returns {String}
|
|
|
*/
|
|
|
function directionStr(direction) {
|
|
|
if (direction == DIRECTION_DOWN) {
|
|
|
return 'down';
|
|
|
} else if (direction == DIRECTION_UP) {
|
|
|
return 'up';
|
|
|
} else if (direction == DIRECTION_LEFT) {
|
|
|
return 'left';
|
|
|
} else if (direction == DIRECTION_RIGHT) {
|
|
|
return 'right';
|
|
|
}
|
|
|
return '';
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* get a recognizer by name if it is bound to a manager
|
|
|
* @param {Recognizer|String} otherRecognizer
|
|
|
* @param {Recognizer} recognizer
|
|
|
* @returns {Recognizer}
|
|
|
*/
|
|
|
function getRecognizerByNameIfManager(otherRecognizer, recognizer) {
|
|
|
var manager = recognizer.manager;
|
|
|
if (manager) {
|
|
|
return manager.get(otherRecognizer);
|
|
|
}
|
|
|
return otherRecognizer;
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* This recognizer is just used as a base for the simple attribute recognizers.
|
|
|
* @constructor
|
|
|
* @extends Recognizer
|
|
|
*/
|
|
|
function AttrRecognizer() {
|
|
|
Recognizer.apply(this, arguments);
|
|
|
}
|
|
|
|
|
|
inherit(AttrRecognizer, Recognizer, {
|
|
|
/**
|
|
|
* @namespace
|
|
|
* @memberof AttrRecognizer
|
|
|
*/
|
|
|
defaults: {
|
|
|
/**
|
|
|
* @type {Number}
|
|
|
* @default 1
|
|
|
*/
|
|
|
pointers: 1
|
|
|
},
|
|
|
|
|
|
/**
|
|
|
* Used to check if it the recognizer receives valid input, like input.distance > 10.
|
|
|
* @memberof AttrRecognizer
|
|
|
* @param {Object} input
|
|
|
* @returns {Boolean} recognized
|
|
|
*/
|
|
|
attrTest: function(input) {
|
|
|
var optionPointers = this.options.pointers;
|
|
|
return optionPointers === 0 || input.pointers.length === optionPointers;
|
|
|
},
|
|
|
|
|
|
/**
|
|
|
* Process the input and return the state for the recognizer
|
|
|
* @memberof AttrRecognizer
|
|
|
* @param {Object} input
|
|
|
* @returns {*} State
|
|
|
*/
|
|
|
process: function(input) {
|
|
|
var state = this.state;
|
|
|
var eventType = input.eventType;
|
|
|
|
|
|
var isRecognized = state & (STATE_BEGAN | STATE_CHANGED);
|
|
|
var isValid = this.attrTest(input);
|
|
|
|
|
|
// on cancel input and we've recognized before, return STATE_CANCELLED
|
|
|
if (isRecognized && (eventType & INPUT_CANCEL || !isValid)) {
|
|
|
return state | STATE_CANCELLED;
|
|
|
} else if (isRecognized || isValid) {
|
|
|
if (eventType & INPUT_END) {
|
|
|
return state | STATE_ENDED;
|
|
|
} else if (!(state & STATE_BEGAN)) {
|
|
|
return STATE_BEGAN;
|
|
|
}
|
|
|
return state | STATE_CHANGED;
|
|
|
}
|
|
|
return STATE_FAILED;
|
|
|
}
|
|
|
});
|
|
|
|
|
|
/**
|
|
|
* Pan
|
|
|
* Recognized when the pointer is down and moved in the allowed direction.
|
|
|
* @constructor
|
|
|
* @extends AttrRecognizer
|
|
|
*/
|
|
|
function PanRecognizer() {
|
|
|
AttrRecognizer.apply(this, arguments);
|
|
|
|
|
|
this.pX = null;
|
|
|
this.pY = null;
|
|
|
}
|
|
|
|
|
|
inherit(PanRecognizer, AttrRecognizer, {
|
|
|
/**
|
|
|
* @namespace
|
|
|
* @memberof PanRecognizer
|
|
|
*/
|
|
|
defaults: {
|
|
|
event: 'pan',
|
|
|
threshold: 10,
|
|
|
pointers: 1,
|
|
|
direction: DIRECTION_ALL
|
|
|
},
|
|
|
|
|
|
getTouchAction: function() {
|
|
|
var direction = this.options.direction;
|
|
|
var actions = [];
|
|
|
if (direction & DIRECTION_HORIZONTAL) {
|
|
|
actions.push(TOUCH_ACTION_PAN_Y);
|
|
|
}
|
|
|
if (direction & DIRECTION_VERTICAL) {
|
|
|
actions.push(TOUCH_ACTION_PAN_X);
|
|
|
}
|
|
|
return actions;
|
|
|
},
|
|
|
|
|
|
directionTest: function(input) {
|
|
|
var options = this.options;
|
|
|
var hasMoved = true;
|
|
|
var distance = input.distance;
|
|
|
var direction = input.direction;
|
|
|
var x = input.deltaX;
|
|
|
var y = input.deltaY;
|
|
|
|
|
|
// lock to axis?
|
|
|
if (!(direction & options.direction)) {
|
|
|
if (options.direction & DIRECTION_HORIZONTAL) {
|
|
|
direction = (x === 0) ? DIRECTION_NONE : (x < 0) ? DIRECTION_LEFT : DIRECTION_RIGHT;
|
|
|
hasMoved = x != this.pX;
|
|
|
distance = Math.abs(input.deltaX);
|
|
|
} else {
|
|
|
direction = (y === 0) ? DIRECTION_NONE : (y < 0) ? DIRECTION_UP : DIRECTION_DOWN;
|
|
|
hasMoved = y != this.pY;
|
|
|
distance = Math.abs(input.deltaY);
|
|
|
}
|
|
|
}
|
|
|
input.direction = direction;
|
|
|
return hasMoved && distance > options.threshold && direction & options.direction;
|
|
|
},
|
|
|
|
|
|
attrTest: function(input) {
|
|
|
return AttrRecognizer.prototype.attrTest.call(this, input) &&
|
|
|
(this.state & STATE_BEGAN || (!(this.state & STATE_BEGAN) && this.directionTest(input)));
|
|
|
},
|
|
|
|
|
|
emit: function(input) {
|
|
|
|
|
|
this.pX = input.deltaX;
|
|
|
this.pY = input.deltaY;
|
|
|
|
|
|
var direction = directionStr(input.direction);
|
|
|
|
|
|
if (direction) {
|
|
|
input.additionalEvent = this.options.event + direction;
|
|
|
}
|
|
|
this._super.emit.call(this, input);
|
|
|
}
|
|
|
});
|
|
|
|
|
|
/**
|
|
|
* Pinch
|
|
|
* Recognized when two or more pointers are moving toward (zoom-in) or away from each other (zoom-out).
|
|
|
* @constructor
|
|
|
* @extends AttrRecognizer
|
|
|
*/
|
|
|
function PinchRecognizer() {
|
|
|
AttrRecognizer.apply(this, arguments);
|
|
|
}
|
|
|
|
|
|
inherit(PinchRecognizer, AttrRecognizer, {
|
|
|
/**
|
|
|
* @namespace
|
|
|
* @memberof PinchRecognizer
|
|
|
*/
|
|
|
defaults: {
|
|
|
event: 'pinch',
|
|
|
threshold: 0,
|
|
|
pointers: 2
|
|
|
},
|
|
|
|
|
|
getTouchAction: function() {
|
|
|
return [TOUCH_ACTION_NONE];
|
|
|
},
|
|
|
|
|
|
attrTest: function(input) {
|
|
|
return this._super.attrTest.call(this, input) &&
|
|
|
(Math.abs(input.scale - 1) > this.options.threshold || this.state & STATE_BEGAN);
|
|
|
},
|
|
|
|
|
|
emit: function(input) {
|
|
|
if (input.scale !== 1) {
|
|
|
var inOut = input.scale < 1 ? 'in' : 'out';
|
|
|
input.additionalEvent = this.options.event + inOut;
|
|
|
}
|
|
|
this._super.emit.call(this, input);
|
|
|
}
|
|
|
});
|
|
|
|
|
|
/**
|
|
|
* Press
|
|
|
* Recognized when the pointer is down for x ms without any movement.
|
|
|
* @constructor
|
|
|
* @extends Recognizer
|
|
|
*/
|
|
|
function PressRecognizer() {
|
|
|
Recognizer.apply(this, arguments);
|
|
|
|
|
|
this._timer = null;
|
|
|
this._input = null;
|
|
|
}
|
|
|
|
|
|
inherit(PressRecognizer, Recognizer, {
|
|
|
/**
|
|
|
* @namespace
|
|
|
* @memberof PressRecognizer
|
|
|
*/
|
|
|
defaults: {
|
|
|
event: 'press',
|
|
|
pointers: 1,
|
|
|
time: 251, // minimal time of the pointer to be pressed
|
|
|
threshold: 9 // a minimal movement is ok, but keep it low
|
|
|
},
|
|
|
|
|
|
getTouchAction: function() {
|
|
|
return [TOUCH_ACTION_AUTO];
|
|
|
},
|
|
|
|
|
|
process: function(input) {
|
|
|
var options = this.options;
|
|
|
var validPointers = input.pointers.length === options.pointers;
|
|
|
var validMovement = input.distance < options.threshold;
|
|
|
var validTime = input.deltaTime > options.time;
|
|
|
|
|
|
this._input = input;
|
|
|
|
|
|
// we only allow little movement
|
|
|
// and we've reached an end event, so a tap is possible
|
|
|
if (!validMovement || !validPointers || (input.eventType & (INPUT_END | INPUT_CANCEL) && !validTime)) {
|
|
|
this.reset();
|
|
|
} else if (input.eventType & INPUT_START) {
|
|
|
this.reset();
|
|
|
this._timer = setTimeoutContext(function() {
|
|
|
this.state = STATE_RECOGNIZED;
|
|
|
this.tryEmit();
|
|
|
}, options.time, this);
|
|
|
} else if (input.eventType & INPUT_END) {
|
|
|
return STATE_RECOGNIZED;
|
|
|
}
|
|
|
return STATE_FAILED;
|
|
|
},
|
|
|
|
|
|
reset: function() {
|
|
|
clearTimeout(this._timer);
|
|
|
},
|
|
|
|
|
|
emit: function(input) {
|
|
|
if (this.state !== STATE_RECOGNIZED) {
|
|
|
return;
|
|
|
}
|
|
|
|
|
|
if (input && (input.eventType & INPUT_END)) {
|
|
|
this.manager.emit(this.options.event + 'up', input);
|
|
|
} else {
|
|
|
this._input.timeStamp = now();
|
|
|
this.manager.emit(this.options.event, this._input);
|
|
|
}
|
|
|
}
|
|
|
});
|
|
|
|
|
|
/**
|
|
|
* Rotate
|
|
|
* Recognized when two or more pointer are moving in a circular motion.
|
|
|
* @constructor
|
|
|
* @extends AttrRecognizer
|
|
|
*/
|
|
|
function RotateRecognizer() {
|
|
|
AttrRecognizer.apply(this, arguments);
|
|
|
}
|
|
|
|
|
|
inherit(RotateRecognizer, AttrRecognizer, {
|
|
|
/**
|
|
|
* @namespace
|
|
|
* @memberof RotateRecognizer
|
|
|
*/
|
|
|
defaults: {
|
|
|
event: 'rotate',
|
|
|
threshold: 0,
|
|
|
pointers: 2
|
|
|
},
|
|
|
|
|
|
getTouchAction: function() {
|
|
|
return [TOUCH_ACTION_NONE];
|
|
|
},
|
|
|
|
|
|
attrTest: function(input) {
|
|
|
return this._super.attrTest.call(this, input) &&
|
|
|
(Math.abs(input.rotation) > this.options.threshold || this.state & STATE_BEGAN);
|
|
|
}
|
|
|
});
|
|
|
|
|
|
/**
|
|
|
* Swipe
|
|
|
* Recognized when the pointer is moving fast (velocity), with enough distance in the allowed direction.
|
|
|
* @constructor
|
|
|
* @extends AttrRecognizer
|
|
|
*/
|
|
|
function SwipeRecognizer() {
|
|
|
AttrRecognizer.apply(this, arguments);
|
|
|
}
|
|
|
|
|
|
inherit(SwipeRecognizer, AttrRecognizer, {
|
|
|
/**
|
|
|
* @namespace
|
|
|
* @memberof SwipeRecognizer
|
|
|
*/
|
|
|
defaults: {
|
|
|
event: 'swipe',
|
|
|
threshold: 10,
|
|
|
velocity: 0.3,
|
|
|
direction: DIRECTION_HORIZONTAL | DIRECTION_VERTICAL,
|
|
|
pointers: 1
|
|
|
},
|
|
|
|
|
|
getTouchAction: function() {
|
|
|
return PanRecognizer.prototype.getTouchAction.call(this);
|
|
|
},
|
|
|
|
|
|
attrTest: function(input) {
|
|
|
var direction = this.options.direction;
|
|
|
var velocity;
|
|
|
|
|
|
if (direction & (DIRECTION_HORIZONTAL | DIRECTION_VERTICAL)) {
|
|
|
velocity = input.overallVelocity;
|
|
|
} else if (direction & DIRECTION_HORIZONTAL) {
|
|
|
velocity = input.overallVelocityX;
|
|
|
} else if (direction & DIRECTION_VERTICAL) {
|
|
|
velocity = input.overallVelocityY;
|
|
|
}
|
|
|
|
|
|
return this._super.attrTest.call(this, input) &&
|
|
|
direction & input.offsetDirection &&
|
|
|
input.distance > this.options.threshold &&
|
|
|
input.maxPointers == this.options.pointers &&
|
|
|
abs(velocity) > this.options.velocity && input.eventType & INPUT_END;
|
|
|
},
|
|
|
|
|
|
emit: function(input) {
|
|
|
var direction = directionStr(input.offsetDirection);
|
|
|
if (direction) {
|
|
|
this.manager.emit(this.options.event + direction, input);
|
|
|
}
|
|
|
|
|
|
this.manager.emit(this.options.event, input);
|
|
|
}
|
|
|
});
|
|
|
|
|
|
/**
|
|
|
* A tap is ecognized when the pointer is doing a small tap/click. Multiple taps are recognized if they occur
|
|
|
* between the given interval and position. The delay option can be used to recognize multi-taps without firing
|
|
|
* a single tap.
|
|
|
*
|
|
|
* The eventData from the emitted event contains the property `tapCount`, which contains the amount of
|
|
|
* multi-taps being recognized.
|
|
|
* @constructor
|
|
|
* @extends Recognizer
|
|
|
*/
|
|
|
function TapRecognizer() {
|
|
|
Recognizer.apply(this, arguments);
|
|
|
|
|
|
// previous time and center,
|
|
|
// used for tap counting
|
|
|
this.pTime = false;
|
|
|
this.pCenter = false;
|
|
|
|
|
|
this._timer = null;
|
|
|
this._input = null;
|
|
|
this.count = 0;
|
|
|
}
|
|
|
|
|
|
inherit(TapRecognizer, Recognizer, {
|
|
|
/**
|
|
|
* @namespace
|
|
|
* @memberof PinchRecognizer
|
|
|
*/
|
|
|
defaults: {
|
|
|
event: 'tap',
|
|
|
pointers: 1,
|
|
|
taps: 1,
|
|
|
interval: 300, // max time between the multi-tap taps
|
|
|
time: 250, // max time of the pointer to be down (like finger on the screen)
|
|
|
threshold: 9, // a minimal movement is ok, but keep it low
|
|
|
posThreshold: 10 // a multi-tap can be a bit off the initial position
|
|
|
},
|
|
|
|
|
|
getTouchAction: function() {
|
|
|
return [TOUCH_ACTION_MANIPULATION];
|
|
|
},
|
|
|
|
|
|
process: function(input) {
|
|
|
var options = this.options;
|
|
|
|
|
|
var validPointers = input.pointers.length === options.pointers;
|
|
|
var validMovement = input.distance < options.threshold;
|
|
|
var validTouchTime = input.deltaTime < options.time;
|
|
|
|
|
|
this.reset();
|
|
|
|
|
|
if ((input.eventType & INPUT_START) && (this.count === 0)) {
|
|
|
return this.failTimeout();
|
|
|
}
|
|
|
|
|
|
// we only allow little movement
|
|
|
// and we've reached an end event, so a tap is possible
|
|
|
if (validMovement && validTouchTime && validPointers) {
|
|
|
if (input.eventType != INPUT_END) {
|
|
|
return this.failTimeout();
|
|
|
}
|
|
|
|
|
|
var validInterval = this.pTime ? (input.timeStamp - this.pTime < options.interval) : true;
|
|
|
var validMultiTap = !this.pCenter || getDistance(this.pCenter, input.center) < options.posThreshold;
|
|
|
|
|
|
this.pTime = input.timeStamp;
|
|
|
this.pCenter = input.center;
|
|
|
|
|
|
if (!validMultiTap || !validInterval) {
|
|
|
this.count = 1;
|
|
|
} else {
|
|
|
this.count += 1;
|
|
|
}
|
|
|
|
|
|
this._input = input;
|
|
|
|
|
|
// if tap count matches we have recognized it,
|
|
|
// else it has began recognizing...
|
|
|
var tapCount = this.count % options.taps;
|
|
|
if (tapCount === 0) {
|
|
|
// no failing requirements, immediately trigger the tap event
|
|
|
// or wait as long as the multitap interval to trigger
|
|
|
if (!this.hasRequireFailures()) {
|
|
|
return STATE_RECOGNIZED;
|
|
|
} else {
|
|
|
this._timer = setTimeoutContext(function() {
|
|
|
this.state = STATE_RECOGNIZED;
|
|
|
this.tryEmit();
|
|
|
}, options.interval, this);
|
|
|
return STATE_BEGAN;
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
return STATE_FAILED;
|
|
|
},
|
|
|
|
|
|
failTimeout: function() {
|
|
|
this._timer = setTimeoutContext(function() {
|
|
|
this.state = STATE_FAILED;
|
|
|
}, this.options.interval, this);
|
|
|
return STATE_FAILED;
|
|
|
},
|
|
|
|
|
|
reset: function() {
|
|
|
clearTimeout(this._timer);
|
|
|
},
|
|
|
|
|
|
emit: function() {
|
|
|
if (this.state == STATE_RECOGNIZED) {
|
|
|
this._input.tapCount = this.count;
|
|
|
this.manager.emit(this.options.event, this._input);
|
|
|
}
|
|
|
}
|
|
|
});
|
|
|
|
|
|
/**
|
|
|
* Simple way to create a manager with a default set of recognizers.
|
|
|
* @param {HTMLElement} element
|
|
|
* @param {Object} [options]
|
|
|
* @constructor
|
|
|
*/
|
|
|
function Hammer(element, options) {
|
|
|
options = options || {};
|
|
|
options.recognizers = ifUndefined(options.recognizers, Hammer.defaults.preset);
|
|
|
return new Manager(element, options);
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* @const {string}
|
|
|
*/
|
|
|
Hammer.VERSION = '2.0.8';
|
|
|
|
|
|
/**
|
|
|
* default settings
|
|
|
* @namespace
|
|
|
*/
|
|
|
Hammer.defaults = {
|
|
|
/**
|
|
|
* set if DOM events are being triggered.
|
|
|
* But this is slower and unused by simple implementations, so disabled by default.
|
|
|
* @type {Boolean}
|
|
|
* @default false
|
|
|
*/
|
|
|
domEvents: false,
|
|
|
|
|
|
/**
|
|
|
* The value for the touchAction property/fallback.
|
|
|
* When set to `compute` it will magically set the correct value based on the added recognizers.
|
|
|
* @type {String}
|
|
|
* @default compute
|
|
|
*/
|
|
|
touchAction: TOUCH_ACTION_COMPUTE,
|
|
|
|
|
|
/**
|
|
|
* @type {Boolean}
|
|
|
* @default true
|
|
|
*/
|
|
|
enable: true,
|
|
|
|
|
|
/**
|
|
|
* EXPERIMENTAL FEATURE -- can be removed/changed
|
|
|
* Change the parent input target element.
|
|
|
* If Null, then it is being set the to main element.
|
|
|
* @type {Null|EventTarget}
|
|
|
* @default null
|
|
|
*/
|
|
|
inputTarget: null,
|
|
|
|
|
|
/**
|
|
|
* force an input class
|
|
|
* @type {Null|Function}
|
|
|
* @default null
|
|
|
*/
|
|
|
inputClass: null,
|
|
|
|
|
|
/**
|
|
|
* Default recognizer setup when calling `Hammer()`
|
|
|
* When creating a new Manager these will be skipped.
|
|
|
* @type {Array}
|
|
|
*/
|
|
|
preset: [
|
|
|
// RecognizerClass, options, [recognizeWith, ...], [requireFailure, ...]
|
|
|
[RotateRecognizer, {enable: false}],
|
|
|
[PinchRecognizer, {enable: false}, ['rotate']],
|
|
|
[SwipeRecognizer, {direction: DIRECTION_HORIZONTAL}],
|
|
|
[PanRecognizer, {direction: DIRECTION_HORIZONTAL}, ['swipe']],
|
|
|
[TapRecognizer],
|
|
|
[TapRecognizer, {event: 'doubletap', taps: 2}, ['tap']],
|
|
|
[PressRecognizer]
|
|
|
],
|
|
|
|
|
|
/**
|
|
|
* Some CSS properties can be used to improve the working of Hammer.
|
|
|
* Add them to this method and they will be set when creating a new Manager.
|
|
|
* @namespace
|
|
|
*/
|
|
|
cssProps: {
|
|
|
/**
|
|
|
* Disables text selection to improve the dragging gesture. Mainly for desktop browsers.
|
|
|
* @type {String}
|
|
|
* @default 'none'
|
|
|
*/
|
|
|
userSelect: 'none',
|
|
|
|
|
|
/**
|
|
|
* Disable the Windows Phone grippers when pressing an element.
|
|
|
* @type {String}
|
|
|
* @default 'none'
|
|
|
*/
|
|
|
touchSelect: 'none',
|
|
|
|
|
|
/**
|
|
|
* Disables the default callout shown when you touch and hold a touch target.
|
|
|
* On iOS, when you touch and hold a touch target such as a link, Safari displays
|
|
|
* a callout containing information about the link. This property allows you to disable that callout.
|
|
|
* @type {String}
|
|
|
* @default 'none'
|
|
|
*/
|
|
|
touchCallout: 'none',
|
|
|
|
|
|
/**
|
|
|
* Specifies whether zooming is enabled. Used by IE10>
|
|
|
* @type {String}
|
|
|
* @default 'none'
|
|
|
*/
|
|
|
contentZooming: 'none',
|
|
|
|
|
|
/**
|
|
|
* Specifies that an entire element should be draggable instead of its contents. Mainly for desktop browsers.
|
|
|
* @type {String}
|
|
|
* @default 'none'
|
|
|
*/
|
|
|
userDrag: 'none',
|
|
|
|
|
|
/**
|
|
|
* Overrides the highlight color shown when the user taps a link or a JavaScript
|
|
|
* clickable element in iOS. This property obeys the alpha value, if specified.
|
|
|
* @type {String}
|
|
|
* @default 'rgba(0,0,0,0)'
|
|
|
*/
|
|
|
tapHighlightColor: 'rgba(0,0,0,0)'
|
|
|
}
|
|
|
};
|
|
|
|
|
|
var STOP = 1;
|
|
|
var FORCED_STOP = 2;
|
|
|
|
|
|
/**
|
|
|
* Manager
|
|
|
* @param {HTMLElement} element
|
|
|
* @param {Object} [options]
|
|
|
* @constructor
|
|
|
*/
|
|
|
function Manager(element, options) {
|
|
|
this.options = assign({}, Hammer.defaults, options || {});
|
|
|
|
|
|
this.options.inputTarget = this.options.inputTarget || element;
|
|
|
|
|
|
this.handlers = {};
|
|
|
this.session = {};
|
|
|
this.recognizers = [];
|
|
|
this.oldCssProps = {};
|
|
|
|
|
|
this.element = element;
|
|
|
this.input = createInputInstance(this);
|
|
|
this.touchAction = new TouchAction(this, this.options.touchAction);
|
|
|
|
|
|
toggleCssProps(this, true);
|
|
|
|
|
|
each(this.options.recognizers, function(item) {
|
|
|
var recognizer = this.add(new (item[0])(item[1]));
|
|
|
item[2] && recognizer.recognizeWith(item[2]);
|
|
|
item[3] && recognizer.requireFailure(item[3]);
|
|
|
}, this);
|
|
|
}
|
|
|
|
|
|
Manager.prototype = {
|
|
|
/**
|
|
|
* set options
|
|
|
* @param {Object} options
|
|
|
* @returns {Manager}
|
|
|
*/
|
|
|
set: function(options) {
|
|
|
assign(this.options, options);
|
|
|
|
|
|
// Options that need a little more setup
|
|
|
if (options.touchAction) {
|
|
|
this.touchAction.update();
|
|
|
}
|
|
|
if (options.inputTarget) {
|
|
|
// Clean up existing event listeners and reinitialize
|
|
|
this.input.destroy();
|
|
|
this.input.target = options.inputTarget;
|
|
|
this.input.init();
|
|
|
}
|
|
|
return this;
|
|
|
},
|
|
|
|
|
|
/**
|
|
|
* stop recognizing for this session.
|
|
|
* This session will be discarded, when a new [input]start event is fired.
|
|
|
* When forced, the recognizer cycle is stopped immediately.
|
|
|
* @param {Boolean} [force]
|
|
|
*/
|
|
|
stop: function(force) {
|
|
|
this.session.stopped = force ? FORCED_STOP : STOP;
|
|
|
},
|
|
|
|
|
|
/**
|
|
|
* run the recognizers!
|
|
|
* called by the inputHandler function on every movement of the pointers (touches)
|
|
|
* it walks through all the recognizers and tries to detect the gesture that is being made
|
|
|
* @param {Object} inputData
|
|
|
*/
|
|
|
recognize: function(inputData) {
|
|
|
var session = this.session;
|
|
|
if (session.stopped) {
|
|
|
return;
|
|
|
}
|
|
|
|
|
|
// run the touch-action polyfill
|
|
|
this.touchAction.preventDefaults(inputData);
|
|
|
|
|
|
var recognizer;
|
|
|
var recognizers = this.recognizers;
|
|
|
|
|
|
// this holds the recognizer that is being recognized.
|
|
|
// so the recognizer's state needs to be BEGAN, CHANGED, ENDED or RECOGNIZED
|
|
|
// if no recognizer is detecting a thing, it is set to `null`
|
|
|
var curRecognizer = session.curRecognizer;
|
|
|
|
|
|
// reset when the last recognizer is recognized
|
|
|
// or when we're in a new session
|
|
|
if (!curRecognizer || (curRecognizer && curRecognizer.state & STATE_RECOGNIZED)) {
|
|
|
curRecognizer = session.curRecognizer = null;
|
|
|
}
|
|
|
|
|
|
var i = 0;
|
|
|
while (i < recognizers.length) {
|
|
|
recognizer = recognizers[i];
|
|
|
|
|
|
// find out if we are allowed try to recognize the input for this one.
|
|
|
// 1. allow if the session is NOT forced stopped (see the .stop() method)
|
|
|
// 2. allow if we still haven't recognized a gesture in this session, or the this recognizer is the one
|
|
|
// that is being recognized.
|
|
|
// 3. allow if the recognizer is allowed to run simultaneous with the current recognized recognizer.
|
|
|
// this can be setup with the `recognizeWith()` method on the recognizer.
|
|
|
if (session.stopped !== FORCED_STOP && ( // 1
|
|
|
!curRecognizer || recognizer == curRecognizer || // 2
|
|
|
recognizer.canRecognizeWith(curRecognizer))) { // 3
|
|
|
recognizer.recognize(inputData);
|
|
|
} else {
|
|
|
recognizer.reset();
|
|
|
}
|
|
|
|
|
|
// if the recognizer has been recognizing the input as a valid gesture, we want to store this one as the
|
|
|
// current active recognizer. but only if we don't already have an active recognizer
|
|
|
if (!curRecognizer && recognizer.state & (STATE_BEGAN | STATE_CHANGED | STATE_ENDED)) {
|
|
|
curRecognizer = session.curRecognizer = recognizer;
|
|
|
}
|
|
|
i++;
|
|
|
}
|
|
|
},
|
|
|
|
|
|
/**
|
|
|
* get a recognizer by its event name.
|
|
|
* @param {Recognizer|String} recognizer
|
|
|
* @returns {Recognizer|Null}
|
|
|
*/
|
|
|
get: function(recognizer) {
|
|
|
if (recognizer instanceof Recognizer) {
|
|
|
return recognizer;
|
|
|
}
|
|
|
|
|
|
var recognizers = this.recognizers;
|
|
|
for (var i = 0; i < recognizers.length; i++) {
|
|
|
if (recognizers[i].options.event == recognizer) {
|
|
|
return recognizers[i];
|
|
|
}
|
|
|
}
|
|
|
return null;
|
|
|
},
|
|
|
|
|
|
/**
|
|
|
* add a recognizer to the manager
|
|
|
* existing recognizers with the same event name will be removed
|
|
|
* @param {Recognizer} recognizer
|
|
|
* @returns {Recognizer|Manager}
|
|
|
*/
|
|
|
add: function(recognizer) {
|
|
|
if (invokeArrayArg(recognizer, 'add', this)) {
|
|
|
return this;
|
|
|
}
|
|
|
|
|
|
// remove existing
|
|
|
var existing = this.get(recognizer.options.event);
|
|
|
if (existing) {
|
|
|
this.remove(existing);
|
|
|
}
|
|
|
|
|
|
this.recognizers.push(recognizer);
|
|
|
recognizer.manager = this;
|
|
|
|
|
|
this.touchAction.update();
|
|
|
return recognizer;
|
|
|
},
|
|
|
|
|
|
/**
|
|
|
* remove a recognizer by name or instance
|
|
|
* @param {Recognizer|String} recognizer
|
|
|
* @returns {Manager}
|
|
|
*/
|
|
|
remove: function(recognizer) {
|
|
|
if (invokeArrayArg(recognizer, 'remove', this)) {
|
|
|
return this;
|
|
|
}
|
|
|
|
|
|
recognizer = this.get(recognizer);
|
|
|
|
|
|
// let's make sure this recognizer exists
|
|
|
if (recognizer) {
|
|
|
var recognizers = this.recognizers;
|
|
|
var index = inArray(recognizers, recognizer);
|
|
|
|
|
|
if (index !== -1) {
|
|
|
recognizers.splice(index, 1);
|
|
|
this.touchAction.update();
|
|
|
}
|
|
|
}
|
|
|
|
|
|
return this;
|
|
|
},
|
|
|
|
|
|
/**
|
|
|
* bind event
|
|
|
* @param {String} events
|
|
|
* @param {Function} handler
|
|
|
* @returns {EventEmitter} this
|
|
|
*/
|
|
|
on: function(events, handler) {
|
|
|
if (events === undefined) {
|
|
|
return;
|
|
|
}
|
|
|
if (handler === undefined) {
|
|
|
return;
|
|
|
}
|
|
|
|
|
|
var handlers = this.handlers;
|
|
|
each(splitStr(events), function(event) {
|
|
|
handlers[event] = handlers[event] || [];
|
|
|
handlers[event].push(handler);
|
|
|
});
|
|
|
return this;
|
|
|
},
|
|
|
|
|
|
/**
|
|
|
* unbind event, leave emit blank to remove all handlers
|
|
|
* @param {String} events
|
|
|
* @param {Function} [handler]
|
|
|
* @returns {EventEmitter} this
|
|
|
*/
|
|
|
off: function(events, handler) {
|
|
|
if (events === undefined) {
|
|
|
return;
|
|
|
}
|
|
|
|
|
|
var handlers = this.handlers;
|
|
|
each(splitStr(events), function(event) {
|
|
|
if (!handler) {
|
|
|
delete handlers[event];
|
|
|
} else {
|
|
|
handlers[event] && handlers[event].splice(inArray(handlers[event], handler), 1);
|
|
|
}
|
|
|
});
|
|
|
return this;
|
|
|
},
|
|
|
|
|
|
/**
|
|
|
* emit event to the listeners
|
|
|
* @param {String} event
|
|
|
* @param {Object} data
|
|
|
*/
|
|
|
emit: function(event, data) {
|
|
|
// we also want to trigger dom events
|
|
|
if (this.options.domEvents) {
|
|
|
triggerDomEvent(event, data);
|
|
|
}
|
|
|
|
|
|
// no handlers, so skip it all
|
|
|
var handlers = this.handlers[event] && this.handlers[event].slice();
|
|
|
if (!handlers || !handlers.length) {
|
|
|
return;
|
|
|
}
|
|
|
|
|
|
data.type = event;
|
|
|
data.preventDefault = function() {
|
|
|
data.srcEvent.preventDefault();
|
|
|
};
|
|
|
|
|
|
var i = 0;
|
|
|
while (i < handlers.length) {
|
|
|
handlers[i](data);
|
|
|
i++;
|
|
|
}
|
|
|
},
|
|
|
|
|
|
/**
|
|
|
* destroy the manager and unbinds all events
|
|
|
* it doesn't unbind dom events, that is the user own responsibility
|
|
|
*/
|
|
|
destroy: function() {
|
|
|
this.element && toggleCssProps(this, false);
|
|
|
|
|
|
this.handlers = {};
|
|
|
this.session = {};
|
|
|
this.input.destroy();
|
|
|
this.element = null;
|
|
|
}
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* add/remove the css properties as defined in manager.options.cssProps
|
|
|
* @param {Manager} manager
|
|
|
* @param {Boolean} add
|
|
|
*/
|
|
|
function toggleCssProps(manager, add) {
|
|
|
var element = manager.element;
|
|
|
if (!element.style) {
|
|
|
return;
|
|
|
}
|
|
|
var prop;
|
|
|
each(manager.options.cssProps, function(value, name) {
|
|
|
prop = prefixed(element.style, name);
|
|
|
if (add) {
|
|
|
manager.oldCssProps[prop] = element.style[prop];
|
|
|
element.style[prop] = value;
|
|
|
} else {
|
|
|
element.style[prop] = manager.oldCssProps[prop] || '';
|
|
|
}
|
|
|
});
|
|
|
if (!add) {
|
|
|
manager.oldCssProps = {};
|
|
|
}
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* trigger dom event
|
|
|
* @param {String} event
|
|
|
* @param {Object} data
|
|
|
*/
|
|
|
function triggerDomEvent(event, data) {
|
|
|
var gestureEvent = document.createEvent('Event');
|
|
|
gestureEvent.initEvent(event, true, true);
|
|
|
gestureEvent.gesture = data;
|
|
|
data.target.dispatchEvent(gestureEvent);
|
|
|
}
|
|
|
|
|
|
assign(Hammer, {
|
|
|
INPUT_START: INPUT_START,
|
|
|
INPUT_MOVE: INPUT_MOVE,
|
|
|
INPUT_END: INPUT_END,
|
|
|
INPUT_CANCEL: INPUT_CANCEL,
|
|
|
|
|
|
STATE_POSSIBLE: STATE_POSSIBLE,
|
|
|
STATE_BEGAN: STATE_BEGAN,
|
|
|
STATE_CHANGED: STATE_CHANGED,
|
|
|
STATE_ENDED: STATE_ENDED,
|
|
|
STATE_RECOGNIZED: STATE_RECOGNIZED,
|
|
|
STATE_CANCELLED: STATE_CANCELLED,
|
|
|
STATE_FAILED: STATE_FAILED,
|
|
|
|
|
|
DIRECTION_NONE: DIRECTION_NONE,
|
|
|
DIRECTION_LEFT: DIRECTION_LEFT,
|
|
|
DIRECTION_RIGHT: DIRECTION_RIGHT,
|
|
|
DIRECTION_UP: DIRECTION_UP,
|
|
|
DIRECTION_DOWN: DIRECTION_DOWN,
|
|
|
DIRECTION_HORIZONTAL: DIRECTION_HORIZONTAL,
|
|
|
DIRECTION_VERTICAL: DIRECTION_VERTICAL,
|
|
|
DIRECTION_ALL: DIRECTION_ALL,
|
|
|
|
|
|
Manager: Manager,
|
|
|
Input: Input,
|
|
|
TouchAction: TouchAction,
|
|
|
|
|
|
TouchInput: TouchInput,
|
|
|
MouseInput: MouseInput,
|
|
|
PointerEventInput: PointerEventInput,
|
|
|
TouchMouseInput: TouchMouseInput,
|
|
|
SingleTouchInput: SingleTouchInput,
|
|
|
|
|
|
Recognizer: Recognizer,
|
|
|
AttrRecognizer: AttrRecognizer,
|
|
|
Tap: TapRecognizer,
|
|
|
Pan: PanRecognizer,
|
|
|
Swipe: SwipeRecognizer,
|
|
|
Pinch: PinchRecognizer,
|
|
|
Rotate: RotateRecognizer,
|
|
|
Press: PressRecognizer,
|
|
|
|
|
|
on: addEventListeners,
|
|
|
off: removeEventListeners,
|
|
|
each: each,
|
|
|
merge: merge,
|
|
|
extend: extend,
|
|
|
assign: assign,
|
|
|
inherit: inherit,
|
|
|
bindFn: bindFn,
|
|
|
prefixed: prefixed
|
|
|
});
|
|
|
|
|
|
// this prevents errors when Hammer is loaded in the presence of an AMD
|
|
|
// style loader but by script tag, not by the loader.
|
|
|
var freeGlobal = (typeof window !== 'undefined' ? window : (typeof self !== 'undefined' ? self : {})); // jshint ignore:line
|
|
|
freeGlobal.Hammer = Hammer;
|
|
|
|
|
|
if (typeof define === 'function' && define.amd) {
|
|
|
define(function() {
|
|
|
return Hammer;
|
|
|
});
|
|
|
} else if (typeof module != 'undefined' && module.exports) {
|
|
|
module.exports = Hammer;
|
|
|
} else {
|
|
|
window[exportName] = Hammer;
|
|
|
}
|
|
|
|
|
|
})(window, document, 'Hammer');
|
|
|
|
|
|
+ function($) {
|
|
|
"use strict";
|
|
|
|
|
|
var defaults;
|
|
|
|
|
|
$.modal = function(params, onOpen) {
|
|
|
params = $.extend({}, defaults, params);
|
|
|
|
|
|
|
|
|
var buttons = params.buttons;
|
|
|
|
|
|
var buttonsHtml = buttons.map(function(d, i) {
|
|
|
return '<a href="javascript:;" class="weui-dialog__btn ' + (d.className || "") + '">' + d.text + '</a>';
|
|
|
}).join("");
|
|
|
|
|
|
var tpl = '<div class="weui-dialog">' +
|
|
|
'<div class="weui-dialog__hd"><strong class="weui-dialog__title">' + params.title + '</strong></div>' +
|
|
|
( params.text ? '<div class="weui-dialog__bd">'+params.text+'</div>' : '')+
|
|
|
'<div class="weui-dialog__ft">' + buttonsHtml + '</div>' +
|
|
|
'</div>';
|
|
|
|
|
|
var dialog = $.openModal(tpl, onOpen);
|
|
|
|
|
|
dialog.find(".weui-dialog__btn").each(function(i, e) {
|
|
|
var el = $(e);
|
|
|
el.click(function() {
|
|
|
//先关闭对话框,再调用回调函数
|
|
|
if(params.autoClose) $.closeModal();
|
|
|
|
|
|
if(buttons[i].onClick) {
|
|
|
buttons[i].onClick.call(dialog);
|
|
|
}
|
|
|
});
|
|
|
});
|
|
|
|
|
|
return dialog;
|
|
|
};
|
|
|
|
|
|
$.openModal = function(tpl, onOpen) {
|
|
|
var mask = $("<div class='weui-mask'></div>").appendTo(document.body);
|
|
|
mask.show();
|
|
|
|
|
|
var dialog = $(tpl).appendTo(document.body);
|
|
|
|
|
|
if (onOpen) {
|
|
|
dialog.transitionEnd(function () {
|
|
|
onOpen.call(dialog);
|
|
|
});
|
|
|
}
|
|
|
|
|
|
dialog.show();
|
|
|
mask.addClass("weui-mask--visible");
|
|
|
dialog.addClass("weui-dialog--visible");
|
|
|
|
|
|
|
|
|
return dialog;
|
|
|
}
|
|
|
|
|
|
$.closeModal = function() {
|
|
|
$(".weui-mask--visible").removeClass("weui-mask--visible").transitionEnd(function() {
|
|
|
$(this).remove();
|
|
|
});
|
|
|
$(".weui-dialog--visible").removeClass("weui-dialog--visible").transitionEnd(function() {
|
|
|
$(this).remove();
|
|
|
});
|
|
|
};
|
|
|
|
|
|
$.alert = function(text, title, onOK) {
|
|
|
var config;
|
|
|
if (typeof text === 'object') {
|
|
|
config = text;
|
|
|
} else {
|
|
|
if (typeof title === 'function') {
|
|
|
onOK = arguments[1];
|
|
|
title = undefined;
|
|
|
}
|
|
|
|
|
|
config = {
|
|
|
text: text,
|
|
|
title: title,
|
|
|
onOK: onOK
|
|
|
}
|
|
|
}
|
|
|
return $.modal({
|
|
|
text: config.text,
|
|
|
title: config.title,
|
|
|
buttons: [{
|
|
|
text: defaults.buttonOK,
|
|
|
className: "primary",
|
|
|
onClick: config.onOK
|
|
|
}]
|
|
|
});
|
|
|
}
|
|
|
|
|
|
$.confirm = function(text, title, onOK, onCancel) {
|
|
|
var config;
|
|
|
if (typeof text === 'object') {
|
|
|
config = text
|
|
|
} else {
|
|
|
if (typeof title === 'function') {
|
|
|
onCancel = arguments[2];
|
|
|
onOK = arguments[1];
|
|
|
title = undefined;
|
|
|
}
|
|
|
|
|
|
config = {
|
|
|
text: text,
|
|
|
title: title,
|
|
|
onOK: onOK,
|
|
|
onCancel: onCancel
|
|
|
}
|
|
|
}
|
|
|
return $.modal({
|
|
|
text: config.text,
|
|
|
title: config.title,
|
|
|
buttons: [
|
|
|
{
|
|
|
text: defaults.buttonCancel,
|
|
|
className: "default",
|
|
|
onClick: config.onCancel
|
|
|
},
|
|
|
{
|
|
|
text: defaults.buttonOK,
|
|
|
className: "primary",
|
|
|
onClick: config.onOK
|
|
|
}]
|
|
|
});
|
|
|
};
|
|
|
|
|
|
//如果参数过多,建议通过 config 对象进行配置,而不是传入多个参数。
|
|
|
$.prompt = function(text, title, onOK, onCancel, input) {
|
|
|
var config;
|
|
|
if (typeof text === 'object') {
|
|
|
config = text;
|
|
|
} else {
|
|
|
if (typeof title === 'function') {
|
|
|
input = arguments[3];
|
|
|
onCancel = arguments[2];
|
|
|
onOK = arguments[1];
|
|
|
title = undefined;
|
|
|
}
|
|
|
config = {
|
|
|
text: text,
|
|
|
title: title,
|
|
|
input: input,
|
|
|
onOK: onOK,
|
|
|
onCancel: onCancel,
|
|
|
empty: false //allow empty
|
|
|
}
|
|
|
}
|
|
|
|
|
|
var modal = $.modal({
|
|
|
text: '<p class="weui-prompt-text">'+(config.text || '')+'</p><input type="text" class="weui-input weui-prompt-input" id="weui-prompt-input" value="' + (config.input || '') + '" />',
|
|
|
title: config.title,
|
|
|
autoClose: false,
|
|
|
buttons: [
|
|
|
{
|
|
|
text: defaults.buttonCancel,
|
|
|
className: "default",
|
|
|
onClick: function () {
|
|
|
$.closeModal();
|
|
|
config.onCancel && config.onCancel.call(modal);
|
|
|
}
|
|
|
},
|
|
|
{
|
|
|
text: defaults.buttonOK,
|
|
|
className: "primary",
|
|
|
onClick: function() {
|
|
|
var input = $("#weui-prompt-input").val();
|
|
|
if (!config.empty && (input === "" || input === null)) {
|
|
|
modal.find('.weui-prompt-input').focus()[0].select();
|
|
|
return false;
|
|
|
}
|
|
|
$.closeModal();
|
|
|
config.onOK && config.onOK.call(modal, input);
|
|
|
}
|
|
|
}]
|
|
|
}, function () {
|
|
|
this.find('.weui-prompt-input').focus()[0].select();
|
|
|
});
|
|
|
|
|
|
return modal;
|
|
|
};
|
|
|
|
|
|
//如果参数过多,建议通过 config 对象进行配置,而不是传入多个参数。
|
|
|
$.login = function(text, title, onOK, onCancel, username, password) {
|
|
|
var config;
|
|
|
if (typeof text === 'object') {
|
|
|
config = text;
|
|
|
} else {
|
|
|
if (typeof title === 'function') {
|
|
|
password = arguments[4];
|
|
|
username = arguments[3];
|
|
|
onCancel = arguments[2];
|
|
|
onOK = arguments[1];
|
|
|
title = undefined;
|
|
|
}
|
|
|
config = {
|
|
|
text: text,
|
|
|
title: title,
|
|
|
username: username,
|
|
|
password: password,
|
|
|
onOK: onOK,
|
|
|
onCancel: onCancel
|
|
|
}
|
|
|
}
|
|
|
|
|
|
var modal = $.modal({
|
|
|
text: '<p class="weui-prompt-text">'+(config.text || '')+'</p>' +
|
|
|
'<input type="text" class="weui-input weui-prompt-input" id="weui-prompt-username" value="' + (config.username || '') + '" placeholder="输入用户名" />' +
|
|
|
'<input type="password" class="weui-input weui-prompt-input" id="weui-prompt-password" value="' + (config.password || '') + '" placeholder="输入密码" />',
|
|
|
title: config.title,
|
|
|
autoClose: false,
|
|
|
buttons: [
|
|
|
{
|
|
|
text: defaults.buttonCancel,
|
|
|
className: "default",
|
|
|
onClick: function () {
|
|
|
$.closeModal();
|
|
|
config.onCancel && config.onCancel.call(modal);
|
|
|
}
|
|
|
}, {
|
|
|
text: defaults.buttonOK,
|
|
|
className: "primary",
|
|
|
onClick: function() {
|
|
|
var username = $("#weui-prompt-username").val();
|
|
|
var password = $("#weui-prompt-password").val();
|
|
|
if (!config.empty && (username === "" || username === null)) {
|
|
|
modal.find('#weui-prompt-username').focus()[0].select();
|
|
|
return false;
|
|
|
}
|
|
|
if (!config.empty && (password === "" || password === null)) {
|
|
|
modal.find('#weui-prompt-password').focus()[0].select();
|
|
|
return false;
|
|
|
}
|
|
|
$.closeModal();
|
|
|
config.onOK && config.onOK.call(modal, username, password);
|
|
|
}
|
|
|
}]
|
|
|
}, function () {
|
|
|
this.find('#weui-prompt-username').focus()[0].select();
|
|
|
});
|
|
|
|
|
|
return modal;
|
|
|
};
|
|
|
|
|
|
defaults = $.modal.prototype.defaults = {
|
|
|
title: "提示",
|
|
|
text: undefined,
|
|
|
buttonOK: "确定",
|
|
|
buttonCancel: "取消",
|
|
|
buttons: [{
|
|
|
text: "确定",
|
|
|
className: "primary"
|
|
|
}],
|
|
|
autoClose: true //点击按钮自动关闭对话框,如果你不希望点击按钮就关闭对话框,可以把这个设置为false
|
|
|
};
|
|
|
|
|
|
}($);
|
|
|
|
|
|
+ function($) {
|
|
|
"use strict";
|
|
|
|
|
|
var defaults;
|
|
|
|
|
|
var show = function(html, className) {
|
|
|
className = className || "";
|
|
|
var mask = $("<div class='weui-mask_transparent'></div>").appendTo(document.body);
|
|
|
|
|
|
var tpl = '<div class="weui-toast ' + className + '">' + html + '</div>';
|
|
|
var dialog = $(tpl).appendTo(document.body);
|
|
|
|
|
|
dialog.addClass("weui-toast--visible");
|
|
|
dialog.show();
|
|
|
};
|
|
|
|
|
|
var hide = function(callback) {
|
|
|
$(".weui-mask_transparent").remove();
|
|
|
var done = false;
|
|
|
var $el = $(".weui-toast--visible").removeClass("weui-toast--visible").transitionEnd(function() {
|
|
|
var $this = $(this);
|
|
|
$this.remove();
|
|
|
callback && callback();
|
|
|
done = true
|
|
|
});
|
|
|
|
|
|
setTimeout(function () {
|
|
|
if (!done) {
|
|
|
$el.remove()
|
|
|
callback && callback();
|
|
|
}
|
|
|
}, 1000)
|
|
|
}
|
|
|
|
|
|
$.toast = function(text, style, callback) {
|
|
|
if(typeof style === "function") {
|
|
|
callback = style;
|
|
|
}
|
|
|
var className, iconClassName = 'weui-icon-success-no-circle';
|
|
|
var duration = toastDefaults.duration;
|
|
|
if(style == "cancel") {
|
|
|
className = "weui-toast_cancel";
|
|
|
iconClassName = 'weui-icon-cancel'
|
|
|
} else if(style == "forbidden") {
|
|
|
className = "weui-toast--forbidden";
|
|
|
iconClassName = 'weui-icon-warn'
|
|
|
} else if(style == "text") {
|
|
|
className = "weui-toast--text";
|
|
|
} else if(typeof style === typeof 1) {
|
|
|
duration = style
|
|
|
}
|
|
|
show('<i class="' + iconClassName + ' weui-icon_toast"></i><p class="weui-toast_content">' + (text || "已经完成") + '</p>', className);
|
|
|
|
|
|
setTimeout(function() {
|
|
|
hide(callback);
|
|
|
}, duration);
|
|
|
}
|
|
|
|
|
|
$.showLoading = function(text) {
|
|
|
var html = '<div class="weui_loading">';
|
|
|
html += '<i class="weui-loading weui-icon_toast"></i>';
|
|
|
html += '</div>';
|
|
|
html += '<p class="weui-toast_content">' + (text || "数据加载中") + '</p>';
|
|
|
show(html, 'weui_loading_toast');
|
|
|
}
|
|
|
|
|
|
$.hideLoading = function() {
|
|
|
hide();
|
|
|
}
|
|
|
|
|
|
var toastDefaults = $.toast.prototype.defaults = {
|
|
|
duration: 2500
|
|
|
}
|
|
|
|
|
|
}($);
|
|
|
|
|
|
+ function($) {
|
|
|
"use strict";
|
|
|
|
|
|
var defaults;
|
|
|
|
|
|
var show = function(params) {
|
|
|
|
|
|
var mask = $("<div class='weui-mask weui-actions_mask'></div>").appendTo(document.body);
|
|
|
|
|
|
var actions = params.actions || [];
|
|
|
|
|
|
var actionsHtml = actions.map(function(d, i) {
|
|
|
return '<div class="weui-actionsheet__cell ' + (d.className || "") + '">' + d.text + '</div>';
|
|
|
}).join("");
|
|
|
|
|
|
var titleHtml = "";
|
|
|
|
|
|
if (params.title) {
|
|
|
titleHtml = '<div class="weui-actionsheet__title"><p class="weui-actionsheet__title-text">' + params.title + '</p></div>';
|
|
|
}
|
|
|
|
|
|
var tpl = '<div class="weui-actionsheet " id="weui-actionsheet">'+
|
|
|
titleHtml +
|
|
|
'<div class="weui-actionsheet__menu">'+
|
|
|
actionsHtml +
|
|
|
'</div>'+
|
|
|
'<div class="weui-actionsheet__action">'+
|
|
|
'<div class="weui-actionsheet__cell weui-actionsheet_cancel">取消</div>'+
|
|
|
'</div>'+
|
|
|
'</div>';
|
|
|
var dialog = $(tpl).appendTo(document.body);
|
|
|
|
|
|
dialog.find(".weui-actionsheet__menu .weui-actionsheet__cell, .weui-actionsheet__action .weui-actionsheet__cell").each(function(i, e) {
|
|
|
$(e).click(function() {
|
|
|
$.closeActions();
|
|
|
params.onClose && params.onClose();
|
|
|
if(actions[i] && actions[i].onClick) {
|
|
|
actions[i].onClick();
|
|
|
}
|
|
|
})
|
|
|
});
|
|
|
|
|
|
mask.show();
|
|
|
dialog.show();
|
|
|
mask.addClass("weui-mask--visible");
|
|
|
dialog.addClass("weui-actionsheet_toggle");
|
|
|
};
|
|
|
|
|
|
var hide = function() {
|
|
|
$(".weui-mask").removeClass("weui-mask--visible").transitionEnd(function() {
|
|
|
$(this).remove();
|
|
|
});
|
|
|
$(".weui-actionsheet").removeClass("weui-actionsheet_toggle").transitionEnd(function() {
|
|
|
$(this).remove();
|
|
|
});
|
|
|
}
|
|
|
|
|
|
$.actions = function(params) {
|
|
|
params = $.extend({}, defaults, params);
|
|
|
show(params);
|
|
|
}
|
|
|
|
|
|
$.closeActions = function() {
|
|
|
hide();
|
|
|
}
|
|
|
|
|
|
$(document).on("click", ".weui-actions_mask", function() {
|
|
|
$.closeActions();
|
|
|
});
|
|
|
|
|
|
var defaults = $.actions.prototype.defaults = {
|
|
|
title: undefined,
|
|
|
onClose: undefined,
|
|
|
/*actions: [{
|
|
|
text: "菜单",
|
|
|
className: "color-danger",
|
|
|
onClick: function() {
|
|
|
console.log(1);
|
|
|
}
|
|
|
},{
|
|
|
text: "菜单2",
|
|
|
className: "color-success",
|
|
|
onClick: function() {
|
|
|
console.log(2);
|
|
|
}
|
|
|
}]*/
|
|
|
}
|
|
|
|
|
|
}($);
|
|
|
|
|
|
/* ===============================================================================
|
|
|
************ Pull to refreh ************
|
|
|
=============================================================================== */
|
|
|
/* global $:true */
|
|
|
|
|
|
+function ($) {
|
|
|
"use strict";
|
|
|
|
|
|
var PTR = function(el, opt) {
|
|
|
if (typeof opt === typeof function () {}) {
|
|
|
opt = {
|
|
|
onRefresh: opt
|
|
|
}
|
|
|
}
|
|
|
if (typeof opt === typeof 'a') {
|
|
|
opt = undefined
|
|
|
}
|
|
|
this.opt = $.extend(PTR.defaults, opt || {});
|
|
|
this.container = $(el);
|
|
|
this.attachEvents();
|
|
|
}
|
|
|
|
|
|
PTR.defaults = {
|
|
|
distance: 50,
|
|
|
onRefresh: undefined,
|
|
|
onPull: undefined
|
|
|
}
|
|
|
|
|
|
PTR.prototype.touchStart = function(e) {
|
|
|
if(this.container.hasClass("refreshing")) return;
|
|
|
var p = $.getTouchPosition(e);
|
|
|
this.start = p;
|
|
|
this.diffX = this.diffY = 0;
|
|
|
};
|
|
|
|
|
|
PTR.prototype.touchMove= function(e) {
|
|
|
if(this.container.hasClass("refreshing")) return;
|
|
|
if(!this.start) return false;
|
|
|
if(this.container.scrollTop() > 0) return;
|
|
|
var p = $.getTouchPosition(e);
|
|
|
this.diffX = p.x - this.start.x;
|
|
|
this.diffY = p.y - this.start.y;
|
|
|
if (Math.abs(this.diffX) > Math.abs(this.diffY)) return true; // 说明是左右方向的拖动
|
|
|
if(this.diffY < 0) return;
|
|
|
this.container.addClass("touching");
|
|
|
e.preventDefault();
|
|
|
e.stopPropagation();
|
|
|
this.diffY = Math.pow(this.diffY, 0.75);
|
|
|
this.container.css("transform", "translate3d(0, "+this.diffY+"px, 0)");
|
|
|
this.triggerPull(this.diffY)
|
|
|
};
|
|
|
PTR.prototype.touchEnd = function() {
|
|
|
this.start = false;
|
|
|
if(this.diffY <= 0 || this.container.hasClass("refreshing")) return;
|
|
|
this.container.removeClass("touching");
|
|
|
this.container.removeClass("pull-down pull-up");
|
|
|
this.container.css("transform", "");
|
|
|
if(Math.abs(this.diffY) <= this.opt.distance) {
|
|
|
} else {
|
|
|
this.triggerPullToRefresh();
|
|
|
}
|
|
|
};
|
|
|
|
|
|
PTR.prototype.triggerPullToRefresh = function() {
|
|
|
this.triggerPull(this.opt.distance)
|
|
|
this.container.removeClass('pull-up').addClass("refreshing");
|
|
|
if (this.opt.onRefresh) {
|
|
|
this.opt.onRefresh.call(this)
|
|
|
}
|
|
|
this.container.trigger("pull-to-refresh");
|
|
|
}
|
|
|
|
|
|
PTR.prototype.triggerPull = function(diffY) {
|
|
|
|
|
|
if(diffY < this.opt.distance) {
|
|
|
this.container.removeClass("pull-up").addClass("pull-down");
|
|
|
} else {
|
|
|
this.container.removeClass("pull-down").addClass("pull-up");
|
|
|
}
|
|
|
|
|
|
if (this.opt.onPull) {
|
|
|
this.opt.onPull.call(this, Math.floor(diffY / this.opt.distance * 100))
|
|
|
}
|
|
|
this.container.trigger("pull");
|
|
|
}
|
|
|
|
|
|
PTR.prototype.pullToRefreshDone = function() {
|
|
|
this.container.removeClass("refreshing");
|
|
|
}
|
|
|
|
|
|
PTR.prototype.attachEvents = function() {
|
|
|
var el = this.container;
|
|
|
el.addClass("weui-pull-to-refresh");
|
|
|
el.on($.touchEvents.start, $.proxy(this.touchStart, this));
|
|
|
el.on($.touchEvents.move, $.proxy(this.touchMove, this));
|
|
|
el.on($.touchEvents.end, $.proxy(this.touchEnd, this));
|
|
|
};
|
|
|
|
|
|
var pullToRefreshDone = function(el) {
|
|
|
$(el).removeClass("refreshing");
|
|
|
}
|
|
|
|
|
|
$.fn.pullToRefresh = function(opt) {
|
|
|
return this.each(function() {
|
|
|
var $this = $(this)
|
|
|
var ptr = $this.data('ptr')
|
|
|
if (!ptr) $this.data('ptr', ptr = new PTR(this, opt))
|
|
|
if (typeof opt === typeof 'a') {
|
|
|
ptr[opt].call(ptr)
|
|
|
}
|
|
|
});
|
|
|
}
|
|
|
|
|
|
$.fn.pullToRefreshDone = function() {
|
|
|
return this.each(function() {
|
|
|
pullToRefreshDone(this);
|
|
|
});
|
|
|
}
|
|
|
|
|
|
}($);
|
|
|
|
|
|
/* ===============================================================================
|
|
|
************ Infinite ************
|
|
|
=============================================================================== */
|
|
|
/* global $:true */
|
|
|
+function ($) {
|
|
|
"use strict";
|
|
|
|
|
|
// fix https://github.com/lihongxun945/jquery-weui/issues/442
|
|
|
// chrome will always return 0, when use document.body.scrollTop
|
|
|
// https://stackoverflow.com/questions/43717316/google-chrome-document-body-scrolltop-always-returns-0
|
|
|
var getOffset = function (container) {
|
|
|
var tagName = container[0].tagName.toUpperCase()
|
|
|
var scrollTop
|
|
|
if (tagName === 'BODY' || tagName === 'HTML') {
|
|
|
scrollTop = container.scrollTop() || $(window).scrollTop()
|
|
|
} else {
|
|
|
scrollTop = container.scrollTop()
|
|
|
}
|
|
|
var offset = container.scrollHeight() - ($(window).height() + scrollTop)
|
|
|
console.log(offset)
|
|
|
return offset
|
|
|
}
|
|
|
|
|
|
var Infinite = function(el, distance) {
|
|
|
this.container = $(el);
|
|
|
this.container.data("infinite", this);
|
|
|
this.distance = distance || 50;
|
|
|
this.attachEvents();
|
|
|
}
|
|
|
|
|
|
Infinite.prototype.scroll = function() {
|
|
|
var container = this.container;
|
|
|
this._check();
|
|
|
}
|
|
|
|
|
|
Infinite.prototype.attachEvents = function(off) {
|
|
|
var el = this.container;
|
|
|
var scrollContainer = (el[0].tagName.toUpperCase() === "BODY" ? $(document) : el);
|
|
|
scrollContainer[off ? "off" : "on"]("scroll", $.proxy(this.scroll, this));
|
|
|
};
|
|
|
Infinite.prototype.detachEvents = function(off) {
|
|
|
this.attachEvents(true);
|
|
|
}
|
|
|
Infinite.prototype._check = function() {
|
|
|
var offset = getOffset(this.container);
|
|
|
if(Math.abs(offset) <= this.distance) {
|
|
|
this.container.trigger("infinite");
|
|
|
}
|
|
|
}
|
|
|
|
|
|
var infinite = function(el) {
|
|
|
attachEvents(el);
|
|
|
}
|
|
|
|
|
|
$.fn.infinite = function(distance) {
|
|
|
return this.each(function() {
|
|
|
new Infinite(this, distance);
|
|
|
});
|
|
|
}
|
|
|
$.fn.destroyInfinite = function() {
|
|
|
return this.each(function() {
|
|
|
var infinite = $(this).data("infinite");
|
|
|
if(infinite && infinite.detachEvents) infinite.detachEvents();
|
|
|
});
|
|
|
}
|
|
|
|
|
|
}($);
|
|
|
|
|
|
/* global $:true */
|
|
|
+function ($) {
|
|
|
"use strict";
|
|
|
|
|
|
var ITEM_ON = "weui-bar__item--on";
|
|
|
|
|
|
var showTab = function(a) {
|
|
|
var $a = $(a);
|
|
|
if($a.hasClass(ITEM_ON)) return;
|
|
|
var href = $a.attr("href");
|
|
|
|
|
|
if(!/^#/.test(href)) return ;
|
|
|
|
|
|
$a.parent().find("."+ITEM_ON).removeClass(ITEM_ON);
|
|
|
$a.addClass(ITEM_ON);
|
|
|
|
|
|
var bd = $a.parents(".weui-tab").find(".weui-tab__bd");
|
|
|
|
|
|
bd.find(".weui-tab__bd-item--active").removeClass("weui-tab__bd-item--active");
|
|
|
|
|
|
$(href).addClass("weui-tab__bd-item--active");
|
|
|
}
|
|
|
|
|
|
$.showTab = showTab;
|
|
|
|
|
|
$(document).on("click", ".weui-navbar__item, .weui-tabbar__item", function(e) {
|
|
|
var $a = $(e.currentTarget);
|
|
|
var href = $a.attr("href");
|
|
|
if($a.hasClass(ITEM_ON)) return;
|
|
|
if(!/^#/.test(href)) return;
|
|
|
|
|
|
e.preventDefault();
|
|
|
|
|
|
showTab($a);
|
|
|
});
|
|
|
|
|
|
}($);
|
|
|
|
|
|
/* global $:true */
|
|
|
+ function($) {
|
|
|
"use strict";
|
|
|
|
|
|
$(document).on("click touchstart", ".weui-search-bar__label", function(e) {
|
|
|
$(e.target).parents(".weui-search-bar").addClass("weui-search-bar_focusing").find('input').focus();
|
|
|
})
|
|
|
/*
|
|
|
.on("blur", ".weui-search-bar__input", function(e) {
|
|
|
var $input = $(e.target);
|
|
|
if(!$input.val()) $input.parents(".weui-search-bar").removeClass("weui-search-bar_focusing");
|
|
|
})
|
|
|
*/
|
|
|
.on("click", ".weui-search-bar__cancel-btn", function(e) {
|
|
|
var $input = $(e.target).parents(".weui-search-bar").removeClass("weui-search-bar_focusing").find(".weui-search-bar__input").val("").blur();
|
|
|
})
|
|
|
.on("click", ".weui-icon-clear", function(e) {
|
|
|
var $input = $(e.target).parents(".weui-search-bar").find(".weui-search-bar__input").val("").focus();
|
|
|
});
|
|
|
|
|
|
}($);
|
|
|
|
|
|
/*===========================
|
|
|
Device/OS Detection
|
|
|
===========================*/
|
|
|
/* global $:true */
|
|
|
;(function ($) {
|
|
|
"use strict";
|
|
|
var device = {};
|
|
|
var ua = navigator.userAgent;
|
|
|
|
|
|
var android = ua.match(/(Android);?[\s\/]+([\d.]+)?/);
|
|
|
var ipad = ua.match(/(iPad).*OS\s([\d_]+)/);
|
|
|
var ipod = ua.match(/(iPod)(.*OS\s([\d_]+))?/);
|
|
|
var iphone = !ipad && ua.match(/(iPhone\sOS)\s([\d_]+)/);
|
|
|
|
|
|
device.ios = device.android = device.iphone = device.ipad = device.androidChrome = false;
|
|
|
|
|
|
// Android
|
|
|
if (android) {
|
|
|
device.os = 'android';
|
|
|
device.osVersion = android[2];
|
|
|
device.android = true;
|
|
|
device.androidChrome = ua.toLowerCase().indexOf('chrome') >= 0;
|
|
|
}
|
|
|
if (ipad || iphone || ipod) {
|
|
|
device.os = 'ios';
|
|
|
device.ios = true;
|
|
|
}
|
|
|
// iOS
|
|
|
if (iphone && !ipod) {
|
|
|
device.osVersion = iphone[2].replace(/_/g, '.');
|
|
|
device.iphone = true;
|
|
|
}
|
|
|
if (ipad) {
|
|
|
device.osVersion = ipad[2].replace(/_/g, '.');
|
|
|
device.ipad = true;
|
|
|
}
|
|
|
if (ipod) {
|
|
|
device.osVersion = ipod[3] ? ipod[3].replace(/_/g, '.') : null;
|
|
|
device.iphone = true;
|
|
|
}
|
|
|
// iOS 8+ changed UA
|
|
|
if (device.ios && device.osVersion && ua.indexOf('Version/') >= 0) {
|
|
|
if (device.osVersion.split('.')[0] === '10') {
|
|
|
device.osVersion = ua.toLowerCase().split('version/')[1].split(' ')[0];
|
|
|
}
|
|
|
}
|
|
|
|
|
|
// Webview
|
|
|
device.webView = (iphone || ipad || ipod) && ua.match(/.*AppleWebKit(?!.*Safari)/i);
|
|
|
|
|
|
// Minimal UI
|
|
|
if (device.os && device.os === 'ios') {
|
|
|
var osVersionArr = device.osVersion.split('.');
|
|
|
device.minimalUi = !device.webView &&
|
|
|
(ipod || iphone) &&
|
|
|
(osVersionArr[0] * 1 === 7 ? osVersionArr[1] * 1 >= 1 : osVersionArr[0] * 1 > 7) &&
|
|
|
$('meta[name="viewport"]').length > 0 && $('meta[name="viewport"]').attr('content').indexOf('minimal-ui') >= 0;
|
|
|
}
|
|
|
|
|
|
// Check for status bar and fullscreen app mode
|
|
|
var windowWidth = $(window).width();
|
|
|
var windowHeight = $(window).height();
|
|
|
device.statusBar = false;
|
|
|
if (device.webView && (windowWidth * windowHeight === screen.width * screen.height)) {
|
|
|
device.statusBar = true;
|
|
|
}
|
|
|
else {
|
|
|
device.statusBar = false;
|
|
|
}
|
|
|
|
|
|
// Classes
|
|
|
var classNames = [];
|
|
|
|
|
|
// Pixel Ratio
|
|
|
device.pixelRatio = window.devicePixelRatio || 1;
|
|
|
classNames.push('pixel-ratio-' + Math.floor(device.pixelRatio));
|
|
|
if (device.pixelRatio >= 2) {
|
|
|
classNames.push('retina');
|
|
|
}
|
|
|
|
|
|
// OS classes
|
|
|
if (device.os) {
|
|
|
classNames.push(device.os, device.os + '-' + device.osVersion.split('.')[0], device.os + '-' + device.osVersion.replace(/\./g, '-'));
|
|
|
if (device.os === 'ios') {
|
|
|
var major = parseInt(device.osVersion.split('.')[0], 10);
|
|
|
for (var i = major - 1; i >= 6; i--) {
|
|
|
classNames.push('ios-gt-' + i);
|
|
|
}
|
|
|
}
|
|
|
|
|
|
}
|
|
|
// Status bar classes
|
|
|
if (device.statusBar) {
|
|
|
classNames.push('with-statusbar-overlay');
|
|
|
}
|
|
|
else {
|
|
|
$('html').removeClass('with-statusbar-overlay');
|
|
|
}
|
|
|
|
|
|
// Add html classes
|
|
|
if (classNames.length > 0) $('html').addClass(classNames.join(' '));
|
|
|
|
|
|
$.device = device;
|
|
|
})($);
|
|
|
|
|
|
/*======================================================
|
|
|
************ Picker ************
|
|
|
======================================================*/
|
|
|
/* global $:true */
|
|
|
/* jshint unused:false */
|
|
|
/* jshint multistr:true */
|
|
|
+ function($) {
|
|
|
"use strict";
|
|
|
var Picker = function (params) {
|
|
|
var p = this;
|
|
|
var defaults = {
|
|
|
updateValuesOnMomentum: false,
|
|
|
updateValuesOnTouchmove: true,
|
|
|
rotateEffect: false,
|
|
|
momentumRatio: 7,
|
|
|
freeMode: false,
|
|
|
// Common settings
|
|
|
scrollToInput: true,
|
|
|
inputReadOnly: true,
|
|
|
toolbar: true,
|
|
|
toolbarCloseText: '完成',
|
|
|
title: '请选择',
|
|
|
toolbarTemplate: '<div class="toolbar">\
|
|
|
<div class="toolbar-inner">\
|
|
|
<a href="javascript:;" class="picker-button close-picker">{{closeText}}</a>\
|
|
|
<h1 class="title">{{title}}</h1>\
|
|
|
</div>\
|
|
|
</div>',
|
|
|
};
|
|
|
params = params || {};
|
|
|
for (var def in defaults) {
|
|
|
if (typeof params[def] === 'undefined') {
|
|
|
params[def] = defaults[def];
|
|
|
}
|
|
|
}
|
|
|
p.params = params;
|
|
|
p.cols = [];
|
|
|
p.initialized = false;
|
|
|
|
|
|
// Inline flag
|
|
|
p.inline = p.params.container ? true : false;
|
|
|
|
|
|
// 3D Transforms origin bug, only on safari
|
|
|
var originBug = $.device.ios || (navigator.userAgent.toLowerCase().indexOf('safari') >= 0 && navigator.userAgent.toLowerCase().indexOf('chrome') < 0) && !$.device.android;
|
|
|
|
|
|
// Should be converted to popover
|
|
|
function isPopover() {
|
|
|
var toPopover = false;
|
|
|
if (!p.params.convertToPopover && !p.params.onlyInPopover) return toPopover;
|
|
|
if (!p.inline && p.params.input) {
|
|
|
if (p.params.onlyInPopover) toPopover = true;
|
|
|
else {
|
|
|
if ($.device.ios) {
|
|
|
toPopover = $.device.ipad ? true : false;
|
|
|
}
|
|
|
else {
|
|
|
if ($(window).width() >= 768) toPopover = true;
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
return toPopover;
|
|
|
}
|
|
|
function inPopover() {
|
|
|
if (p.opened && p.container && p.container.length > 0 && p.container.parents('.popover').length > 0) return true;
|
|
|
else return false;
|
|
|
}
|
|
|
|
|
|
// Value
|
|
|
p.setValue = function (arrValues, transition) {
|
|
|
var valueIndex = 0;
|
|
|
for (var i = 0; i < p.cols.length; i++) {
|
|
|
if (p.cols[i] && !p.cols[i].divider) {
|
|
|
p.cols[i].setValue(arrValues[valueIndex], transition);
|
|
|
valueIndex++;
|
|
|
}
|
|
|
}
|
|
|
};
|
|
|
p.updateValue = function () {
|
|
|
var newValue = [];
|
|
|
var newDisplayValue = [];
|
|
|
for (var i = 0; i < p.cols.length; i++) {
|
|
|
if (!p.cols[i].divider) {
|
|
|
newValue.push(p.cols[i].value);
|
|
|
newDisplayValue.push(p.cols[i].displayValue);
|
|
|
}
|
|
|
}
|
|
|
if (newValue.indexOf(undefined) >= 0) {
|
|
|
return;
|
|
|
}
|
|
|
p.value = newValue;
|
|
|
p.displayValue = newDisplayValue;
|
|
|
if (p.params.onChange) {
|
|
|
p.params.onChange(p, p.value, p.displayValue);
|
|
|
}
|
|
|
if (p.input && p.input.length > 0) {
|
|
|
$(p.input).val(p.params.formatValue ? p.params.formatValue(p, p.value, p.displayValue) : p.value.join(' '));
|
|
|
$(p.input).trigger('change');
|
|
|
}
|
|
|
};
|
|
|
|
|
|
// Columns Handlers
|
|
|
p.initPickerCol = function (colElement, updateItems) {
|
|
|
var colContainer = $(colElement);
|
|
|
var colIndex = colContainer.index();
|
|
|
var col = p.cols[colIndex];
|
|
|
if (col.divider) return;
|
|
|
col.container = colContainer;
|
|
|
col.wrapper = col.container.find('.picker-items-col-wrapper');
|
|
|
col.items = col.wrapper.find('.picker-item');
|
|
|
|
|
|
var i, j;
|
|
|
var wrapperHeight, itemHeight, itemsHeight, minTranslate, maxTranslate;
|
|
|
col.replaceValues = function (values, displayValues) {
|
|
|
col.destroyEvents();
|
|
|
col.values = values;
|
|
|
col.displayValues = displayValues;
|
|
|
var newItemsHTML = p.columnHTML(col, true);
|
|
|
col.wrapper.html(newItemsHTML);
|
|
|
col.items = col.wrapper.find('.picker-item');
|
|
|
col.calcSize();
|
|
|
col.setValue(col.values[0] || '', 0, true);
|
|
|
col.initEvents();
|
|
|
};
|
|
|
col.calcSize = function () {
|
|
|
if (!col.values.length) return;
|
|
|
if (p.params.rotateEffect) {
|
|
|
col.container.removeClass('picker-items-col-absolute');
|
|
|
if (!col.width) col.container.css({width:''});
|
|
|
}
|
|
|
var colWidth, colHeight;
|
|
|
colWidth = 0;
|
|
|
colHeight = col.container[0].offsetHeight;
|
|
|
wrapperHeight = col.wrapper[0].offsetHeight;
|
|
|
itemHeight = col.items[0].offsetHeight;
|
|
|
itemsHeight = itemHeight * col.items.length;
|
|
|
minTranslate = colHeight / 2 - itemsHeight + itemHeight / 2;
|
|
|
maxTranslate = colHeight / 2 - itemHeight / 2;
|
|
|
if (col.width) {
|
|
|
colWidth = col.width;
|
|
|
if (parseInt(colWidth, 10) === colWidth) colWidth = colWidth + 'px';
|
|
|
col.container.css({width: colWidth});
|
|
|
}
|
|
|
if (p.params.rotateEffect) {
|
|
|
if (!col.width) {
|
|
|
col.items.each(function () {
|
|
|
var item = $(this);
|
|
|
item.css({width:'auto'});
|
|
|
colWidth = Math.max(colWidth, item[0].offsetWidth);
|
|
|
item.css({width:''});
|
|
|
});
|
|
|
col.container.css({width: (colWidth + 2) + 'px'});
|
|
|
}
|
|
|
col.container.addClass('picker-items-col-absolute');
|
|
|
}
|
|
|
};
|
|
|
col.calcSize();
|
|
|
|
|
|
col.wrapper.transform('translate3d(0,' + maxTranslate + 'px,0)').transition(0);
|
|
|
|
|
|
|
|
|
var activeIndex = 0;
|
|
|
var animationFrameId;
|
|
|
|
|
|
// Set Value Function
|
|
|
col.setValue = function (newValue, transition, valueCallbacks) {
|
|
|
if (typeof transition === 'undefined') transition = '';
|
|
|
var newActiveIndex = col.wrapper.find('.picker-item[data-picker-value="' + newValue + '"]').index();
|
|
|
if(typeof newActiveIndex === 'undefined' || newActiveIndex === -1) {
|
|
|
col.value = col.displayValue = newValue;
|
|
|
return;
|
|
|
}
|
|
|
var newTranslate = -newActiveIndex * itemHeight + maxTranslate;
|
|
|
// Update wrapper
|
|
|
col.wrapper.transition(transition);
|
|
|
col.wrapper.transform('translate3d(0,' + (newTranslate) + 'px,0)');
|
|
|
|
|
|
// Watch items
|
|
|
if (p.params.updateValuesOnMomentum && col.activeIndex && col.activeIndex !== newActiveIndex ) {
|
|
|
$.cancelAnimationFrame(animationFrameId);
|
|
|
col.wrapper.transitionEnd(function(){
|
|
|
$.cancelAnimationFrame(animationFrameId);
|
|
|
});
|
|
|
updateDuringScroll();
|
|
|
}
|
|
|
|
|
|
// Update items
|
|
|
col.updateItems(newActiveIndex, newTranslate, transition, valueCallbacks);
|
|
|
};
|
|
|
|
|
|
col.updateItems = function (activeIndex, translate, transition, valueCallbacks) {
|
|
|
if (typeof translate === 'undefined') {
|
|
|
translate = $.getTranslate(col.wrapper[0], 'y');
|
|
|
}
|
|
|
if(typeof activeIndex === 'undefined') activeIndex = -Math.round((translate - maxTranslate)/itemHeight);
|
|
|
if (activeIndex < 0) activeIndex = 0;
|
|
|
if (activeIndex >= col.items.length) activeIndex = col.items.length - 1;
|
|
|
var previousActiveIndex = col.activeIndex;
|
|
|
col.activeIndex = activeIndex;
|
|
|
/*
|
|
|
col.wrapper.find('.picker-selected, .picker-after-selected, .picker-before-selected').removeClass('picker-selected picker-after-selected picker-before-selected');
|
|
|
|
|
|
col.items.transition(transition);
|
|
|
var selectedItem = col.items.eq(activeIndex).addClass('picker-selected').transform('');
|
|
|
var prevItems = selectedItem.prevAll().addClass('picker-before-selected');
|
|
|
var nextItems = selectedItem.nextAll().addClass('picker-after-selected');
|
|
|
*/
|
|
|
//去掉 .picker-after-selected, .picker-before-selected 以提高性能
|
|
|
col.wrapper.find('.picker-selected').removeClass('picker-selected');
|
|
|
if (p.params.rotateEffect) {
|
|
|
col.items.transition(transition);
|
|
|
}
|
|
|
var selectedItem = col.items.eq(activeIndex).addClass('picker-selected').transform('');
|
|
|
|
|
|
if (valueCallbacks || typeof valueCallbacks === 'undefined') {
|
|
|
// Update values
|
|
|
col.value = selectedItem.attr('data-picker-value');
|
|
|
col.displayValue = col.displayValues ? col.displayValues[activeIndex] : col.value;
|
|
|
// On change callback
|
|
|
if (previousActiveIndex !== activeIndex) {
|
|
|
if (col.onChange) {
|
|
|
col.onChange(p, col.value, col.displayValue);
|
|
|
}
|
|
|
p.updateValue();
|
|
|
}
|
|
|
}
|
|
|
|
|
|
// Set 3D rotate effect
|
|
|
if (!p.params.rotateEffect) {
|
|
|
return;
|
|
|
}
|
|
|
var percentage = (translate - (Math.floor((translate - maxTranslate)/itemHeight) * itemHeight + maxTranslate)) / itemHeight;
|
|
|
|
|
|
col.items.each(function () {
|
|
|
var item = $(this);
|
|
|
var itemOffsetTop = item.index() * itemHeight;
|
|
|
var translateOffset = maxTranslate - translate;
|
|
|
var itemOffset = itemOffsetTop - translateOffset;
|
|
|
var percentage = itemOffset / itemHeight;
|
|
|
|
|
|
var itemsFit = Math.ceil(col.height / itemHeight / 2) + 1;
|
|
|
|
|
|
var angle = (-18*percentage);
|
|
|
if (angle > 180) angle = 180;
|
|
|
if (angle < -180) angle = -180;
|
|
|
// Far class
|
|
|
if (Math.abs(percentage) > itemsFit) item.addClass('picker-item-far');
|
|
|
else item.removeClass('picker-item-far');
|
|
|
// Set transform
|
|
|
item.transform('translate3d(0, ' + (-translate + maxTranslate) + 'px, ' + (originBug ? -110 : 0) + 'px) rotateX(' + angle + 'deg)');
|
|
|
});
|
|
|
};
|
|
|
|
|
|
function updateDuringScroll() {
|
|
|
animationFrameId = $.requestAnimationFrame(function () {
|
|
|
col.updateItems(undefined, undefined, 0);
|
|
|
updateDuringScroll();
|
|
|
});
|
|
|
}
|
|
|
|
|
|
// Update items on init
|
|
|
if (updateItems) col.updateItems(0, maxTranslate, 0);
|
|
|
|
|
|
var allowItemClick = true;
|
|
|
var isTouched, isMoved, touchStartY, touchCurrentY, touchStartTime, touchEndTime, startTranslate, returnTo, currentTranslate, prevTranslate, velocityTranslate, velocityTime;
|
|
|
function handleTouchStart (e) {
|
|
|
if (isMoved || isTouched) return;
|
|
|
e.preventDefault();
|
|
|
isTouched = true;
|
|
|
var position = $.getTouchPosition(e);
|
|
|
touchStartY = touchCurrentY = position.y;
|
|
|
touchStartTime = (new Date()).getTime();
|
|
|
|
|
|
allowItemClick = true;
|
|
|
startTranslate = currentTranslate = $.getTranslate(col.wrapper[0], 'y');
|
|
|
}
|
|
|
function handleTouchMove (e) {
|
|
|
if (!isTouched) return;
|
|
|
e.preventDefault();
|
|
|
allowItemClick = false;
|
|
|
var position = $.getTouchPosition(e);
|
|
|
touchCurrentY = position.y;
|
|
|
if (!isMoved) {
|
|
|
// First move
|
|
|
$.cancelAnimationFrame(animationFrameId);
|
|
|
isMoved = true;
|
|
|
startTranslate = currentTranslate = $.getTranslate(col.wrapper[0], 'y');
|
|
|
col.wrapper.transition(0);
|
|
|
}
|
|
|
e.preventDefault();
|
|
|
|
|
|
var diff = touchCurrentY - touchStartY;
|
|
|
currentTranslate = startTranslate + diff;
|
|
|
returnTo = undefined;
|
|
|
|
|
|
// Normalize translate
|
|
|
if (currentTranslate < minTranslate) {
|
|
|
currentTranslate = minTranslate - Math.pow(minTranslate - currentTranslate, 0.8);
|
|
|
returnTo = 'min';
|
|
|
}
|
|
|
if (currentTranslate > maxTranslate) {
|
|
|
currentTranslate = maxTranslate + Math.pow(currentTranslate - maxTranslate, 0.8);
|
|
|
returnTo = 'max';
|
|
|
}
|
|
|
// Transform wrapper
|
|
|
col.wrapper.transform('translate3d(0,' + currentTranslate + 'px,0)');
|
|
|
|
|
|
// Update items
|
|
|
col.updateItems(undefined, currentTranslate, 0, p.params.updateValuesOnTouchmove);
|
|
|
|
|
|
// Calc velocity
|
|
|
velocityTranslate = currentTranslate - prevTranslate || currentTranslate;
|
|
|
velocityTime = (new Date()).getTime();
|
|
|
prevTranslate = currentTranslate;
|
|
|
}
|
|
|
function handleTouchEnd (e) {
|
|
|
if (!isTouched || !isMoved) {
|
|
|
isTouched = isMoved = false;
|
|
|
return;
|
|
|
}
|
|
|
isTouched = isMoved = false;
|
|
|
col.wrapper.transition('');
|
|
|
if (returnTo) {
|
|
|
if (returnTo === 'min') {
|
|
|
col.wrapper.transform('translate3d(0,' + minTranslate + 'px,0)');
|
|
|
}
|
|
|
else col.wrapper.transform('translate3d(0,' + maxTranslate + 'px,0)');
|
|
|
}
|
|
|
touchEndTime = new Date().getTime();
|
|
|
var velocity, newTranslate;
|
|
|
if (touchEndTime - touchStartTime > 300) {
|
|
|
newTranslate = currentTranslate;
|
|
|
}
|
|
|
else {
|
|
|
velocity = Math.abs(velocityTranslate / (touchEndTime - velocityTime));
|
|
|
newTranslate = currentTranslate + velocityTranslate * p.params.momentumRatio;
|
|
|
}
|
|
|
|
|
|
newTranslate = Math.max(Math.min(newTranslate, maxTranslate), minTranslate);
|
|
|
|
|
|
// Active Index
|
|
|
var activeIndex = -Math.floor((newTranslate - maxTranslate)/itemHeight);
|
|
|
|
|
|
// Normalize translate
|
|
|
if (!p.params.freeMode) newTranslate = -activeIndex * itemHeight + maxTranslate;
|
|
|
|
|
|
// Transform wrapper
|
|
|
col.wrapper.transform('translate3d(0,' + (parseInt(newTranslate,10)) + 'px,0)');
|
|
|
|
|
|
// Update items
|
|
|
col.updateItems(activeIndex, newTranslate, '', true);
|
|
|
|
|
|
// Watch items
|
|
|
if (p.params.updateValuesOnMomentum) {
|
|
|
updateDuringScroll();
|
|
|
col.wrapper.transitionEnd(function(){
|
|
|
$.cancelAnimationFrame(animationFrameId);
|
|
|
});
|
|
|
}
|
|
|
|
|
|
// Allow click
|
|
|
setTimeout(function () {
|
|
|
allowItemClick = true;
|
|
|
}, 100);
|
|
|
}
|
|
|
|
|
|
function handleClick(e) {
|
|
|
if (!allowItemClick) return;
|
|
|
$.cancelAnimationFrame(animationFrameId);
|
|
|
/*jshint validthis:true */
|
|
|
var value = $(this).attr('data-picker-value');
|
|
|
col.setValue(value);
|
|
|
}
|
|
|
|
|
|
col.initEvents = function (detach) {
|
|
|
var method = detach ? 'off' : 'on';
|
|
|
col.container[method]($.touchEvents.start, handleTouchStart);
|
|
|
col.container[method]($.touchEvents.move, handleTouchMove);
|
|
|
col.container[method]($.touchEvents.end, handleTouchEnd);
|
|
|
col.items[method]('click', handleClick);
|
|
|
};
|
|
|
col.destroyEvents = function () {
|
|
|
col.initEvents(true);
|
|
|
};
|
|
|
|
|
|
col.container[0].f7DestroyPickerCol = function () {
|
|
|
col.destroyEvents();
|
|
|
};
|
|
|
|
|
|
col.initEvents();
|
|
|
|
|
|
};
|
|
|
p.destroyPickerCol = function (colContainer) {
|
|
|
colContainer = $(colContainer);
|
|
|
if ('f7DestroyPickerCol' in colContainer[0]) colContainer[0].f7DestroyPickerCol();
|
|
|
};
|
|
|
// Resize cols
|
|
|
function resizeCols() {
|
|
|
if (!p.opened) return;
|
|
|
for (var i = 0; i < p.cols.length; i++) {
|
|
|
if (!p.cols[i].divider) {
|
|
|
p.cols[i].calcSize();
|
|
|
p.cols[i].setValue(p.cols[i].value, 0, false);
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
$(window).on('resize', resizeCols);
|
|
|
|
|
|
// HTML Layout
|
|
|
p.columnHTML = function (col, onlyItems) {
|
|
|
var columnItemsHTML = '';
|
|
|
var columnHTML = '';
|
|
|
if (col.divider) {
|
|
|
columnHTML += '<div class="picker-items-col picker-items-col-divider ' + (col.textAlign ? 'picker-items-col-' + col.textAlign : '') + ' ' + (col.cssClass || '') + '">' + col.content + '</div>';
|
|
|
}
|
|
|
else {
|
|
|
for (var j = 0; j < col.values.length; j++) {
|
|
|
columnItemsHTML += '<div class="picker-item" data-picker-value="' + col.values[j] + '">' + (col.displayValues ? col.displayValues[j] : col.values[j]) + '</div>';
|
|
|
}
|
|
|
columnHTML += '<div class="picker-items-col ' + (col.textAlign ? 'picker-items-col-' + col.textAlign : '') + ' ' + (col.cssClass || '') + '"><div class="picker-items-col-wrapper">' + columnItemsHTML + '</div></div>';
|
|
|
}
|
|
|
return onlyItems ? columnItemsHTML : columnHTML;
|
|
|
};
|
|
|
p.layout = function () {
|
|
|
var pickerHTML = '';
|
|
|
var pickerClass = '';
|
|
|
var i;
|
|
|
p.cols = [];
|
|
|
var colsHTML = '';
|
|
|
for (i = 0; i < p.params.cols.length; i++) {
|
|
|
var col = p.params.cols[i];
|
|
|
colsHTML += p.columnHTML(p.params.cols[i]);
|
|
|
p.cols.push(col);
|
|
|
}
|
|
|
pickerClass = 'weui-picker-modal picker-columns ' + (p.params.cssClass || '') + (p.params.rotateEffect ? ' picker-3d' : '') + (p.params.cols.length === 1 ? ' picker-columns-single' : '');
|
|
|
pickerHTML =
|
|
|
'<div class="' + (pickerClass) + '">' +
|
|
|
(p.params.toolbar ? p.params.toolbarTemplate.replace(/{{closeText}}/g, p.params.toolbarCloseText).replace(/{{title}}/g, p.params.title) : '') +
|
|
|
'<div class="picker-modal-inner picker-items">' +
|
|
|
colsHTML +
|
|
|
'<div class="picker-center-highlight"></div>' +
|
|
|
'</div>' +
|
|
|
'</div>';
|
|
|
|
|
|
p.pickerHTML = pickerHTML;
|
|
|
};
|
|
|
|
|
|
// Input Events
|
|
|
function openOnInput(e) {
|
|
|
e.preventDefault();
|
|
|
if (p.opened) return;
|
|
|
p.open();
|
|
|
if (p.params.scrollToInput && !isPopover()) {
|
|
|
var pageContent = p.input.parents('.content');
|
|
|
if (pageContent.length === 0) return;
|
|
|
|
|
|
var paddingTop = parseInt(pageContent.css('padding-top'), 10),
|
|
|
paddingBottom = parseInt(pageContent.css('padding-bottom'), 10),
|
|
|
pageHeight = pageContent[0].offsetHeight - paddingTop - p.container.height(),
|
|
|
pageScrollHeight = pageContent[0].scrollHeight - paddingTop - p.container.height(),
|
|
|
newPaddingBottom;
|
|
|
var inputTop = p.input.offset().top - paddingTop + p.input[0].offsetHeight;
|
|
|
if (inputTop > pageHeight) {
|
|
|
var scrollTop = pageContent.scrollTop() + inputTop - pageHeight;
|
|
|
if (scrollTop + pageHeight > pageScrollHeight) {
|
|
|
newPaddingBottom = scrollTop + pageHeight - pageScrollHeight + paddingBottom;
|
|
|
if (pageHeight === pageScrollHeight) {
|
|
|
newPaddingBottom = p.container.height();
|
|
|
}
|
|
|
pageContent.css({'padding-bottom': (newPaddingBottom) + 'px'});
|
|
|
}
|
|
|
pageContent.scrollTop(scrollTop, 300);
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
function closeOnHTMLClick(e) {
|
|
|
if (inPopover()) return;
|
|
|
if (p.input && p.input.length > 0) {
|
|
|
if (e.target !== p.input[0] && $(e.target).parents('.weui-picker-modal').length === 0) p.close();
|
|
|
}
|
|
|
else {
|
|
|
if ($(e.target).parents('.weui-picker-modal').length === 0) p.close();
|
|
|
}
|
|
|
}
|
|
|
|
|
|
if (p.params.input) {
|
|
|
p.input = $(p.params.input);
|
|
|
if (p.input.length > 0) {
|
|
|
if (p.params.inputReadOnly) p.input.prop('readOnly', true);
|
|
|
if (!p.inline) {
|
|
|
p.input.on('click', openOnInput);
|
|
|
}
|
|
|
if (p.params.inputReadOnly) {
|
|
|
p.input.on('focus mousedown', function (e) {
|
|
|
e.preventDefault();
|
|
|
});
|
|
|
}
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
if (!p.inline) $('html').on('click', closeOnHTMLClick);
|
|
|
|
|
|
// Open
|
|
|
function onPickerClose() {
|
|
|
p.opened = false;
|
|
|
if (p.input && p.input.length > 0) p.input.parents('.page-content').css({'padding-bottom': ''});
|
|
|
if (p.params.onClose) p.params.onClose(p);
|
|
|
|
|
|
// Destroy events
|
|
|
p.container.find('.picker-items-col').each(function () {
|
|
|
p.destroyPickerCol(this);
|
|
|
});
|
|
|
}
|
|
|
|
|
|
p.opened = false;
|
|
|
p.open = function () {
|
|
|
var toPopover = isPopover();
|
|
|
|
|
|
if (!p.opened) {
|
|
|
|
|
|
// Layout
|
|
|
p.layout();
|
|
|
|
|
|
// Append
|
|
|
if (toPopover) {
|
|
|
p.pickerHTML = '<div class="popover popover-picker-columns"><div class="popover-inner">' + p.pickerHTML + '</div></div>';
|
|
|
p.popover = $.popover(p.pickerHTML, p.params.input, true);
|
|
|
p.container = $(p.popover).find('.weui-picker-modal');
|
|
|
$(p.popover).on('close', function () {
|
|
|
onPickerClose();
|
|
|
});
|
|
|
}
|
|
|
else if (p.inline) {
|
|
|
p.container = $(p.pickerHTML);
|
|
|
p.container.addClass('picker-modal-inline');
|
|
|
$(p.params.container).append(p.container);
|
|
|
}
|
|
|
else {
|
|
|
p.container = $($.openPicker(p.pickerHTML));
|
|
|
$(p.container)
|
|
|
.on('close', function () {
|
|
|
onPickerClose();
|
|
|
});
|
|
|
}
|
|
|
|
|
|
// Store picker instance
|
|
|
p.container[0].f7Picker = p;
|
|
|
|
|
|
// Init Events
|
|
|
p.container.find('.picker-items-col').each(function () {
|
|
|
var updateItems = true;
|
|
|
if ((!p.initialized && p.params.value) || (p.initialized && p.value)) updateItems = false;
|
|
|
p.initPickerCol(this, updateItems);
|
|
|
});
|
|
|
|
|
|
// Set value
|
|
|
if (!p.initialized) {
|
|
|
if (p.params.value) {
|
|
|
p.setValue(p.params.value, 0);
|
|
|
}
|
|
|
}
|
|
|
else {
|
|
|
if (p.value) p.setValue(p.value, 0);
|
|
|
}
|
|
|
}
|
|
|
|
|
|
// Set flag
|
|
|
p.opened = true;
|
|
|
p.initialized = true;
|
|
|
|
|
|
if (p.params.onOpen) p.params.onOpen(p);
|
|
|
};
|
|
|
|
|
|
// Close
|
|
|
p.close = function (force) {
|
|
|
if (!p.opened || p.inline) return;
|
|
|
if (inPopover()) {
|
|
|
$.closePicker(p.popover);
|
|
|
return;
|
|
|
}
|
|
|
else {
|
|
|
$.closePicker(p.container);
|
|
|
return;
|
|
|
}
|
|
|
};
|
|
|
|
|
|
// Destroy
|
|
|
p.destroy = function () {
|
|
|
p.close();
|
|
|
if (p.params.input && p.input.length > 0) {
|
|
|
p.input.off('click focus', openOnInput);
|
|
|
$(p.input).data('picker', null);
|
|
|
}
|
|
|
$('html').off('click', closeOnHTMLClick);
|
|
|
$(window).off('resize', resizeCols);
|
|
|
};
|
|
|
|
|
|
if (p.inline) {
|
|
|
p.open();
|
|
|
}
|
|
|
|
|
|
return p;
|
|
|
};
|
|
|
|
|
|
$(document).on("click", ".close-picker", function() {
|
|
|
var pickerToClose = $('.weui-picker-modal.weui-picker-modal-visible');
|
|
|
if (pickerToClose.length > 0) {
|
|
|
$.closePicker(pickerToClose);
|
|
|
}
|
|
|
});
|
|
|
|
|
|
//修复picker会滚动页面的bug
|
|
|
$(document).on($.touchEvents.move, ".picker-modal-inner", function(e) {
|
|
|
e.preventDefault();
|
|
|
});
|
|
|
|
|
|
|
|
|
$.openPicker = function(tpl, className, callback) {
|
|
|
|
|
|
if(typeof className === "function") {
|
|
|
callback = className;
|
|
|
className = undefined;
|
|
|
}
|
|
|
|
|
|
$.closePicker();
|
|
|
|
|
|
var container = $("<div class='weui-picker-container "+ (className || "") + "'></div>").appendTo(document.body);
|
|
|
container.show();
|
|
|
|
|
|
container.addClass("weui-picker-container-visible");
|
|
|
|
|
|
//关于布局的问题,如果直接放在body上,则做动画的时候会撑开body高度而导致滚动条变化。
|
|
|
var dialog = $(tpl).appendTo(container);
|
|
|
|
|
|
dialog.width(); //通过取一次CSS值,强制浏览器不能把上下两行代码合并执行,因为合并之后会导致无法出现动画。
|
|
|
|
|
|
dialog.addClass("weui-picker-modal-visible");
|
|
|
|
|
|
callback && container.on("close", callback);
|
|
|
|
|
|
return dialog;
|
|
|
}
|
|
|
|
|
|
$.updatePicker = function(tpl) {
|
|
|
var container = $(".weui-picker-container-visible");
|
|
|
if(!container[0]) return false;
|
|
|
|
|
|
container.html("");
|
|
|
|
|
|
var dialog = $(tpl).appendTo(container);
|
|
|
|
|
|
dialog.addClass("weui-picker-modal-visible");
|
|
|
|
|
|
return dialog;
|
|
|
}
|
|
|
|
|
|
$.closePicker = function(container, callback) {
|
|
|
if(typeof container === "function") callback = container;
|
|
|
$(".weui-picker-modal-visible").removeClass("weui-picker-modal-visible").transitionEnd(function() {
|
|
|
$(this).parent().remove();
|
|
|
callback && callback();
|
|
|
}).trigger("close");
|
|
|
};
|
|
|
|
|
|
$.fn.picker = function(params) {
|
|
|
var args = arguments;
|
|
|
return this.each(function() {
|
|
|
if(!this) return;
|
|
|
var $this = $(this);
|
|
|
|
|
|
var picker = $this.data("picker");
|
|
|
if(!picker) {
|
|
|
params = $.extend({ input: this }, params || {}) // https://github.com/lihongxun945/jquery-weui/issues/432
|
|
|
var inputValue = $this.val();
|
|
|
if(params.value === undefined && inputValue !== "") {
|
|
|
params.value = (params.cols && params.cols.length > 1) ? inputValue.split(" ") : [inputValue];
|
|
|
}
|
|
|
var p = $.extend({input: this}, params);
|
|
|
picker = new Picker(p);
|
|
|
$this.data("picker", picker);
|
|
|
}
|
|
|
if(typeof params === typeof "a") {
|
|
|
picker[params].apply(picker, Array.prototype.slice.call(args, 1));
|
|
|
}
|
|
|
});
|
|
|
};
|
|
|
}($);
|
|
|
|
|
|
/* global $:true */
|
|
|
+ function($) {
|
|
|
"use strict";
|
|
|
|
|
|
var defaults;
|
|
|
|
|
|
var selects = [];
|
|
|
|
|
|
var Select = function(input, config) {
|
|
|
|
|
|
var self = this;
|
|
|
this.config = config;
|
|
|
|
|
|
//init empty data
|
|
|
this.data = {
|
|
|
values: '',
|
|
|
titles: '',
|
|
|
origins: [],
|
|
|
length: 0
|
|
|
};
|
|
|
|
|
|
this.$input = $(input);
|
|
|
this.$input.prop("readOnly", true);
|
|
|
|
|
|
this.initConfig();
|
|
|
|
|
|
config = this.config;
|
|
|
|
|
|
this.$input.click($.proxy(this.open, this));
|
|
|
selects.push(this)
|
|
|
}
|
|
|
|
|
|
Select.prototype.initConfig = function() {
|
|
|
this.config = $.extend({}, defaults, this.config);
|
|
|
|
|
|
var config = this.config;
|
|
|
|
|
|
if(!config.items || !config.items.length) return;
|
|
|
|
|
|
config.items = config.items.map(function(d, i) {
|
|
|
if(typeof d == typeof "a") {
|
|
|
return {
|
|
|
title: d,
|
|
|
value: d
|
|
|
};
|
|
|
}
|
|
|
|
|
|
return d;
|
|
|
});
|
|
|
|
|
|
|
|
|
this.tpl = $.t7.compile("<div class='weui-picker-modal weui-select-modal'>" + config.toolbarTemplate + (config.multi ? config.checkboxTemplate : config.radioTemplate) + "</div>");
|
|
|
|
|
|
if(config.input !== undefined) this.$input.val(config.input);
|
|
|
|
|
|
this.parseInitValue();
|
|
|
|
|
|
this._init = true;
|
|
|
}
|
|
|
|
|
|
Select.prototype.updateInputValue = function(values, titles) {
|
|
|
var v, t;
|
|
|
if(this.config.multi) {
|
|
|
v = values.join(this.config.split);
|
|
|
t = titles.join(this.config.split);
|
|
|
} else {
|
|
|
v = values[0];
|
|
|
t = titles[0];
|
|
|
}
|
|
|
|
|
|
//caculate origin data
|
|
|
var origins = [];
|
|
|
|
|
|
this.config.items.forEach(function(d) {
|
|
|
values.each(function(i, dd) {
|
|
|
if(d.value == dd) origins.push(d);
|
|
|
});
|
|
|
});
|
|
|
|
|
|
this.$input.val(t).data("values", v);
|
|
|
this.$input.attr("value", t).attr("data-values", v);
|
|
|
|
|
|
var data = {
|
|
|
values: v,
|
|
|
titles: t,
|
|
|
valuesArray: values,
|
|
|
titlesArray: titles,
|
|
|
origins: origins,
|
|
|
length: origins.length
|
|
|
};
|
|
|
this.data = data;
|
|
|
this.$input.trigger("change", data);
|
|
|
this.config.onChange && this.config.onChange.call(this, data);
|
|
|
}
|
|
|
|
|
|
Select.prototype.parseInitValue = function() {
|
|
|
var value = this.$input.val();
|
|
|
var items = this.config.items;
|
|
|
|
|
|
//如果input为空,只有在第一次初始化的时候才保留默认选择。因为后来就是用户自己取消了全部选择,不能再为他选中默认值。
|
|
|
if( !this._init && (value === undefined || value == null || value === "")) return;
|
|
|
|
|
|
var titles = this.config.multi ? value.split(this.config.split) : [value];
|
|
|
for(var i=0;i<items.length;i++) {
|
|
|
items[i].checked = false;
|
|
|
for(var j=0;j<titles.length;j++) {
|
|
|
if(items[i].title === titles[j]) {
|
|
|
items[i].checked = true;
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
|
|
|
Select.prototype._bind = function(dialog) {
|
|
|
var self = this,
|
|
|
config = this.config;
|
|
|
dialog.on("change", function(e) {
|
|
|
var checked = dialog.find("input:checked");
|
|
|
var values = checked.map(function() {
|
|
|
return $(this).val();
|
|
|
});
|
|
|
var titles = checked.map(function() {
|
|
|
return $(this).data("title");
|
|
|
});
|
|
|
self.updateInputValue(values, titles);
|
|
|
|
|
|
if(config.autoClose && !config.multi) self.close();
|
|
|
})
|
|
|
.trigger('change')
|
|
|
.on("click", ".close-select", function() {
|
|
|
self.close();
|
|
|
});
|
|
|
}
|
|
|
|
|
|
//更新数据
|
|
|
Select.prototype.update = function(config) {
|
|
|
this.config = $.extend({}, this.config, config);
|
|
|
this.initConfig();
|
|
|
if(this._open) {
|
|
|
this._bind($.updatePicker(this.getHTML()));
|
|
|
}
|
|
|
}
|
|
|
|
|
|
Select.prototype.open = function(values, titles) {
|
|
|
|
|
|
if(this._open) return;
|
|
|
|
|
|
// open picker 会默认关掉其他的,但是 onClose 不会被调用,所以这里先关掉其他select
|
|
|
for (var i = 0; i < selects.length; i++ ) {
|
|
|
var s = selects[i];
|
|
|
if (s === this) continue;
|
|
|
if (s._open) {
|
|
|
if(!s.close()) return false; // 其他的select由于某些条件限制关闭失败。
|
|
|
}
|
|
|
}
|
|
|
|
|
|
this.parseInitValue();
|
|
|
|
|
|
var config = this.config;
|
|
|
|
|
|
var dialog = this.dialog = $.openPicker(this.getHTML());
|
|
|
|
|
|
this._bind(dialog);
|
|
|
|
|
|
this._open = true;
|
|
|
if(config.onOpen) config.onOpen(this);
|
|
|
}
|
|
|
|
|
|
Select.prototype.close = function(callback, force) {
|
|
|
if (!this._open) return false;
|
|
|
var self = this,
|
|
|
beforeClose = this.config.beforeClose;
|
|
|
|
|
|
if(typeof callback === typeof true) {
|
|
|
force === callback;
|
|
|
}
|
|
|
if(!force) {
|
|
|
if(beforeClose && typeof beforeClose === 'function' && beforeClose.call(this, this.data.values, this.data.titles) === false) {
|
|
|
return false
|
|
|
}
|
|
|
if(this.config.multi) {
|
|
|
if(this.config.min !== undefined && this.data.length < this.config.min) {
|
|
|
$.toast("请至少选择"+this.config.min+"个", "text");
|
|
|
return false
|
|
|
}
|
|
|
if(this.config.max !== undefined && this.data.length > this.config.max) {
|
|
|
$.toast("最多只能选择"+this.config.max+"个", "text");
|
|
|
return false
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
$.closePicker(function() {
|
|
|
self.onClose();
|
|
|
callback && callback();
|
|
|
});
|
|
|
|
|
|
return true
|
|
|
}
|
|
|
|
|
|
Select.prototype.onClose = function() {
|
|
|
this._open = false;
|
|
|
if(this.config.onClose) this.config.onClose(this);
|
|
|
}
|
|
|
|
|
|
Select.prototype.getHTML = function(callback) {
|
|
|
var config = this.config;
|
|
|
return this.tpl({
|
|
|
items: config.items,
|
|
|
title: config.title,
|
|
|
closeText: config.closeText
|
|
|
})
|
|
|
}
|
|
|
|
|
|
|
|
|
$.fn.select = function(params, args) {
|
|
|
|
|
|
return this.each(function() {
|
|
|
var $this = $(this);
|
|
|
if(!$this.data("weui-select")) $this.data("weui-select", new Select(this, params));
|
|
|
|
|
|
var select = $this.data("weui-select");
|
|
|
|
|
|
if(typeof params === typeof "a") select[params].call(select, args);
|
|
|
|
|
|
return select;
|
|
|
});
|
|
|
}
|
|
|
|
|
|
defaults = $.fn.select.prototype.defaults = {
|
|
|
items: [],
|
|
|
input: undefined, //输入框的初始值
|
|
|
title: "请选择",
|
|
|
multi: false,
|
|
|
closeText: "确定",
|
|
|
autoClose: true, //是否选择完成后自动关闭,只有单选模式下才有效
|
|
|
onChange: undefined, //function
|
|
|
beforeClose: undefined, // function 关闭之前,如果返回false则阻止关闭
|
|
|
onClose: undefined, //function
|
|
|
onOpen: undefined, //function
|
|
|
split: ",", //多选模式下的分隔符
|
|
|
min: undefined, //多选模式下可用,最少选择数
|
|
|
max: undefined, //单选模式下可用,最多选择数
|
|
|
toolbarTemplate: '<div class="toolbar">\
|
|
|
<div class="toolbar-inner">\
|
|
|
<a href="javascript:;" class="picker-button close-select">{{closeText}}</a>\
|
|
|
<h1 class="title">{{title}}</h1>\
|
|
|
</div>\
|
|
|
</div>',
|
|
|
radioTemplate:
|
|
|
'<div class="weui-cells weui-cells_radio">\
|
|
|
{{#items}}\
|
|
|
<label class="weui-cell weui-check_label" for="weui-select-id-{{this.title}}">\
|
|
|
<div class="weui-cell__bd weui-cell_primary">\
|
|
|
<p>{{this.title}}</p>\
|
|
|
</div>\
|
|
|
<div class="weui-cell__ft">\
|
|
|
<input type="radio" class="weui-check" name="weui-select" id="weui-select-id-{{this.title}}" value="{{this.value}}" {{#if this.checked}}checked="checked"{{/if}} data-title="{{this.title}}">\
|
|
|
<span class="weui-icon-checked"></span>\
|
|
|
</div>\
|
|
|
</label>\
|
|
|
{{/items}}\
|
|
|
</div>',
|
|
|
checkboxTemplate:
|
|
|
'<div class="weui-cells weui-cells_checkbox">\
|
|
|
{{#items}}\
|
|
|
<label class="weui-cell weui-check_label" for="weui-select-id-{{this.title}}">\
|
|
|
<div class="weui-cell__bd weui-cell_primary">\
|
|
|
<p>{{this.title}}</p>\
|
|
|
</div>\
|
|
|
<div class="weui-cell__ft">\
|
|
|
<input type="checkbox" class="weui-check" name="weui-select" id="weui-select-id-{{this.title}}" value="{{this.value}}" {{#if this.checked}}checked="checked"{{/if}} data-title="{{this.title}}" >\
|
|
|
<span class="weui-icon-checked"></span>\
|
|
|
</div>\
|
|
|
</label>\
|
|
|
{{/items}}\
|
|
|
</div>',
|
|
|
}
|
|
|
|
|
|
}($);
|
|
|
|
|
|
/*======================================================
|
|
|
************ Calendar ************
|
|
|
======================================================*/
|
|
|
/* global $:true */
|
|
|
/*jshint unused: false*/
|
|
|
+function ($) {
|
|
|
"use strict";
|
|
|
var rtl = false;
|
|
|
var defaults;
|
|
|
var isSameDate = function (a, b) {
|
|
|
var a = new Date(a),
|
|
|
b = new Date(b);
|
|
|
return a.getFullYear() === b.getFullYear() && a.getMonth() === b.getMonth() && a.getDate() === b.getDate()
|
|
|
}
|
|
|
var Calendar = function (params) {
|
|
|
var p = this;
|
|
|
params = params || {};
|
|
|
for (var def in defaults) {
|
|
|
if (typeof params[def] === 'undefined') {
|
|
|
params[def] = defaults[def];
|
|
|
}
|
|
|
}
|
|
|
p.params = params;
|
|
|
p.initialized = false;
|
|
|
|
|
|
// Inline flag
|
|
|
p.inline = p.params.container ? true : false;
|
|
|
|
|
|
// Is horizontal
|
|
|
p.isH = p.params.direction === 'horizontal';
|
|
|
|
|
|
// RTL inverter
|
|
|
var inverter = p.isH ? (rtl ? -1 : 1) : 1;
|
|
|
|
|
|
// Animating flag
|
|
|
p.animating = false;
|
|
|
|
|
|
// Should be converted to popover
|
|
|
function isPopover() {
|
|
|
var toPopover = false;
|
|
|
if (!p.params.convertToPopover && !p.params.onlyInPopover) return toPopover;
|
|
|
if (!p.inline && p.params.input) {
|
|
|
if (p.params.onlyInPopover) toPopover = true;
|
|
|
else {
|
|
|
if ($.device.ios) {
|
|
|
toPopover = $.device.ipad ? true : false;
|
|
|
}
|
|
|
else {
|
|
|
if ($(window).width() >= 768) toPopover = true;
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
return toPopover;
|
|
|
}
|
|
|
function inPopover() {
|
|
|
if (p.opened && p.container && p.container.length > 0 && p.container.parents('.popover').length > 0) return true;
|
|
|
else return false;
|
|
|
}
|
|
|
|
|
|
// Format date
|
|
|
function formatDate(date) {
|
|
|
date = new Date(date);
|
|
|
var year = date.getFullYear();
|
|
|
var month = date.getMonth();
|
|
|
var month1 = month + 1;
|
|
|
var day = date.getDate();
|
|
|
var weekDay = date.getDay();
|
|
|
return p.params.dateFormat
|
|
|
.replace(/yyyy/g, year)
|
|
|
.replace(/yy/g, (year + '').substring(2))
|
|
|
.replace(/mm/g, month1 < 10 ? '0' + month1 : month1)
|
|
|
.replace(/m/g, month1)
|
|
|
.replace(/MM/g, p.params.monthNames[month])
|
|
|
.replace(/M/g, p.params.monthNamesShort[month])
|
|
|
.replace(/dd/g, day < 10 ? '0' + day : day)
|
|
|
.replace(/d/g, day)
|
|
|
.replace(/DD/g, p.params.dayNames[weekDay])
|
|
|
.replace(/D/g, p.params.dayNamesShort[weekDay]);
|
|
|
}
|
|
|
|
|
|
|
|
|
// Value
|
|
|
p.addValue = function (value) {
|
|
|
if (p.params.multiple) {
|
|
|
if (!p.value) p.value = [];
|
|
|
var inValuesIndex;
|
|
|
for (var i = 0; i < p.value.length; i++) {
|
|
|
if (isSameDate(value, p.value[i])) {
|
|
|
inValuesIndex = i;
|
|
|
}
|
|
|
}
|
|
|
if (typeof inValuesIndex === 'undefined') {
|
|
|
p.value.push(value);
|
|
|
}
|
|
|
else {
|
|
|
p.value.splice(inValuesIndex, 1);
|
|
|
}
|
|
|
p.updateValue();
|
|
|
}
|
|
|
else {
|
|
|
p.value = [value];
|
|
|
p.updateValue();
|
|
|
}
|
|
|
};
|
|
|
p.setValue = function (arrValues) {
|
|
|
var date = new Date(arrValues[0]);
|
|
|
p.setYearMonth(date.getFullYear(), date.getMonth());
|
|
|
p.addValue(+ date);
|
|
|
};
|
|
|
p.updateValue = function () {
|
|
|
p.wrapper.find('.picker-calendar-day-selected').removeClass('picker-calendar-day-selected');
|
|
|
var i, inputValue;
|
|
|
for (i = 0; i < p.value.length; i++) {
|
|
|
var valueDate = new Date(p.value[i]);
|
|
|
p.wrapper.find('.picker-calendar-day[data-date="' + valueDate.getFullYear() + '-' + valueDate.getMonth() + '-' + valueDate.getDate() + '"]').addClass('picker-calendar-day-selected');
|
|
|
}
|
|
|
if (p.params.onChange) {
|
|
|
p.params.onChange(p, p.value.map(formatDate), p.value.map(function (d) {
|
|
|
return + new Date(typeof d === typeof 'a' ? d.split(/\D/).filter(function (a) { return !!a; }).join("-") : d);
|
|
|
}));
|
|
|
}
|
|
|
if (p.input && p.input.length > 0) {
|
|
|
if (p.params.formatValue) inputValue = p.params.formatValue(p, p.value);
|
|
|
else {
|
|
|
inputValue = [];
|
|
|
for (i = 0; i < p.value.length; i++) {
|
|
|
inputValue.push(formatDate(p.value[i]));
|
|
|
}
|
|
|
inputValue = inputValue.join(', ');
|
|
|
}
|
|
|
$(p.input).val(inputValue);
|
|
|
$(p.input).trigger('change');
|
|
|
}
|
|
|
};
|
|
|
|
|
|
// Columns Handlers
|
|
|
p.initCalendarEvents = function () {
|
|
|
var col;
|
|
|
var allowItemClick = true;
|
|
|
var isTouched, isMoved, touchStartX, touchStartY, touchCurrentX, touchCurrentY, touchStartTime, touchEndTime, startTranslate, currentTranslate, wrapperWidth, wrapperHeight, percentage, touchesDiff, isScrolling;
|
|
|
function handleTouchStart (e) {
|
|
|
if (isMoved || isTouched) return;
|
|
|
// e.preventDefault();
|
|
|
isTouched = true;
|
|
|
var position = $.getTouchPosition(e);
|
|
|
touchStartX = touchCurrentY = position.x;
|
|
|
touchStartY = touchCurrentY = position.y;
|
|
|
touchStartTime = (new Date()).getTime();
|
|
|
percentage = 0;
|
|
|
allowItemClick = true;
|
|
|
isScrolling = undefined;
|
|
|
startTranslate = currentTranslate = p.monthsTranslate;
|
|
|
}
|
|
|
function handleTouchMove (e) {
|
|
|
if (!isTouched) return;
|
|
|
var position = $.getTouchPosition(e);
|
|
|
touchCurrentX = position.x;
|
|
|
touchCurrentY = position.y;
|
|
|
if (typeof isScrolling === 'undefined') {
|
|
|
isScrolling = !!(isScrolling || Math.abs(touchCurrentY - touchStartY) > Math.abs(touchCurrentX - touchStartX));
|
|
|
}
|
|
|
if (p.isH && isScrolling) {
|
|
|
isTouched = false;
|
|
|
return;
|
|
|
}
|
|
|
e.preventDefault();
|
|
|
if (p.animating) {
|
|
|
isTouched = false;
|
|
|
return;
|
|
|
}
|
|
|
allowItemClick = false;
|
|
|
if (!isMoved) {
|
|
|
// First move
|
|
|
isMoved = true;
|
|
|
wrapperWidth = p.wrapper[0].offsetWidth;
|
|
|
wrapperHeight = p.wrapper[0].offsetHeight;
|
|
|
p.wrapper.transition(0);
|
|
|
}
|
|
|
e.preventDefault();
|
|
|
|
|
|
touchesDiff = p.isH ? touchCurrentX - touchStartX : touchCurrentY - touchStartY;
|
|
|
percentage = touchesDiff/(p.isH ? wrapperWidth : wrapperHeight);
|
|
|
currentTranslate = (p.monthsTranslate * inverter + percentage) * 100;
|
|
|
|
|
|
// Transform wrapper
|
|
|
p.wrapper.transform('translate3d(' + (p.isH ? currentTranslate : 0) + '%, ' + (p.isH ? 0 : currentTranslate) + '%, 0)');
|
|
|
|
|
|
}
|
|
|
function handleTouchEnd (e) {
|
|
|
if (!isTouched || !isMoved) {
|
|
|
isTouched = isMoved = false;
|
|
|
return;
|
|
|
}
|
|
|
isTouched = isMoved = false;
|
|
|
|
|
|
touchEndTime = new Date().getTime();
|
|
|
if (touchEndTime - touchStartTime < 300) {
|
|
|
if (Math.abs(touchesDiff) < 10) {
|
|
|
p.resetMonth();
|
|
|
}
|
|
|
else if (touchesDiff >= 10) {
|
|
|
if (rtl) p.nextMonth();
|
|
|
else p.prevMonth();
|
|
|
}
|
|
|
else {
|
|
|
if (rtl) p.prevMonth();
|
|
|
else p.nextMonth();
|
|
|
}
|
|
|
}
|
|
|
else {
|
|
|
if (percentage <= -0.5) {
|
|
|
if (rtl) p.prevMonth();
|
|
|
else p.nextMonth();
|
|
|
}
|
|
|
else if (percentage >= 0.5) {
|
|
|
if (rtl) p.nextMonth();
|
|
|
else p.prevMonth();
|
|
|
}
|
|
|
else {
|
|
|
p.resetMonth();
|
|
|
}
|
|
|
}
|
|
|
|
|
|
// Allow click
|
|
|
setTimeout(function () {
|
|
|
allowItemClick = true;
|
|
|
}, 100);
|
|
|
}
|
|
|
|
|
|
function handleDayClick(e) {
|
|
|
if (!allowItemClick) return;
|
|
|
var day = $(e.target).parents('.picker-calendar-day');
|
|
|
if (day.length === 0 && $(e.target).hasClass('picker-calendar-day')) {
|
|
|
day = $(e.target);
|
|
|
}
|
|
|
if (day.length === 0) return;
|
|
|
// if (day.hasClass('picker-calendar-day-selected') && !p.params.multiple) return;
|
|
|
if (day.hasClass('picker-calendar-day-disabled')) return;
|
|
|
if (day.hasClass('picker-calendar-day-next')) p.nextMonth();
|
|
|
if (day.hasClass('picker-calendar-day-prev')) p.prevMonth();
|
|
|
var dateYear = day.attr('data-year');
|
|
|
var dateMonth = day.attr('data-month');
|
|
|
var dateDay = day.attr('data-day');
|
|
|
if (p.params.onDayClick) {
|
|
|
p.params.onDayClick(p, day[0], dateYear, dateMonth, dateDay);
|
|
|
}
|
|
|
p.addValue(new Date(dateYear, dateMonth, dateDay).getTime());
|
|
|
if (p.params.closeOnSelect && !p.params.multiple) p.close();
|
|
|
}
|
|
|
|
|
|
p.container.find('.picker-calendar-prev-month').on('click', p.prevMonth);
|
|
|
p.container.find('.picker-calendar-next-month').on('click', p.nextMonth);
|
|
|
p.container.find('.picker-calendar-prev-year').on('click', p.prevYear);
|
|
|
p.container.find('.picker-calendar-next-year').on('click', p.nextYear);
|
|
|
p.wrapper.on('click', handleDayClick);
|
|
|
if (p.params.touchMove) {
|
|
|
p.wrapper.on($.touchEvents.start, handleTouchStart);
|
|
|
p.wrapper.on($.touchEvents.move, handleTouchMove);
|
|
|
p.wrapper.on($.touchEvents.end, handleTouchEnd);
|
|
|
}
|
|
|
|
|
|
p.container[0].f7DestroyCalendarEvents = function () {
|
|
|
p.container.find('.picker-calendar-prev-month').off('click', p.prevMonth);
|
|
|
p.container.find('.picker-calendar-next-month').off('click', p.nextMonth);
|
|
|
p.container.find('.picker-calendar-prev-year').off('click', p.prevYear);
|
|
|
p.container.find('.picker-calendar-next-year').off('click', p.nextYear);
|
|
|
p.wrapper.off('click', handleDayClick);
|
|
|
if (p.params.touchMove) {
|
|
|
p.wrapper.off($.touchEvents.start, handleTouchStart);
|
|
|
p.wrapper.off($.touchEvents.move, handleTouchMove);
|
|
|
p.wrapper.off($.touchEvents.end, handleTouchEnd);
|
|
|
}
|
|
|
};
|
|
|
|
|
|
|
|
|
};
|
|
|
p.destroyCalendarEvents = function (colContainer) {
|
|
|
if ('f7DestroyCalendarEvents' in p.container[0]) p.container[0].f7DestroyCalendarEvents();
|
|
|
};
|
|
|
|
|
|
// Calendar Methods
|
|
|
p.daysInMonth = function (date) {
|
|
|
var d = new Date(date);
|
|
|
return new Date(d.getFullYear(), d.getMonth() + 1, 0).getDate();
|
|
|
};
|
|
|
p.monthHTML = function (date, offset) {
|
|
|
date = new Date(date);
|
|
|
var year = date.getFullYear(),
|
|
|
month = date.getMonth(),
|
|
|
day = date.getDate();
|
|
|
if (offset === 'next') {
|
|
|
if (month === 11) date = new Date(year + 1, 0);
|
|
|
else date = new Date(year, month + 1, 1);
|
|
|
}
|
|
|
if (offset === 'prev') {
|
|
|
if (month === 0) date = new Date(year - 1, 11);
|
|
|
else date = new Date(year, month - 1, 1);
|
|
|
}
|
|
|
if (offset === 'next' || offset === 'prev') {
|
|
|
month = date.getMonth();
|
|
|
year = date.getFullYear();
|
|
|
}
|
|
|
var daysInPrevMonth = p.daysInMonth(new Date(date.getFullYear(), date.getMonth()).getTime() - 10 * 24 * 60 * 60 * 1000),
|
|
|
daysInMonth = p.daysInMonth(date),
|
|
|
firstDayOfMonthIndex = new Date(date.getFullYear(), date.getMonth()).getDay();
|
|
|
if (firstDayOfMonthIndex === 0) firstDayOfMonthIndex = 7;
|
|
|
|
|
|
var dayDate, currentValues = [], i, j,
|
|
|
rows = 6, cols = 7,
|
|
|
monthHTML = '',
|
|
|
dayIndex = 0 + (p.params.firstDay - 1),
|
|
|
today = new Date().setHours(0,0,0,0),
|
|
|
minDate = p.params.minDate ? new Date(p.params.minDate).getTime() : null,
|
|
|
maxDate = p.params.maxDate ? new Date(p.params.maxDate).getTime() : null;
|
|
|
|
|
|
if (p.value && p.value.length) {
|
|
|
for (i = 0; i < p.value.length; i++) {
|
|
|
currentValues.push(new Date(p.value[i]).setHours(0,0,0,0));
|
|
|
}
|
|
|
}
|
|
|
|
|
|
for (i = 1; i <= rows; i++) {
|
|
|
var rowHTML = '';
|
|
|
var row = i;
|
|
|
for (j = 1; j <= cols; j++) {
|
|
|
var col = j;
|
|
|
dayIndex ++;
|
|
|
var dayNumber = dayIndex - firstDayOfMonthIndex;
|
|
|
var addClass = '';
|
|
|
if (dayNumber < 0) {
|
|
|
dayNumber = daysInPrevMonth + dayNumber + 1;
|
|
|
addClass += ' picker-calendar-day-prev';
|
|
|
dayDate = new Date(month - 1 < 0 ? year - 1 : year, month - 1 < 0 ? 11 : month - 1, dayNumber).getTime();
|
|
|
}
|
|
|
else {
|
|
|
dayNumber = dayNumber + 1;
|
|
|
if (dayNumber > daysInMonth) {
|
|
|
dayNumber = dayNumber - daysInMonth;
|
|
|
addClass += ' picker-calendar-day-next';
|
|
|
dayDate = new Date(month + 1 > 11 ? year + 1 : year, month + 1 > 11 ? 0 : month + 1, dayNumber).getTime();
|
|
|
}
|
|
|
else {
|
|
|
dayDate = new Date(year, month, dayNumber).getTime();
|
|
|
}
|
|
|
}
|
|
|
// Today
|
|
|
if (dayDate === today) addClass += ' picker-calendar-day-today';
|
|
|
// Selected
|
|
|
if (currentValues.indexOf(dayDate) >= 0) addClass += ' picker-calendar-day-selected';
|
|
|
// Weekend
|
|
|
if (p.params.weekendDays.indexOf(col - 1) >= 0) {
|
|
|
addClass += ' picker-calendar-day-weekend';
|
|
|
}
|
|
|
// Disabled
|
|
|
if ((minDate && dayDate < minDate) || (maxDate && dayDate > maxDate)) {
|
|
|
addClass += ' picker-calendar-day-disabled';
|
|
|
}
|
|
|
|
|
|
dayDate = new Date(dayDate);
|
|
|
var dayYear = dayDate.getFullYear();
|
|
|
var dayMonth = dayDate.getMonth();
|
|
|
rowHTML += '<div data-year="' + dayYear + '" data-month="' + dayMonth + '" data-day="' + dayNumber + '" class="picker-calendar-day' + (addClass) + '" data-date="' + (dayYear + '-' + dayMonth + '-' + dayNumber) + '"><span>'+dayNumber+'</span></div>';
|
|
|
}
|
|
|
monthHTML += '<div class="picker-calendar-row">' + rowHTML + '</div>';
|
|
|
}
|
|
|
monthHTML = '<div class="picker-calendar-month" data-year="' + year + '" data-month="' + month + '">' + monthHTML + '</div>';
|
|
|
return monthHTML;
|
|
|
};
|
|
|
p.animating = false;
|
|
|
p.updateCurrentMonthYear = function (dir) {
|
|
|
if (typeof dir === 'undefined') {
|
|
|
p.currentMonth = parseInt(p.months.eq(1).attr('data-month'), 10);
|
|
|
p.currentYear = parseInt(p.months.eq(1).attr('data-year'), 10);
|
|
|
}
|
|
|
else {
|
|
|
p.currentMonth = parseInt(p.months.eq(dir === 'next' ? (p.months.length - 1) : 0).attr('data-month'), 10);
|
|
|
p.currentYear = parseInt(p.months.eq(dir === 'next' ? (p.months.length - 1) : 0).attr('data-year'), 10);
|
|
|
}
|
|
|
p.container.find('.current-month-value').text(p.params.monthNames[p.currentMonth]);
|
|
|
p.container.find('.current-year-value').text(p.currentYear);
|
|
|
|
|
|
};
|
|
|
p.onMonthChangeStart = function (dir) {
|
|
|
p.updateCurrentMonthYear(dir);
|
|
|
p.months.removeClass('picker-calendar-month-current picker-calendar-month-prev picker-calendar-month-next');
|
|
|
var currentIndex = dir === 'next' ? p.months.length - 1 : 0;
|
|
|
|
|
|
p.months.eq(currentIndex).addClass('picker-calendar-month-current');
|
|
|
p.months.eq(dir === 'next' ? currentIndex - 1 : currentIndex + 1).addClass(dir === 'next' ? 'picker-calendar-month-prev' : 'picker-calendar-month-next');
|
|
|
|
|
|
if (p.params.onMonthYearChangeStart) {
|
|
|
p.params.onMonthYearChangeStart(p, p.currentYear, p.currentMonth);
|
|
|
}
|
|
|
};
|
|
|
p.onMonthChangeEnd = function (dir, rebuildBoth) {
|
|
|
p.animating = false;
|
|
|
var nextMonthHTML, prevMonthHTML, newMonthHTML;
|
|
|
p.wrapper.find('.picker-calendar-month:not(.picker-calendar-month-prev):not(.picker-calendar-month-current):not(.picker-calendar-month-next)').remove();
|
|
|
|
|
|
if (typeof dir === 'undefined') {
|
|
|
dir = 'next';
|
|
|
rebuildBoth = true;
|
|
|
}
|
|
|
if (!rebuildBoth) {
|
|
|
newMonthHTML = p.monthHTML(new Date(p.currentYear, p.currentMonth), dir);
|
|
|
}
|
|
|
else {
|
|
|
p.wrapper.find('.picker-calendar-month-next, .picker-calendar-month-prev').remove();
|
|
|
prevMonthHTML = p.monthHTML(new Date(p.currentYear, p.currentMonth), 'prev');
|
|
|
nextMonthHTML = p.monthHTML(new Date(p.currentYear, p.currentMonth), 'next');
|
|
|
}
|
|
|
if (dir === 'next' || rebuildBoth) {
|
|
|
p.wrapper.append(newMonthHTML || nextMonthHTML);
|
|
|
}
|
|
|
if (dir === 'prev' || rebuildBoth) {
|
|
|
p.wrapper.prepend(newMonthHTML || prevMonthHTML);
|
|
|
}
|
|
|
p.months = p.wrapper.find('.picker-calendar-month');
|
|
|
p.setMonthsTranslate(p.monthsTranslate);
|
|
|
if (p.params.onMonthAdd) {
|
|
|
p.params.onMonthAdd(p, dir === 'next' ? p.months.eq(p.months.length - 1)[0] : p.months.eq(0)[0]);
|
|
|
}
|
|
|
if (p.params.onMonthYearChangeEnd) {
|
|
|
p.params.onMonthYearChangeEnd(p, p.currentYear, p.currentMonth);
|
|
|
}
|
|
|
};
|
|
|
p.setMonthsTranslate = function (translate) {
|
|
|
translate = translate || p.monthsTranslate || 0;
|
|
|
if (typeof p.monthsTranslate === 'undefined') p.monthsTranslate = translate;
|
|
|
p.months.removeClass('picker-calendar-month-current picker-calendar-month-prev picker-calendar-month-next');
|
|
|
var prevMonthTranslate = -(translate + 1) * 100 * inverter;
|
|
|
var currentMonthTranslate = -translate * 100 * inverter;
|
|
|
var nextMonthTranslate = -(translate - 1) * 100 * inverter;
|
|
|
p.months.eq(0).transform('translate3d(' + (p.isH ? prevMonthTranslate : 0) + '%, ' + (p.isH ? 0 : prevMonthTranslate) + '%, 0)').addClass('picker-calendar-month-prev');
|
|
|
p.months.eq(1).transform('translate3d(' + (p.isH ? currentMonthTranslate : 0) + '%, ' + (p.isH ? 0 : currentMonthTranslate) + '%, 0)').addClass('picker-calendar-month-current');
|
|
|
p.months.eq(2).transform('translate3d(' + (p.isH ? nextMonthTranslate : 0) + '%, ' + (p.isH ? 0 : nextMonthTranslate) + '%, 0)').addClass('picker-calendar-month-next');
|
|
|
};
|
|
|
p.nextMonth = function (transition) {
|
|
|
if (typeof transition === 'undefined' || typeof transition === 'object') {
|
|
|
transition = '';
|
|
|
if (!p.params.animate) transition = 0;
|
|
|
}
|
|
|
var nextMonth = parseInt(p.months.eq(p.months.length - 1).attr('data-month'), 10);
|
|
|
var nextYear = parseInt(p.months.eq(p.months.length - 1).attr('data-year'), 10);
|
|
|
var nextDate = new Date(nextYear, nextMonth);
|
|
|
var nextDateTime = nextDate.getTime();
|
|
|
var transitionEndCallback = p.animating ? false : true;
|
|
|
if (p.params.maxDate) {
|
|
|
if (nextDateTime > new Date(p.params.maxDate).getTime()) {
|
|
|
return p.resetMonth();
|
|
|
}
|
|
|
}
|
|
|
p.monthsTranslate --;
|
|
|
if (nextMonth === p.currentMonth) {
|
|
|
var nextMonthTranslate = -(p.monthsTranslate) * 100 * inverter;
|
|
|
var nextMonthHTML = $(p.monthHTML(nextDateTime, 'next')).transform('translate3d(' + (p.isH ? nextMonthTranslate : 0) + '%, ' + (p.isH ? 0 : nextMonthTranslate) + '%, 0)').addClass('picker-calendar-month-next');
|
|
|
p.wrapper.append(nextMonthHTML[0]);
|
|
|
p.months = p.wrapper.find('.picker-calendar-month');
|
|
|
if (p.params.onMonthAdd) {
|
|
|
p.params.onMonthAdd(p, p.months.eq(p.months.length - 1)[0]);
|
|
|
}
|
|
|
}
|
|
|
p.animating = true;
|
|
|
p.onMonthChangeStart('next');
|
|
|
var translate = (p.monthsTranslate * 100) * inverter;
|
|
|
|
|
|
p.wrapper.transition(transition).transform('translate3d(' + (p.isH ? translate : 0) + '%, ' + (p.isH ? 0 : translate) + '%, 0)');
|
|
|
if (transitionEndCallback) {
|
|
|
p.wrapper.transitionEnd(function () {
|
|
|
p.onMonthChangeEnd('next');
|
|
|
});
|
|
|
}
|
|
|
if (!p.params.animate) {
|
|
|
p.onMonthChangeEnd('next');
|
|
|
}
|
|
|
};
|
|
|
p.prevMonth = function (transition) {
|
|
|
if (typeof transition === 'undefined' || typeof transition === 'object') {
|
|
|
transition = '';
|
|
|
if (!p.params.animate) transition = 0;
|
|
|
}
|
|
|
var prevMonth = parseInt(p.months.eq(0).attr('data-month'), 10);
|
|
|
var prevYear = parseInt(p.months.eq(0).attr('data-year'), 10);
|
|
|
var prevDate = new Date(prevYear, prevMonth + 1, -1);
|
|
|
var prevDateTime = prevDate.getTime();
|
|
|
var transitionEndCallback = p.animating ? false : true;
|
|
|
if (p.params.minDate) {
|
|
|
if (prevDateTime < new Date(p.params.minDate).getTime()) {
|
|
|
return p.resetMonth();
|
|
|
}
|
|
|
}
|
|
|
p.monthsTranslate ++;
|
|
|
if (prevMonth === p.currentMonth) {
|
|
|
var prevMonthTranslate = -(p.monthsTranslate) * 100 * inverter;
|
|
|
var prevMonthHTML = $(p.monthHTML(prevDateTime, 'prev')).transform('translate3d(' + (p.isH ? prevMonthTranslate : 0) + '%, ' + (p.isH ? 0 : prevMonthTranslate) + '%, 0)').addClass('picker-calendar-month-prev');
|
|
|
p.wrapper.prepend(prevMonthHTML[0]);
|
|
|
p.months = p.wrapper.find('.picker-calendar-month');
|
|
|
if (p.params.onMonthAdd) {
|
|
|
p.params.onMonthAdd(p, p.months.eq(0)[0]);
|
|
|
}
|
|
|
}
|
|
|
p.animating = true;
|
|
|
p.onMonthChangeStart('prev');
|
|
|
var translate = (p.monthsTranslate * 100) * inverter;
|
|
|
p.wrapper.transition(transition).transform('translate3d(' + (p.isH ? translate : 0) + '%, ' + (p.isH ? 0 : translate) + '%, 0)');
|
|
|
if (transitionEndCallback) {
|
|
|
p.wrapper.transitionEnd(function () {
|
|
|
p.onMonthChangeEnd('prev');
|
|
|
});
|
|
|
}
|
|
|
if (!p.params.animate) {
|
|
|
p.onMonthChangeEnd('prev');
|
|
|
}
|
|
|
};
|
|
|
p.resetMonth = function (transition) {
|
|
|
if (typeof transition === 'undefined') transition = '';
|
|
|
var translate = (p.monthsTranslate * 100) * inverter;
|
|
|
p.wrapper.transition(transition).transform('translate3d(' + (p.isH ? translate : 0) + '%, ' + (p.isH ? 0 : translate) + '%, 0)');
|
|
|
};
|
|
|
p.setYearMonth = function (year, month, transition) {
|
|
|
if (typeof year === 'undefined') year = p.currentYear;
|
|
|
if (typeof month === 'undefined') month = p.currentMonth;
|
|
|
if (typeof transition === 'undefined' || typeof transition === 'object') {
|
|
|
transition = '';
|
|
|
if (!p.params.animate) transition = 0;
|
|
|
}
|
|
|
var targetDate;
|
|
|
if (year < p.currentYear) {
|
|
|
targetDate = new Date(year, month + 1, -1).getTime();
|
|
|
}
|
|
|
else {
|
|
|
targetDate = new Date(year, month).getTime();
|
|
|
}
|
|
|
if (p.params.maxDate && targetDate > new Date(p.params.maxDate).getTime()) {
|
|
|
return false;
|
|
|
}
|
|
|
if (p.params.minDate && targetDate < new Date(p.params.minDate).getTime()) {
|
|
|
return false;
|
|
|
}
|
|
|
var currentDate = new Date(p.currentYear, p.currentMonth).getTime();
|
|
|
var dir = targetDate > currentDate ? 'next' : 'prev';
|
|
|
var newMonthHTML = p.monthHTML(new Date(year, month));
|
|
|
p.monthsTranslate = p.monthsTranslate || 0;
|
|
|
var prevTranslate = p.monthsTranslate;
|
|
|
var monthTranslate, wrapperTranslate;
|
|
|
var transitionEndCallback = p.animating ? false : true;
|
|
|
if (targetDate > currentDate) {
|
|
|
// To next
|
|
|
p.monthsTranslate --;
|
|
|
if (!p.animating) p.months.eq(p.months.length - 1).remove();
|
|
|
p.wrapper.append(newMonthHTML);
|
|
|
p.months = p.wrapper.find('.picker-calendar-month');
|
|
|
monthTranslate = -(prevTranslate - 1) * 100 * inverter;
|
|
|
p.months.eq(p.months.length - 1).transform('translate3d(' + (p.isH ? monthTranslate : 0) + '%, ' + (p.isH ? 0 : monthTranslate) + '%, 0)').addClass('picker-calendar-month-next');
|
|
|
}
|
|
|
else {
|
|
|
// To prev
|
|
|
p.monthsTranslate ++;
|
|
|
if (!p.animating) p.months.eq(0).remove();
|
|
|
p.wrapper.prepend(newMonthHTML);
|
|
|
p.months = p.wrapper.find('.picker-calendar-month');
|
|
|
monthTranslate = -(prevTranslate + 1) * 100 * inverter;
|
|
|
p.months.eq(0).transform('translate3d(' + (p.isH ? monthTranslate : 0) + '%, ' + (p.isH ? 0 : monthTranslate) + '%, 0)').addClass('picker-calendar-month-prev');
|
|
|
}
|
|
|
if (p.params.onMonthAdd) {
|
|
|
p.params.onMonthAdd(p, dir === 'next' ? p.months.eq(p.months.length - 1)[0] : p.months.eq(0)[0]);
|
|
|
}
|
|
|
p.animating = true;
|
|
|
p.onMonthChangeStart(dir);
|
|
|
wrapperTranslate = (p.monthsTranslate * 100) * inverter;
|
|
|
p.wrapper.transition(transition).transform('translate3d(' + (p.isH ? wrapperTranslate : 0) + '%, ' + (p.isH ? 0 : wrapperTranslate) + '%, 0)');
|
|
|
if (transitionEndCallback) {
|
|
|
p.wrapper.transitionEnd(function () {
|
|
|
p.onMonthChangeEnd(dir, true);
|
|
|
});
|
|
|
}
|
|
|
if (!p.params.animate) {
|
|
|
p.onMonthChangeEnd(dir);
|
|
|
}
|
|
|
};
|
|
|
p.nextYear = function () {
|
|
|
p.setYearMonth(p.currentYear + 1);
|
|
|
};
|
|
|
p.prevYear = function () {
|
|
|
p.setYearMonth(p.currentYear - 1);
|
|
|
};
|
|
|
|
|
|
|
|
|
// HTML Layout
|
|
|
p.layout = function () {
|
|
|
var pickerHTML = '';
|
|
|
var pickerClass = '';
|
|
|
var i;
|
|
|
|
|
|
var layoutDate = p.value && p.value.length ? p.value[0] : new Date().setHours(0,0,0,0);
|
|
|
var prevMonthHTML = p.monthHTML(layoutDate, 'prev');
|
|
|
var currentMonthHTML = p.monthHTML(layoutDate);
|
|
|
var nextMonthHTML = p.monthHTML(layoutDate, 'next');
|
|
|
var monthsHTML = '<div class="picker-calendar-months"><div class="picker-calendar-months-wrapper">' + (prevMonthHTML + currentMonthHTML + nextMonthHTML) + '</div></div>';
|
|
|
// Week days header
|
|
|
var weekHeaderHTML = '';
|
|
|
if (p.params.weekHeader) {
|
|
|
for (i = 0; i < 7; i++) {
|
|
|
var weekDayIndex = (i + p.params.firstDay > 6) ? (i - 7 + p.params.firstDay) : (i + p.params.firstDay);
|
|
|
var dayName = p.params.dayNamesShort[weekDayIndex];
|
|
|
weekHeaderHTML += '<div class="picker-calendar-week-day ' + ((p.params.weekendDays.indexOf(weekDayIndex) >= 0) ? 'picker-calendar-week-day-weekend' : '') + '"> ' + dayName + '</div>';
|
|
|
|
|
|
}
|
|
|
weekHeaderHTML = '<div class="picker-calendar-week-days">' + weekHeaderHTML + '</div>';
|
|
|
}
|
|
|
pickerClass = 'weui-picker-calendar ' + (p.params.cssClass || '');
|
|
|
if(!p.inline) pickerClass = 'weui-picker-modal ' + pickerClass;
|
|
|
var toolbarHTML = p.params.toolbar ? p.params.toolbarTemplate.replace(/{{closeText}}/g, p.params.toolbarCloseText) : '';
|
|
|
if (p.params.toolbar) {
|
|
|
toolbarHTML = p.params.toolbarTemplate
|
|
|
.replace(/{{closeText}}/g, p.params.toolbarCloseText)
|
|
|
.replace(/{{monthPicker}}/g, (p.params.monthPicker ? p.params.monthPickerTemplate : ''))
|
|
|
.replace(/{{yearPicker}}/g, (p.params.yearPicker ? p.params.yearPickerTemplate : ''));
|
|
|
}
|
|
|
|
|
|
pickerHTML =
|
|
|
'<div class="' + (pickerClass) + '">' +
|
|
|
toolbarHTML +
|
|
|
'<div class="picker-modal-inner">' +
|
|
|
weekHeaderHTML +
|
|
|
monthsHTML +
|
|
|
'</div>' +
|
|
|
'</div>';
|
|
|
|
|
|
|
|
|
p.pickerHTML = pickerHTML;
|
|
|
};
|
|
|
|
|
|
// Input Events
|
|
|
function openOnInput(e) {
|
|
|
e.preventDefault();
|
|
|
if (p.opened) return;
|
|
|
p.open();
|
|
|
if (p.params.scrollToInput && !isPopover()) {
|
|
|
var pageContent = p.input.parents('.page-content');
|
|
|
if (pageContent.length === 0) return;
|
|
|
|
|
|
var paddingTop = parseInt(pageContent.css('padding-top'), 10),
|
|
|
paddingBottom = parseInt(pageContent.css('padding-bottom'), 10),
|
|
|
pageHeight = pageContent[0].offsetHeight - paddingTop - p.container.height(),
|
|
|
pageScrollHeight = pageContent[0].scrollHeight - paddingTop - p.container.height(),
|
|
|
newPaddingBottom;
|
|
|
|
|
|
var inputTop = p.input.offset().top - paddingTop + p.input[0].offsetHeight;
|
|
|
if (inputTop > pageHeight) {
|
|
|
var scrollTop = pageContent.scrollTop() + inputTop - pageHeight;
|
|
|
if (scrollTop + pageHeight > pageScrollHeight) {
|
|
|
newPaddingBottom = scrollTop + pageHeight - pageScrollHeight + paddingBottom;
|
|
|
if (pageHeight === pageScrollHeight) {
|
|
|
newPaddingBottom = p.container.height();
|
|
|
}
|
|
|
pageContent.css({'padding-bottom': (newPaddingBottom) + 'px'});
|
|
|
}
|
|
|
pageContent.scrollTop(scrollTop, 300);
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
function closeOnHTMLClick(e) {
|
|
|
if (inPopover()) return;
|
|
|
if (p.input && p.input.length > 0) {
|
|
|
if (e.target !== p.input[0] && $(e.target).parents('.weui-picker-modal').length === 0) p.close();
|
|
|
}
|
|
|
else {
|
|
|
if ($(e.target).parents('.weui-picker-modal').length === 0) p.close();
|
|
|
}
|
|
|
}
|
|
|
|
|
|
if (p.params.input) {
|
|
|
p.input = $(p.params.input);
|
|
|
if (p.input.length > 0) {
|
|
|
if (p.params.inputReadOnly) p.input.prop('readOnly', true);
|
|
|
if (!p.inline) {
|
|
|
p.input.on('click', openOnInput);
|
|
|
}
|
|
|
if (p.params.inputReadOnly) {
|
|
|
p.input.on('focus mousedown', function (e) {
|
|
|
e.preventDefault();
|
|
|
});
|
|
|
}
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
//iphone 上无法正确触发 click,会导致点击外面无法关闭
|
|
|
if (!p.inline) $(document).on('click touchend', closeOnHTMLClick);
|
|
|
|
|
|
// Open
|
|
|
function onPickerClose() {
|
|
|
p.opened = false;
|
|
|
if (p.input && p.input.length > 0) p.input.parents('.page-content').css({'padding-bottom': ''});
|
|
|
if (p.params.onClose) p.params.onClose(p);
|
|
|
|
|
|
// Destroy events
|
|
|
p.destroyCalendarEvents();
|
|
|
}
|
|
|
|
|
|
p.opened = false;
|
|
|
p.open = function () {
|
|
|
var toPopover = isPopover() && false;
|
|
|
var updateValue = false;
|
|
|
if (!p.opened) {
|
|
|
// Set date value
|
|
|
if (!p.value) {
|
|
|
if (p.params.value) {
|
|
|
p.value = p.params.value;
|
|
|
updateValue = true;
|
|
|
}
|
|
|
}
|
|
|
|
|
|
// Layout
|
|
|
p.layout();
|
|
|
|
|
|
// Append
|
|
|
if (toPopover) {
|
|
|
p.pickerHTML = '<div class="popover popover-picker-calendar"><div class="popover-inner">' + p.pickerHTML + '</div></div>';
|
|
|
p.popover = $.popover(p.pickerHTML, p.params.input, true);
|
|
|
p.container = $(p.popover).find('.weui-picker-modal');
|
|
|
$(p.popover).on('close', function () {
|
|
|
onPickerClose();
|
|
|
});
|
|
|
}
|
|
|
else if (p.inline) {
|
|
|
p.container = $(p.pickerHTML);
|
|
|
p.container.addClass('picker-modal-inline');
|
|
|
$(p.params.container).append(p.container);
|
|
|
}
|
|
|
else {
|
|
|
p.container = $($.openPicker(p.pickerHTML));
|
|
|
$(p.container)
|
|
|
.on('close', function () {
|
|
|
onPickerClose();
|
|
|
});
|
|
|
}
|
|
|
|
|
|
// Store calendar instance
|
|
|
p.container[0].f7Calendar = p;
|
|
|
p.wrapper = p.container.find('.picker-calendar-months-wrapper');
|
|
|
|
|
|
// Months
|
|
|
p.months = p.wrapper.find('.picker-calendar-month');
|
|
|
|
|
|
// Update current month and year
|
|
|
p.updateCurrentMonthYear();
|
|
|
|
|
|
// Set initial translate
|
|
|
p.monthsTranslate = 0;
|
|
|
p.setMonthsTranslate();
|
|
|
|
|
|
// Init events
|
|
|
p.initCalendarEvents();
|
|
|
|
|
|
// Update input value
|
|
|
if (updateValue) p.updateValue();
|
|
|
|
|
|
}
|
|
|
|
|
|
// Set flag
|
|
|
p.opened = true;
|
|
|
p.initialized = true;
|
|
|
if (p.params.onMonthAdd) {
|
|
|
p.months.each(function () {
|
|
|
p.params.onMonthAdd(p, this);
|
|
|
});
|
|
|
}
|
|
|
if (p.params.onOpen) p.params.onOpen(p);
|
|
|
};
|
|
|
|
|
|
// Close
|
|
|
p.close = function () {
|
|
|
if (!p.opened || p.inline) return;
|
|
|
p.animating = false; //有可能还有动画没做完,因此animating设置还没改。
|
|
|
if (inPopover()) {
|
|
|
$.closePicker(p.popover);
|
|
|
return;
|
|
|
}
|
|
|
else {
|
|
|
$.closePicker(p.container);
|
|
|
return;
|
|
|
}
|
|
|
};
|
|
|
|
|
|
// Destroy
|
|
|
p.destroy = function () {
|
|
|
p.close();
|
|
|
if (p.params.input && p.input.length > 0) {
|
|
|
p.input.off('click focus', openOnInput);
|
|
|
p.input.data("calendar", null);
|
|
|
}
|
|
|
$('html').off('click', closeOnHTMLClick);
|
|
|
};
|
|
|
|
|
|
if (p.inline) {
|
|
|
p.open();
|
|
|
}
|
|
|
|
|
|
return p;
|
|
|
};
|
|
|
|
|
|
var format = function(d) {
|
|
|
return d < 10 ? "0"+d : d;
|
|
|
}
|
|
|
|
|
|
|
|
|
$.fn.calendar = function (params, args) {
|
|
|
params = params || {};
|
|
|
return this.each(function() {
|
|
|
var $this = $(this);
|
|
|
if(!$this[0]) return;
|
|
|
var p = {};
|
|
|
if($this[0].tagName.toUpperCase() === "INPUT") {
|
|
|
p.input = $this;
|
|
|
} else {
|
|
|
p.container = $this;
|
|
|
}
|
|
|
|
|
|
var calendar = $this.data("calendar");
|
|
|
|
|
|
if(!calendar) {
|
|
|
if(typeof params === typeof "a") {
|
|
|
} else {
|
|
|
if(!params.value && $this.val()) params.value = [$this.val()];
|
|
|
//默认显示今天
|
|
|
if(!params.value) {
|
|
|
var today = new Date();
|
|
|
params.value = [today.getFullYear() + "/" + format(today.getMonth() + 1) + "/" + format(today.getDate())];
|
|
|
}
|
|
|
calendar = $this.data("calendar", new Calendar($.extend(p, params)));
|
|
|
}
|
|
|
}
|
|
|
|
|
|
if(typeof params === typeof "a") {
|
|
|
calendar[params].call(calendar, args);
|
|
|
}
|
|
|
});
|
|
|
};
|
|
|
|
|
|
defaults = $.fn.calendar.prototype.defaults = {
|
|
|
value: undefined, // 通过JS赋值,注意是数组
|
|
|
monthNames: ['一月', '二月', '三月', '四月', '五月', '六月', '七月', '八月', '九月', '十月', '十一月', '十二月'],
|
|
|
monthNamesShort: ['一月', '二月', '三月', '四月', '五月', '六月', '七月', '八月', '九月', '十月', '十一月', '十二月'],
|
|
|
dayNames: ['周日', '周一', '周二', '周三', '周四', '周五', '周六'],
|
|
|
dayNamesShort: ['周日', '周一', '周二', '周三', '周四', '周五', '周六'],
|
|
|
firstDay: 1, // First day of the week, Monday
|
|
|
weekendDays: [0, 6], // Sunday and Saturday
|
|
|
multiple: false,
|
|
|
dateFormat: 'yyyy/mm/dd',
|
|
|
direction: 'horizontal', // or 'vertical'
|
|
|
minDate: null,
|
|
|
maxDate: null,
|
|
|
touchMove: true,
|
|
|
animate: true,
|
|
|
closeOnSelect: true,
|
|
|
monthPicker: true,
|
|
|
monthPickerTemplate:
|
|
|
'<div class="picker-calendar-month-picker">' +
|
|
|
'<a href="javascript:;" class="link icon-only picker-calendar-prev-month"><i class="icon icon-prev"></i></a>' +
|
|
|
'<div class="current-month-value"></div>' +
|
|
|
'<a href="javascript:;" class="link icon-only picker-calendar-next-month"><i class="icon icon-next"></i></a>' +
|
|
|
'</div>',
|
|
|
yearPicker: true,
|
|
|
yearPickerTemplate:
|
|
|
'<div class="picker-calendar-year-picker">' +
|
|
|
'<a href="javascript:;" class="link icon-only picker-calendar-prev-year"><i class="icon icon-prev"></i></a>' +
|
|
|
'<span class="current-year-value"></span>' +
|
|
|
'<a href="javascript:;" class="link icon-only picker-calendar-next-year"><i class="icon icon-next"></i></a>' +
|
|
|
'</div>',
|
|
|
weekHeader: true,
|
|
|
// Common settings
|
|
|
scrollToInput: true,
|
|
|
inputReadOnly: true,
|
|
|
convertToPopover: true,
|
|
|
onlyInPopover: false,
|
|
|
toolbar: true,
|
|
|
toolbarCloseText: 'Done',
|
|
|
toolbarTemplate:
|
|
|
'<div class="toolbar">' +
|
|
|
'<div class="toolbar-inner">' +
|
|
|
'{{yearPicker}}' +
|
|
|
'{{monthPicker}}' +
|
|
|
// '<a href="#" class="link close-picker">{{closeText}}</a>' +
|
|
|
'</div>' +
|
|
|
'</div>',
|
|
|
/* Callbacks
|
|
|
onMonthAdd
|
|
|
onChange
|
|
|
onOpen
|
|
|
onClose
|
|
|
onDayClick
|
|
|
onMonthYearChangeStart
|
|
|
onMonthYearChangeEnd
|
|
|
*/
|
|
|
};
|
|
|
|
|
|
}($);
|
|
|
|
|
|
/* global $:true */
|
|
|
/* jshint unused:false*/
|
|
|
|
|
|
+ function($) {
|
|
|
"use strict";
|
|
|
|
|
|
|
|
|
var defaults;
|
|
|
|
|
|
var formatNumber = function (n) {
|
|
|
return n < 10 ? "0" + n : n;
|
|
|
}
|
|
|
|
|
|
var Datetime = function(input, params) {
|
|
|
this.input = $(input);
|
|
|
this.params = params || {};
|
|
|
|
|
|
this.initMonthes = params.monthes
|
|
|
|
|
|
this.initYears = params.years
|
|
|
|
|
|
var p = $.extend({}, params, this.getConfig());
|
|
|
$(this.input).picker(p);
|
|
|
}
|
|
|
|
|
|
Datetime.prototype = {
|
|
|
getDays : function(max) {
|
|
|
var days = [];
|
|
|
for(var i=1; i<= (max||31);i++) {
|
|
|
days.push(i < 10 ? "0"+i : i);
|
|
|
}
|
|
|
return days;
|
|
|
},
|
|
|
|
|
|
getDaysByMonthAndYear : function(month, year) {
|
|
|
var int_d = new Date(year, parseInt(month)+1-1, 1);
|
|
|
var d = new Date(int_d - 1);
|
|
|
return this.getDays(d.getDate());
|
|
|
},
|
|
|
getConfig: function() {
|
|
|
var today = new Date(),
|
|
|
params = this.params,
|
|
|
self = this,
|
|
|
lastValidValues;
|
|
|
|
|
|
var config = {
|
|
|
rotateEffect: false, //为了性能
|
|
|
cssClass: 'datetime-picker',
|
|
|
|
|
|
value: [today.getFullYear(), formatNumber(today.getMonth()+1), formatNumber(today.getDate()), formatNumber(today.getHours()), (formatNumber(today.getMinutes()))],
|
|
|
|
|
|
onChange: function (picker, values, displayValues) {
|
|
|
var cols = picker.cols;
|
|
|
var days = self.getDaysByMonthAndYear(values[1], values[0]);
|
|
|
var currentValue = values[2];
|
|
|
if(currentValue > days.length) currentValue = days.length;
|
|
|
picker.cols[4].setValue(currentValue);
|
|
|
|
|
|
//check min and max
|
|
|
var current = new Date(values[0]+'-'+values[1]+'-'+values[2]);
|
|
|
var valid = true;
|
|
|
if(params.min) {
|
|
|
var min = new Date(typeof params.min === "function" ? params.min() : params.min);
|
|
|
|
|
|
if(current < +min) {
|
|
|
picker.setValue(lastValidValues);
|
|
|
valid = false;
|
|
|
}
|
|
|
}
|
|
|
if(params.max) {
|
|
|
var max = new Date(typeof params.max === "function" ? params.max() : params.max);
|
|
|
if(current > +max) {
|
|
|
picker.setValue(lastValidValues);
|
|
|
valid = false;
|
|
|
}
|
|
|
}
|
|
|
|
|
|
valid && (lastValidValues = values);
|
|
|
|
|
|
if (self.params.onChange) {
|
|
|
self.params.onChange.apply(this, arguments);
|
|
|
}
|
|
|
},
|
|
|
|
|
|
formatValue: function (p, values, displayValues) {
|
|
|
return self.params.format(p, values, displayValues);
|
|
|
},
|
|
|
|
|
|
cols: [
|
|
|
{
|
|
|
values: this.initYears
|
|
|
},
|
|
|
{
|
|
|
divider: true, // 这是一个分隔符
|
|
|
content: params.yearSplit
|
|
|
},
|
|
|
{
|
|
|
values: this.initMonthes
|
|
|
},
|
|
|
{
|
|
|
divider: true, // 这是一个分隔符
|
|
|
content: params.monthSplit
|
|
|
},
|
|
|
{
|
|
|
values: (function () {
|
|
|
var dates = [];
|
|
|
for (var i=1; i<=31; i++) dates.push(formatNumber(i));
|
|
|
return dates;
|
|
|
})()
|
|
|
},
|
|
|
|
|
|
]
|
|
|
}
|
|
|
|
|
|
if (params.dateSplit) {
|
|
|
config.cols.push({
|
|
|
divider: true,
|
|
|
content: params.dateSplit
|
|
|
})
|
|
|
}
|
|
|
|
|
|
config.cols.push({
|
|
|
divider: true,
|
|
|
content: params.datetimeSplit
|
|
|
})
|
|
|
|
|
|
var times = self.params.times();
|
|
|
if (times && times.length) {
|
|
|
config.cols = config.cols.concat(times);
|
|
|
}
|
|
|
|
|
|
var inputValue = this.input.val();
|
|
|
if(inputValue) config.value = params.parse(inputValue);
|
|
|
if(this.params.value) {
|
|
|
this.input.val(this.params.value);
|
|
|
config.value = params.parse(this.params.value);
|
|
|
}
|
|
|
|
|
|
return config;
|
|
|
}
|
|
|
}
|
|
|
|
|
|
$.fn.datetimePicker = function(params) {
|
|
|
params = $.extend({}, defaults, params);
|
|
|
return this.each(function() {
|
|
|
if(!this) return;
|
|
|
var $this = $(this);
|
|
|
var datetime = $this.data("datetime");
|
|
|
if(!datetime) $this.data("datetime", new Datetime(this, params));
|
|
|
return datetime;
|
|
|
});
|
|
|
};
|
|
|
|
|
|
defaults = $.fn.datetimePicker.prototype.defaults = {
|
|
|
input: undefined, // 默认值
|
|
|
min: undefined, // YYYY-MM-DD 最大最小值只比较年月日,不比较时分秒
|
|
|
max: undefined, // YYYY-MM-DD
|
|
|
yearSplit: '-',
|
|
|
monthSplit: '-',
|
|
|
dateSplit: '', // 默认为空
|
|
|
datetimeSplit: ' ', // 日期和时间之间的分隔符,不可为空
|
|
|
monthes: ('01 02 03 04 05 06 07 08 09 10 11 12').split(' '),
|
|
|
years: (function () {
|
|
|
var arr = [];
|
|
|
for (var i = 1950; i <= 2030; i++) { arr.push(i); }
|
|
|
return arr;
|
|
|
})(),
|
|
|
times: function () {
|
|
|
return [ // 自定义的时间
|
|
|
{
|
|
|
values: (function () {
|
|
|
var hours = [];
|
|
|
for (var i=0; i<24; i++) hours.push(formatNumber(i));
|
|
|
return hours;
|
|
|
})()
|
|
|
},
|
|
|
{
|
|
|
divider: true, // 这是一个分隔符
|
|
|
content: ':'
|
|
|
},
|
|
|
{
|
|
|
values: (function () {
|
|
|
var minutes = [];
|
|
|
for (var i=0; i<60; i++) minutes.push(formatNumber(i));
|
|
|
return minutes;
|
|
|
})()
|
|
|
}
|
|
|
];
|
|
|
},
|
|
|
format: function (p, values) { // 数组转换成字符串
|
|
|
return p.cols.map(function (col) {
|
|
|
return col.value || col.content;
|
|
|
}).join('');
|
|
|
},
|
|
|
parse: function (str) {
|
|
|
// 把字符串转换成数组,用来解析初始值
|
|
|
// 如果你的定制的初始值格式无法被这个默认函数解析,请自定义这个函数。比如你的时间是 '子时' 那么默认情况这个'时'会被当做分隔符而导致错误,所以你需要自己定义parse函数
|
|
|
// 默认兼容的分隔符
|
|
|
var t = str.split(this.datetimeSplit);
|
|
|
return t[0].split(/\D/).concat(t[1].split(/:|时|分|秒/)).filter(function (d) {
|
|
|
return !!d;
|
|
|
})
|
|
|
}
|
|
|
}
|
|
|
|
|
|
}($);
|
|
|
|
|
|
/*======================================================
|
|
|
************ Picker ************
|
|
|
======================================================*/
|
|
|
/* global $:true */
|
|
|
|
|
|
+ function($) {
|
|
|
"use strict";
|
|
|
|
|
|
|
|
|
//Popup 和 picker 之类的不要共用一个弹出方法,因为这样会导致 在 popup 中再弹出 picker 的时候会有问题。
|
|
|
|
|
|
$.openPopup = function(popup, className) {
|
|
|
|
|
|
$.closePopup();
|
|
|
|
|
|
popup = $(popup);
|
|
|
popup.show();
|
|
|
popup.width();
|
|
|
popup.addClass("weui-popup__container--visible");
|
|
|
var modal = popup.find(".weui-popup__modal");
|
|
|
modal.width();
|
|
|
modal.transitionEnd(function() {
|
|
|
modal.trigger("open");
|
|
|
});
|
|
|
}
|
|
|
|
|
|
|
|
|
$.closePopup = function(container, remove) {
|
|
|
container = $(container || ".weui-popup__container--visible");
|
|
|
container.find('.weui-popup__modal').transitionEnd(function() {
|
|
|
var $this = $(this);
|
|
|
$this.trigger("close");
|
|
|
container.hide();
|
|
|
remove && container.remove();
|
|
|
})
|
|
|
container.removeClass("weui-popup__container--visible")
|
|
|
};
|
|
|
|
|
|
|
|
|
$(document).on("click", ".close-popup, .weui-popup__overlay", function() {
|
|
|
$.closePopup();
|
|
|
})
|
|
|
.on("click", ".open-popup", function() {
|
|
|
$($(this).data("target")).popup();
|
|
|
})
|
|
|
.on("click", ".weui-popup__container", function(e) {
|
|
|
if($(e.target).hasClass("weui-popup__container")) $.closePopup();
|
|
|
})
|
|
|
|
|
|
$.fn.popup = function() {
|
|
|
return this.each(function() {
|
|
|
$.openPopup(this);
|
|
|
});
|
|
|
};
|
|
|
|
|
|
}($);
|
|
|
|
|
|
/* ===============================================================================
|
|
|
************ Notification ************
|
|
|
=============================================================================== */
|
|
|
/* global $:true */
|
|
|
+function ($) {
|
|
|
"use strict";
|
|
|
|
|
|
var noti, defaults, timeout, start, diffX, diffY;
|
|
|
|
|
|
var touchStart = function(e) {
|
|
|
var p = $.getTouchPosition(e);
|
|
|
start = p;
|
|
|
diffX = diffY = 0;
|
|
|
noti.addClass("touching");
|
|
|
};
|
|
|
var touchMove = function(e) {
|
|
|
if(!start) return false;
|
|
|
e.preventDefault();
|
|
|
e.stopPropagation();
|
|
|
var p = $.getTouchPosition(e);
|
|
|
diffX = p.x - start.x;
|
|
|
diffY = p.y - start.y;
|
|
|
if(diffY > 0) {
|
|
|
diffY = Math.sqrt(diffY);
|
|
|
}
|
|
|
|
|
|
noti.css("transform", "translate3d(0, "+diffY+"px, 0)");
|
|
|
};
|
|
|
var touchEnd = function() {
|
|
|
noti.removeClass("touching");
|
|
|
noti.attr("style", "");
|
|
|
if(diffY < 0 && (Math.abs(diffY) > noti.height()*0.38)) {
|
|
|
$.closeNotification();
|
|
|
}
|
|
|
|
|
|
if(Math.abs(diffX) <= 1 && Math.abs(diffY) <= 1) {
|
|
|
noti.trigger("noti-click");
|
|
|
}
|
|
|
|
|
|
start = false;
|
|
|
};
|
|
|
|
|
|
var attachEvents = function(el) {
|
|
|
el.on($.touchEvents.start, touchStart);
|
|
|
el.on($.touchEvents.move, touchMove);
|
|
|
el.on($.touchEvents.end, touchEnd);
|
|
|
};
|
|
|
|
|
|
$.notification = $.noti = function(params) {
|
|
|
params = $.extend({}, defaults, params);
|
|
|
noti = $(".weui-notification");
|
|
|
if(!noti[0]) { // create a new notification
|
|
|
noti = $('<div class="weui-notification"></div>').appendTo(document.body);
|
|
|
attachEvents(noti);
|
|
|
}
|
|
|
|
|
|
noti.off("noti-click"); //the click event is not correct sometime: it will trigger when user is draging.
|
|
|
if(params.onClick) noti.on("noti-click", function() {
|
|
|
params.onClick(params.data);
|
|
|
});
|
|
|
|
|
|
noti.html($.t7.compile(params.tpl)(params));
|
|
|
|
|
|
noti.show();
|
|
|
|
|
|
noti.addClass("weui-notification--in");
|
|
|
noti.data("params", params);
|
|
|
|
|
|
var startTimeout = function() {
|
|
|
if(timeout) {
|
|
|
clearTimeout(timeout);
|
|
|
timeout = null;
|
|
|
}
|
|
|
|
|
|
timeout = setTimeout(function() {
|
|
|
if(noti.hasClass("weui-notification--touching")) {
|
|
|
startTimeout();
|
|
|
} else {
|
|
|
$.closeNotification();
|
|
|
}
|
|
|
}, params.time);
|
|
|
};
|
|
|
|
|
|
startTimeout();
|
|
|
|
|
|
};
|
|
|
|
|
|
$.closeNotification = function() {
|
|
|
timeout && clearTimeout(timeout);
|
|
|
timeout = null;
|
|
|
var noti = $(".weui-notification").removeClass("weui-notification--in").transitionEnd(function() {
|
|
|
$(this).remove();
|
|
|
});
|
|
|
|
|
|
if(noti[0]) {
|
|
|
var params = $(".weui-notification").data("params");
|
|
|
if(params && params.onClose) {
|
|
|
params.onClose(params.data);
|
|
|
}
|
|
|
}
|
|
|
};
|
|
|
|
|
|
defaults = $.noti.prototype.defaults = {
|
|
|
title: undefined,
|
|
|
text: undefined,
|
|
|
media: undefined,
|
|
|
time: 4000,
|
|
|
onClick: undefined,
|
|
|
onClose: undefined,
|
|
|
data: undefined,
|
|
|
tpl: '<div class="weui-notification__inner">' +
|
|
|
'{{#if media}}<div class="weui-notification__media">{{media}}</div>{{/if}}' +
|
|
|
'<div class="weui-notification__content">' +
|
|
|
'{{#if title}}<div class="weui-notification__title">{{title}}</div>{{/if}}' +
|
|
|
'{{#if text}}<div class="weui-notification__text">{{text}}</div>{{/if}}' +
|
|
|
'</div>' +
|
|
|
'<div class="weui-notification__handle-bar"></div>' +
|
|
|
'</div>'
|
|
|
};
|
|
|
|
|
|
}($);
|
|
|
|
|
|
+ function($) {
|
|
|
"use strict";
|
|
|
|
|
|
var timeout;
|
|
|
|
|
|
$.toptip = function(text, duration, type) {
|
|
|
if(!text) return;
|
|
|
if(typeof duration === typeof "a") {
|
|
|
type = duration;
|
|
|
duration = undefined;
|
|
|
}
|
|
|
duration = duration || 3000;
|
|
|
var className = type ? 'bg-' + type : 'bg-danger';
|
|
|
var $t = $('.weui-toptips').remove();
|
|
|
$t = $('<div class="weui-toptips"></div>').appendTo(document.body);
|
|
|
$t.html(text);
|
|
|
$t[0].className = 'weui-toptips ' + className
|
|
|
|
|
|
clearTimeout(timeout);
|
|
|
|
|
|
if(!$t.hasClass('weui-toptips_visible')) {
|
|
|
$t.show().width();
|
|
|
$t.addClass('weui-toptips_visible');
|
|
|
}
|
|
|
|
|
|
timeout = setTimeout(function() {
|
|
|
$t.removeClass('weui-toptips_visible').transitionEnd(function() {
|
|
|
$t.remove();
|
|
|
});
|
|
|
}, duration);
|
|
|
}
|
|
|
}($);
|
|
|
|
|
|
/* global $:true */
|
|
|
+ function($) {
|
|
|
"use strict";
|
|
|
var Slider = function (container, arg) {
|
|
|
this.container = $(container);
|
|
|
this.handler = this.container.find('.weui-slider__handler')
|
|
|
this.track = this.container.find('.weui-slider__track')
|
|
|
this.value = this.container.find('.weui-slider-box__value')
|
|
|
this.bind()
|
|
|
if (typeof arg === 'function') {
|
|
|
this.callback = arg
|
|
|
}
|
|
|
}
|
|
|
|
|
|
Slider.prototype.bind = function () {
|
|
|
this.container
|
|
|
.on($.touchEvents.start, $.proxy(this.touchStart, this))
|
|
|
.on($.touchEvents.end, $.proxy(this.touchEnd, this));
|
|
|
$(document.body).on($.touchEvents.move, $.proxy(this.touchMove, this)) // move even outside container
|
|
|
}
|
|
|
|
|
|
Slider.prototype.touchStart = function (e) {
|
|
|
e.preventDefault()
|
|
|
this.start = $.getTouchPosition(e)
|
|
|
this.width = this.container.find('.weui-slider__inner').width()
|
|
|
this.left = parseInt(this.container.find('.weui-slider__handler').css('left'))
|
|
|
this.touching = true
|
|
|
}
|
|
|
|
|
|
Slider.prototype.touchMove = function (e) {
|
|
|
if (!this.touching) return true
|
|
|
var p = $.getTouchPosition(e)
|
|
|
var distance = p.x - this.start.x
|
|
|
var left = distance + this.left
|
|
|
var per = parseInt(left / this.width * 100)
|
|
|
if (per < 0) per = 0
|
|
|
if (per > 100) per = 100
|
|
|
this.handler.css('left', per + '%')
|
|
|
this.track.css('width', per + '%')
|
|
|
this.value.text(per)
|
|
|
this.callback && this.callback.call(this, per)
|
|
|
this.container.trigger('change', per)
|
|
|
}
|
|
|
|
|
|
Slider.prototype.touchEnd = function (e) {
|
|
|
this.touching = false
|
|
|
}
|
|
|
|
|
|
$.fn.slider = function (arg) {
|
|
|
this.each(function () {
|
|
|
var $this = $(this)
|
|
|
var slider = $this.data('slider')
|
|
|
if (slider) return slider;
|
|
|
else $this.data('slider', new Slider(this, arg))
|
|
|
})
|
|
|
}
|
|
|
}($);
|
|
|
|
|
|
/* ===============================================================================
|
|
|
************ Swipeout ************
|
|
|
=============================================================================== */
|
|
|
/* global $:true */
|
|
|
|
|
|
+function ($) {
|
|
|
"use strict";
|
|
|
|
|
|
var cache = [];
|
|
|
var TOUCHING = 'swipeout-touching'
|
|
|
|
|
|
var Swipeout = function(el) {
|
|
|
this.container = $(el);
|
|
|
this.mover = this.container.find('>.weui-cell__bd')
|
|
|
this.attachEvents();
|
|
|
cache.push(this)
|
|
|
}
|
|
|
|
|
|
Swipeout.prototype.touchStart = function(e) {
|
|
|
var p = $.getTouchPosition(e);
|
|
|
this.container.addClass(TOUCHING);
|
|
|
this.start = p;
|
|
|
this.startX = 0;
|
|
|
this.startTime = + new Date;
|
|
|
var transform = this.mover.css('transform').match(/-?[\d\.]+/g)
|
|
|
if (transform && transform.length) this.startX = parseInt(transform[4])
|
|
|
this.diffX = this.diffY = 0;
|
|
|
this._closeOthers()
|
|
|
this.limit = this.container.find('>.weui-cell__ft').width() || 68; // 因为有的时候初始化的时候元素是隐藏的(比如在对话框内),所以在touchstart的时候计算宽度而不是初始化的时候
|
|
|
};
|
|
|
|
|
|
Swipeout.prototype.touchMove= function(e) {
|
|
|
if(!this.start) return true;
|
|
|
var p = $.getTouchPosition(e);
|
|
|
this.diffX = p.x - this.start.x;
|
|
|
this.diffY = p.y - this.start.y;
|
|
|
if (Math.abs(this.diffX) < Math.abs(this.diffY)) { // 说明是上下方向在拖动
|
|
|
this.close()
|
|
|
this.start = false
|
|
|
return true;
|
|
|
}
|
|
|
e.preventDefault();
|
|
|
e.stopPropagation();
|
|
|
var x = this.diffX + this.startX
|
|
|
if (x > 0) x = 0;
|
|
|
if (Math.abs(x) > this.limit) x = - (Math.pow(-(x+this.limit), .7) + this.limit)
|
|
|
this.mover.css("transform", "translate3d("+x+"px, 0, 0)");
|
|
|
};
|
|
|
Swipeout.prototype.touchEnd = function() {
|
|
|
if (!this.start) return true;
|
|
|
this.start = false;
|
|
|
var x = this.diffX + this.startX
|
|
|
var t = new Date - this.startTime;
|
|
|
if (this.diffX < -5 && t < 200) { // 向左快速滑动,则打开
|
|
|
this.open()
|
|
|
} else if (this.diffX >= 0 && t < 200) { // 向右快速滑动,或者单击,则关闭
|
|
|
this.close()
|
|
|
} else if (x > 0 || -x <= this.limit / 2) {
|
|
|
this.close()
|
|
|
} else {
|
|
|
this.open()
|
|
|
}
|
|
|
};
|
|
|
|
|
|
|
|
|
Swipeout.prototype.close = function() {
|
|
|
this.container.removeClass(TOUCHING);
|
|
|
this.mover.css("transform", "translate3d(0, 0, 0)");
|
|
|
this.container.trigger('swipeout-close');
|
|
|
}
|
|
|
|
|
|
Swipeout.prototype.open = function() {
|
|
|
this.container.removeClass(TOUCHING);
|
|
|
this._closeOthers()
|
|
|
this.mover.css("transform", "translate3d(" + (-this.limit) + "px, 0, 0)");
|
|
|
this.container.trigger('swipeout-open');
|
|
|
}
|
|
|
|
|
|
Swipeout.prototype.attachEvents = function() {
|
|
|
var el = this.mover;
|
|
|
el.on($.touchEvents.start, $.proxy(this.touchStart, this));
|
|
|
el.on($.touchEvents.move, $.proxy(this.touchMove, this));
|
|
|
el.on($.touchEvents.end, $.proxy(this.touchEnd, this));
|
|
|
}
|
|
|
Swipeout.prototype._closeOthers = function() {
|
|
|
//close others
|
|
|
var self = this
|
|
|
cache.forEach(function (s) {
|
|
|
if (s !== self) s.close()
|
|
|
})
|
|
|
}
|
|
|
|
|
|
var swipeout = function(el) {
|
|
|
return new Swipeout(el);
|
|
|
};
|
|
|
|
|
|
$.fn.swipeout = function (arg) {
|
|
|
return this.each(function() {
|
|
|
var $this = $(this)
|
|
|
var s = $this.data('swipeout') || swipeout(this);
|
|
|
$this.data('swipeout', s);
|
|
|
|
|
|
if (typeof arg === typeof 'a') {
|
|
|
s[arg]()
|
|
|
}
|
|
|
});
|
|
|
}
|
|
|
|
|
|
$('.weui-cell_swiped').swipeout() // auto init
|
|
|
}($);
|