mirror of
https://github.com/opensourcepos/opensourcepos.git
synced 2026-01-21 07:48:06 -05:00
50025 lines
1.5 MiB
50025 lines
1.5 MiB
/*!
|
|
* jQuery JavaScript Library v1.12.3
|
|
* http://jquery.com/
|
|
*
|
|
* Includes Sizzle.js
|
|
* http://sizzlejs.com/
|
|
*
|
|
* Copyright jQuery Foundation and other contributors
|
|
* Released under the MIT license
|
|
* http://jquery.org/license
|
|
*
|
|
* Date: 2016-04-05T19:16Z
|
|
*/
|
|
|
|
(function( global, factory ) {
|
|
|
|
if ( typeof module === "object" && typeof module.exports === "object" ) {
|
|
// For CommonJS and CommonJS-like environments where a proper `window`
|
|
// is present, execute the factory and get jQuery.
|
|
// For environments that do not have a `window` with a `document`
|
|
// (such as Node.js), expose a factory as module.exports.
|
|
// This accentuates the need for the creation of a real `window`.
|
|
// e.g. var jQuery = require("jquery")(window);
|
|
// See ticket #14549 for more info.
|
|
module.exports = global.document ?
|
|
factory( global, true ) :
|
|
function( w ) {
|
|
if ( !w.document ) {
|
|
throw new Error( "jQuery requires a window with a document" );
|
|
}
|
|
return factory( w );
|
|
};
|
|
} else {
|
|
factory( global );
|
|
}
|
|
|
|
// Pass this if window is not defined yet
|
|
}(typeof window !== "undefined" ? window : this, function( window, noGlobal ) {
|
|
|
|
// Support: Firefox 18+
|
|
// Can't be in strict mode, several libs including ASP.NET trace
|
|
// the stack via arguments.caller.callee and Firefox dies if
|
|
// you try to trace through "use strict" call chains. (#13335)
|
|
//"use strict";
|
|
var deletedIds = [];
|
|
|
|
var document = window.document;
|
|
|
|
var slice = deletedIds.slice;
|
|
|
|
var concat = deletedIds.concat;
|
|
|
|
var push = deletedIds.push;
|
|
|
|
var indexOf = deletedIds.indexOf;
|
|
|
|
var class2type = {};
|
|
|
|
var toString = class2type.toString;
|
|
|
|
var hasOwn = class2type.hasOwnProperty;
|
|
|
|
var support = {};
|
|
|
|
|
|
|
|
var
|
|
version = "1.12.3",
|
|
|
|
// Define a local copy of jQuery
|
|
jQuery = function( selector, context ) {
|
|
|
|
// The jQuery object is actually just the init constructor 'enhanced'
|
|
// Need init if jQuery is called (just allow error to be thrown if not included)
|
|
return new jQuery.fn.init( selector, context );
|
|
},
|
|
|
|
// Support: Android<4.1, IE<9
|
|
// Make sure we trim BOM and NBSP
|
|
rtrim = /^[\s\uFEFF\xA0]+|[\s\uFEFF\xA0]+$/g,
|
|
|
|
// Matches dashed string for camelizing
|
|
rmsPrefix = /^-ms-/,
|
|
rdashAlpha = /-([\da-z])/gi,
|
|
|
|
// Used by jQuery.camelCase as callback to replace()
|
|
fcamelCase = function( all, letter ) {
|
|
return letter.toUpperCase();
|
|
};
|
|
|
|
jQuery.fn = jQuery.prototype = {
|
|
|
|
// The current version of jQuery being used
|
|
jquery: version,
|
|
|
|
constructor: jQuery,
|
|
|
|
// Start with an empty selector
|
|
selector: "",
|
|
|
|
// The default length of a jQuery object is 0
|
|
length: 0,
|
|
|
|
toArray: function() {
|
|
return slice.call( this );
|
|
},
|
|
|
|
// Get the Nth element in the matched element set OR
|
|
// Get the whole matched element set as a clean array
|
|
get: function( num ) {
|
|
return num != null ?
|
|
|
|
// Return just the one element from the set
|
|
( num < 0 ? this[ num + this.length ] : this[ num ] ) :
|
|
|
|
// Return all the elements in a clean array
|
|
slice.call( this );
|
|
},
|
|
|
|
// Take an array of elements and push it onto the stack
|
|
// (returning the new matched element set)
|
|
pushStack: function( elems ) {
|
|
|
|
// Build a new jQuery matched element set
|
|
var ret = jQuery.merge( this.constructor(), elems );
|
|
|
|
// Add the old object onto the stack (as a reference)
|
|
ret.prevObject = this;
|
|
ret.context = this.context;
|
|
|
|
// Return the newly-formed element set
|
|
return ret;
|
|
},
|
|
|
|
// Execute a callback for every element in the matched set.
|
|
each: function( callback ) {
|
|
return jQuery.each( this, callback );
|
|
},
|
|
|
|
map: function( callback ) {
|
|
return this.pushStack( jQuery.map( this, function( elem, i ) {
|
|
return callback.call( elem, i, elem );
|
|
} ) );
|
|
},
|
|
|
|
slice: function() {
|
|
return this.pushStack( slice.apply( this, arguments ) );
|
|
},
|
|
|
|
first: function() {
|
|
return this.eq( 0 );
|
|
},
|
|
|
|
last: function() {
|
|
return this.eq( -1 );
|
|
},
|
|
|
|
eq: function( i ) {
|
|
var len = this.length,
|
|
j = +i + ( i < 0 ? len : 0 );
|
|
return this.pushStack( j >= 0 && j < len ? [ this[ j ] ] : [] );
|
|
},
|
|
|
|
end: function() {
|
|
return this.prevObject || this.constructor();
|
|
},
|
|
|
|
// For internal use only.
|
|
// Behaves like an Array's method, not like a jQuery method.
|
|
push: push,
|
|
sort: deletedIds.sort,
|
|
splice: deletedIds.splice
|
|
};
|
|
|
|
jQuery.extend = jQuery.fn.extend = function() {
|
|
var src, copyIsArray, copy, name, options, clone,
|
|
target = arguments[ 0 ] || {},
|
|
i = 1,
|
|
length = arguments.length,
|
|
deep = false;
|
|
|
|
// Handle a deep copy situation
|
|
if ( typeof target === "boolean" ) {
|
|
deep = target;
|
|
|
|
// skip the boolean and the target
|
|
target = arguments[ i ] || {};
|
|
i++;
|
|
}
|
|
|
|
// Handle case when target is a string or something (possible in deep copy)
|
|
if ( typeof target !== "object" && !jQuery.isFunction( target ) ) {
|
|
target = {};
|
|
}
|
|
|
|
// extend jQuery itself if only one argument is passed
|
|
if ( i === length ) {
|
|
target = this;
|
|
i--;
|
|
}
|
|
|
|
for ( ; i < length; i++ ) {
|
|
|
|
// Only deal with non-null/undefined values
|
|
if ( ( options = arguments[ i ] ) != null ) {
|
|
|
|
// Extend the base object
|
|
for ( name in options ) {
|
|
src = target[ name ];
|
|
copy = options[ name ];
|
|
|
|
// Prevent never-ending loop
|
|
if ( target === copy ) {
|
|
continue;
|
|
}
|
|
|
|
// Recurse if we're merging plain objects or arrays
|
|
if ( deep && copy && ( jQuery.isPlainObject( copy ) ||
|
|
( copyIsArray = jQuery.isArray( copy ) ) ) ) {
|
|
|
|
if ( copyIsArray ) {
|
|
copyIsArray = false;
|
|
clone = src && jQuery.isArray( src ) ? src : [];
|
|
|
|
} else {
|
|
clone = src && jQuery.isPlainObject( src ) ? src : {};
|
|
}
|
|
|
|
// Never move original objects, clone them
|
|
target[ name ] = jQuery.extend( deep, clone, copy );
|
|
|
|
// Don't bring in undefined values
|
|
} else if ( copy !== undefined ) {
|
|
target[ name ] = copy;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// Return the modified object
|
|
return target;
|
|
};
|
|
|
|
jQuery.extend( {
|
|
|
|
// Unique for each copy of jQuery on the page
|
|
expando: "jQuery" + ( version + Math.random() ).replace( /\D/g, "" ),
|
|
|
|
// Assume jQuery is ready without the ready module
|
|
isReady: true,
|
|
|
|
error: function( msg ) {
|
|
throw new Error( msg );
|
|
},
|
|
|
|
noop: function() {},
|
|
|
|
// See test/unit/core.js for details concerning isFunction.
|
|
// Since version 1.3, DOM methods and functions like alert
|
|
// aren't supported. They return false on IE (#2968).
|
|
isFunction: function( obj ) {
|
|
return jQuery.type( obj ) === "function";
|
|
},
|
|
|
|
isArray: Array.isArray || function( obj ) {
|
|
return jQuery.type( obj ) === "array";
|
|
},
|
|
|
|
isWindow: function( obj ) {
|
|
/* jshint eqeqeq: false */
|
|
return obj != null && obj == obj.window;
|
|
},
|
|
|
|
isNumeric: function( obj ) {
|
|
|
|
// parseFloat NaNs numeric-cast false positives (null|true|false|"")
|
|
// ...but misinterprets leading-number strings, particularly hex literals ("0x...")
|
|
// subtraction forces infinities to NaN
|
|
// adding 1 corrects loss of precision from parseFloat (#15100)
|
|
var realStringObj = obj && obj.toString();
|
|
return !jQuery.isArray( obj ) && ( realStringObj - parseFloat( realStringObj ) + 1 ) >= 0;
|
|
},
|
|
|
|
isEmptyObject: function( obj ) {
|
|
var name;
|
|
for ( name in obj ) {
|
|
return false;
|
|
}
|
|
return true;
|
|
},
|
|
|
|
isPlainObject: function( obj ) {
|
|
var key;
|
|
|
|
// Must be an Object.
|
|
// Because of IE, we also have to check the presence of the constructor property.
|
|
// Make sure that DOM nodes and window objects don't pass through, as well
|
|
if ( !obj || jQuery.type( obj ) !== "object" || obj.nodeType || jQuery.isWindow( obj ) ) {
|
|
return false;
|
|
}
|
|
|
|
try {
|
|
|
|
// Not own constructor property must be Object
|
|
if ( obj.constructor &&
|
|
!hasOwn.call( obj, "constructor" ) &&
|
|
!hasOwn.call( obj.constructor.prototype, "isPrototypeOf" ) ) {
|
|
return false;
|
|
}
|
|
} catch ( e ) {
|
|
|
|
// IE8,9 Will throw exceptions on certain host objects #9897
|
|
return false;
|
|
}
|
|
|
|
// Support: IE<9
|
|
// Handle iteration over inherited properties before own properties.
|
|
if ( !support.ownFirst ) {
|
|
for ( key in obj ) {
|
|
return hasOwn.call( obj, key );
|
|
}
|
|
}
|
|
|
|
// Own properties are enumerated firstly, so to speed up,
|
|
// if last one is own, then all properties are own.
|
|
for ( key in obj ) {}
|
|
|
|
return key === undefined || hasOwn.call( obj, key );
|
|
},
|
|
|
|
type: function( obj ) {
|
|
if ( obj == null ) {
|
|
return obj + "";
|
|
}
|
|
return typeof obj === "object" || typeof obj === "function" ?
|
|
class2type[ toString.call( obj ) ] || "object" :
|
|
typeof obj;
|
|
},
|
|
|
|
// Workarounds based on findings by Jim Driscoll
|
|
// http://weblogs.java.net/blog/driscoll/archive/2009/09/08/eval-javascript-global-context
|
|
globalEval: function( data ) {
|
|
if ( data && jQuery.trim( data ) ) {
|
|
|
|
// We use execScript on Internet Explorer
|
|
// We use an anonymous function so that context is window
|
|
// rather than jQuery in Firefox
|
|
( window.execScript || function( data ) {
|
|
window[ "eval" ].call( window, data ); // jscs:ignore requireDotNotation
|
|
} )( data );
|
|
}
|
|
},
|
|
|
|
// Convert dashed to camelCase; used by the css and data modules
|
|
// Microsoft forgot to hump their vendor prefix (#9572)
|
|
camelCase: function( string ) {
|
|
return string.replace( rmsPrefix, "ms-" ).replace( rdashAlpha, fcamelCase );
|
|
},
|
|
|
|
nodeName: function( elem, name ) {
|
|
return elem.nodeName && elem.nodeName.toLowerCase() === name.toLowerCase();
|
|
},
|
|
|
|
each: function( obj, callback ) {
|
|
var length, i = 0;
|
|
|
|
if ( isArrayLike( obj ) ) {
|
|
length = obj.length;
|
|
for ( ; i < length; i++ ) {
|
|
if ( callback.call( obj[ i ], i, obj[ i ] ) === false ) {
|
|
break;
|
|
}
|
|
}
|
|
} else {
|
|
for ( i in obj ) {
|
|
if ( callback.call( obj[ i ], i, obj[ i ] ) === false ) {
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
return obj;
|
|
},
|
|
|
|
// Support: Android<4.1, IE<9
|
|
trim: function( text ) {
|
|
return text == null ?
|
|
"" :
|
|
( text + "" ).replace( rtrim, "" );
|
|
},
|
|
|
|
// results is for internal usage only
|
|
makeArray: function( arr, results ) {
|
|
var ret = results || [];
|
|
|
|
if ( arr != null ) {
|
|
if ( isArrayLike( Object( arr ) ) ) {
|
|
jQuery.merge( ret,
|
|
typeof arr === "string" ?
|
|
[ arr ] : arr
|
|
);
|
|
} else {
|
|
push.call( ret, arr );
|
|
}
|
|
}
|
|
|
|
return ret;
|
|
},
|
|
|
|
inArray: function( elem, arr, i ) {
|
|
var len;
|
|
|
|
if ( arr ) {
|
|
if ( indexOf ) {
|
|
return indexOf.call( arr, elem, i );
|
|
}
|
|
|
|
len = arr.length;
|
|
i = i ? i < 0 ? Math.max( 0, len + i ) : i : 0;
|
|
|
|
for ( ; i < len; i++ ) {
|
|
|
|
// Skip accessing in sparse arrays
|
|
if ( i in arr && arr[ i ] === elem ) {
|
|
return i;
|
|
}
|
|
}
|
|
}
|
|
|
|
return -1;
|
|
},
|
|
|
|
merge: function( first, second ) {
|
|
var len = +second.length,
|
|
j = 0,
|
|
i = first.length;
|
|
|
|
while ( j < len ) {
|
|
first[ i++ ] = second[ j++ ];
|
|
}
|
|
|
|
// Support: IE<9
|
|
// Workaround casting of .length to NaN on otherwise arraylike objects (e.g., NodeLists)
|
|
if ( len !== len ) {
|
|
while ( second[ j ] !== undefined ) {
|
|
first[ i++ ] = second[ j++ ];
|
|
}
|
|
}
|
|
|
|
first.length = i;
|
|
|
|
return first;
|
|
},
|
|
|
|
grep: function( elems, callback, invert ) {
|
|
var callbackInverse,
|
|
matches = [],
|
|
i = 0,
|
|
length = elems.length,
|
|
callbackExpect = !invert;
|
|
|
|
// Go through the array, only saving the items
|
|
// that pass the validator function
|
|
for ( ; i < length; i++ ) {
|
|
callbackInverse = !callback( elems[ i ], i );
|
|
if ( callbackInverse !== callbackExpect ) {
|
|
matches.push( elems[ i ] );
|
|
}
|
|
}
|
|
|
|
return matches;
|
|
},
|
|
|
|
// arg is for internal usage only
|
|
map: function( elems, callback, arg ) {
|
|
var length, value,
|
|
i = 0,
|
|
ret = [];
|
|
|
|
// Go through the array, translating each of the items to their new values
|
|
if ( isArrayLike( elems ) ) {
|
|
length = elems.length;
|
|
for ( ; i < length; i++ ) {
|
|
value = callback( elems[ i ], i, arg );
|
|
|
|
if ( value != null ) {
|
|
ret.push( value );
|
|
}
|
|
}
|
|
|
|
// Go through every key on the object,
|
|
} else {
|
|
for ( i in elems ) {
|
|
value = callback( elems[ i ], i, arg );
|
|
|
|
if ( value != null ) {
|
|
ret.push( value );
|
|
}
|
|
}
|
|
}
|
|
|
|
// Flatten any nested arrays
|
|
return concat.apply( [], ret );
|
|
},
|
|
|
|
// A global GUID counter for objects
|
|
guid: 1,
|
|
|
|
// Bind a function to a context, optionally partially applying any
|
|
// arguments.
|
|
proxy: function( fn, context ) {
|
|
var args, proxy, tmp;
|
|
|
|
if ( typeof context === "string" ) {
|
|
tmp = fn[ context ];
|
|
context = fn;
|
|
fn = tmp;
|
|
}
|
|
|
|
// Quick check to determine if target is callable, in the spec
|
|
// this throws a TypeError, but we will just return undefined.
|
|
if ( !jQuery.isFunction( fn ) ) {
|
|
return undefined;
|
|
}
|
|
|
|
// Simulated bind
|
|
args = slice.call( arguments, 2 );
|
|
proxy = function() {
|
|
return fn.apply( context || this, args.concat( slice.call( arguments ) ) );
|
|
};
|
|
|
|
// Set the guid of unique handler to the same of original handler, so it can be removed
|
|
proxy.guid = fn.guid = fn.guid || jQuery.guid++;
|
|
|
|
return proxy;
|
|
},
|
|
|
|
now: function() {
|
|
return +( new Date() );
|
|
},
|
|
|
|
// jQuery.support is not used in Core but other projects attach their
|
|
// properties to it so it needs to exist.
|
|
support: support
|
|
} );
|
|
|
|
// JSHint would error on this code due to the Symbol not being defined in ES5.
|
|
// Defining this global in .jshintrc would create a danger of using the global
|
|
// unguarded in another place, it seems safer to just disable JSHint for these
|
|
// three lines.
|
|
/* jshint ignore: start */
|
|
if ( typeof Symbol === "function" ) {
|
|
jQuery.fn[ Symbol.iterator ] = deletedIds[ Symbol.iterator ];
|
|
}
|
|
/* jshint ignore: end */
|
|
|
|
// Populate the class2type map
|
|
jQuery.each( "Boolean Number String Function Array Date RegExp Object Error Symbol".split( " " ),
|
|
function( i, name ) {
|
|
class2type[ "[object " + name + "]" ] = name.toLowerCase();
|
|
} );
|
|
|
|
function isArrayLike( obj ) {
|
|
|
|
// Support: iOS 8.2 (not reproducible in simulator)
|
|
// `in` check used to prevent JIT error (gh-2145)
|
|
// hasOwn isn't used here due to false negatives
|
|
// regarding Nodelist length in IE
|
|
var length = !!obj && "length" in obj && obj.length,
|
|
type = jQuery.type( obj );
|
|
|
|
if ( type === "function" || jQuery.isWindow( obj ) ) {
|
|
return false;
|
|
}
|
|
|
|
return type === "array" || length === 0 ||
|
|
typeof length === "number" && length > 0 && ( length - 1 ) in obj;
|
|
}
|
|
var Sizzle =
|
|
/*!
|
|
* Sizzle CSS Selector Engine v2.2.1
|
|
* http://sizzlejs.com/
|
|
*
|
|
* Copyright jQuery Foundation and other contributors
|
|
* Released under the MIT license
|
|
* http://jquery.org/license
|
|
*
|
|
* Date: 2015-10-17
|
|
*/
|
|
(function( window ) {
|
|
|
|
var i,
|
|
support,
|
|
Expr,
|
|
getText,
|
|
isXML,
|
|
tokenize,
|
|
compile,
|
|
select,
|
|
outermostContext,
|
|
sortInput,
|
|
hasDuplicate,
|
|
|
|
// Local document vars
|
|
setDocument,
|
|
document,
|
|
docElem,
|
|
documentIsHTML,
|
|
rbuggyQSA,
|
|
rbuggyMatches,
|
|
matches,
|
|
contains,
|
|
|
|
// Instance-specific data
|
|
expando = "sizzle" + 1 * new Date(),
|
|
preferredDoc = window.document,
|
|
dirruns = 0,
|
|
done = 0,
|
|
classCache = createCache(),
|
|
tokenCache = createCache(),
|
|
compilerCache = createCache(),
|
|
sortOrder = function( a, b ) {
|
|
if ( a === b ) {
|
|
hasDuplicate = true;
|
|
}
|
|
return 0;
|
|
},
|
|
|
|
// General-purpose constants
|
|
MAX_NEGATIVE = 1 << 31,
|
|
|
|
// Instance methods
|
|
hasOwn = ({}).hasOwnProperty,
|
|
arr = [],
|
|
pop = arr.pop,
|
|
push_native = arr.push,
|
|
push = arr.push,
|
|
slice = arr.slice,
|
|
// Use a stripped-down indexOf as it's faster than native
|
|
// http://jsperf.com/thor-indexof-vs-for/5
|
|
indexOf = function( list, elem ) {
|
|
var i = 0,
|
|
len = list.length;
|
|
for ( ; i < len; i++ ) {
|
|
if ( list[i] === elem ) {
|
|
return i;
|
|
}
|
|
}
|
|
return -1;
|
|
},
|
|
|
|
booleans = "checked|selected|async|autofocus|autoplay|controls|defer|disabled|hidden|ismap|loop|multiple|open|readonly|required|scoped",
|
|
|
|
// Regular expressions
|
|
|
|
// http://www.w3.org/TR/css3-selectors/#whitespace
|
|
whitespace = "[\\x20\\t\\r\\n\\f]",
|
|
|
|
// http://www.w3.org/TR/CSS21/syndata.html#value-def-identifier
|
|
identifier = "(?:\\\\.|[\\w-]|[^\\x00-\\xa0])+",
|
|
|
|
// Attribute selectors: http://www.w3.org/TR/selectors/#attribute-selectors
|
|
attributes = "\\[" + whitespace + "*(" + identifier + ")(?:" + whitespace +
|
|
// Operator (capture 2)
|
|
"*([*^$|!~]?=)" + whitespace +
|
|
// "Attribute values must be CSS identifiers [capture 5] or strings [capture 3 or capture 4]"
|
|
"*(?:'((?:\\\\.|[^\\\\'])*)'|\"((?:\\\\.|[^\\\\\"])*)\"|(" + identifier + "))|)" + whitespace +
|
|
"*\\]",
|
|
|
|
pseudos = ":(" + identifier + ")(?:\\((" +
|
|
// To reduce the number of selectors needing tokenize in the preFilter, prefer arguments:
|
|
// 1. quoted (capture 3; capture 4 or capture 5)
|
|
"('((?:\\\\.|[^\\\\'])*)'|\"((?:\\\\.|[^\\\\\"])*)\")|" +
|
|
// 2. simple (capture 6)
|
|
"((?:\\\\.|[^\\\\()[\\]]|" + attributes + ")*)|" +
|
|
// 3. anything else (capture 2)
|
|
".*" +
|
|
")\\)|)",
|
|
|
|
// Leading and non-escaped trailing whitespace, capturing some non-whitespace characters preceding the latter
|
|
rwhitespace = new RegExp( whitespace + "+", "g" ),
|
|
rtrim = new RegExp( "^" + whitespace + "+|((?:^|[^\\\\])(?:\\\\.)*)" + whitespace + "+$", "g" ),
|
|
|
|
rcomma = new RegExp( "^" + whitespace + "*," + whitespace + "*" ),
|
|
rcombinators = new RegExp( "^" + whitespace + "*([>+~]|" + whitespace + ")" + whitespace + "*" ),
|
|
|
|
rattributeQuotes = new RegExp( "=" + whitespace + "*([^\\]'\"]*?)" + whitespace + "*\\]", "g" ),
|
|
|
|
rpseudo = new RegExp( pseudos ),
|
|
ridentifier = new RegExp( "^" + identifier + "$" ),
|
|
|
|
matchExpr = {
|
|
"ID": new RegExp( "^#(" + identifier + ")" ),
|
|
"CLASS": new RegExp( "^\\.(" + identifier + ")" ),
|
|
"TAG": new RegExp( "^(" + identifier + "|[*])" ),
|
|
"ATTR": new RegExp( "^" + attributes ),
|
|
"PSEUDO": new RegExp( "^" + pseudos ),
|
|
"CHILD": new RegExp( "^:(only|first|last|nth|nth-last)-(child|of-type)(?:\\(" + whitespace +
|
|
"*(even|odd|(([+-]|)(\\d*)n|)" + whitespace + "*(?:([+-]|)" + whitespace +
|
|
"*(\\d+)|))" + whitespace + "*\\)|)", "i" ),
|
|
"bool": new RegExp( "^(?:" + booleans + ")$", "i" ),
|
|
// For use in libraries implementing .is()
|
|
// We use this for POS matching in `select`
|
|
"needsContext": new RegExp( "^" + whitespace + "*[>+~]|:(even|odd|eq|gt|lt|nth|first|last)(?:\\(" +
|
|
whitespace + "*((?:-\\d)?\\d*)" + whitespace + "*\\)|)(?=[^-]|$)", "i" )
|
|
},
|
|
|
|
rinputs = /^(?:input|select|textarea|button)$/i,
|
|
rheader = /^h\d$/i,
|
|
|
|
rnative = /^[^{]+\{\s*\[native \w/,
|
|
|
|
// Easily-parseable/retrievable ID or TAG or CLASS selectors
|
|
rquickExpr = /^(?:#([\w-]+)|(\w+)|\.([\w-]+))$/,
|
|
|
|
rsibling = /[+~]/,
|
|
rescape = /'|\\/g,
|
|
|
|
// CSS escapes http://www.w3.org/TR/CSS21/syndata.html#escaped-characters
|
|
runescape = new RegExp( "\\\\([\\da-f]{1,6}" + whitespace + "?|(" + whitespace + ")|.)", "ig" ),
|
|
funescape = function( _, escaped, escapedWhitespace ) {
|
|
var high = "0x" + escaped - 0x10000;
|
|
// NaN means non-codepoint
|
|
// Support: Firefox<24
|
|
// Workaround erroneous numeric interpretation of +"0x"
|
|
return high !== high || escapedWhitespace ?
|
|
escaped :
|
|
high < 0 ?
|
|
// BMP codepoint
|
|
String.fromCharCode( high + 0x10000 ) :
|
|
// Supplemental Plane codepoint (surrogate pair)
|
|
String.fromCharCode( high >> 10 | 0xD800, high & 0x3FF | 0xDC00 );
|
|
},
|
|
|
|
// Used for iframes
|
|
// See setDocument()
|
|
// Removing the function wrapper causes a "Permission Denied"
|
|
// error in IE
|
|
unloadHandler = function() {
|
|
setDocument();
|
|
};
|
|
|
|
// Optimize for push.apply( _, NodeList )
|
|
try {
|
|
push.apply(
|
|
(arr = slice.call( preferredDoc.childNodes )),
|
|
preferredDoc.childNodes
|
|
);
|
|
// Support: Android<4.0
|
|
// Detect silently failing push.apply
|
|
arr[ preferredDoc.childNodes.length ].nodeType;
|
|
} catch ( e ) {
|
|
push = { apply: arr.length ?
|
|
|
|
// Leverage slice if possible
|
|
function( target, els ) {
|
|
push_native.apply( target, slice.call(els) );
|
|
} :
|
|
|
|
// Support: IE<9
|
|
// Otherwise append directly
|
|
function( target, els ) {
|
|
var j = target.length,
|
|
i = 0;
|
|
// Can't trust NodeList.length
|
|
while ( (target[j++] = els[i++]) ) {}
|
|
target.length = j - 1;
|
|
}
|
|
};
|
|
}
|
|
|
|
function Sizzle( selector, context, results, seed ) {
|
|
var m, i, elem, nid, nidselect, match, groups, newSelector,
|
|
newContext = context && context.ownerDocument,
|
|
|
|
// nodeType defaults to 9, since context defaults to document
|
|
nodeType = context ? context.nodeType : 9;
|
|
|
|
results = results || [];
|
|
|
|
// Return early from calls with invalid selector or context
|
|
if ( typeof selector !== "string" || !selector ||
|
|
nodeType !== 1 && nodeType !== 9 && nodeType !== 11 ) {
|
|
|
|
return results;
|
|
}
|
|
|
|
// Try to shortcut find operations (as opposed to filters) in HTML documents
|
|
if ( !seed ) {
|
|
|
|
if ( ( context ? context.ownerDocument || context : preferredDoc ) !== document ) {
|
|
setDocument( context );
|
|
}
|
|
context = context || document;
|
|
|
|
if ( documentIsHTML ) {
|
|
|
|
// If the selector is sufficiently simple, try using a "get*By*" DOM method
|
|
// (excepting DocumentFragment context, where the methods don't exist)
|
|
if ( nodeType !== 11 && (match = rquickExpr.exec( selector )) ) {
|
|
|
|
// ID selector
|
|
if ( (m = match[1]) ) {
|
|
|
|
// Document context
|
|
if ( nodeType === 9 ) {
|
|
if ( (elem = context.getElementById( m )) ) {
|
|
|
|
// Support: IE, Opera, Webkit
|
|
// TODO: identify versions
|
|
// getElementById can match elements by name instead of ID
|
|
if ( elem.id === m ) {
|
|
results.push( elem );
|
|
return results;
|
|
}
|
|
} else {
|
|
return results;
|
|
}
|
|
|
|
// Element context
|
|
} else {
|
|
|
|
// Support: IE, Opera, Webkit
|
|
// TODO: identify versions
|
|
// getElementById can match elements by name instead of ID
|
|
if ( newContext && (elem = newContext.getElementById( m )) &&
|
|
contains( context, elem ) &&
|
|
elem.id === m ) {
|
|
|
|
results.push( elem );
|
|
return results;
|
|
}
|
|
}
|
|
|
|
// Type selector
|
|
} else if ( match[2] ) {
|
|
push.apply( results, context.getElementsByTagName( selector ) );
|
|
return results;
|
|
|
|
// Class selector
|
|
} else if ( (m = match[3]) && support.getElementsByClassName &&
|
|
context.getElementsByClassName ) {
|
|
|
|
push.apply( results, context.getElementsByClassName( m ) );
|
|
return results;
|
|
}
|
|
}
|
|
|
|
// Take advantage of querySelectorAll
|
|
if ( support.qsa &&
|
|
!compilerCache[ selector + " " ] &&
|
|
(!rbuggyQSA || !rbuggyQSA.test( selector )) ) {
|
|
|
|
if ( nodeType !== 1 ) {
|
|
newContext = context;
|
|
newSelector = selector;
|
|
|
|
// qSA looks outside Element context, which is not what we want
|
|
// Thanks to Andrew Dupont for this workaround technique
|
|
// Support: IE <=8
|
|
// Exclude object elements
|
|
} else if ( context.nodeName.toLowerCase() !== "object" ) {
|
|
|
|
// Capture the context ID, setting it first if necessary
|
|
if ( (nid = context.getAttribute( "id" )) ) {
|
|
nid = nid.replace( rescape, "\\$&" );
|
|
} else {
|
|
context.setAttribute( "id", (nid = expando) );
|
|
}
|
|
|
|
// Prefix every selector in the list
|
|
groups = tokenize( selector );
|
|
i = groups.length;
|
|
nidselect = ridentifier.test( nid ) ? "#" + nid : "[id='" + nid + "']";
|
|
while ( i-- ) {
|
|
groups[i] = nidselect + " " + toSelector( groups[i] );
|
|
}
|
|
newSelector = groups.join( "," );
|
|
|
|
// Expand context for sibling selectors
|
|
newContext = rsibling.test( selector ) && testContext( context.parentNode ) ||
|
|
context;
|
|
}
|
|
|
|
if ( newSelector ) {
|
|
try {
|
|
push.apply( results,
|
|
newContext.querySelectorAll( newSelector )
|
|
);
|
|
return results;
|
|
} catch ( qsaError ) {
|
|
} finally {
|
|
if ( nid === expando ) {
|
|
context.removeAttribute( "id" );
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// All others
|
|
return select( selector.replace( rtrim, "$1" ), context, results, seed );
|
|
}
|
|
|
|
/**
|
|
* Create key-value caches of limited size
|
|
* @returns {function(string, object)} Returns the Object data after storing it on itself with
|
|
* property name the (space-suffixed) string and (if the cache is larger than Expr.cacheLength)
|
|
* deleting the oldest entry
|
|
*/
|
|
function createCache() {
|
|
var keys = [];
|
|
|
|
function cache( key, value ) {
|
|
// Use (key + " ") to avoid collision with native prototype properties (see Issue #157)
|
|
if ( keys.push( key + " " ) > Expr.cacheLength ) {
|
|
// Only keep the most recent entries
|
|
delete cache[ keys.shift() ];
|
|
}
|
|
return (cache[ key + " " ] = value);
|
|
}
|
|
return cache;
|
|
}
|
|
|
|
/**
|
|
* Mark a function for special use by Sizzle
|
|
* @param {Function} fn The function to mark
|
|
*/
|
|
function markFunction( fn ) {
|
|
fn[ expando ] = true;
|
|
return fn;
|
|
}
|
|
|
|
/**
|
|
* Support testing using an element
|
|
* @param {Function} fn Passed the created div and expects a boolean result
|
|
*/
|
|
function assert( fn ) {
|
|
var div = document.createElement("div");
|
|
|
|
try {
|
|
return !!fn( div );
|
|
} catch (e) {
|
|
return false;
|
|
} finally {
|
|
// Remove from its parent by default
|
|
if ( div.parentNode ) {
|
|
div.parentNode.removeChild( div );
|
|
}
|
|
// release memory in IE
|
|
div = null;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Adds the same handler for all of the specified attrs
|
|
* @param {String} attrs Pipe-separated list of attributes
|
|
* @param {Function} handler The method that will be applied
|
|
*/
|
|
function addHandle( attrs, handler ) {
|
|
var arr = attrs.split("|"),
|
|
i = arr.length;
|
|
|
|
while ( i-- ) {
|
|
Expr.attrHandle[ arr[i] ] = handler;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Checks document order of two siblings
|
|
* @param {Element} a
|
|
* @param {Element} b
|
|
* @returns {Number} Returns less than 0 if a precedes b, greater than 0 if a follows b
|
|
*/
|
|
function siblingCheck( a, b ) {
|
|
var cur = b && a,
|
|
diff = cur && a.nodeType === 1 && b.nodeType === 1 &&
|
|
( ~b.sourceIndex || MAX_NEGATIVE ) -
|
|
( ~a.sourceIndex || MAX_NEGATIVE );
|
|
|
|
// Use IE sourceIndex if available on both nodes
|
|
if ( diff ) {
|
|
return diff;
|
|
}
|
|
|
|
// Check if b follows a
|
|
if ( cur ) {
|
|
while ( (cur = cur.nextSibling) ) {
|
|
if ( cur === b ) {
|
|
return -1;
|
|
}
|
|
}
|
|
}
|
|
|
|
return a ? 1 : -1;
|
|
}
|
|
|
|
/**
|
|
* Returns a function to use in pseudos for input types
|
|
* @param {String} type
|
|
*/
|
|
function createInputPseudo( type ) {
|
|
return function( elem ) {
|
|
var name = elem.nodeName.toLowerCase();
|
|
return name === "input" && elem.type === type;
|
|
};
|
|
}
|
|
|
|
/**
|
|
* Returns a function to use in pseudos for buttons
|
|
* @param {String} type
|
|
*/
|
|
function createButtonPseudo( type ) {
|
|
return function( elem ) {
|
|
var name = elem.nodeName.toLowerCase();
|
|
return (name === "input" || name === "button") && elem.type === type;
|
|
};
|
|
}
|
|
|
|
/**
|
|
* Returns a function to use in pseudos for positionals
|
|
* @param {Function} fn
|
|
*/
|
|
function createPositionalPseudo( fn ) {
|
|
return markFunction(function( argument ) {
|
|
argument = +argument;
|
|
return markFunction(function( seed, matches ) {
|
|
var j,
|
|
matchIndexes = fn( [], seed.length, argument ),
|
|
i = matchIndexes.length;
|
|
|
|
// Match elements found at the specified indexes
|
|
while ( i-- ) {
|
|
if ( seed[ (j = matchIndexes[i]) ] ) {
|
|
seed[j] = !(matches[j] = seed[j]);
|
|
}
|
|
}
|
|
});
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Checks a node for validity as a Sizzle context
|
|
* @param {Element|Object=} context
|
|
* @returns {Element|Object|Boolean} The input node if acceptable, otherwise a falsy value
|
|
*/
|
|
function testContext( context ) {
|
|
return context && typeof context.getElementsByTagName !== "undefined" && context;
|
|
}
|
|
|
|
// Expose support vars for convenience
|
|
support = Sizzle.support = {};
|
|
|
|
/**
|
|
* Detects XML nodes
|
|
* @param {Element|Object} elem An element or a document
|
|
* @returns {Boolean} True iff elem is a non-HTML XML node
|
|
*/
|
|
isXML = Sizzle.isXML = function( elem ) {
|
|
// documentElement is verified for cases where it doesn't yet exist
|
|
// (such as loading iframes in IE - #4833)
|
|
var documentElement = elem && (elem.ownerDocument || elem).documentElement;
|
|
return documentElement ? documentElement.nodeName !== "HTML" : false;
|
|
};
|
|
|
|
/**
|
|
* Sets document-related variables once based on the current document
|
|
* @param {Element|Object} [doc] An element or document object to use to set the document
|
|
* @returns {Object} Returns the current document
|
|
*/
|
|
setDocument = Sizzle.setDocument = function( node ) {
|
|
var hasCompare, parent,
|
|
doc = node ? node.ownerDocument || node : preferredDoc;
|
|
|
|
// Return early if doc is invalid or already selected
|
|
if ( doc === document || doc.nodeType !== 9 || !doc.documentElement ) {
|
|
return document;
|
|
}
|
|
|
|
// Update global variables
|
|
document = doc;
|
|
docElem = document.documentElement;
|
|
documentIsHTML = !isXML( document );
|
|
|
|
// Support: IE 9-11, Edge
|
|
// Accessing iframe documents after unload throws "permission denied" errors (jQuery #13936)
|
|
if ( (parent = document.defaultView) && parent.top !== parent ) {
|
|
// Support: IE 11
|
|
if ( parent.addEventListener ) {
|
|
parent.addEventListener( "unload", unloadHandler, false );
|
|
|
|
// Support: IE 9 - 10 only
|
|
} else if ( parent.attachEvent ) {
|
|
parent.attachEvent( "onunload", unloadHandler );
|
|
}
|
|
}
|
|
|
|
/* Attributes
|
|
---------------------------------------------------------------------- */
|
|
|
|
// Support: IE<8
|
|
// Verify that getAttribute really returns attributes and not properties
|
|
// (excepting IE8 booleans)
|
|
support.attributes = assert(function( div ) {
|
|
div.className = "i";
|
|
return !div.getAttribute("className");
|
|
});
|
|
|
|
/* getElement(s)By*
|
|
---------------------------------------------------------------------- */
|
|
|
|
// Check if getElementsByTagName("*") returns only elements
|
|
support.getElementsByTagName = assert(function( div ) {
|
|
div.appendChild( document.createComment("") );
|
|
return !div.getElementsByTagName("*").length;
|
|
});
|
|
|
|
// Support: IE<9
|
|
support.getElementsByClassName = rnative.test( document.getElementsByClassName );
|
|
|
|
// Support: IE<10
|
|
// Check if getElementById returns elements by name
|
|
// The broken getElementById methods don't pick up programatically-set names,
|
|
// so use a roundabout getElementsByName test
|
|
support.getById = assert(function( div ) {
|
|
docElem.appendChild( div ).id = expando;
|
|
return !document.getElementsByName || !document.getElementsByName( expando ).length;
|
|
});
|
|
|
|
// ID find and filter
|
|
if ( support.getById ) {
|
|
Expr.find["ID"] = function( id, context ) {
|
|
if ( typeof context.getElementById !== "undefined" && documentIsHTML ) {
|
|
var m = context.getElementById( id );
|
|
return m ? [ m ] : [];
|
|
}
|
|
};
|
|
Expr.filter["ID"] = function( id ) {
|
|
var attrId = id.replace( runescape, funescape );
|
|
return function( elem ) {
|
|
return elem.getAttribute("id") === attrId;
|
|
};
|
|
};
|
|
} else {
|
|
// Support: IE6/7
|
|
// getElementById is not reliable as a find shortcut
|
|
delete Expr.find["ID"];
|
|
|
|
Expr.filter["ID"] = function( id ) {
|
|
var attrId = id.replace( runescape, funescape );
|
|
return function( elem ) {
|
|
var node = typeof elem.getAttributeNode !== "undefined" &&
|
|
elem.getAttributeNode("id");
|
|
return node && node.value === attrId;
|
|
};
|
|
};
|
|
}
|
|
|
|
// Tag
|
|
Expr.find["TAG"] = support.getElementsByTagName ?
|
|
function( tag, context ) {
|
|
if ( typeof context.getElementsByTagName !== "undefined" ) {
|
|
return context.getElementsByTagName( tag );
|
|
|
|
// DocumentFragment nodes don't have gEBTN
|
|
} else if ( support.qsa ) {
|
|
return context.querySelectorAll( tag );
|
|
}
|
|
} :
|
|
|
|
function( tag, context ) {
|
|
var elem,
|
|
tmp = [],
|
|
i = 0,
|
|
// By happy coincidence, a (broken) gEBTN appears on DocumentFragment nodes too
|
|
results = context.getElementsByTagName( tag );
|
|
|
|
// Filter out possible comments
|
|
if ( tag === "*" ) {
|
|
while ( (elem = results[i++]) ) {
|
|
if ( elem.nodeType === 1 ) {
|
|
tmp.push( elem );
|
|
}
|
|
}
|
|
|
|
return tmp;
|
|
}
|
|
return results;
|
|
};
|
|
|
|
// Class
|
|
Expr.find["CLASS"] = support.getElementsByClassName && function( className, context ) {
|
|
if ( typeof context.getElementsByClassName !== "undefined" && documentIsHTML ) {
|
|
return context.getElementsByClassName( className );
|
|
}
|
|
};
|
|
|
|
/* QSA/matchesSelector
|
|
---------------------------------------------------------------------- */
|
|
|
|
// QSA and matchesSelector support
|
|
|
|
// matchesSelector(:active) reports false when true (IE9/Opera 11.5)
|
|
rbuggyMatches = [];
|
|
|
|
// qSa(:focus) reports false when true (Chrome 21)
|
|
// We allow this because of a bug in IE8/9 that throws an error
|
|
// whenever `document.activeElement` is accessed on an iframe
|
|
// So, we allow :focus to pass through QSA all the time to avoid the IE error
|
|
// See http://bugs.jquery.com/ticket/13378
|
|
rbuggyQSA = [];
|
|
|
|
if ( (support.qsa = rnative.test( document.querySelectorAll )) ) {
|
|
// Build QSA regex
|
|
// Regex strategy adopted from Diego Perini
|
|
assert(function( div ) {
|
|
// Select is set to empty string on purpose
|
|
// This is to test IE's treatment of not explicitly
|
|
// setting a boolean content attribute,
|
|
// since its presence should be enough
|
|
// http://bugs.jquery.com/ticket/12359
|
|
docElem.appendChild( div ).innerHTML = "<a id='" + expando + "'></a>" +
|
|
"<select id='" + expando + "-\r\\' msallowcapture=''>" +
|
|
"<option selected=''></option></select>";
|
|
|
|
// Support: IE8, Opera 11-12.16
|
|
// Nothing should be selected when empty strings follow ^= or $= or *=
|
|
// The test attribute must be unknown in Opera but "safe" for WinRT
|
|
// http://msdn.microsoft.com/en-us/library/ie/hh465388.aspx#attribute_section
|
|
if ( div.querySelectorAll("[msallowcapture^='']").length ) {
|
|
rbuggyQSA.push( "[*^$]=" + whitespace + "*(?:''|\"\")" );
|
|
}
|
|
|
|
// Support: IE8
|
|
// Boolean attributes and "value" are not treated correctly
|
|
if ( !div.querySelectorAll("[selected]").length ) {
|
|
rbuggyQSA.push( "\\[" + whitespace + "*(?:value|" + booleans + ")" );
|
|
}
|
|
|
|
// Support: Chrome<29, Android<4.4, Safari<7.0+, iOS<7.0+, PhantomJS<1.9.8+
|
|
if ( !div.querySelectorAll( "[id~=" + expando + "-]" ).length ) {
|
|
rbuggyQSA.push("~=");
|
|
}
|
|
|
|
// Webkit/Opera - :checked should return selected option elements
|
|
// http://www.w3.org/TR/2011/REC-css3-selectors-20110929/#checked
|
|
// IE8 throws error here and will not see later tests
|
|
if ( !div.querySelectorAll(":checked").length ) {
|
|
rbuggyQSA.push(":checked");
|
|
}
|
|
|
|
// Support: Safari 8+, iOS 8+
|
|
// https://bugs.webkit.org/show_bug.cgi?id=136851
|
|
// In-page `selector#id sibing-combinator selector` fails
|
|
if ( !div.querySelectorAll( "a#" + expando + "+*" ).length ) {
|
|
rbuggyQSA.push(".#.+[+~]");
|
|
}
|
|
});
|
|
|
|
assert(function( div ) {
|
|
// Support: Windows 8 Native Apps
|
|
// The type and name attributes are restricted during .innerHTML assignment
|
|
var input = document.createElement("input");
|
|
input.setAttribute( "type", "hidden" );
|
|
div.appendChild( input ).setAttribute( "name", "D" );
|
|
|
|
// Support: IE8
|
|
// Enforce case-sensitivity of name attribute
|
|
if ( div.querySelectorAll("[name=d]").length ) {
|
|
rbuggyQSA.push( "name" + whitespace + "*[*^$|!~]?=" );
|
|
}
|
|
|
|
// FF 3.5 - :enabled/:disabled and hidden elements (hidden elements are still enabled)
|
|
// IE8 throws error here and will not see later tests
|
|
if ( !div.querySelectorAll(":enabled").length ) {
|
|
rbuggyQSA.push( ":enabled", ":disabled" );
|
|
}
|
|
|
|
// Opera 10-11 does not throw on post-comma invalid pseudos
|
|
div.querySelectorAll("*,:x");
|
|
rbuggyQSA.push(",.*:");
|
|
});
|
|
}
|
|
|
|
if ( (support.matchesSelector = rnative.test( (matches = docElem.matches ||
|
|
docElem.webkitMatchesSelector ||
|
|
docElem.mozMatchesSelector ||
|
|
docElem.oMatchesSelector ||
|
|
docElem.msMatchesSelector) )) ) {
|
|
|
|
assert(function( div ) {
|
|
// Check to see if it's possible to do matchesSelector
|
|
// on a disconnected node (IE 9)
|
|
support.disconnectedMatch = matches.call( div, "div" );
|
|
|
|
// This should fail with an exception
|
|
// Gecko does not error, returns false instead
|
|
matches.call( div, "[s!='']:x" );
|
|
rbuggyMatches.push( "!=", pseudos );
|
|
});
|
|
}
|
|
|
|
rbuggyQSA = rbuggyQSA.length && new RegExp( rbuggyQSA.join("|") );
|
|
rbuggyMatches = rbuggyMatches.length && new RegExp( rbuggyMatches.join("|") );
|
|
|
|
/* Contains
|
|
---------------------------------------------------------------------- */
|
|
hasCompare = rnative.test( docElem.compareDocumentPosition );
|
|
|
|
// Element contains another
|
|
// Purposefully self-exclusive
|
|
// As in, an element does not contain itself
|
|
contains = hasCompare || rnative.test( docElem.contains ) ?
|
|
function( a, b ) {
|
|
var adown = a.nodeType === 9 ? a.documentElement : a,
|
|
bup = b && b.parentNode;
|
|
return a === bup || !!( bup && bup.nodeType === 1 && (
|
|
adown.contains ?
|
|
adown.contains( bup ) :
|
|
a.compareDocumentPosition && a.compareDocumentPosition( bup ) & 16
|
|
));
|
|
} :
|
|
function( a, b ) {
|
|
if ( b ) {
|
|
while ( (b = b.parentNode) ) {
|
|
if ( b === a ) {
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
return false;
|
|
};
|
|
|
|
/* Sorting
|
|
---------------------------------------------------------------------- */
|
|
|
|
// Document order sorting
|
|
sortOrder = hasCompare ?
|
|
function( a, b ) {
|
|
|
|
// Flag for duplicate removal
|
|
if ( a === b ) {
|
|
hasDuplicate = true;
|
|
return 0;
|
|
}
|
|
|
|
// Sort on method existence if only one input has compareDocumentPosition
|
|
var compare = !a.compareDocumentPosition - !b.compareDocumentPosition;
|
|
if ( compare ) {
|
|
return compare;
|
|
}
|
|
|
|
// Calculate position if both inputs belong to the same document
|
|
compare = ( a.ownerDocument || a ) === ( b.ownerDocument || b ) ?
|
|
a.compareDocumentPosition( b ) :
|
|
|
|
// Otherwise we know they are disconnected
|
|
1;
|
|
|
|
// Disconnected nodes
|
|
if ( compare & 1 ||
|
|
(!support.sortDetached && b.compareDocumentPosition( a ) === compare) ) {
|
|
|
|
// Choose the first element that is related to our preferred document
|
|
if ( a === document || a.ownerDocument === preferredDoc && contains(preferredDoc, a) ) {
|
|
return -1;
|
|
}
|
|
if ( b === document || b.ownerDocument === preferredDoc && contains(preferredDoc, b) ) {
|
|
return 1;
|
|
}
|
|
|
|
// Maintain original order
|
|
return sortInput ?
|
|
( indexOf( sortInput, a ) - indexOf( sortInput, b ) ) :
|
|
0;
|
|
}
|
|
|
|
return compare & 4 ? -1 : 1;
|
|
} :
|
|
function( a, b ) {
|
|
// Exit early if the nodes are identical
|
|
if ( a === b ) {
|
|
hasDuplicate = true;
|
|
return 0;
|
|
}
|
|
|
|
var cur,
|
|
i = 0,
|
|
aup = a.parentNode,
|
|
bup = b.parentNode,
|
|
ap = [ a ],
|
|
bp = [ b ];
|
|
|
|
// Parentless nodes are either documents or disconnected
|
|
if ( !aup || !bup ) {
|
|
return a === document ? -1 :
|
|
b === document ? 1 :
|
|
aup ? -1 :
|
|
bup ? 1 :
|
|
sortInput ?
|
|
( indexOf( sortInput, a ) - indexOf( sortInput, b ) ) :
|
|
0;
|
|
|
|
// If the nodes are siblings, we can do a quick check
|
|
} else if ( aup === bup ) {
|
|
return siblingCheck( a, b );
|
|
}
|
|
|
|
// Otherwise we need full lists of their ancestors for comparison
|
|
cur = a;
|
|
while ( (cur = cur.parentNode) ) {
|
|
ap.unshift( cur );
|
|
}
|
|
cur = b;
|
|
while ( (cur = cur.parentNode) ) {
|
|
bp.unshift( cur );
|
|
}
|
|
|
|
// Walk down the tree looking for a discrepancy
|
|
while ( ap[i] === bp[i] ) {
|
|
i++;
|
|
}
|
|
|
|
return i ?
|
|
// Do a sibling check if the nodes have a common ancestor
|
|
siblingCheck( ap[i], bp[i] ) :
|
|
|
|
// Otherwise nodes in our document sort first
|
|
ap[i] === preferredDoc ? -1 :
|
|
bp[i] === preferredDoc ? 1 :
|
|
0;
|
|
};
|
|
|
|
return document;
|
|
};
|
|
|
|
Sizzle.matches = function( expr, elements ) {
|
|
return Sizzle( expr, null, null, elements );
|
|
};
|
|
|
|
Sizzle.matchesSelector = function( elem, expr ) {
|
|
// Set document vars if needed
|
|
if ( ( elem.ownerDocument || elem ) !== document ) {
|
|
setDocument( elem );
|
|
}
|
|
|
|
// Make sure that attribute selectors are quoted
|
|
expr = expr.replace( rattributeQuotes, "='$1']" );
|
|
|
|
if ( support.matchesSelector && documentIsHTML &&
|
|
!compilerCache[ expr + " " ] &&
|
|
( !rbuggyMatches || !rbuggyMatches.test( expr ) ) &&
|
|
( !rbuggyQSA || !rbuggyQSA.test( expr ) ) ) {
|
|
|
|
try {
|
|
var ret = matches.call( elem, expr );
|
|
|
|
// IE 9's matchesSelector returns false on disconnected nodes
|
|
if ( ret || support.disconnectedMatch ||
|
|
// As well, disconnected nodes are said to be in a document
|
|
// fragment in IE 9
|
|
elem.document && elem.document.nodeType !== 11 ) {
|
|
return ret;
|
|
}
|
|
} catch (e) {}
|
|
}
|
|
|
|
return Sizzle( expr, document, null, [ elem ] ).length > 0;
|
|
};
|
|
|
|
Sizzle.contains = function( context, elem ) {
|
|
// Set document vars if needed
|
|
if ( ( context.ownerDocument || context ) !== document ) {
|
|
setDocument( context );
|
|
}
|
|
return contains( context, elem );
|
|
};
|
|
|
|
Sizzle.attr = function( elem, name ) {
|
|
// Set document vars if needed
|
|
if ( ( elem.ownerDocument || elem ) !== document ) {
|
|
setDocument( elem );
|
|
}
|
|
|
|
var fn = Expr.attrHandle[ name.toLowerCase() ],
|
|
// Don't get fooled by Object.prototype properties (jQuery #13807)
|
|
val = fn && hasOwn.call( Expr.attrHandle, name.toLowerCase() ) ?
|
|
fn( elem, name, !documentIsHTML ) :
|
|
undefined;
|
|
|
|
return val !== undefined ?
|
|
val :
|
|
support.attributes || !documentIsHTML ?
|
|
elem.getAttribute( name ) :
|
|
(val = elem.getAttributeNode(name)) && val.specified ?
|
|
val.value :
|
|
null;
|
|
};
|
|
|
|
Sizzle.error = function( msg ) {
|
|
throw new Error( "Syntax error, unrecognized expression: " + msg );
|
|
};
|
|
|
|
/**
|
|
* Document sorting and removing duplicates
|
|
* @param {ArrayLike} results
|
|
*/
|
|
Sizzle.uniqueSort = function( results ) {
|
|
var elem,
|
|
duplicates = [],
|
|
j = 0,
|
|
i = 0;
|
|
|
|
// Unless we *know* we can detect duplicates, assume their presence
|
|
hasDuplicate = !support.detectDuplicates;
|
|
sortInput = !support.sortStable && results.slice( 0 );
|
|
results.sort( sortOrder );
|
|
|
|
if ( hasDuplicate ) {
|
|
while ( (elem = results[i++]) ) {
|
|
if ( elem === results[ i ] ) {
|
|
j = duplicates.push( i );
|
|
}
|
|
}
|
|
while ( j-- ) {
|
|
results.splice( duplicates[ j ], 1 );
|
|
}
|
|
}
|
|
|
|
// Clear input after sorting to release objects
|
|
// See https://github.com/jquery/sizzle/pull/225
|
|
sortInput = null;
|
|
|
|
return results;
|
|
};
|
|
|
|
/**
|
|
* Utility function for retrieving the text value of an array of DOM nodes
|
|
* @param {Array|Element} elem
|
|
*/
|
|
getText = Sizzle.getText = function( elem ) {
|
|
var node,
|
|
ret = "",
|
|
i = 0,
|
|
nodeType = elem.nodeType;
|
|
|
|
if ( !nodeType ) {
|
|
// If no nodeType, this is expected to be an array
|
|
while ( (node = elem[i++]) ) {
|
|
// Do not traverse comment nodes
|
|
ret += getText( node );
|
|
}
|
|
} else if ( nodeType === 1 || nodeType === 9 || nodeType === 11 ) {
|
|
// Use textContent for elements
|
|
// innerText usage removed for consistency of new lines (jQuery #11153)
|
|
if ( typeof elem.textContent === "string" ) {
|
|
return elem.textContent;
|
|
} else {
|
|
// Traverse its children
|
|
for ( elem = elem.firstChild; elem; elem = elem.nextSibling ) {
|
|
ret += getText( elem );
|
|
}
|
|
}
|
|
} else if ( nodeType === 3 || nodeType === 4 ) {
|
|
return elem.nodeValue;
|
|
}
|
|
// Do not include comment or processing instruction nodes
|
|
|
|
return ret;
|
|
};
|
|
|
|
Expr = Sizzle.selectors = {
|
|
|
|
// Can be adjusted by the user
|
|
cacheLength: 50,
|
|
|
|
createPseudo: markFunction,
|
|
|
|
match: matchExpr,
|
|
|
|
attrHandle: {},
|
|
|
|
find: {},
|
|
|
|
relative: {
|
|
">": { dir: "parentNode", first: true },
|
|
" ": { dir: "parentNode" },
|
|
"+": { dir: "previousSibling", first: true },
|
|
"~": { dir: "previousSibling" }
|
|
},
|
|
|
|
preFilter: {
|
|
"ATTR": function( match ) {
|
|
match[1] = match[1].replace( runescape, funescape );
|
|
|
|
// Move the given value to match[3] whether quoted or unquoted
|
|
match[3] = ( match[3] || match[4] || match[5] || "" ).replace( runescape, funescape );
|
|
|
|
if ( match[2] === "~=" ) {
|
|
match[3] = " " + match[3] + " ";
|
|
}
|
|
|
|
return match.slice( 0, 4 );
|
|
},
|
|
|
|
"CHILD": function( match ) {
|
|
/* matches from matchExpr["CHILD"]
|
|
1 type (only|nth|...)
|
|
2 what (child|of-type)
|
|
3 argument (even|odd|\d*|\d*n([+-]\d+)?|...)
|
|
4 xn-component of xn+y argument ([+-]?\d*n|)
|
|
5 sign of xn-component
|
|
6 x of xn-component
|
|
7 sign of y-component
|
|
8 y of y-component
|
|
*/
|
|
match[1] = match[1].toLowerCase();
|
|
|
|
if ( match[1].slice( 0, 3 ) === "nth" ) {
|
|
// nth-* requires argument
|
|
if ( !match[3] ) {
|
|
Sizzle.error( match[0] );
|
|
}
|
|
|
|
// numeric x and y parameters for Expr.filter.CHILD
|
|
// remember that false/true cast respectively to 0/1
|
|
match[4] = +( match[4] ? match[5] + (match[6] || 1) : 2 * ( match[3] === "even" || match[3] === "odd" ) );
|
|
match[5] = +( ( match[7] + match[8] ) || match[3] === "odd" );
|
|
|
|
// other types prohibit arguments
|
|
} else if ( match[3] ) {
|
|
Sizzle.error( match[0] );
|
|
}
|
|
|
|
return match;
|
|
},
|
|
|
|
"PSEUDO": function( match ) {
|
|
var excess,
|
|
unquoted = !match[6] && match[2];
|
|
|
|
if ( matchExpr["CHILD"].test( match[0] ) ) {
|
|
return null;
|
|
}
|
|
|
|
// Accept quoted arguments as-is
|
|
if ( match[3] ) {
|
|
match[2] = match[4] || match[5] || "";
|
|
|
|
// Strip excess characters from unquoted arguments
|
|
} else if ( unquoted && rpseudo.test( unquoted ) &&
|
|
// Get excess from tokenize (recursively)
|
|
(excess = tokenize( unquoted, true )) &&
|
|
// advance to the next closing parenthesis
|
|
(excess = unquoted.indexOf( ")", unquoted.length - excess ) - unquoted.length) ) {
|
|
|
|
// excess is a negative index
|
|
match[0] = match[0].slice( 0, excess );
|
|
match[2] = unquoted.slice( 0, excess );
|
|
}
|
|
|
|
// Return only captures needed by the pseudo filter method (type and argument)
|
|
return match.slice( 0, 3 );
|
|
}
|
|
},
|
|
|
|
filter: {
|
|
|
|
"TAG": function( nodeNameSelector ) {
|
|
var nodeName = nodeNameSelector.replace( runescape, funescape ).toLowerCase();
|
|
return nodeNameSelector === "*" ?
|
|
function() { return true; } :
|
|
function( elem ) {
|
|
return elem.nodeName && elem.nodeName.toLowerCase() === nodeName;
|
|
};
|
|
},
|
|
|
|
"CLASS": function( className ) {
|
|
var pattern = classCache[ className + " " ];
|
|
|
|
return pattern ||
|
|
(pattern = new RegExp( "(^|" + whitespace + ")" + className + "(" + whitespace + "|$)" )) &&
|
|
classCache( className, function( elem ) {
|
|
return pattern.test( typeof elem.className === "string" && elem.className || typeof elem.getAttribute !== "undefined" && elem.getAttribute("class") || "" );
|
|
});
|
|
},
|
|
|
|
"ATTR": function( name, operator, check ) {
|
|
return function( elem ) {
|
|
var result = Sizzle.attr( elem, name );
|
|
|
|
if ( result == null ) {
|
|
return operator === "!=";
|
|
}
|
|
if ( !operator ) {
|
|
return true;
|
|
}
|
|
|
|
result += "";
|
|
|
|
return operator === "=" ? result === check :
|
|
operator === "!=" ? result !== check :
|
|
operator === "^=" ? check && result.indexOf( check ) === 0 :
|
|
operator === "*=" ? check && result.indexOf( check ) > -1 :
|
|
operator === "$=" ? check && result.slice( -check.length ) === check :
|
|
operator === "~=" ? ( " " + result.replace( rwhitespace, " " ) + " " ).indexOf( check ) > -1 :
|
|
operator === "|=" ? result === check || result.slice( 0, check.length + 1 ) === check + "-" :
|
|
false;
|
|
};
|
|
},
|
|
|
|
"CHILD": function( type, what, argument, first, last ) {
|
|
var simple = type.slice( 0, 3 ) !== "nth",
|
|
forward = type.slice( -4 ) !== "last",
|
|
ofType = what === "of-type";
|
|
|
|
return first === 1 && last === 0 ?
|
|
|
|
// Shortcut for :nth-*(n)
|
|
function( elem ) {
|
|
return !!elem.parentNode;
|
|
} :
|
|
|
|
function( elem, context, xml ) {
|
|
var cache, uniqueCache, outerCache, node, nodeIndex, start,
|
|
dir = simple !== forward ? "nextSibling" : "previousSibling",
|
|
parent = elem.parentNode,
|
|
name = ofType && elem.nodeName.toLowerCase(),
|
|
useCache = !xml && !ofType,
|
|
diff = false;
|
|
|
|
if ( parent ) {
|
|
|
|
// :(first|last|only)-(child|of-type)
|
|
if ( simple ) {
|
|
while ( dir ) {
|
|
node = elem;
|
|
while ( (node = node[ dir ]) ) {
|
|
if ( ofType ?
|
|
node.nodeName.toLowerCase() === name :
|
|
node.nodeType === 1 ) {
|
|
|
|
return false;
|
|
}
|
|
}
|
|
// Reverse direction for :only-* (if we haven't yet done so)
|
|
start = dir = type === "only" && !start && "nextSibling";
|
|
}
|
|
return true;
|
|
}
|
|
|
|
start = [ forward ? parent.firstChild : parent.lastChild ];
|
|
|
|
// non-xml :nth-child(...) stores cache data on `parent`
|
|
if ( forward && useCache ) {
|
|
|
|
// Seek `elem` from a previously-cached index
|
|
|
|
// ...in a gzip-friendly way
|
|
node = parent;
|
|
outerCache = node[ expando ] || (node[ expando ] = {});
|
|
|
|
// Support: IE <9 only
|
|
// Defend against cloned attroperties (jQuery gh-1709)
|
|
uniqueCache = outerCache[ node.uniqueID ] ||
|
|
(outerCache[ node.uniqueID ] = {});
|
|
|
|
cache = uniqueCache[ type ] || [];
|
|
nodeIndex = cache[ 0 ] === dirruns && cache[ 1 ];
|
|
diff = nodeIndex && cache[ 2 ];
|
|
node = nodeIndex && parent.childNodes[ nodeIndex ];
|
|
|
|
while ( (node = ++nodeIndex && node && node[ dir ] ||
|
|
|
|
// Fallback to seeking `elem` from the start
|
|
(diff = nodeIndex = 0) || start.pop()) ) {
|
|
|
|
// When found, cache indexes on `parent` and break
|
|
if ( node.nodeType === 1 && ++diff && node === elem ) {
|
|
uniqueCache[ type ] = [ dirruns, nodeIndex, diff ];
|
|
break;
|
|
}
|
|
}
|
|
|
|
} else {
|
|
// Use previously-cached element index if available
|
|
if ( useCache ) {
|
|
// ...in a gzip-friendly way
|
|
node = elem;
|
|
outerCache = node[ expando ] || (node[ expando ] = {});
|
|
|
|
// Support: IE <9 only
|
|
// Defend against cloned attroperties (jQuery gh-1709)
|
|
uniqueCache = outerCache[ node.uniqueID ] ||
|
|
(outerCache[ node.uniqueID ] = {});
|
|
|
|
cache = uniqueCache[ type ] || [];
|
|
nodeIndex = cache[ 0 ] === dirruns && cache[ 1 ];
|
|
diff = nodeIndex;
|
|
}
|
|
|
|
// xml :nth-child(...)
|
|
// or :nth-last-child(...) or :nth(-last)?-of-type(...)
|
|
if ( diff === false ) {
|
|
// Use the same loop as above to seek `elem` from the start
|
|
while ( (node = ++nodeIndex && node && node[ dir ] ||
|
|
(diff = nodeIndex = 0) || start.pop()) ) {
|
|
|
|
if ( ( ofType ?
|
|
node.nodeName.toLowerCase() === name :
|
|
node.nodeType === 1 ) &&
|
|
++diff ) {
|
|
|
|
// Cache the index of each encountered element
|
|
if ( useCache ) {
|
|
outerCache = node[ expando ] || (node[ expando ] = {});
|
|
|
|
// Support: IE <9 only
|
|
// Defend against cloned attroperties (jQuery gh-1709)
|
|
uniqueCache = outerCache[ node.uniqueID ] ||
|
|
(outerCache[ node.uniqueID ] = {});
|
|
|
|
uniqueCache[ type ] = [ dirruns, diff ];
|
|
}
|
|
|
|
if ( node === elem ) {
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// Incorporate the offset, then check against cycle size
|
|
diff -= last;
|
|
return diff === first || ( diff % first === 0 && diff / first >= 0 );
|
|
}
|
|
};
|
|
},
|
|
|
|
"PSEUDO": function( pseudo, argument ) {
|
|
// pseudo-class names are case-insensitive
|
|
// http://www.w3.org/TR/selectors/#pseudo-classes
|
|
// Prioritize by case sensitivity in case custom pseudos are added with uppercase letters
|
|
// Remember that setFilters inherits from pseudos
|
|
var args,
|
|
fn = Expr.pseudos[ pseudo ] || Expr.setFilters[ pseudo.toLowerCase() ] ||
|
|
Sizzle.error( "unsupported pseudo: " + pseudo );
|
|
|
|
// The user may use createPseudo to indicate that
|
|
// arguments are needed to create the filter function
|
|
// just as Sizzle does
|
|
if ( fn[ expando ] ) {
|
|
return fn( argument );
|
|
}
|
|
|
|
// But maintain support for old signatures
|
|
if ( fn.length > 1 ) {
|
|
args = [ pseudo, pseudo, "", argument ];
|
|
return Expr.setFilters.hasOwnProperty( pseudo.toLowerCase() ) ?
|
|
markFunction(function( seed, matches ) {
|
|
var idx,
|
|
matched = fn( seed, argument ),
|
|
i = matched.length;
|
|
while ( i-- ) {
|
|
idx = indexOf( seed, matched[i] );
|
|
seed[ idx ] = !( matches[ idx ] = matched[i] );
|
|
}
|
|
}) :
|
|
function( elem ) {
|
|
return fn( elem, 0, args );
|
|
};
|
|
}
|
|
|
|
return fn;
|
|
}
|
|
},
|
|
|
|
pseudos: {
|
|
// Potentially complex pseudos
|
|
"not": markFunction(function( selector ) {
|
|
// Trim the selector passed to compile
|
|
// to avoid treating leading and trailing
|
|
// spaces as combinators
|
|
var input = [],
|
|
results = [],
|
|
matcher = compile( selector.replace( rtrim, "$1" ) );
|
|
|
|
return matcher[ expando ] ?
|
|
markFunction(function( seed, matches, context, xml ) {
|
|
var elem,
|
|
unmatched = matcher( seed, null, xml, [] ),
|
|
i = seed.length;
|
|
|
|
// Match elements unmatched by `matcher`
|
|
while ( i-- ) {
|
|
if ( (elem = unmatched[i]) ) {
|
|
seed[i] = !(matches[i] = elem);
|
|
}
|
|
}
|
|
}) :
|
|
function( elem, context, xml ) {
|
|
input[0] = elem;
|
|
matcher( input, null, xml, results );
|
|
// Don't keep the element (issue #299)
|
|
input[0] = null;
|
|
return !results.pop();
|
|
};
|
|
}),
|
|
|
|
"has": markFunction(function( selector ) {
|
|
return function( elem ) {
|
|
return Sizzle( selector, elem ).length > 0;
|
|
};
|
|
}),
|
|
|
|
"contains": markFunction(function( text ) {
|
|
text = text.replace( runescape, funescape );
|
|
return function( elem ) {
|
|
return ( elem.textContent || elem.innerText || getText( elem ) ).indexOf( text ) > -1;
|
|
};
|
|
}),
|
|
|
|
// "Whether an element is represented by a :lang() selector
|
|
// is based solely on the element's language value
|
|
// being equal to the identifier C,
|
|
// or beginning with the identifier C immediately followed by "-".
|
|
// The matching of C against the element's language value is performed case-insensitively.
|
|
// The identifier C does not have to be a valid language name."
|
|
// http://www.w3.org/TR/selectors/#lang-pseudo
|
|
"lang": markFunction( function( lang ) {
|
|
// lang value must be a valid identifier
|
|
if ( !ridentifier.test(lang || "") ) {
|
|
Sizzle.error( "unsupported lang: " + lang );
|
|
}
|
|
lang = lang.replace( runescape, funescape ).toLowerCase();
|
|
return function( elem ) {
|
|
var elemLang;
|
|
do {
|
|
if ( (elemLang = documentIsHTML ?
|
|
elem.lang :
|
|
elem.getAttribute("xml:lang") || elem.getAttribute("lang")) ) {
|
|
|
|
elemLang = elemLang.toLowerCase();
|
|
return elemLang === lang || elemLang.indexOf( lang + "-" ) === 0;
|
|
}
|
|
} while ( (elem = elem.parentNode) && elem.nodeType === 1 );
|
|
return false;
|
|
};
|
|
}),
|
|
|
|
// Miscellaneous
|
|
"target": function( elem ) {
|
|
var hash = window.location && window.location.hash;
|
|
return hash && hash.slice( 1 ) === elem.id;
|
|
},
|
|
|
|
"root": function( elem ) {
|
|
return elem === docElem;
|
|
},
|
|
|
|
"focus": function( elem ) {
|
|
return elem === document.activeElement && (!document.hasFocus || document.hasFocus()) && !!(elem.type || elem.href || ~elem.tabIndex);
|
|
},
|
|
|
|
// Boolean properties
|
|
"enabled": function( elem ) {
|
|
return elem.disabled === false;
|
|
},
|
|
|
|
"disabled": function( elem ) {
|
|
return elem.disabled === true;
|
|
},
|
|
|
|
"checked": function( elem ) {
|
|
// In CSS3, :checked should return both checked and selected elements
|
|
// http://www.w3.org/TR/2011/REC-css3-selectors-20110929/#checked
|
|
var nodeName = elem.nodeName.toLowerCase();
|
|
return (nodeName === "input" && !!elem.checked) || (nodeName === "option" && !!elem.selected);
|
|
},
|
|
|
|
"selected": function( elem ) {
|
|
// Accessing this property makes selected-by-default
|
|
// options in Safari work properly
|
|
if ( elem.parentNode ) {
|
|
elem.parentNode.selectedIndex;
|
|
}
|
|
|
|
return elem.selected === true;
|
|
},
|
|
|
|
// Contents
|
|
"empty": function( elem ) {
|
|
// http://www.w3.org/TR/selectors/#empty-pseudo
|
|
// :empty is negated by element (1) or content nodes (text: 3; cdata: 4; entity ref: 5),
|
|
// but not by others (comment: 8; processing instruction: 7; etc.)
|
|
// nodeType < 6 works because attributes (2) do not appear as children
|
|
for ( elem = elem.firstChild; elem; elem = elem.nextSibling ) {
|
|
if ( elem.nodeType < 6 ) {
|
|
return false;
|
|
}
|
|
}
|
|
return true;
|
|
},
|
|
|
|
"parent": function( elem ) {
|
|
return !Expr.pseudos["empty"]( elem );
|
|
},
|
|
|
|
// Element/input types
|
|
"header": function( elem ) {
|
|
return rheader.test( elem.nodeName );
|
|
},
|
|
|
|
"input": function( elem ) {
|
|
return rinputs.test( elem.nodeName );
|
|
},
|
|
|
|
"button": function( elem ) {
|
|
var name = elem.nodeName.toLowerCase();
|
|
return name === "input" && elem.type === "button" || name === "button";
|
|
},
|
|
|
|
"text": function( elem ) {
|
|
var attr;
|
|
return elem.nodeName.toLowerCase() === "input" &&
|
|
elem.type === "text" &&
|
|
|
|
// Support: IE<8
|
|
// New HTML5 attribute values (e.g., "search") appear with elem.type === "text"
|
|
( (attr = elem.getAttribute("type")) == null || attr.toLowerCase() === "text" );
|
|
},
|
|
|
|
// Position-in-collection
|
|
"first": createPositionalPseudo(function() {
|
|
return [ 0 ];
|
|
}),
|
|
|
|
"last": createPositionalPseudo(function( matchIndexes, length ) {
|
|
return [ length - 1 ];
|
|
}),
|
|
|
|
"eq": createPositionalPseudo(function( matchIndexes, length, argument ) {
|
|
return [ argument < 0 ? argument + length : argument ];
|
|
}),
|
|
|
|
"even": createPositionalPseudo(function( matchIndexes, length ) {
|
|
var i = 0;
|
|
for ( ; i < length; i += 2 ) {
|
|
matchIndexes.push( i );
|
|
}
|
|
return matchIndexes;
|
|
}),
|
|
|
|
"odd": createPositionalPseudo(function( matchIndexes, length ) {
|
|
var i = 1;
|
|
for ( ; i < length; i += 2 ) {
|
|
matchIndexes.push( i );
|
|
}
|
|
return matchIndexes;
|
|
}),
|
|
|
|
"lt": createPositionalPseudo(function( matchIndexes, length, argument ) {
|
|
var i = argument < 0 ? argument + length : argument;
|
|
for ( ; --i >= 0; ) {
|
|
matchIndexes.push( i );
|
|
}
|
|
return matchIndexes;
|
|
}),
|
|
|
|
"gt": createPositionalPseudo(function( matchIndexes, length, argument ) {
|
|
var i = argument < 0 ? argument + length : argument;
|
|
for ( ; ++i < length; ) {
|
|
matchIndexes.push( i );
|
|
}
|
|
return matchIndexes;
|
|
})
|
|
}
|
|
};
|
|
|
|
Expr.pseudos["nth"] = Expr.pseudos["eq"];
|
|
|
|
// Add button/input type pseudos
|
|
for ( i in { radio: true, checkbox: true, file: true, password: true, image: true } ) {
|
|
Expr.pseudos[ i ] = createInputPseudo( i );
|
|
}
|
|
for ( i in { submit: true, reset: true } ) {
|
|
Expr.pseudos[ i ] = createButtonPseudo( i );
|
|
}
|
|
|
|
// Easy API for creating new setFilters
|
|
function setFilters() {}
|
|
setFilters.prototype = Expr.filters = Expr.pseudos;
|
|
Expr.setFilters = new setFilters();
|
|
|
|
tokenize = Sizzle.tokenize = function( selector, parseOnly ) {
|
|
var matched, match, tokens, type,
|
|
soFar, groups, preFilters,
|
|
cached = tokenCache[ selector + " " ];
|
|
|
|
if ( cached ) {
|
|
return parseOnly ? 0 : cached.slice( 0 );
|
|
}
|
|
|
|
soFar = selector;
|
|
groups = [];
|
|
preFilters = Expr.preFilter;
|
|
|
|
while ( soFar ) {
|
|
|
|
// Comma and first run
|
|
if ( !matched || (match = rcomma.exec( soFar )) ) {
|
|
if ( match ) {
|
|
// Don't consume trailing commas as valid
|
|
soFar = soFar.slice( match[0].length ) || soFar;
|
|
}
|
|
groups.push( (tokens = []) );
|
|
}
|
|
|
|
matched = false;
|
|
|
|
// Combinators
|
|
if ( (match = rcombinators.exec( soFar )) ) {
|
|
matched = match.shift();
|
|
tokens.push({
|
|
value: matched,
|
|
// Cast descendant combinators to space
|
|
type: match[0].replace( rtrim, " " )
|
|
});
|
|
soFar = soFar.slice( matched.length );
|
|
}
|
|
|
|
// Filters
|
|
for ( type in Expr.filter ) {
|
|
if ( (match = matchExpr[ type ].exec( soFar )) && (!preFilters[ type ] ||
|
|
(match = preFilters[ type ]( match ))) ) {
|
|
matched = match.shift();
|
|
tokens.push({
|
|
value: matched,
|
|
type: type,
|
|
matches: match
|
|
});
|
|
soFar = soFar.slice( matched.length );
|
|
}
|
|
}
|
|
|
|
if ( !matched ) {
|
|
break;
|
|
}
|
|
}
|
|
|
|
// Return the length of the invalid excess
|
|
// if we're just parsing
|
|
// Otherwise, throw an error or return tokens
|
|
return parseOnly ?
|
|
soFar.length :
|
|
soFar ?
|
|
Sizzle.error( selector ) :
|
|
// Cache the tokens
|
|
tokenCache( selector, groups ).slice( 0 );
|
|
};
|
|
|
|
function toSelector( tokens ) {
|
|
var i = 0,
|
|
len = tokens.length,
|
|
selector = "";
|
|
for ( ; i < len; i++ ) {
|
|
selector += tokens[i].value;
|
|
}
|
|
return selector;
|
|
}
|
|
|
|
function addCombinator( matcher, combinator, base ) {
|
|
var dir = combinator.dir,
|
|
checkNonElements = base && dir === "parentNode",
|
|
doneName = done++;
|
|
|
|
return combinator.first ?
|
|
// Check against closest ancestor/preceding element
|
|
function( elem, context, xml ) {
|
|
while ( (elem = elem[ dir ]) ) {
|
|
if ( elem.nodeType === 1 || checkNonElements ) {
|
|
return matcher( elem, context, xml );
|
|
}
|
|
}
|
|
} :
|
|
|
|
// Check against all ancestor/preceding elements
|
|
function( elem, context, xml ) {
|
|
var oldCache, uniqueCache, outerCache,
|
|
newCache = [ dirruns, doneName ];
|
|
|
|
// We can't set arbitrary data on XML nodes, so they don't benefit from combinator caching
|
|
if ( xml ) {
|
|
while ( (elem = elem[ dir ]) ) {
|
|
if ( elem.nodeType === 1 || checkNonElements ) {
|
|
if ( matcher( elem, context, xml ) ) {
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
} else {
|
|
while ( (elem = elem[ dir ]) ) {
|
|
if ( elem.nodeType === 1 || checkNonElements ) {
|
|
outerCache = elem[ expando ] || (elem[ expando ] = {});
|
|
|
|
// Support: IE <9 only
|
|
// Defend against cloned attroperties (jQuery gh-1709)
|
|
uniqueCache = outerCache[ elem.uniqueID ] || (outerCache[ elem.uniqueID ] = {});
|
|
|
|
if ( (oldCache = uniqueCache[ dir ]) &&
|
|
oldCache[ 0 ] === dirruns && oldCache[ 1 ] === doneName ) {
|
|
|
|
// Assign to newCache so results back-propagate to previous elements
|
|
return (newCache[ 2 ] = oldCache[ 2 ]);
|
|
} else {
|
|
// Reuse newcache so results back-propagate to previous elements
|
|
uniqueCache[ dir ] = newCache;
|
|
|
|
// A match means we're done; a fail means we have to keep checking
|
|
if ( (newCache[ 2 ] = matcher( elem, context, xml )) ) {
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
};
|
|
}
|
|
|
|
function elementMatcher( matchers ) {
|
|
return matchers.length > 1 ?
|
|
function( elem, context, xml ) {
|
|
var i = matchers.length;
|
|
while ( i-- ) {
|
|
if ( !matchers[i]( elem, context, xml ) ) {
|
|
return false;
|
|
}
|
|
}
|
|
return true;
|
|
} :
|
|
matchers[0];
|
|
}
|
|
|
|
function multipleContexts( selector, contexts, results ) {
|
|
var i = 0,
|
|
len = contexts.length;
|
|
for ( ; i < len; i++ ) {
|
|
Sizzle( selector, contexts[i], results );
|
|
}
|
|
return results;
|
|
}
|
|
|
|
function condense( unmatched, map, filter, context, xml ) {
|
|
var elem,
|
|
newUnmatched = [],
|
|
i = 0,
|
|
len = unmatched.length,
|
|
mapped = map != null;
|
|
|
|
for ( ; i < len; i++ ) {
|
|
if ( (elem = unmatched[i]) ) {
|
|
if ( !filter || filter( elem, context, xml ) ) {
|
|
newUnmatched.push( elem );
|
|
if ( mapped ) {
|
|
map.push( i );
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return newUnmatched;
|
|
}
|
|
|
|
function setMatcher( preFilter, selector, matcher, postFilter, postFinder, postSelector ) {
|
|
if ( postFilter && !postFilter[ expando ] ) {
|
|
postFilter = setMatcher( postFilter );
|
|
}
|
|
if ( postFinder && !postFinder[ expando ] ) {
|
|
postFinder = setMatcher( postFinder, postSelector );
|
|
}
|
|
return markFunction(function( seed, results, context, xml ) {
|
|
var temp, i, elem,
|
|
preMap = [],
|
|
postMap = [],
|
|
preexisting = results.length,
|
|
|
|
// Get initial elements from seed or context
|
|
elems = seed || multipleContexts( selector || "*", context.nodeType ? [ context ] : context, [] ),
|
|
|
|
// Prefilter to get matcher input, preserving a map for seed-results synchronization
|
|
matcherIn = preFilter && ( seed || !selector ) ?
|
|
condense( elems, preMap, preFilter, context, xml ) :
|
|
elems,
|
|
|
|
matcherOut = matcher ?
|
|
// If we have a postFinder, or filtered seed, or non-seed postFilter or preexisting results,
|
|
postFinder || ( seed ? preFilter : preexisting || postFilter ) ?
|
|
|
|
// ...intermediate processing is necessary
|
|
[] :
|
|
|
|
// ...otherwise use results directly
|
|
results :
|
|
matcherIn;
|
|
|
|
// Find primary matches
|
|
if ( matcher ) {
|
|
matcher( matcherIn, matcherOut, context, xml );
|
|
}
|
|
|
|
// Apply postFilter
|
|
if ( postFilter ) {
|
|
temp = condense( matcherOut, postMap );
|
|
postFilter( temp, [], context, xml );
|
|
|
|
// Un-match failing elements by moving them back to matcherIn
|
|
i = temp.length;
|
|
while ( i-- ) {
|
|
if ( (elem = temp[i]) ) {
|
|
matcherOut[ postMap[i] ] = !(matcherIn[ postMap[i] ] = elem);
|
|
}
|
|
}
|
|
}
|
|
|
|
if ( seed ) {
|
|
if ( postFinder || preFilter ) {
|
|
if ( postFinder ) {
|
|
// Get the final matcherOut by condensing this intermediate into postFinder contexts
|
|
temp = [];
|
|
i = matcherOut.length;
|
|
while ( i-- ) {
|
|
if ( (elem = matcherOut[i]) ) {
|
|
// Restore matcherIn since elem is not yet a final match
|
|
temp.push( (matcherIn[i] = elem) );
|
|
}
|
|
}
|
|
postFinder( null, (matcherOut = []), temp, xml );
|
|
}
|
|
|
|
// Move matched elements from seed to results to keep them synchronized
|
|
i = matcherOut.length;
|
|
while ( i-- ) {
|
|
if ( (elem = matcherOut[i]) &&
|
|
(temp = postFinder ? indexOf( seed, elem ) : preMap[i]) > -1 ) {
|
|
|
|
seed[temp] = !(results[temp] = elem);
|
|
}
|
|
}
|
|
}
|
|
|
|
// Add elements to results, through postFinder if defined
|
|
} else {
|
|
matcherOut = condense(
|
|
matcherOut === results ?
|
|
matcherOut.splice( preexisting, matcherOut.length ) :
|
|
matcherOut
|
|
);
|
|
if ( postFinder ) {
|
|
postFinder( null, results, matcherOut, xml );
|
|
} else {
|
|
push.apply( results, matcherOut );
|
|
}
|
|
}
|
|
});
|
|
}
|
|
|
|
function matcherFromTokens( tokens ) {
|
|
var checkContext, matcher, j,
|
|
len = tokens.length,
|
|
leadingRelative = Expr.relative[ tokens[0].type ],
|
|
implicitRelative = leadingRelative || Expr.relative[" "],
|
|
i = leadingRelative ? 1 : 0,
|
|
|
|
// The foundational matcher ensures that elements are reachable from top-level context(s)
|
|
matchContext = addCombinator( function( elem ) {
|
|
return elem === checkContext;
|
|
}, implicitRelative, true ),
|
|
matchAnyContext = addCombinator( function( elem ) {
|
|
return indexOf( checkContext, elem ) > -1;
|
|
}, implicitRelative, true ),
|
|
matchers = [ function( elem, context, xml ) {
|
|
var ret = ( !leadingRelative && ( xml || context !== outermostContext ) ) || (
|
|
(checkContext = context).nodeType ?
|
|
matchContext( elem, context, xml ) :
|
|
matchAnyContext( elem, context, xml ) );
|
|
// Avoid hanging onto element (issue #299)
|
|
checkContext = null;
|
|
return ret;
|
|
} ];
|
|
|
|
for ( ; i < len; i++ ) {
|
|
if ( (matcher = Expr.relative[ tokens[i].type ]) ) {
|
|
matchers = [ addCombinator(elementMatcher( matchers ), matcher) ];
|
|
} else {
|
|
matcher = Expr.filter[ tokens[i].type ].apply( null, tokens[i].matches );
|
|
|
|
// Return special upon seeing a positional matcher
|
|
if ( matcher[ expando ] ) {
|
|
// Find the next relative operator (if any) for proper handling
|
|
j = ++i;
|
|
for ( ; j < len; j++ ) {
|
|
if ( Expr.relative[ tokens[j].type ] ) {
|
|
break;
|
|
}
|
|
}
|
|
return setMatcher(
|
|
i > 1 && elementMatcher( matchers ),
|
|
i > 1 && toSelector(
|
|
// If the preceding token was a descendant combinator, insert an implicit any-element `*`
|
|
tokens.slice( 0, i - 1 ).concat({ value: tokens[ i - 2 ].type === " " ? "*" : "" })
|
|
).replace( rtrim, "$1" ),
|
|
matcher,
|
|
i < j && matcherFromTokens( tokens.slice( i, j ) ),
|
|
j < len && matcherFromTokens( (tokens = tokens.slice( j )) ),
|
|
j < len && toSelector( tokens )
|
|
);
|
|
}
|
|
matchers.push( matcher );
|
|
}
|
|
}
|
|
|
|
return elementMatcher( matchers );
|
|
}
|
|
|
|
function matcherFromGroupMatchers( elementMatchers, setMatchers ) {
|
|
var bySet = setMatchers.length > 0,
|
|
byElement = elementMatchers.length > 0,
|
|
superMatcher = function( seed, context, xml, results, outermost ) {
|
|
var elem, j, matcher,
|
|
matchedCount = 0,
|
|
i = "0",
|
|
unmatched = seed && [],
|
|
setMatched = [],
|
|
contextBackup = outermostContext,
|
|
// We must always have either seed elements or outermost context
|
|
elems = seed || byElement && Expr.find["TAG"]( "*", outermost ),
|
|
// Use integer dirruns iff this is the outermost matcher
|
|
dirrunsUnique = (dirruns += contextBackup == null ? 1 : Math.random() || 0.1),
|
|
len = elems.length;
|
|
|
|
if ( outermost ) {
|
|
outermostContext = context === document || context || outermost;
|
|
}
|
|
|
|
// Add elements passing elementMatchers directly to results
|
|
// Support: IE<9, Safari
|
|
// Tolerate NodeList properties (IE: "length"; Safari: <number>) matching elements by id
|
|
for ( ; i !== len && (elem = elems[i]) != null; i++ ) {
|
|
if ( byElement && elem ) {
|
|
j = 0;
|
|
if ( !context && elem.ownerDocument !== document ) {
|
|
setDocument( elem );
|
|
xml = !documentIsHTML;
|
|
}
|
|
while ( (matcher = elementMatchers[j++]) ) {
|
|
if ( matcher( elem, context || document, xml) ) {
|
|
results.push( elem );
|
|
break;
|
|
}
|
|
}
|
|
if ( outermost ) {
|
|
dirruns = dirrunsUnique;
|
|
}
|
|
}
|
|
|
|
// Track unmatched elements for set filters
|
|
if ( bySet ) {
|
|
// They will have gone through all possible matchers
|
|
if ( (elem = !matcher && elem) ) {
|
|
matchedCount--;
|
|
}
|
|
|
|
// Lengthen the array for every element, matched or not
|
|
if ( seed ) {
|
|
unmatched.push( elem );
|
|
}
|
|
}
|
|
}
|
|
|
|
// `i` is now the count of elements visited above, and adding it to `matchedCount`
|
|
// makes the latter nonnegative.
|
|
matchedCount += i;
|
|
|
|
// Apply set filters to unmatched elements
|
|
// NOTE: This can be skipped if there are no unmatched elements (i.e., `matchedCount`
|
|
// equals `i`), unless we didn't visit _any_ elements in the above loop because we have
|
|
// no element matchers and no seed.
|
|
// Incrementing an initially-string "0" `i` allows `i` to remain a string only in that
|
|
// case, which will result in a "00" `matchedCount` that differs from `i` but is also
|
|
// numerically zero.
|
|
if ( bySet && i !== matchedCount ) {
|
|
j = 0;
|
|
while ( (matcher = setMatchers[j++]) ) {
|
|
matcher( unmatched, setMatched, context, xml );
|
|
}
|
|
|
|
if ( seed ) {
|
|
// Reintegrate element matches to eliminate the need for sorting
|
|
if ( matchedCount > 0 ) {
|
|
while ( i-- ) {
|
|
if ( !(unmatched[i] || setMatched[i]) ) {
|
|
setMatched[i] = pop.call( results );
|
|
}
|
|
}
|
|
}
|
|
|
|
// Discard index placeholder values to get only actual matches
|
|
setMatched = condense( setMatched );
|
|
}
|
|
|
|
// Add matches to results
|
|
push.apply( results, setMatched );
|
|
|
|
// Seedless set matches succeeding multiple successful matchers stipulate sorting
|
|
if ( outermost && !seed && setMatched.length > 0 &&
|
|
( matchedCount + setMatchers.length ) > 1 ) {
|
|
|
|
Sizzle.uniqueSort( results );
|
|
}
|
|
}
|
|
|
|
// Override manipulation of globals by nested matchers
|
|
if ( outermost ) {
|
|
dirruns = dirrunsUnique;
|
|
outermostContext = contextBackup;
|
|
}
|
|
|
|
return unmatched;
|
|
};
|
|
|
|
return bySet ?
|
|
markFunction( superMatcher ) :
|
|
superMatcher;
|
|
}
|
|
|
|
compile = Sizzle.compile = function( selector, match /* Internal Use Only */ ) {
|
|
var i,
|
|
setMatchers = [],
|
|
elementMatchers = [],
|
|
cached = compilerCache[ selector + " " ];
|
|
|
|
if ( !cached ) {
|
|
// Generate a function of recursive functions that can be used to check each element
|
|
if ( !match ) {
|
|
match = tokenize( selector );
|
|
}
|
|
i = match.length;
|
|
while ( i-- ) {
|
|
cached = matcherFromTokens( match[i] );
|
|
if ( cached[ expando ] ) {
|
|
setMatchers.push( cached );
|
|
} else {
|
|
elementMatchers.push( cached );
|
|
}
|
|
}
|
|
|
|
// Cache the compiled function
|
|
cached = compilerCache( selector, matcherFromGroupMatchers( elementMatchers, setMatchers ) );
|
|
|
|
// Save selector and tokenization
|
|
cached.selector = selector;
|
|
}
|
|
return cached;
|
|
};
|
|
|
|
/**
|
|
* A low-level selection function that works with Sizzle's compiled
|
|
* selector functions
|
|
* @param {String|Function} selector A selector or a pre-compiled
|
|
* selector function built with Sizzle.compile
|
|
* @param {Element} context
|
|
* @param {Array} [results]
|
|
* @param {Array} [seed] A set of elements to match against
|
|
*/
|
|
select = Sizzle.select = function( selector, context, results, seed ) {
|
|
var i, tokens, token, type, find,
|
|
compiled = typeof selector === "function" && selector,
|
|
match = !seed && tokenize( (selector = compiled.selector || selector) );
|
|
|
|
results = results || [];
|
|
|
|
// Try to minimize operations if there is only one selector in the list and no seed
|
|
// (the latter of which guarantees us context)
|
|
if ( match.length === 1 ) {
|
|
|
|
// Reduce context if the leading compound selector is an ID
|
|
tokens = match[0] = match[0].slice( 0 );
|
|
if ( tokens.length > 2 && (token = tokens[0]).type === "ID" &&
|
|
support.getById && context.nodeType === 9 && documentIsHTML &&
|
|
Expr.relative[ tokens[1].type ] ) {
|
|
|
|
context = ( Expr.find["ID"]( token.matches[0].replace(runescape, funescape), context ) || [] )[0];
|
|
if ( !context ) {
|
|
return results;
|
|
|
|
// Precompiled matchers will still verify ancestry, so step up a level
|
|
} else if ( compiled ) {
|
|
context = context.parentNode;
|
|
}
|
|
|
|
selector = selector.slice( tokens.shift().value.length );
|
|
}
|
|
|
|
// Fetch a seed set for right-to-left matching
|
|
i = matchExpr["needsContext"].test( selector ) ? 0 : tokens.length;
|
|
while ( i-- ) {
|
|
token = tokens[i];
|
|
|
|
// Abort if we hit a combinator
|
|
if ( Expr.relative[ (type = token.type) ] ) {
|
|
break;
|
|
}
|
|
if ( (find = Expr.find[ type ]) ) {
|
|
// Search, expanding context for leading sibling combinators
|
|
if ( (seed = find(
|
|
token.matches[0].replace( runescape, funescape ),
|
|
rsibling.test( tokens[0].type ) && testContext( context.parentNode ) || context
|
|
)) ) {
|
|
|
|
// If seed is empty or no tokens remain, we can return early
|
|
tokens.splice( i, 1 );
|
|
selector = seed.length && toSelector( tokens );
|
|
if ( !selector ) {
|
|
push.apply( results, seed );
|
|
return results;
|
|
}
|
|
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// Compile and execute a filtering function if one is not provided
|
|
// Provide `match` to avoid retokenization if we modified the selector above
|
|
( compiled || compile( selector, match ) )(
|
|
seed,
|
|
context,
|
|
!documentIsHTML,
|
|
results,
|
|
!context || rsibling.test( selector ) && testContext( context.parentNode ) || context
|
|
);
|
|
return results;
|
|
};
|
|
|
|
// One-time assignments
|
|
|
|
// Sort stability
|
|
support.sortStable = expando.split("").sort( sortOrder ).join("") === expando;
|
|
|
|
// Support: Chrome 14-35+
|
|
// Always assume duplicates if they aren't passed to the comparison function
|
|
support.detectDuplicates = !!hasDuplicate;
|
|
|
|
// Initialize against the default document
|
|
setDocument();
|
|
|
|
// Support: Webkit<537.32 - Safari 6.0.3/Chrome 25 (fixed in Chrome 27)
|
|
// Detached nodes confoundingly follow *each other*
|
|
support.sortDetached = assert(function( div1 ) {
|
|
// Should return 1, but returns 4 (following)
|
|
return div1.compareDocumentPosition( document.createElement("div") ) & 1;
|
|
});
|
|
|
|
// Support: IE<8
|
|
// Prevent attribute/property "interpolation"
|
|
// http://msdn.microsoft.com/en-us/library/ms536429%28VS.85%29.aspx
|
|
if ( !assert(function( div ) {
|
|
div.innerHTML = "<a href='#'></a>";
|
|
return div.firstChild.getAttribute("href") === "#" ;
|
|
}) ) {
|
|
addHandle( "type|href|height|width", function( elem, name, isXML ) {
|
|
if ( !isXML ) {
|
|
return elem.getAttribute( name, name.toLowerCase() === "type" ? 1 : 2 );
|
|
}
|
|
});
|
|
}
|
|
|
|
// Support: IE<9
|
|
// Use defaultValue in place of getAttribute("value")
|
|
if ( !support.attributes || !assert(function( div ) {
|
|
div.innerHTML = "<input/>";
|
|
div.firstChild.setAttribute( "value", "" );
|
|
return div.firstChild.getAttribute( "value" ) === "";
|
|
}) ) {
|
|
addHandle( "value", function( elem, name, isXML ) {
|
|
if ( !isXML && elem.nodeName.toLowerCase() === "input" ) {
|
|
return elem.defaultValue;
|
|
}
|
|
});
|
|
}
|
|
|
|
// Support: IE<9
|
|
// Use getAttributeNode to fetch booleans when getAttribute lies
|
|
if ( !assert(function( div ) {
|
|
return div.getAttribute("disabled") == null;
|
|
}) ) {
|
|
addHandle( booleans, function( elem, name, isXML ) {
|
|
var val;
|
|
if ( !isXML ) {
|
|
return elem[ name ] === true ? name.toLowerCase() :
|
|
(val = elem.getAttributeNode( name )) && val.specified ?
|
|
val.value :
|
|
null;
|
|
}
|
|
});
|
|
}
|
|
|
|
return Sizzle;
|
|
|
|
})( window );
|
|
|
|
|
|
|
|
jQuery.find = Sizzle;
|
|
jQuery.expr = Sizzle.selectors;
|
|
jQuery.expr[ ":" ] = jQuery.expr.pseudos;
|
|
jQuery.uniqueSort = jQuery.unique = Sizzle.uniqueSort;
|
|
jQuery.text = Sizzle.getText;
|
|
jQuery.isXMLDoc = Sizzle.isXML;
|
|
jQuery.contains = Sizzle.contains;
|
|
|
|
|
|
|
|
var dir = function( elem, dir, until ) {
|
|
var matched = [],
|
|
truncate = until !== undefined;
|
|
|
|
while ( ( elem = elem[ dir ] ) && elem.nodeType !== 9 ) {
|
|
if ( elem.nodeType === 1 ) {
|
|
if ( truncate && jQuery( elem ).is( until ) ) {
|
|
break;
|
|
}
|
|
matched.push( elem );
|
|
}
|
|
}
|
|
return matched;
|
|
};
|
|
|
|
|
|
var siblings = function( n, elem ) {
|
|
var matched = [];
|
|
|
|
for ( ; n; n = n.nextSibling ) {
|
|
if ( n.nodeType === 1 && n !== elem ) {
|
|
matched.push( n );
|
|
}
|
|
}
|
|
|
|
return matched;
|
|
};
|
|
|
|
|
|
var rneedsContext = jQuery.expr.match.needsContext;
|
|
|
|
var rsingleTag = ( /^<([\w-]+)\s*\/?>(?:<\/\1>|)$/ );
|
|
|
|
|
|
|
|
var risSimple = /^.[^:#\[\.,]*$/;
|
|
|
|
// Implement the identical functionality for filter and not
|
|
function winnow( elements, qualifier, not ) {
|
|
if ( jQuery.isFunction( qualifier ) ) {
|
|
return jQuery.grep( elements, function( elem, i ) {
|
|
/* jshint -W018 */
|
|
return !!qualifier.call( elem, i, elem ) !== not;
|
|
} );
|
|
|
|
}
|
|
|
|
if ( qualifier.nodeType ) {
|
|
return jQuery.grep( elements, function( elem ) {
|
|
return ( elem === qualifier ) !== not;
|
|
} );
|
|
|
|
}
|
|
|
|
if ( typeof qualifier === "string" ) {
|
|
if ( risSimple.test( qualifier ) ) {
|
|
return jQuery.filter( qualifier, elements, not );
|
|
}
|
|
|
|
qualifier = jQuery.filter( qualifier, elements );
|
|
}
|
|
|
|
return jQuery.grep( elements, function( elem ) {
|
|
return ( jQuery.inArray( elem, qualifier ) > -1 ) !== not;
|
|
} );
|
|
}
|
|
|
|
jQuery.filter = function( expr, elems, not ) {
|
|
var elem = elems[ 0 ];
|
|
|
|
if ( not ) {
|
|
expr = ":not(" + expr + ")";
|
|
}
|
|
|
|
return elems.length === 1 && elem.nodeType === 1 ?
|
|
jQuery.find.matchesSelector( elem, expr ) ? [ elem ] : [] :
|
|
jQuery.find.matches( expr, jQuery.grep( elems, function( elem ) {
|
|
return elem.nodeType === 1;
|
|
} ) );
|
|
};
|
|
|
|
jQuery.fn.extend( {
|
|
find: function( selector ) {
|
|
var i,
|
|
ret = [],
|
|
self = this,
|
|
len = self.length;
|
|
|
|
if ( typeof selector !== "string" ) {
|
|
return this.pushStack( jQuery( selector ).filter( function() {
|
|
for ( i = 0; i < len; i++ ) {
|
|
if ( jQuery.contains( self[ i ], this ) ) {
|
|
return true;
|
|
}
|
|
}
|
|
} ) );
|
|
}
|
|
|
|
for ( i = 0; i < len; i++ ) {
|
|
jQuery.find( selector, self[ i ], ret );
|
|
}
|
|
|
|
// Needed because $( selector, context ) becomes $( context ).find( selector )
|
|
ret = this.pushStack( len > 1 ? jQuery.unique( ret ) : ret );
|
|
ret.selector = this.selector ? this.selector + " " + selector : selector;
|
|
return ret;
|
|
},
|
|
filter: function( selector ) {
|
|
return this.pushStack( winnow( this, selector || [], false ) );
|
|
},
|
|
not: function( selector ) {
|
|
return this.pushStack( winnow( this, selector || [], true ) );
|
|
},
|
|
is: function( selector ) {
|
|
return !!winnow(
|
|
this,
|
|
|
|
// If this is a positional/relative selector, check membership in the returned set
|
|
// so $("p:first").is("p:last") won't return true for a doc with two "p".
|
|
typeof selector === "string" && rneedsContext.test( selector ) ?
|
|
jQuery( selector ) :
|
|
selector || [],
|
|
false
|
|
).length;
|
|
}
|
|
} );
|
|
|
|
|
|
// Initialize a jQuery object
|
|
|
|
|
|
// A central reference to the root jQuery(document)
|
|
var rootjQuery,
|
|
|
|
// A simple way to check for HTML strings
|
|
// Prioritize #id over <tag> to avoid XSS via location.hash (#9521)
|
|
// Strict HTML recognition (#11290: must start with <)
|
|
rquickExpr = /^(?:\s*(<[\w\W]+>)[^>]*|#([\w-]*))$/,
|
|
|
|
init = jQuery.fn.init = function( selector, context, root ) {
|
|
var match, elem;
|
|
|
|
// HANDLE: $(""), $(null), $(undefined), $(false)
|
|
if ( !selector ) {
|
|
return this;
|
|
}
|
|
|
|
// init accepts an alternate rootjQuery
|
|
// so migrate can support jQuery.sub (gh-2101)
|
|
root = root || rootjQuery;
|
|
|
|
// Handle HTML strings
|
|
if ( typeof selector === "string" ) {
|
|
if ( selector.charAt( 0 ) === "<" &&
|
|
selector.charAt( selector.length - 1 ) === ">" &&
|
|
selector.length >= 3 ) {
|
|
|
|
// Assume that strings that start and end with <> are HTML and skip the regex check
|
|
match = [ null, selector, null ];
|
|
|
|
} else {
|
|
match = rquickExpr.exec( selector );
|
|
}
|
|
|
|
// Match html or make sure no context is specified for #id
|
|
if ( match && ( match[ 1 ] || !context ) ) {
|
|
|
|
// HANDLE: $(html) -> $(array)
|
|
if ( match[ 1 ] ) {
|
|
context = context instanceof jQuery ? context[ 0 ] : context;
|
|
|
|
// scripts is true for back-compat
|
|
// Intentionally let the error be thrown if parseHTML is not present
|
|
jQuery.merge( this, jQuery.parseHTML(
|
|
match[ 1 ],
|
|
context && context.nodeType ? context.ownerDocument || context : document,
|
|
true
|
|
) );
|
|
|
|
// HANDLE: $(html, props)
|
|
if ( rsingleTag.test( match[ 1 ] ) && jQuery.isPlainObject( context ) ) {
|
|
for ( match in context ) {
|
|
|
|
// Properties of context are called as methods if possible
|
|
if ( jQuery.isFunction( this[ match ] ) ) {
|
|
this[ match ]( context[ match ] );
|
|
|
|
// ...and otherwise set as attributes
|
|
} else {
|
|
this.attr( match, context[ match ] );
|
|
}
|
|
}
|
|
}
|
|
|
|
return this;
|
|
|
|
// HANDLE: $(#id)
|
|
} else {
|
|
elem = document.getElementById( match[ 2 ] );
|
|
|
|
// Check parentNode to catch when Blackberry 4.6 returns
|
|
// nodes that are no longer in the document #6963
|
|
if ( elem && elem.parentNode ) {
|
|
|
|
// Handle the case where IE and Opera return items
|
|
// by name instead of ID
|
|
if ( elem.id !== match[ 2 ] ) {
|
|
return rootjQuery.find( selector );
|
|
}
|
|
|
|
// Otherwise, we inject the element directly into the jQuery object
|
|
this.length = 1;
|
|
this[ 0 ] = elem;
|
|
}
|
|
|
|
this.context = document;
|
|
this.selector = selector;
|
|
return this;
|
|
}
|
|
|
|
// HANDLE: $(expr, $(...))
|
|
} else if ( !context || context.jquery ) {
|
|
return ( context || root ).find( selector );
|
|
|
|
// HANDLE: $(expr, context)
|
|
// (which is just equivalent to: $(context).find(expr)
|
|
} else {
|
|
return this.constructor( context ).find( selector );
|
|
}
|
|
|
|
// HANDLE: $(DOMElement)
|
|
} else if ( selector.nodeType ) {
|
|
this.context = this[ 0 ] = selector;
|
|
this.length = 1;
|
|
return this;
|
|
|
|
// HANDLE: $(function)
|
|
// Shortcut for document ready
|
|
} else if ( jQuery.isFunction( selector ) ) {
|
|
return typeof root.ready !== "undefined" ?
|
|
root.ready( selector ) :
|
|
|
|
// Execute immediately if ready is not present
|
|
selector( jQuery );
|
|
}
|
|
|
|
if ( selector.selector !== undefined ) {
|
|
this.selector = selector.selector;
|
|
this.context = selector.context;
|
|
}
|
|
|
|
return jQuery.makeArray( selector, this );
|
|
};
|
|
|
|
// Give the init function the jQuery prototype for later instantiation
|
|
init.prototype = jQuery.fn;
|
|
|
|
// Initialize central reference
|
|
rootjQuery = jQuery( document );
|
|
|
|
|
|
var rparentsprev = /^(?:parents|prev(?:Until|All))/,
|
|
|
|
// methods guaranteed to produce a unique set when starting from a unique set
|
|
guaranteedUnique = {
|
|
children: true,
|
|
contents: true,
|
|
next: true,
|
|
prev: true
|
|
};
|
|
|
|
jQuery.fn.extend( {
|
|
has: function( target ) {
|
|
var i,
|
|
targets = jQuery( target, this ),
|
|
len = targets.length;
|
|
|
|
return this.filter( function() {
|
|
for ( i = 0; i < len; i++ ) {
|
|
if ( jQuery.contains( this, targets[ i ] ) ) {
|
|
return true;
|
|
}
|
|
}
|
|
} );
|
|
},
|
|
|
|
closest: function( selectors, context ) {
|
|
var cur,
|
|
i = 0,
|
|
l = this.length,
|
|
matched = [],
|
|
pos = rneedsContext.test( selectors ) || typeof selectors !== "string" ?
|
|
jQuery( selectors, context || this.context ) :
|
|
0;
|
|
|
|
for ( ; i < l; i++ ) {
|
|
for ( cur = this[ i ]; cur && cur !== context; cur = cur.parentNode ) {
|
|
|
|
// Always skip document fragments
|
|
if ( cur.nodeType < 11 && ( pos ?
|
|
pos.index( cur ) > -1 :
|
|
|
|
// Don't pass non-elements to Sizzle
|
|
cur.nodeType === 1 &&
|
|
jQuery.find.matchesSelector( cur, selectors ) ) ) {
|
|
|
|
matched.push( cur );
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
return this.pushStack( matched.length > 1 ? jQuery.uniqueSort( matched ) : matched );
|
|
},
|
|
|
|
// Determine the position of an element within
|
|
// the matched set of elements
|
|
index: function( elem ) {
|
|
|
|
// No argument, return index in parent
|
|
if ( !elem ) {
|
|
return ( this[ 0 ] && this[ 0 ].parentNode ) ? this.first().prevAll().length : -1;
|
|
}
|
|
|
|
// index in selector
|
|
if ( typeof elem === "string" ) {
|
|
return jQuery.inArray( this[ 0 ], jQuery( elem ) );
|
|
}
|
|
|
|
// Locate the position of the desired element
|
|
return jQuery.inArray(
|
|
|
|
// If it receives a jQuery object, the first element is used
|
|
elem.jquery ? elem[ 0 ] : elem, this );
|
|
},
|
|
|
|
add: function( selector, context ) {
|
|
return this.pushStack(
|
|
jQuery.uniqueSort(
|
|
jQuery.merge( this.get(), jQuery( selector, context ) )
|
|
)
|
|
);
|
|
},
|
|
|
|
addBack: function( selector ) {
|
|
return this.add( selector == null ?
|
|
this.prevObject : this.prevObject.filter( selector )
|
|
);
|
|
}
|
|
} );
|
|
|
|
function sibling( cur, dir ) {
|
|
do {
|
|
cur = cur[ dir ];
|
|
} while ( cur && cur.nodeType !== 1 );
|
|
|
|
return cur;
|
|
}
|
|
|
|
jQuery.each( {
|
|
parent: function( elem ) {
|
|
var parent = elem.parentNode;
|
|
return parent && parent.nodeType !== 11 ? parent : null;
|
|
},
|
|
parents: function( elem ) {
|
|
return dir( elem, "parentNode" );
|
|
},
|
|
parentsUntil: function( elem, i, until ) {
|
|
return dir( elem, "parentNode", until );
|
|
},
|
|
next: function( elem ) {
|
|
return sibling( elem, "nextSibling" );
|
|
},
|
|
prev: function( elem ) {
|
|
return sibling( elem, "previousSibling" );
|
|
},
|
|
nextAll: function( elem ) {
|
|
return dir( elem, "nextSibling" );
|
|
},
|
|
prevAll: function( elem ) {
|
|
return dir( elem, "previousSibling" );
|
|
},
|
|
nextUntil: function( elem, i, until ) {
|
|
return dir( elem, "nextSibling", until );
|
|
},
|
|
prevUntil: function( elem, i, until ) {
|
|
return dir( elem, "previousSibling", until );
|
|
},
|
|
siblings: function( elem ) {
|
|
return siblings( ( elem.parentNode || {} ).firstChild, elem );
|
|
},
|
|
children: function( elem ) {
|
|
return siblings( elem.firstChild );
|
|
},
|
|
contents: function( elem ) {
|
|
return jQuery.nodeName( elem, "iframe" ) ?
|
|
elem.contentDocument || elem.contentWindow.document :
|
|
jQuery.merge( [], elem.childNodes );
|
|
}
|
|
}, function( name, fn ) {
|
|
jQuery.fn[ name ] = function( until, selector ) {
|
|
var ret = jQuery.map( this, fn, until );
|
|
|
|
if ( name.slice( -5 ) !== "Until" ) {
|
|
selector = until;
|
|
}
|
|
|
|
if ( selector && typeof selector === "string" ) {
|
|
ret = jQuery.filter( selector, ret );
|
|
}
|
|
|
|
if ( this.length > 1 ) {
|
|
|
|
// Remove duplicates
|
|
if ( !guaranteedUnique[ name ] ) {
|
|
ret = jQuery.uniqueSort( ret );
|
|
}
|
|
|
|
// Reverse order for parents* and prev-derivatives
|
|
if ( rparentsprev.test( name ) ) {
|
|
ret = ret.reverse();
|
|
}
|
|
}
|
|
|
|
return this.pushStack( ret );
|
|
};
|
|
} );
|
|
var rnotwhite = ( /\S+/g );
|
|
|
|
|
|
|
|
// Convert String-formatted options into Object-formatted ones
|
|
function createOptions( options ) {
|
|
var object = {};
|
|
jQuery.each( options.match( rnotwhite ) || [], function( _, flag ) {
|
|
object[ flag ] = true;
|
|
} );
|
|
return object;
|
|
}
|
|
|
|
/*
|
|
* Create a callback list using the following parameters:
|
|
*
|
|
* options: an optional list of space-separated options that will change how
|
|
* the callback list behaves or a more traditional option object
|
|
*
|
|
* By default a callback list will act like an event callback list and can be
|
|
* "fired" multiple times.
|
|
*
|
|
* Possible options:
|
|
*
|
|
* once: will ensure the callback list can only be fired once (like a Deferred)
|
|
*
|
|
* memory: will keep track of previous values and will call any callback added
|
|
* after the list has been fired right away with the latest "memorized"
|
|
* values (like a Deferred)
|
|
*
|
|
* unique: will ensure a callback can only be added once (no duplicate in the list)
|
|
*
|
|
* stopOnFalse: interrupt callings when a callback returns false
|
|
*
|
|
*/
|
|
jQuery.Callbacks = function( options ) {
|
|
|
|
// Convert options from String-formatted to Object-formatted if needed
|
|
// (we check in cache first)
|
|
options = typeof options === "string" ?
|
|
createOptions( options ) :
|
|
jQuery.extend( {}, options );
|
|
|
|
var // Flag to know if list is currently firing
|
|
firing,
|
|
|
|
// Last fire value for non-forgettable lists
|
|
memory,
|
|
|
|
// Flag to know if list was already fired
|
|
fired,
|
|
|
|
// Flag to prevent firing
|
|
locked,
|
|
|
|
// Actual callback list
|
|
list = [],
|
|
|
|
// Queue of execution data for repeatable lists
|
|
queue = [],
|
|
|
|
// Index of currently firing callback (modified by add/remove as needed)
|
|
firingIndex = -1,
|
|
|
|
// Fire callbacks
|
|
fire = function() {
|
|
|
|
// Enforce single-firing
|
|
locked = options.once;
|
|
|
|
// Execute callbacks for all pending executions,
|
|
// respecting firingIndex overrides and runtime changes
|
|
fired = firing = true;
|
|
for ( ; queue.length; firingIndex = -1 ) {
|
|
memory = queue.shift();
|
|
while ( ++firingIndex < list.length ) {
|
|
|
|
// Run callback and check for early termination
|
|
if ( list[ firingIndex ].apply( memory[ 0 ], memory[ 1 ] ) === false &&
|
|
options.stopOnFalse ) {
|
|
|
|
// Jump to end and forget the data so .add doesn't re-fire
|
|
firingIndex = list.length;
|
|
memory = false;
|
|
}
|
|
}
|
|
}
|
|
|
|
// Forget the data if we're done with it
|
|
if ( !options.memory ) {
|
|
memory = false;
|
|
}
|
|
|
|
firing = false;
|
|
|
|
// Clean up if we're done firing for good
|
|
if ( locked ) {
|
|
|
|
// Keep an empty list if we have data for future add calls
|
|
if ( memory ) {
|
|
list = [];
|
|
|
|
// Otherwise, this object is spent
|
|
} else {
|
|
list = "";
|
|
}
|
|
}
|
|
},
|
|
|
|
// Actual Callbacks object
|
|
self = {
|
|
|
|
// Add a callback or a collection of callbacks to the list
|
|
add: function() {
|
|
if ( list ) {
|
|
|
|
// If we have memory from a past run, we should fire after adding
|
|
if ( memory && !firing ) {
|
|
firingIndex = list.length - 1;
|
|
queue.push( memory );
|
|
}
|
|
|
|
( function add( args ) {
|
|
jQuery.each( args, function( _, arg ) {
|
|
if ( jQuery.isFunction( arg ) ) {
|
|
if ( !options.unique || !self.has( arg ) ) {
|
|
list.push( arg );
|
|
}
|
|
} else if ( arg && arg.length && jQuery.type( arg ) !== "string" ) {
|
|
|
|
// Inspect recursively
|
|
add( arg );
|
|
}
|
|
} );
|
|
} )( arguments );
|
|
|
|
if ( memory && !firing ) {
|
|
fire();
|
|
}
|
|
}
|
|
return this;
|
|
},
|
|
|
|
// Remove a callback from the list
|
|
remove: function() {
|
|
jQuery.each( arguments, function( _, arg ) {
|
|
var index;
|
|
while ( ( index = jQuery.inArray( arg, list, index ) ) > -1 ) {
|
|
list.splice( index, 1 );
|
|
|
|
// Handle firing indexes
|
|
if ( index <= firingIndex ) {
|
|
firingIndex--;
|
|
}
|
|
}
|
|
} );
|
|
return this;
|
|
},
|
|
|
|
// Check if a given callback is in the list.
|
|
// If no argument is given, return whether or not list has callbacks attached.
|
|
has: function( fn ) {
|
|
return fn ?
|
|
jQuery.inArray( fn, list ) > -1 :
|
|
list.length > 0;
|
|
},
|
|
|
|
// Remove all callbacks from the list
|
|
empty: function() {
|
|
if ( list ) {
|
|
list = [];
|
|
}
|
|
return this;
|
|
},
|
|
|
|
// Disable .fire and .add
|
|
// Abort any current/pending executions
|
|
// Clear all callbacks and values
|
|
disable: function() {
|
|
locked = queue = [];
|
|
list = memory = "";
|
|
return this;
|
|
},
|
|
disabled: function() {
|
|
return !list;
|
|
},
|
|
|
|
// Disable .fire
|
|
// Also disable .add unless we have memory (since it would have no effect)
|
|
// Abort any pending executions
|
|
lock: function() {
|
|
locked = true;
|
|
if ( !memory ) {
|
|
self.disable();
|
|
}
|
|
return this;
|
|
},
|
|
locked: function() {
|
|
return !!locked;
|
|
},
|
|
|
|
// Call all callbacks with the given context and arguments
|
|
fireWith: function( context, args ) {
|
|
if ( !locked ) {
|
|
args = args || [];
|
|
args = [ context, args.slice ? args.slice() : args ];
|
|
queue.push( args );
|
|
if ( !firing ) {
|
|
fire();
|
|
}
|
|
}
|
|
return this;
|
|
},
|
|
|
|
// Call all the callbacks with the given arguments
|
|
fire: function() {
|
|
self.fireWith( this, arguments );
|
|
return this;
|
|
},
|
|
|
|
// To know if the callbacks have already been called at least once
|
|
fired: function() {
|
|
return !!fired;
|
|
}
|
|
};
|
|
|
|
return self;
|
|
};
|
|
|
|
|
|
jQuery.extend( {
|
|
|
|
Deferred: function( func ) {
|
|
var tuples = [
|
|
|
|
// action, add listener, listener list, final state
|
|
[ "resolve", "done", jQuery.Callbacks( "once memory" ), "resolved" ],
|
|
[ "reject", "fail", jQuery.Callbacks( "once memory" ), "rejected" ],
|
|
[ "notify", "progress", jQuery.Callbacks( "memory" ) ]
|
|
],
|
|
state = "pending",
|
|
promise = {
|
|
state: function() {
|
|
return state;
|
|
},
|
|
always: function() {
|
|
deferred.done( arguments ).fail( arguments );
|
|
return this;
|
|
},
|
|
then: function( /* fnDone, fnFail, fnProgress */ ) {
|
|
var fns = arguments;
|
|
return jQuery.Deferred( function( newDefer ) {
|
|
jQuery.each( tuples, function( i, tuple ) {
|
|
var fn = jQuery.isFunction( fns[ i ] ) && fns[ i ];
|
|
|
|
// deferred[ done | fail | progress ] for forwarding actions to newDefer
|
|
deferred[ tuple[ 1 ] ]( function() {
|
|
var returned = fn && fn.apply( this, arguments );
|
|
if ( returned && jQuery.isFunction( returned.promise ) ) {
|
|
returned.promise()
|
|
.progress( newDefer.notify )
|
|
.done( newDefer.resolve )
|
|
.fail( newDefer.reject );
|
|
} else {
|
|
newDefer[ tuple[ 0 ] + "With" ](
|
|
this === promise ? newDefer.promise() : this,
|
|
fn ? [ returned ] : arguments
|
|
);
|
|
}
|
|
} );
|
|
} );
|
|
fns = null;
|
|
} ).promise();
|
|
},
|
|
|
|
// Get a promise for this deferred
|
|
// If obj is provided, the promise aspect is added to the object
|
|
promise: function( obj ) {
|
|
return obj != null ? jQuery.extend( obj, promise ) : promise;
|
|
}
|
|
},
|
|
deferred = {};
|
|
|
|
// Keep pipe for back-compat
|
|
promise.pipe = promise.then;
|
|
|
|
// Add list-specific methods
|
|
jQuery.each( tuples, function( i, tuple ) {
|
|
var list = tuple[ 2 ],
|
|
stateString = tuple[ 3 ];
|
|
|
|
// promise[ done | fail | progress ] = list.add
|
|
promise[ tuple[ 1 ] ] = list.add;
|
|
|
|
// Handle state
|
|
if ( stateString ) {
|
|
list.add( function() {
|
|
|
|
// state = [ resolved | rejected ]
|
|
state = stateString;
|
|
|
|
// [ reject_list | resolve_list ].disable; progress_list.lock
|
|
}, tuples[ i ^ 1 ][ 2 ].disable, tuples[ 2 ][ 2 ].lock );
|
|
}
|
|
|
|
// deferred[ resolve | reject | notify ]
|
|
deferred[ tuple[ 0 ] ] = function() {
|
|
deferred[ tuple[ 0 ] + "With" ]( this === deferred ? promise : this, arguments );
|
|
return this;
|
|
};
|
|
deferred[ tuple[ 0 ] + "With" ] = list.fireWith;
|
|
} );
|
|
|
|
// Make the deferred a promise
|
|
promise.promise( deferred );
|
|
|
|
// Call given func if any
|
|
if ( func ) {
|
|
func.call( deferred, deferred );
|
|
}
|
|
|
|
// All done!
|
|
return deferred;
|
|
},
|
|
|
|
// Deferred helper
|
|
when: function( subordinate /* , ..., subordinateN */ ) {
|
|
var i = 0,
|
|
resolveValues = slice.call( arguments ),
|
|
length = resolveValues.length,
|
|
|
|
// the count of uncompleted subordinates
|
|
remaining = length !== 1 ||
|
|
( subordinate && jQuery.isFunction( subordinate.promise ) ) ? length : 0,
|
|
|
|
// the master Deferred.
|
|
// If resolveValues consist of only a single Deferred, just use that.
|
|
deferred = remaining === 1 ? subordinate : jQuery.Deferred(),
|
|
|
|
// Update function for both resolve and progress values
|
|
updateFunc = function( i, contexts, values ) {
|
|
return function( value ) {
|
|
contexts[ i ] = this;
|
|
values[ i ] = arguments.length > 1 ? slice.call( arguments ) : value;
|
|
if ( values === progressValues ) {
|
|
deferred.notifyWith( contexts, values );
|
|
|
|
} else if ( !( --remaining ) ) {
|
|
deferred.resolveWith( contexts, values );
|
|
}
|
|
};
|
|
},
|
|
|
|
progressValues, progressContexts, resolveContexts;
|
|
|
|
// add listeners to Deferred subordinates; treat others as resolved
|
|
if ( length > 1 ) {
|
|
progressValues = new Array( length );
|
|
progressContexts = new Array( length );
|
|
resolveContexts = new Array( length );
|
|
for ( ; i < length; i++ ) {
|
|
if ( resolveValues[ i ] && jQuery.isFunction( resolveValues[ i ].promise ) ) {
|
|
resolveValues[ i ].promise()
|
|
.progress( updateFunc( i, progressContexts, progressValues ) )
|
|
.done( updateFunc( i, resolveContexts, resolveValues ) )
|
|
.fail( deferred.reject );
|
|
} else {
|
|
--remaining;
|
|
}
|
|
}
|
|
}
|
|
|
|
// if we're not waiting on anything, resolve the master
|
|
if ( !remaining ) {
|
|
deferred.resolveWith( resolveContexts, resolveValues );
|
|
}
|
|
|
|
return deferred.promise();
|
|
}
|
|
} );
|
|
|
|
|
|
// The deferred used on DOM ready
|
|
var readyList;
|
|
|
|
jQuery.fn.ready = function( fn ) {
|
|
|
|
// Add the callback
|
|
jQuery.ready.promise().done( fn );
|
|
|
|
return this;
|
|
};
|
|
|
|
jQuery.extend( {
|
|
|
|
// Is the DOM ready to be used? Set to true once it occurs.
|
|
isReady: false,
|
|
|
|
// A counter to track how many items to wait for before
|
|
// the ready event fires. See #6781
|
|
readyWait: 1,
|
|
|
|
// Hold (or release) the ready event
|
|
holdReady: function( hold ) {
|
|
if ( hold ) {
|
|
jQuery.readyWait++;
|
|
} else {
|
|
jQuery.ready( true );
|
|
}
|
|
},
|
|
|
|
// Handle when the DOM is ready
|
|
ready: function( wait ) {
|
|
|
|
// Abort if there are pending holds or we're already ready
|
|
if ( wait === true ? --jQuery.readyWait : jQuery.isReady ) {
|
|
return;
|
|
}
|
|
|
|
// Remember that the DOM is ready
|
|
jQuery.isReady = true;
|
|
|
|
// If a normal DOM Ready event fired, decrement, and wait if need be
|
|
if ( wait !== true && --jQuery.readyWait > 0 ) {
|
|
return;
|
|
}
|
|
|
|
// If there are functions bound, to execute
|
|
readyList.resolveWith( document, [ jQuery ] );
|
|
|
|
// Trigger any bound ready events
|
|
if ( jQuery.fn.triggerHandler ) {
|
|
jQuery( document ).triggerHandler( "ready" );
|
|
jQuery( document ).off( "ready" );
|
|
}
|
|
}
|
|
} );
|
|
|
|
/**
|
|
* Clean-up method for dom ready events
|
|
*/
|
|
function detach() {
|
|
if ( document.addEventListener ) {
|
|
document.removeEventListener( "DOMContentLoaded", completed );
|
|
window.removeEventListener( "load", completed );
|
|
|
|
} else {
|
|
document.detachEvent( "onreadystatechange", completed );
|
|
window.detachEvent( "onload", completed );
|
|
}
|
|
}
|
|
|
|
/**
|
|
* The ready event handler and self cleanup method
|
|
*/
|
|
function completed() {
|
|
|
|
// readyState === "complete" is good enough for us to call the dom ready in oldIE
|
|
if ( document.addEventListener ||
|
|
window.event.type === "load" ||
|
|
document.readyState === "complete" ) {
|
|
|
|
detach();
|
|
jQuery.ready();
|
|
}
|
|
}
|
|
|
|
jQuery.ready.promise = function( obj ) {
|
|
if ( !readyList ) {
|
|
|
|
readyList = jQuery.Deferred();
|
|
|
|
// Catch cases where $(document).ready() is called
|
|
// after the browser event has already occurred.
|
|
// Support: IE6-10
|
|
// Older IE sometimes signals "interactive" too soon
|
|
if ( document.readyState === "complete" ||
|
|
( document.readyState !== "loading" && !document.documentElement.doScroll ) ) {
|
|
|
|
// Handle it asynchronously to allow scripts the opportunity to delay ready
|
|
window.setTimeout( jQuery.ready );
|
|
|
|
// Standards-based browsers support DOMContentLoaded
|
|
} else if ( document.addEventListener ) {
|
|
|
|
// Use the handy event callback
|
|
document.addEventListener( "DOMContentLoaded", completed );
|
|
|
|
// A fallback to window.onload, that will always work
|
|
window.addEventListener( "load", completed );
|
|
|
|
// If IE event model is used
|
|
} else {
|
|
|
|
// Ensure firing before onload, maybe late but safe also for iframes
|
|
document.attachEvent( "onreadystatechange", completed );
|
|
|
|
// A fallback to window.onload, that will always work
|
|
window.attachEvent( "onload", completed );
|
|
|
|
// If IE and not a frame
|
|
// continually check to see if the document is ready
|
|
var top = false;
|
|
|
|
try {
|
|
top = window.frameElement == null && document.documentElement;
|
|
} catch ( e ) {}
|
|
|
|
if ( top && top.doScroll ) {
|
|
( function doScrollCheck() {
|
|
if ( !jQuery.isReady ) {
|
|
|
|
try {
|
|
|
|
// Use the trick by Diego Perini
|
|
// http://javascript.nwbox.com/IEContentLoaded/
|
|
top.doScroll( "left" );
|
|
} catch ( e ) {
|
|
return window.setTimeout( doScrollCheck, 50 );
|
|
}
|
|
|
|
// detach all dom ready events
|
|
detach();
|
|
|
|
// and execute any waiting functions
|
|
jQuery.ready();
|
|
}
|
|
} )();
|
|
}
|
|
}
|
|
}
|
|
return readyList.promise( obj );
|
|
};
|
|
|
|
// Kick off the DOM ready check even if the user does not
|
|
jQuery.ready.promise();
|
|
|
|
|
|
|
|
|
|
// Support: IE<9
|
|
// Iteration over object's inherited properties before its own
|
|
var i;
|
|
for ( i in jQuery( support ) ) {
|
|
break;
|
|
}
|
|
support.ownFirst = i === "0";
|
|
|
|
// Note: most support tests are defined in their respective modules.
|
|
// false until the test is run
|
|
support.inlineBlockNeedsLayout = false;
|
|
|
|
// Execute ASAP in case we need to set body.style.zoom
|
|
jQuery( function() {
|
|
|
|
// Minified: var a,b,c,d
|
|
var val, div, body, container;
|
|
|
|
body = document.getElementsByTagName( "body" )[ 0 ];
|
|
if ( !body || !body.style ) {
|
|
|
|
// Return for frameset docs that don't have a body
|
|
return;
|
|
}
|
|
|
|
// Setup
|
|
div = document.createElement( "div" );
|
|
container = document.createElement( "div" );
|
|
container.style.cssText = "position:absolute;border:0;width:0;height:0;top:0;left:-9999px";
|
|
body.appendChild( container ).appendChild( div );
|
|
|
|
if ( typeof div.style.zoom !== "undefined" ) {
|
|
|
|
// Support: IE<8
|
|
// Check if natively block-level elements act like inline-block
|
|
// elements when setting their display to 'inline' and giving
|
|
// them layout
|
|
div.style.cssText = "display:inline;margin:0;border:0;padding:1px;width:1px;zoom:1";
|
|
|
|
support.inlineBlockNeedsLayout = val = div.offsetWidth === 3;
|
|
if ( val ) {
|
|
|
|
// Prevent IE 6 from affecting layout for positioned elements #11048
|
|
// Prevent IE from shrinking the body in IE 7 mode #12869
|
|
// Support: IE<8
|
|
body.style.zoom = 1;
|
|
}
|
|
}
|
|
|
|
body.removeChild( container );
|
|
} );
|
|
|
|
|
|
( function() {
|
|
var div = document.createElement( "div" );
|
|
|
|
// Support: IE<9
|
|
support.deleteExpando = true;
|
|
try {
|
|
delete div.test;
|
|
} catch ( e ) {
|
|
support.deleteExpando = false;
|
|
}
|
|
|
|
// Null elements to avoid leaks in IE.
|
|
div = null;
|
|
} )();
|
|
var acceptData = function( elem ) {
|
|
var noData = jQuery.noData[ ( elem.nodeName + " " ).toLowerCase() ],
|
|
nodeType = +elem.nodeType || 1;
|
|
|
|
// Do not set data on non-element DOM nodes because it will not be cleared (#8335).
|
|
return nodeType !== 1 && nodeType !== 9 ?
|
|
false :
|
|
|
|
// Nodes accept data unless otherwise specified; rejection can be conditional
|
|
!noData || noData !== true && elem.getAttribute( "classid" ) === noData;
|
|
};
|
|
|
|
|
|
|
|
|
|
var rbrace = /^(?:\{[\w\W]*\}|\[[\w\W]*\])$/,
|
|
rmultiDash = /([A-Z])/g;
|
|
|
|
function dataAttr( elem, key, data ) {
|
|
|
|
// If nothing was found internally, try to fetch any
|
|
// data from the HTML5 data-* attribute
|
|
if ( data === undefined && elem.nodeType === 1 ) {
|
|
|
|
var name = "data-" + key.replace( rmultiDash, "-$1" ).toLowerCase();
|
|
|
|
data = elem.getAttribute( name );
|
|
|
|
if ( typeof data === "string" ) {
|
|
try {
|
|
data = data === "true" ? true :
|
|
data === "false" ? false :
|
|
data === "null" ? null :
|
|
|
|
// Only convert to a number if it doesn't change the string
|
|
+data + "" === data ? +data :
|
|
rbrace.test( data ) ? jQuery.parseJSON( data ) :
|
|
data;
|
|
} catch ( e ) {}
|
|
|
|
// Make sure we set the data so it isn't changed later
|
|
jQuery.data( elem, key, data );
|
|
|
|
} else {
|
|
data = undefined;
|
|
}
|
|
}
|
|
|
|
return data;
|
|
}
|
|
|
|
// checks a cache object for emptiness
|
|
function isEmptyDataObject( obj ) {
|
|
var name;
|
|
for ( name in obj ) {
|
|
|
|
// if the public data object is empty, the private is still empty
|
|
if ( name === "data" && jQuery.isEmptyObject( obj[ name ] ) ) {
|
|
continue;
|
|
}
|
|
if ( name !== "toJSON" ) {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
function internalData( elem, name, data, pvt /* Internal Use Only */ ) {
|
|
if ( !acceptData( elem ) ) {
|
|
return;
|
|
}
|
|
|
|
var ret, thisCache,
|
|
internalKey = jQuery.expando,
|
|
|
|
// We have to handle DOM nodes and JS objects differently because IE6-7
|
|
// can't GC object references properly across the DOM-JS boundary
|
|
isNode = elem.nodeType,
|
|
|
|
// Only DOM nodes need the global jQuery cache; JS object data is
|
|
// attached directly to the object so GC can occur automatically
|
|
cache = isNode ? jQuery.cache : elem,
|
|
|
|
// Only defining an ID for JS objects if its cache already exists allows
|
|
// the code to shortcut on the same path as a DOM node with no cache
|
|
id = isNode ? elem[ internalKey ] : elem[ internalKey ] && internalKey;
|
|
|
|
// Avoid doing any more work than we need to when trying to get data on an
|
|
// object that has no data at all
|
|
if ( ( !id || !cache[ id ] || ( !pvt && !cache[ id ].data ) ) &&
|
|
data === undefined && typeof name === "string" ) {
|
|
return;
|
|
}
|
|
|
|
if ( !id ) {
|
|
|
|
// Only DOM nodes need a new unique ID for each element since their data
|
|
// ends up in the global cache
|
|
if ( isNode ) {
|
|
id = elem[ internalKey ] = deletedIds.pop() || jQuery.guid++;
|
|
} else {
|
|
id = internalKey;
|
|
}
|
|
}
|
|
|
|
if ( !cache[ id ] ) {
|
|
|
|
// Avoid exposing jQuery metadata on plain JS objects when the object
|
|
// is serialized using JSON.stringify
|
|
cache[ id ] = isNode ? {} : { toJSON: jQuery.noop };
|
|
}
|
|
|
|
// An object can be passed to jQuery.data instead of a key/value pair; this gets
|
|
// shallow copied over onto the existing cache
|
|
if ( typeof name === "object" || typeof name === "function" ) {
|
|
if ( pvt ) {
|
|
cache[ id ] = jQuery.extend( cache[ id ], name );
|
|
} else {
|
|
cache[ id ].data = jQuery.extend( cache[ id ].data, name );
|
|
}
|
|
}
|
|
|
|
thisCache = cache[ id ];
|
|
|
|
// jQuery data() is stored in a separate object inside the object's internal data
|
|
// cache in order to avoid key collisions between internal data and user-defined
|
|
// data.
|
|
if ( !pvt ) {
|
|
if ( !thisCache.data ) {
|
|
thisCache.data = {};
|
|
}
|
|
|
|
thisCache = thisCache.data;
|
|
}
|
|
|
|
if ( data !== undefined ) {
|
|
thisCache[ jQuery.camelCase( name ) ] = data;
|
|
}
|
|
|
|
// Check for both converted-to-camel and non-converted data property names
|
|
// If a data property was specified
|
|
if ( typeof name === "string" ) {
|
|
|
|
// First Try to find as-is property data
|
|
ret = thisCache[ name ];
|
|
|
|
// Test for null|undefined property data
|
|
if ( ret == null ) {
|
|
|
|
// Try to find the camelCased property
|
|
ret = thisCache[ jQuery.camelCase( name ) ];
|
|
}
|
|
} else {
|
|
ret = thisCache;
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
function internalRemoveData( elem, name, pvt ) {
|
|
if ( !acceptData( elem ) ) {
|
|
return;
|
|
}
|
|
|
|
var thisCache, i,
|
|
isNode = elem.nodeType,
|
|
|
|
// See jQuery.data for more information
|
|
cache = isNode ? jQuery.cache : elem,
|
|
id = isNode ? elem[ jQuery.expando ] : jQuery.expando;
|
|
|
|
// If there is already no cache entry for this object, there is no
|
|
// purpose in continuing
|
|
if ( !cache[ id ] ) {
|
|
return;
|
|
}
|
|
|
|
if ( name ) {
|
|
|
|
thisCache = pvt ? cache[ id ] : cache[ id ].data;
|
|
|
|
if ( thisCache ) {
|
|
|
|
// Support array or space separated string names for data keys
|
|
if ( !jQuery.isArray( name ) ) {
|
|
|
|
// try the string as a key before any manipulation
|
|
if ( name in thisCache ) {
|
|
name = [ name ];
|
|
} else {
|
|
|
|
// split the camel cased version by spaces unless a key with the spaces exists
|
|
name = jQuery.camelCase( name );
|
|
if ( name in thisCache ) {
|
|
name = [ name ];
|
|
} else {
|
|
name = name.split( " " );
|
|
}
|
|
}
|
|
} else {
|
|
|
|
// If "name" is an array of keys...
|
|
// When data is initially created, via ("key", "val") signature,
|
|
// keys will be converted to camelCase.
|
|
// Since there is no way to tell _how_ a key was added, remove
|
|
// both plain key and camelCase key. #12786
|
|
// This will only penalize the array argument path.
|
|
name = name.concat( jQuery.map( name, jQuery.camelCase ) );
|
|
}
|
|
|
|
i = name.length;
|
|
while ( i-- ) {
|
|
delete thisCache[ name[ i ] ];
|
|
}
|
|
|
|
// If there is no data left in the cache, we want to continue
|
|
// and let the cache object itself get destroyed
|
|
if ( pvt ? !isEmptyDataObject( thisCache ) : !jQuery.isEmptyObject( thisCache ) ) {
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
// See jQuery.data for more information
|
|
if ( !pvt ) {
|
|
delete cache[ id ].data;
|
|
|
|
// Don't destroy the parent cache unless the internal data object
|
|
// had been the only thing left in it
|
|
if ( !isEmptyDataObject( cache[ id ] ) ) {
|
|
return;
|
|
}
|
|
}
|
|
|
|
// Destroy the cache
|
|
if ( isNode ) {
|
|
jQuery.cleanData( [ elem ], true );
|
|
|
|
// Use delete when supported for expandos or `cache` is not a window per isWindow (#10080)
|
|
/* jshint eqeqeq: false */
|
|
} else if ( support.deleteExpando || cache != cache.window ) {
|
|
/* jshint eqeqeq: true */
|
|
delete cache[ id ];
|
|
|
|
// When all else fails, undefined
|
|
} else {
|
|
cache[ id ] = undefined;
|
|
}
|
|
}
|
|
|
|
jQuery.extend( {
|
|
cache: {},
|
|
|
|
// The following elements (space-suffixed to avoid Object.prototype collisions)
|
|
// throw uncatchable exceptions if you attempt to set expando properties
|
|
noData: {
|
|
"applet ": true,
|
|
"embed ": true,
|
|
|
|
// ...but Flash objects (which have this classid) *can* handle expandos
|
|
"object ": "clsid:D27CDB6E-AE6D-11cf-96B8-444553540000"
|
|
},
|
|
|
|
hasData: function( elem ) {
|
|
elem = elem.nodeType ? jQuery.cache[ elem[ jQuery.expando ] ] : elem[ jQuery.expando ];
|
|
return !!elem && !isEmptyDataObject( elem );
|
|
},
|
|
|
|
data: function( elem, name, data ) {
|
|
return internalData( elem, name, data );
|
|
},
|
|
|
|
removeData: function( elem, name ) {
|
|
return internalRemoveData( elem, name );
|
|
},
|
|
|
|
// For internal use only.
|
|
_data: function( elem, name, data ) {
|
|
return internalData( elem, name, data, true );
|
|
},
|
|
|
|
_removeData: function( elem, name ) {
|
|
return internalRemoveData( elem, name, true );
|
|
}
|
|
} );
|
|
|
|
jQuery.fn.extend( {
|
|
data: function( key, value ) {
|
|
var i, name, data,
|
|
elem = this[ 0 ],
|
|
attrs = elem && elem.attributes;
|
|
|
|
// Special expections of .data basically thwart jQuery.access,
|
|
// so implement the relevant behavior ourselves
|
|
|
|
// Gets all values
|
|
if ( key === undefined ) {
|
|
if ( this.length ) {
|
|
data = jQuery.data( elem );
|
|
|
|
if ( elem.nodeType === 1 && !jQuery._data( elem, "parsedAttrs" ) ) {
|
|
i = attrs.length;
|
|
while ( i-- ) {
|
|
|
|
// Support: IE11+
|
|
// The attrs elements can be null (#14894)
|
|
if ( attrs[ i ] ) {
|
|
name = attrs[ i ].name;
|
|
if ( name.indexOf( "data-" ) === 0 ) {
|
|
name = jQuery.camelCase( name.slice( 5 ) );
|
|
dataAttr( elem, name, data[ name ] );
|
|
}
|
|
}
|
|
}
|
|
jQuery._data( elem, "parsedAttrs", true );
|
|
}
|
|
}
|
|
|
|
return data;
|
|
}
|
|
|
|
// Sets multiple values
|
|
if ( typeof key === "object" ) {
|
|
return this.each( function() {
|
|
jQuery.data( this, key );
|
|
} );
|
|
}
|
|
|
|
return arguments.length > 1 ?
|
|
|
|
// Sets one value
|
|
this.each( function() {
|
|
jQuery.data( this, key, value );
|
|
} ) :
|
|
|
|
// Gets one value
|
|
// Try to fetch any internally stored data first
|
|
elem ? dataAttr( elem, key, jQuery.data( elem, key ) ) : undefined;
|
|
},
|
|
|
|
removeData: function( key ) {
|
|
return this.each( function() {
|
|
jQuery.removeData( this, key );
|
|
} );
|
|
}
|
|
} );
|
|
|
|
|
|
jQuery.extend( {
|
|
queue: function( elem, type, data ) {
|
|
var queue;
|
|
|
|
if ( elem ) {
|
|
type = ( type || "fx" ) + "queue";
|
|
queue = jQuery._data( elem, type );
|
|
|
|
// Speed up dequeue by getting out quickly if this is just a lookup
|
|
if ( data ) {
|
|
if ( !queue || jQuery.isArray( data ) ) {
|
|
queue = jQuery._data( elem, type, jQuery.makeArray( data ) );
|
|
} else {
|
|
queue.push( data );
|
|
}
|
|
}
|
|
return queue || [];
|
|
}
|
|
},
|
|
|
|
dequeue: function( elem, type ) {
|
|
type = type || "fx";
|
|
|
|
var queue = jQuery.queue( elem, type ),
|
|
startLength = queue.length,
|
|
fn = queue.shift(),
|
|
hooks = jQuery._queueHooks( elem, type ),
|
|
next = function() {
|
|
jQuery.dequeue( elem, type );
|
|
};
|
|
|
|
// If the fx queue is dequeued, always remove the progress sentinel
|
|
if ( fn === "inprogress" ) {
|
|
fn = queue.shift();
|
|
startLength--;
|
|
}
|
|
|
|
if ( fn ) {
|
|
|
|
// Add a progress sentinel to prevent the fx queue from being
|
|
// automatically dequeued
|
|
if ( type === "fx" ) {
|
|
queue.unshift( "inprogress" );
|
|
}
|
|
|
|
// clear up the last queue stop function
|
|
delete hooks.stop;
|
|
fn.call( elem, next, hooks );
|
|
}
|
|
|
|
if ( !startLength && hooks ) {
|
|
hooks.empty.fire();
|
|
}
|
|
},
|
|
|
|
// not intended for public consumption - generates a queueHooks object,
|
|
// or returns the current one
|
|
_queueHooks: function( elem, type ) {
|
|
var key = type + "queueHooks";
|
|
return jQuery._data( elem, key ) || jQuery._data( elem, key, {
|
|
empty: jQuery.Callbacks( "once memory" ).add( function() {
|
|
jQuery._removeData( elem, type + "queue" );
|
|
jQuery._removeData( elem, key );
|
|
} )
|
|
} );
|
|
}
|
|
} );
|
|
|
|
jQuery.fn.extend( {
|
|
queue: function( type, data ) {
|
|
var setter = 2;
|
|
|
|
if ( typeof type !== "string" ) {
|
|
data = type;
|
|
type = "fx";
|
|
setter--;
|
|
}
|
|
|
|
if ( arguments.length < setter ) {
|
|
return jQuery.queue( this[ 0 ], type );
|
|
}
|
|
|
|
return data === undefined ?
|
|
this :
|
|
this.each( function() {
|
|
var queue = jQuery.queue( this, type, data );
|
|
|
|
// ensure a hooks for this queue
|
|
jQuery._queueHooks( this, type );
|
|
|
|
if ( type === "fx" && queue[ 0 ] !== "inprogress" ) {
|
|
jQuery.dequeue( this, type );
|
|
}
|
|
} );
|
|
},
|
|
dequeue: function( type ) {
|
|
return this.each( function() {
|
|
jQuery.dequeue( this, type );
|
|
} );
|
|
},
|
|
clearQueue: function( type ) {
|
|
return this.queue( type || "fx", [] );
|
|
},
|
|
|
|
// Get a promise resolved when queues of a certain type
|
|
// are emptied (fx is the type by default)
|
|
promise: function( type, obj ) {
|
|
var tmp,
|
|
count = 1,
|
|
defer = jQuery.Deferred(),
|
|
elements = this,
|
|
i = this.length,
|
|
resolve = function() {
|
|
if ( !( --count ) ) {
|
|
defer.resolveWith( elements, [ elements ] );
|
|
}
|
|
};
|
|
|
|
if ( typeof type !== "string" ) {
|
|
obj = type;
|
|
type = undefined;
|
|
}
|
|
type = type || "fx";
|
|
|
|
while ( i-- ) {
|
|
tmp = jQuery._data( elements[ i ], type + "queueHooks" );
|
|
if ( tmp && tmp.empty ) {
|
|
count++;
|
|
tmp.empty.add( resolve );
|
|
}
|
|
}
|
|
resolve();
|
|
return defer.promise( obj );
|
|
}
|
|
} );
|
|
|
|
|
|
( function() {
|
|
var shrinkWrapBlocksVal;
|
|
|
|
support.shrinkWrapBlocks = function() {
|
|
if ( shrinkWrapBlocksVal != null ) {
|
|
return shrinkWrapBlocksVal;
|
|
}
|
|
|
|
// Will be changed later if needed.
|
|
shrinkWrapBlocksVal = false;
|
|
|
|
// Minified: var b,c,d
|
|
var div, body, container;
|
|
|
|
body = document.getElementsByTagName( "body" )[ 0 ];
|
|
if ( !body || !body.style ) {
|
|
|
|
// Test fired too early or in an unsupported environment, exit.
|
|
return;
|
|
}
|
|
|
|
// Setup
|
|
div = document.createElement( "div" );
|
|
container = document.createElement( "div" );
|
|
container.style.cssText = "position:absolute;border:0;width:0;height:0;top:0;left:-9999px";
|
|
body.appendChild( container ).appendChild( div );
|
|
|
|
// Support: IE6
|
|
// Check if elements with layout shrink-wrap their children
|
|
if ( typeof div.style.zoom !== "undefined" ) {
|
|
|
|
// Reset CSS: box-sizing; display; margin; border
|
|
div.style.cssText =
|
|
|
|
// Support: Firefox<29, Android 2.3
|
|
// Vendor-prefix box-sizing
|
|
"-webkit-box-sizing:content-box;-moz-box-sizing:content-box;" +
|
|
"box-sizing:content-box;display:block;margin:0;border:0;" +
|
|
"padding:1px;width:1px;zoom:1";
|
|
div.appendChild( document.createElement( "div" ) ).style.width = "5px";
|
|
shrinkWrapBlocksVal = div.offsetWidth !== 3;
|
|
}
|
|
|
|
body.removeChild( container );
|
|
|
|
return shrinkWrapBlocksVal;
|
|
};
|
|
|
|
} )();
|
|
var pnum = ( /[+-]?(?:\d*\.|)\d+(?:[eE][+-]?\d+|)/ ).source;
|
|
|
|
var rcssNum = new RegExp( "^(?:([+-])=|)(" + pnum + ")([a-z%]*)$", "i" );
|
|
|
|
|
|
var cssExpand = [ "Top", "Right", "Bottom", "Left" ];
|
|
|
|
var isHidden = function( elem, el ) {
|
|
|
|
// isHidden might be called from jQuery#filter function;
|
|
// in that case, element will be second argument
|
|
elem = el || elem;
|
|
return jQuery.css( elem, "display" ) === "none" ||
|
|
!jQuery.contains( elem.ownerDocument, elem );
|
|
};
|
|
|
|
|
|
|
|
function adjustCSS( elem, prop, valueParts, tween ) {
|
|
var adjusted,
|
|
scale = 1,
|
|
maxIterations = 20,
|
|
currentValue = tween ?
|
|
function() { return tween.cur(); } :
|
|
function() { return jQuery.css( elem, prop, "" ); },
|
|
initial = currentValue(),
|
|
unit = valueParts && valueParts[ 3 ] || ( jQuery.cssNumber[ prop ] ? "" : "px" ),
|
|
|
|
// Starting value computation is required for potential unit mismatches
|
|
initialInUnit = ( jQuery.cssNumber[ prop ] || unit !== "px" && +initial ) &&
|
|
rcssNum.exec( jQuery.css( elem, prop ) );
|
|
|
|
if ( initialInUnit && initialInUnit[ 3 ] !== unit ) {
|
|
|
|
// Trust units reported by jQuery.css
|
|
unit = unit || initialInUnit[ 3 ];
|
|
|
|
// Make sure we update the tween properties later on
|
|
valueParts = valueParts || [];
|
|
|
|
// Iteratively approximate from a nonzero starting point
|
|
initialInUnit = +initial || 1;
|
|
|
|
do {
|
|
|
|
// If previous iteration zeroed out, double until we get *something*.
|
|
// Use string for doubling so we don't accidentally see scale as unchanged below
|
|
scale = scale || ".5";
|
|
|
|
// Adjust and apply
|
|
initialInUnit = initialInUnit / scale;
|
|
jQuery.style( elem, prop, initialInUnit + unit );
|
|
|
|
// Update scale, tolerating zero or NaN from tween.cur()
|
|
// Break the loop if scale is unchanged or perfect, or if we've just had enough.
|
|
} while (
|
|
scale !== ( scale = currentValue() / initial ) && scale !== 1 && --maxIterations
|
|
);
|
|
}
|
|
|
|
if ( valueParts ) {
|
|
initialInUnit = +initialInUnit || +initial || 0;
|
|
|
|
// Apply relative offset (+=/-=) if specified
|
|
adjusted = valueParts[ 1 ] ?
|
|
initialInUnit + ( valueParts[ 1 ] + 1 ) * valueParts[ 2 ] :
|
|
+valueParts[ 2 ];
|
|
if ( tween ) {
|
|
tween.unit = unit;
|
|
tween.start = initialInUnit;
|
|
tween.end = adjusted;
|
|
}
|
|
}
|
|
return adjusted;
|
|
}
|
|
|
|
|
|
// Multifunctional method to get and set values of a collection
|
|
// The value/s can optionally be executed if it's a function
|
|
var access = function( elems, fn, key, value, chainable, emptyGet, raw ) {
|
|
var i = 0,
|
|
length = elems.length,
|
|
bulk = key == null;
|
|
|
|
// Sets many values
|
|
if ( jQuery.type( key ) === "object" ) {
|
|
chainable = true;
|
|
for ( i in key ) {
|
|
access( elems, fn, i, key[ i ], true, emptyGet, raw );
|
|
}
|
|
|
|
// Sets one value
|
|
} else if ( value !== undefined ) {
|
|
chainable = true;
|
|
|
|
if ( !jQuery.isFunction( value ) ) {
|
|
raw = true;
|
|
}
|
|
|
|
if ( bulk ) {
|
|
|
|
// Bulk operations run against the entire set
|
|
if ( raw ) {
|
|
fn.call( elems, value );
|
|
fn = null;
|
|
|
|
// ...except when executing function values
|
|
} else {
|
|
bulk = fn;
|
|
fn = function( elem, key, value ) {
|
|
return bulk.call( jQuery( elem ), value );
|
|
};
|
|
}
|
|
}
|
|
|
|
if ( fn ) {
|
|
for ( ; i < length; i++ ) {
|
|
fn(
|
|
elems[ i ],
|
|
key,
|
|
raw ? value : value.call( elems[ i ], i, fn( elems[ i ], key ) )
|
|
);
|
|
}
|
|
}
|
|
}
|
|
|
|
return chainable ?
|
|
elems :
|
|
|
|
// Gets
|
|
bulk ?
|
|
fn.call( elems ) :
|
|
length ? fn( elems[ 0 ], key ) : emptyGet;
|
|
};
|
|
var rcheckableType = ( /^(?:checkbox|radio)$/i );
|
|
|
|
var rtagName = ( /<([\w:-]+)/ );
|
|
|
|
var rscriptType = ( /^$|\/(?:java|ecma)script/i );
|
|
|
|
var rleadingWhitespace = ( /^\s+/ );
|
|
|
|
var nodeNames = "abbr|article|aside|audio|bdi|canvas|data|datalist|" +
|
|
"details|dialog|figcaption|figure|footer|header|hgroup|main|" +
|
|
"mark|meter|nav|output|picture|progress|section|summary|template|time|video";
|
|
|
|
|
|
|
|
function createSafeFragment( document ) {
|
|
var list = nodeNames.split( "|" ),
|
|
safeFrag = document.createDocumentFragment();
|
|
|
|
if ( safeFrag.createElement ) {
|
|
while ( list.length ) {
|
|
safeFrag.createElement(
|
|
list.pop()
|
|
);
|
|
}
|
|
}
|
|
return safeFrag;
|
|
}
|
|
|
|
|
|
( function() {
|
|
var div = document.createElement( "div" ),
|
|
fragment = document.createDocumentFragment(),
|
|
input = document.createElement( "input" );
|
|
|
|
// Setup
|
|
div.innerHTML = " <link/><table></table><a href='/a'>a</a><input type='checkbox'/>";
|
|
|
|
// IE strips leading whitespace when .innerHTML is used
|
|
support.leadingWhitespace = div.firstChild.nodeType === 3;
|
|
|
|
// Make sure that tbody elements aren't automatically inserted
|
|
// IE will insert them into empty tables
|
|
support.tbody = !div.getElementsByTagName( "tbody" ).length;
|
|
|
|
// Make sure that link elements get serialized correctly by innerHTML
|
|
// This requires a wrapper element in IE
|
|
support.htmlSerialize = !!div.getElementsByTagName( "link" ).length;
|
|
|
|
// Makes sure cloning an html5 element does not cause problems
|
|
// Where outerHTML is undefined, this still works
|
|
support.html5Clone =
|
|
document.createElement( "nav" ).cloneNode( true ).outerHTML !== "<:nav></:nav>";
|
|
|
|
// Check if a disconnected checkbox will retain its checked
|
|
// value of true after appended to the DOM (IE6/7)
|
|
input.type = "checkbox";
|
|
input.checked = true;
|
|
fragment.appendChild( input );
|
|
support.appendChecked = input.checked;
|
|
|
|
// Make sure textarea (and checkbox) defaultValue is properly cloned
|
|
// Support: IE6-IE11+
|
|
div.innerHTML = "<textarea>x</textarea>";
|
|
support.noCloneChecked = !!div.cloneNode( true ).lastChild.defaultValue;
|
|
|
|
// #11217 - WebKit loses check when the name is after the checked attribute
|
|
fragment.appendChild( div );
|
|
|
|
// Support: Windows Web Apps (WWA)
|
|
// `name` and `type` must use .setAttribute for WWA (#14901)
|
|
input = document.createElement( "input" );
|
|
input.setAttribute( "type", "radio" );
|
|
input.setAttribute( "checked", "checked" );
|
|
input.setAttribute( "name", "t" );
|
|
|
|
div.appendChild( input );
|
|
|
|
// Support: Safari 5.1, iOS 5.1, Android 4.x, Android 2.3
|
|
// old WebKit doesn't clone checked state correctly in fragments
|
|
support.checkClone = div.cloneNode( true ).cloneNode( true ).lastChild.checked;
|
|
|
|
// Support: IE<9
|
|
// Cloned elements keep attachEvent handlers, we use addEventListener on IE9+
|
|
support.noCloneEvent = !!div.addEventListener;
|
|
|
|
// Support: IE<9
|
|
// Since attributes and properties are the same in IE,
|
|
// cleanData must set properties to undefined rather than use removeAttribute
|
|
div[ jQuery.expando ] = 1;
|
|
support.attributes = !div.getAttribute( jQuery.expando );
|
|
} )();
|
|
|
|
|
|
// We have to close these tags to support XHTML (#13200)
|
|
var wrapMap = {
|
|
option: [ 1, "<select multiple='multiple'>", "</select>" ],
|
|
legend: [ 1, "<fieldset>", "</fieldset>" ],
|
|
area: [ 1, "<map>", "</map>" ],
|
|
|
|
// Support: IE8
|
|
param: [ 1, "<object>", "</object>" ],
|
|
thead: [ 1, "<table>", "</table>" ],
|
|
tr: [ 2, "<table><tbody>", "</tbody></table>" ],
|
|
col: [ 2, "<table><tbody></tbody><colgroup>", "</colgroup></table>" ],
|
|
td: [ 3, "<table><tbody><tr>", "</tr></tbody></table>" ],
|
|
|
|
// IE6-8 can't serialize link, script, style, or any html5 (NoScope) tags,
|
|
// unless wrapped in a div with non-breaking characters in front of it.
|
|
_default: support.htmlSerialize ? [ 0, "", "" ] : [ 1, "X<div>", "</div>" ]
|
|
};
|
|
|
|
// Support: IE8-IE9
|
|
wrapMap.optgroup = wrapMap.option;
|
|
|
|
wrapMap.tbody = wrapMap.tfoot = wrapMap.colgroup = wrapMap.caption = wrapMap.thead;
|
|
wrapMap.th = wrapMap.td;
|
|
|
|
|
|
function getAll( context, tag ) {
|
|
var elems, elem,
|
|
i = 0,
|
|
found = typeof context.getElementsByTagName !== "undefined" ?
|
|
context.getElementsByTagName( tag || "*" ) :
|
|
typeof context.querySelectorAll !== "undefined" ?
|
|
context.querySelectorAll( tag || "*" ) :
|
|
undefined;
|
|
|
|
if ( !found ) {
|
|
for ( found = [], elems = context.childNodes || context;
|
|
( elem = elems[ i ] ) != null;
|
|
i++
|
|
) {
|
|
if ( !tag || jQuery.nodeName( elem, tag ) ) {
|
|
found.push( elem );
|
|
} else {
|
|
jQuery.merge( found, getAll( elem, tag ) );
|
|
}
|
|
}
|
|
}
|
|
|
|
return tag === undefined || tag && jQuery.nodeName( context, tag ) ?
|
|
jQuery.merge( [ context ], found ) :
|
|
found;
|
|
}
|
|
|
|
|
|
// Mark scripts as having already been evaluated
|
|
function setGlobalEval( elems, refElements ) {
|
|
var elem,
|
|
i = 0;
|
|
for ( ; ( elem = elems[ i ] ) != null; i++ ) {
|
|
jQuery._data(
|
|
elem,
|
|
"globalEval",
|
|
!refElements || jQuery._data( refElements[ i ], "globalEval" )
|
|
);
|
|
}
|
|
}
|
|
|
|
|
|
var rhtml = /<|&#?\w+;/,
|
|
rtbody = /<tbody/i;
|
|
|
|
function fixDefaultChecked( elem ) {
|
|
if ( rcheckableType.test( elem.type ) ) {
|
|
elem.defaultChecked = elem.checked;
|
|
}
|
|
}
|
|
|
|
function buildFragment( elems, context, scripts, selection, ignored ) {
|
|
var j, elem, contains,
|
|
tmp, tag, tbody, wrap,
|
|
l = elems.length,
|
|
|
|
// Ensure a safe fragment
|
|
safe = createSafeFragment( context ),
|
|
|
|
nodes = [],
|
|
i = 0;
|
|
|
|
for ( ; i < l; i++ ) {
|
|
elem = elems[ i ];
|
|
|
|
if ( elem || elem === 0 ) {
|
|
|
|
// Add nodes directly
|
|
if ( jQuery.type( elem ) === "object" ) {
|
|
jQuery.merge( nodes, elem.nodeType ? [ elem ] : elem );
|
|
|
|
// Convert non-html into a text node
|
|
} else if ( !rhtml.test( elem ) ) {
|
|
nodes.push( context.createTextNode( elem ) );
|
|
|
|
// Convert html into DOM nodes
|
|
} else {
|
|
tmp = tmp || safe.appendChild( context.createElement( "div" ) );
|
|
|
|
// Deserialize a standard representation
|
|
tag = ( rtagName.exec( elem ) || [ "", "" ] )[ 1 ].toLowerCase();
|
|
wrap = wrapMap[ tag ] || wrapMap._default;
|
|
|
|
tmp.innerHTML = wrap[ 1 ] + jQuery.htmlPrefilter( elem ) + wrap[ 2 ];
|
|
|
|
// Descend through wrappers to the right content
|
|
j = wrap[ 0 ];
|
|
while ( j-- ) {
|
|
tmp = tmp.lastChild;
|
|
}
|
|
|
|
// Manually add leading whitespace removed by IE
|
|
if ( !support.leadingWhitespace && rleadingWhitespace.test( elem ) ) {
|
|
nodes.push( context.createTextNode( rleadingWhitespace.exec( elem )[ 0 ] ) );
|
|
}
|
|
|
|
// Remove IE's autoinserted <tbody> from table fragments
|
|
if ( !support.tbody ) {
|
|
|
|
// String was a <table>, *may* have spurious <tbody>
|
|
elem = tag === "table" && !rtbody.test( elem ) ?
|
|
tmp.firstChild :
|
|
|
|
// String was a bare <thead> or <tfoot>
|
|
wrap[ 1 ] === "<table>" && !rtbody.test( elem ) ?
|
|
tmp :
|
|
0;
|
|
|
|
j = elem && elem.childNodes.length;
|
|
while ( j-- ) {
|
|
if ( jQuery.nodeName( ( tbody = elem.childNodes[ j ] ), "tbody" ) &&
|
|
!tbody.childNodes.length ) {
|
|
|
|
elem.removeChild( tbody );
|
|
}
|
|
}
|
|
}
|
|
|
|
jQuery.merge( nodes, tmp.childNodes );
|
|
|
|
// Fix #12392 for WebKit and IE > 9
|
|
tmp.textContent = "";
|
|
|
|
// Fix #12392 for oldIE
|
|
while ( tmp.firstChild ) {
|
|
tmp.removeChild( tmp.firstChild );
|
|
}
|
|
|
|
// Remember the top-level container for proper cleanup
|
|
tmp = safe.lastChild;
|
|
}
|
|
}
|
|
}
|
|
|
|
// Fix #11356: Clear elements from fragment
|
|
if ( tmp ) {
|
|
safe.removeChild( tmp );
|
|
}
|
|
|
|
// Reset defaultChecked for any radios and checkboxes
|
|
// about to be appended to the DOM in IE 6/7 (#8060)
|
|
if ( !support.appendChecked ) {
|
|
jQuery.grep( getAll( nodes, "input" ), fixDefaultChecked );
|
|
}
|
|
|
|
i = 0;
|
|
while ( ( elem = nodes[ i++ ] ) ) {
|
|
|
|
// Skip elements already in the context collection (trac-4087)
|
|
if ( selection && jQuery.inArray( elem, selection ) > -1 ) {
|
|
if ( ignored ) {
|
|
ignored.push( elem );
|
|
}
|
|
|
|
continue;
|
|
}
|
|
|
|
contains = jQuery.contains( elem.ownerDocument, elem );
|
|
|
|
// Append to fragment
|
|
tmp = getAll( safe.appendChild( elem ), "script" );
|
|
|
|
// Preserve script evaluation history
|
|
if ( contains ) {
|
|
setGlobalEval( tmp );
|
|
}
|
|
|
|
// Capture executables
|
|
if ( scripts ) {
|
|
j = 0;
|
|
while ( ( elem = tmp[ j++ ] ) ) {
|
|
if ( rscriptType.test( elem.type || "" ) ) {
|
|
scripts.push( elem );
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
tmp = null;
|
|
|
|
return safe;
|
|
}
|
|
|
|
|
|
( function() {
|
|
var i, eventName,
|
|
div = document.createElement( "div" );
|
|
|
|
// Support: IE<9 (lack submit/change bubble), Firefox (lack focus(in | out) events)
|
|
for ( i in { submit: true, change: true, focusin: true } ) {
|
|
eventName = "on" + i;
|
|
|
|
if ( !( support[ i ] = eventName in window ) ) {
|
|
|
|
// Beware of CSP restrictions (https://developer.mozilla.org/en/Security/CSP)
|
|
div.setAttribute( eventName, "t" );
|
|
support[ i ] = div.attributes[ eventName ].expando === false;
|
|
}
|
|
}
|
|
|
|
// Null elements to avoid leaks in IE.
|
|
div = null;
|
|
} )();
|
|
|
|
|
|
var rformElems = /^(?:input|select|textarea)$/i,
|
|
rkeyEvent = /^key/,
|
|
rmouseEvent = /^(?:mouse|pointer|contextmenu|drag|drop)|click/,
|
|
rfocusMorph = /^(?:focusinfocus|focusoutblur)$/,
|
|
rtypenamespace = /^([^.]*)(?:\.(.+)|)/;
|
|
|
|
function returnTrue() {
|
|
return true;
|
|
}
|
|
|
|
function returnFalse() {
|
|
return false;
|
|
}
|
|
|
|
// Support: IE9
|
|
// See #13393 for more info
|
|
function safeActiveElement() {
|
|
try {
|
|
return document.activeElement;
|
|
} catch ( err ) { }
|
|
}
|
|
|
|
function on( elem, types, selector, data, fn, one ) {
|
|
var origFn, type;
|
|
|
|
// Types can be a map of types/handlers
|
|
if ( typeof types === "object" ) {
|
|
|
|
// ( types-Object, selector, data )
|
|
if ( typeof selector !== "string" ) {
|
|
|
|
// ( types-Object, data )
|
|
data = data || selector;
|
|
selector = undefined;
|
|
}
|
|
for ( type in types ) {
|
|
on( elem, type, selector, data, types[ type ], one );
|
|
}
|
|
return elem;
|
|
}
|
|
|
|
if ( data == null && fn == null ) {
|
|
|
|
// ( types, fn )
|
|
fn = selector;
|
|
data = selector = undefined;
|
|
} else if ( fn == null ) {
|
|
if ( typeof selector === "string" ) {
|
|
|
|
// ( types, selector, fn )
|
|
fn = data;
|
|
data = undefined;
|
|
} else {
|
|
|
|
// ( types, data, fn )
|
|
fn = data;
|
|
data = selector;
|
|
selector = undefined;
|
|
}
|
|
}
|
|
if ( fn === false ) {
|
|
fn = returnFalse;
|
|
} else if ( !fn ) {
|
|
return elem;
|
|
}
|
|
|
|
if ( one === 1 ) {
|
|
origFn = fn;
|
|
fn = function( event ) {
|
|
|
|
// Can use an empty set, since event contains the info
|
|
jQuery().off( event );
|
|
return origFn.apply( this, arguments );
|
|
};
|
|
|
|
// Use same guid so caller can remove using origFn
|
|
fn.guid = origFn.guid || ( origFn.guid = jQuery.guid++ );
|
|
}
|
|
return elem.each( function() {
|
|
jQuery.event.add( this, types, fn, data, selector );
|
|
} );
|
|
}
|
|
|
|
/*
|
|
* Helper functions for managing events -- not part of the public interface.
|
|
* Props to Dean Edwards' addEvent library for many of the ideas.
|
|
*/
|
|
jQuery.event = {
|
|
|
|
global: {},
|
|
|
|
add: function( elem, types, handler, data, selector ) {
|
|
var tmp, events, t, handleObjIn,
|
|
special, eventHandle, handleObj,
|
|
handlers, type, namespaces, origType,
|
|
elemData = jQuery._data( elem );
|
|
|
|
// Don't attach events to noData or text/comment nodes (but allow plain objects)
|
|
if ( !elemData ) {
|
|
return;
|
|
}
|
|
|
|
// Caller can pass in an object of custom data in lieu of the handler
|
|
if ( handler.handler ) {
|
|
handleObjIn = handler;
|
|
handler = handleObjIn.handler;
|
|
selector = handleObjIn.selector;
|
|
}
|
|
|
|
// Make sure that the handler has a unique ID, used to find/remove it later
|
|
if ( !handler.guid ) {
|
|
handler.guid = jQuery.guid++;
|
|
}
|
|
|
|
// Init the element's event structure and main handler, if this is the first
|
|
if ( !( events = elemData.events ) ) {
|
|
events = elemData.events = {};
|
|
}
|
|
if ( !( eventHandle = elemData.handle ) ) {
|
|
eventHandle = elemData.handle = function( e ) {
|
|
|
|
// Discard the second event of a jQuery.event.trigger() and
|
|
// when an event is called after a page has unloaded
|
|
return typeof jQuery !== "undefined" &&
|
|
( !e || jQuery.event.triggered !== e.type ) ?
|
|
jQuery.event.dispatch.apply( eventHandle.elem, arguments ) :
|
|
undefined;
|
|
};
|
|
|
|
// Add elem as a property of the handle fn to prevent a memory leak
|
|
// with IE non-native events
|
|
eventHandle.elem = elem;
|
|
}
|
|
|
|
// Handle multiple events separated by a space
|
|
types = ( types || "" ).match( rnotwhite ) || [ "" ];
|
|
t = types.length;
|
|
while ( t-- ) {
|
|
tmp = rtypenamespace.exec( types[ t ] ) || [];
|
|
type = origType = tmp[ 1 ];
|
|
namespaces = ( tmp[ 2 ] || "" ).split( "." ).sort();
|
|
|
|
// There *must* be a type, no attaching namespace-only handlers
|
|
if ( !type ) {
|
|
continue;
|
|
}
|
|
|
|
// If event changes its type, use the special event handlers for the changed type
|
|
special = jQuery.event.special[ type ] || {};
|
|
|
|
// If selector defined, determine special event api type, otherwise given type
|
|
type = ( selector ? special.delegateType : special.bindType ) || type;
|
|
|
|
// Update special based on newly reset type
|
|
special = jQuery.event.special[ type ] || {};
|
|
|
|
// handleObj is passed to all event handlers
|
|
handleObj = jQuery.extend( {
|
|
type: type,
|
|
origType: origType,
|
|
data: data,
|
|
handler: handler,
|
|
guid: handler.guid,
|
|
selector: selector,
|
|
needsContext: selector && jQuery.expr.match.needsContext.test( selector ),
|
|
namespace: namespaces.join( "." )
|
|
}, handleObjIn );
|
|
|
|
// Init the event handler queue if we're the first
|
|
if ( !( handlers = events[ type ] ) ) {
|
|
handlers = events[ type ] = [];
|
|
handlers.delegateCount = 0;
|
|
|
|
// Only use addEventListener/attachEvent if the special events handler returns false
|
|
if ( !special.setup ||
|
|
special.setup.call( elem, data, namespaces, eventHandle ) === false ) {
|
|
|
|
// Bind the global event handler to the element
|
|
if ( elem.addEventListener ) {
|
|
elem.addEventListener( type, eventHandle, false );
|
|
|
|
} else if ( elem.attachEvent ) {
|
|
elem.attachEvent( "on" + type, eventHandle );
|
|
}
|
|
}
|
|
}
|
|
|
|
if ( special.add ) {
|
|
special.add.call( elem, handleObj );
|
|
|
|
if ( !handleObj.handler.guid ) {
|
|
handleObj.handler.guid = handler.guid;
|
|
}
|
|
}
|
|
|
|
// Add to the element's handler list, delegates in front
|
|
if ( selector ) {
|
|
handlers.splice( handlers.delegateCount++, 0, handleObj );
|
|
} else {
|
|
handlers.push( handleObj );
|
|
}
|
|
|
|
// Keep track of which events have ever been used, for event optimization
|
|
jQuery.event.global[ type ] = true;
|
|
}
|
|
|
|
// Nullify elem to prevent memory leaks in IE
|
|
elem = null;
|
|
},
|
|
|
|
// Detach an event or set of events from an element
|
|
remove: function( elem, types, handler, selector, mappedTypes ) {
|
|
var j, handleObj, tmp,
|
|
origCount, t, events,
|
|
special, handlers, type,
|
|
namespaces, origType,
|
|
elemData = jQuery.hasData( elem ) && jQuery._data( elem );
|
|
|
|
if ( !elemData || !( events = elemData.events ) ) {
|
|
return;
|
|
}
|
|
|
|
// Once for each type.namespace in types; type may be omitted
|
|
types = ( types || "" ).match( rnotwhite ) || [ "" ];
|
|
t = types.length;
|
|
while ( t-- ) {
|
|
tmp = rtypenamespace.exec( types[ t ] ) || [];
|
|
type = origType = tmp[ 1 ];
|
|
namespaces = ( tmp[ 2 ] || "" ).split( "." ).sort();
|
|
|
|
// Unbind all events (on this namespace, if provided) for the element
|
|
if ( !type ) {
|
|
for ( type in events ) {
|
|
jQuery.event.remove( elem, type + types[ t ], handler, selector, true );
|
|
}
|
|
continue;
|
|
}
|
|
|
|
special = jQuery.event.special[ type ] || {};
|
|
type = ( selector ? special.delegateType : special.bindType ) || type;
|
|
handlers = events[ type ] || [];
|
|
tmp = tmp[ 2 ] &&
|
|
new RegExp( "(^|\\.)" + namespaces.join( "\\.(?:.*\\.|)" ) + "(\\.|$)" );
|
|
|
|
// Remove matching events
|
|
origCount = j = handlers.length;
|
|
while ( j-- ) {
|
|
handleObj = handlers[ j ];
|
|
|
|
if ( ( mappedTypes || origType === handleObj.origType ) &&
|
|
( !handler || handler.guid === handleObj.guid ) &&
|
|
( !tmp || tmp.test( handleObj.namespace ) ) &&
|
|
( !selector || selector === handleObj.selector ||
|
|
selector === "**" && handleObj.selector ) ) {
|
|
handlers.splice( j, 1 );
|
|
|
|
if ( handleObj.selector ) {
|
|
handlers.delegateCount--;
|
|
}
|
|
if ( special.remove ) {
|
|
special.remove.call( elem, handleObj );
|
|
}
|
|
}
|
|
}
|
|
|
|
// Remove generic event handler if we removed something and no more handlers exist
|
|
// (avoids potential for endless recursion during removal of special event handlers)
|
|
if ( origCount && !handlers.length ) {
|
|
if ( !special.teardown ||
|
|
special.teardown.call( elem, namespaces, elemData.handle ) === false ) {
|
|
|
|
jQuery.removeEvent( elem, type, elemData.handle );
|
|
}
|
|
|
|
delete events[ type ];
|
|
}
|
|
}
|
|
|
|
// Remove the expando if it's no longer used
|
|
if ( jQuery.isEmptyObject( events ) ) {
|
|
delete elemData.handle;
|
|
|
|
// removeData also checks for emptiness and clears the expando if empty
|
|
// so use it instead of delete
|
|
jQuery._removeData( elem, "events" );
|
|
}
|
|
},
|
|
|
|
trigger: function( event, data, elem, onlyHandlers ) {
|
|
var handle, ontype, cur,
|
|
bubbleType, special, tmp, i,
|
|
eventPath = [ elem || document ],
|
|
type = hasOwn.call( event, "type" ) ? event.type : event,
|
|
namespaces = hasOwn.call( event, "namespace" ) ? event.namespace.split( "." ) : [];
|
|
|
|
cur = tmp = elem = elem || document;
|
|
|
|
// Don't do events on text and comment nodes
|
|
if ( elem.nodeType === 3 || elem.nodeType === 8 ) {
|
|
return;
|
|
}
|
|
|
|
// focus/blur morphs to focusin/out; ensure we're not firing them right now
|
|
if ( rfocusMorph.test( type + jQuery.event.triggered ) ) {
|
|
return;
|
|
}
|
|
|
|
if ( type.indexOf( "." ) > -1 ) {
|
|
|
|
// Namespaced trigger; create a regexp to match event type in handle()
|
|
namespaces = type.split( "." );
|
|
type = namespaces.shift();
|
|
namespaces.sort();
|
|
}
|
|
ontype = type.indexOf( ":" ) < 0 && "on" + type;
|
|
|
|
// Caller can pass in a jQuery.Event object, Object, or just an event type string
|
|
event = event[ jQuery.expando ] ?
|
|
event :
|
|
new jQuery.Event( type, typeof event === "object" && event );
|
|
|
|
// Trigger bitmask: & 1 for native handlers; & 2 for jQuery (always true)
|
|
event.isTrigger = onlyHandlers ? 2 : 3;
|
|
event.namespace = namespaces.join( "." );
|
|
event.rnamespace = event.namespace ?
|
|
new RegExp( "(^|\\.)" + namespaces.join( "\\.(?:.*\\.|)" ) + "(\\.|$)" ) :
|
|
null;
|
|
|
|
// Clean up the event in case it is being reused
|
|
event.result = undefined;
|
|
if ( !event.target ) {
|
|
event.target = elem;
|
|
}
|
|
|
|
// Clone any incoming data and prepend the event, creating the handler arg list
|
|
data = data == null ?
|
|
[ event ] :
|
|
jQuery.makeArray( data, [ event ] );
|
|
|
|
// Allow special events to draw outside the lines
|
|
special = jQuery.event.special[ type ] || {};
|
|
if ( !onlyHandlers && special.trigger && special.trigger.apply( elem, data ) === false ) {
|
|
return;
|
|
}
|
|
|
|
// Determine event propagation path in advance, per W3C events spec (#9951)
|
|
// Bubble up to document, then to window; watch for a global ownerDocument var (#9724)
|
|
if ( !onlyHandlers && !special.noBubble && !jQuery.isWindow( elem ) ) {
|
|
|
|
bubbleType = special.delegateType || type;
|
|
if ( !rfocusMorph.test( bubbleType + type ) ) {
|
|
cur = cur.parentNode;
|
|
}
|
|
for ( ; cur; cur = cur.parentNode ) {
|
|
eventPath.push( cur );
|
|
tmp = cur;
|
|
}
|
|
|
|
// Only add window if we got to document (e.g., not plain obj or detached DOM)
|
|
if ( tmp === ( elem.ownerDocument || document ) ) {
|
|
eventPath.push( tmp.defaultView || tmp.parentWindow || window );
|
|
}
|
|
}
|
|
|
|
// Fire handlers on the event path
|
|
i = 0;
|
|
while ( ( cur = eventPath[ i++ ] ) && !event.isPropagationStopped() ) {
|
|
|
|
event.type = i > 1 ?
|
|
bubbleType :
|
|
special.bindType || type;
|
|
|
|
// jQuery handler
|
|
handle = ( jQuery._data( cur, "events" ) || {} )[ event.type ] &&
|
|
jQuery._data( cur, "handle" );
|
|
|
|
if ( handle ) {
|
|
handle.apply( cur, data );
|
|
}
|
|
|
|
// Native handler
|
|
handle = ontype && cur[ ontype ];
|
|
if ( handle && handle.apply && acceptData( cur ) ) {
|
|
event.result = handle.apply( cur, data );
|
|
if ( event.result === false ) {
|
|
event.preventDefault();
|
|
}
|
|
}
|
|
}
|
|
event.type = type;
|
|
|
|
// If nobody prevented the default action, do it now
|
|
if ( !onlyHandlers && !event.isDefaultPrevented() ) {
|
|
|
|
if (
|
|
( !special._default ||
|
|
special._default.apply( eventPath.pop(), data ) === false
|
|
) && acceptData( elem )
|
|
) {
|
|
|
|
// Call a native DOM method on the target with the same name name as the event.
|
|
// Can't use an .isFunction() check here because IE6/7 fails that test.
|
|
// Don't do default actions on window, that's where global variables be (#6170)
|
|
if ( ontype && elem[ type ] && !jQuery.isWindow( elem ) ) {
|
|
|
|
// Don't re-trigger an onFOO event when we call its FOO() method
|
|
tmp = elem[ ontype ];
|
|
|
|
if ( tmp ) {
|
|
elem[ ontype ] = null;
|
|
}
|
|
|
|
// Prevent re-triggering of the same event, since we already bubbled it above
|
|
jQuery.event.triggered = type;
|
|
try {
|
|
elem[ type ]();
|
|
} catch ( e ) {
|
|
|
|
// IE<9 dies on focus/blur to hidden element (#1486,#12518)
|
|
// only reproducible on winXP IE8 native, not IE9 in IE8 mode
|
|
}
|
|
jQuery.event.triggered = undefined;
|
|
|
|
if ( tmp ) {
|
|
elem[ ontype ] = tmp;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return event.result;
|
|
},
|
|
|
|
dispatch: function( event ) {
|
|
|
|
// Make a writable jQuery.Event from the native event object
|
|
event = jQuery.event.fix( event );
|
|
|
|
var i, j, ret, matched, handleObj,
|
|
handlerQueue = [],
|
|
args = slice.call( arguments ),
|
|
handlers = ( jQuery._data( this, "events" ) || {} )[ event.type ] || [],
|
|
special = jQuery.event.special[ event.type ] || {};
|
|
|
|
// Use the fix-ed jQuery.Event rather than the (read-only) native event
|
|
args[ 0 ] = event;
|
|
event.delegateTarget = this;
|
|
|
|
// Call the preDispatch hook for the mapped type, and let it bail if desired
|
|
if ( special.preDispatch && special.preDispatch.call( this, event ) === false ) {
|
|
return;
|
|
}
|
|
|
|
// Determine handlers
|
|
handlerQueue = jQuery.event.handlers.call( this, event, handlers );
|
|
|
|
// Run delegates first; they may want to stop propagation beneath us
|
|
i = 0;
|
|
while ( ( matched = handlerQueue[ i++ ] ) && !event.isPropagationStopped() ) {
|
|
event.currentTarget = matched.elem;
|
|
|
|
j = 0;
|
|
while ( ( handleObj = matched.handlers[ j++ ] ) &&
|
|
!event.isImmediatePropagationStopped() ) {
|
|
|
|
// Triggered event must either 1) have no namespace, or 2) have namespace(s)
|
|
// a subset or equal to those in the bound event (both can have no namespace).
|
|
if ( !event.rnamespace || event.rnamespace.test( handleObj.namespace ) ) {
|
|
|
|
event.handleObj = handleObj;
|
|
event.data = handleObj.data;
|
|
|
|
ret = ( ( jQuery.event.special[ handleObj.origType ] || {} ).handle ||
|
|
handleObj.handler ).apply( matched.elem, args );
|
|
|
|
if ( ret !== undefined ) {
|
|
if ( ( event.result = ret ) === false ) {
|
|
event.preventDefault();
|
|
event.stopPropagation();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// Call the postDispatch hook for the mapped type
|
|
if ( special.postDispatch ) {
|
|
special.postDispatch.call( this, event );
|
|
}
|
|
|
|
return event.result;
|
|
},
|
|
|
|
handlers: function( event, handlers ) {
|
|
var i, matches, sel, handleObj,
|
|
handlerQueue = [],
|
|
delegateCount = handlers.delegateCount,
|
|
cur = event.target;
|
|
|
|
// Support (at least): Chrome, IE9
|
|
// Find delegate handlers
|
|
// Black-hole SVG <use> instance trees (#13180)
|
|
//
|
|
// Support: Firefox<=42+
|
|
// Avoid non-left-click in FF but don't block IE radio events (#3861, gh-2343)
|
|
if ( delegateCount && cur.nodeType &&
|
|
( event.type !== "click" || isNaN( event.button ) || event.button < 1 ) ) {
|
|
|
|
/* jshint eqeqeq: false */
|
|
for ( ; cur != this; cur = cur.parentNode || this ) {
|
|
/* jshint eqeqeq: true */
|
|
|
|
// Don't check non-elements (#13208)
|
|
// Don't process clicks on disabled elements (#6911, #8165, #11382, #11764)
|
|
if ( cur.nodeType === 1 && ( cur.disabled !== true || event.type !== "click" ) ) {
|
|
matches = [];
|
|
for ( i = 0; i < delegateCount; i++ ) {
|
|
handleObj = handlers[ i ];
|
|
|
|
// Don't conflict with Object.prototype properties (#13203)
|
|
sel = handleObj.selector + " ";
|
|
|
|
if ( matches[ sel ] === undefined ) {
|
|
matches[ sel ] = handleObj.needsContext ?
|
|
jQuery( sel, this ).index( cur ) > -1 :
|
|
jQuery.find( sel, this, null, [ cur ] ).length;
|
|
}
|
|
if ( matches[ sel ] ) {
|
|
matches.push( handleObj );
|
|
}
|
|
}
|
|
if ( matches.length ) {
|
|
handlerQueue.push( { elem: cur, handlers: matches } );
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// Add the remaining (directly-bound) handlers
|
|
if ( delegateCount < handlers.length ) {
|
|
handlerQueue.push( { elem: this, handlers: handlers.slice( delegateCount ) } );
|
|
}
|
|
|
|
return handlerQueue;
|
|
},
|
|
|
|
fix: function( event ) {
|
|
if ( event[ jQuery.expando ] ) {
|
|
return event;
|
|
}
|
|
|
|
// Create a writable copy of the event object and normalize some properties
|
|
var i, prop, copy,
|
|
type = event.type,
|
|
originalEvent = event,
|
|
fixHook = this.fixHooks[ type ];
|
|
|
|
if ( !fixHook ) {
|
|
this.fixHooks[ type ] = fixHook =
|
|
rmouseEvent.test( type ) ? this.mouseHooks :
|
|
rkeyEvent.test( type ) ? this.keyHooks :
|
|
{};
|
|
}
|
|
copy = fixHook.props ? this.props.concat( fixHook.props ) : this.props;
|
|
|
|
event = new jQuery.Event( originalEvent );
|
|
|
|
i = copy.length;
|
|
while ( i-- ) {
|
|
prop = copy[ i ];
|
|
event[ prop ] = originalEvent[ prop ];
|
|
}
|
|
|
|
// Support: IE<9
|
|
// Fix target property (#1925)
|
|
if ( !event.target ) {
|
|
event.target = originalEvent.srcElement || document;
|
|
}
|
|
|
|
// Support: Safari 6-8+
|
|
// Target should not be a text node (#504, #13143)
|
|
if ( event.target.nodeType === 3 ) {
|
|
event.target = event.target.parentNode;
|
|
}
|
|
|
|
// Support: IE<9
|
|
// For mouse/key events, metaKey==false if it's undefined (#3368, #11328)
|
|
event.metaKey = !!event.metaKey;
|
|
|
|
return fixHook.filter ? fixHook.filter( event, originalEvent ) : event;
|
|
},
|
|
|
|
// Includes some event props shared by KeyEvent and MouseEvent
|
|
props: ( "altKey bubbles cancelable ctrlKey currentTarget detail eventPhase " +
|
|
"metaKey relatedTarget shiftKey target timeStamp view which" ).split( " " ),
|
|
|
|
fixHooks: {},
|
|
|
|
keyHooks: {
|
|
props: "char charCode key keyCode".split( " " ),
|
|
filter: function( event, original ) {
|
|
|
|
// Add which for key events
|
|
if ( event.which == null ) {
|
|
event.which = original.charCode != null ? original.charCode : original.keyCode;
|
|
}
|
|
|
|
return event;
|
|
}
|
|
},
|
|
|
|
mouseHooks: {
|
|
props: ( "button buttons clientX clientY fromElement offsetX offsetY " +
|
|
"pageX pageY screenX screenY toElement" ).split( " " ),
|
|
filter: function( event, original ) {
|
|
var body, eventDoc, doc,
|
|
button = original.button,
|
|
fromElement = original.fromElement;
|
|
|
|
// Calculate pageX/Y if missing and clientX/Y available
|
|
if ( event.pageX == null && original.clientX != null ) {
|
|
eventDoc = event.target.ownerDocument || document;
|
|
doc = eventDoc.documentElement;
|
|
body = eventDoc.body;
|
|
|
|
event.pageX = original.clientX +
|
|
( doc && doc.scrollLeft || body && body.scrollLeft || 0 ) -
|
|
( doc && doc.clientLeft || body && body.clientLeft || 0 );
|
|
event.pageY = original.clientY +
|
|
( doc && doc.scrollTop || body && body.scrollTop || 0 ) -
|
|
( doc && doc.clientTop || body && body.clientTop || 0 );
|
|
}
|
|
|
|
// Add relatedTarget, if necessary
|
|
if ( !event.relatedTarget && fromElement ) {
|
|
event.relatedTarget = fromElement === event.target ?
|
|
original.toElement :
|
|
fromElement;
|
|
}
|
|
|
|
// Add which for click: 1 === left; 2 === middle; 3 === right
|
|
// Note: button is not normalized, so don't use it
|
|
if ( !event.which && button !== undefined ) {
|
|
event.which = ( button & 1 ? 1 : ( button & 2 ? 3 : ( button & 4 ? 2 : 0 ) ) );
|
|
}
|
|
|
|
return event;
|
|
}
|
|
},
|
|
|
|
special: {
|
|
load: {
|
|
|
|
// Prevent triggered image.load events from bubbling to window.load
|
|
noBubble: true
|
|
},
|
|
focus: {
|
|
|
|
// Fire native event if possible so blur/focus sequence is correct
|
|
trigger: function() {
|
|
if ( this !== safeActiveElement() && this.focus ) {
|
|
try {
|
|
this.focus();
|
|
return false;
|
|
} catch ( e ) {
|
|
|
|
// Support: IE<9
|
|
// If we error on focus to hidden element (#1486, #12518),
|
|
// let .trigger() run the handlers
|
|
}
|
|
}
|
|
},
|
|
delegateType: "focusin"
|
|
},
|
|
blur: {
|
|
trigger: function() {
|
|
if ( this === safeActiveElement() && this.blur ) {
|
|
this.blur();
|
|
return false;
|
|
}
|
|
},
|
|
delegateType: "focusout"
|
|
},
|
|
click: {
|
|
|
|
// For checkbox, fire native event so checked state will be right
|
|
trigger: function() {
|
|
if ( jQuery.nodeName( this, "input" ) && this.type === "checkbox" && this.click ) {
|
|
this.click();
|
|
return false;
|
|
}
|
|
},
|
|
|
|
// For cross-browser consistency, don't fire native .click() on links
|
|
_default: function( event ) {
|
|
return jQuery.nodeName( event.target, "a" );
|
|
}
|
|
},
|
|
|
|
beforeunload: {
|
|
postDispatch: function( event ) {
|
|
|
|
// Support: Firefox 20+
|
|
// Firefox doesn't alert if the returnValue field is not set.
|
|
if ( event.result !== undefined && event.originalEvent ) {
|
|
event.originalEvent.returnValue = event.result;
|
|
}
|
|
}
|
|
}
|
|
},
|
|
|
|
// Piggyback on a donor event to simulate a different one
|
|
simulate: function( type, elem, event ) {
|
|
var e = jQuery.extend(
|
|
new jQuery.Event(),
|
|
event,
|
|
{
|
|
type: type,
|
|
isSimulated: true
|
|
|
|
// Previously, `originalEvent: {}` was set here, so stopPropagation call
|
|
// would not be triggered on donor event, since in our own
|
|
// jQuery.event.stopPropagation function we had a check for existence of
|
|
// originalEvent.stopPropagation method, so, consequently it would be a noop.
|
|
//
|
|
// Guard for simulated events was moved to jQuery.event.stopPropagation function
|
|
// since `originalEvent` should point to the original event for the
|
|
// constancy with other events and for more focused logic
|
|
}
|
|
);
|
|
|
|
jQuery.event.trigger( e, null, elem );
|
|
|
|
if ( e.isDefaultPrevented() ) {
|
|
event.preventDefault();
|
|
}
|
|
}
|
|
};
|
|
|
|
jQuery.removeEvent = document.removeEventListener ?
|
|
function( elem, type, handle ) {
|
|
|
|
// This "if" is needed for plain objects
|
|
if ( elem.removeEventListener ) {
|
|
elem.removeEventListener( type, handle );
|
|
}
|
|
} :
|
|
function( elem, type, handle ) {
|
|
var name = "on" + type;
|
|
|
|
if ( elem.detachEvent ) {
|
|
|
|
// #8545, #7054, preventing memory leaks for custom events in IE6-8
|
|
// detachEvent needed property on element, by name of that event,
|
|
// to properly expose it to GC
|
|
if ( typeof elem[ name ] === "undefined" ) {
|
|
elem[ name ] = null;
|
|
}
|
|
|
|
elem.detachEvent( name, handle );
|
|
}
|
|
};
|
|
|
|
jQuery.Event = function( src, props ) {
|
|
|
|
// Allow instantiation without the 'new' keyword
|
|
if ( !( this instanceof jQuery.Event ) ) {
|
|
return new jQuery.Event( src, props );
|
|
}
|
|
|
|
// Event object
|
|
if ( src && src.type ) {
|
|
this.originalEvent = src;
|
|
this.type = src.type;
|
|
|
|
// Events bubbling up the document may have been marked as prevented
|
|
// by a handler lower down the tree; reflect the correct value.
|
|
this.isDefaultPrevented = src.defaultPrevented ||
|
|
src.defaultPrevented === undefined &&
|
|
|
|
// Support: IE < 9, Android < 4.0
|
|
src.returnValue === false ?
|
|
returnTrue :
|
|
returnFalse;
|
|
|
|
// Event type
|
|
} else {
|
|
this.type = src;
|
|
}
|
|
|
|
// Put explicitly provided properties onto the event object
|
|
if ( props ) {
|
|
jQuery.extend( this, props );
|
|
}
|
|
|
|
// Create a timestamp if incoming event doesn't have one
|
|
this.timeStamp = src && src.timeStamp || jQuery.now();
|
|
|
|
// Mark it as fixed
|
|
this[ jQuery.expando ] = true;
|
|
};
|
|
|
|
// jQuery.Event is based on DOM3 Events as specified by the ECMAScript Language Binding
|
|
// http://www.w3.org/TR/2003/WD-DOM-Level-3-Events-20030331/ecma-script-binding.html
|
|
jQuery.Event.prototype = {
|
|
constructor: jQuery.Event,
|
|
isDefaultPrevented: returnFalse,
|
|
isPropagationStopped: returnFalse,
|
|
isImmediatePropagationStopped: returnFalse,
|
|
|
|
preventDefault: function() {
|
|
var e = this.originalEvent;
|
|
|
|
this.isDefaultPrevented = returnTrue;
|
|
if ( !e ) {
|
|
return;
|
|
}
|
|
|
|
// If preventDefault exists, run it on the original event
|
|
if ( e.preventDefault ) {
|
|
e.preventDefault();
|
|
|
|
// Support: IE
|
|
// Otherwise set the returnValue property of the original event to false
|
|
} else {
|
|
e.returnValue = false;
|
|
}
|
|
},
|
|
stopPropagation: function() {
|
|
var e = this.originalEvent;
|
|
|
|
this.isPropagationStopped = returnTrue;
|
|
|
|
if ( !e || this.isSimulated ) {
|
|
return;
|
|
}
|
|
|
|
// If stopPropagation exists, run it on the original event
|
|
if ( e.stopPropagation ) {
|
|
e.stopPropagation();
|
|
}
|
|
|
|
// Support: IE
|
|
// Set the cancelBubble property of the original event to true
|
|
e.cancelBubble = true;
|
|
},
|
|
stopImmediatePropagation: function() {
|
|
var e = this.originalEvent;
|
|
|
|
this.isImmediatePropagationStopped = returnTrue;
|
|
|
|
if ( e && e.stopImmediatePropagation ) {
|
|
e.stopImmediatePropagation();
|
|
}
|
|
|
|
this.stopPropagation();
|
|
}
|
|
};
|
|
|
|
// Create mouseenter/leave events using mouseover/out and event-time checks
|
|
// so that event delegation works in jQuery.
|
|
// Do the same for pointerenter/pointerleave and pointerover/pointerout
|
|
//
|
|
// Support: Safari 7 only
|
|
// Safari sends mouseenter too often; see:
|
|
// https://code.google.com/p/chromium/issues/detail?id=470258
|
|
// for the description of the bug (it existed in older Chrome versions as well).
|
|
jQuery.each( {
|
|
mouseenter: "mouseover",
|
|
mouseleave: "mouseout",
|
|
pointerenter: "pointerover",
|
|
pointerleave: "pointerout"
|
|
}, function( orig, fix ) {
|
|
jQuery.event.special[ orig ] = {
|
|
delegateType: fix,
|
|
bindType: fix,
|
|
|
|
handle: function( event ) {
|
|
var ret,
|
|
target = this,
|
|
related = event.relatedTarget,
|
|
handleObj = event.handleObj;
|
|
|
|
// For mouseenter/leave call the handler if related is outside the target.
|
|
// NB: No relatedTarget if the mouse left/entered the browser window
|
|
if ( !related || ( related !== target && !jQuery.contains( target, related ) ) ) {
|
|
event.type = handleObj.origType;
|
|
ret = handleObj.handler.apply( this, arguments );
|
|
event.type = fix;
|
|
}
|
|
return ret;
|
|
}
|
|
};
|
|
} );
|
|
|
|
// IE submit delegation
|
|
if ( !support.submit ) {
|
|
|
|
jQuery.event.special.submit = {
|
|
setup: function() {
|
|
|
|
// Only need this for delegated form submit events
|
|
if ( jQuery.nodeName( this, "form" ) ) {
|
|
return false;
|
|
}
|
|
|
|
// Lazy-add a submit handler when a descendant form may potentially be submitted
|
|
jQuery.event.add( this, "click._submit keypress._submit", function( e ) {
|
|
|
|
// Node name check avoids a VML-related crash in IE (#9807)
|
|
var elem = e.target,
|
|
form = jQuery.nodeName( elem, "input" ) || jQuery.nodeName( elem, "button" ) ?
|
|
|
|
// Support: IE <=8
|
|
// We use jQuery.prop instead of elem.form
|
|
// to allow fixing the IE8 delegated submit issue (gh-2332)
|
|
// by 3rd party polyfills/workarounds.
|
|
jQuery.prop( elem, "form" ) :
|
|
undefined;
|
|
|
|
if ( form && !jQuery._data( form, "submit" ) ) {
|
|
jQuery.event.add( form, "submit._submit", function( event ) {
|
|
event._submitBubble = true;
|
|
} );
|
|
jQuery._data( form, "submit", true );
|
|
}
|
|
} );
|
|
|
|
// return undefined since we don't need an event listener
|
|
},
|
|
|
|
postDispatch: function( event ) {
|
|
|
|
// If form was submitted by the user, bubble the event up the tree
|
|
if ( event._submitBubble ) {
|
|
delete event._submitBubble;
|
|
if ( this.parentNode && !event.isTrigger ) {
|
|
jQuery.event.simulate( "submit", this.parentNode, event );
|
|
}
|
|
}
|
|
},
|
|
|
|
teardown: function() {
|
|
|
|
// Only need this for delegated form submit events
|
|
if ( jQuery.nodeName( this, "form" ) ) {
|
|
return false;
|
|
}
|
|
|
|
// Remove delegated handlers; cleanData eventually reaps submit handlers attached above
|
|
jQuery.event.remove( this, "._submit" );
|
|
}
|
|
};
|
|
}
|
|
|
|
// IE change delegation and checkbox/radio fix
|
|
if ( !support.change ) {
|
|
|
|
jQuery.event.special.change = {
|
|
|
|
setup: function() {
|
|
|
|
if ( rformElems.test( this.nodeName ) ) {
|
|
|
|
// IE doesn't fire change on a check/radio until blur; trigger it on click
|
|
// after a propertychange. Eat the blur-change in special.change.handle.
|
|
// This still fires onchange a second time for check/radio after blur.
|
|
if ( this.type === "checkbox" || this.type === "radio" ) {
|
|
jQuery.event.add( this, "propertychange._change", function( event ) {
|
|
if ( event.originalEvent.propertyName === "checked" ) {
|
|
this._justChanged = true;
|
|
}
|
|
} );
|
|
jQuery.event.add( this, "click._change", function( event ) {
|
|
if ( this._justChanged && !event.isTrigger ) {
|
|
this._justChanged = false;
|
|
}
|
|
|
|
// Allow triggered, simulated change events (#11500)
|
|
jQuery.event.simulate( "change", this, event );
|
|
} );
|
|
}
|
|
return false;
|
|
}
|
|
|
|
// Delegated event; lazy-add a change handler on descendant inputs
|
|
jQuery.event.add( this, "beforeactivate._change", function( e ) {
|
|
var elem = e.target;
|
|
|
|
if ( rformElems.test( elem.nodeName ) && !jQuery._data( elem, "change" ) ) {
|
|
jQuery.event.add( elem, "change._change", function( event ) {
|
|
if ( this.parentNode && !event.isSimulated && !event.isTrigger ) {
|
|
jQuery.event.simulate( "change", this.parentNode, event );
|
|
}
|
|
} );
|
|
jQuery._data( elem, "change", true );
|
|
}
|
|
} );
|
|
},
|
|
|
|
handle: function( event ) {
|
|
var elem = event.target;
|
|
|
|
// Swallow native change events from checkbox/radio, we already triggered them above
|
|
if ( this !== elem || event.isSimulated || event.isTrigger ||
|
|
( elem.type !== "radio" && elem.type !== "checkbox" ) ) {
|
|
|
|
return event.handleObj.handler.apply( this, arguments );
|
|
}
|
|
},
|
|
|
|
teardown: function() {
|
|
jQuery.event.remove( this, "._change" );
|
|
|
|
return !rformElems.test( this.nodeName );
|
|
}
|
|
};
|
|
}
|
|
|
|
// Support: Firefox
|
|
// Firefox doesn't have focus(in | out) events
|
|
// Related ticket - https://bugzilla.mozilla.org/show_bug.cgi?id=687787
|
|
//
|
|
// Support: Chrome, Safari
|
|
// focus(in | out) events fire after focus & blur events,
|
|
// which is spec violation - http://www.w3.org/TR/DOM-Level-3-Events/#events-focusevent-event-order
|
|
// Related ticket - https://code.google.com/p/chromium/issues/detail?id=449857
|
|
if ( !support.focusin ) {
|
|
jQuery.each( { focus: "focusin", blur: "focusout" }, function( orig, fix ) {
|
|
|
|
// Attach a single capturing handler on the document while someone wants focusin/focusout
|
|
var handler = function( event ) {
|
|
jQuery.event.simulate( fix, event.target, jQuery.event.fix( event ) );
|
|
};
|
|
|
|
jQuery.event.special[ fix ] = {
|
|
setup: function() {
|
|
var doc = this.ownerDocument || this,
|
|
attaches = jQuery._data( doc, fix );
|
|
|
|
if ( !attaches ) {
|
|
doc.addEventListener( orig, handler, true );
|
|
}
|
|
jQuery._data( doc, fix, ( attaches || 0 ) + 1 );
|
|
},
|
|
teardown: function() {
|
|
var doc = this.ownerDocument || this,
|
|
attaches = jQuery._data( doc, fix ) - 1;
|
|
|
|
if ( !attaches ) {
|
|
doc.removeEventListener( orig, handler, true );
|
|
jQuery._removeData( doc, fix );
|
|
} else {
|
|
jQuery._data( doc, fix, attaches );
|
|
}
|
|
}
|
|
};
|
|
} );
|
|
}
|
|
|
|
jQuery.fn.extend( {
|
|
|
|
on: function( types, selector, data, fn ) {
|
|
return on( this, types, selector, data, fn );
|
|
},
|
|
one: function( types, selector, data, fn ) {
|
|
return on( this, types, selector, data, fn, 1 );
|
|
},
|
|
off: function( types, selector, fn ) {
|
|
var handleObj, type;
|
|
if ( types && types.preventDefault && types.handleObj ) {
|
|
|
|
// ( event ) dispatched jQuery.Event
|
|
handleObj = types.handleObj;
|
|
jQuery( types.delegateTarget ).off(
|
|
handleObj.namespace ?
|
|
handleObj.origType + "." + handleObj.namespace :
|
|
handleObj.origType,
|
|
handleObj.selector,
|
|
handleObj.handler
|
|
);
|
|
return this;
|
|
}
|
|
if ( typeof types === "object" ) {
|
|
|
|
// ( types-object [, selector] )
|
|
for ( type in types ) {
|
|
this.off( type, selector, types[ type ] );
|
|
}
|
|
return this;
|
|
}
|
|
if ( selector === false || typeof selector === "function" ) {
|
|
|
|
// ( types [, fn] )
|
|
fn = selector;
|
|
selector = undefined;
|
|
}
|
|
if ( fn === false ) {
|
|
fn = returnFalse;
|
|
}
|
|
return this.each( function() {
|
|
jQuery.event.remove( this, types, fn, selector );
|
|
} );
|
|
},
|
|
|
|
trigger: function( type, data ) {
|
|
return this.each( function() {
|
|
jQuery.event.trigger( type, data, this );
|
|
} );
|
|
},
|
|
triggerHandler: function( type, data ) {
|
|
var elem = this[ 0 ];
|
|
if ( elem ) {
|
|
return jQuery.event.trigger( type, data, elem, true );
|
|
}
|
|
}
|
|
} );
|
|
|
|
|
|
var rinlinejQuery = / jQuery\d+="(?:null|\d+)"/g,
|
|
rnoshimcache = new RegExp( "<(?:" + nodeNames + ")[\\s/>]", "i" ),
|
|
rxhtmlTag = /<(?!area|br|col|embed|hr|img|input|link|meta|param)(([\w:-]+)[^>]*)\/>/gi,
|
|
|
|
// Support: IE 10-11, Edge 10240+
|
|
// In IE/Edge using regex groups here causes severe slowdowns.
|
|
// See https://connect.microsoft.com/IE/feedback/details/1736512/
|
|
rnoInnerhtml = /<script|<style|<link/i,
|
|
|
|
// checked="checked" or checked
|
|
rchecked = /checked\s*(?:[^=]|=\s*.checked.)/i,
|
|
rscriptTypeMasked = /^true\/(.*)/,
|
|
rcleanScript = /^\s*<!(?:\[CDATA\[|--)|(?:\]\]|--)>\s*$/g,
|
|
safeFragment = createSafeFragment( document ),
|
|
fragmentDiv = safeFragment.appendChild( document.createElement( "div" ) );
|
|
|
|
// Support: IE<8
|
|
// Manipulating tables requires a tbody
|
|
function manipulationTarget( elem, content ) {
|
|
return jQuery.nodeName( elem, "table" ) &&
|
|
jQuery.nodeName( content.nodeType !== 11 ? content : content.firstChild, "tr" ) ?
|
|
|
|
elem.getElementsByTagName( "tbody" )[ 0 ] ||
|
|
elem.appendChild( elem.ownerDocument.createElement( "tbody" ) ) :
|
|
elem;
|
|
}
|
|
|
|
// Replace/restore the type attribute of script elements for safe DOM manipulation
|
|
function disableScript( elem ) {
|
|
elem.type = ( jQuery.find.attr( elem, "type" ) !== null ) + "/" + elem.type;
|
|
return elem;
|
|
}
|
|
function restoreScript( elem ) {
|
|
var match = rscriptTypeMasked.exec( elem.type );
|
|
if ( match ) {
|
|
elem.type = match[ 1 ];
|
|
} else {
|
|
elem.removeAttribute( "type" );
|
|
}
|
|
return elem;
|
|
}
|
|
|
|
function cloneCopyEvent( src, dest ) {
|
|
if ( dest.nodeType !== 1 || !jQuery.hasData( src ) ) {
|
|
return;
|
|
}
|
|
|
|
var type, i, l,
|
|
oldData = jQuery._data( src ),
|
|
curData = jQuery._data( dest, oldData ),
|
|
events = oldData.events;
|
|
|
|
if ( events ) {
|
|
delete curData.handle;
|
|
curData.events = {};
|
|
|
|
for ( type in events ) {
|
|
for ( i = 0, l = events[ type ].length; i < l; i++ ) {
|
|
jQuery.event.add( dest, type, events[ type ][ i ] );
|
|
}
|
|
}
|
|
}
|
|
|
|
// make the cloned public data object a copy from the original
|
|
if ( curData.data ) {
|
|
curData.data = jQuery.extend( {}, curData.data );
|
|
}
|
|
}
|
|
|
|
function fixCloneNodeIssues( src, dest ) {
|
|
var nodeName, e, data;
|
|
|
|
// We do not need to do anything for non-Elements
|
|
if ( dest.nodeType !== 1 ) {
|
|
return;
|
|
}
|
|
|
|
nodeName = dest.nodeName.toLowerCase();
|
|
|
|
// IE6-8 copies events bound via attachEvent when using cloneNode.
|
|
if ( !support.noCloneEvent && dest[ jQuery.expando ] ) {
|
|
data = jQuery._data( dest );
|
|
|
|
for ( e in data.events ) {
|
|
jQuery.removeEvent( dest, e, data.handle );
|
|
}
|
|
|
|
// Event data gets referenced instead of copied if the expando gets copied too
|
|
dest.removeAttribute( jQuery.expando );
|
|
}
|
|
|
|
// IE blanks contents when cloning scripts, and tries to evaluate newly-set text
|
|
if ( nodeName === "script" && dest.text !== src.text ) {
|
|
disableScript( dest ).text = src.text;
|
|
restoreScript( dest );
|
|
|
|
// IE6-10 improperly clones children of object elements using classid.
|
|
// IE10 throws NoModificationAllowedError if parent is null, #12132.
|
|
} else if ( nodeName === "object" ) {
|
|
if ( dest.parentNode ) {
|
|
dest.outerHTML = src.outerHTML;
|
|
}
|
|
|
|
// This path appears unavoidable for IE9. When cloning an object
|
|
// element in IE9, the outerHTML strategy above is not sufficient.
|
|
// If the src has innerHTML and the destination does not,
|
|
// copy the src.innerHTML into the dest.innerHTML. #10324
|
|
if ( support.html5Clone && ( src.innerHTML && !jQuery.trim( dest.innerHTML ) ) ) {
|
|
dest.innerHTML = src.innerHTML;
|
|
}
|
|
|
|
} else if ( nodeName === "input" && rcheckableType.test( src.type ) ) {
|
|
|
|
// IE6-8 fails to persist the checked state of a cloned checkbox
|
|
// or radio button. Worse, IE6-7 fail to give the cloned element
|
|
// a checked appearance if the defaultChecked value isn't also set
|
|
|
|
dest.defaultChecked = dest.checked = src.checked;
|
|
|
|
// IE6-7 get confused and end up setting the value of a cloned
|
|
// checkbox/radio button to an empty string instead of "on"
|
|
if ( dest.value !== src.value ) {
|
|
dest.value = src.value;
|
|
}
|
|
|
|
// IE6-8 fails to return the selected option to the default selected
|
|
// state when cloning options
|
|
} else if ( nodeName === "option" ) {
|
|
dest.defaultSelected = dest.selected = src.defaultSelected;
|
|
|
|
// IE6-8 fails to set the defaultValue to the correct value when
|
|
// cloning other types of input fields
|
|
} else if ( nodeName === "input" || nodeName === "textarea" ) {
|
|
dest.defaultValue = src.defaultValue;
|
|
}
|
|
}
|
|
|
|
function domManip( collection, args, callback, ignored ) {
|
|
|
|
// Flatten any nested arrays
|
|
args = concat.apply( [], args );
|
|
|
|
var first, node, hasScripts,
|
|
scripts, doc, fragment,
|
|
i = 0,
|
|
l = collection.length,
|
|
iNoClone = l - 1,
|
|
value = args[ 0 ],
|
|
isFunction = jQuery.isFunction( value );
|
|
|
|
// We can't cloneNode fragments that contain checked, in WebKit
|
|
if ( isFunction ||
|
|
( l > 1 && typeof value === "string" &&
|
|
!support.checkClone && rchecked.test( value ) ) ) {
|
|
return collection.each( function( index ) {
|
|
var self = collection.eq( index );
|
|
if ( isFunction ) {
|
|
args[ 0 ] = value.call( this, index, self.html() );
|
|
}
|
|
domManip( self, args, callback, ignored );
|
|
} );
|
|
}
|
|
|
|
if ( l ) {
|
|
fragment = buildFragment( args, collection[ 0 ].ownerDocument, false, collection, ignored );
|
|
first = fragment.firstChild;
|
|
|
|
if ( fragment.childNodes.length === 1 ) {
|
|
fragment = first;
|
|
}
|
|
|
|
// Require either new content or an interest in ignored elements to invoke the callback
|
|
if ( first || ignored ) {
|
|
scripts = jQuery.map( getAll( fragment, "script" ), disableScript );
|
|
hasScripts = scripts.length;
|
|
|
|
// Use the original fragment for the last item
|
|
// instead of the first because it can end up
|
|
// being emptied incorrectly in certain situations (#8070).
|
|
for ( ; i < l; i++ ) {
|
|
node = fragment;
|
|
|
|
if ( i !== iNoClone ) {
|
|
node = jQuery.clone( node, true, true );
|
|
|
|
// Keep references to cloned scripts for later restoration
|
|
if ( hasScripts ) {
|
|
|
|
// Support: Android<4.1, PhantomJS<2
|
|
// push.apply(_, arraylike) throws on ancient WebKit
|
|
jQuery.merge( scripts, getAll( node, "script" ) );
|
|
}
|
|
}
|
|
|
|
callback.call( collection[ i ], node, i );
|
|
}
|
|
|
|
if ( hasScripts ) {
|
|
doc = scripts[ scripts.length - 1 ].ownerDocument;
|
|
|
|
// Reenable scripts
|
|
jQuery.map( scripts, restoreScript );
|
|
|
|
// Evaluate executable scripts on first document insertion
|
|
for ( i = 0; i < hasScripts; i++ ) {
|
|
node = scripts[ i ];
|
|
if ( rscriptType.test( node.type || "" ) &&
|
|
!jQuery._data( node, "globalEval" ) &&
|
|
jQuery.contains( doc, node ) ) {
|
|
|
|
if ( node.src ) {
|
|
|
|
// Optional AJAX dependency, but won't run scripts if not present
|
|
if ( jQuery._evalUrl ) {
|
|
jQuery._evalUrl( node.src );
|
|
}
|
|
} else {
|
|
jQuery.globalEval(
|
|
( node.text || node.textContent || node.innerHTML || "" )
|
|
.replace( rcleanScript, "" )
|
|
);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// Fix #11809: Avoid leaking memory
|
|
fragment = first = null;
|
|
}
|
|
}
|
|
|
|
return collection;
|
|
}
|
|
|
|
function remove( elem, selector, keepData ) {
|
|
var node,
|
|
elems = selector ? jQuery.filter( selector, elem ) : elem,
|
|
i = 0;
|
|
|
|
for ( ; ( node = elems[ i ] ) != null; i++ ) {
|
|
|
|
if ( !keepData && node.nodeType === 1 ) {
|
|
jQuery.cleanData( getAll( node ) );
|
|
}
|
|
|
|
if ( node.parentNode ) {
|
|
if ( keepData && jQuery.contains( node.ownerDocument, node ) ) {
|
|
setGlobalEval( getAll( node, "script" ) );
|
|
}
|
|
node.parentNode.removeChild( node );
|
|
}
|
|
}
|
|
|
|
return elem;
|
|
}
|
|
|
|
jQuery.extend( {
|
|
htmlPrefilter: function( html ) {
|
|
return html.replace( rxhtmlTag, "<$1></$2>" );
|
|
},
|
|
|
|
clone: function( elem, dataAndEvents, deepDataAndEvents ) {
|
|
var destElements, node, clone, i, srcElements,
|
|
inPage = jQuery.contains( elem.ownerDocument, elem );
|
|
|
|
if ( support.html5Clone || jQuery.isXMLDoc( elem ) ||
|
|
!rnoshimcache.test( "<" + elem.nodeName + ">" ) ) {
|
|
|
|
clone = elem.cloneNode( true );
|
|
|
|
// IE<=8 does not properly clone detached, unknown element nodes
|
|
} else {
|
|
fragmentDiv.innerHTML = elem.outerHTML;
|
|
fragmentDiv.removeChild( clone = fragmentDiv.firstChild );
|
|
}
|
|
|
|
if ( ( !support.noCloneEvent || !support.noCloneChecked ) &&
|
|
( elem.nodeType === 1 || elem.nodeType === 11 ) && !jQuery.isXMLDoc( elem ) ) {
|
|
|
|
// We eschew Sizzle here for performance reasons: http://jsperf.com/getall-vs-sizzle/2
|
|
destElements = getAll( clone );
|
|
srcElements = getAll( elem );
|
|
|
|
// Fix all IE cloning issues
|
|
for ( i = 0; ( node = srcElements[ i ] ) != null; ++i ) {
|
|
|
|
// Ensure that the destination node is not null; Fixes #9587
|
|
if ( destElements[ i ] ) {
|
|
fixCloneNodeIssues( node, destElements[ i ] );
|
|
}
|
|
}
|
|
}
|
|
|
|
// Copy the events from the original to the clone
|
|
if ( dataAndEvents ) {
|
|
if ( deepDataAndEvents ) {
|
|
srcElements = srcElements || getAll( elem );
|
|
destElements = destElements || getAll( clone );
|
|
|
|
for ( i = 0; ( node = srcElements[ i ] ) != null; i++ ) {
|
|
cloneCopyEvent( node, destElements[ i ] );
|
|
}
|
|
} else {
|
|
cloneCopyEvent( elem, clone );
|
|
}
|
|
}
|
|
|
|
// Preserve script evaluation history
|
|
destElements = getAll( clone, "script" );
|
|
if ( destElements.length > 0 ) {
|
|
setGlobalEval( destElements, !inPage && getAll( elem, "script" ) );
|
|
}
|
|
|
|
destElements = srcElements = node = null;
|
|
|
|
// Return the cloned set
|
|
return clone;
|
|
},
|
|
|
|
cleanData: function( elems, /* internal */ forceAcceptData ) {
|
|
var elem, type, id, data,
|
|
i = 0,
|
|
internalKey = jQuery.expando,
|
|
cache = jQuery.cache,
|
|
attributes = support.attributes,
|
|
special = jQuery.event.special;
|
|
|
|
for ( ; ( elem = elems[ i ] ) != null; i++ ) {
|
|
if ( forceAcceptData || acceptData( elem ) ) {
|
|
|
|
id = elem[ internalKey ];
|
|
data = id && cache[ id ];
|
|
|
|
if ( data ) {
|
|
if ( data.events ) {
|
|
for ( type in data.events ) {
|
|
if ( special[ type ] ) {
|
|
jQuery.event.remove( elem, type );
|
|
|
|
// This is a shortcut to avoid jQuery.event.remove's overhead
|
|
} else {
|
|
jQuery.removeEvent( elem, type, data.handle );
|
|
}
|
|
}
|
|
}
|
|
|
|
// Remove cache only if it was not already removed by jQuery.event.remove
|
|
if ( cache[ id ] ) {
|
|
|
|
delete cache[ id ];
|
|
|
|
// Support: IE<9
|
|
// IE does not allow us to delete expando properties from nodes
|
|
// IE creates expando attributes along with the property
|
|
// IE does not have a removeAttribute function on Document nodes
|
|
if ( !attributes && typeof elem.removeAttribute !== "undefined" ) {
|
|
elem.removeAttribute( internalKey );
|
|
|
|
// Webkit & Blink performance suffers when deleting properties
|
|
// from DOM nodes, so set to undefined instead
|
|
// https://code.google.com/p/chromium/issues/detail?id=378607
|
|
} else {
|
|
elem[ internalKey ] = undefined;
|
|
}
|
|
|
|
deletedIds.push( id );
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
} );
|
|
|
|
jQuery.fn.extend( {
|
|
|
|
// Keep domManip exposed until 3.0 (gh-2225)
|
|
domManip: domManip,
|
|
|
|
detach: function( selector ) {
|
|
return remove( this, selector, true );
|
|
},
|
|
|
|
remove: function( selector ) {
|
|
return remove( this, selector );
|
|
},
|
|
|
|
text: function( value ) {
|
|
return access( this, function( value ) {
|
|
return value === undefined ?
|
|
jQuery.text( this ) :
|
|
this.empty().append(
|
|
( this[ 0 ] && this[ 0 ].ownerDocument || document ).createTextNode( value )
|
|
);
|
|
}, null, value, arguments.length );
|
|
},
|
|
|
|
append: function() {
|
|
return domManip( this, arguments, function( elem ) {
|
|
if ( this.nodeType === 1 || this.nodeType === 11 || this.nodeType === 9 ) {
|
|
var target = manipulationTarget( this, elem );
|
|
target.appendChild( elem );
|
|
}
|
|
} );
|
|
},
|
|
|
|
prepend: function() {
|
|
return domManip( this, arguments, function( elem ) {
|
|
if ( this.nodeType === 1 || this.nodeType === 11 || this.nodeType === 9 ) {
|
|
var target = manipulationTarget( this, elem );
|
|
target.insertBefore( elem, target.firstChild );
|
|
}
|
|
} );
|
|
},
|
|
|
|
before: function() {
|
|
return domManip( this, arguments, function( elem ) {
|
|
if ( this.parentNode ) {
|
|
this.parentNode.insertBefore( elem, this );
|
|
}
|
|
} );
|
|
},
|
|
|
|
after: function() {
|
|
return domManip( this, arguments, function( elem ) {
|
|
if ( this.parentNode ) {
|
|
this.parentNode.insertBefore( elem, this.nextSibling );
|
|
}
|
|
} );
|
|
},
|
|
|
|
empty: function() {
|
|
var elem,
|
|
i = 0;
|
|
|
|
for ( ; ( elem = this[ i ] ) != null; i++ ) {
|
|
|
|
// Remove element nodes and prevent memory leaks
|
|
if ( elem.nodeType === 1 ) {
|
|
jQuery.cleanData( getAll( elem, false ) );
|
|
}
|
|
|
|
// Remove any remaining nodes
|
|
while ( elem.firstChild ) {
|
|
elem.removeChild( elem.firstChild );
|
|
}
|
|
|
|
// If this is a select, ensure that it displays empty (#12336)
|
|
// Support: IE<9
|
|
if ( elem.options && jQuery.nodeName( elem, "select" ) ) {
|
|
elem.options.length = 0;
|
|
}
|
|
}
|
|
|
|
return this;
|
|
},
|
|
|
|
clone: function( dataAndEvents, deepDataAndEvents ) {
|
|
dataAndEvents = dataAndEvents == null ? false : dataAndEvents;
|
|
deepDataAndEvents = deepDataAndEvents == null ? dataAndEvents : deepDataAndEvents;
|
|
|
|
return this.map( function() {
|
|
return jQuery.clone( this, dataAndEvents, deepDataAndEvents );
|
|
} );
|
|
},
|
|
|
|
html: function( value ) {
|
|
return access( this, function( value ) {
|
|
var elem = this[ 0 ] || {},
|
|
i = 0,
|
|
l = this.length;
|
|
|
|
if ( value === undefined ) {
|
|
return elem.nodeType === 1 ?
|
|
elem.innerHTML.replace( rinlinejQuery, "" ) :
|
|
undefined;
|
|
}
|
|
|
|
// See if we can take a shortcut and just use innerHTML
|
|
if ( typeof value === "string" && !rnoInnerhtml.test( value ) &&
|
|
( support.htmlSerialize || !rnoshimcache.test( value ) ) &&
|
|
( support.leadingWhitespace || !rleadingWhitespace.test( value ) ) &&
|
|
!wrapMap[ ( rtagName.exec( value ) || [ "", "" ] )[ 1 ].toLowerCase() ] ) {
|
|
|
|
value = jQuery.htmlPrefilter( value );
|
|
|
|
try {
|
|
for ( ; i < l; i++ ) {
|
|
|
|
// Remove element nodes and prevent memory leaks
|
|
elem = this[ i ] || {};
|
|
if ( elem.nodeType === 1 ) {
|
|
jQuery.cleanData( getAll( elem, false ) );
|
|
elem.innerHTML = value;
|
|
}
|
|
}
|
|
|
|
elem = 0;
|
|
|
|
// If using innerHTML throws an exception, use the fallback method
|
|
} catch ( e ) {}
|
|
}
|
|
|
|
if ( elem ) {
|
|
this.empty().append( value );
|
|
}
|
|
}, null, value, arguments.length );
|
|
},
|
|
|
|
replaceWith: function() {
|
|
var ignored = [];
|
|
|
|
// Make the changes, replacing each non-ignored context element with the new content
|
|
return domManip( this, arguments, function( elem ) {
|
|
var parent = this.parentNode;
|
|
|
|
if ( jQuery.inArray( this, ignored ) < 0 ) {
|
|
jQuery.cleanData( getAll( this ) );
|
|
if ( parent ) {
|
|
parent.replaceChild( elem, this );
|
|
}
|
|
}
|
|
|
|
// Force callback invocation
|
|
}, ignored );
|
|
}
|
|
} );
|
|
|
|
jQuery.each( {
|
|
appendTo: "append",
|
|
prependTo: "prepend",
|
|
insertBefore: "before",
|
|
insertAfter: "after",
|
|
replaceAll: "replaceWith"
|
|
}, function( name, original ) {
|
|
jQuery.fn[ name ] = function( selector ) {
|
|
var elems,
|
|
i = 0,
|
|
ret = [],
|
|
insert = jQuery( selector ),
|
|
last = insert.length - 1;
|
|
|
|
for ( ; i <= last; i++ ) {
|
|
elems = i === last ? this : this.clone( true );
|
|
jQuery( insert[ i ] )[ original ]( elems );
|
|
|
|
// Modern browsers can apply jQuery collections as arrays, but oldIE needs a .get()
|
|
push.apply( ret, elems.get() );
|
|
}
|
|
|
|
return this.pushStack( ret );
|
|
};
|
|
} );
|
|
|
|
|
|
var iframe,
|
|
elemdisplay = {
|
|
|
|
// Support: Firefox
|
|
// We have to pre-define these values for FF (#10227)
|
|
HTML: "block",
|
|
BODY: "block"
|
|
};
|
|
|
|
/**
|
|
* Retrieve the actual display of a element
|
|
* @param {String} name nodeName of the element
|
|
* @param {Object} doc Document object
|
|
*/
|
|
|
|
// Called only from within defaultDisplay
|
|
function actualDisplay( name, doc ) {
|
|
var elem = jQuery( doc.createElement( name ) ).appendTo( doc.body ),
|
|
|
|
display = jQuery.css( elem[ 0 ], "display" );
|
|
|
|
// We don't have any data stored on the element,
|
|
// so use "detach" method as fast way to get rid of the element
|
|
elem.detach();
|
|
|
|
return display;
|
|
}
|
|
|
|
/**
|
|
* Try to determine the default display value of an element
|
|
* @param {String} nodeName
|
|
*/
|
|
function defaultDisplay( nodeName ) {
|
|
var doc = document,
|
|
display = elemdisplay[ nodeName ];
|
|
|
|
if ( !display ) {
|
|
display = actualDisplay( nodeName, doc );
|
|
|
|
// If the simple way fails, read from inside an iframe
|
|
if ( display === "none" || !display ) {
|
|
|
|
// Use the already-created iframe if possible
|
|
iframe = ( iframe || jQuery( "<iframe frameborder='0' width='0' height='0'/>" ) )
|
|
.appendTo( doc.documentElement );
|
|
|
|
// Always write a new HTML skeleton so Webkit and Firefox don't choke on reuse
|
|
doc = ( iframe[ 0 ].contentWindow || iframe[ 0 ].contentDocument ).document;
|
|
|
|
// Support: IE
|
|
doc.write();
|
|
doc.close();
|
|
|
|
display = actualDisplay( nodeName, doc );
|
|
iframe.detach();
|
|
}
|
|
|
|
// Store the correct default display
|
|
elemdisplay[ nodeName ] = display;
|
|
}
|
|
|
|
return display;
|
|
}
|
|
var rmargin = ( /^margin/ );
|
|
|
|
var rnumnonpx = new RegExp( "^(" + pnum + ")(?!px)[a-z%]+$", "i" );
|
|
|
|
var swap = function( elem, options, callback, args ) {
|
|
var ret, name,
|
|
old = {};
|
|
|
|
// Remember the old values, and insert the new ones
|
|
for ( name in options ) {
|
|
old[ name ] = elem.style[ name ];
|
|
elem.style[ name ] = options[ name ];
|
|
}
|
|
|
|
ret = callback.apply( elem, args || [] );
|
|
|
|
// Revert the old values
|
|
for ( name in options ) {
|
|
elem.style[ name ] = old[ name ];
|
|
}
|
|
|
|
return ret;
|
|
};
|
|
|
|
|
|
var documentElement = document.documentElement;
|
|
|
|
|
|
|
|
( function() {
|
|
var pixelPositionVal, pixelMarginRightVal, boxSizingReliableVal,
|
|
reliableHiddenOffsetsVal, reliableMarginRightVal, reliableMarginLeftVal,
|
|
container = document.createElement( "div" ),
|
|
div = document.createElement( "div" );
|
|
|
|
// Finish early in limited (non-browser) environments
|
|
if ( !div.style ) {
|
|
return;
|
|
}
|
|
|
|
div.style.cssText = "float:left;opacity:.5";
|
|
|
|
// Support: IE<9
|
|
// Make sure that element opacity exists (as opposed to filter)
|
|
support.opacity = div.style.opacity === "0.5";
|
|
|
|
// Verify style float existence
|
|
// (IE uses styleFloat instead of cssFloat)
|
|
support.cssFloat = !!div.style.cssFloat;
|
|
|
|
div.style.backgroundClip = "content-box";
|
|
div.cloneNode( true ).style.backgroundClip = "";
|
|
support.clearCloneStyle = div.style.backgroundClip === "content-box";
|
|
|
|
container = document.createElement( "div" );
|
|
container.style.cssText = "border:0;width:8px;height:0;top:0;left:-9999px;" +
|
|
"padding:0;margin-top:1px;position:absolute";
|
|
div.innerHTML = "";
|
|
container.appendChild( div );
|
|
|
|
// Support: Firefox<29, Android 2.3
|
|
// Vendor-prefix box-sizing
|
|
support.boxSizing = div.style.boxSizing === "" || div.style.MozBoxSizing === "" ||
|
|
div.style.WebkitBoxSizing === "";
|
|
|
|
jQuery.extend( support, {
|
|
reliableHiddenOffsets: function() {
|
|
if ( pixelPositionVal == null ) {
|
|
computeStyleTests();
|
|
}
|
|
return reliableHiddenOffsetsVal;
|
|
},
|
|
|
|
boxSizingReliable: function() {
|
|
|
|
// We're checking for pixelPositionVal here instead of boxSizingReliableVal
|
|
// since that compresses better and they're computed together anyway.
|
|
if ( pixelPositionVal == null ) {
|
|
computeStyleTests();
|
|
}
|
|
return boxSizingReliableVal;
|
|
},
|
|
|
|
pixelMarginRight: function() {
|
|
|
|
// Support: Android 4.0-4.3
|
|
if ( pixelPositionVal == null ) {
|
|
computeStyleTests();
|
|
}
|
|
return pixelMarginRightVal;
|
|
},
|
|
|
|
pixelPosition: function() {
|
|
if ( pixelPositionVal == null ) {
|
|
computeStyleTests();
|
|
}
|
|
return pixelPositionVal;
|
|
},
|
|
|
|
reliableMarginRight: function() {
|
|
|
|
// Support: Android 2.3
|
|
if ( pixelPositionVal == null ) {
|
|
computeStyleTests();
|
|
}
|
|
return reliableMarginRightVal;
|
|
},
|
|
|
|
reliableMarginLeft: function() {
|
|
|
|
// Support: IE <=8 only, Android 4.0 - 4.3 only, Firefox <=3 - 37
|
|
if ( pixelPositionVal == null ) {
|
|
computeStyleTests();
|
|
}
|
|
return reliableMarginLeftVal;
|
|
}
|
|
} );
|
|
|
|
function computeStyleTests() {
|
|
var contents, divStyle,
|
|
documentElement = document.documentElement;
|
|
|
|
// Setup
|
|
documentElement.appendChild( container );
|
|
|
|
div.style.cssText =
|
|
|
|
// Support: Android 2.3
|
|
// Vendor-prefix box-sizing
|
|
"-webkit-box-sizing:border-box;box-sizing:border-box;" +
|
|
"position:relative;display:block;" +
|
|
"margin:auto;border:1px;padding:1px;" +
|
|
"top:1%;width:50%";
|
|
|
|
// Support: IE<9
|
|
// Assume reasonable values in the absence of getComputedStyle
|
|
pixelPositionVal = boxSizingReliableVal = reliableMarginLeftVal = false;
|
|
pixelMarginRightVal = reliableMarginRightVal = true;
|
|
|
|
// Check for getComputedStyle so that this code is not run in IE<9.
|
|
if ( window.getComputedStyle ) {
|
|
divStyle = window.getComputedStyle( div );
|
|
pixelPositionVal = ( divStyle || {} ).top !== "1%";
|
|
reliableMarginLeftVal = ( divStyle || {} ).marginLeft === "2px";
|
|
boxSizingReliableVal = ( divStyle || { width: "4px" } ).width === "4px";
|
|
|
|
// Support: Android 4.0 - 4.3 only
|
|
// Some styles come back with percentage values, even though they shouldn't
|
|
div.style.marginRight = "50%";
|
|
pixelMarginRightVal = ( divStyle || { marginRight: "4px" } ).marginRight === "4px";
|
|
|
|
// Support: Android 2.3 only
|
|
// Div with explicit width and no margin-right incorrectly
|
|
// gets computed margin-right based on width of container (#3333)
|
|
// WebKit Bug 13343 - getComputedStyle returns wrong value for margin-right
|
|
contents = div.appendChild( document.createElement( "div" ) );
|
|
|
|
// Reset CSS: box-sizing; display; margin; border; padding
|
|
contents.style.cssText = div.style.cssText =
|
|
|
|
// Support: Android 2.3
|
|
// Vendor-prefix box-sizing
|
|
"-webkit-box-sizing:content-box;-moz-box-sizing:content-box;" +
|
|
"box-sizing:content-box;display:block;margin:0;border:0;padding:0";
|
|
contents.style.marginRight = contents.style.width = "0";
|
|
div.style.width = "1px";
|
|
|
|
reliableMarginRightVal =
|
|
!parseFloat( ( window.getComputedStyle( contents ) || {} ).marginRight );
|
|
|
|
div.removeChild( contents );
|
|
}
|
|
|
|
// Support: IE6-8
|
|
// First check that getClientRects works as expected
|
|
// Check if table cells still have offsetWidth/Height when they are set
|
|
// to display:none and there are still other visible table cells in a
|
|
// table row; if so, offsetWidth/Height are not reliable for use when
|
|
// determining if an element has been hidden directly using
|
|
// display:none (it is still safe to use offsets if a parent element is
|
|
// hidden; don safety goggles and see bug #4512 for more information).
|
|
div.style.display = "none";
|
|
reliableHiddenOffsetsVal = div.getClientRects().length === 0;
|
|
if ( reliableHiddenOffsetsVal ) {
|
|
div.style.display = "";
|
|
div.innerHTML = "<table><tr><td></td><td>t</td></tr></table>";
|
|
contents = div.getElementsByTagName( "td" );
|
|
contents[ 0 ].style.cssText = "margin:0;border:0;padding:0;display:none";
|
|
reliableHiddenOffsetsVal = contents[ 0 ].offsetHeight === 0;
|
|
if ( reliableHiddenOffsetsVal ) {
|
|
contents[ 0 ].style.display = "";
|
|
contents[ 1 ].style.display = "none";
|
|
reliableHiddenOffsetsVal = contents[ 0 ].offsetHeight === 0;
|
|
}
|
|
}
|
|
|
|
// Teardown
|
|
documentElement.removeChild( container );
|
|
}
|
|
|
|
} )();
|
|
|
|
|
|
var getStyles, curCSS,
|
|
rposition = /^(top|right|bottom|left)$/;
|
|
|
|
if ( window.getComputedStyle ) {
|
|
getStyles = function( elem ) {
|
|
|
|
// Support: IE<=11+, Firefox<=30+ (#15098, #14150)
|
|
// IE throws on elements created in popups
|
|
// FF meanwhile throws on frame elements through "defaultView.getComputedStyle"
|
|
var view = elem.ownerDocument.defaultView;
|
|
|
|
if ( !view || !view.opener ) {
|
|
view = window;
|
|
}
|
|
|
|
return view.getComputedStyle( elem );
|
|
};
|
|
|
|
curCSS = function( elem, name, computed ) {
|
|
var width, minWidth, maxWidth, ret,
|
|
style = elem.style;
|
|
|
|
computed = computed || getStyles( elem );
|
|
|
|
// getPropertyValue is only needed for .css('filter') in IE9, see #12537
|
|
ret = computed ? computed.getPropertyValue( name ) || computed[ name ] : undefined;
|
|
|
|
// Support: Opera 12.1x only
|
|
// Fall back to style even without computed
|
|
// computed is undefined for elems on document fragments
|
|
if ( ( ret === "" || ret === undefined ) && !jQuery.contains( elem.ownerDocument, elem ) ) {
|
|
ret = jQuery.style( elem, name );
|
|
}
|
|
|
|
if ( computed ) {
|
|
|
|
// A tribute to the "awesome hack by Dean Edwards"
|
|
// Chrome < 17 and Safari 5.0 uses "computed value"
|
|
// instead of "used value" for margin-right
|
|
// Safari 5.1.7 (at least) returns percentage for a larger set of values,
|
|
// but width seems to be reliably pixels
|
|
// this is against the CSSOM draft spec:
|
|
// http://dev.w3.org/csswg/cssom/#resolved-values
|
|
if ( !support.pixelMarginRight() && rnumnonpx.test( ret ) && rmargin.test( name ) ) {
|
|
|
|
// Remember the original values
|
|
width = style.width;
|
|
minWidth = style.minWidth;
|
|
maxWidth = style.maxWidth;
|
|
|
|
// Put in the new values to get a computed value out
|
|
style.minWidth = style.maxWidth = style.width = ret;
|
|
ret = computed.width;
|
|
|
|
// Revert the changed values
|
|
style.width = width;
|
|
style.minWidth = minWidth;
|
|
style.maxWidth = maxWidth;
|
|
}
|
|
}
|
|
|
|
// Support: IE
|
|
// IE returns zIndex value as an integer.
|
|
return ret === undefined ?
|
|
ret :
|
|
ret + "";
|
|
};
|
|
} else if ( documentElement.currentStyle ) {
|
|
getStyles = function( elem ) {
|
|
return elem.currentStyle;
|
|
};
|
|
|
|
curCSS = function( elem, name, computed ) {
|
|
var left, rs, rsLeft, ret,
|
|
style = elem.style;
|
|
|
|
computed = computed || getStyles( elem );
|
|
ret = computed ? computed[ name ] : undefined;
|
|
|
|
// Avoid setting ret to empty string here
|
|
// so we don't default to auto
|
|
if ( ret == null && style && style[ name ] ) {
|
|
ret = style[ name ];
|
|
}
|
|
|
|
// From the awesome hack by Dean Edwards
|
|
// http://erik.eae.net/archives/2007/07/27/18.54.15/#comment-102291
|
|
|
|
// If we're not dealing with a regular pixel number
|
|
// but a number that has a weird ending, we need to convert it to pixels
|
|
// but not position css attributes, as those are
|
|
// proportional to the parent element instead
|
|
// and we can't measure the parent instead because it
|
|
// might trigger a "stacking dolls" problem
|
|
if ( rnumnonpx.test( ret ) && !rposition.test( name ) ) {
|
|
|
|
// Remember the original values
|
|
left = style.left;
|
|
rs = elem.runtimeStyle;
|
|
rsLeft = rs && rs.left;
|
|
|
|
// Put in the new values to get a computed value out
|
|
if ( rsLeft ) {
|
|
rs.left = elem.currentStyle.left;
|
|
}
|
|
style.left = name === "fontSize" ? "1em" : ret;
|
|
ret = style.pixelLeft + "px";
|
|
|
|
// Revert the changed values
|
|
style.left = left;
|
|
if ( rsLeft ) {
|
|
rs.left = rsLeft;
|
|
}
|
|
}
|
|
|
|
// Support: IE
|
|
// IE returns zIndex value as an integer.
|
|
return ret === undefined ?
|
|
ret :
|
|
ret + "" || "auto";
|
|
};
|
|
}
|
|
|
|
|
|
|
|
|
|
function addGetHookIf( conditionFn, hookFn ) {
|
|
|
|
// Define the hook, we'll check on the first run if it's really needed.
|
|
return {
|
|
get: function() {
|
|
if ( conditionFn() ) {
|
|
|
|
// Hook not needed (or it's not possible to use it due
|
|
// to missing dependency), remove it.
|
|
delete this.get;
|
|
return;
|
|
}
|
|
|
|
// Hook needed; redefine it so that the support test is not executed again.
|
|
return ( this.get = hookFn ).apply( this, arguments );
|
|
}
|
|
};
|
|
}
|
|
|
|
|
|
var
|
|
|
|
ralpha = /alpha\([^)]*\)/i,
|
|
ropacity = /opacity\s*=\s*([^)]*)/i,
|
|
|
|
// swappable if display is none or starts with table except
|
|
// "table", "table-cell", or "table-caption"
|
|
// see here for display values:
|
|
// https://developer.mozilla.org/en-US/docs/CSS/display
|
|
rdisplayswap = /^(none|table(?!-c[ea]).+)/,
|
|
rnumsplit = new RegExp( "^(" + pnum + ")(.*)$", "i" ),
|
|
|
|
cssShow = { position: "absolute", visibility: "hidden", display: "block" },
|
|
cssNormalTransform = {
|
|
letterSpacing: "0",
|
|
fontWeight: "400"
|
|
},
|
|
|
|
cssPrefixes = [ "Webkit", "O", "Moz", "ms" ],
|
|
emptyStyle = document.createElement( "div" ).style;
|
|
|
|
|
|
// return a css property mapped to a potentially vendor prefixed property
|
|
function vendorPropName( name ) {
|
|
|
|
// shortcut for names that are not vendor prefixed
|
|
if ( name in emptyStyle ) {
|
|
return name;
|
|
}
|
|
|
|
// check for vendor prefixed names
|
|
var capName = name.charAt( 0 ).toUpperCase() + name.slice( 1 ),
|
|
i = cssPrefixes.length;
|
|
|
|
while ( i-- ) {
|
|
name = cssPrefixes[ i ] + capName;
|
|
if ( name in emptyStyle ) {
|
|
return name;
|
|
}
|
|
}
|
|
}
|
|
|
|
function showHide( elements, show ) {
|
|
var display, elem, hidden,
|
|
values = [],
|
|
index = 0,
|
|
length = elements.length;
|
|
|
|
for ( ; index < length; index++ ) {
|
|
elem = elements[ index ];
|
|
if ( !elem.style ) {
|
|
continue;
|
|
}
|
|
|
|
values[ index ] = jQuery._data( elem, "olddisplay" );
|
|
display = elem.style.display;
|
|
if ( show ) {
|
|
|
|
// Reset the inline display of this element to learn if it is
|
|
// being hidden by cascaded rules or not
|
|
if ( !values[ index ] && display === "none" ) {
|
|
elem.style.display = "";
|
|
}
|
|
|
|
// Set elements which have been overridden with display: none
|
|
// in a stylesheet to whatever the default browser style is
|
|
// for such an element
|
|
if ( elem.style.display === "" && isHidden( elem ) ) {
|
|
values[ index ] =
|
|
jQuery._data( elem, "olddisplay", defaultDisplay( elem.nodeName ) );
|
|
}
|
|
} else {
|
|
hidden = isHidden( elem );
|
|
|
|
if ( display && display !== "none" || !hidden ) {
|
|
jQuery._data(
|
|
elem,
|
|
"olddisplay",
|
|
hidden ? display : jQuery.css( elem, "display" )
|
|
);
|
|
}
|
|
}
|
|
}
|
|
|
|
// Set the display of most of the elements in a second loop
|
|
// to avoid the constant reflow
|
|
for ( index = 0; index < length; index++ ) {
|
|
elem = elements[ index ];
|
|
if ( !elem.style ) {
|
|
continue;
|
|
}
|
|
if ( !show || elem.style.display === "none" || elem.style.display === "" ) {
|
|
elem.style.display = show ? values[ index ] || "" : "none";
|
|
}
|
|
}
|
|
|
|
return elements;
|
|
}
|
|
|
|
function setPositiveNumber( elem, value, subtract ) {
|
|
var matches = rnumsplit.exec( value );
|
|
return matches ?
|
|
|
|
// Guard against undefined "subtract", e.g., when used as in cssHooks
|
|
Math.max( 0, matches[ 1 ] - ( subtract || 0 ) ) + ( matches[ 2 ] || "px" ) :
|
|
value;
|
|
}
|
|
|
|
function augmentWidthOrHeight( elem, name, extra, isBorderBox, styles ) {
|
|
var i = extra === ( isBorderBox ? "border" : "content" ) ?
|
|
|
|
// If we already have the right measurement, avoid augmentation
|
|
4 :
|
|
|
|
// Otherwise initialize for horizontal or vertical properties
|
|
name === "width" ? 1 : 0,
|
|
|
|
val = 0;
|
|
|
|
for ( ; i < 4; i += 2 ) {
|
|
|
|
// both box models exclude margin, so add it if we want it
|
|
if ( extra === "margin" ) {
|
|
val += jQuery.css( elem, extra + cssExpand[ i ], true, styles );
|
|
}
|
|
|
|
if ( isBorderBox ) {
|
|
|
|
// border-box includes padding, so remove it if we want content
|
|
if ( extra === "content" ) {
|
|
val -= jQuery.css( elem, "padding" + cssExpand[ i ], true, styles );
|
|
}
|
|
|
|
// at this point, extra isn't border nor margin, so remove border
|
|
if ( extra !== "margin" ) {
|
|
val -= jQuery.css( elem, "border" + cssExpand[ i ] + "Width", true, styles );
|
|
}
|
|
} else {
|
|
|
|
// at this point, extra isn't content, so add padding
|
|
val += jQuery.css( elem, "padding" + cssExpand[ i ], true, styles );
|
|
|
|
// at this point, extra isn't content nor padding, so add border
|
|
if ( extra !== "padding" ) {
|
|
val += jQuery.css( elem, "border" + cssExpand[ i ] + "Width", true, styles );
|
|
}
|
|
}
|
|
}
|
|
|
|
return val;
|
|
}
|
|
|
|
function getWidthOrHeight( elem, name, extra ) {
|
|
|
|
// Start with offset property, which is equivalent to the border-box value
|
|
var valueIsBorderBox = true,
|
|
val = name === "width" ? elem.offsetWidth : elem.offsetHeight,
|
|
styles = getStyles( elem ),
|
|
isBorderBox = support.boxSizing &&
|
|
jQuery.css( elem, "boxSizing", false, styles ) === "border-box";
|
|
|
|
// Support: IE11 only
|
|
// In IE 11 fullscreen elements inside of an iframe have
|
|
// 100x too small dimensions (gh-1764).
|
|
if ( document.msFullscreenElement && window.top !== window ) {
|
|
|
|
// Support: IE11 only
|
|
// Running getBoundingClientRect on a disconnected node
|
|
// in IE throws an error.
|
|
if ( elem.getClientRects().length ) {
|
|
val = Math.round( elem.getBoundingClientRect()[ name ] * 100 );
|
|
}
|
|
}
|
|
|
|
// some non-html elements return undefined for offsetWidth, so check for null/undefined
|
|
// svg - https://bugzilla.mozilla.org/show_bug.cgi?id=649285
|
|
// MathML - https://bugzilla.mozilla.org/show_bug.cgi?id=491668
|
|
if ( val <= 0 || val == null ) {
|
|
|
|
// Fall back to computed then uncomputed css if necessary
|
|
val = curCSS( elem, name, styles );
|
|
if ( val < 0 || val == null ) {
|
|
val = elem.style[ name ];
|
|
}
|
|
|
|
// Computed unit is not pixels. Stop here and return.
|
|
if ( rnumnonpx.test( val ) ) {
|
|
return val;
|
|
}
|
|
|
|
// we need the check for style in case a browser which returns unreliable values
|
|
// for getComputedStyle silently falls back to the reliable elem.style
|
|
valueIsBorderBox = isBorderBox &&
|
|
( support.boxSizingReliable() || val === elem.style[ name ] );
|
|
|
|
// Normalize "", auto, and prepare for extra
|
|
val = parseFloat( val ) || 0;
|
|
}
|
|
|
|
// use the active box-sizing model to add/subtract irrelevant styles
|
|
return ( val +
|
|
augmentWidthOrHeight(
|
|
elem,
|
|
name,
|
|
extra || ( isBorderBox ? "border" : "content" ),
|
|
valueIsBorderBox,
|
|
styles
|
|
)
|
|
) + "px";
|
|
}
|
|
|
|
jQuery.extend( {
|
|
|
|
// Add in style property hooks for overriding the default
|
|
// behavior of getting and setting a style property
|
|
cssHooks: {
|
|
opacity: {
|
|
get: function( elem, computed ) {
|
|
if ( computed ) {
|
|
|
|
// We should always get a number back from opacity
|
|
var ret = curCSS( elem, "opacity" );
|
|
return ret === "" ? "1" : ret;
|
|
}
|
|
}
|
|
}
|
|
},
|
|
|
|
// Don't automatically add "px" to these possibly-unitless properties
|
|
cssNumber: {
|
|
"animationIterationCount": true,
|
|
"columnCount": true,
|
|
"fillOpacity": true,
|
|
"flexGrow": true,
|
|
"flexShrink": true,
|
|
"fontWeight": true,
|
|
"lineHeight": true,
|
|
"opacity": true,
|
|
"order": true,
|
|
"orphans": true,
|
|
"widows": true,
|
|
"zIndex": true,
|
|
"zoom": true
|
|
},
|
|
|
|
// Add in properties whose names you wish to fix before
|
|
// setting or getting the value
|
|
cssProps: {
|
|
|
|
// normalize float css property
|
|
"float": support.cssFloat ? "cssFloat" : "styleFloat"
|
|
},
|
|
|
|
// Get and set the style property on a DOM Node
|
|
style: function( elem, name, value, extra ) {
|
|
|
|
// Don't set styles on text and comment nodes
|
|
if ( !elem || elem.nodeType === 3 || elem.nodeType === 8 || !elem.style ) {
|
|
return;
|
|
}
|
|
|
|
// Make sure that we're working with the right name
|
|
var ret, type, hooks,
|
|
origName = jQuery.camelCase( name ),
|
|
style = elem.style;
|
|
|
|
name = jQuery.cssProps[ origName ] ||
|
|
( jQuery.cssProps[ origName ] = vendorPropName( origName ) || origName );
|
|
|
|
// gets hook for the prefixed version
|
|
// followed by the unprefixed version
|
|
hooks = jQuery.cssHooks[ name ] || jQuery.cssHooks[ origName ];
|
|
|
|
// Check if we're setting a value
|
|
if ( value !== undefined ) {
|
|
type = typeof value;
|
|
|
|
// Convert "+=" or "-=" to relative numbers (#7345)
|
|
if ( type === "string" && ( ret = rcssNum.exec( value ) ) && ret[ 1 ] ) {
|
|
value = adjustCSS( elem, name, ret );
|
|
|
|
// Fixes bug #9237
|
|
type = "number";
|
|
}
|
|
|
|
// Make sure that null and NaN values aren't set. See: #7116
|
|
if ( value == null || value !== value ) {
|
|
return;
|
|
}
|
|
|
|
// If a number was passed in, add the unit (except for certain CSS properties)
|
|
if ( type === "number" ) {
|
|
value += ret && ret[ 3 ] || ( jQuery.cssNumber[ origName ] ? "" : "px" );
|
|
}
|
|
|
|
// Fixes #8908, it can be done more correctly by specifing setters in cssHooks,
|
|
// but it would mean to define eight
|
|
// (for every problematic property) identical functions
|
|
if ( !support.clearCloneStyle && value === "" && name.indexOf( "background" ) === 0 ) {
|
|
style[ name ] = "inherit";
|
|
}
|
|
|
|
// If a hook was provided, use that value, otherwise just set the specified value
|
|
if ( !hooks || !( "set" in hooks ) ||
|
|
( value = hooks.set( elem, value, extra ) ) !== undefined ) {
|
|
|
|
// Support: IE
|
|
// Swallow errors from 'invalid' CSS values (#5509)
|
|
try {
|
|
style[ name ] = value;
|
|
} catch ( e ) {}
|
|
}
|
|
|
|
} else {
|
|
|
|
// If a hook was provided get the non-computed value from there
|
|
if ( hooks && "get" in hooks &&
|
|
( ret = hooks.get( elem, false, extra ) ) !== undefined ) {
|
|
|
|
return ret;
|
|
}
|
|
|
|
// Otherwise just get the value from the style object
|
|
return style[ name ];
|
|
}
|
|
},
|
|
|
|
css: function( elem, name, extra, styles ) {
|
|
var num, val, hooks,
|
|
origName = jQuery.camelCase( name );
|
|
|
|
// Make sure that we're working with the right name
|
|
name = jQuery.cssProps[ origName ] ||
|
|
( jQuery.cssProps[ origName ] = vendorPropName( origName ) || origName );
|
|
|
|
// gets hook for the prefixed version
|
|
// followed by the unprefixed version
|
|
hooks = jQuery.cssHooks[ name ] || jQuery.cssHooks[ origName ];
|
|
|
|
// If a hook was provided get the computed value from there
|
|
if ( hooks && "get" in hooks ) {
|
|
val = hooks.get( elem, true, extra );
|
|
}
|
|
|
|
// Otherwise, if a way to get the computed value exists, use that
|
|
if ( val === undefined ) {
|
|
val = curCSS( elem, name, styles );
|
|
}
|
|
|
|
//convert "normal" to computed value
|
|
if ( val === "normal" && name in cssNormalTransform ) {
|
|
val = cssNormalTransform[ name ];
|
|
}
|
|
|
|
// Return, converting to number if forced or a qualifier was provided and val looks numeric
|
|
if ( extra === "" || extra ) {
|
|
num = parseFloat( val );
|
|
return extra === true || isFinite( num ) ? num || 0 : val;
|
|
}
|
|
return val;
|
|
}
|
|
} );
|
|
|
|
jQuery.each( [ "height", "width" ], function( i, name ) {
|
|
jQuery.cssHooks[ name ] = {
|
|
get: function( elem, computed, extra ) {
|
|
if ( computed ) {
|
|
|
|
// certain elements can have dimension info if we invisibly show them
|
|
// however, it must have a current display style that would benefit from this
|
|
return rdisplayswap.test( jQuery.css( elem, "display" ) ) &&
|
|
elem.offsetWidth === 0 ?
|
|
swap( elem, cssShow, function() {
|
|
return getWidthOrHeight( elem, name, extra );
|
|
} ) :
|
|
getWidthOrHeight( elem, name, extra );
|
|
}
|
|
},
|
|
|
|
set: function( elem, value, extra ) {
|
|
var styles = extra && getStyles( elem );
|
|
return setPositiveNumber( elem, value, extra ?
|
|
augmentWidthOrHeight(
|
|
elem,
|
|
name,
|
|
extra,
|
|
support.boxSizing &&
|
|
jQuery.css( elem, "boxSizing", false, styles ) === "border-box",
|
|
styles
|
|
) : 0
|
|
);
|
|
}
|
|
};
|
|
} );
|
|
|
|
if ( !support.opacity ) {
|
|
jQuery.cssHooks.opacity = {
|
|
get: function( elem, computed ) {
|
|
|
|
// IE uses filters for opacity
|
|
return ropacity.test( ( computed && elem.currentStyle ?
|
|
elem.currentStyle.filter :
|
|
elem.style.filter ) || "" ) ?
|
|
( 0.01 * parseFloat( RegExp.$1 ) ) + "" :
|
|
computed ? "1" : "";
|
|
},
|
|
|
|
set: function( elem, value ) {
|
|
var style = elem.style,
|
|
currentStyle = elem.currentStyle,
|
|
opacity = jQuery.isNumeric( value ) ? "alpha(opacity=" + value * 100 + ")" : "",
|
|
filter = currentStyle && currentStyle.filter || style.filter || "";
|
|
|
|
// IE has trouble with opacity if it does not have layout
|
|
// Force it by setting the zoom level
|
|
style.zoom = 1;
|
|
|
|
// if setting opacity to 1, and no other filters exist -
|
|
// attempt to remove filter attribute #6652
|
|
// if value === "", then remove inline opacity #12685
|
|
if ( ( value >= 1 || value === "" ) &&
|
|
jQuery.trim( filter.replace( ralpha, "" ) ) === "" &&
|
|
style.removeAttribute ) {
|
|
|
|
// Setting style.filter to null, "" & " " still leave "filter:" in the cssText
|
|
// if "filter:" is present at all, clearType is disabled, we want to avoid this
|
|
// style.removeAttribute is IE Only, but so apparently is this code path...
|
|
style.removeAttribute( "filter" );
|
|
|
|
// if there is no filter style applied in a css rule
|
|
// or unset inline opacity, we are done
|
|
if ( value === "" || currentStyle && !currentStyle.filter ) {
|
|
return;
|
|
}
|
|
}
|
|
|
|
// otherwise, set new filter values
|
|
style.filter = ralpha.test( filter ) ?
|
|
filter.replace( ralpha, opacity ) :
|
|
filter + " " + opacity;
|
|
}
|
|
};
|
|
}
|
|
|
|
jQuery.cssHooks.marginRight = addGetHookIf( support.reliableMarginRight,
|
|
function( elem, computed ) {
|
|
if ( computed ) {
|
|
return swap( elem, { "display": "inline-block" },
|
|
curCSS, [ elem, "marginRight" ] );
|
|
}
|
|
}
|
|
);
|
|
|
|
jQuery.cssHooks.marginLeft = addGetHookIf( support.reliableMarginLeft,
|
|
function( elem, computed ) {
|
|
if ( computed ) {
|
|
return (
|
|
parseFloat( curCSS( elem, "marginLeft" ) ) ||
|
|
|
|
// Support: IE<=11+
|
|
// Running getBoundingClientRect on a disconnected node in IE throws an error
|
|
// Support: IE8 only
|
|
// getClientRects() errors on disconnected elems
|
|
( jQuery.contains( elem.ownerDocument, elem ) ?
|
|
elem.getBoundingClientRect().left -
|
|
swap( elem, { marginLeft: 0 }, function() {
|
|
return elem.getBoundingClientRect().left;
|
|
} ) :
|
|
0
|
|
)
|
|
) + "px";
|
|
}
|
|
}
|
|
);
|
|
|
|
// These hooks are used by animate to expand properties
|
|
jQuery.each( {
|
|
margin: "",
|
|
padding: "",
|
|
border: "Width"
|
|
}, function( prefix, suffix ) {
|
|
jQuery.cssHooks[ prefix + suffix ] = {
|
|
expand: function( value ) {
|
|
var i = 0,
|
|
expanded = {},
|
|
|
|
// assumes a single number if not a string
|
|
parts = typeof value === "string" ? value.split( " " ) : [ value ];
|
|
|
|
for ( ; i < 4; i++ ) {
|
|
expanded[ prefix + cssExpand[ i ] + suffix ] =
|
|
parts[ i ] || parts[ i - 2 ] || parts[ 0 ];
|
|
}
|
|
|
|
return expanded;
|
|
}
|
|
};
|
|
|
|
if ( !rmargin.test( prefix ) ) {
|
|
jQuery.cssHooks[ prefix + suffix ].set = setPositiveNumber;
|
|
}
|
|
} );
|
|
|
|
jQuery.fn.extend( {
|
|
css: function( name, value ) {
|
|
return access( this, function( elem, name, value ) {
|
|
var styles, len,
|
|
map = {},
|
|
i = 0;
|
|
|
|
if ( jQuery.isArray( name ) ) {
|
|
styles = getStyles( elem );
|
|
len = name.length;
|
|
|
|
for ( ; i < len; i++ ) {
|
|
map[ name[ i ] ] = jQuery.css( elem, name[ i ], false, styles );
|
|
}
|
|
|
|
return map;
|
|
}
|
|
|
|
return value !== undefined ?
|
|
jQuery.style( elem, name, value ) :
|
|
jQuery.css( elem, name );
|
|
}, name, value, arguments.length > 1 );
|
|
},
|
|
show: function() {
|
|
return showHide( this, true );
|
|
},
|
|
hide: function() {
|
|
return showHide( this );
|
|
},
|
|
toggle: function( state ) {
|
|
if ( typeof state === "boolean" ) {
|
|
return state ? this.show() : this.hide();
|
|
}
|
|
|
|
return this.each( function() {
|
|
if ( isHidden( this ) ) {
|
|
jQuery( this ).show();
|
|
} else {
|
|
jQuery( this ).hide();
|
|
}
|
|
} );
|
|
}
|
|
} );
|
|
|
|
|
|
function Tween( elem, options, prop, end, easing ) {
|
|
return new Tween.prototype.init( elem, options, prop, end, easing );
|
|
}
|
|
jQuery.Tween = Tween;
|
|
|
|
Tween.prototype = {
|
|
constructor: Tween,
|
|
init: function( elem, options, prop, end, easing, unit ) {
|
|
this.elem = elem;
|
|
this.prop = prop;
|
|
this.easing = easing || jQuery.easing._default;
|
|
this.options = options;
|
|
this.start = this.now = this.cur();
|
|
this.end = end;
|
|
this.unit = unit || ( jQuery.cssNumber[ prop ] ? "" : "px" );
|
|
},
|
|
cur: function() {
|
|
var hooks = Tween.propHooks[ this.prop ];
|
|
|
|
return hooks && hooks.get ?
|
|
hooks.get( this ) :
|
|
Tween.propHooks._default.get( this );
|
|
},
|
|
run: function( percent ) {
|
|
var eased,
|
|
hooks = Tween.propHooks[ this.prop ];
|
|
|
|
if ( this.options.duration ) {
|
|
this.pos = eased = jQuery.easing[ this.easing ](
|
|
percent, this.options.duration * percent, 0, 1, this.options.duration
|
|
);
|
|
} else {
|
|
this.pos = eased = percent;
|
|
}
|
|
this.now = ( this.end - this.start ) * eased + this.start;
|
|
|
|
if ( this.options.step ) {
|
|
this.options.step.call( this.elem, this.now, this );
|
|
}
|
|
|
|
if ( hooks && hooks.set ) {
|
|
hooks.set( this );
|
|
} else {
|
|
Tween.propHooks._default.set( this );
|
|
}
|
|
return this;
|
|
}
|
|
};
|
|
|
|
Tween.prototype.init.prototype = Tween.prototype;
|
|
|
|
Tween.propHooks = {
|
|
_default: {
|
|
get: function( tween ) {
|
|
var result;
|
|
|
|
// Use a property on the element directly when it is not a DOM element,
|
|
// or when there is no matching style property that exists.
|
|
if ( tween.elem.nodeType !== 1 ||
|
|
tween.elem[ tween.prop ] != null && tween.elem.style[ tween.prop ] == null ) {
|
|
return tween.elem[ tween.prop ];
|
|
}
|
|
|
|
// passing an empty string as a 3rd parameter to .css will automatically
|
|
// attempt a parseFloat and fallback to a string if the parse fails
|
|
// so, simple values such as "10px" are parsed to Float.
|
|
// complex values such as "rotate(1rad)" are returned as is.
|
|
result = jQuery.css( tween.elem, tween.prop, "" );
|
|
|
|
// Empty strings, null, undefined and "auto" are converted to 0.
|
|
return !result || result === "auto" ? 0 : result;
|
|
},
|
|
set: function( tween ) {
|
|
|
|
// use step hook for back compat - use cssHook if its there - use .style if its
|
|
// available and use plain properties where available
|
|
if ( jQuery.fx.step[ tween.prop ] ) {
|
|
jQuery.fx.step[ tween.prop ]( tween );
|
|
} else if ( tween.elem.nodeType === 1 &&
|
|
( tween.elem.style[ jQuery.cssProps[ tween.prop ] ] != null ||
|
|
jQuery.cssHooks[ tween.prop ] ) ) {
|
|
jQuery.style( tween.elem, tween.prop, tween.now + tween.unit );
|
|
} else {
|
|
tween.elem[ tween.prop ] = tween.now;
|
|
}
|
|
}
|
|
}
|
|
};
|
|
|
|
// Support: IE <=9
|
|
// Panic based approach to setting things on disconnected nodes
|
|
|
|
Tween.propHooks.scrollTop = Tween.propHooks.scrollLeft = {
|
|
set: function( tween ) {
|
|
if ( tween.elem.nodeType && tween.elem.parentNode ) {
|
|
tween.elem[ tween.prop ] = tween.now;
|
|
}
|
|
}
|
|
};
|
|
|
|
jQuery.easing = {
|
|
linear: function( p ) {
|
|
return p;
|
|
},
|
|
swing: function( p ) {
|
|
return 0.5 - Math.cos( p * Math.PI ) / 2;
|
|
},
|
|
_default: "swing"
|
|
};
|
|
|
|
jQuery.fx = Tween.prototype.init;
|
|
|
|
// Back Compat <1.8 extension point
|
|
jQuery.fx.step = {};
|
|
|
|
|
|
|
|
|
|
var
|
|
fxNow, timerId,
|
|
rfxtypes = /^(?:toggle|show|hide)$/,
|
|
rrun = /queueHooks$/;
|
|
|
|
// Animations created synchronously will run synchronously
|
|
function createFxNow() {
|
|
window.setTimeout( function() {
|
|
fxNow = undefined;
|
|
} );
|
|
return ( fxNow = jQuery.now() );
|
|
}
|
|
|
|
// Generate parameters to create a standard animation
|
|
function genFx( type, includeWidth ) {
|
|
var which,
|
|
attrs = { height: type },
|
|
i = 0;
|
|
|
|
// if we include width, step value is 1 to do all cssExpand values,
|
|
// if we don't include width, step value is 2 to skip over Left and Right
|
|
includeWidth = includeWidth ? 1 : 0;
|
|
for ( ; i < 4 ; i += 2 - includeWidth ) {
|
|
which = cssExpand[ i ];
|
|
attrs[ "margin" + which ] = attrs[ "padding" + which ] = type;
|
|
}
|
|
|
|
if ( includeWidth ) {
|
|
attrs.opacity = attrs.width = type;
|
|
}
|
|
|
|
return attrs;
|
|
}
|
|
|
|
function createTween( value, prop, animation ) {
|
|
var tween,
|
|
collection = ( Animation.tweeners[ prop ] || [] ).concat( Animation.tweeners[ "*" ] ),
|
|
index = 0,
|
|
length = collection.length;
|
|
for ( ; index < length; index++ ) {
|
|
if ( ( tween = collection[ index ].call( animation, prop, value ) ) ) {
|
|
|
|
// we're done with this property
|
|
return tween;
|
|
}
|
|
}
|
|
}
|
|
|
|
function defaultPrefilter( elem, props, opts ) {
|
|
/* jshint validthis: true */
|
|
var prop, value, toggle, tween, hooks, oldfire, display, checkDisplay,
|
|
anim = this,
|
|
orig = {},
|
|
style = elem.style,
|
|
hidden = elem.nodeType && isHidden( elem ),
|
|
dataShow = jQuery._data( elem, "fxshow" );
|
|
|
|
// handle queue: false promises
|
|
if ( !opts.queue ) {
|
|
hooks = jQuery._queueHooks( elem, "fx" );
|
|
if ( hooks.unqueued == null ) {
|
|
hooks.unqueued = 0;
|
|
oldfire = hooks.empty.fire;
|
|
hooks.empty.fire = function() {
|
|
if ( !hooks.unqueued ) {
|
|
oldfire();
|
|
}
|
|
};
|
|
}
|
|
hooks.unqueued++;
|
|
|
|
anim.always( function() {
|
|
|
|
// doing this makes sure that the complete handler will be called
|
|
// before this completes
|
|
anim.always( function() {
|
|
hooks.unqueued--;
|
|
if ( !jQuery.queue( elem, "fx" ).length ) {
|
|
hooks.empty.fire();
|
|
}
|
|
} );
|
|
} );
|
|
}
|
|
|
|
// height/width overflow pass
|
|
if ( elem.nodeType === 1 && ( "height" in props || "width" in props ) ) {
|
|
|
|
// Make sure that nothing sneaks out
|
|
// Record all 3 overflow attributes because IE does not
|
|
// change the overflow attribute when overflowX and
|
|
// overflowY are set to the same value
|
|
opts.overflow = [ style.overflow, style.overflowX, style.overflowY ];
|
|
|
|
// Set display property to inline-block for height/width
|
|
// animations on inline elements that are having width/height animated
|
|
display = jQuery.css( elem, "display" );
|
|
|
|
// Test default display if display is currently "none"
|
|
checkDisplay = display === "none" ?
|
|
jQuery._data( elem, "olddisplay" ) || defaultDisplay( elem.nodeName ) : display;
|
|
|
|
if ( checkDisplay === "inline" && jQuery.css( elem, "float" ) === "none" ) {
|
|
|
|
// inline-level elements accept inline-block;
|
|
// block-level elements need to be inline with layout
|
|
if ( !support.inlineBlockNeedsLayout || defaultDisplay( elem.nodeName ) === "inline" ) {
|
|
style.display = "inline-block";
|
|
} else {
|
|
style.zoom = 1;
|
|
}
|
|
}
|
|
}
|
|
|
|
if ( opts.overflow ) {
|
|
style.overflow = "hidden";
|
|
if ( !support.shrinkWrapBlocks() ) {
|
|
anim.always( function() {
|
|
style.overflow = opts.overflow[ 0 ];
|
|
style.overflowX = opts.overflow[ 1 ];
|
|
style.overflowY = opts.overflow[ 2 ];
|
|
} );
|
|
}
|
|
}
|
|
|
|
// show/hide pass
|
|
for ( prop in props ) {
|
|
value = props[ prop ];
|
|
if ( rfxtypes.exec( value ) ) {
|
|
delete props[ prop ];
|
|
toggle = toggle || value === "toggle";
|
|
if ( value === ( hidden ? "hide" : "show" ) ) {
|
|
|
|
// If there is dataShow left over from a stopped hide or show
|
|
// and we are going to proceed with show, we should pretend to be hidden
|
|
if ( value === "show" && dataShow && dataShow[ prop ] !== undefined ) {
|
|
hidden = true;
|
|
} else {
|
|
continue;
|
|
}
|
|
}
|
|
orig[ prop ] = dataShow && dataShow[ prop ] || jQuery.style( elem, prop );
|
|
|
|
// Any non-fx value stops us from restoring the original display value
|
|
} else {
|
|
display = undefined;
|
|
}
|
|
}
|
|
|
|
if ( !jQuery.isEmptyObject( orig ) ) {
|
|
if ( dataShow ) {
|
|
if ( "hidden" in dataShow ) {
|
|
hidden = dataShow.hidden;
|
|
}
|
|
} else {
|
|
dataShow = jQuery._data( elem, "fxshow", {} );
|
|
}
|
|
|
|
// store state if its toggle - enables .stop().toggle() to "reverse"
|
|
if ( toggle ) {
|
|
dataShow.hidden = !hidden;
|
|
}
|
|
if ( hidden ) {
|
|
jQuery( elem ).show();
|
|
} else {
|
|
anim.done( function() {
|
|
jQuery( elem ).hide();
|
|
} );
|
|
}
|
|
anim.done( function() {
|
|
var prop;
|
|
jQuery._removeData( elem, "fxshow" );
|
|
for ( prop in orig ) {
|
|
jQuery.style( elem, prop, orig[ prop ] );
|
|
}
|
|
} );
|
|
for ( prop in orig ) {
|
|
tween = createTween( hidden ? dataShow[ prop ] : 0, prop, anim );
|
|
|
|
if ( !( prop in dataShow ) ) {
|
|
dataShow[ prop ] = tween.start;
|
|
if ( hidden ) {
|
|
tween.end = tween.start;
|
|
tween.start = prop === "width" || prop === "height" ? 1 : 0;
|
|
}
|
|
}
|
|
}
|
|
|
|
// If this is a noop like .hide().hide(), restore an overwritten display value
|
|
} else if ( ( display === "none" ? defaultDisplay( elem.nodeName ) : display ) === "inline" ) {
|
|
style.display = display;
|
|
}
|
|
}
|
|
|
|
function propFilter( props, specialEasing ) {
|
|
var index, name, easing, value, hooks;
|
|
|
|
// camelCase, specialEasing and expand cssHook pass
|
|
for ( index in props ) {
|
|
name = jQuery.camelCase( index );
|
|
easing = specialEasing[ name ];
|
|
value = props[ index ];
|
|
if ( jQuery.isArray( value ) ) {
|
|
easing = value[ 1 ];
|
|
value = props[ index ] = value[ 0 ];
|
|
}
|
|
|
|
if ( index !== name ) {
|
|
props[ name ] = value;
|
|
delete props[ index ];
|
|
}
|
|
|
|
hooks = jQuery.cssHooks[ name ];
|
|
if ( hooks && "expand" in hooks ) {
|
|
value = hooks.expand( value );
|
|
delete props[ name ];
|
|
|
|
// not quite $.extend, this wont overwrite keys already present.
|
|
// also - reusing 'index' from above because we have the correct "name"
|
|
for ( index in value ) {
|
|
if ( !( index in props ) ) {
|
|
props[ index ] = value[ index ];
|
|
specialEasing[ index ] = easing;
|
|
}
|
|
}
|
|
} else {
|
|
specialEasing[ name ] = easing;
|
|
}
|
|
}
|
|
}
|
|
|
|
function Animation( elem, properties, options ) {
|
|
var result,
|
|
stopped,
|
|
index = 0,
|
|
length = Animation.prefilters.length,
|
|
deferred = jQuery.Deferred().always( function() {
|
|
|
|
// don't match elem in the :animated selector
|
|
delete tick.elem;
|
|
} ),
|
|
tick = function() {
|
|
if ( stopped ) {
|
|
return false;
|
|
}
|
|
var currentTime = fxNow || createFxNow(),
|
|
remaining = Math.max( 0, animation.startTime + animation.duration - currentTime ),
|
|
|
|
// Support: Android 2.3
|
|
// Archaic crash bug won't allow us to use `1 - ( 0.5 || 0 )` (#12497)
|
|
temp = remaining / animation.duration || 0,
|
|
percent = 1 - temp,
|
|
index = 0,
|
|
length = animation.tweens.length;
|
|
|
|
for ( ; index < length ; index++ ) {
|
|
animation.tweens[ index ].run( percent );
|
|
}
|
|
|
|
deferred.notifyWith( elem, [ animation, percent, remaining ] );
|
|
|
|
if ( percent < 1 && length ) {
|
|
return remaining;
|
|
} else {
|
|
deferred.resolveWith( elem, [ animation ] );
|
|
return false;
|
|
}
|
|
},
|
|
animation = deferred.promise( {
|
|
elem: elem,
|
|
props: jQuery.extend( {}, properties ),
|
|
opts: jQuery.extend( true, {
|
|
specialEasing: {},
|
|
easing: jQuery.easing._default
|
|
}, options ),
|
|
originalProperties: properties,
|
|
originalOptions: options,
|
|
startTime: fxNow || createFxNow(),
|
|
duration: options.duration,
|
|
tweens: [],
|
|
createTween: function( prop, end ) {
|
|
var tween = jQuery.Tween( elem, animation.opts, prop, end,
|
|
animation.opts.specialEasing[ prop ] || animation.opts.easing );
|
|
animation.tweens.push( tween );
|
|
return tween;
|
|
},
|
|
stop: function( gotoEnd ) {
|
|
var index = 0,
|
|
|
|
// if we are going to the end, we want to run all the tweens
|
|
// otherwise we skip this part
|
|
length = gotoEnd ? animation.tweens.length : 0;
|
|
if ( stopped ) {
|
|
return this;
|
|
}
|
|
stopped = true;
|
|
for ( ; index < length ; index++ ) {
|
|
animation.tweens[ index ].run( 1 );
|
|
}
|
|
|
|
// resolve when we played the last frame
|
|
// otherwise, reject
|
|
if ( gotoEnd ) {
|
|
deferred.notifyWith( elem, [ animation, 1, 0 ] );
|
|
deferred.resolveWith( elem, [ animation, gotoEnd ] );
|
|
} else {
|
|
deferred.rejectWith( elem, [ animation, gotoEnd ] );
|
|
}
|
|
return this;
|
|
}
|
|
} ),
|
|
props = animation.props;
|
|
|
|
propFilter( props, animation.opts.specialEasing );
|
|
|
|
for ( ; index < length ; index++ ) {
|
|
result = Animation.prefilters[ index ].call( animation, elem, props, animation.opts );
|
|
if ( result ) {
|
|
if ( jQuery.isFunction( result.stop ) ) {
|
|
jQuery._queueHooks( animation.elem, animation.opts.queue ).stop =
|
|
jQuery.proxy( result.stop, result );
|
|
}
|
|
return result;
|
|
}
|
|
}
|
|
|
|
jQuery.map( props, createTween, animation );
|
|
|
|
if ( jQuery.isFunction( animation.opts.start ) ) {
|
|
animation.opts.start.call( elem, animation );
|
|
}
|
|
|
|
jQuery.fx.timer(
|
|
jQuery.extend( tick, {
|
|
elem: elem,
|
|
anim: animation,
|
|
queue: animation.opts.queue
|
|
} )
|
|
);
|
|
|
|
// attach callbacks from options
|
|
return animation.progress( animation.opts.progress )
|
|
.done( animation.opts.done, animation.opts.complete )
|
|
.fail( animation.opts.fail )
|
|
.always( animation.opts.always );
|
|
}
|
|
|
|
jQuery.Animation = jQuery.extend( Animation, {
|
|
|
|
tweeners: {
|
|
"*": [ function( prop, value ) {
|
|
var tween = this.createTween( prop, value );
|
|
adjustCSS( tween.elem, prop, rcssNum.exec( value ), tween );
|
|
return tween;
|
|
} ]
|
|
},
|
|
|
|
tweener: function( props, callback ) {
|
|
if ( jQuery.isFunction( props ) ) {
|
|
callback = props;
|
|
props = [ "*" ];
|
|
} else {
|
|
props = props.match( rnotwhite );
|
|
}
|
|
|
|
var prop,
|
|
index = 0,
|
|
length = props.length;
|
|
|
|
for ( ; index < length ; index++ ) {
|
|
prop = props[ index ];
|
|
Animation.tweeners[ prop ] = Animation.tweeners[ prop ] || [];
|
|
Animation.tweeners[ prop ].unshift( callback );
|
|
}
|
|
},
|
|
|
|
prefilters: [ defaultPrefilter ],
|
|
|
|
prefilter: function( callback, prepend ) {
|
|
if ( prepend ) {
|
|
Animation.prefilters.unshift( callback );
|
|
} else {
|
|
Animation.prefilters.push( callback );
|
|
}
|
|
}
|
|
} );
|
|
|
|
jQuery.speed = function( speed, easing, fn ) {
|
|
var opt = speed && typeof speed === "object" ? jQuery.extend( {}, speed ) : {
|
|
complete: fn || !fn && easing ||
|
|
jQuery.isFunction( speed ) && speed,
|
|
duration: speed,
|
|
easing: fn && easing || easing && !jQuery.isFunction( easing ) && easing
|
|
};
|
|
|
|
opt.duration = jQuery.fx.off ? 0 : typeof opt.duration === "number" ? opt.duration :
|
|
opt.duration in jQuery.fx.speeds ?
|
|
jQuery.fx.speeds[ opt.duration ] : jQuery.fx.speeds._default;
|
|
|
|
// normalize opt.queue - true/undefined/null -> "fx"
|
|
if ( opt.queue == null || opt.queue === true ) {
|
|
opt.queue = "fx";
|
|
}
|
|
|
|
// Queueing
|
|
opt.old = opt.complete;
|
|
|
|
opt.complete = function() {
|
|
if ( jQuery.isFunction( opt.old ) ) {
|
|
opt.old.call( this );
|
|
}
|
|
|
|
if ( opt.queue ) {
|
|
jQuery.dequeue( this, opt.queue );
|
|
}
|
|
};
|
|
|
|
return opt;
|
|
};
|
|
|
|
jQuery.fn.extend( {
|
|
fadeTo: function( speed, to, easing, callback ) {
|
|
|
|
// show any hidden elements after setting opacity to 0
|
|
return this.filter( isHidden ).css( "opacity", 0 ).show()
|
|
|
|
// animate to the value specified
|
|
.end().animate( { opacity: to }, speed, easing, callback );
|
|
},
|
|
animate: function( prop, speed, easing, callback ) {
|
|
var empty = jQuery.isEmptyObject( prop ),
|
|
optall = jQuery.speed( speed, easing, callback ),
|
|
doAnimation = function() {
|
|
|
|
// Operate on a copy of prop so per-property easing won't be lost
|
|
var anim = Animation( this, jQuery.extend( {}, prop ), optall );
|
|
|
|
// Empty animations, or finishing resolves immediately
|
|
if ( empty || jQuery._data( this, "finish" ) ) {
|
|
anim.stop( true );
|
|
}
|
|
};
|
|
doAnimation.finish = doAnimation;
|
|
|
|
return empty || optall.queue === false ?
|
|
this.each( doAnimation ) :
|
|
this.queue( optall.queue, doAnimation );
|
|
},
|
|
stop: function( type, clearQueue, gotoEnd ) {
|
|
var stopQueue = function( hooks ) {
|
|
var stop = hooks.stop;
|
|
delete hooks.stop;
|
|
stop( gotoEnd );
|
|
};
|
|
|
|
if ( typeof type !== "string" ) {
|
|
gotoEnd = clearQueue;
|
|
clearQueue = type;
|
|
type = undefined;
|
|
}
|
|
if ( clearQueue && type !== false ) {
|
|
this.queue( type || "fx", [] );
|
|
}
|
|
|
|
return this.each( function() {
|
|
var dequeue = true,
|
|
index = type != null && type + "queueHooks",
|
|
timers = jQuery.timers,
|
|
data = jQuery._data( this );
|
|
|
|
if ( index ) {
|
|
if ( data[ index ] && data[ index ].stop ) {
|
|
stopQueue( data[ index ] );
|
|
}
|
|
} else {
|
|
for ( index in data ) {
|
|
if ( data[ index ] && data[ index ].stop && rrun.test( index ) ) {
|
|
stopQueue( data[ index ] );
|
|
}
|
|
}
|
|
}
|
|
|
|
for ( index = timers.length; index--; ) {
|
|
if ( timers[ index ].elem === this &&
|
|
( type == null || timers[ index ].queue === type ) ) {
|
|
|
|
timers[ index ].anim.stop( gotoEnd );
|
|
dequeue = false;
|
|
timers.splice( index, 1 );
|
|
}
|
|
}
|
|
|
|
// start the next in the queue if the last step wasn't forced
|
|
// timers currently will call their complete callbacks, which will dequeue
|
|
// but only if they were gotoEnd
|
|
if ( dequeue || !gotoEnd ) {
|
|
jQuery.dequeue( this, type );
|
|
}
|
|
} );
|
|
},
|
|
finish: function( type ) {
|
|
if ( type !== false ) {
|
|
type = type || "fx";
|
|
}
|
|
return this.each( function() {
|
|
var index,
|
|
data = jQuery._data( this ),
|
|
queue = data[ type + "queue" ],
|
|
hooks = data[ type + "queueHooks" ],
|
|
timers = jQuery.timers,
|
|
length = queue ? queue.length : 0;
|
|
|
|
// enable finishing flag on private data
|
|
data.finish = true;
|
|
|
|
// empty the queue first
|
|
jQuery.queue( this, type, [] );
|
|
|
|
if ( hooks && hooks.stop ) {
|
|
hooks.stop.call( this, true );
|
|
}
|
|
|
|
// look for any active animations, and finish them
|
|
for ( index = timers.length; index--; ) {
|
|
if ( timers[ index ].elem === this && timers[ index ].queue === type ) {
|
|
timers[ index ].anim.stop( true );
|
|
timers.splice( index, 1 );
|
|
}
|
|
}
|
|
|
|
// look for any animations in the old queue and finish them
|
|
for ( index = 0; index < length; index++ ) {
|
|
if ( queue[ index ] && queue[ index ].finish ) {
|
|
queue[ index ].finish.call( this );
|
|
}
|
|
}
|
|
|
|
// turn off finishing flag
|
|
delete data.finish;
|
|
} );
|
|
}
|
|
} );
|
|
|
|
jQuery.each( [ "toggle", "show", "hide" ], function( i, name ) {
|
|
var cssFn = jQuery.fn[ name ];
|
|
jQuery.fn[ name ] = function( speed, easing, callback ) {
|
|
return speed == null || typeof speed === "boolean" ?
|
|
cssFn.apply( this, arguments ) :
|
|
this.animate( genFx( name, true ), speed, easing, callback );
|
|
};
|
|
} );
|
|
|
|
// Generate shortcuts for custom animations
|
|
jQuery.each( {
|
|
slideDown: genFx( "show" ),
|
|
slideUp: genFx( "hide" ),
|
|
slideToggle: genFx( "toggle" ),
|
|
fadeIn: { opacity: "show" },
|
|
fadeOut: { opacity: "hide" },
|
|
fadeToggle: { opacity: "toggle" }
|
|
}, function( name, props ) {
|
|
jQuery.fn[ name ] = function( speed, easing, callback ) {
|
|
return this.animate( props, speed, easing, callback );
|
|
};
|
|
} );
|
|
|
|
jQuery.timers = [];
|
|
jQuery.fx.tick = function() {
|
|
var timer,
|
|
timers = jQuery.timers,
|
|
i = 0;
|
|
|
|
fxNow = jQuery.now();
|
|
|
|
for ( ; i < timers.length; i++ ) {
|
|
timer = timers[ i ];
|
|
|
|
// Checks the timer has not already been removed
|
|
if ( !timer() && timers[ i ] === timer ) {
|
|
timers.splice( i--, 1 );
|
|
}
|
|
}
|
|
|
|
if ( !timers.length ) {
|
|
jQuery.fx.stop();
|
|
}
|
|
fxNow = undefined;
|
|
};
|
|
|
|
jQuery.fx.timer = function( timer ) {
|
|
jQuery.timers.push( timer );
|
|
if ( timer() ) {
|
|
jQuery.fx.start();
|
|
} else {
|
|
jQuery.timers.pop();
|
|
}
|
|
};
|
|
|
|
jQuery.fx.interval = 13;
|
|
|
|
jQuery.fx.start = function() {
|
|
if ( !timerId ) {
|
|
timerId = window.setInterval( jQuery.fx.tick, jQuery.fx.interval );
|
|
}
|
|
};
|
|
|
|
jQuery.fx.stop = function() {
|
|
window.clearInterval( timerId );
|
|
timerId = null;
|
|
};
|
|
|
|
jQuery.fx.speeds = {
|
|
slow: 600,
|
|
fast: 200,
|
|
|
|
// Default speed
|
|
_default: 400
|
|
};
|
|
|
|
|
|
// Based off of the plugin by Clint Helfers, with permission.
|
|
// http://web.archive.org/web/20100324014747/http://blindsignals.com/index.php/2009/07/jquery-delay/
|
|
jQuery.fn.delay = function( time, type ) {
|
|
time = jQuery.fx ? jQuery.fx.speeds[ time ] || time : time;
|
|
type = type || "fx";
|
|
|
|
return this.queue( type, function( next, hooks ) {
|
|
var timeout = window.setTimeout( next, time );
|
|
hooks.stop = function() {
|
|
window.clearTimeout( timeout );
|
|
};
|
|
} );
|
|
};
|
|
|
|
|
|
( function() {
|
|
var a,
|
|
input = document.createElement( "input" ),
|
|
div = document.createElement( "div" ),
|
|
select = document.createElement( "select" ),
|
|
opt = select.appendChild( document.createElement( "option" ) );
|
|
|
|
// Setup
|
|
div = document.createElement( "div" );
|
|
div.setAttribute( "className", "t" );
|
|
div.innerHTML = " <link/><table></table><a href='/a'>a</a><input type='checkbox'/>";
|
|
a = div.getElementsByTagName( "a" )[ 0 ];
|
|
|
|
// Support: Windows Web Apps (WWA)
|
|
// `type` must use .setAttribute for WWA (#14901)
|
|
input.setAttribute( "type", "checkbox" );
|
|
div.appendChild( input );
|
|
|
|
a = div.getElementsByTagName( "a" )[ 0 ];
|
|
|
|
// First batch of tests.
|
|
a.style.cssText = "top:1px";
|
|
|
|
// Test setAttribute on camelCase class.
|
|
// If it works, we need attrFixes when doing get/setAttribute (ie6/7)
|
|
support.getSetAttribute = div.className !== "t";
|
|
|
|
// Get the style information from getAttribute
|
|
// (IE uses .cssText instead)
|
|
support.style = /top/.test( a.getAttribute( "style" ) );
|
|
|
|
// Make sure that URLs aren't manipulated
|
|
// (IE normalizes it by default)
|
|
support.hrefNormalized = a.getAttribute( "href" ) === "/a";
|
|
|
|
// Check the default checkbox/radio value ("" on WebKit; "on" elsewhere)
|
|
support.checkOn = !!input.value;
|
|
|
|
// Make sure that a selected-by-default option has a working selected property.
|
|
// (WebKit defaults to false instead of true, IE too, if it's in an optgroup)
|
|
support.optSelected = opt.selected;
|
|
|
|
// Tests for enctype support on a form (#6743)
|
|
support.enctype = !!document.createElement( "form" ).enctype;
|
|
|
|
// Make sure that the options inside disabled selects aren't marked as disabled
|
|
// (WebKit marks them as disabled)
|
|
select.disabled = true;
|
|
support.optDisabled = !opt.disabled;
|
|
|
|
// Support: IE8 only
|
|
// Check if we can trust getAttribute("value")
|
|
input = document.createElement( "input" );
|
|
input.setAttribute( "value", "" );
|
|
support.input = input.getAttribute( "value" ) === "";
|
|
|
|
// Check if an input maintains its value after becoming a radio
|
|
input.value = "t";
|
|
input.setAttribute( "type", "radio" );
|
|
support.radioValue = input.value === "t";
|
|
} )();
|
|
|
|
|
|
var rreturn = /\r/g,
|
|
rspaces = /[\x20\t\r\n\f]+/g;
|
|
|
|
jQuery.fn.extend( {
|
|
val: function( value ) {
|
|
var hooks, ret, isFunction,
|
|
elem = this[ 0 ];
|
|
|
|
if ( !arguments.length ) {
|
|
if ( elem ) {
|
|
hooks = jQuery.valHooks[ elem.type ] ||
|
|
jQuery.valHooks[ elem.nodeName.toLowerCase() ];
|
|
|
|
if (
|
|
hooks &&
|
|
"get" in hooks &&
|
|
( ret = hooks.get( elem, "value" ) ) !== undefined
|
|
) {
|
|
return ret;
|
|
}
|
|
|
|
ret = elem.value;
|
|
|
|
return typeof ret === "string" ?
|
|
|
|
// handle most common string cases
|
|
ret.replace( rreturn, "" ) :
|
|
|
|
// handle cases where value is null/undef or number
|
|
ret == null ? "" : ret;
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
isFunction = jQuery.isFunction( value );
|
|
|
|
return this.each( function( i ) {
|
|
var val;
|
|
|
|
if ( this.nodeType !== 1 ) {
|
|
return;
|
|
}
|
|
|
|
if ( isFunction ) {
|
|
val = value.call( this, i, jQuery( this ).val() );
|
|
} else {
|
|
val = value;
|
|
}
|
|
|
|
// Treat null/undefined as ""; convert numbers to string
|
|
if ( val == null ) {
|
|
val = "";
|
|
} else if ( typeof val === "number" ) {
|
|
val += "";
|
|
} else if ( jQuery.isArray( val ) ) {
|
|
val = jQuery.map( val, function( value ) {
|
|
return value == null ? "" : value + "";
|
|
} );
|
|
}
|
|
|
|
hooks = jQuery.valHooks[ this.type ] || jQuery.valHooks[ this.nodeName.toLowerCase() ];
|
|
|
|
// If set returns undefined, fall back to normal setting
|
|
if ( !hooks || !( "set" in hooks ) || hooks.set( this, val, "value" ) === undefined ) {
|
|
this.value = val;
|
|
}
|
|
} );
|
|
}
|
|
} );
|
|
|
|
jQuery.extend( {
|
|
valHooks: {
|
|
option: {
|
|
get: function( elem ) {
|
|
var val = jQuery.find.attr( elem, "value" );
|
|
return val != null ?
|
|
val :
|
|
|
|
// Support: IE10-11+
|
|
// option.text throws exceptions (#14686, #14858)
|
|
// Strip and collapse whitespace
|
|
// https://html.spec.whatwg.org/#strip-and-collapse-whitespace
|
|
jQuery.trim( jQuery.text( elem ) ).replace( rspaces, " " );
|
|
}
|
|
},
|
|
select: {
|
|
get: function( elem ) {
|
|
var value, option,
|
|
options = elem.options,
|
|
index = elem.selectedIndex,
|
|
one = elem.type === "select-one" || index < 0,
|
|
values = one ? null : [],
|
|
max = one ? index + 1 : options.length,
|
|
i = index < 0 ?
|
|
max :
|
|
one ? index : 0;
|
|
|
|
// Loop through all the selected options
|
|
for ( ; i < max; i++ ) {
|
|
option = options[ i ];
|
|
|
|
// oldIE doesn't update selected after form reset (#2551)
|
|
if ( ( option.selected || i === index ) &&
|
|
|
|
// Don't return options that are disabled or in a disabled optgroup
|
|
( support.optDisabled ?
|
|
!option.disabled :
|
|
option.getAttribute( "disabled" ) === null ) &&
|
|
( !option.parentNode.disabled ||
|
|
!jQuery.nodeName( option.parentNode, "optgroup" ) ) ) {
|
|
|
|
// Get the specific value for the option
|
|
value = jQuery( option ).val();
|
|
|
|
// We don't need an array for one selects
|
|
if ( one ) {
|
|
return value;
|
|
}
|
|
|
|
// Multi-Selects return an array
|
|
values.push( value );
|
|
}
|
|
}
|
|
|
|
return values;
|
|
},
|
|
|
|
set: function( elem, value ) {
|
|
var optionSet, option,
|
|
options = elem.options,
|
|
values = jQuery.makeArray( value ),
|
|
i = options.length;
|
|
|
|
while ( i-- ) {
|
|
option = options[ i ];
|
|
|
|
if ( jQuery.inArray( jQuery.valHooks.option.get( option ), values ) > -1 ) {
|
|
|
|
// Support: IE6
|
|
// When new option element is added to select box we need to
|
|
// force reflow of newly added node in order to workaround delay
|
|
// of initialization properties
|
|
try {
|
|
option.selected = optionSet = true;
|
|
|
|
} catch ( _ ) {
|
|
|
|
// Will be executed only in IE6
|
|
option.scrollHeight;
|
|
}
|
|
|
|
} else {
|
|
option.selected = false;
|
|
}
|
|
}
|
|
|
|
// Force browsers to behave consistently when non-matching value is set
|
|
if ( !optionSet ) {
|
|
elem.selectedIndex = -1;
|
|
}
|
|
|
|
return options;
|
|
}
|
|
}
|
|
}
|
|
} );
|
|
|
|
// Radios and checkboxes getter/setter
|
|
jQuery.each( [ "radio", "checkbox" ], function() {
|
|
jQuery.valHooks[ this ] = {
|
|
set: function( elem, value ) {
|
|
if ( jQuery.isArray( value ) ) {
|
|
return ( elem.checked = jQuery.inArray( jQuery( elem ).val(), value ) > -1 );
|
|
}
|
|
}
|
|
};
|
|
if ( !support.checkOn ) {
|
|
jQuery.valHooks[ this ].get = function( elem ) {
|
|
return elem.getAttribute( "value" ) === null ? "on" : elem.value;
|
|
};
|
|
}
|
|
} );
|
|
|
|
|
|
|
|
|
|
var nodeHook, boolHook,
|
|
attrHandle = jQuery.expr.attrHandle,
|
|
ruseDefault = /^(?:checked|selected)$/i,
|
|
getSetAttribute = support.getSetAttribute,
|
|
getSetInput = support.input;
|
|
|
|
jQuery.fn.extend( {
|
|
attr: function( name, value ) {
|
|
return access( this, jQuery.attr, name, value, arguments.length > 1 );
|
|
},
|
|
|
|
removeAttr: function( name ) {
|
|
return this.each( function() {
|
|
jQuery.removeAttr( this, name );
|
|
} );
|
|
}
|
|
} );
|
|
|
|
jQuery.extend( {
|
|
attr: function( elem, name, value ) {
|
|
var ret, hooks,
|
|
nType = elem.nodeType;
|
|
|
|
// Don't get/set attributes on text, comment and attribute nodes
|
|
if ( nType === 3 || nType === 8 || nType === 2 ) {
|
|
return;
|
|
}
|
|
|
|
// Fallback to prop when attributes are not supported
|
|
if ( typeof elem.getAttribute === "undefined" ) {
|
|
return jQuery.prop( elem, name, value );
|
|
}
|
|
|
|
// All attributes are lowercase
|
|
// Grab necessary hook if one is defined
|
|
if ( nType !== 1 || !jQuery.isXMLDoc( elem ) ) {
|
|
name = name.toLowerCase();
|
|
hooks = jQuery.attrHooks[ name ] ||
|
|
( jQuery.expr.match.bool.test( name ) ? boolHook : nodeHook );
|
|
}
|
|
|
|
if ( value !== undefined ) {
|
|
if ( value === null ) {
|
|
jQuery.removeAttr( elem, name );
|
|
return;
|
|
}
|
|
|
|
if ( hooks && "set" in hooks &&
|
|
( ret = hooks.set( elem, value, name ) ) !== undefined ) {
|
|
return ret;
|
|
}
|
|
|
|
elem.setAttribute( name, value + "" );
|
|
return value;
|
|
}
|
|
|
|
if ( hooks && "get" in hooks && ( ret = hooks.get( elem, name ) ) !== null ) {
|
|
return ret;
|
|
}
|
|
|
|
ret = jQuery.find.attr( elem, name );
|
|
|
|
// Non-existent attributes return null, we normalize to undefined
|
|
return ret == null ? undefined : ret;
|
|
},
|
|
|
|
attrHooks: {
|
|
type: {
|
|
set: function( elem, value ) {
|
|
if ( !support.radioValue && value === "radio" &&
|
|
jQuery.nodeName( elem, "input" ) ) {
|
|
|
|
// Setting the type on a radio button after the value resets the value in IE8-9
|
|
// Reset value to default in case type is set after value during creation
|
|
var val = elem.value;
|
|
elem.setAttribute( "type", value );
|
|
if ( val ) {
|
|
elem.value = val;
|
|
}
|
|
return value;
|
|
}
|
|
}
|
|
}
|
|
},
|
|
|
|
removeAttr: function( elem, value ) {
|
|
var name, propName,
|
|
i = 0,
|
|
attrNames = value && value.match( rnotwhite );
|
|
|
|
if ( attrNames && elem.nodeType === 1 ) {
|
|
while ( ( name = attrNames[ i++ ] ) ) {
|
|
propName = jQuery.propFix[ name ] || name;
|
|
|
|
// Boolean attributes get special treatment (#10870)
|
|
if ( jQuery.expr.match.bool.test( name ) ) {
|
|
|
|
// Set corresponding property to false
|
|
if ( getSetInput && getSetAttribute || !ruseDefault.test( name ) ) {
|
|
elem[ propName ] = false;
|
|
|
|
// Support: IE<9
|
|
// Also clear defaultChecked/defaultSelected (if appropriate)
|
|
} else {
|
|
elem[ jQuery.camelCase( "default-" + name ) ] =
|
|
elem[ propName ] = false;
|
|
}
|
|
|
|
// See #9699 for explanation of this approach (setting first, then removal)
|
|
} else {
|
|
jQuery.attr( elem, name, "" );
|
|
}
|
|
|
|
elem.removeAttribute( getSetAttribute ? name : propName );
|
|
}
|
|
}
|
|
}
|
|
} );
|
|
|
|
// Hooks for boolean attributes
|
|
boolHook = {
|
|
set: function( elem, value, name ) {
|
|
if ( value === false ) {
|
|
|
|
// Remove boolean attributes when set to false
|
|
jQuery.removeAttr( elem, name );
|
|
} else if ( getSetInput && getSetAttribute || !ruseDefault.test( name ) ) {
|
|
|
|
// IE<8 needs the *property* name
|
|
elem.setAttribute( !getSetAttribute && jQuery.propFix[ name ] || name, name );
|
|
|
|
} else {
|
|
|
|
// Support: IE<9
|
|
// Use defaultChecked and defaultSelected for oldIE
|
|
elem[ jQuery.camelCase( "default-" + name ) ] = elem[ name ] = true;
|
|
}
|
|
return name;
|
|
}
|
|
};
|
|
|
|
jQuery.each( jQuery.expr.match.bool.source.match( /\w+/g ), function( i, name ) {
|
|
var getter = attrHandle[ name ] || jQuery.find.attr;
|
|
|
|
if ( getSetInput && getSetAttribute || !ruseDefault.test( name ) ) {
|
|
attrHandle[ name ] = function( elem, name, isXML ) {
|
|
var ret, handle;
|
|
if ( !isXML ) {
|
|
|
|
// Avoid an infinite loop by temporarily removing this function from the getter
|
|
handle = attrHandle[ name ];
|
|
attrHandle[ name ] = ret;
|
|
ret = getter( elem, name, isXML ) != null ?
|
|
name.toLowerCase() :
|
|
null;
|
|
attrHandle[ name ] = handle;
|
|
}
|
|
return ret;
|
|
};
|
|
} else {
|
|
attrHandle[ name ] = function( elem, name, isXML ) {
|
|
if ( !isXML ) {
|
|
return elem[ jQuery.camelCase( "default-" + name ) ] ?
|
|
name.toLowerCase() :
|
|
null;
|
|
}
|
|
};
|
|
}
|
|
} );
|
|
|
|
// fix oldIE attroperties
|
|
if ( !getSetInput || !getSetAttribute ) {
|
|
jQuery.attrHooks.value = {
|
|
set: function( elem, value, name ) {
|
|
if ( jQuery.nodeName( elem, "input" ) ) {
|
|
|
|
// Does not return so that setAttribute is also used
|
|
elem.defaultValue = value;
|
|
} else {
|
|
|
|
// Use nodeHook if defined (#1954); otherwise setAttribute is fine
|
|
return nodeHook && nodeHook.set( elem, value, name );
|
|
}
|
|
}
|
|
};
|
|
}
|
|
|
|
// IE6/7 do not support getting/setting some attributes with get/setAttribute
|
|
if ( !getSetAttribute ) {
|
|
|
|
// Use this for any attribute in IE6/7
|
|
// This fixes almost every IE6/7 issue
|
|
nodeHook = {
|
|
set: function( elem, value, name ) {
|
|
|
|
// Set the existing or create a new attribute node
|
|
var ret = elem.getAttributeNode( name );
|
|
if ( !ret ) {
|
|
elem.setAttributeNode(
|
|
( ret = elem.ownerDocument.createAttribute( name ) )
|
|
);
|
|
}
|
|
|
|
ret.value = value += "";
|
|
|
|
// Break association with cloned elements by also using setAttribute (#9646)
|
|
if ( name === "value" || value === elem.getAttribute( name ) ) {
|
|
return value;
|
|
}
|
|
}
|
|
};
|
|
|
|
// Some attributes are constructed with empty-string values when not defined
|
|
attrHandle.id = attrHandle.name = attrHandle.coords =
|
|
function( elem, name, isXML ) {
|
|
var ret;
|
|
if ( !isXML ) {
|
|
return ( ret = elem.getAttributeNode( name ) ) && ret.value !== "" ?
|
|
ret.value :
|
|
null;
|
|
}
|
|
};
|
|
|
|
// Fixing value retrieval on a button requires this module
|
|
jQuery.valHooks.button = {
|
|
get: function( elem, name ) {
|
|
var ret = elem.getAttributeNode( name );
|
|
if ( ret && ret.specified ) {
|
|
return ret.value;
|
|
}
|
|
},
|
|
set: nodeHook.set
|
|
};
|
|
|
|
// Set contenteditable to false on removals(#10429)
|
|
// Setting to empty string throws an error as an invalid value
|
|
jQuery.attrHooks.contenteditable = {
|
|
set: function( elem, value, name ) {
|
|
nodeHook.set( elem, value === "" ? false : value, name );
|
|
}
|
|
};
|
|
|
|
// Set width and height to auto instead of 0 on empty string( Bug #8150 )
|
|
// This is for removals
|
|
jQuery.each( [ "width", "height" ], function( i, name ) {
|
|
jQuery.attrHooks[ name ] = {
|
|
set: function( elem, value ) {
|
|
if ( value === "" ) {
|
|
elem.setAttribute( name, "auto" );
|
|
return value;
|
|
}
|
|
}
|
|
};
|
|
} );
|
|
}
|
|
|
|
if ( !support.style ) {
|
|
jQuery.attrHooks.style = {
|
|
get: function( elem ) {
|
|
|
|
// Return undefined in the case of empty string
|
|
// Note: IE uppercases css property names, but if we were to .toLowerCase()
|
|
// .cssText, that would destroy case sensitivity in URL's, like in "background"
|
|
return elem.style.cssText || undefined;
|
|
},
|
|
set: function( elem, value ) {
|
|
return ( elem.style.cssText = value + "" );
|
|
}
|
|
};
|
|
}
|
|
|
|
|
|
|
|
|
|
var rfocusable = /^(?:input|select|textarea|button|object)$/i,
|
|
rclickable = /^(?:a|area)$/i;
|
|
|
|
jQuery.fn.extend( {
|
|
prop: function( name, value ) {
|
|
return access( this, jQuery.prop, name, value, arguments.length > 1 );
|
|
},
|
|
|
|
removeProp: function( name ) {
|
|
name = jQuery.propFix[ name ] || name;
|
|
return this.each( function() {
|
|
|
|
// try/catch handles cases where IE balks (such as removing a property on window)
|
|
try {
|
|
this[ name ] = undefined;
|
|
delete this[ name ];
|
|
} catch ( e ) {}
|
|
} );
|
|
}
|
|
} );
|
|
|
|
jQuery.extend( {
|
|
prop: function( elem, name, value ) {
|
|
var ret, hooks,
|
|
nType = elem.nodeType;
|
|
|
|
// Don't get/set properties on text, comment and attribute nodes
|
|
if ( nType === 3 || nType === 8 || nType === 2 ) {
|
|
return;
|
|
}
|
|
|
|
if ( nType !== 1 || !jQuery.isXMLDoc( elem ) ) {
|
|
|
|
// Fix name and attach hooks
|
|
name = jQuery.propFix[ name ] || name;
|
|
hooks = jQuery.propHooks[ name ];
|
|
}
|
|
|
|
if ( value !== undefined ) {
|
|
if ( hooks && "set" in hooks &&
|
|
( ret = hooks.set( elem, value, name ) ) !== undefined ) {
|
|
return ret;
|
|
}
|
|
|
|
return ( elem[ name ] = value );
|
|
}
|
|
|
|
if ( hooks && "get" in hooks && ( ret = hooks.get( elem, name ) ) !== null ) {
|
|
return ret;
|
|
}
|
|
|
|
return elem[ name ];
|
|
},
|
|
|
|
propHooks: {
|
|
tabIndex: {
|
|
get: function( elem ) {
|
|
|
|
// elem.tabIndex doesn't always return the
|
|
// correct value when it hasn't been explicitly set
|
|
// http://fluidproject.org/blog/2008/01/09/getting-setting-and-removing-tabindex-values-with-javascript/
|
|
// Use proper attribute retrieval(#12072)
|
|
var tabindex = jQuery.find.attr( elem, "tabindex" );
|
|
|
|
return tabindex ?
|
|
parseInt( tabindex, 10 ) :
|
|
rfocusable.test( elem.nodeName ) ||
|
|
rclickable.test( elem.nodeName ) && elem.href ?
|
|
0 :
|
|
-1;
|
|
}
|
|
}
|
|
},
|
|
|
|
propFix: {
|
|
"for": "htmlFor",
|
|
"class": "className"
|
|
}
|
|
} );
|
|
|
|
// Some attributes require a special call on IE
|
|
// http://msdn.microsoft.com/en-us/library/ms536429%28VS.85%29.aspx
|
|
if ( !support.hrefNormalized ) {
|
|
|
|
// href/src property should get the full normalized URL (#10299/#12915)
|
|
jQuery.each( [ "href", "src" ], function( i, name ) {
|
|
jQuery.propHooks[ name ] = {
|
|
get: function( elem ) {
|
|
return elem.getAttribute( name, 4 );
|
|
}
|
|
};
|
|
} );
|
|
}
|
|
|
|
// Support: Safari, IE9+
|
|
// Accessing the selectedIndex property
|
|
// forces the browser to respect setting selected
|
|
// on the option
|
|
// The getter ensures a default option is selected
|
|
// when in an optgroup
|
|
if ( !support.optSelected ) {
|
|
jQuery.propHooks.selected = {
|
|
get: function( elem ) {
|
|
var parent = elem.parentNode;
|
|
|
|
if ( parent ) {
|
|
parent.selectedIndex;
|
|
|
|
// Make sure that it also works with optgroups, see #5701
|
|
if ( parent.parentNode ) {
|
|
parent.parentNode.selectedIndex;
|
|
}
|
|
}
|
|
return null;
|
|
},
|
|
set: function( elem ) {
|
|
var parent = elem.parentNode;
|
|
if ( parent ) {
|
|
parent.selectedIndex;
|
|
|
|
if ( parent.parentNode ) {
|
|
parent.parentNode.selectedIndex;
|
|
}
|
|
}
|
|
}
|
|
};
|
|
}
|
|
|
|
jQuery.each( [
|
|
"tabIndex",
|
|
"readOnly",
|
|
"maxLength",
|
|
"cellSpacing",
|
|
"cellPadding",
|
|
"rowSpan",
|
|
"colSpan",
|
|
"useMap",
|
|
"frameBorder",
|
|
"contentEditable"
|
|
], function() {
|
|
jQuery.propFix[ this.toLowerCase() ] = this;
|
|
} );
|
|
|
|
// IE6/7 call enctype encoding
|
|
if ( !support.enctype ) {
|
|
jQuery.propFix.enctype = "encoding";
|
|
}
|
|
|
|
|
|
|
|
|
|
var rclass = /[\t\r\n\f]/g;
|
|
|
|
function getClass( elem ) {
|
|
return jQuery.attr( elem, "class" ) || "";
|
|
}
|
|
|
|
jQuery.fn.extend( {
|
|
addClass: function( value ) {
|
|
var classes, elem, cur, curValue, clazz, j, finalValue,
|
|
i = 0;
|
|
|
|
if ( jQuery.isFunction( value ) ) {
|
|
return this.each( function( j ) {
|
|
jQuery( this ).addClass( value.call( this, j, getClass( this ) ) );
|
|
} );
|
|
}
|
|
|
|
if ( typeof value === "string" && value ) {
|
|
classes = value.match( rnotwhite ) || [];
|
|
|
|
while ( ( elem = this[ i++ ] ) ) {
|
|
curValue = getClass( elem );
|
|
cur = elem.nodeType === 1 &&
|
|
( " " + curValue + " " ).replace( rclass, " " );
|
|
|
|
if ( cur ) {
|
|
j = 0;
|
|
while ( ( clazz = classes[ j++ ] ) ) {
|
|
if ( cur.indexOf( " " + clazz + " " ) < 0 ) {
|
|
cur += clazz + " ";
|
|
}
|
|
}
|
|
|
|
// only assign if different to avoid unneeded rendering.
|
|
finalValue = jQuery.trim( cur );
|
|
if ( curValue !== finalValue ) {
|
|
jQuery.attr( elem, "class", finalValue );
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return this;
|
|
},
|
|
|
|
removeClass: function( value ) {
|
|
var classes, elem, cur, curValue, clazz, j, finalValue,
|
|
i = 0;
|
|
|
|
if ( jQuery.isFunction( value ) ) {
|
|
return this.each( function( j ) {
|
|
jQuery( this ).removeClass( value.call( this, j, getClass( this ) ) );
|
|
} );
|
|
}
|
|
|
|
if ( !arguments.length ) {
|
|
return this.attr( "class", "" );
|
|
}
|
|
|
|
if ( typeof value === "string" && value ) {
|
|
classes = value.match( rnotwhite ) || [];
|
|
|
|
while ( ( elem = this[ i++ ] ) ) {
|
|
curValue = getClass( elem );
|
|
|
|
// This expression is here for better compressibility (see addClass)
|
|
cur = elem.nodeType === 1 &&
|
|
( " " + curValue + " " ).replace( rclass, " " );
|
|
|
|
if ( cur ) {
|
|
j = 0;
|
|
while ( ( clazz = classes[ j++ ] ) ) {
|
|
|
|
// Remove *all* instances
|
|
while ( cur.indexOf( " " + clazz + " " ) > -1 ) {
|
|
cur = cur.replace( " " + clazz + " ", " " );
|
|
}
|
|
}
|
|
|
|
// Only assign if different to avoid unneeded rendering.
|
|
finalValue = jQuery.trim( cur );
|
|
if ( curValue !== finalValue ) {
|
|
jQuery.attr( elem, "class", finalValue );
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return this;
|
|
},
|
|
|
|
toggleClass: function( value, stateVal ) {
|
|
var type = typeof value;
|
|
|
|
if ( typeof stateVal === "boolean" && type === "string" ) {
|
|
return stateVal ? this.addClass( value ) : this.removeClass( value );
|
|
}
|
|
|
|
if ( jQuery.isFunction( value ) ) {
|
|
return this.each( function( i ) {
|
|
jQuery( this ).toggleClass(
|
|
value.call( this, i, getClass( this ), stateVal ),
|
|
stateVal
|
|
);
|
|
} );
|
|
}
|
|
|
|
return this.each( function() {
|
|
var className, i, self, classNames;
|
|
|
|
if ( type === "string" ) {
|
|
|
|
// Toggle individual class names
|
|
i = 0;
|
|
self = jQuery( this );
|
|
classNames = value.match( rnotwhite ) || [];
|
|
|
|
while ( ( className = classNames[ i++ ] ) ) {
|
|
|
|
// Check each className given, space separated list
|
|
if ( self.hasClass( className ) ) {
|
|
self.removeClass( className );
|
|
} else {
|
|
self.addClass( className );
|
|
}
|
|
}
|
|
|
|
// Toggle whole class name
|
|
} else if ( value === undefined || type === "boolean" ) {
|
|
className = getClass( this );
|
|
if ( className ) {
|
|
|
|
// store className if set
|
|
jQuery._data( this, "__className__", className );
|
|
}
|
|
|
|
// If the element has a class name or if we're passed "false",
|
|
// then remove the whole classname (if there was one, the above saved it).
|
|
// Otherwise bring back whatever was previously saved (if anything),
|
|
// falling back to the empty string if nothing was stored.
|
|
jQuery.attr( this, "class",
|
|
className || value === false ?
|
|
"" :
|
|
jQuery._data( this, "__className__" ) || ""
|
|
);
|
|
}
|
|
} );
|
|
},
|
|
|
|
hasClass: function( selector ) {
|
|
var className, elem,
|
|
i = 0;
|
|
|
|
className = " " + selector + " ";
|
|
while ( ( elem = this[ i++ ] ) ) {
|
|
if ( elem.nodeType === 1 &&
|
|
( " " + getClass( elem ) + " " ).replace( rclass, " " )
|
|
.indexOf( className ) > -1
|
|
) {
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
} );
|
|
|
|
|
|
|
|
|
|
// Return jQuery for attributes-only inclusion
|
|
|
|
|
|
jQuery.each( ( "blur focus focusin focusout load resize scroll unload click dblclick " +
|
|
"mousedown mouseup mousemove mouseover mouseout mouseenter mouseleave " +
|
|
"change select submit keydown keypress keyup error contextmenu" ).split( " " ),
|
|
function( i, name ) {
|
|
|
|
// Handle event binding
|
|
jQuery.fn[ name ] = function( data, fn ) {
|
|
return arguments.length > 0 ?
|
|
this.on( name, null, data, fn ) :
|
|
this.trigger( name );
|
|
};
|
|
} );
|
|
|
|
jQuery.fn.extend( {
|
|
hover: function( fnOver, fnOut ) {
|
|
return this.mouseenter( fnOver ).mouseleave( fnOut || fnOver );
|
|
}
|
|
} );
|
|
|
|
|
|
var location = window.location;
|
|
|
|
var nonce = jQuery.now();
|
|
|
|
var rquery = ( /\?/ );
|
|
|
|
|
|
|
|
var rvalidtokens = /(,)|(\[|{)|(}|])|"(?:[^"\\\r\n]|\\["\\\/bfnrt]|\\u[\da-fA-F]{4})*"\s*:?|true|false|null|-?(?!0\d)\d+(?:\.\d+|)(?:[eE][+-]?\d+|)/g;
|
|
|
|
jQuery.parseJSON = function( data ) {
|
|
|
|
// Attempt to parse using the native JSON parser first
|
|
if ( window.JSON && window.JSON.parse ) {
|
|
|
|
// Support: Android 2.3
|
|
// Workaround failure to string-cast null input
|
|
return window.JSON.parse( data + "" );
|
|
}
|
|
|
|
var requireNonComma,
|
|
depth = null,
|
|
str = jQuery.trim( data + "" );
|
|
|
|
// Guard against invalid (and possibly dangerous) input by ensuring that nothing remains
|
|
// after removing valid tokens
|
|
return str && !jQuery.trim( str.replace( rvalidtokens, function( token, comma, open, close ) {
|
|
|
|
// Force termination if we see a misplaced comma
|
|
if ( requireNonComma && comma ) {
|
|
depth = 0;
|
|
}
|
|
|
|
// Perform no more replacements after returning to outermost depth
|
|
if ( depth === 0 ) {
|
|
return token;
|
|
}
|
|
|
|
// Commas must not follow "[", "{", or ","
|
|
requireNonComma = open || comma;
|
|
|
|
// Determine new depth
|
|
// array/object open ("[" or "{"): depth += true - false (increment)
|
|
// array/object close ("]" or "}"): depth += false - true (decrement)
|
|
// other cases ("," or primitive): depth += true - true (numeric cast)
|
|
depth += !close - !open;
|
|
|
|
// Remove this token
|
|
return "";
|
|
} ) ) ?
|
|
( Function( "return " + str ) )() :
|
|
jQuery.error( "Invalid JSON: " + data );
|
|
};
|
|
|
|
|
|
// Cross-browser xml parsing
|
|
jQuery.parseXML = function( data ) {
|
|
var xml, tmp;
|
|
if ( !data || typeof data !== "string" ) {
|
|
return null;
|
|
}
|
|
try {
|
|
if ( window.DOMParser ) { // Standard
|
|
tmp = new window.DOMParser();
|
|
xml = tmp.parseFromString( data, "text/xml" );
|
|
} else { // IE
|
|
xml = new window.ActiveXObject( "Microsoft.XMLDOM" );
|
|
xml.async = "false";
|
|
xml.loadXML( data );
|
|
}
|
|
} catch ( e ) {
|
|
xml = undefined;
|
|
}
|
|
if ( !xml || !xml.documentElement || xml.getElementsByTagName( "parsererror" ).length ) {
|
|
jQuery.error( "Invalid XML: " + data );
|
|
}
|
|
return xml;
|
|
};
|
|
|
|
|
|
var
|
|
rhash = /#.*$/,
|
|
rts = /([?&])_=[^&]*/,
|
|
|
|
// IE leaves an \r character at EOL
|
|
rheaders = /^(.*?):[ \t]*([^\r\n]*)\r?$/mg,
|
|
|
|
// #7653, #8125, #8152: local protocol detection
|
|
rlocalProtocol = /^(?:about|app|app-storage|.+-extension|file|res|widget):$/,
|
|
rnoContent = /^(?:GET|HEAD)$/,
|
|
rprotocol = /^\/\//,
|
|
rurl = /^([\w.+-]+:)(?:\/\/(?:[^\/?#]*@|)([^\/?#:]*)(?::(\d+)|)|)/,
|
|
|
|
/* Prefilters
|
|
* 1) They are useful to introduce custom dataTypes (see ajax/jsonp.js for an example)
|
|
* 2) These are called:
|
|
* - BEFORE asking for a transport
|
|
* - AFTER param serialization (s.data is a string if s.processData is true)
|
|
* 3) key is the dataType
|
|
* 4) the catchall symbol "*" can be used
|
|
* 5) execution will start with transport dataType and THEN continue down to "*" if needed
|
|
*/
|
|
prefilters = {},
|
|
|
|
/* Transports bindings
|
|
* 1) key is the dataType
|
|
* 2) the catchall symbol "*" can be used
|
|
* 3) selection will start with transport dataType and THEN go to "*" if needed
|
|
*/
|
|
transports = {},
|
|
|
|
// Avoid comment-prolog char sequence (#10098); must appease lint and evade compression
|
|
allTypes = "*/".concat( "*" ),
|
|
|
|
// Document location
|
|
ajaxLocation = location.href,
|
|
|
|
// Segment location into parts
|
|
ajaxLocParts = rurl.exec( ajaxLocation.toLowerCase() ) || [];
|
|
|
|
// Base "constructor" for jQuery.ajaxPrefilter and jQuery.ajaxTransport
|
|
function addToPrefiltersOrTransports( structure ) {
|
|
|
|
// dataTypeExpression is optional and defaults to "*"
|
|
return function( dataTypeExpression, func ) {
|
|
|
|
if ( typeof dataTypeExpression !== "string" ) {
|
|
func = dataTypeExpression;
|
|
dataTypeExpression = "*";
|
|
}
|
|
|
|
var dataType,
|
|
i = 0,
|
|
dataTypes = dataTypeExpression.toLowerCase().match( rnotwhite ) || [];
|
|
|
|
if ( jQuery.isFunction( func ) ) {
|
|
|
|
// For each dataType in the dataTypeExpression
|
|
while ( ( dataType = dataTypes[ i++ ] ) ) {
|
|
|
|
// Prepend if requested
|
|
if ( dataType.charAt( 0 ) === "+" ) {
|
|
dataType = dataType.slice( 1 ) || "*";
|
|
( structure[ dataType ] = structure[ dataType ] || [] ).unshift( func );
|
|
|
|
// Otherwise append
|
|
} else {
|
|
( structure[ dataType ] = structure[ dataType ] || [] ).push( func );
|
|
}
|
|
}
|
|
}
|
|
};
|
|
}
|
|
|
|
// Base inspection function for prefilters and transports
|
|
function inspectPrefiltersOrTransports( structure, options, originalOptions, jqXHR ) {
|
|
|
|
var inspected = {},
|
|
seekingTransport = ( structure === transports );
|
|
|
|
function inspect( dataType ) {
|
|
var selected;
|
|
inspected[ dataType ] = true;
|
|
jQuery.each( structure[ dataType ] || [], function( _, prefilterOrFactory ) {
|
|
var dataTypeOrTransport = prefilterOrFactory( options, originalOptions, jqXHR );
|
|
if ( typeof dataTypeOrTransport === "string" &&
|
|
!seekingTransport && !inspected[ dataTypeOrTransport ] ) {
|
|
|
|
options.dataTypes.unshift( dataTypeOrTransport );
|
|
inspect( dataTypeOrTransport );
|
|
return false;
|
|
} else if ( seekingTransport ) {
|
|
return !( selected = dataTypeOrTransport );
|
|
}
|
|
} );
|
|
return selected;
|
|
}
|
|
|
|
return inspect( options.dataTypes[ 0 ] ) || !inspected[ "*" ] && inspect( "*" );
|
|
}
|
|
|
|
// A special extend for ajax options
|
|
// that takes "flat" options (not to be deep extended)
|
|
// Fixes #9887
|
|
function ajaxExtend( target, src ) {
|
|
var deep, key,
|
|
flatOptions = jQuery.ajaxSettings.flatOptions || {};
|
|
|
|
for ( key in src ) {
|
|
if ( src[ key ] !== undefined ) {
|
|
( flatOptions[ key ] ? target : ( deep || ( deep = {} ) ) )[ key ] = src[ key ];
|
|
}
|
|
}
|
|
if ( deep ) {
|
|
jQuery.extend( true, target, deep );
|
|
}
|
|
|
|
return target;
|
|
}
|
|
|
|
/* Handles responses to an ajax request:
|
|
* - finds the right dataType (mediates between content-type and expected dataType)
|
|
* - returns the corresponding response
|
|
*/
|
|
function ajaxHandleResponses( s, jqXHR, responses ) {
|
|
var firstDataType, ct, finalDataType, type,
|
|
contents = s.contents,
|
|
dataTypes = s.dataTypes;
|
|
|
|
// Remove auto dataType and get content-type in the process
|
|
while ( dataTypes[ 0 ] === "*" ) {
|
|
dataTypes.shift();
|
|
if ( ct === undefined ) {
|
|
ct = s.mimeType || jqXHR.getResponseHeader( "Content-Type" );
|
|
}
|
|
}
|
|
|
|
// Check if we're dealing with a known content-type
|
|
if ( ct ) {
|
|
for ( type in contents ) {
|
|
if ( contents[ type ] && contents[ type ].test( ct ) ) {
|
|
dataTypes.unshift( type );
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
// Check to see if we have a response for the expected dataType
|
|
if ( dataTypes[ 0 ] in responses ) {
|
|
finalDataType = dataTypes[ 0 ];
|
|
} else {
|
|
|
|
// Try convertible dataTypes
|
|
for ( type in responses ) {
|
|
if ( !dataTypes[ 0 ] || s.converters[ type + " " + dataTypes[ 0 ] ] ) {
|
|
finalDataType = type;
|
|
break;
|
|
}
|
|
if ( !firstDataType ) {
|
|
firstDataType = type;
|
|
}
|
|
}
|
|
|
|
// Or just use first one
|
|
finalDataType = finalDataType || firstDataType;
|
|
}
|
|
|
|
// If we found a dataType
|
|
// We add the dataType to the list if needed
|
|
// and return the corresponding response
|
|
if ( finalDataType ) {
|
|
if ( finalDataType !== dataTypes[ 0 ] ) {
|
|
dataTypes.unshift( finalDataType );
|
|
}
|
|
return responses[ finalDataType ];
|
|
}
|
|
}
|
|
|
|
/* Chain conversions given the request and the original response
|
|
* Also sets the responseXXX fields on the jqXHR instance
|
|
*/
|
|
function ajaxConvert( s, response, jqXHR, isSuccess ) {
|
|
var conv2, current, conv, tmp, prev,
|
|
converters = {},
|
|
|
|
// Work with a copy of dataTypes in case we need to modify it for conversion
|
|
dataTypes = s.dataTypes.slice();
|
|
|
|
// Create converters map with lowercased keys
|
|
if ( dataTypes[ 1 ] ) {
|
|
for ( conv in s.converters ) {
|
|
converters[ conv.toLowerCase() ] = s.converters[ conv ];
|
|
}
|
|
}
|
|
|
|
current = dataTypes.shift();
|
|
|
|
// Convert to each sequential dataType
|
|
while ( current ) {
|
|
|
|
if ( s.responseFields[ current ] ) {
|
|
jqXHR[ s.responseFields[ current ] ] = response;
|
|
}
|
|
|
|
// Apply the dataFilter if provided
|
|
if ( !prev && isSuccess && s.dataFilter ) {
|
|
response = s.dataFilter( response, s.dataType );
|
|
}
|
|
|
|
prev = current;
|
|
current = dataTypes.shift();
|
|
|
|
if ( current ) {
|
|
|
|
// There's only work to do if current dataType is non-auto
|
|
if ( current === "*" ) {
|
|
|
|
current = prev;
|
|
|
|
// Convert response if prev dataType is non-auto and differs from current
|
|
} else if ( prev !== "*" && prev !== current ) {
|
|
|
|
// Seek a direct converter
|
|
conv = converters[ prev + " " + current ] || converters[ "* " + current ];
|
|
|
|
// If none found, seek a pair
|
|
if ( !conv ) {
|
|
for ( conv2 in converters ) {
|
|
|
|
// If conv2 outputs current
|
|
tmp = conv2.split( " " );
|
|
if ( tmp[ 1 ] === current ) {
|
|
|
|
// If prev can be converted to accepted input
|
|
conv = converters[ prev + " " + tmp[ 0 ] ] ||
|
|
converters[ "* " + tmp[ 0 ] ];
|
|
if ( conv ) {
|
|
|
|
// Condense equivalence converters
|
|
if ( conv === true ) {
|
|
conv = converters[ conv2 ];
|
|
|
|
// Otherwise, insert the intermediate dataType
|
|
} else if ( converters[ conv2 ] !== true ) {
|
|
current = tmp[ 0 ];
|
|
dataTypes.unshift( tmp[ 1 ] );
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// Apply converter (if not an equivalence)
|
|
if ( conv !== true ) {
|
|
|
|
// Unless errors are allowed to bubble, catch and return them
|
|
if ( conv && s[ "throws" ] ) { // jscs:ignore requireDotNotation
|
|
response = conv( response );
|
|
} else {
|
|
try {
|
|
response = conv( response );
|
|
} catch ( e ) {
|
|
return {
|
|
state: "parsererror",
|
|
error: conv ? e : "No conversion from " + prev + " to " + current
|
|
};
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return { state: "success", data: response };
|
|
}
|
|
|
|
jQuery.extend( {
|
|
|
|
// Counter for holding the number of active queries
|
|
active: 0,
|
|
|
|
// Last-Modified header cache for next request
|
|
lastModified: {},
|
|
etag: {},
|
|
|
|
ajaxSettings: {
|
|
url: ajaxLocation,
|
|
type: "GET",
|
|
isLocal: rlocalProtocol.test( ajaxLocParts[ 1 ] ),
|
|
global: true,
|
|
processData: true,
|
|
async: true,
|
|
contentType: "application/x-www-form-urlencoded; charset=UTF-8",
|
|
/*
|
|
timeout: 0,
|
|
data: null,
|
|
dataType: null,
|
|
username: null,
|
|
password: null,
|
|
cache: null,
|
|
throws: false,
|
|
traditional: false,
|
|
headers: {},
|
|
*/
|
|
|
|
accepts: {
|
|
"*": allTypes,
|
|
text: "text/plain",
|
|
html: "text/html",
|
|
xml: "application/xml, text/xml",
|
|
json: "application/json, text/javascript"
|
|
},
|
|
|
|
contents: {
|
|
xml: /\bxml\b/,
|
|
html: /\bhtml/,
|
|
json: /\bjson\b/
|
|
},
|
|
|
|
responseFields: {
|
|
xml: "responseXML",
|
|
text: "responseText",
|
|
json: "responseJSON"
|
|
},
|
|
|
|
// Data converters
|
|
// Keys separate source (or catchall "*") and destination types with a single space
|
|
converters: {
|
|
|
|
// Convert anything to text
|
|
"* text": String,
|
|
|
|
// Text to html (true = no transformation)
|
|
"text html": true,
|
|
|
|
// Evaluate text as a json expression
|
|
"text json": jQuery.parseJSON,
|
|
|
|
// Parse text as xml
|
|
"text xml": jQuery.parseXML
|
|
},
|
|
|
|
// For options that shouldn't be deep extended:
|
|
// you can add your own custom options here if
|
|
// and when you create one that shouldn't be
|
|
// deep extended (see ajaxExtend)
|
|
flatOptions: {
|
|
url: true,
|
|
context: true
|
|
}
|
|
},
|
|
|
|
// Creates a full fledged settings object into target
|
|
// with both ajaxSettings and settings fields.
|
|
// If target is omitted, writes into ajaxSettings.
|
|
ajaxSetup: function( target, settings ) {
|
|
return settings ?
|
|
|
|
// Building a settings object
|
|
ajaxExtend( ajaxExtend( target, jQuery.ajaxSettings ), settings ) :
|
|
|
|
// Extending ajaxSettings
|
|
ajaxExtend( jQuery.ajaxSettings, target );
|
|
},
|
|
|
|
ajaxPrefilter: addToPrefiltersOrTransports( prefilters ),
|
|
ajaxTransport: addToPrefiltersOrTransports( transports ),
|
|
|
|
// Main method
|
|
ajax: function( url, options ) {
|
|
|
|
// If url is an object, simulate pre-1.5 signature
|
|
if ( typeof url === "object" ) {
|
|
options = url;
|
|
url = undefined;
|
|
}
|
|
|
|
// Force options to be an object
|
|
options = options || {};
|
|
|
|
var
|
|
|
|
// Cross-domain detection vars
|
|
parts,
|
|
|
|
// Loop variable
|
|
i,
|
|
|
|
// URL without anti-cache param
|
|
cacheURL,
|
|
|
|
// Response headers as string
|
|
responseHeadersString,
|
|
|
|
// timeout handle
|
|
timeoutTimer,
|
|
|
|
// To know if global events are to be dispatched
|
|
fireGlobals,
|
|
|
|
transport,
|
|
|
|
// Response headers
|
|
responseHeaders,
|
|
|
|
// Create the final options object
|
|
s = jQuery.ajaxSetup( {}, options ),
|
|
|
|
// Callbacks context
|
|
callbackContext = s.context || s,
|
|
|
|
// Context for global events is callbackContext if it is a DOM node or jQuery collection
|
|
globalEventContext = s.context &&
|
|
( callbackContext.nodeType || callbackContext.jquery ) ?
|
|
jQuery( callbackContext ) :
|
|
jQuery.event,
|
|
|
|
// Deferreds
|
|
deferred = jQuery.Deferred(),
|
|
completeDeferred = jQuery.Callbacks( "once memory" ),
|
|
|
|
// Status-dependent callbacks
|
|
statusCode = s.statusCode || {},
|
|
|
|
// Headers (they are sent all at once)
|
|
requestHeaders = {},
|
|
requestHeadersNames = {},
|
|
|
|
// The jqXHR state
|
|
state = 0,
|
|
|
|
// Default abort message
|
|
strAbort = "canceled",
|
|
|
|
// Fake xhr
|
|
jqXHR = {
|
|
readyState: 0,
|
|
|
|
// Builds headers hashtable if needed
|
|
getResponseHeader: function( key ) {
|
|
var match;
|
|
if ( state === 2 ) {
|
|
if ( !responseHeaders ) {
|
|
responseHeaders = {};
|
|
while ( ( match = rheaders.exec( responseHeadersString ) ) ) {
|
|
responseHeaders[ match[ 1 ].toLowerCase() ] = match[ 2 ];
|
|
}
|
|
}
|
|
match = responseHeaders[ key.toLowerCase() ];
|
|
}
|
|
return match == null ? null : match;
|
|
},
|
|
|
|
// Raw string
|
|
getAllResponseHeaders: function() {
|
|
return state === 2 ? responseHeadersString : null;
|
|
},
|
|
|
|
// Caches the header
|
|
setRequestHeader: function( name, value ) {
|
|
var lname = name.toLowerCase();
|
|
if ( !state ) {
|
|
name = requestHeadersNames[ lname ] = requestHeadersNames[ lname ] || name;
|
|
requestHeaders[ name ] = value;
|
|
}
|
|
return this;
|
|
},
|
|
|
|
// Overrides response content-type header
|
|
overrideMimeType: function( type ) {
|
|
if ( !state ) {
|
|
s.mimeType = type;
|
|
}
|
|
return this;
|
|
},
|
|
|
|
// Status-dependent callbacks
|
|
statusCode: function( map ) {
|
|
var code;
|
|
if ( map ) {
|
|
if ( state < 2 ) {
|
|
for ( code in map ) {
|
|
|
|
// Lazy-add the new callback in a way that preserves old ones
|
|
statusCode[ code ] = [ statusCode[ code ], map[ code ] ];
|
|
}
|
|
} else {
|
|
|
|
// Execute the appropriate callbacks
|
|
jqXHR.always( map[ jqXHR.status ] );
|
|
}
|
|
}
|
|
return this;
|
|
},
|
|
|
|
// Cancel the request
|
|
abort: function( statusText ) {
|
|
var finalText = statusText || strAbort;
|
|
if ( transport ) {
|
|
transport.abort( finalText );
|
|
}
|
|
done( 0, finalText );
|
|
return this;
|
|
}
|
|
};
|
|
|
|
// Attach deferreds
|
|
deferred.promise( jqXHR ).complete = completeDeferred.add;
|
|
jqXHR.success = jqXHR.done;
|
|
jqXHR.error = jqXHR.fail;
|
|
|
|
// Remove hash character (#7531: and string promotion)
|
|
// Add protocol if not provided (#5866: IE7 issue with protocol-less urls)
|
|
// Handle falsy url in the settings object (#10093: consistency with old signature)
|
|
// We also use the url parameter if available
|
|
s.url = ( ( url || s.url || ajaxLocation ) + "" )
|
|
.replace( rhash, "" )
|
|
.replace( rprotocol, ajaxLocParts[ 1 ] + "//" );
|
|
|
|
// Alias method option to type as per ticket #12004
|
|
s.type = options.method || options.type || s.method || s.type;
|
|
|
|
// Extract dataTypes list
|
|
s.dataTypes = jQuery.trim( s.dataType || "*" ).toLowerCase().match( rnotwhite ) || [ "" ];
|
|
|
|
// A cross-domain request is in order when we have a protocol:host:port mismatch
|
|
if ( s.crossDomain == null ) {
|
|
parts = rurl.exec( s.url.toLowerCase() );
|
|
s.crossDomain = !!( parts &&
|
|
( parts[ 1 ] !== ajaxLocParts[ 1 ] || parts[ 2 ] !== ajaxLocParts[ 2 ] ||
|
|
( parts[ 3 ] || ( parts[ 1 ] === "http:" ? "80" : "443" ) ) !==
|
|
( ajaxLocParts[ 3 ] || ( ajaxLocParts[ 1 ] === "http:" ? "80" : "443" ) ) )
|
|
);
|
|
}
|
|
|
|
// Convert data if not already a string
|
|
if ( s.data && s.processData && typeof s.data !== "string" ) {
|
|
s.data = jQuery.param( s.data, s.traditional );
|
|
}
|
|
|
|
// Apply prefilters
|
|
inspectPrefiltersOrTransports( prefilters, s, options, jqXHR );
|
|
|
|
// If request was aborted inside a prefilter, stop there
|
|
if ( state === 2 ) {
|
|
return jqXHR;
|
|
}
|
|
|
|
// We can fire global events as of now if asked to
|
|
// Don't fire events if jQuery.event is undefined in an AMD-usage scenario (#15118)
|
|
fireGlobals = jQuery.event && s.global;
|
|
|
|
// Watch for a new set of requests
|
|
if ( fireGlobals && jQuery.active++ === 0 ) {
|
|
jQuery.event.trigger( "ajaxStart" );
|
|
}
|
|
|
|
// Uppercase the type
|
|
s.type = s.type.toUpperCase();
|
|
|
|
// Determine if request has content
|
|
s.hasContent = !rnoContent.test( s.type );
|
|
|
|
// Save the URL in case we're toying with the If-Modified-Since
|
|
// and/or If-None-Match header later on
|
|
cacheURL = s.url;
|
|
|
|
// More options handling for requests with no content
|
|
if ( !s.hasContent ) {
|
|
|
|
// If data is available, append data to url
|
|
if ( s.data ) {
|
|
cacheURL = ( s.url += ( rquery.test( cacheURL ) ? "&" : "?" ) + s.data );
|
|
|
|
// #9682: remove data so that it's not used in an eventual retry
|
|
delete s.data;
|
|
}
|
|
|
|
// Add anti-cache in url if needed
|
|
if ( s.cache === false ) {
|
|
s.url = rts.test( cacheURL ) ?
|
|
|
|
// If there is already a '_' parameter, set its value
|
|
cacheURL.replace( rts, "$1_=" + nonce++ ) :
|
|
|
|
// Otherwise add one to the end
|
|
cacheURL + ( rquery.test( cacheURL ) ? "&" : "?" ) + "_=" + nonce++;
|
|
}
|
|
}
|
|
|
|
// Set the If-Modified-Since and/or If-None-Match header, if in ifModified mode.
|
|
if ( s.ifModified ) {
|
|
if ( jQuery.lastModified[ cacheURL ] ) {
|
|
jqXHR.setRequestHeader( "If-Modified-Since", jQuery.lastModified[ cacheURL ] );
|
|
}
|
|
if ( jQuery.etag[ cacheURL ] ) {
|
|
jqXHR.setRequestHeader( "If-None-Match", jQuery.etag[ cacheURL ] );
|
|
}
|
|
}
|
|
|
|
// Set the correct header, if data is being sent
|
|
if ( s.data && s.hasContent && s.contentType !== false || options.contentType ) {
|
|
jqXHR.setRequestHeader( "Content-Type", s.contentType );
|
|
}
|
|
|
|
// Set the Accepts header for the server, depending on the dataType
|
|
jqXHR.setRequestHeader(
|
|
"Accept",
|
|
s.dataTypes[ 0 ] && s.accepts[ s.dataTypes[ 0 ] ] ?
|
|
s.accepts[ s.dataTypes[ 0 ] ] +
|
|
( s.dataTypes[ 0 ] !== "*" ? ", " + allTypes + "; q=0.01" : "" ) :
|
|
s.accepts[ "*" ]
|
|
);
|
|
|
|
// Check for headers option
|
|
for ( i in s.headers ) {
|
|
jqXHR.setRequestHeader( i, s.headers[ i ] );
|
|
}
|
|
|
|
// Allow custom headers/mimetypes and early abort
|
|
if ( s.beforeSend &&
|
|
( s.beforeSend.call( callbackContext, jqXHR, s ) === false || state === 2 ) ) {
|
|
|
|
// Abort if not done already and return
|
|
return jqXHR.abort();
|
|
}
|
|
|
|
// aborting is no longer a cancellation
|
|
strAbort = "abort";
|
|
|
|
// Install callbacks on deferreds
|
|
for ( i in { success: 1, error: 1, complete: 1 } ) {
|
|
jqXHR[ i ]( s[ i ] );
|
|
}
|
|
|
|
// Get transport
|
|
transport = inspectPrefiltersOrTransports( transports, s, options, jqXHR );
|
|
|
|
// If no transport, we auto-abort
|
|
if ( !transport ) {
|
|
done( -1, "No Transport" );
|
|
} else {
|
|
jqXHR.readyState = 1;
|
|
|
|
// Send global event
|
|
if ( fireGlobals ) {
|
|
globalEventContext.trigger( "ajaxSend", [ jqXHR, s ] );
|
|
}
|
|
|
|
// If request was aborted inside ajaxSend, stop there
|
|
if ( state === 2 ) {
|
|
return jqXHR;
|
|
}
|
|
|
|
// Timeout
|
|
if ( s.async && s.timeout > 0 ) {
|
|
timeoutTimer = window.setTimeout( function() {
|
|
jqXHR.abort( "timeout" );
|
|
}, s.timeout );
|
|
}
|
|
|
|
try {
|
|
state = 1;
|
|
transport.send( requestHeaders, done );
|
|
} catch ( e ) {
|
|
|
|
// Propagate exception as error if not done
|
|
if ( state < 2 ) {
|
|
done( -1, e );
|
|
|
|
// Simply rethrow otherwise
|
|
} else {
|
|
throw e;
|
|
}
|
|
}
|
|
}
|
|
|
|
// Callback for when everything is done
|
|
function done( status, nativeStatusText, responses, headers ) {
|
|
var isSuccess, success, error, response, modified,
|
|
statusText = nativeStatusText;
|
|
|
|
// Called once
|
|
if ( state === 2 ) {
|
|
return;
|
|
}
|
|
|
|
// State is "done" now
|
|
state = 2;
|
|
|
|
// Clear timeout if it exists
|
|
if ( timeoutTimer ) {
|
|
window.clearTimeout( timeoutTimer );
|
|
}
|
|
|
|
// Dereference transport for early garbage collection
|
|
// (no matter how long the jqXHR object will be used)
|
|
transport = undefined;
|
|
|
|
// Cache response headers
|
|
responseHeadersString = headers || "";
|
|
|
|
// Set readyState
|
|
jqXHR.readyState = status > 0 ? 4 : 0;
|
|
|
|
// Determine if successful
|
|
isSuccess = status >= 200 && status < 300 || status === 304;
|
|
|
|
// Get response data
|
|
if ( responses ) {
|
|
response = ajaxHandleResponses( s, jqXHR, responses );
|
|
}
|
|
|
|
// Convert no matter what (that way responseXXX fields are always set)
|
|
response = ajaxConvert( s, response, jqXHR, isSuccess );
|
|
|
|
// If successful, handle type chaining
|
|
if ( isSuccess ) {
|
|
|
|
// Set the If-Modified-Since and/or If-None-Match header, if in ifModified mode.
|
|
if ( s.ifModified ) {
|
|
modified = jqXHR.getResponseHeader( "Last-Modified" );
|
|
if ( modified ) {
|
|
jQuery.lastModified[ cacheURL ] = modified;
|
|
}
|
|
modified = jqXHR.getResponseHeader( "etag" );
|
|
if ( modified ) {
|
|
jQuery.etag[ cacheURL ] = modified;
|
|
}
|
|
}
|
|
|
|
// if no content
|
|
if ( status === 204 || s.type === "HEAD" ) {
|
|
statusText = "nocontent";
|
|
|
|
// if not modified
|
|
} else if ( status === 304 ) {
|
|
statusText = "notmodified";
|
|
|
|
// If we have data, let's convert it
|
|
} else {
|
|
statusText = response.state;
|
|
success = response.data;
|
|
error = response.error;
|
|
isSuccess = !error;
|
|
}
|
|
} else {
|
|
|
|
// We extract error from statusText
|
|
// then normalize statusText and status for non-aborts
|
|
error = statusText;
|
|
if ( status || !statusText ) {
|
|
statusText = "error";
|
|
if ( status < 0 ) {
|
|
status = 0;
|
|
}
|
|
}
|
|
}
|
|
|
|
// Set data for the fake xhr object
|
|
jqXHR.status = status;
|
|
jqXHR.statusText = ( nativeStatusText || statusText ) + "";
|
|
|
|
// Success/Error
|
|
if ( isSuccess ) {
|
|
deferred.resolveWith( callbackContext, [ success, statusText, jqXHR ] );
|
|
} else {
|
|
deferred.rejectWith( callbackContext, [ jqXHR, statusText, error ] );
|
|
}
|
|
|
|
// Status-dependent callbacks
|
|
jqXHR.statusCode( statusCode );
|
|
statusCode = undefined;
|
|
|
|
if ( fireGlobals ) {
|
|
globalEventContext.trigger( isSuccess ? "ajaxSuccess" : "ajaxError",
|
|
[ jqXHR, s, isSuccess ? success : error ] );
|
|
}
|
|
|
|
// Complete
|
|
completeDeferred.fireWith( callbackContext, [ jqXHR, statusText ] );
|
|
|
|
if ( fireGlobals ) {
|
|
globalEventContext.trigger( "ajaxComplete", [ jqXHR, s ] );
|
|
|
|
// Handle the global AJAX counter
|
|
if ( !( --jQuery.active ) ) {
|
|
jQuery.event.trigger( "ajaxStop" );
|
|
}
|
|
}
|
|
}
|
|
|
|
return jqXHR;
|
|
},
|
|
|
|
getJSON: function( url, data, callback ) {
|
|
return jQuery.get( url, data, callback, "json" );
|
|
},
|
|
|
|
getScript: function( url, callback ) {
|
|
return jQuery.get( url, undefined, callback, "script" );
|
|
}
|
|
} );
|
|
|
|
jQuery.each( [ "get", "post" ], function( i, method ) {
|
|
jQuery[ method ] = function( url, data, callback, type ) {
|
|
|
|
// shift arguments if data argument was omitted
|
|
if ( jQuery.isFunction( data ) ) {
|
|
type = type || callback;
|
|
callback = data;
|
|
data = undefined;
|
|
}
|
|
|
|
// The url can be an options object (which then must have .url)
|
|
return jQuery.ajax( jQuery.extend( {
|
|
url: url,
|
|
type: method,
|
|
dataType: type,
|
|
data: data,
|
|
success: callback
|
|
}, jQuery.isPlainObject( url ) && url ) );
|
|
};
|
|
} );
|
|
|
|
|
|
jQuery._evalUrl = function( url ) {
|
|
return jQuery.ajax( {
|
|
url: url,
|
|
|
|
// Make this explicit, since user can override this through ajaxSetup (#11264)
|
|
type: "GET",
|
|
dataType: "script",
|
|
cache: true,
|
|
async: false,
|
|
global: false,
|
|
"throws": true
|
|
} );
|
|
};
|
|
|
|
|
|
jQuery.fn.extend( {
|
|
wrapAll: function( html ) {
|
|
if ( jQuery.isFunction( html ) ) {
|
|
return this.each( function( i ) {
|
|
jQuery( this ).wrapAll( html.call( this, i ) );
|
|
} );
|
|
}
|
|
|
|
if ( this[ 0 ] ) {
|
|
|
|
// The elements to wrap the target around
|
|
var wrap = jQuery( html, this[ 0 ].ownerDocument ).eq( 0 ).clone( true );
|
|
|
|
if ( this[ 0 ].parentNode ) {
|
|
wrap.insertBefore( this[ 0 ] );
|
|
}
|
|
|
|
wrap.map( function() {
|
|
var elem = this;
|
|
|
|
while ( elem.firstChild && elem.firstChild.nodeType === 1 ) {
|
|
elem = elem.firstChild;
|
|
}
|
|
|
|
return elem;
|
|
} ).append( this );
|
|
}
|
|
|
|
return this;
|
|
},
|
|
|
|
wrapInner: function( html ) {
|
|
if ( jQuery.isFunction( html ) ) {
|
|
return this.each( function( i ) {
|
|
jQuery( this ).wrapInner( html.call( this, i ) );
|
|
} );
|
|
}
|
|
|
|
return this.each( function() {
|
|
var self = jQuery( this ),
|
|
contents = self.contents();
|
|
|
|
if ( contents.length ) {
|
|
contents.wrapAll( html );
|
|
|
|
} else {
|
|
self.append( html );
|
|
}
|
|
} );
|
|
},
|
|
|
|
wrap: function( html ) {
|
|
var isFunction = jQuery.isFunction( html );
|
|
|
|
return this.each( function( i ) {
|
|
jQuery( this ).wrapAll( isFunction ? html.call( this, i ) : html );
|
|
} );
|
|
},
|
|
|
|
unwrap: function() {
|
|
return this.parent().each( function() {
|
|
if ( !jQuery.nodeName( this, "body" ) ) {
|
|
jQuery( this ).replaceWith( this.childNodes );
|
|
}
|
|
} ).end();
|
|
}
|
|
} );
|
|
|
|
|
|
function getDisplay( elem ) {
|
|
return elem.style && elem.style.display || jQuery.css( elem, "display" );
|
|
}
|
|
|
|
function filterHidden( elem ) {
|
|
while ( elem && elem.nodeType === 1 ) {
|
|
if ( getDisplay( elem ) === "none" || elem.type === "hidden" ) {
|
|
return true;
|
|
}
|
|
elem = elem.parentNode;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
jQuery.expr.filters.hidden = function( elem ) {
|
|
|
|
// Support: Opera <= 12.12
|
|
// Opera reports offsetWidths and offsetHeights less than zero on some elements
|
|
return support.reliableHiddenOffsets() ?
|
|
( elem.offsetWidth <= 0 && elem.offsetHeight <= 0 &&
|
|
!elem.getClientRects().length ) :
|
|
filterHidden( elem );
|
|
};
|
|
|
|
jQuery.expr.filters.visible = function( elem ) {
|
|
return !jQuery.expr.filters.hidden( elem );
|
|
};
|
|
|
|
|
|
|
|
|
|
var r20 = /%20/g,
|
|
rbracket = /\[\]$/,
|
|
rCRLF = /\r?\n/g,
|
|
rsubmitterTypes = /^(?:submit|button|image|reset|file)$/i,
|
|
rsubmittable = /^(?:input|select|textarea|keygen)/i;
|
|
|
|
function buildParams( prefix, obj, traditional, add ) {
|
|
var name;
|
|
|
|
if ( jQuery.isArray( obj ) ) {
|
|
|
|
// Serialize array item.
|
|
jQuery.each( obj, function( i, v ) {
|
|
if ( traditional || rbracket.test( prefix ) ) {
|
|
|
|
// Treat each array item as a scalar.
|
|
add( prefix, v );
|
|
|
|
} else {
|
|
|
|
// Item is non-scalar (array or object), encode its numeric index.
|
|
buildParams(
|
|
prefix + "[" + ( typeof v === "object" && v != null ? i : "" ) + "]",
|
|
v,
|
|
traditional,
|
|
add
|
|
);
|
|
}
|
|
} );
|
|
|
|
} else if ( !traditional && jQuery.type( obj ) === "object" ) {
|
|
|
|
// Serialize object item.
|
|
for ( name in obj ) {
|
|
buildParams( prefix + "[" + name + "]", obj[ name ], traditional, add );
|
|
}
|
|
|
|
} else {
|
|
|
|
// Serialize scalar item.
|
|
add( prefix, obj );
|
|
}
|
|
}
|
|
|
|
// Serialize an array of form elements or a set of
|
|
// key/values into a query string
|
|
jQuery.param = function( a, traditional ) {
|
|
var prefix,
|
|
s = [],
|
|
add = function( key, value ) {
|
|
|
|
// If value is a function, invoke it and return its value
|
|
value = jQuery.isFunction( value ) ? value() : ( value == null ? "" : value );
|
|
s[ s.length ] = encodeURIComponent( key ) + "=" + encodeURIComponent( value );
|
|
};
|
|
|
|
// Set traditional to true for jQuery <= 1.3.2 behavior.
|
|
if ( traditional === undefined ) {
|
|
traditional = jQuery.ajaxSettings && jQuery.ajaxSettings.traditional;
|
|
}
|
|
|
|
// If an array was passed in, assume that it is an array of form elements.
|
|
if ( jQuery.isArray( a ) || ( a.jquery && !jQuery.isPlainObject( a ) ) ) {
|
|
|
|
// Serialize the form elements
|
|
jQuery.each( a, function() {
|
|
add( this.name, this.value );
|
|
} );
|
|
|
|
} else {
|
|
|
|
// If traditional, encode the "old" way (the way 1.3.2 or older
|
|
// did it), otherwise encode params recursively.
|
|
for ( prefix in a ) {
|
|
buildParams( prefix, a[ prefix ], traditional, add );
|
|
}
|
|
}
|
|
|
|
// Return the resulting serialization
|
|
return s.join( "&" ).replace( r20, "+" );
|
|
};
|
|
|
|
jQuery.fn.extend( {
|
|
serialize: function() {
|
|
return jQuery.param( this.serializeArray() );
|
|
},
|
|
serializeArray: function() {
|
|
return this.map( function() {
|
|
|
|
// Can add propHook for "elements" to filter or add form elements
|
|
var elements = jQuery.prop( this, "elements" );
|
|
return elements ? jQuery.makeArray( elements ) : this;
|
|
} )
|
|
.filter( function() {
|
|
var type = this.type;
|
|
|
|
// Use .is(":disabled") so that fieldset[disabled] works
|
|
return this.name && !jQuery( this ).is( ":disabled" ) &&
|
|
rsubmittable.test( this.nodeName ) && !rsubmitterTypes.test( type ) &&
|
|
( this.checked || !rcheckableType.test( type ) );
|
|
} )
|
|
.map( function( i, elem ) {
|
|
var val = jQuery( this ).val();
|
|
|
|
return val == null ?
|
|
null :
|
|
jQuery.isArray( val ) ?
|
|
jQuery.map( val, function( val ) {
|
|
return { name: elem.name, value: val.replace( rCRLF, "\r\n" ) };
|
|
} ) :
|
|
{ name: elem.name, value: val.replace( rCRLF, "\r\n" ) };
|
|
} ).get();
|
|
}
|
|
} );
|
|
|
|
|
|
// Create the request object
|
|
// (This is still attached to ajaxSettings for backward compatibility)
|
|
jQuery.ajaxSettings.xhr = window.ActiveXObject !== undefined ?
|
|
|
|
// Support: IE6-IE8
|
|
function() {
|
|
|
|
// XHR cannot access local files, always use ActiveX for that case
|
|
if ( this.isLocal ) {
|
|
return createActiveXHR();
|
|
}
|
|
|
|
// Support: IE 9-11
|
|
// IE seems to error on cross-domain PATCH requests when ActiveX XHR
|
|
// is used. In IE 9+ always use the native XHR.
|
|
// Note: this condition won't catch Edge as it doesn't define
|
|
// document.documentMode but it also doesn't support ActiveX so it won't
|
|
// reach this code.
|
|
if ( document.documentMode > 8 ) {
|
|
return createStandardXHR();
|
|
}
|
|
|
|
// Support: IE<9
|
|
// oldIE XHR does not support non-RFC2616 methods (#13240)
|
|
// See http://msdn.microsoft.com/en-us/library/ie/ms536648(v=vs.85).aspx
|
|
// and http://www.w3.org/Protocols/rfc2616/rfc2616-sec9.html#sec9
|
|
// Although this check for six methods instead of eight
|
|
// since IE also does not support "trace" and "connect"
|
|
return /^(get|post|head|put|delete|options)$/i.test( this.type ) &&
|
|
createStandardXHR() || createActiveXHR();
|
|
} :
|
|
|
|
// For all other browsers, use the standard XMLHttpRequest object
|
|
createStandardXHR;
|
|
|
|
var xhrId = 0,
|
|
xhrCallbacks = {},
|
|
xhrSupported = jQuery.ajaxSettings.xhr();
|
|
|
|
// Support: IE<10
|
|
// Open requests must be manually aborted on unload (#5280)
|
|
// See https://support.microsoft.com/kb/2856746 for more info
|
|
if ( window.attachEvent ) {
|
|
window.attachEvent( "onunload", function() {
|
|
for ( var key in xhrCallbacks ) {
|
|
xhrCallbacks[ key ]( undefined, true );
|
|
}
|
|
} );
|
|
}
|
|
|
|
// Determine support properties
|
|
support.cors = !!xhrSupported && ( "withCredentials" in xhrSupported );
|
|
xhrSupported = support.ajax = !!xhrSupported;
|
|
|
|
// Create transport if the browser can provide an xhr
|
|
if ( xhrSupported ) {
|
|
|
|
jQuery.ajaxTransport( function( options ) {
|
|
|
|
// Cross domain only allowed if supported through XMLHttpRequest
|
|
if ( !options.crossDomain || support.cors ) {
|
|
|
|
var callback;
|
|
|
|
return {
|
|
send: function( headers, complete ) {
|
|
var i,
|
|
xhr = options.xhr(),
|
|
id = ++xhrId;
|
|
|
|
// Open the socket
|
|
xhr.open(
|
|
options.type,
|
|
options.url,
|
|
options.async,
|
|
options.username,
|
|
options.password
|
|
);
|
|
|
|
// Apply custom fields if provided
|
|
if ( options.xhrFields ) {
|
|
for ( i in options.xhrFields ) {
|
|
xhr[ i ] = options.xhrFields[ i ];
|
|
}
|
|
}
|
|
|
|
// Override mime type if needed
|
|
if ( options.mimeType && xhr.overrideMimeType ) {
|
|
xhr.overrideMimeType( options.mimeType );
|
|
}
|
|
|
|
// X-Requested-With header
|
|
// For cross-domain requests, seeing as conditions for a preflight are
|
|
// akin to a jigsaw puzzle, we simply never set it to be sure.
|
|
// (it can always be set on a per-request basis or even using ajaxSetup)
|
|
// For same-domain requests, won't change header if already provided.
|
|
if ( !options.crossDomain && !headers[ "X-Requested-With" ] ) {
|
|
headers[ "X-Requested-With" ] = "XMLHttpRequest";
|
|
}
|
|
|
|
// Set headers
|
|
for ( i in headers ) {
|
|
|
|
// Support: IE<9
|
|
// IE's ActiveXObject throws a 'Type Mismatch' exception when setting
|
|
// request header to a null-value.
|
|
//
|
|
// To keep consistent with other XHR implementations, cast the value
|
|
// to string and ignore `undefined`.
|
|
if ( headers[ i ] !== undefined ) {
|
|
xhr.setRequestHeader( i, headers[ i ] + "" );
|
|
}
|
|
}
|
|
|
|
// Do send the request
|
|
// This may raise an exception which is actually
|
|
// handled in jQuery.ajax (so no try/catch here)
|
|
xhr.send( ( options.hasContent && options.data ) || null );
|
|
|
|
// Listener
|
|
callback = function( _, isAbort ) {
|
|
var status, statusText, responses;
|
|
|
|
// Was never called and is aborted or complete
|
|
if ( callback && ( isAbort || xhr.readyState === 4 ) ) {
|
|
|
|
// Clean up
|
|
delete xhrCallbacks[ id ];
|
|
callback = undefined;
|
|
xhr.onreadystatechange = jQuery.noop;
|
|
|
|
// Abort manually if needed
|
|
if ( isAbort ) {
|
|
if ( xhr.readyState !== 4 ) {
|
|
xhr.abort();
|
|
}
|
|
} else {
|
|
responses = {};
|
|
status = xhr.status;
|
|
|
|
// Support: IE<10
|
|
// Accessing binary-data responseText throws an exception
|
|
// (#11426)
|
|
if ( typeof xhr.responseText === "string" ) {
|
|
responses.text = xhr.responseText;
|
|
}
|
|
|
|
// Firefox throws an exception when accessing
|
|
// statusText for faulty cross-domain requests
|
|
try {
|
|
statusText = xhr.statusText;
|
|
} catch ( e ) {
|
|
|
|
// We normalize with Webkit giving an empty statusText
|
|
statusText = "";
|
|
}
|
|
|
|
// Filter status for non standard behaviors
|
|
|
|
// If the request is local and we have data: assume a success
|
|
// (success with no data won't get notified, that's the best we
|
|
// can do given current implementations)
|
|
if ( !status && options.isLocal && !options.crossDomain ) {
|
|
status = responses.text ? 200 : 404;
|
|
|
|
// IE - #1450: sometimes returns 1223 when it should be 204
|
|
} else if ( status === 1223 ) {
|
|
status = 204;
|
|
}
|
|
}
|
|
}
|
|
|
|
// Call complete if needed
|
|
if ( responses ) {
|
|
complete( status, statusText, responses, xhr.getAllResponseHeaders() );
|
|
}
|
|
};
|
|
|
|
// Do send the request
|
|
// `xhr.send` may raise an exception, but it will be
|
|
// handled in jQuery.ajax (so no try/catch here)
|
|
if ( !options.async ) {
|
|
|
|
// If we're in sync mode we fire the callback
|
|
callback();
|
|
} else if ( xhr.readyState === 4 ) {
|
|
|
|
// (IE6 & IE7) if it's in cache and has been
|
|
// retrieved directly we need to fire the callback
|
|
window.setTimeout( callback );
|
|
} else {
|
|
|
|
// Register the callback, but delay it in case `xhr.send` throws
|
|
// Add to the list of active xhr callbacks
|
|
xhr.onreadystatechange = xhrCallbacks[ id ] = callback;
|
|
}
|
|
},
|
|
|
|
abort: function() {
|
|
if ( callback ) {
|
|
callback( undefined, true );
|
|
}
|
|
}
|
|
};
|
|
}
|
|
} );
|
|
}
|
|
|
|
// Functions to create xhrs
|
|
function createStandardXHR() {
|
|
try {
|
|
return new window.XMLHttpRequest();
|
|
} catch ( e ) {}
|
|
}
|
|
|
|
function createActiveXHR() {
|
|
try {
|
|
return new window.ActiveXObject( "Microsoft.XMLHTTP" );
|
|
} catch ( e ) {}
|
|
}
|
|
|
|
|
|
|
|
|
|
// Install script dataType
|
|
jQuery.ajaxSetup( {
|
|
accepts: {
|
|
script: "text/javascript, application/javascript, " +
|
|
"application/ecmascript, application/x-ecmascript"
|
|
},
|
|
contents: {
|
|
script: /\b(?:java|ecma)script\b/
|
|
},
|
|
converters: {
|
|
"text script": function( text ) {
|
|
jQuery.globalEval( text );
|
|
return text;
|
|
}
|
|
}
|
|
} );
|
|
|
|
// Handle cache's special case and global
|
|
jQuery.ajaxPrefilter( "script", function( s ) {
|
|
if ( s.cache === undefined ) {
|
|
s.cache = false;
|
|
}
|
|
if ( s.crossDomain ) {
|
|
s.type = "GET";
|
|
s.global = false;
|
|
}
|
|
} );
|
|
|
|
// Bind script tag hack transport
|
|
jQuery.ajaxTransport( "script", function( s ) {
|
|
|
|
// This transport only deals with cross domain requests
|
|
if ( s.crossDomain ) {
|
|
|
|
var script,
|
|
head = document.head || jQuery( "head" )[ 0 ] || document.documentElement;
|
|
|
|
return {
|
|
|
|
send: function( _, callback ) {
|
|
|
|
script = document.createElement( "script" );
|
|
|
|
script.async = true;
|
|
|
|
if ( s.scriptCharset ) {
|
|
script.charset = s.scriptCharset;
|
|
}
|
|
|
|
script.src = s.url;
|
|
|
|
// Attach handlers for all browsers
|
|
script.onload = script.onreadystatechange = function( _, isAbort ) {
|
|
|
|
if ( isAbort || !script.readyState || /loaded|complete/.test( script.readyState ) ) {
|
|
|
|
// Handle memory leak in IE
|
|
script.onload = script.onreadystatechange = null;
|
|
|
|
// Remove the script
|
|
if ( script.parentNode ) {
|
|
script.parentNode.removeChild( script );
|
|
}
|
|
|
|
// Dereference the script
|
|
script = null;
|
|
|
|
// Callback if not abort
|
|
if ( !isAbort ) {
|
|
callback( 200, "success" );
|
|
}
|
|
}
|
|
};
|
|
|
|
// Circumvent IE6 bugs with base elements (#2709 and #4378) by prepending
|
|
// Use native DOM manipulation to avoid our domManip AJAX trickery
|
|
head.insertBefore( script, head.firstChild );
|
|
},
|
|
|
|
abort: function() {
|
|
if ( script ) {
|
|
script.onload( undefined, true );
|
|
}
|
|
}
|
|
};
|
|
}
|
|
} );
|
|
|
|
|
|
|
|
|
|
var oldCallbacks = [],
|
|
rjsonp = /(=)\?(?=&|$)|\?\?/;
|
|
|
|
// Default jsonp settings
|
|
jQuery.ajaxSetup( {
|
|
jsonp: "callback",
|
|
jsonpCallback: function() {
|
|
var callback = oldCallbacks.pop() || ( jQuery.expando + "_" + ( nonce++ ) );
|
|
this[ callback ] = true;
|
|
return callback;
|
|
}
|
|
} );
|
|
|
|
// Detect, normalize options and install callbacks for jsonp requests
|
|
jQuery.ajaxPrefilter( "json jsonp", function( s, originalSettings, jqXHR ) {
|
|
|
|
var callbackName, overwritten, responseContainer,
|
|
jsonProp = s.jsonp !== false && ( rjsonp.test( s.url ) ?
|
|
"url" :
|
|
typeof s.data === "string" &&
|
|
( s.contentType || "" )
|
|
.indexOf( "application/x-www-form-urlencoded" ) === 0 &&
|
|
rjsonp.test( s.data ) && "data"
|
|
);
|
|
|
|
// Handle iff the expected data type is "jsonp" or we have a parameter to set
|
|
if ( jsonProp || s.dataTypes[ 0 ] === "jsonp" ) {
|
|
|
|
// Get callback name, remembering preexisting value associated with it
|
|
callbackName = s.jsonpCallback = jQuery.isFunction( s.jsonpCallback ) ?
|
|
s.jsonpCallback() :
|
|
s.jsonpCallback;
|
|
|
|
// Insert callback into url or form data
|
|
if ( jsonProp ) {
|
|
s[ jsonProp ] = s[ jsonProp ].replace( rjsonp, "$1" + callbackName );
|
|
} else if ( s.jsonp !== false ) {
|
|
s.url += ( rquery.test( s.url ) ? "&" : "?" ) + s.jsonp + "=" + callbackName;
|
|
}
|
|
|
|
// Use data converter to retrieve json after script execution
|
|
s.converters[ "script json" ] = function() {
|
|
if ( !responseContainer ) {
|
|
jQuery.error( callbackName + " was not called" );
|
|
}
|
|
return responseContainer[ 0 ];
|
|
};
|
|
|
|
// force json dataType
|
|
s.dataTypes[ 0 ] = "json";
|
|
|
|
// Install callback
|
|
overwritten = window[ callbackName ];
|
|
window[ callbackName ] = function() {
|
|
responseContainer = arguments;
|
|
};
|
|
|
|
// Clean-up function (fires after converters)
|
|
jqXHR.always( function() {
|
|
|
|
// If previous value didn't exist - remove it
|
|
if ( overwritten === undefined ) {
|
|
jQuery( window ).removeProp( callbackName );
|
|
|
|
// Otherwise restore preexisting value
|
|
} else {
|
|
window[ callbackName ] = overwritten;
|
|
}
|
|
|
|
// Save back as free
|
|
if ( s[ callbackName ] ) {
|
|
|
|
// make sure that re-using the options doesn't screw things around
|
|
s.jsonpCallback = originalSettings.jsonpCallback;
|
|
|
|
// save the callback name for future use
|
|
oldCallbacks.push( callbackName );
|
|
}
|
|
|
|
// Call if it was a function and we have a response
|
|
if ( responseContainer && jQuery.isFunction( overwritten ) ) {
|
|
overwritten( responseContainer[ 0 ] );
|
|
}
|
|
|
|
responseContainer = overwritten = undefined;
|
|
} );
|
|
|
|
// Delegate to script
|
|
return "script";
|
|
}
|
|
} );
|
|
|
|
|
|
|
|
|
|
// data: string of html
|
|
// context (optional): If specified, the fragment will be created in this context,
|
|
// defaults to document
|
|
// keepScripts (optional): If true, will include scripts passed in the html string
|
|
jQuery.parseHTML = function( data, context, keepScripts ) {
|
|
if ( !data || typeof data !== "string" ) {
|
|
return null;
|
|
}
|
|
if ( typeof context === "boolean" ) {
|
|
keepScripts = context;
|
|
context = false;
|
|
}
|
|
context = context || document;
|
|
|
|
var parsed = rsingleTag.exec( data ),
|
|
scripts = !keepScripts && [];
|
|
|
|
// Single tag
|
|
if ( parsed ) {
|
|
return [ context.createElement( parsed[ 1 ] ) ];
|
|
}
|
|
|
|
parsed = buildFragment( [ data ], context, scripts );
|
|
|
|
if ( scripts && scripts.length ) {
|
|
jQuery( scripts ).remove();
|
|
}
|
|
|
|
return jQuery.merge( [], parsed.childNodes );
|
|
};
|
|
|
|
|
|
// Keep a copy of the old load method
|
|
var _load = jQuery.fn.load;
|
|
|
|
/**
|
|
* Load a url into a page
|
|
*/
|
|
jQuery.fn.load = function( url, params, callback ) {
|
|
if ( typeof url !== "string" && _load ) {
|
|
return _load.apply( this, arguments );
|
|
}
|
|
|
|
var selector, type, response,
|
|
self = this,
|
|
off = url.indexOf( " " );
|
|
|
|
if ( off > -1 ) {
|
|
selector = jQuery.trim( url.slice( off, url.length ) );
|
|
url = url.slice( 0, off );
|
|
}
|
|
|
|
// If it's a function
|
|
if ( jQuery.isFunction( params ) ) {
|
|
|
|
// We assume that it's the callback
|
|
callback = params;
|
|
params = undefined;
|
|
|
|
// Otherwise, build a param string
|
|
} else if ( params && typeof params === "object" ) {
|
|
type = "POST";
|
|
}
|
|
|
|
// If we have elements to modify, make the request
|
|
if ( self.length > 0 ) {
|
|
jQuery.ajax( {
|
|
url: url,
|
|
|
|
// If "type" variable is undefined, then "GET" method will be used.
|
|
// Make value of this field explicit since
|
|
// user can override it through ajaxSetup method
|
|
type: type || "GET",
|
|
dataType: "html",
|
|
data: params
|
|
} ).done( function( responseText ) {
|
|
|
|
// Save response for use in complete callback
|
|
response = arguments;
|
|
|
|
self.html( selector ?
|
|
|
|
// If a selector was specified, locate the right elements in a dummy div
|
|
// Exclude scripts to avoid IE 'Permission Denied' errors
|
|
jQuery( "<div>" ).append( jQuery.parseHTML( responseText ) ).find( selector ) :
|
|
|
|
// Otherwise use the full result
|
|
responseText );
|
|
|
|
// If the request succeeds, this function gets "data", "status", "jqXHR"
|
|
// but they are ignored because response was set above.
|
|
// If it fails, this function gets "jqXHR", "status", "error"
|
|
} ).always( callback && function( jqXHR, status ) {
|
|
self.each( function() {
|
|
callback.apply( this, response || [ jqXHR.responseText, status, jqXHR ] );
|
|
} );
|
|
} );
|
|
}
|
|
|
|
return this;
|
|
};
|
|
|
|
|
|
|
|
|
|
// Attach a bunch of functions for handling common AJAX events
|
|
jQuery.each( [
|
|
"ajaxStart",
|
|
"ajaxStop",
|
|
"ajaxComplete",
|
|
"ajaxError",
|
|
"ajaxSuccess",
|
|
"ajaxSend"
|
|
], function( i, type ) {
|
|
jQuery.fn[ type ] = function( fn ) {
|
|
return this.on( type, fn );
|
|
};
|
|
} );
|
|
|
|
|
|
|
|
|
|
jQuery.expr.filters.animated = function( elem ) {
|
|
return jQuery.grep( jQuery.timers, function( fn ) {
|
|
return elem === fn.elem;
|
|
} ).length;
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
* Gets a window from an element
|
|
*/
|
|
function getWindow( elem ) {
|
|
return jQuery.isWindow( elem ) ?
|
|
elem :
|
|
elem.nodeType === 9 ?
|
|
elem.defaultView || elem.parentWindow :
|
|
false;
|
|
}
|
|
|
|
jQuery.offset = {
|
|
setOffset: function( elem, options, i ) {
|
|
var curPosition, curLeft, curCSSTop, curTop, curOffset, curCSSLeft, calculatePosition,
|
|
position = jQuery.css( elem, "position" ),
|
|
curElem = jQuery( elem ),
|
|
props = {};
|
|
|
|
// set position first, in-case top/left are set even on static elem
|
|
if ( position === "static" ) {
|
|
elem.style.position = "relative";
|
|
}
|
|
|
|
curOffset = curElem.offset();
|
|
curCSSTop = jQuery.css( elem, "top" );
|
|
curCSSLeft = jQuery.css( elem, "left" );
|
|
calculatePosition = ( position === "absolute" || position === "fixed" ) &&
|
|
jQuery.inArray( "auto", [ curCSSTop, curCSSLeft ] ) > -1;
|
|
|
|
// need to be able to calculate position if either top or left
|
|
// is auto and position is either absolute or fixed
|
|
if ( calculatePosition ) {
|
|
curPosition = curElem.position();
|
|
curTop = curPosition.top;
|
|
curLeft = curPosition.left;
|
|
} else {
|
|
curTop = parseFloat( curCSSTop ) || 0;
|
|
curLeft = parseFloat( curCSSLeft ) || 0;
|
|
}
|
|
|
|
if ( jQuery.isFunction( options ) ) {
|
|
|
|
// Use jQuery.extend here to allow modification of coordinates argument (gh-1848)
|
|
options = options.call( elem, i, jQuery.extend( {}, curOffset ) );
|
|
}
|
|
|
|
if ( options.top != null ) {
|
|
props.top = ( options.top - curOffset.top ) + curTop;
|
|
}
|
|
if ( options.left != null ) {
|
|
props.left = ( options.left - curOffset.left ) + curLeft;
|
|
}
|
|
|
|
if ( "using" in options ) {
|
|
options.using.call( elem, props );
|
|
} else {
|
|
curElem.css( props );
|
|
}
|
|
}
|
|
};
|
|
|
|
jQuery.fn.extend( {
|
|
offset: function( options ) {
|
|
if ( arguments.length ) {
|
|
return options === undefined ?
|
|
this :
|
|
this.each( function( i ) {
|
|
jQuery.offset.setOffset( this, options, i );
|
|
} );
|
|
}
|
|
|
|
var docElem, win,
|
|
box = { top: 0, left: 0 },
|
|
elem = this[ 0 ],
|
|
doc = elem && elem.ownerDocument;
|
|
|
|
if ( !doc ) {
|
|
return;
|
|
}
|
|
|
|
docElem = doc.documentElement;
|
|
|
|
// Make sure it's not a disconnected DOM node
|
|
if ( !jQuery.contains( docElem, elem ) ) {
|
|
return box;
|
|
}
|
|
|
|
// If we don't have gBCR, just use 0,0 rather than error
|
|
// BlackBerry 5, iOS 3 (original iPhone)
|
|
if ( typeof elem.getBoundingClientRect !== "undefined" ) {
|
|
box = elem.getBoundingClientRect();
|
|
}
|
|
win = getWindow( doc );
|
|
return {
|
|
top: box.top + ( win.pageYOffset || docElem.scrollTop ) - ( docElem.clientTop || 0 ),
|
|
left: box.left + ( win.pageXOffset || docElem.scrollLeft ) - ( docElem.clientLeft || 0 )
|
|
};
|
|
},
|
|
|
|
position: function() {
|
|
if ( !this[ 0 ] ) {
|
|
return;
|
|
}
|
|
|
|
var offsetParent, offset,
|
|
parentOffset = { top: 0, left: 0 },
|
|
elem = this[ 0 ];
|
|
|
|
// Fixed elements are offset from window (parentOffset = {top:0, left: 0},
|
|
// because it is its only offset parent
|
|
if ( jQuery.css( elem, "position" ) === "fixed" ) {
|
|
|
|
// we assume that getBoundingClientRect is available when computed position is fixed
|
|
offset = elem.getBoundingClientRect();
|
|
} else {
|
|
|
|
// Get *real* offsetParent
|
|
offsetParent = this.offsetParent();
|
|
|
|
// Get correct offsets
|
|
offset = this.offset();
|
|
if ( !jQuery.nodeName( offsetParent[ 0 ], "html" ) ) {
|
|
parentOffset = offsetParent.offset();
|
|
}
|
|
|
|
// Add offsetParent borders
|
|
parentOffset.top += jQuery.css( offsetParent[ 0 ], "borderTopWidth", true );
|
|
parentOffset.left += jQuery.css( offsetParent[ 0 ], "borderLeftWidth", true );
|
|
}
|
|
|
|
// Subtract parent offsets and element margins
|
|
// note: when an element has margin: auto the offsetLeft and marginLeft
|
|
// are the same in Safari causing offset.left to incorrectly be 0
|
|
return {
|
|
top: offset.top - parentOffset.top - jQuery.css( elem, "marginTop", true ),
|
|
left: offset.left - parentOffset.left - jQuery.css( elem, "marginLeft", true )
|
|
};
|
|
},
|
|
|
|
offsetParent: function() {
|
|
return this.map( function() {
|
|
var offsetParent = this.offsetParent;
|
|
|
|
while ( offsetParent && ( !jQuery.nodeName( offsetParent, "html" ) &&
|
|
jQuery.css( offsetParent, "position" ) === "static" ) ) {
|
|
offsetParent = offsetParent.offsetParent;
|
|
}
|
|
return offsetParent || documentElement;
|
|
} );
|
|
}
|
|
} );
|
|
|
|
// Create scrollLeft and scrollTop methods
|
|
jQuery.each( { scrollLeft: "pageXOffset", scrollTop: "pageYOffset" }, function( method, prop ) {
|
|
var top = /Y/.test( prop );
|
|
|
|
jQuery.fn[ method ] = function( val ) {
|
|
return access( this, function( elem, method, val ) {
|
|
var win = getWindow( elem );
|
|
|
|
if ( val === undefined ) {
|
|
return win ? ( prop in win ) ? win[ prop ] :
|
|
win.document.documentElement[ method ] :
|
|
elem[ method ];
|
|
}
|
|
|
|
if ( win ) {
|
|
win.scrollTo(
|
|
!top ? val : jQuery( win ).scrollLeft(),
|
|
top ? val : jQuery( win ).scrollTop()
|
|
);
|
|
|
|
} else {
|
|
elem[ method ] = val;
|
|
}
|
|
}, method, val, arguments.length, null );
|
|
};
|
|
} );
|
|
|
|
// Support: Safari<7-8+, Chrome<37-44+
|
|
// Add the top/left cssHooks using jQuery.fn.position
|
|
// Webkit bug: https://bugs.webkit.org/show_bug.cgi?id=29084
|
|
// getComputedStyle returns percent when specified for top/left/bottom/right
|
|
// rather than make the css module depend on the offset module, we just check for it here
|
|
jQuery.each( [ "top", "left" ], function( i, prop ) {
|
|
jQuery.cssHooks[ prop ] = addGetHookIf( support.pixelPosition,
|
|
function( elem, computed ) {
|
|
if ( computed ) {
|
|
computed = curCSS( elem, prop );
|
|
|
|
// if curCSS returns percentage, fallback to offset
|
|
return rnumnonpx.test( computed ) ?
|
|
jQuery( elem ).position()[ prop ] + "px" :
|
|
computed;
|
|
}
|
|
}
|
|
);
|
|
} );
|
|
|
|
|
|
// Create innerHeight, innerWidth, height, width, outerHeight and outerWidth methods
|
|
jQuery.each( { Height: "height", Width: "width" }, function( name, type ) {
|
|
jQuery.each( { padding: "inner" + name, content: type, "": "outer" + name },
|
|
function( defaultExtra, funcName ) {
|
|
|
|
// margin is only for outerHeight, outerWidth
|
|
jQuery.fn[ funcName ] = function( margin, value ) {
|
|
var chainable = arguments.length && ( defaultExtra || typeof margin !== "boolean" ),
|
|
extra = defaultExtra || ( margin === true || value === true ? "margin" : "border" );
|
|
|
|
return access( this, function( elem, type, value ) {
|
|
var doc;
|
|
|
|
if ( jQuery.isWindow( elem ) ) {
|
|
|
|
// As of 5/8/2012 this will yield incorrect results for Mobile Safari, but there
|
|
// isn't a whole lot we can do. See pull request at this URL for discussion:
|
|
// https://github.com/jquery/jquery/pull/764
|
|
return elem.document.documentElement[ "client" + name ];
|
|
}
|
|
|
|
// Get document width or height
|
|
if ( elem.nodeType === 9 ) {
|
|
doc = elem.documentElement;
|
|
|
|
// Either scroll[Width/Height] or offset[Width/Height] or client[Width/Height],
|
|
// whichever is greatest
|
|
// unfortunately, this causes bug #3838 in IE6/8 only,
|
|
// but there is currently no good, small way to fix it.
|
|
return Math.max(
|
|
elem.body[ "scroll" + name ], doc[ "scroll" + name ],
|
|
elem.body[ "offset" + name ], doc[ "offset" + name ],
|
|
doc[ "client" + name ]
|
|
);
|
|
}
|
|
|
|
return value === undefined ?
|
|
|
|
// Get width or height on the element, requesting but not forcing parseFloat
|
|
jQuery.css( elem, type, extra ) :
|
|
|
|
// Set width or height on the element
|
|
jQuery.style( elem, type, value, extra );
|
|
}, type, chainable ? margin : undefined, chainable, null );
|
|
};
|
|
} );
|
|
} );
|
|
|
|
|
|
jQuery.fn.extend( {
|
|
|
|
bind: function( types, data, fn ) {
|
|
return this.on( types, null, data, fn );
|
|
},
|
|
unbind: function( types, fn ) {
|
|
return this.off( types, null, fn );
|
|
},
|
|
|
|
delegate: function( selector, types, data, fn ) {
|
|
return this.on( types, selector, data, fn );
|
|
},
|
|
undelegate: function( selector, types, fn ) {
|
|
|
|
// ( namespace ) or ( selector, types [, fn] )
|
|
return arguments.length === 1 ?
|
|
this.off( selector, "**" ) :
|
|
this.off( types, selector || "**", fn );
|
|
}
|
|
} );
|
|
|
|
// The number of elements contained in the matched element set
|
|
jQuery.fn.size = function() {
|
|
return this.length;
|
|
};
|
|
|
|
jQuery.fn.andSelf = jQuery.fn.addBack;
|
|
|
|
|
|
|
|
|
|
// Register as a named AMD module, since jQuery can be concatenated with other
|
|
// files that may use define, but not via a proper concatenation script that
|
|
// understands anonymous AMD modules. A named AMD is safest and most robust
|
|
// way to register. Lowercase jquery is used because AMD module names are
|
|
// derived from file names, and jQuery is normally delivered in a lowercase
|
|
// file name. Do this after creating the global so that if an AMD module wants
|
|
// to call noConflict to hide this version of jQuery, it will work.
|
|
|
|
// Note that for maximum portability, libraries that are not jQuery should
|
|
// declare themselves as anonymous modules, and avoid setting a global if an
|
|
// AMD loader is present. jQuery is a special case. For more information, see
|
|
// https://github.com/jrburke/requirejs/wiki/Updating-existing-libraries#wiki-anon
|
|
|
|
if ( typeof define === "function" && define.amd ) {
|
|
define( "jquery", [], function() {
|
|
return jQuery;
|
|
} );
|
|
}
|
|
|
|
|
|
|
|
var
|
|
|
|
// Map over jQuery in case of overwrite
|
|
_jQuery = window.jQuery,
|
|
|
|
// Map over the $ in case of overwrite
|
|
_$ = window.$;
|
|
|
|
jQuery.noConflict = function( deep ) {
|
|
if ( window.$ === jQuery ) {
|
|
window.$ = _$;
|
|
}
|
|
|
|
if ( deep && window.jQuery === jQuery ) {
|
|
window.jQuery = _jQuery;
|
|
}
|
|
|
|
return jQuery;
|
|
};
|
|
|
|
// Expose jQuery and $ identifiers, even in
|
|
// AMD (#7102#comment:10, https://github.com/jquery/jquery/pull/557)
|
|
// and CommonJS for browser emulators (#13566)
|
|
if ( !noGlobal ) {
|
|
window.jQuery = window.$ = jQuery;
|
|
}
|
|
|
|
return jQuery;
|
|
}));
|
|
|
|
//! moment.js
|
|
//! version : 2.12.0
|
|
//! authors : Tim Wood, Iskren Chernev, Moment.js contributors
|
|
//! license : MIT
|
|
//! momentjs.com
|
|
|
|
;(function (global, factory) {
|
|
typeof exports === 'object' && typeof module !== 'undefined' ? module.exports = factory() :
|
|
typeof define === 'function' && define.amd ? define(factory) :
|
|
global.moment = factory()
|
|
}(this, function () { 'use strict';
|
|
|
|
var hookCallback;
|
|
|
|
function utils_hooks__hooks () {
|
|
return hookCallback.apply(null, arguments);
|
|
}
|
|
|
|
// This is done to register the method called with moment()
|
|
// without creating circular dependencies.
|
|
function setHookCallback (callback) {
|
|
hookCallback = callback;
|
|
}
|
|
|
|
function isArray(input) {
|
|
return input instanceof Array || Object.prototype.toString.call(input) === '[object Array]';
|
|
}
|
|
|
|
function isDate(input) {
|
|
return input instanceof Date || Object.prototype.toString.call(input) === '[object Date]';
|
|
}
|
|
|
|
function map(arr, fn) {
|
|
var res = [], i;
|
|
for (i = 0; i < arr.length; ++i) {
|
|
res.push(fn(arr[i], i));
|
|
}
|
|
return res;
|
|
}
|
|
|
|
function hasOwnProp(a, b) {
|
|
return Object.prototype.hasOwnProperty.call(a, b);
|
|
}
|
|
|
|
function extend(a, b) {
|
|
for (var i in b) {
|
|
if (hasOwnProp(b, i)) {
|
|
a[i] = b[i];
|
|
}
|
|
}
|
|
|
|
if (hasOwnProp(b, 'toString')) {
|
|
a.toString = b.toString;
|
|
}
|
|
|
|
if (hasOwnProp(b, 'valueOf')) {
|
|
a.valueOf = b.valueOf;
|
|
}
|
|
|
|
return a;
|
|
}
|
|
|
|
function create_utc__createUTC (input, format, locale, strict) {
|
|
return createLocalOrUTC(input, format, locale, strict, true).utc();
|
|
}
|
|
|
|
function defaultParsingFlags() {
|
|
// We need to deep clone this object.
|
|
return {
|
|
empty : false,
|
|
unusedTokens : [],
|
|
unusedInput : [],
|
|
overflow : -2,
|
|
charsLeftOver : 0,
|
|
nullInput : false,
|
|
invalidMonth : null,
|
|
invalidFormat : false,
|
|
userInvalidated : false,
|
|
iso : false
|
|
};
|
|
}
|
|
|
|
function getParsingFlags(m) {
|
|
if (m._pf == null) {
|
|
m._pf = defaultParsingFlags();
|
|
}
|
|
return m._pf;
|
|
}
|
|
|
|
function valid__isValid(m) {
|
|
if (m._isValid == null) {
|
|
var flags = getParsingFlags(m);
|
|
m._isValid = !isNaN(m._d.getTime()) &&
|
|
flags.overflow < 0 &&
|
|
!flags.empty &&
|
|
!flags.invalidMonth &&
|
|
!flags.invalidWeekday &&
|
|
!flags.nullInput &&
|
|
!flags.invalidFormat &&
|
|
!flags.userInvalidated;
|
|
|
|
if (m._strict) {
|
|
m._isValid = m._isValid &&
|
|
flags.charsLeftOver === 0 &&
|
|
flags.unusedTokens.length === 0 &&
|
|
flags.bigHour === undefined;
|
|
}
|
|
}
|
|
return m._isValid;
|
|
}
|
|
|
|
function valid__createInvalid (flags) {
|
|
var m = create_utc__createUTC(NaN);
|
|
if (flags != null) {
|
|
extend(getParsingFlags(m), flags);
|
|
}
|
|
else {
|
|
getParsingFlags(m).userInvalidated = true;
|
|
}
|
|
|
|
return m;
|
|
}
|
|
|
|
function isUndefined(input) {
|
|
return input === void 0;
|
|
}
|
|
|
|
// Plugins that add properties should also add the key here (null value),
|
|
// so we can properly clone ourselves.
|
|
var momentProperties = utils_hooks__hooks.momentProperties = [];
|
|
|
|
function copyConfig(to, from) {
|
|
var i, prop, val;
|
|
|
|
if (!isUndefined(from._isAMomentObject)) {
|
|
to._isAMomentObject = from._isAMomentObject;
|
|
}
|
|
if (!isUndefined(from._i)) {
|
|
to._i = from._i;
|
|
}
|
|
if (!isUndefined(from._f)) {
|
|
to._f = from._f;
|
|
}
|
|
if (!isUndefined(from._l)) {
|
|
to._l = from._l;
|
|
}
|
|
if (!isUndefined(from._strict)) {
|
|
to._strict = from._strict;
|
|
}
|
|
if (!isUndefined(from._tzm)) {
|
|
to._tzm = from._tzm;
|
|
}
|
|
if (!isUndefined(from._isUTC)) {
|
|
to._isUTC = from._isUTC;
|
|
}
|
|
if (!isUndefined(from._offset)) {
|
|
to._offset = from._offset;
|
|
}
|
|
if (!isUndefined(from._pf)) {
|
|
to._pf = getParsingFlags(from);
|
|
}
|
|
if (!isUndefined(from._locale)) {
|
|
to._locale = from._locale;
|
|
}
|
|
|
|
if (momentProperties.length > 0) {
|
|
for (i in momentProperties) {
|
|
prop = momentProperties[i];
|
|
val = from[prop];
|
|
if (!isUndefined(val)) {
|
|
to[prop] = val;
|
|
}
|
|
}
|
|
}
|
|
|
|
return to;
|
|
}
|
|
|
|
var updateInProgress = false;
|
|
|
|
// Moment prototype object
|
|
function Moment(config) {
|
|
copyConfig(this, config);
|
|
this._d = new Date(config._d != null ? config._d.getTime() : NaN);
|
|
// Prevent infinite loop in case updateOffset creates new moment
|
|
// objects.
|
|
if (updateInProgress === false) {
|
|
updateInProgress = true;
|
|
utils_hooks__hooks.updateOffset(this);
|
|
updateInProgress = false;
|
|
}
|
|
}
|
|
|
|
function isMoment (obj) {
|
|
return obj instanceof Moment || (obj != null && obj._isAMomentObject != null);
|
|
}
|
|
|
|
function absFloor (number) {
|
|
if (number < 0) {
|
|
return Math.ceil(number);
|
|
} else {
|
|
return Math.floor(number);
|
|
}
|
|
}
|
|
|
|
function toInt(argumentForCoercion) {
|
|
var coercedNumber = +argumentForCoercion,
|
|
value = 0;
|
|
|
|
if (coercedNumber !== 0 && isFinite(coercedNumber)) {
|
|
value = absFloor(coercedNumber);
|
|
}
|
|
|
|
return value;
|
|
}
|
|
|
|
// compare two arrays, return the number of differences
|
|
function compareArrays(array1, array2, dontConvert) {
|
|
var len = Math.min(array1.length, array2.length),
|
|
lengthDiff = Math.abs(array1.length - array2.length),
|
|
diffs = 0,
|
|
i;
|
|
for (i = 0; i < len; i++) {
|
|
if ((dontConvert && array1[i] !== array2[i]) ||
|
|
(!dontConvert && toInt(array1[i]) !== toInt(array2[i]))) {
|
|
diffs++;
|
|
}
|
|
}
|
|
return diffs + lengthDiff;
|
|
}
|
|
|
|
function warn(msg) {
|
|
if (utils_hooks__hooks.suppressDeprecationWarnings === false &&
|
|
(typeof console !== 'undefined') && console.warn) {
|
|
console.warn('Deprecation warning: ' + msg);
|
|
}
|
|
}
|
|
|
|
function deprecate(msg, fn) {
|
|
var firstTime = true;
|
|
|
|
return extend(function () {
|
|
if (firstTime) {
|
|
warn(msg + '\nArguments: ' + Array.prototype.slice.call(arguments).join(', ') + '\n' + (new Error()).stack);
|
|
firstTime = false;
|
|
}
|
|
return fn.apply(this, arguments);
|
|
}, fn);
|
|
}
|
|
|
|
var deprecations = {};
|
|
|
|
function deprecateSimple(name, msg) {
|
|
if (!deprecations[name]) {
|
|
warn(msg);
|
|
deprecations[name] = true;
|
|
}
|
|
}
|
|
|
|
utils_hooks__hooks.suppressDeprecationWarnings = false;
|
|
|
|
function isFunction(input) {
|
|
return input instanceof Function || Object.prototype.toString.call(input) === '[object Function]';
|
|
}
|
|
|
|
function isObject(input) {
|
|
return Object.prototype.toString.call(input) === '[object Object]';
|
|
}
|
|
|
|
function locale_set__set (config) {
|
|
var prop, i;
|
|
for (i in config) {
|
|
prop = config[i];
|
|
if (isFunction(prop)) {
|
|
this[i] = prop;
|
|
} else {
|
|
this['_' + i] = prop;
|
|
}
|
|
}
|
|
this._config = config;
|
|
// Lenient ordinal parsing accepts just a number in addition to
|
|
// number + (possibly) stuff coming from _ordinalParseLenient.
|
|
this._ordinalParseLenient = new RegExp(this._ordinalParse.source + '|' + (/\d{1,2}/).source);
|
|
}
|
|
|
|
function mergeConfigs(parentConfig, childConfig) {
|
|
var res = extend({}, parentConfig), prop;
|
|
for (prop in childConfig) {
|
|
if (hasOwnProp(childConfig, prop)) {
|
|
if (isObject(parentConfig[prop]) && isObject(childConfig[prop])) {
|
|
res[prop] = {};
|
|
extend(res[prop], parentConfig[prop]);
|
|
extend(res[prop], childConfig[prop]);
|
|
} else if (childConfig[prop] != null) {
|
|
res[prop] = childConfig[prop];
|
|
} else {
|
|
delete res[prop];
|
|
}
|
|
}
|
|
}
|
|
return res;
|
|
}
|
|
|
|
function Locale(config) {
|
|
if (config != null) {
|
|
this.set(config);
|
|
}
|
|
}
|
|
|
|
// internal storage for locale config files
|
|
var locales = {};
|
|
var globalLocale;
|
|
|
|
function normalizeLocale(key) {
|
|
return key ? key.toLowerCase().replace('_', '-') : key;
|
|
}
|
|
|
|
// pick the locale from the array
|
|
// try ['en-au', 'en-gb'] as 'en-au', 'en-gb', 'en', as in move through the list trying each
|
|
// substring from most specific to least, but move to the next array item if it's a more specific variant than the current root
|
|
function chooseLocale(names) {
|
|
var i = 0, j, next, locale, split;
|
|
|
|
while (i < names.length) {
|
|
split = normalizeLocale(names[i]).split('-');
|
|
j = split.length;
|
|
next = normalizeLocale(names[i + 1]);
|
|
next = next ? next.split('-') : null;
|
|
while (j > 0) {
|
|
locale = loadLocale(split.slice(0, j).join('-'));
|
|
if (locale) {
|
|
return locale;
|
|
}
|
|
if (next && next.length >= j && compareArrays(split, next, true) >= j - 1) {
|
|
//the next array item is better than a shallower substring of this one
|
|
break;
|
|
}
|
|
j--;
|
|
}
|
|
i++;
|
|
}
|
|
return null;
|
|
}
|
|
|
|
function loadLocale(name) {
|
|
var oldLocale = null;
|
|
// TODO: Find a better way to register and load all the locales in Node
|
|
if (!locales[name] && (typeof module !== 'undefined') &&
|
|
module && module.exports) {
|
|
try {
|
|
oldLocale = globalLocale._abbr;
|
|
require('./locale/' + name);
|
|
// because defineLocale currently also sets the global locale, we
|
|
// want to undo that for lazy loaded locales
|
|
locale_locales__getSetGlobalLocale(oldLocale);
|
|
} catch (e) { }
|
|
}
|
|
return locales[name];
|
|
}
|
|
|
|
// This function will load locale and then set the global locale. If
|
|
// no arguments are passed in, it will simply return the current global
|
|
// locale key.
|
|
function locale_locales__getSetGlobalLocale (key, values) {
|
|
var data;
|
|
if (key) {
|
|
if (isUndefined(values)) {
|
|
data = locale_locales__getLocale(key);
|
|
}
|
|
else {
|
|
data = defineLocale(key, values);
|
|
}
|
|
|
|
if (data) {
|
|
// moment.duration._locale = moment._locale = data;
|
|
globalLocale = data;
|
|
}
|
|
}
|
|
|
|
return globalLocale._abbr;
|
|
}
|
|
|
|
function defineLocale (name, config) {
|
|
if (config !== null) {
|
|
config.abbr = name;
|
|
if (locales[name] != null) {
|
|
deprecateSimple('defineLocaleOverride',
|
|
'use moment.updateLocale(localeName, config) to change ' +
|
|
'an existing locale. moment.defineLocale(localeName, ' +
|
|
'config) should only be used for creating a new locale');
|
|
config = mergeConfigs(locales[name]._config, config);
|
|
} else if (config.parentLocale != null) {
|
|
if (locales[config.parentLocale] != null) {
|
|
config = mergeConfigs(locales[config.parentLocale]._config, config);
|
|
} else {
|
|
// treat as if there is no base config
|
|
deprecateSimple('parentLocaleUndefined',
|
|
'specified parentLocale is not defined yet');
|
|
}
|
|
}
|
|
locales[name] = new Locale(config);
|
|
|
|
// backwards compat for now: also set the locale
|
|
locale_locales__getSetGlobalLocale(name);
|
|
|
|
return locales[name];
|
|
} else {
|
|
// useful for testing
|
|
delete locales[name];
|
|
return null;
|
|
}
|
|
}
|
|
|
|
function updateLocale(name, config) {
|
|
if (config != null) {
|
|
var locale;
|
|
if (locales[name] != null) {
|
|
config = mergeConfigs(locales[name]._config, config);
|
|
}
|
|
locale = new Locale(config);
|
|
locale.parentLocale = locales[name];
|
|
locales[name] = locale;
|
|
|
|
// backwards compat for now: also set the locale
|
|
locale_locales__getSetGlobalLocale(name);
|
|
} else {
|
|
// pass null for config to unupdate, useful for tests
|
|
if (locales[name] != null) {
|
|
if (locales[name].parentLocale != null) {
|
|
locales[name] = locales[name].parentLocale;
|
|
} else if (locales[name] != null) {
|
|
delete locales[name];
|
|
}
|
|
}
|
|
}
|
|
return locales[name];
|
|
}
|
|
|
|
// returns locale data
|
|
function locale_locales__getLocale (key) {
|
|
var locale;
|
|
|
|
if (key && key._locale && key._locale._abbr) {
|
|
key = key._locale._abbr;
|
|
}
|
|
|
|
if (!key) {
|
|
return globalLocale;
|
|
}
|
|
|
|
if (!isArray(key)) {
|
|
//short-circuit everything else
|
|
locale = loadLocale(key);
|
|
if (locale) {
|
|
return locale;
|
|
}
|
|
key = [key];
|
|
}
|
|
|
|
return chooseLocale(key);
|
|
}
|
|
|
|
function locale_locales__listLocales() {
|
|
return Object.keys(locales);
|
|
}
|
|
|
|
var aliases = {};
|
|
|
|
function addUnitAlias (unit, shorthand) {
|
|
var lowerCase = unit.toLowerCase();
|
|
aliases[lowerCase] = aliases[lowerCase + 's'] = aliases[shorthand] = unit;
|
|
}
|
|
|
|
function normalizeUnits(units) {
|
|
return typeof units === 'string' ? aliases[units] || aliases[units.toLowerCase()] : undefined;
|
|
}
|
|
|
|
function normalizeObjectUnits(inputObject) {
|
|
var normalizedInput = {},
|
|
normalizedProp,
|
|
prop;
|
|
|
|
for (prop in inputObject) {
|
|
if (hasOwnProp(inputObject, prop)) {
|
|
normalizedProp = normalizeUnits(prop);
|
|
if (normalizedProp) {
|
|
normalizedInput[normalizedProp] = inputObject[prop];
|
|
}
|
|
}
|
|
}
|
|
|
|
return normalizedInput;
|
|
}
|
|
|
|
function makeGetSet (unit, keepTime) {
|
|
return function (value) {
|
|
if (value != null) {
|
|
get_set__set(this, unit, value);
|
|
utils_hooks__hooks.updateOffset(this, keepTime);
|
|
return this;
|
|
} else {
|
|
return get_set__get(this, unit);
|
|
}
|
|
};
|
|
}
|
|
|
|
function get_set__get (mom, unit) {
|
|
return mom.isValid() ?
|
|
mom._d['get' + (mom._isUTC ? 'UTC' : '') + unit]() : NaN;
|
|
}
|
|
|
|
function get_set__set (mom, unit, value) {
|
|
if (mom.isValid()) {
|
|
mom._d['set' + (mom._isUTC ? 'UTC' : '') + unit](value);
|
|
}
|
|
}
|
|
|
|
// MOMENTS
|
|
|
|
function getSet (units, value) {
|
|
var unit;
|
|
if (typeof units === 'object') {
|
|
for (unit in units) {
|
|
this.set(unit, units[unit]);
|
|
}
|
|
} else {
|
|
units = normalizeUnits(units);
|
|
if (isFunction(this[units])) {
|
|
return this[units](value);
|
|
}
|
|
}
|
|
return this;
|
|
}
|
|
|
|
function zeroFill(number, targetLength, forceSign) {
|
|
var absNumber = '' + Math.abs(number),
|
|
zerosToFill = targetLength - absNumber.length,
|
|
sign = number >= 0;
|
|
return (sign ? (forceSign ? '+' : '') : '-') +
|
|
Math.pow(10, Math.max(0, zerosToFill)).toString().substr(1) + absNumber;
|
|
}
|
|
|
|
var formattingTokens = /(\[[^\[]*\])|(\\)?([Hh]mm(ss)?|Mo|MM?M?M?|Do|DDDo|DD?D?D?|ddd?d?|do?|w[o|w]?|W[o|W]?|Qo?|YYYYYY|YYYYY|YYYY|YY|gg(ggg?)?|GG(GGG?)?|e|E|a|A|hh?|HH?|mm?|ss?|S{1,9}|x|X|zz?|ZZ?|.)/g;
|
|
|
|
var localFormattingTokens = /(\[[^\[]*\])|(\\)?(LTS|LT|LL?L?L?|l{1,4})/g;
|
|
|
|
var formatFunctions = {};
|
|
|
|
var formatTokenFunctions = {};
|
|
|
|
// token: 'M'
|
|
// padded: ['MM', 2]
|
|
// ordinal: 'Mo'
|
|
// callback: function () { this.month() + 1 }
|
|
function addFormatToken (token, padded, ordinal, callback) {
|
|
var func = callback;
|
|
if (typeof callback === 'string') {
|
|
func = function () {
|
|
return this[callback]();
|
|
};
|
|
}
|
|
if (token) {
|
|
formatTokenFunctions[token] = func;
|
|
}
|
|
if (padded) {
|
|
formatTokenFunctions[padded[0]] = function () {
|
|
return zeroFill(func.apply(this, arguments), padded[1], padded[2]);
|
|
};
|
|
}
|
|
if (ordinal) {
|
|
formatTokenFunctions[ordinal] = function () {
|
|
return this.localeData().ordinal(func.apply(this, arguments), token);
|
|
};
|
|
}
|
|
}
|
|
|
|
function removeFormattingTokens(input) {
|
|
if (input.match(/\[[\s\S]/)) {
|
|
return input.replace(/^\[|\]$/g, '');
|
|
}
|
|
return input.replace(/\\/g, '');
|
|
}
|
|
|
|
function makeFormatFunction(format) {
|
|
var array = format.match(formattingTokens), i, length;
|
|
|
|
for (i = 0, length = array.length; i < length; i++) {
|
|
if (formatTokenFunctions[array[i]]) {
|
|
array[i] = formatTokenFunctions[array[i]];
|
|
} else {
|
|
array[i] = removeFormattingTokens(array[i]);
|
|
}
|
|
}
|
|
|
|
return function (mom) {
|
|
var output = '';
|
|
for (i = 0; i < length; i++) {
|
|
output += array[i] instanceof Function ? array[i].call(mom, format) : array[i];
|
|
}
|
|
return output;
|
|
};
|
|
}
|
|
|
|
// format date using native date object
|
|
function formatMoment(m, format) {
|
|
if (!m.isValid()) {
|
|
return m.localeData().invalidDate();
|
|
}
|
|
|
|
format = expandFormat(format, m.localeData());
|
|
formatFunctions[format] = formatFunctions[format] || makeFormatFunction(format);
|
|
|
|
return formatFunctions[format](m);
|
|
}
|
|
|
|
function expandFormat(format, locale) {
|
|
var i = 5;
|
|
|
|
function replaceLongDateFormatTokens(input) {
|
|
return locale.longDateFormat(input) || input;
|
|
}
|
|
|
|
localFormattingTokens.lastIndex = 0;
|
|
while (i >= 0 && localFormattingTokens.test(format)) {
|
|
format = format.replace(localFormattingTokens, replaceLongDateFormatTokens);
|
|
localFormattingTokens.lastIndex = 0;
|
|
i -= 1;
|
|
}
|
|
|
|
return format;
|
|
}
|
|
|
|
var match1 = /\d/; // 0 - 9
|
|
var match2 = /\d\d/; // 00 - 99
|
|
var match3 = /\d{3}/; // 000 - 999
|
|
var match4 = /\d{4}/; // 0000 - 9999
|
|
var match6 = /[+-]?\d{6}/; // -999999 - 999999
|
|
var match1to2 = /\d\d?/; // 0 - 99
|
|
var match3to4 = /\d\d\d\d?/; // 999 - 9999
|
|
var match5to6 = /\d\d\d\d\d\d?/; // 99999 - 999999
|
|
var match1to3 = /\d{1,3}/; // 0 - 999
|
|
var match1to4 = /\d{1,4}/; // 0 - 9999
|
|
var match1to6 = /[+-]?\d{1,6}/; // -999999 - 999999
|
|
|
|
var matchUnsigned = /\d+/; // 0 - inf
|
|
var matchSigned = /[+-]?\d+/; // -inf - inf
|
|
|
|
var matchOffset = /Z|[+-]\d\d:?\d\d/gi; // +00:00 -00:00 +0000 -0000 or Z
|
|
var matchShortOffset = /Z|[+-]\d\d(?::?\d\d)?/gi; // +00 -00 +00:00 -00:00 +0000 -0000 or Z
|
|
|
|
var matchTimestamp = /[+-]?\d+(\.\d{1,3})?/; // 123456789 123456789.123
|
|
|
|
// any word (or two) characters or numbers including two/three word month in arabic.
|
|
// includes scottish gaelic two word and hyphenated months
|
|
var matchWord = /[0-9]*['a-z\u00A0-\u05FF\u0700-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF]+|[\u0600-\u06FF\/]+(\s*?[\u0600-\u06FF]+){1,2}/i;
|
|
|
|
|
|
var regexes = {};
|
|
|
|
function addRegexToken (token, regex, strictRegex) {
|
|
regexes[token] = isFunction(regex) ? regex : function (isStrict, localeData) {
|
|
return (isStrict && strictRegex) ? strictRegex : regex;
|
|
};
|
|
}
|
|
|
|
function getParseRegexForToken (token, config) {
|
|
if (!hasOwnProp(regexes, token)) {
|
|
return new RegExp(unescapeFormat(token));
|
|
}
|
|
|
|
return regexes[token](config._strict, config._locale);
|
|
}
|
|
|
|
// Code from http://stackoverflow.com/questions/3561493/is-there-a-regexp-escape-function-in-javascript
|
|
function unescapeFormat(s) {
|
|
return regexEscape(s.replace('\\', '').replace(/\\(\[)|\\(\])|\[([^\]\[]*)\]|\\(.)/g, function (matched, p1, p2, p3, p4) {
|
|
return p1 || p2 || p3 || p4;
|
|
}));
|
|
}
|
|
|
|
function regexEscape(s) {
|
|
return s.replace(/[-\/\\^$*+?.()|[\]{}]/g, '\\$&');
|
|
}
|
|
|
|
var tokens = {};
|
|
|
|
function addParseToken (token, callback) {
|
|
var i, func = callback;
|
|
if (typeof token === 'string') {
|
|
token = [token];
|
|
}
|
|
if (typeof callback === 'number') {
|
|
func = function (input, array) {
|
|
array[callback] = toInt(input);
|
|
};
|
|
}
|
|
for (i = 0; i < token.length; i++) {
|
|
tokens[token[i]] = func;
|
|
}
|
|
}
|
|
|
|
function addWeekParseToken (token, callback) {
|
|
addParseToken(token, function (input, array, config, token) {
|
|
config._w = config._w || {};
|
|
callback(input, config._w, config, token);
|
|
});
|
|
}
|
|
|
|
function addTimeToArrayFromToken(token, input, config) {
|
|
if (input != null && hasOwnProp(tokens, token)) {
|
|
tokens[token](input, config._a, config, token);
|
|
}
|
|
}
|
|
|
|
var YEAR = 0;
|
|
var MONTH = 1;
|
|
var DATE = 2;
|
|
var HOUR = 3;
|
|
var MINUTE = 4;
|
|
var SECOND = 5;
|
|
var MILLISECOND = 6;
|
|
var WEEK = 7;
|
|
var WEEKDAY = 8;
|
|
|
|
function daysInMonth(year, month) {
|
|
return new Date(Date.UTC(year, month + 1, 0)).getUTCDate();
|
|
}
|
|
|
|
// FORMATTING
|
|
|
|
addFormatToken('M', ['MM', 2], 'Mo', function () {
|
|
return this.month() + 1;
|
|
});
|
|
|
|
addFormatToken('MMM', 0, 0, function (format) {
|
|
return this.localeData().monthsShort(this, format);
|
|
});
|
|
|
|
addFormatToken('MMMM', 0, 0, function (format) {
|
|
return this.localeData().months(this, format);
|
|
});
|
|
|
|
// ALIASES
|
|
|
|
addUnitAlias('month', 'M');
|
|
|
|
// PARSING
|
|
|
|
addRegexToken('M', match1to2);
|
|
addRegexToken('MM', match1to2, match2);
|
|
addRegexToken('MMM', function (isStrict, locale) {
|
|
return locale.monthsShortRegex(isStrict);
|
|
});
|
|
addRegexToken('MMMM', function (isStrict, locale) {
|
|
return locale.monthsRegex(isStrict);
|
|
});
|
|
|
|
addParseToken(['M', 'MM'], function (input, array) {
|
|
array[MONTH] = toInt(input) - 1;
|
|
});
|
|
|
|
addParseToken(['MMM', 'MMMM'], function (input, array, config, token) {
|
|
var month = config._locale.monthsParse(input, token, config._strict);
|
|
// if we didn't find a month name, mark the date as invalid.
|
|
if (month != null) {
|
|
array[MONTH] = month;
|
|
} else {
|
|
getParsingFlags(config).invalidMonth = input;
|
|
}
|
|
});
|
|
|
|
// LOCALES
|
|
|
|
var MONTHS_IN_FORMAT = /D[oD]?(\[[^\[\]]*\]|\s+)+MMMM?/;
|
|
var defaultLocaleMonths = 'January_February_March_April_May_June_July_August_September_October_November_December'.split('_');
|
|
function localeMonths (m, format) {
|
|
return isArray(this._months) ? this._months[m.month()] :
|
|
this._months[MONTHS_IN_FORMAT.test(format) ? 'format' : 'standalone'][m.month()];
|
|
}
|
|
|
|
var defaultLocaleMonthsShort = 'Jan_Feb_Mar_Apr_May_Jun_Jul_Aug_Sep_Oct_Nov_Dec'.split('_');
|
|
function localeMonthsShort (m, format) {
|
|
return isArray(this._monthsShort) ? this._monthsShort[m.month()] :
|
|
this._monthsShort[MONTHS_IN_FORMAT.test(format) ? 'format' : 'standalone'][m.month()];
|
|
}
|
|
|
|
function localeMonthsParse (monthName, format, strict) {
|
|
var i, mom, regex;
|
|
|
|
if (!this._monthsParse) {
|
|
this._monthsParse = [];
|
|
this._longMonthsParse = [];
|
|
this._shortMonthsParse = [];
|
|
}
|
|
|
|
for (i = 0; i < 12; i++) {
|
|
// make the regex if we don't have it already
|
|
mom = create_utc__createUTC([2000, i]);
|
|
if (strict && !this._longMonthsParse[i]) {
|
|
this._longMonthsParse[i] = new RegExp('^' + this.months(mom, '').replace('.', '') + '$', 'i');
|
|
this._shortMonthsParse[i] = new RegExp('^' + this.monthsShort(mom, '').replace('.', '') + '$', 'i');
|
|
}
|
|
if (!strict && !this._monthsParse[i]) {
|
|
regex = '^' + this.months(mom, '') + '|^' + this.monthsShort(mom, '');
|
|
this._monthsParse[i] = new RegExp(regex.replace('.', ''), 'i');
|
|
}
|
|
// test the regex
|
|
if (strict && format === 'MMMM' && this._longMonthsParse[i].test(monthName)) {
|
|
return i;
|
|
} else if (strict && format === 'MMM' && this._shortMonthsParse[i].test(monthName)) {
|
|
return i;
|
|
} else if (!strict && this._monthsParse[i].test(monthName)) {
|
|
return i;
|
|
}
|
|
}
|
|
}
|
|
|
|
// MOMENTS
|
|
|
|
function setMonth (mom, value) {
|
|
var dayOfMonth;
|
|
|
|
if (!mom.isValid()) {
|
|
// No op
|
|
return mom;
|
|
}
|
|
|
|
if (typeof value === 'string') {
|
|
if (/^\d+$/.test(value)) {
|
|
value = toInt(value);
|
|
} else {
|
|
value = mom.localeData().monthsParse(value);
|
|
// TODO: Another silent failure?
|
|
if (typeof value !== 'number') {
|
|
return mom;
|
|
}
|
|
}
|
|
}
|
|
|
|
dayOfMonth = Math.min(mom.date(), daysInMonth(mom.year(), value));
|
|
mom._d['set' + (mom._isUTC ? 'UTC' : '') + 'Month'](value, dayOfMonth);
|
|
return mom;
|
|
}
|
|
|
|
function getSetMonth (value) {
|
|
if (value != null) {
|
|
setMonth(this, value);
|
|
utils_hooks__hooks.updateOffset(this, true);
|
|
return this;
|
|
} else {
|
|
return get_set__get(this, 'Month');
|
|
}
|
|
}
|
|
|
|
function getDaysInMonth () {
|
|
return daysInMonth(this.year(), this.month());
|
|
}
|
|
|
|
var defaultMonthsShortRegex = matchWord;
|
|
function monthsShortRegex (isStrict) {
|
|
if (this._monthsParseExact) {
|
|
if (!hasOwnProp(this, '_monthsRegex')) {
|
|
computeMonthsParse.call(this);
|
|
}
|
|
if (isStrict) {
|
|
return this._monthsShortStrictRegex;
|
|
} else {
|
|
return this._monthsShortRegex;
|
|
}
|
|
} else {
|
|
return this._monthsShortStrictRegex && isStrict ?
|
|
this._monthsShortStrictRegex : this._monthsShortRegex;
|
|
}
|
|
}
|
|
|
|
var defaultMonthsRegex = matchWord;
|
|
function monthsRegex (isStrict) {
|
|
if (this._monthsParseExact) {
|
|
if (!hasOwnProp(this, '_monthsRegex')) {
|
|
computeMonthsParse.call(this);
|
|
}
|
|
if (isStrict) {
|
|
return this._monthsStrictRegex;
|
|
} else {
|
|
return this._monthsRegex;
|
|
}
|
|
} else {
|
|
return this._monthsStrictRegex && isStrict ?
|
|
this._monthsStrictRegex : this._monthsRegex;
|
|
}
|
|
}
|
|
|
|
function computeMonthsParse () {
|
|
function cmpLenRev(a, b) {
|
|
return b.length - a.length;
|
|
}
|
|
|
|
var shortPieces = [], longPieces = [], mixedPieces = [],
|
|
i, mom;
|
|
for (i = 0; i < 12; i++) {
|
|
// make the regex if we don't have it already
|
|
mom = create_utc__createUTC([2000, i]);
|
|
shortPieces.push(this.monthsShort(mom, ''));
|
|
longPieces.push(this.months(mom, ''));
|
|
mixedPieces.push(this.months(mom, ''));
|
|
mixedPieces.push(this.monthsShort(mom, ''));
|
|
}
|
|
// Sorting makes sure if one month (or abbr) is a prefix of another it
|
|
// will match the longer piece.
|
|
shortPieces.sort(cmpLenRev);
|
|
longPieces.sort(cmpLenRev);
|
|
mixedPieces.sort(cmpLenRev);
|
|
for (i = 0; i < 12; i++) {
|
|
shortPieces[i] = regexEscape(shortPieces[i]);
|
|
longPieces[i] = regexEscape(longPieces[i]);
|
|
mixedPieces[i] = regexEscape(mixedPieces[i]);
|
|
}
|
|
|
|
this._monthsRegex = new RegExp('^(' + mixedPieces.join('|') + ')', 'i');
|
|
this._monthsShortRegex = this._monthsRegex;
|
|
this._monthsStrictRegex = new RegExp('^(' + longPieces.join('|') + ')$', 'i');
|
|
this._monthsShortStrictRegex = new RegExp('^(' + shortPieces.join('|') + ')$', 'i');
|
|
}
|
|
|
|
function checkOverflow (m) {
|
|
var overflow;
|
|
var a = m._a;
|
|
|
|
if (a && getParsingFlags(m).overflow === -2) {
|
|
overflow =
|
|
a[MONTH] < 0 || a[MONTH] > 11 ? MONTH :
|
|
a[DATE] < 1 || a[DATE] > daysInMonth(a[YEAR], a[MONTH]) ? DATE :
|
|
a[HOUR] < 0 || a[HOUR] > 24 || (a[HOUR] === 24 && (a[MINUTE] !== 0 || a[SECOND] !== 0 || a[MILLISECOND] !== 0)) ? HOUR :
|
|
a[MINUTE] < 0 || a[MINUTE] > 59 ? MINUTE :
|
|
a[SECOND] < 0 || a[SECOND] > 59 ? SECOND :
|
|
a[MILLISECOND] < 0 || a[MILLISECOND] > 999 ? MILLISECOND :
|
|
-1;
|
|
|
|
if (getParsingFlags(m)._overflowDayOfYear && (overflow < YEAR || overflow > DATE)) {
|
|
overflow = DATE;
|
|
}
|
|
if (getParsingFlags(m)._overflowWeeks && overflow === -1) {
|
|
overflow = WEEK;
|
|
}
|
|
if (getParsingFlags(m)._overflowWeekday && overflow === -1) {
|
|
overflow = WEEKDAY;
|
|
}
|
|
|
|
getParsingFlags(m).overflow = overflow;
|
|
}
|
|
|
|
return m;
|
|
}
|
|
|
|
// iso 8601 regex
|
|
// 0000-00-00 0000-W00 or 0000-W00-0 + T + 00 or 00:00 or 00:00:00 or 00:00:00.000 + +00:00 or +0000 or +00)
|
|
var extendedIsoRegex = /^\s*((?:[+-]\d{6}|\d{4})-(?:\d\d-\d\d|W\d\d-\d|W\d\d|\d\d\d|\d\d))(?:(T| )(\d\d(?::\d\d(?::\d\d(?:[.,]\d+)?)?)?)([\+\-]\d\d(?::?\d\d)?|\s*Z)?)?/;
|
|
var basicIsoRegex = /^\s*((?:[+-]\d{6}|\d{4})(?:\d\d\d\d|W\d\d\d|W\d\d|\d\d\d|\d\d))(?:(T| )(\d\d(?:\d\d(?:\d\d(?:[.,]\d+)?)?)?)([\+\-]\d\d(?::?\d\d)?|\s*Z)?)?/;
|
|
|
|
var tzRegex = /Z|[+-]\d\d(?::?\d\d)?/;
|
|
|
|
var isoDates = [
|
|
['YYYYYY-MM-DD', /[+-]\d{6}-\d\d-\d\d/],
|
|
['YYYY-MM-DD', /\d{4}-\d\d-\d\d/],
|
|
['GGGG-[W]WW-E', /\d{4}-W\d\d-\d/],
|
|
['GGGG-[W]WW', /\d{4}-W\d\d/, false],
|
|
['YYYY-DDD', /\d{4}-\d{3}/],
|
|
['YYYY-MM', /\d{4}-\d\d/, false],
|
|
['YYYYYYMMDD', /[+-]\d{10}/],
|
|
['YYYYMMDD', /\d{8}/],
|
|
// YYYYMM is NOT allowed by the standard
|
|
['GGGG[W]WWE', /\d{4}W\d{3}/],
|
|
['GGGG[W]WW', /\d{4}W\d{2}/, false],
|
|
['YYYYDDD', /\d{7}/]
|
|
];
|
|
|
|
// iso time formats and regexes
|
|
var isoTimes = [
|
|
['HH:mm:ss.SSSS', /\d\d:\d\d:\d\d\.\d+/],
|
|
['HH:mm:ss,SSSS', /\d\d:\d\d:\d\d,\d+/],
|
|
['HH:mm:ss', /\d\d:\d\d:\d\d/],
|
|
['HH:mm', /\d\d:\d\d/],
|
|
['HHmmss.SSSS', /\d\d\d\d\d\d\.\d+/],
|
|
['HHmmss,SSSS', /\d\d\d\d\d\d,\d+/],
|
|
['HHmmss', /\d\d\d\d\d\d/],
|
|
['HHmm', /\d\d\d\d/],
|
|
['HH', /\d\d/]
|
|
];
|
|
|
|
var aspNetJsonRegex = /^\/?Date\((\-?\d+)/i;
|
|
|
|
// date from iso format
|
|
function configFromISO(config) {
|
|
var i, l,
|
|
string = config._i,
|
|
match = extendedIsoRegex.exec(string) || basicIsoRegex.exec(string),
|
|
allowTime, dateFormat, timeFormat, tzFormat;
|
|
|
|
if (match) {
|
|
getParsingFlags(config).iso = true;
|
|
|
|
for (i = 0, l = isoDates.length; i < l; i++) {
|
|
if (isoDates[i][1].exec(match[1])) {
|
|
dateFormat = isoDates[i][0];
|
|
allowTime = isoDates[i][2] !== false;
|
|
break;
|
|
}
|
|
}
|
|
if (dateFormat == null) {
|
|
config._isValid = false;
|
|
return;
|
|
}
|
|
if (match[3]) {
|
|
for (i = 0, l = isoTimes.length; i < l; i++) {
|
|
if (isoTimes[i][1].exec(match[3])) {
|
|
// match[2] should be 'T' or space
|
|
timeFormat = (match[2] || ' ') + isoTimes[i][0];
|
|
break;
|
|
}
|
|
}
|
|
if (timeFormat == null) {
|
|
config._isValid = false;
|
|
return;
|
|
}
|
|
}
|
|
if (!allowTime && timeFormat != null) {
|
|
config._isValid = false;
|
|
return;
|
|
}
|
|
if (match[4]) {
|
|
if (tzRegex.exec(match[4])) {
|
|
tzFormat = 'Z';
|
|
} else {
|
|
config._isValid = false;
|
|
return;
|
|
}
|
|
}
|
|
config._f = dateFormat + (timeFormat || '') + (tzFormat || '');
|
|
configFromStringAndFormat(config);
|
|
} else {
|
|
config._isValid = false;
|
|
}
|
|
}
|
|
|
|
// date from iso format or fallback
|
|
function configFromString(config) {
|
|
var matched = aspNetJsonRegex.exec(config._i);
|
|
|
|
if (matched !== null) {
|
|
config._d = new Date(+matched[1]);
|
|
return;
|
|
}
|
|
|
|
configFromISO(config);
|
|
if (config._isValid === false) {
|
|
delete config._isValid;
|
|
utils_hooks__hooks.createFromInputFallback(config);
|
|
}
|
|
}
|
|
|
|
utils_hooks__hooks.createFromInputFallback = deprecate(
|
|
'moment construction falls back to js Date. This is ' +
|
|
'discouraged and will be removed in upcoming major ' +
|
|
'release. Please refer to ' +
|
|
'https://github.com/moment/moment/issues/1407 for more info.',
|
|
function (config) {
|
|
config._d = new Date(config._i + (config._useUTC ? ' UTC' : ''));
|
|
}
|
|
);
|
|
|
|
function createDate (y, m, d, h, M, s, ms) {
|
|
//can't just apply() to create a date:
|
|
//http://stackoverflow.com/questions/181348/instantiating-a-javascript-object-by-calling-prototype-constructor-apply
|
|
var date = new Date(y, m, d, h, M, s, ms);
|
|
|
|
//the date constructor remaps years 0-99 to 1900-1999
|
|
if (y < 100 && y >= 0 && isFinite(date.getFullYear())) {
|
|
date.setFullYear(y);
|
|
}
|
|
return date;
|
|
}
|
|
|
|
function createUTCDate (y) {
|
|
var date = new Date(Date.UTC.apply(null, arguments));
|
|
|
|
//the Date.UTC function remaps years 0-99 to 1900-1999
|
|
if (y < 100 && y >= 0 && isFinite(date.getUTCFullYear())) {
|
|
date.setUTCFullYear(y);
|
|
}
|
|
return date;
|
|
}
|
|
|
|
// FORMATTING
|
|
|
|
addFormatToken('Y', 0, 0, function () {
|
|
var y = this.year();
|
|
return y <= 9999 ? '' + y : '+' + y;
|
|
});
|
|
|
|
addFormatToken(0, ['YY', 2], 0, function () {
|
|
return this.year() % 100;
|
|
});
|
|
|
|
addFormatToken(0, ['YYYY', 4], 0, 'year');
|
|
addFormatToken(0, ['YYYYY', 5], 0, 'year');
|
|
addFormatToken(0, ['YYYYYY', 6, true], 0, 'year');
|
|
|
|
// ALIASES
|
|
|
|
addUnitAlias('year', 'y');
|
|
|
|
// PARSING
|
|
|
|
addRegexToken('Y', matchSigned);
|
|
addRegexToken('YY', match1to2, match2);
|
|
addRegexToken('YYYY', match1to4, match4);
|
|
addRegexToken('YYYYY', match1to6, match6);
|
|
addRegexToken('YYYYYY', match1to6, match6);
|
|
|
|
addParseToken(['YYYYY', 'YYYYYY'], YEAR);
|
|
addParseToken('YYYY', function (input, array) {
|
|
array[YEAR] = input.length === 2 ? utils_hooks__hooks.parseTwoDigitYear(input) : toInt(input);
|
|
});
|
|
addParseToken('YY', function (input, array) {
|
|
array[YEAR] = utils_hooks__hooks.parseTwoDigitYear(input);
|
|
});
|
|
addParseToken('Y', function (input, array) {
|
|
array[YEAR] = parseInt(input, 10);
|
|
});
|
|
|
|
// HELPERS
|
|
|
|
function daysInYear(year) {
|
|
return isLeapYear(year) ? 366 : 365;
|
|
}
|
|
|
|
function isLeapYear(year) {
|
|
return (year % 4 === 0 && year % 100 !== 0) || year % 400 === 0;
|
|
}
|
|
|
|
// HOOKS
|
|
|
|
utils_hooks__hooks.parseTwoDigitYear = function (input) {
|
|
return toInt(input) + (toInt(input) > 68 ? 1900 : 2000);
|
|
};
|
|
|
|
// MOMENTS
|
|
|
|
var getSetYear = makeGetSet('FullYear', false);
|
|
|
|
function getIsLeapYear () {
|
|
return isLeapYear(this.year());
|
|
}
|
|
|
|
// start-of-first-week - start-of-year
|
|
function firstWeekOffset(year, dow, doy) {
|
|
var // first-week day -- which january is always in the first week (4 for iso, 1 for other)
|
|
fwd = 7 + dow - doy,
|
|
// first-week day local weekday -- which local weekday is fwd
|
|
fwdlw = (7 + createUTCDate(year, 0, fwd).getUTCDay() - dow) % 7;
|
|
|
|
return -fwdlw + fwd - 1;
|
|
}
|
|
|
|
//http://en.wikipedia.org/wiki/ISO_week_date#Calculating_a_date_given_the_year.2C_week_number_and_weekday
|
|
function dayOfYearFromWeeks(year, week, weekday, dow, doy) {
|
|
var localWeekday = (7 + weekday - dow) % 7,
|
|
weekOffset = firstWeekOffset(year, dow, doy),
|
|
dayOfYear = 1 + 7 * (week - 1) + localWeekday + weekOffset,
|
|
resYear, resDayOfYear;
|
|
|
|
if (dayOfYear <= 0) {
|
|
resYear = year - 1;
|
|
resDayOfYear = daysInYear(resYear) + dayOfYear;
|
|
} else if (dayOfYear > daysInYear(year)) {
|
|
resYear = year + 1;
|
|
resDayOfYear = dayOfYear - daysInYear(year);
|
|
} else {
|
|
resYear = year;
|
|
resDayOfYear = dayOfYear;
|
|
}
|
|
|
|
return {
|
|
year: resYear,
|
|
dayOfYear: resDayOfYear
|
|
};
|
|
}
|
|
|
|
function weekOfYear(mom, dow, doy) {
|
|
var weekOffset = firstWeekOffset(mom.year(), dow, doy),
|
|
week = Math.floor((mom.dayOfYear() - weekOffset - 1) / 7) + 1,
|
|
resWeek, resYear;
|
|
|
|
if (week < 1) {
|
|
resYear = mom.year() - 1;
|
|
resWeek = week + weeksInYear(resYear, dow, doy);
|
|
} else if (week > weeksInYear(mom.year(), dow, doy)) {
|
|
resWeek = week - weeksInYear(mom.year(), dow, doy);
|
|
resYear = mom.year() + 1;
|
|
} else {
|
|
resYear = mom.year();
|
|
resWeek = week;
|
|
}
|
|
|
|
return {
|
|
week: resWeek,
|
|
year: resYear
|
|
};
|
|
}
|
|
|
|
function weeksInYear(year, dow, doy) {
|
|
var weekOffset = firstWeekOffset(year, dow, doy),
|
|
weekOffsetNext = firstWeekOffset(year + 1, dow, doy);
|
|
return (daysInYear(year) - weekOffset + weekOffsetNext) / 7;
|
|
}
|
|
|
|
// Pick the first defined of two or three arguments.
|
|
function defaults(a, b, c) {
|
|
if (a != null) {
|
|
return a;
|
|
}
|
|
if (b != null) {
|
|
return b;
|
|
}
|
|
return c;
|
|
}
|
|
|
|
function currentDateArray(config) {
|
|
// hooks is actually the exported moment object
|
|
var nowValue = new Date(utils_hooks__hooks.now());
|
|
if (config._useUTC) {
|
|
return [nowValue.getUTCFullYear(), nowValue.getUTCMonth(), nowValue.getUTCDate()];
|
|
}
|
|
return [nowValue.getFullYear(), nowValue.getMonth(), nowValue.getDate()];
|
|
}
|
|
|
|
// convert an array to a date.
|
|
// the array should mirror the parameters below
|
|
// note: all values past the year are optional and will default to the lowest possible value.
|
|
// [year, month, day , hour, minute, second, millisecond]
|
|
function configFromArray (config) {
|
|
var i, date, input = [], currentDate, yearToUse;
|
|
|
|
if (config._d) {
|
|
return;
|
|
}
|
|
|
|
currentDate = currentDateArray(config);
|
|
|
|
//compute day of the year from weeks and weekdays
|
|
if (config._w && config._a[DATE] == null && config._a[MONTH] == null) {
|
|
dayOfYearFromWeekInfo(config);
|
|
}
|
|
|
|
//if the day of the year is set, figure out what it is
|
|
if (config._dayOfYear) {
|
|
yearToUse = defaults(config._a[YEAR], currentDate[YEAR]);
|
|
|
|
if (config._dayOfYear > daysInYear(yearToUse)) {
|
|
getParsingFlags(config)._overflowDayOfYear = true;
|
|
}
|
|
|
|
date = createUTCDate(yearToUse, 0, config._dayOfYear);
|
|
config._a[MONTH] = date.getUTCMonth();
|
|
config._a[DATE] = date.getUTCDate();
|
|
}
|
|
|
|
// Default to current date.
|
|
// * if no year, month, day of month are given, default to today
|
|
// * if day of month is given, default month and year
|
|
// * if month is given, default only year
|
|
// * if year is given, don't default anything
|
|
for (i = 0; i < 3 && config._a[i] == null; ++i) {
|
|
config._a[i] = input[i] = currentDate[i];
|
|
}
|
|
|
|
// Zero out whatever was not defaulted, including time
|
|
for (; i < 7; i++) {
|
|
config._a[i] = input[i] = (config._a[i] == null) ? (i === 2 ? 1 : 0) : config._a[i];
|
|
}
|
|
|
|
// Check for 24:00:00.000
|
|
if (config._a[HOUR] === 24 &&
|
|
config._a[MINUTE] === 0 &&
|
|
config._a[SECOND] === 0 &&
|
|
config._a[MILLISECOND] === 0) {
|
|
config._nextDay = true;
|
|
config._a[HOUR] = 0;
|
|
}
|
|
|
|
config._d = (config._useUTC ? createUTCDate : createDate).apply(null, input);
|
|
// Apply timezone offset from input. The actual utcOffset can be changed
|
|
// with parseZone.
|
|
if (config._tzm != null) {
|
|
config._d.setUTCMinutes(config._d.getUTCMinutes() - config._tzm);
|
|
}
|
|
|
|
if (config._nextDay) {
|
|
config._a[HOUR] = 24;
|
|
}
|
|
}
|
|
|
|
function dayOfYearFromWeekInfo(config) {
|
|
var w, weekYear, week, weekday, dow, doy, temp, weekdayOverflow;
|
|
|
|
w = config._w;
|
|
if (w.GG != null || w.W != null || w.E != null) {
|
|
dow = 1;
|
|
doy = 4;
|
|
|
|
// TODO: We need to take the current isoWeekYear, but that depends on
|
|
// how we interpret now (local, utc, fixed offset). So create
|
|
// a now version of current config (take local/utc/offset flags, and
|
|
// create now).
|
|
weekYear = defaults(w.GG, config._a[YEAR], weekOfYear(local__createLocal(), 1, 4).year);
|
|
week = defaults(w.W, 1);
|
|
weekday = defaults(w.E, 1);
|
|
if (weekday < 1 || weekday > 7) {
|
|
weekdayOverflow = true;
|
|
}
|
|
} else {
|
|
dow = config._locale._week.dow;
|
|
doy = config._locale._week.doy;
|
|
|
|
weekYear = defaults(w.gg, config._a[YEAR], weekOfYear(local__createLocal(), dow, doy).year);
|
|
week = defaults(w.w, 1);
|
|
|
|
if (w.d != null) {
|
|
// weekday -- low day numbers are considered next week
|
|
weekday = w.d;
|
|
if (weekday < 0 || weekday > 6) {
|
|
weekdayOverflow = true;
|
|
}
|
|
} else if (w.e != null) {
|
|
// local weekday -- counting starts from begining of week
|
|
weekday = w.e + dow;
|
|
if (w.e < 0 || w.e > 6) {
|
|
weekdayOverflow = true;
|
|
}
|
|
} else {
|
|
// default to begining of week
|
|
weekday = dow;
|
|
}
|
|
}
|
|
if (week < 1 || week > weeksInYear(weekYear, dow, doy)) {
|
|
getParsingFlags(config)._overflowWeeks = true;
|
|
} else if (weekdayOverflow != null) {
|
|
getParsingFlags(config)._overflowWeekday = true;
|
|
} else {
|
|
temp = dayOfYearFromWeeks(weekYear, week, weekday, dow, doy);
|
|
config._a[YEAR] = temp.year;
|
|
config._dayOfYear = temp.dayOfYear;
|
|
}
|
|
}
|
|
|
|
// constant that refers to the ISO standard
|
|
utils_hooks__hooks.ISO_8601 = function () {};
|
|
|
|
// date from string and format string
|
|
function configFromStringAndFormat(config) {
|
|
// TODO: Move this to another part of the creation flow to prevent circular deps
|
|
if (config._f === utils_hooks__hooks.ISO_8601) {
|
|
configFromISO(config);
|
|
return;
|
|
}
|
|
|
|
config._a = [];
|
|
getParsingFlags(config).empty = true;
|
|
|
|
// This array is used to make a Date, either with `new Date` or `Date.UTC`
|
|
var string = '' + config._i,
|
|
i, parsedInput, tokens, token, skipped,
|
|
stringLength = string.length,
|
|
totalParsedInputLength = 0;
|
|
|
|
tokens = expandFormat(config._f, config._locale).match(formattingTokens) || [];
|
|
|
|
for (i = 0; i < tokens.length; i++) {
|
|
token = tokens[i];
|
|
parsedInput = (string.match(getParseRegexForToken(token, config)) || [])[0];
|
|
// console.log('token', token, 'parsedInput', parsedInput,
|
|
// 'regex', getParseRegexForToken(token, config));
|
|
if (parsedInput) {
|
|
skipped = string.substr(0, string.indexOf(parsedInput));
|
|
if (skipped.length > 0) {
|
|
getParsingFlags(config).unusedInput.push(skipped);
|
|
}
|
|
string = string.slice(string.indexOf(parsedInput) + parsedInput.length);
|
|
totalParsedInputLength += parsedInput.length;
|
|
}
|
|
// don't parse if it's not a known token
|
|
if (formatTokenFunctions[token]) {
|
|
if (parsedInput) {
|
|
getParsingFlags(config).empty = false;
|
|
}
|
|
else {
|
|
getParsingFlags(config).unusedTokens.push(token);
|
|
}
|
|
addTimeToArrayFromToken(token, parsedInput, config);
|
|
}
|
|
else if (config._strict && !parsedInput) {
|
|
getParsingFlags(config).unusedTokens.push(token);
|
|
}
|
|
}
|
|
|
|
// add remaining unparsed input length to the string
|
|
getParsingFlags(config).charsLeftOver = stringLength - totalParsedInputLength;
|
|
if (string.length > 0) {
|
|
getParsingFlags(config).unusedInput.push(string);
|
|
}
|
|
|
|
// clear _12h flag if hour is <= 12
|
|
if (getParsingFlags(config).bigHour === true &&
|
|
config._a[HOUR] <= 12 &&
|
|
config._a[HOUR] > 0) {
|
|
getParsingFlags(config).bigHour = undefined;
|
|
}
|
|
// handle meridiem
|
|
config._a[HOUR] = meridiemFixWrap(config._locale, config._a[HOUR], config._meridiem);
|
|
|
|
configFromArray(config);
|
|
checkOverflow(config);
|
|
}
|
|
|
|
|
|
function meridiemFixWrap (locale, hour, meridiem) {
|
|
var isPm;
|
|
|
|
if (meridiem == null) {
|
|
// nothing to do
|
|
return hour;
|
|
}
|
|
if (locale.meridiemHour != null) {
|
|
return locale.meridiemHour(hour, meridiem);
|
|
} else if (locale.isPM != null) {
|
|
// Fallback
|
|
isPm = locale.isPM(meridiem);
|
|
if (isPm && hour < 12) {
|
|
hour += 12;
|
|
}
|
|
if (!isPm && hour === 12) {
|
|
hour = 0;
|
|
}
|
|
return hour;
|
|
} else {
|
|
// this is not supposed to happen
|
|
return hour;
|
|
}
|
|
}
|
|
|
|
// date from string and array of format strings
|
|
function configFromStringAndArray(config) {
|
|
var tempConfig,
|
|
bestMoment,
|
|
|
|
scoreToBeat,
|
|
i,
|
|
currentScore;
|
|
|
|
if (config._f.length === 0) {
|
|
getParsingFlags(config).invalidFormat = true;
|
|
config._d = new Date(NaN);
|
|
return;
|
|
}
|
|
|
|
for (i = 0; i < config._f.length; i++) {
|
|
currentScore = 0;
|
|
tempConfig = copyConfig({}, config);
|
|
if (config._useUTC != null) {
|
|
tempConfig._useUTC = config._useUTC;
|
|
}
|
|
tempConfig._f = config._f[i];
|
|
configFromStringAndFormat(tempConfig);
|
|
|
|
if (!valid__isValid(tempConfig)) {
|
|
continue;
|
|
}
|
|
|
|
// if there is any input that was not parsed add a penalty for that format
|
|
currentScore += getParsingFlags(tempConfig).charsLeftOver;
|
|
|
|
//or tokens
|
|
currentScore += getParsingFlags(tempConfig).unusedTokens.length * 10;
|
|
|
|
getParsingFlags(tempConfig).score = currentScore;
|
|
|
|
if (scoreToBeat == null || currentScore < scoreToBeat) {
|
|
scoreToBeat = currentScore;
|
|
bestMoment = tempConfig;
|
|
}
|
|
}
|
|
|
|
extend(config, bestMoment || tempConfig);
|
|
}
|
|
|
|
function configFromObject(config) {
|
|
if (config._d) {
|
|
return;
|
|
}
|
|
|
|
var i = normalizeObjectUnits(config._i);
|
|
config._a = map([i.year, i.month, i.day || i.date, i.hour, i.minute, i.second, i.millisecond], function (obj) {
|
|
return obj && parseInt(obj, 10);
|
|
});
|
|
|
|
configFromArray(config);
|
|
}
|
|
|
|
function createFromConfig (config) {
|
|
var res = new Moment(checkOverflow(prepareConfig(config)));
|
|
if (res._nextDay) {
|
|
// Adding is smart enough around DST
|
|
res.add(1, 'd');
|
|
res._nextDay = undefined;
|
|
}
|
|
|
|
return res;
|
|
}
|
|
|
|
function prepareConfig (config) {
|
|
var input = config._i,
|
|
format = config._f;
|
|
|
|
config._locale = config._locale || locale_locales__getLocale(config._l);
|
|
|
|
if (input === null || (format === undefined && input === '')) {
|
|
return valid__createInvalid({nullInput: true});
|
|
}
|
|
|
|
if (typeof input === 'string') {
|
|
config._i = input = config._locale.preparse(input);
|
|
}
|
|
|
|
if (isMoment(input)) {
|
|
return new Moment(checkOverflow(input));
|
|
} else if (isArray(format)) {
|
|
configFromStringAndArray(config);
|
|
} else if (format) {
|
|
configFromStringAndFormat(config);
|
|
} else if (isDate(input)) {
|
|
config._d = input;
|
|
} else {
|
|
configFromInput(config);
|
|
}
|
|
|
|
if (!valid__isValid(config)) {
|
|
config._d = null;
|
|
}
|
|
|
|
return config;
|
|
}
|
|
|
|
function configFromInput(config) {
|
|
var input = config._i;
|
|
if (input === undefined) {
|
|
config._d = new Date(utils_hooks__hooks.now());
|
|
} else if (isDate(input)) {
|
|
config._d = new Date(+input);
|
|
} else if (typeof input === 'string') {
|
|
configFromString(config);
|
|
} else if (isArray(input)) {
|
|
config._a = map(input.slice(0), function (obj) {
|
|
return parseInt(obj, 10);
|
|
});
|
|
configFromArray(config);
|
|
} else if (typeof(input) === 'object') {
|
|
configFromObject(config);
|
|
} else if (typeof(input) === 'number') {
|
|
// from milliseconds
|
|
config._d = new Date(input);
|
|
} else {
|
|
utils_hooks__hooks.createFromInputFallback(config);
|
|
}
|
|
}
|
|
|
|
function createLocalOrUTC (input, format, locale, strict, isUTC) {
|
|
var c = {};
|
|
|
|
if (typeof(locale) === 'boolean') {
|
|
strict = locale;
|
|
locale = undefined;
|
|
}
|
|
// object construction must be done this way.
|
|
// https://github.com/moment/moment/issues/1423
|
|
c._isAMomentObject = true;
|
|
c._useUTC = c._isUTC = isUTC;
|
|
c._l = locale;
|
|
c._i = input;
|
|
c._f = format;
|
|
c._strict = strict;
|
|
|
|
return createFromConfig(c);
|
|
}
|
|
|
|
function local__createLocal (input, format, locale, strict) {
|
|
return createLocalOrUTC(input, format, locale, strict, false);
|
|
}
|
|
|
|
var prototypeMin = deprecate(
|
|
'moment().min is deprecated, use moment.max instead. https://github.com/moment/moment/issues/1548',
|
|
function () {
|
|
var other = local__createLocal.apply(null, arguments);
|
|
if (this.isValid() && other.isValid()) {
|
|
return other < this ? this : other;
|
|
} else {
|
|
return valid__createInvalid();
|
|
}
|
|
}
|
|
);
|
|
|
|
var prototypeMax = deprecate(
|
|
'moment().max is deprecated, use moment.min instead. https://github.com/moment/moment/issues/1548',
|
|
function () {
|
|
var other = local__createLocal.apply(null, arguments);
|
|
if (this.isValid() && other.isValid()) {
|
|
return other > this ? this : other;
|
|
} else {
|
|
return valid__createInvalid();
|
|
}
|
|
}
|
|
);
|
|
|
|
// Pick a moment m from moments so that m[fn](other) is true for all
|
|
// other. This relies on the function fn to be transitive.
|
|
//
|
|
// moments should either be an array of moment objects or an array, whose
|
|
// first element is an array of moment objects.
|
|
function pickBy(fn, moments) {
|
|
var res, i;
|
|
if (moments.length === 1 && isArray(moments[0])) {
|
|
moments = moments[0];
|
|
}
|
|
if (!moments.length) {
|
|
return local__createLocal();
|
|
}
|
|
res = moments[0];
|
|
for (i = 1; i < moments.length; ++i) {
|
|
if (!moments[i].isValid() || moments[i][fn](res)) {
|
|
res = moments[i];
|
|
}
|
|
}
|
|
return res;
|
|
}
|
|
|
|
// TODO: Use [].sort instead?
|
|
function min () {
|
|
var args = [].slice.call(arguments, 0);
|
|
|
|
return pickBy('isBefore', args);
|
|
}
|
|
|
|
function max () {
|
|
var args = [].slice.call(arguments, 0);
|
|
|
|
return pickBy('isAfter', args);
|
|
}
|
|
|
|
var now = function () {
|
|
return Date.now ? Date.now() : +(new Date());
|
|
};
|
|
|
|
function Duration (duration) {
|
|
var normalizedInput = normalizeObjectUnits(duration),
|
|
years = normalizedInput.year || 0,
|
|
quarters = normalizedInput.quarter || 0,
|
|
months = normalizedInput.month || 0,
|
|
weeks = normalizedInput.week || 0,
|
|
days = normalizedInput.day || 0,
|
|
hours = normalizedInput.hour || 0,
|
|
minutes = normalizedInput.minute || 0,
|
|
seconds = normalizedInput.second || 0,
|
|
milliseconds = normalizedInput.millisecond || 0;
|
|
|
|
// representation for dateAddRemove
|
|
this._milliseconds = +milliseconds +
|
|
seconds * 1e3 + // 1000
|
|
minutes * 6e4 + // 1000 * 60
|
|
hours * 36e5; // 1000 * 60 * 60
|
|
// Because of dateAddRemove treats 24 hours as different from a
|
|
// day when working around DST, we need to store them separately
|
|
this._days = +days +
|
|
weeks * 7;
|
|
// It is impossible translate months into days without knowing
|
|
// which months you are are talking about, so we have to store
|
|
// it separately.
|
|
this._months = +months +
|
|
quarters * 3 +
|
|
years * 12;
|
|
|
|
this._data = {};
|
|
|
|
this._locale = locale_locales__getLocale();
|
|
|
|
this._bubble();
|
|
}
|
|
|
|
function isDuration (obj) {
|
|
return obj instanceof Duration;
|
|
}
|
|
|
|
// FORMATTING
|
|
|
|
function offset (token, separator) {
|
|
addFormatToken(token, 0, 0, function () {
|
|
var offset = this.utcOffset();
|
|
var sign = '+';
|
|
if (offset < 0) {
|
|
offset = -offset;
|
|
sign = '-';
|
|
}
|
|
return sign + zeroFill(~~(offset / 60), 2) + separator + zeroFill(~~(offset) % 60, 2);
|
|
});
|
|
}
|
|
|
|
offset('Z', ':');
|
|
offset('ZZ', '');
|
|
|
|
// PARSING
|
|
|
|
addRegexToken('Z', matchShortOffset);
|
|
addRegexToken('ZZ', matchShortOffset);
|
|
addParseToken(['Z', 'ZZ'], function (input, array, config) {
|
|
config._useUTC = true;
|
|
config._tzm = offsetFromString(matchShortOffset, input);
|
|
});
|
|
|
|
// HELPERS
|
|
|
|
// timezone chunker
|
|
// '+10:00' > ['10', '00']
|
|
// '-1530' > ['-15', '30']
|
|
var chunkOffset = /([\+\-]|\d\d)/gi;
|
|
|
|
function offsetFromString(matcher, string) {
|
|
var matches = ((string || '').match(matcher) || []);
|
|
var chunk = matches[matches.length - 1] || [];
|
|
var parts = (chunk + '').match(chunkOffset) || ['-', 0, 0];
|
|
var minutes = +(parts[1] * 60) + toInt(parts[2]);
|
|
|
|
return parts[0] === '+' ? minutes : -minutes;
|
|
}
|
|
|
|
// Return a moment from input, that is local/utc/zone equivalent to model.
|
|
function cloneWithOffset(input, model) {
|
|
var res, diff;
|
|
if (model._isUTC) {
|
|
res = model.clone();
|
|
diff = (isMoment(input) || isDate(input) ? +input : +local__createLocal(input)) - (+res);
|
|
// Use low-level api, because this fn is low-level api.
|
|
res._d.setTime(+res._d + diff);
|
|
utils_hooks__hooks.updateOffset(res, false);
|
|
return res;
|
|
} else {
|
|
return local__createLocal(input).local();
|
|
}
|
|
}
|
|
|
|
function getDateOffset (m) {
|
|
// On Firefox.24 Date#getTimezoneOffset returns a floating point.
|
|
// https://github.com/moment/moment/pull/1871
|
|
return -Math.round(m._d.getTimezoneOffset() / 15) * 15;
|
|
}
|
|
|
|
// HOOKS
|
|
|
|
// This function will be called whenever a moment is mutated.
|
|
// It is intended to keep the offset in sync with the timezone.
|
|
utils_hooks__hooks.updateOffset = function () {};
|
|
|
|
// MOMENTS
|
|
|
|
// keepLocalTime = true means only change the timezone, without
|
|
// affecting the local hour. So 5:31:26 +0300 --[utcOffset(2, true)]-->
|
|
// 5:31:26 +0200 It is possible that 5:31:26 doesn't exist with offset
|
|
// +0200, so we adjust the time as needed, to be valid.
|
|
//
|
|
// Keeping the time actually adds/subtracts (one hour)
|
|
// from the actual represented time. That is why we call updateOffset
|
|
// a second time. In case it wants us to change the offset again
|
|
// _changeInProgress == true case, then we have to adjust, because
|
|
// there is no such time in the given timezone.
|
|
function getSetOffset (input, keepLocalTime) {
|
|
var offset = this._offset || 0,
|
|
localAdjust;
|
|
if (!this.isValid()) {
|
|
return input != null ? this : NaN;
|
|
}
|
|
if (input != null) {
|
|
if (typeof input === 'string') {
|
|
input = offsetFromString(matchShortOffset, input);
|
|
} else if (Math.abs(input) < 16) {
|
|
input = input * 60;
|
|
}
|
|
if (!this._isUTC && keepLocalTime) {
|
|
localAdjust = getDateOffset(this);
|
|
}
|
|
this._offset = input;
|
|
this._isUTC = true;
|
|
if (localAdjust != null) {
|
|
this.add(localAdjust, 'm');
|
|
}
|
|
if (offset !== input) {
|
|
if (!keepLocalTime || this._changeInProgress) {
|
|
add_subtract__addSubtract(this, create__createDuration(input - offset, 'm'), 1, false);
|
|
} else if (!this._changeInProgress) {
|
|
this._changeInProgress = true;
|
|
utils_hooks__hooks.updateOffset(this, true);
|
|
this._changeInProgress = null;
|
|
}
|
|
}
|
|
return this;
|
|
} else {
|
|
return this._isUTC ? offset : getDateOffset(this);
|
|
}
|
|
}
|
|
|
|
function getSetZone (input, keepLocalTime) {
|
|
if (input != null) {
|
|
if (typeof input !== 'string') {
|
|
input = -input;
|
|
}
|
|
|
|
this.utcOffset(input, keepLocalTime);
|
|
|
|
return this;
|
|
} else {
|
|
return -this.utcOffset();
|
|
}
|
|
}
|
|
|
|
function setOffsetToUTC (keepLocalTime) {
|
|
return this.utcOffset(0, keepLocalTime);
|
|
}
|
|
|
|
function setOffsetToLocal (keepLocalTime) {
|
|
if (this._isUTC) {
|
|
this.utcOffset(0, keepLocalTime);
|
|
this._isUTC = false;
|
|
|
|
if (keepLocalTime) {
|
|
this.subtract(getDateOffset(this), 'm');
|
|
}
|
|
}
|
|
return this;
|
|
}
|
|
|
|
function setOffsetToParsedOffset () {
|
|
if (this._tzm) {
|
|
this.utcOffset(this._tzm);
|
|
} else if (typeof this._i === 'string') {
|
|
this.utcOffset(offsetFromString(matchOffset, this._i));
|
|
}
|
|
return this;
|
|
}
|
|
|
|
function hasAlignedHourOffset (input) {
|
|
if (!this.isValid()) {
|
|
return false;
|
|
}
|
|
input = input ? local__createLocal(input).utcOffset() : 0;
|
|
|
|
return (this.utcOffset() - input) % 60 === 0;
|
|
}
|
|
|
|
function isDaylightSavingTime () {
|
|
return (
|
|
this.utcOffset() > this.clone().month(0).utcOffset() ||
|
|
this.utcOffset() > this.clone().month(5).utcOffset()
|
|
);
|
|
}
|
|
|
|
function isDaylightSavingTimeShifted () {
|
|
if (!isUndefined(this._isDSTShifted)) {
|
|
return this._isDSTShifted;
|
|
}
|
|
|
|
var c = {};
|
|
|
|
copyConfig(c, this);
|
|
c = prepareConfig(c);
|
|
|
|
if (c._a) {
|
|
var other = c._isUTC ? create_utc__createUTC(c._a) : local__createLocal(c._a);
|
|
this._isDSTShifted = this.isValid() &&
|
|
compareArrays(c._a, other.toArray()) > 0;
|
|
} else {
|
|
this._isDSTShifted = false;
|
|
}
|
|
|
|
return this._isDSTShifted;
|
|
}
|
|
|
|
function isLocal () {
|
|
return this.isValid() ? !this._isUTC : false;
|
|
}
|
|
|
|
function isUtcOffset () {
|
|
return this.isValid() ? this._isUTC : false;
|
|
}
|
|
|
|
function isUtc () {
|
|
return this.isValid() ? this._isUTC && this._offset === 0 : false;
|
|
}
|
|
|
|
// ASP.NET json date format regex
|
|
var aspNetRegex = /^(\-)?(?:(\d*)[. ])?(\d+)\:(\d+)(?:\:(\d+)\.?(\d{3})?\d*)?$/;
|
|
|
|
// from http://docs.closure-library.googlecode.com/git/closure_goog_date_date.js.source.html
|
|
// somewhat more in line with 4.4.3.2 2004 spec, but allows decimal anywhere
|
|
// and further modified to allow for strings containing both week and day
|
|
var isoRegex = /^(-)?P(?:([0-9,.]*)Y)?(?:([0-9,.]*)M)?(?:([0-9,.]*)W)?(?:([0-9,.]*)D)?(?:T(?:([0-9,.]*)H)?(?:([0-9,.]*)M)?(?:([0-9,.]*)S)?)?$/;
|
|
|
|
function create__createDuration (input, key) {
|
|
var duration = input,
|
|
// matching against regexp is expensive, do it on demand
|
|
match = null,
|
|
sign,
|
|
ret,
|
|
diffRes;
|
|
|
|
if (isDuration(input)) {
|
|
duration = {
|
|
ms : input._milliseconds,
|
|
d : input._days,
|
|
M : input._months
|
|
};
|
|
} else if (typeof input === 'number') {
|
|
duration = {};
|
|
if (key) {
|
|
duration[key] = input;
|
|
} else {
|
|
duration.milliseconds = input;
|
|
}
|
|
} else if (!!(match = aspNetRegex.exec(input))) {
|
|
sign = (match[1] === '-') ? -1 : 1;
|
|
duration = {
|
|
y : 0,
|
|
d : toInt(match[DATE]) * sign,
|
|
h : toInt(match[HOUR]) * sign,
|
|
m : toInt(match[MINUTE]) * sign,
|
|
s : toInt(match[SECOND]) * sign,
|
|
ms : toInt(match[MILLISECOND]) * sign
|
|
};
|
|
} else if (!!(match = isoRegex.exec(input))) {
|
|
sign = (match[1] === '-') ? -1 : 1;
|
|
duration = {
|
|
y : parseIso(match[2], sign),
|
|
M : parseIso(match[3], sign),
|
|
w : parseIso(match[4], sign),
|
|
d : parseIso(match[5], sign),
|
|
h : parseIso(match[6], sign),
|
|
m : parseIso(match[7], sign),
|
|
s : parseIso(match[8], sign)
|
|
};
|
|
} else if (duration == null) {// checks for null or undefined
|
|
duration = {};
|
|
} else if (typeof duration === 'object' && ('from' in duration || 'to' in duration)) {
|
|
diffRes = momentsDifference(local__createLocal(duration.from), local__createLocal(duration.to));
|
|
|
|
duration = {};
|
|
duration.ms = diffRes.milliseconds;
|
|
duration.M = diffRes.months;
|
|
}
|
|
|
|
ret = new Duration(duration);
|
|
|
|
if (isDuration(input) && hasOwnProp(input, '_locale')) {
|
|
ret._locale = input._locale;
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
create__createDuration.fn = Duration.prototype;
|
|
|
|
function parseIso (inp, sign) {
|
|
// We'd normally use ~~inp for this, but unfortunately it also
|
|
// converts floats to ints.
|
|
// inp may be undefined, so careful calling replace on it.
|
|
var res = inp && parseFloat(inp.replace(',', '.'));
|
|
// apply sign while we're at it
|
|
return (isNaN(res) ? 0 : res) * sign;
|
|
}
|
|
|
|
function positiveMomentsDifference(base, other) {
|
|
var res = {milliseconds: 0, months: 0};
|
|
|
|
res.months = other.month() - base.month() +
|
|
(other.year() - base.year()) * 12;
|
|
if (base.clone().add(res.months, 'M').isAfter(other)) {
|
|
--res.months;
|
|
}
|
|
|
|
res.milliseconds = +other - +(base.clone().add(res.months, 'M'));
|
|
|
|
return res;
|
|
}
|
|
|
|
function momentsDifference(base, other) {
|
|
var res;
|
|
if (!(base.isValid() && other.isValid())) {
|
|
return {milliseconds: 0, months: 0};
|
|
}
|
|
|
|
other = cloneWithOffset(other, base);
|
|
if (base.isBefore(other)) {
|
|
res = positiveMomentsDifference(base, other);
|
|
} else {
|
|
res = positiveMomentsDifference(other, base);
|
|
res.milliseconds = -res.milliseconds;
|
|
res.months = -res.months;
|
|
}
|
|
|
|
return res;
|
|
}
|
|
|
|
function absRound (number) {
|
|
if (number < 0) {
|
|
return Math.round(-1 * number) * -1;
|
|
} else {
|
|
return Math.round(number);
|
|
}
|
|
}
|
|
|
|
// TODO: remove 'name' arg after deprecation is removed
|
|
function createAdder(direction, name) {
|
|
return function (val, period) {
|
|
var dur, tmp;
|
|
//invert the arguments, but complain about it
|
|
if (period !== null && !isNaN(+period)) {
|
|
deprecateSimple(name, 'moment().' + name + '(period, number) is deprecated. Please use moment().' + name + '(number, period).');
|
|
tmp = val; val = period; period = tmp;
|
|
}
|
|
|
|
val = typeof val === 'string' ? +val : val;
|
|
dur = create__createDuration(val, period);
|
|
add_subtract__addSubtract(this, dur, direction);
|
|
return this;
|
|
};
|
|
}
|
|
|
|
function add_subtract__addSubtract (mom, duration, isAdding, updateOffset) {
|
|
var milliseconds = duration._milliseconds,
|
|
days = absRound(duration._days),
|
|
months = absRound(duration._months);
|
|
|
|
if (!mom.isValid()) {
|
|
// No op
|
|
return;
|
|
}
|
|
|
|
updateOffset = updateOffset == null ? true : updateOffset;
|
|
|
|
if (milliseconds) {
|
|
mom._d.setTime(+mom._d + milliseconds * isAdding);
|
|
}
|
|
if (days) {
|
|
get_set__set(mom, 'Date', get_set__get(mom, 'Date') + days * isAdding);
|
|
}
|
|
if (months) {
|
|
setMonth(mom, get_set__get(mom, 'Month') + months * isAdding);
|
|
}
|
|
if (updateOffset) {
|
|
utils_hooks__hooks.updateOffset(mom, days || months);
|
|
}
|
|
}
|
|
|
|
var add_subtract__add = createAdder(1, 'add');
|
|
var add_subtract__subtract = createAdder(-1, 'subtract');
|
|
|
|
function moment_calendar__calendar (time, formats) {
|
|
// We want to compare the start of today, vs this.
|
|
// Getting start-of-today depends on whether we're local/utc/offset or not.
|
|
var now = time || local__createLocal(),
|
|
sod = cloneWithOffset(now, this).startOf('day'),
|
|
diff = this.diff(sod, 'days', true),
|
|
format = diff < -6 ? 'sameElse' :
|
|
diff < -1 ? 'lastWeek' :
|
|
diff < 0 ? 'lastDay' :
|
|
diff < 1 ? 'sameDay' :
|
|
diff < 2 ? 'nextDay' :
|
|
diff < 7 ? 'nextWeek' : 'sameElse';
|
|
|
|
var output = formats && (isFunction(formats[format]) ? formats[format]() : formats[format]);
|
|
|
|
return this.format(output || this.localeData().calendar(format, this, local__createLocal(now)));
|
|
}
|
|
|
|
function clone () {
|
|
return new Moment(this);
|
|
}
|
|
|
|
function isAfter (input, units) {
|
|
var localInput = isMoment(input) ? input : local__createLocal(input);
|
|
if (!(this.isValid() && localInput.isValid())) {
|
|
return false;
|
|
}
|
|
units = normalizeUnits(!isUndefined(units) ? units : 'millisecond');
|
|
if (units === 'millisecond') {
|
|
return +this > +localInput;
|
|
} else {
|
|
return +localInput < +this.clone().startOf(units);
|
|
}
|
|
}
|
|
|
|
function isBefore (input, units) {
|
|
var localInput = isMoment(input) ? input : local__createLocal(input);
|
|
if (!(this.isValid() && localInput.isValid())) {
|
|
return false;
|
|
}
|
|
units = normalizeUnits(!isUndefined(units) ? units : 'millisecond');
|
|
if (units === 'millisecond') {
|
|
return +this < +localInput;
|
|
} else {
|
|
return +this.clone().endOf(units) < +localInput;
|
|
}
|
|
}
|
|
|
|
function isBetween (from, to, units) {
|
|
return this.isAfter(from, units) && this.isBefore(to, units);
|
|
}
|
|
|
|
function isSame (input, units) {
|
|
var localInput = isMoment(input) ? input : local__createLocal(input),
|
|
inputMs;
|
|
if (!(this.isValid() && localInput.isValid())) {
|
|
return false;
|
|
}
|
|
units = normalizeUnits(units || 'millisecond');
|
|
if (units === 'millisecond') {
|
|
return +this === +localInput;
|
|
} else {
|
|
inputMs = +localInput;
|
|
return +(this.clone().startOf(units)) <= inputMs && inputMs <= +(this.clone().endOf(units));
|
|
}
|
|
}
|
|
|
|
function isSameOrAfter (input, units) {
|
|
return this.isSame(input, units) || this.isAfter(input,units);
|
|
}
|
|
|
|
function isSameOrBefore (input, units) {
|
|
return this.isSame(input, units) || this.isBefore(input,units);
|
|
}
|
|
|
|
function diff (input, units, asFloat) {
|
|
var that,
|
|
zoneDelta,
|
|
delta, output;
|
|
|
|
if (!this.isValid()) {
|
|
return NaN;
|
|
}
|
|
|
|
that = cloneWithOffset(input, this);
|
|
|
|
if (!that.isValid()) {
|
|
return NaN;
|
|
}
|
|
|
|
zoneDelta = (that.utcOffset() - this.utcOffset()) * 6e4;
|
|
|
|
units = normalizeUnits(units);
|
|
|
|
if (units === 'year' || units === 'month' || units === 'quarter') {
|
|
output = monthDiff(this, that);
|
|
if (units === 'quarter') {
|
|
output = output / 3;
|
|
} else if (units === 'year') {
|
|
output = output / 12;
|
|
}
|
|
} else {
|
|
delta = this - that;
|
|
output = units === 'second' ? delta / 1e3 : // 1000
|
|
units === 'minute' ? delta / 6e4 : // 1000 * 60
|
|
units === 'hour' ? delta / 36e5 : // 1000 * 60 * 60
|
|
units === 'day' ? (delta - zoneDelta) / 864e5 : // 1000 * 60 * 60 * 24, negate dst
|
|
units === 'week' ? (delta - zoneDelta) / 6048e5 : // 1000 * 60 * 60 * 24 * 7, negate dst
|
|
delta;
|
|
}
|
|
return asFloat ? output : absFloor(output);
|
|
}
|
|
|
|
function monthDiff (a, b) {
|
|
// difference in months
|
|
var wholeMonthDiff = ((b.year() - a.year()) * 12) + (b.month() - a.month()),
|
|
// b is in (anchor - 1 month, anchor + 1 month)
|
|
anchor = a.clone().add(wholeMonthDiff, 'months'),
|
|
anchor2, adjust;
|
|
|
|
if (b - anchor < 0) {
|
|
anchor2 = a.clone().add(wholeMonthDiff - 1, 'months');
|
|
// linear across the month
|
|
adjust = (b - anchor) / (anchor - anchor2);
|
|
} else {
|
|
anchor2 = a.clone().add(wholeMonthDiff + 1, 'months');
|
|
// linear across the month
|
|
adjust = (b - anchor) / (anchor2 - anchor);
|
|
}
|
|
|
|
return -(wholeMonthDiff + adjust);
|
|
}
|
|
|
|
utils_hooks__hooks.defaultFormat = 'YYYY-MM-DDTHH:mm:ssZ';
|
|
|
|
function toString () {
|
|
return this.clone().locale('en').format('ddd MMM DD YYYY HH:mm:ss [GMT]ZZ');
|
|
}
|
|
|
|
function moment_format__toISOString () {
|
|
var m = this.clone().utc();
|
|
if (0 < m.year() && m.year() <= 9999) {
|
|
if (isFunction(Date.prototype.toISOString)) {
|
|
// native implementation is ~50x faster, use it when we can
|
|
return this.toDate().toISOString();
|
|
} else {
|
|
return formatMoment(m, 'YYYY-MM-DD[T]HH:mm:ss.SSS[Z]');
|
|
}
|
|
} else {
|
|
return formatMoment(m, 'YYYYYY-MM-DD[T]HH:mm:ss.SSS[Z]');
|
|
}
|
|
}
|
|
|
|
function format (inputString) {
|
|
var output = formatMoment(this, inputString || utils_hooks__hooks.defaultFormat);
|
|
return this.localeData().postformat(output);
|
|
}
|
|
|
|
function from (time, withoutSuffix) {
|
|
if (this.isValid() &&
|
|
((isMoment(time) && time.isValid()) ||
|
|
local__createLocal(time).isValid())) {
|
|
return create__createDuration({to: this, from: time}).locale(this.locale()).humanize(!withoutSuffix);
|
|
} else {
|
|
return this.localeData().invalidDate();
|
|
}
|
|
}
|
|
|
|
function fromNow (withoutSuffix) {
|
|
return this.from(local__createLocal(), withoutSuffix);
|
|
}
|
|
|
|
function to (time, withoutSuffix) {
|
|
if (this.isValid() &&
|
|
((isMoment(time) && time.isValid()) ||
|
|
local__createLocal(time).isValid())) {
|
|
return create__createDuration({from: this, to: time}).locale(this.locale()).humanize(!withoutSuffix);
|
|
} else {
|
|
return this.localeData().invalidDate();
|
|
}
|
|
}
|
|
|
|
function toNow (withoutSuffix) {
|
|
return this.to(local__createLocal(), withoutSuffix);
|
|
}
|
|
|
|
// If passed a locale key, it will set the locale for this
|
|
// instance. Otherwise, it will return the locale configuration
|
|
// variables for this instance.
|
|
function locale (key) {
|
|
var newLocaleData;
|
|
|
|
if (key === undefined) {
|
|
return this._locale._abbr;
|
|
} else {
|
|
newLocaleData = locale_locales__getLocale(key);
|
|
if (newLocaleData != null) {
|
|
this._locale = newLocaleData;
|
|
}
|
|
return this;
|
|
}
|
|
}
|
|
|
|
var lang = deprecate(
|
|
'moment().lang() is deprecated. Instead, use moment().localeData() to get the language configuration. Use moment().locale() to change languages.',
|
|
function (key) {
|
|
if (key === undefined) {
|
|
return this.localeData();
|
|
} else {
|
|
return this.locale(key);
|
|
}
|
|
}
|
|
);
|
|
|
|
function localeData () {
|
|
return this._locale;
|
|
}
|
|
|
|
function startOf (units) {
|
|
units = normalizeUnits(units);
|
|
// the following switch intentionally omits break keywords
|
|
// to utilize falling through the cases.
|
|
switch (units) {
|
|
case 'year':
|
|
this.month(0);
|
|
/* falls through */
|
|
case 'quarter':
|
|
case 'month':
|
|
this.date(1);
|
|
/* falls through */
|
|
case 'week':
|
|
case 'isoWeek':
|
|
case 'day':
|
|
this.hours(0);
|
|
/* falls through */
|
|
case 'hour':
|
|
this.minutes(0);
|
|
/* falls through */
|
|
case 'minute':
|
|
this.seconds(0);
|
|
/* falls through */
|
|
case 'second':
|
|
this.milliseconds(0);
|
|
}
|
|
|
|
// weeks are a special case
|
|
if (units === 'week') {
|
|
this.weekday(0);
|
|
}
|
|
if (units === 'isoWeek') {
|
|
this.isoWeekday(1);
|
|
}
|
|
|
|
// quarters are also special
|
|
if (units === 'quarter') {
|
|
this.month(Math.floor(this.month() / 3) * 3);
|
|
}
|
|
|
|
return this;
|
|
}
|
|
|
|
function endOf (units) {
|
|
units = normalizeUnits(units);
|
|
if (units === undefined || units === 'millisecond') {
|
|
return this;
|
|
}
|
|
return this.startOf(units).add(1, (units === 'isoWeek' ? 'week' : units)).subtract(1, 'ms');
|
|
}
|
|
|
|
function to_type__valueOf () {
|
|
return +this._d - ((this._offset || 0) * 60000);
|
|
}
|
|
|
|
function unix () {
|
|
return Math.floor(+this / 1000);
|
|
}
|
|
|
|
function toDate () {
|
|
return this._offset ? new Date(+this) : this._d;
|
|
}
|
|
|
|
function toArray () {
|
|
var m = this;
|
|
return [m.year(), m.month(), m.date(), m.hour(), m.minute(), m.second(), m.millisecond()];
|
|
}
|
|
|
|
function toObject () {
|
|
var m = this;
|
|
return {
|
|
years: m.year(),
|
|
months: m.month(),
|
|
date: m.date(),
|
|
hours: m.hours(),
|
|
minutes: m.minutes(),
|
|
seconds: m.seconds(),
|
|
milliseconds: m.milliseconds()
|
|
};
|
|
}
|
|
|
|
function toJSON () {
|
|
// new Date(NaN).toJSON() === null
|
|
return this.isValid() ? this.toISOString() : null;
|
|
}
|
|
|
|
function moment_valid__isValid () {
|
|
return valid__isValid(this);
|
|
}
|
|
|
|
function parsingFlags () {
|
|
return extend({}, getParsingFlags(this));
|
|
}
|
|
|
|
function invalidAt () {
|
|
return getParsingFlags(this).overflow;
|
|
}
|
|
|
|
function creationData() {
|
|
return {
|
|
input: this._i,
|
|
format: this._f,
|
|
locale: this._locale,
|
|
isUTC: this._isUTC,
|
|
strict: this._strict
|
|
};
|
|
}
|
|
|
|
// FORMATTING
|
|
|
|
addFormatToken(0, ['gg', 2], 0, function () {
|
|
return this.weekYear() % 100;
|
|
});
|
|
|
|
addFormatToken(0, ['GG', 2], 0, function () {
|
|
return this.isoWeekYear() % 100;
|
|
});
|
|
|
|
function addWeekYearFormatToken (token, getter) {
|
|
addFormatToken(0, [token, token.length], 0, getter);
|
|
}
|
|
|
|
addWeekYearFormatToken('gggg', 'weekYear');
|
|
addWeekYearFormatToken('ggggg', 'weekYear');
|
|
addWeekYearFormatToken('GGGG', 'isoWeekYear');
|
|
addWeekYearFormatToken('GGGGG', 'isoWeekYear');
|
|
|
|
// ALIASES
|
|
|
|
addUnitAlias('weekYear', 'gg');
|
|
addUnitAlias('isoWeekYear', 'GG');
|
|
|
|
// PARSING
|
|
|
|
addRegexToken('G', matchSigned);
|
|
addRegexToken('g', matchSigned);
|
|
addRegexToken('GG', match1to2, match2);
|
|
addRegexToken('gg', match1to2, match2);
|
|
addRegexToken('GGGG', match1to4, match4);
|
|
addRegexToken('gggg', match1to4, match4);
|
|
addRegexToken('GGGGG', match1to6, match6);
|
|
addRegexToken('ggggg', match1to6, match6);
|
|
|
|
addWeekParseToken(['gggg', 'ggggg', 'GGGG', 'GGGGG'], function (input, week, config, token) {
|
|
week[token.substr(0, 2)] = toInt(input);
|
|
});
|
|
|
|
addWeekParseToken(['gg', 'GG'], function (input, week, config, token) {
|
|
week[token] = utils_hooks__hooks.parseTwoDigitYear(input);
|
|
});
|
|
|
|
// MOMENTS
|
|
|
|
function getSetWeekYear (input) {
|
|
return getSetWeekYearHelper.call(this,
|
|
input,
|
|
this.week(),
|
|
this.weekday(),
|
|
this.localeData()._week.dow,
|
|
this.localeData()._week.doy);
|
|
}
|
|
|
|
function getSetISOWeekYear (input) {
|
|
return getSetWeekYearHelper.call(this,
|
|
input, this.isoWeek(), this.isoWeekday(), 1, 4);
|
|
}
|
|
|
|
function getISOWeeksInYear () {
|
|
return weeksInYear(this.year(), 1, 4);
|
|
}
|
|
|
|
function getWeeksInYear () {
|
|
var weekInfo = this.localeData()._week;
|
|
return weeksInYear(this.year(), weekInfo.dow, weekInfo.doy);
|
|
}
|
|
|
|
function getSetWeekYearHelper(input, week, weekday, dow, doy) {
|
|
var weeksTarget;
|
|
if (input == null) {
|
|
return weekOfYear(this, dow, doy).year;
|
|
} else {
|
|
weeksTarget = weeksInYear(input, dow, doy);
|
|
if (week > weeksTarget) {
|
|
week = weeksTarget;
|
|
}
|
|
return setWeekAll.call(this, input, week, weekday, dow, doy);
|
|
}
|
|
}
|
|
|
|
function setWeekAll(weekYear, week, weekday, dow, doy) {
|
|
var dayOfYearData = dayOfYearFromWeeks(weekYear, week, weekday, dow, doy),
|
|
date = createUTCDate(dayOfYearData.year, 0, dayOfYearData.dayOfYear);
|
|
|
|
this.year(date.getUTCFullYear());
|
|
this.month(date.getUTCMonth());
|
|
this.date(date.getUTCDate());
|
|
return this;
|
|
}
|
|
|
|
// FORMATTING
|
|
|
|
addFormatToken('Q', 0, 'Qo', 'quarter');
|
|
|
|
// ALIASES
|
|
|
|
addUnitAlias('quarter', 'Q');
|
|
|
|
// PARSING
|
|
|
|
addRegexToken('Q', match1);
|
|
addParseToken('Q', function (input, array) {
|
|
array[MONTH] = (toInt(input) - 1) * 3;
|
|
});
|
|
|
|
// MOMENTS
|
|
|
|
function getSetQuarter (input) {
|
|
return input == null ? Math.ceil((this.month() + 1) / 3) : this.month((input - 1) * 3 + this.month() % 3);
|
|
}
|
|
|
|
// FORMATTING
|
|
|
|
addFormatToken('w', ['ww', 2], 'wo', 'week');
|
|
addFormatToken('W', ['WW', 2], 'Wo', 'isoWeek');
|
|
|
|
// ALIASES
|
|
|
|
addUnitAlias('week', 'w');
|
|
addUnitAlias('isoWeek', 'W');
|
|
|
|
// PARSING
|
|
|
|
addRegexToken('w', match1to2);
|
|
addRegexToken('ww', match1to2, match2);
|
|
addRegexToken('W', match1to2);
|
|
addRegexToken('WW', match1to2, match2);
|
|
|
|
addWeekParseToken(['w', 'ww', 'W', 'WW'], function (input, week, config, token) {
|
|
week[token.substr(0, 1)] = toInt(input);
|
|
});
|
|
|
|
// HELPERS
|
|
|
|
// LOCALES
|
|
|
|
function localeWeek (mom) {
|
|
return weekOfYear(mom, this._week.dow, this._week.doy).week;
|
|
}
|
|
|
|
var defaultLocaleWeek = {
|
|
dow : 0, // Sunday is the first day of the week.
|
|
doy : 6 // The week that contains Jan 1st is the first week of the year.
|
|
};
|
|
|
|
function localeFirstDayOfWeek () {
|
|
return this._week.dow;
|
|
}
|
|
|
|
function localeFirstDayOfYear () {
|
|
return this._week.doy;
|
|
}
|
|
|
|
// MOMENTS
|
|
|
|
function getSetWeek (input) {
|
|
var week = this.localeData().week(this);
|
|
return input == null ? week : this.add((input - week) * 7, 'd');
|
|
}
|
|
|
|
function getSetISOWeek (input) {
|
|
var week = weekOfYear(this, 1, 4).week;
|
|
return input == null ? week : this.add((input - week) * 7, 'd');
|
|
}
|
|
|
|
// FORMATTING
|
|
|
|
addFormatToken('D', ['DD', 2], 'Do', 'date');
|
|
|
|
// ALIASES
|
|
|
|
addUnitAlias('date', 'D');
|
|
|
|
// PARSING
|
|
|
|
addRegexToken('D', match1to2);
|
|
addRegexToken('DD', match1to2, match2);
|
|
addRegexToken('Do', function (isStrict, locale) {
|
|
return isStrict ? locale._ordinalParse : locale._ordinalParseLenient;
|
|
});
|
|
|
|
addParseToken(['D', 'DD'], DATE);
|
|
addParseToken('Do', function (input, array) {
|
|
array[DATE] = toInt(input.match(match1to2)[0], 10);
|
|
});
|
|
|
|
// MOMENTS
|
|
|
|
var getSetDayOfMonth = makeGetSet('Date', true);
|
|
|
|
// FORMATTING
|
|
|
|
addFormatToken('d', 0, 'do', 'day');
|
|
|
|
addFormatToken('dd', 0, 0, function (format) {
|
|
return this.localeData().weekdaysMin(this, format);
|
|
});
|
|
|
|
addFormatToken('ddd', 0, 0, function (format) {
|
|
return this.localeData().weekdaysShort(this, format);
|
|
});
|
|
|
|
addFormatToken('dddd', 0, 0, function (format) {
|
|
return this.localeData().weekdays(this, format);
|
|
});
|
|
|
|
addFormatToken('e', 0, 0, 'weekday');
|
|
addFormatToken('E', 0, 0, 'isoWeekday');
|
|
|
|
// ALIASES
|
|
|
|
addUnitAlias('day', 'd');
|
|
addUnitAlias('weekday', 'e');
|
|
addUnitAlias('isoWeekday', 'E');
|
|
|
|
// PARSING
|
|
|
|
addRegexToken('d', match1to2);
|
|
addRegexToken('e', match1to2);
|
|
addRegexToken('E', match1to2);
|
|
addRegexToken('dd', matchWord);
|
|
addRegexToken('ddd', matchWord);
|
|
addRegexToken('dddd', matchWord);
|
|
|
|
addWeekParseToken(['dd', 'ddd', 'dddd'], function (input, week, config, token) {
|
|
var weekday = config._locale.weekdaysParse(input, token, config._strict);
|
|
// if we didn't get a weekday name, mark the date as invalid
|
|
if (weekday != null) {
|
|
week.d = weekday;
|
|
} else {
|
|
getParsingFlags(config).invalidWeekday = input;
|
|
}
|
|
});
|
|
|
|
addWeekParseToken(['d', 'e', 'E'], function (input, week, config, token) {
|
|
week[token] = toInt(input);
|
|
});
|
|
|
|
// HELPERS
|
|
|
|
function parseWeekday(input, locale) {
|
|
if (typeof input !== 'string') {
|
|
return input;
|
|
}
|
|
|
|
if (!isNaN(input)) {
|
|
return parseInt(input, 10);
|
|
}
|
|
|
|
input = locale.weekdaysParse(input);
|
|
if (typeof input === 'number') {
|
|
return input;
|
|
}
|
|
|
|
return null;
|
|
}
|
|
|
|
// LOCALES
|
|
|
|
var defaultLocaleWeekdays = 'Sunday_Monday_Tuesday_Wednesday_Thursday_Friday_Saturday'.split('_');
|
|
function localeWeekdays (m, format) {
|
|
return isArray(this._weekdays) ? this._weekdays[m.day()] :
|
|
this._weekdays[this._weekdays.isFormat.test(format) ? 'format' : 'standalone'][m.day()];
|
|
}
|
|
|
|
var defaultLocaleWeekdaysShort = 'Sun_Mon_Tue_Wed_Thu_Fri_Sat'.split('_');
|
|
function localeWeekdaysShort (m) {
|
|
return this._weekdaysShort[m.day()];
|
|
}
|
|
|
|
var defaultLocaleWeekdaysMin = 'Su_Mo_Tu_We_Th_Fr_Sa'.split('_');
|
|
function localeWeekdaysMin (m) {
|
|
return this._weekdaysMin[m.day()];
|
|
}
|
|
|
|
function localeWeekdaysParse (weekdayName, format, strict) {
|
|
var i, mom, regex;
|
|
|
|
if (!this._weekdaysParse) {
|
|
this._weekdaysParse = [];
|
|
this._minWeekdaysParse = [];
|
|
this._shortWeekdaysParse = [];
|
|
this._fullWeekdaysParse = [];
|
|
}
|
|
|
|
for (i = 0; i < 7; i++) {
|
|
// make the regex if we don't have it already
|
|
|
|
mom = local__createLocal([2000, 1]).day(i);
|
|
if (strict && !this._fullWeekdaysParse[i]) {
|
|
this._fullWeekdaysParse[i] = new RegExp('^' + this.weekdays(mom, '').replace('.', '\.?') + '$', 'i');
|
|
this._shortWeekdaysParse[i] = new RegExp('^' + this.weekdaysShort(mom, '').replace('.', '\.?') + '$', 'i');
|
|
this._minWeekdaysParse[i] = new RegExp('^' + this.weekdaysMin(mom, '').replace('.', '\.?') + '$', 'i');
|
|
}
|
|
if (!this._weekdaysParse[i]) {
|
|
regex = '^' + this.weekdays(mom, '') + '|^' + this.weekdaysShort(mom, '') + '|^' + this.weekdaysMin(mom, '');
|
|
this._weekdaysParse[i] = new RegExp(regex.replace('.', ''), 'i');
|
|
}
|
|
// test the regex
|
|
if (strict && format === 'dddd' && this._fullWeekdaysParse[i].test(weekdayName)) {
|
|
return i;
|
|
} else if (strict && format === 'ddd' && this._shortWeekdaysParse[i].test(weekdayName)) {
|
|
return i;
|
|
} else if (strict && format === 'dd' && this._minWeekdaysParse[i].test(weekdayName)) {
|
|
return i;
|
|
} else if (!strict && this._weekdaysParse[i].test(weekdayName)) {
|
|
return i;
|
|
}
|
|
}
|
|
}
|
|
|
|
// MOMENTS
|
|
|
|
function getSetDayOfWeek (input) {
|
|
if (!this.isValid()) {
|
|
return input != null ? this : NaN;
|
|
}
|
|
var day = this._isUTC ? this._d.getUTCDay() : this._d.getDay();
|
|
if (input != null) {
|
|
input = parseWeekday(input, this.localeData());
|
|
return this.add(input - day, 'd');
|
|
} else {
|
|
return day;
|
|
}
|
|
}
|
|
|
|
function getSetLocaleDayOfWeek (input) {
|
|
if (!this.isValid()) {
|
|
return input != null ? this : NaN;
|
|
}
|
|
var weekday = (this.day() + 7 - this.localeData()._week.dow) % 7;
|
|
return input == null ? weekday : this.add(input - weekday, 'd');
|
|
}
|
|
|
|
function getSetISODayOfWeek (input) {
|
|
if (!this.isValid()) {
|
|
return input != null ? this : NaN;
|
|
}
|
|
// behaves the same as moment#day except
|
|
// as a getter, returns 7 instead of 0 (1-7 range instead of 0-6)
|
|
// as a setter, sunday should belong to the previous week.
|
|
return input == null ? this.day() || 7 : this.day(this.day() % 7 ? input : input - 7);
|
|
}
|
|
|
|
// FORMATTING
|
|
|
|
addFormatToken('DDD', ['DDDD', 3], 'DDDo', 'dayOfYear');
|
|
|
|
// ALIASES
|
|
|
|
addUnitAlias('dayOfYear', 'DDD');
|
|
|
|
// PARSING
|
|
|
|
addRegexToken('DDD', match1to3);
|
|
addRegexToken('DDDD', match3);
|
|
addParseToken(['DDD', 'DDDD'], function (input, array, config) {
|
|
config._dayOfYear = toInt(input);
|
|
});
|
|
|
|
// HELPERS
|
|
|
|
// MOMENTS
|
|
|
|
function getSetDayOfYear (input) {
|
|
var dayOfYear = Math.round((this.clone().startOf('day') - this.clone().startOf('year')) / 864e5) + 1;
|
|
return input == null ? dayOfYear : this.add((input - dayOfYear), 'd');
|
|
}
|
|
|
|
// FORMATTING
|
|
|
|
function hFormat() {
|
|
return this.hours() % 12 || 12;
|
|
}
|
|
|
|
addFormatToken('H', ['HH', 2], 0, 'hour');
|
|
addFormatToken('h', ['hh', 2], 0, hFormat);
|
|
|
|
addFormatToken('hmm', 0, 0, function () {
|
|
return '' + hFormat.apply(this) + zeroFill(this.minutes(), 2);
|
|
});
|
|
|
|
addFormatToken('hmmss', 0, 0, function () {
|
|
return '' + hFormat.apply(this) + zeroFill(this.minutes(), 2) +
|
|
zeroFill(this.seconds(), 2);
|
|
});
|
|
|
|
addFormatToken('Hmm', 0, 0, function () {
|
|
return '' + this.hours() + zeroFill(this.minutes(), 2);
|
|
});
|
|
|
|
addFormatToken('Hmmss', 0, 0, function () {
|
|
return '' + this.hours() + zeroFill(this.minutes(), 2) +
|
|
zeroFill(this.seconds(), 2);
|
|
});
|
|
|
|
function meridiem (token, lowercase) {
|
|
addFormatToken(token, 0, 0, function () {
|
|
return this.localeData().meridiem(this.hours(), this.minutes(), lowercase);
|
|
});
|
|
}
|
|
|
|
meridiem('a', true);
|
|
meridiem('A', false);
|
|
|
|
// ALIASES
|
|
|
|
addUnitAlias('hour', 'h');
|
|
|
|
// PARSING
|
|
|
|
function matchMeridiem (isStrict, locale) {
|
|
return locale._meridiemParse;
|
|
}
|
|
|
|
addRegexToken('a', matchMeridiem);
|
|
addRegexToken('A', matchMeridiem);
|
|
addRegexToken('H', match1to2);
|
|
addRegexToken('h', match1to2);
|
|
addRegexToken('HH', match1to2, match2);
|
|
addRegexToken('hh', match1to2, match2);
|
|
|
|
addRegexToken('hmm', match3to4);
|
|
addRegexToken('hmmss', match5to6);
|
|
addRegexToken('Hmm', match3to4);
|
|
addRegexToken('Hmmss', match5to6);
|
|
|
|
addParseToken(['H', 'HH'], HOUR);
|
|
addParseToken(['a', 'A'], function (input, array, config) {
|
|
config._isPm = config._locale.isPM(input);
|
|
config._meridiem = input;
|
|
});
|
|
addParseToken(['h', 'hh'], function (input, array, config) {
|
|
array[HOUR] = toInt(input);
|
|
getParsingFlags(config).bigHour = true;
|
|
});
|
|
addParseToken('hmm', function (input, array, config) {
|
|
var pos = input.length - 2;
|
|
array[HOUR] = toInt(input.substr(0, pos));
|
|
array[MINUTE] = toInt(input.substr(pos));
|
|
getParsingFlags(config).bigHour = true;
|
|
});
|
|
addParseToken('hmmss', function (input, array, config) {
|
|
var pos1 = input.length - 4;
|
|
var pos2 = input.length - 2;
|
|
array[HOUR] = toInt(input.substr(0, pos1));
|
|
array[MINUTE] = toInt(input.substr(pos1, 2));
|
|
array[SECOND] = toInt(input.substr(pos2));
|
|
getParsingFlags(config).bigHour = true;
|
|
});
|
|
addParseToken('Hmm', function (input, array, config) {
|
|
var pos = input.length - 2;
|
|
array[HOUR] = toInt(input.substr(0, pos));
|
|
array[MINUTE] = toInt(input.substr(pos));
|
|
});
|
|
addParseToken('Hmmss', function (input, array, config) {
|
|
var pos1 = input.length - 4;
|
|
var pos2 = input.length - 2;
|
|
array[HOUR] = toInt(input.substr(0, pos1));
|
|
array[MINUTE] = toInt(input.substr(pos1, 2));
|
|
array[SECOND] = toInt(input.substr(pos2));
|
|
});
|
|
|
|
// LOCALES
|
|
|
|
function localeIsPM (input) {
|
|
// IE8 Quirks Mode & IE7 Standards Mode do not allow accessing strings like arrays
|
|
// Using charAt should be more compatible.
|
|
return ((input + '').toLowerCase().charAt(0) === 'p');
|
|
}
|
|
|
|
var defaultLocaleMeridiemParse = /[ap]\.?m?\.?/i;
|
|
function localeMeridiem (hours, minutes, isLower) {
|
|
if (hours > 11) {
|
|
return isLower ? 'pm' : 'PM';
|
|
} else {
|
|
return isLower ? 'am' : 'AM';
|
|
}
|
|
}
|
|
|
|
|
|
// MOMENTS
|
|
|
|
// Setting the hour should keep the time, because the user explicitly
|
|
// specified which hour he wants. So trying to maintain the same hour (in
|
|
// a new timezone) makes sense. Adding/subtracting hours does not follow
|
|
// this rule.
|
|
var getSetHour = makeGetSet('Hours', true);
|
|
|
|
// FORMATTING
|
|
|
|
addFormatToken('m', ['mm', 2], 0, 'minute');
|
|
|
|
// ALIASES
|
|
|
|
addUnitAlias('minute', 'm');
|
|
|
|
// PARSING
|
|
|
|
addRegexToken('m', match1to2);
|
|
addRegexToken('mm', match1to2, match2);
|
|
addParseToken(['m', 'mm'], MINUTE);
|
|
|
|
// MOMENTS
|
|
|
|
var getSetMinute = makeGetSet('Minutes', false);
|
|
|
|
// FORMATTING
|
|
|
|
addFormatToken('s', ['ss', 2], 0, 'second');
|
|
|
|
// ALIASES
|
|
|
|
addUnitAlias('second', 's');
|
|
|
|
// PARSING
|
|
|
|
addRegexToken('s', match1to2);
|
|
addRegexToken('ss', match1to2, match2);
|
|
addParseToken(['s', 'ss'], SECOND);
|
|
|
|
// MOMENTS
|
|
|
|
var getSetSecond = makeGetSet('Seconds', false);
|
|
|
|
// FORMATTING
|
|
|
|
addFormatToken('S', 0, 0, function () {
|
|
return ~~(this.millisecond() / 100);
|
|
});
|
|
|
|
addFormatToken(0, ['SS', 2], 0, function () {
|
|
return ~~(this.millisecond() / 10);
|
|
});
|
|
|
|
addFormatToken(0, ['SSS', 3], 0, 'millisecond');
|
|
addFormatToken(0, ['SSSS', 4], 0, function () {
|
|
return this.millisecond() * 10;
|
|
});
|
|
addFormatToken(0, ['SSSSS', 5], 0, function () {
|
|
return this.millisecond() * 100;
|
|
});
|
|
addFormatToken(0, ['SSSSSS', 6], 0, function () {
|
|
return this.millisecond() * 1000;
|
|
});
|
|
addFormatToken(0, ['SSSSSSS', 7], 0, function () {
|
|
return this.millisecond() * 10000;
|
|
});
|
|
addFormatToken(0, ['SSSSSSSS', 8], 0, function () {
|
|
return this.millisecond() * 100000;
|
|
});
|
|
addFormatToken(0, ['SSSSSSSSS', 9], 0, function () {
|
|
return this.millisecond() * 1000000;
|
|
});
|
|
|
|
|
|
// ALIASES
|
|
|
|
addUnitAlias('millisecond', 'ms');
|
|
|
|
// PARSING
|
|
|
|
addRegexToken('S', match1to3, match1);
|
|
addRegexToken('SS', match1to3, match2);
|
|
addRegexToken('SSS', match1to3, match3);
|
|
|
|
var token;
|
|
for (token = 'SSSS'; token.length <= 9; token += 'S') {
|
|
addRegexToken(token, matchUnsigned);
|
|
}
|
|
|
|
function parseMs(input, array) {
|
|
array[MILLISECOND] = toInt(('0.' + input) * 1000);
|
|
}
|
|
|
|
for (token = 'S'; token.length <= 9; token += 'S') {
|
|
addParseToken(token, parseMs);
|
|
}
|
|
// MOMENTS
|
|
|
|
var getSetMillisecond = makeGetSet('Milliseconds', false);
|
|
|
|
// FORMATTING
|
|
|
|
addFormatToken('z', 0, 0, 'zoneAbbr');
|
|
addFormatToken('zz', 0, 0, 'zoneName');
|
|
|
|
// MOMENTS
|
|
|
|
function getZoneAbbr () {
|
|
return this._isUTC ? 'UTC' : '';
|
|
}
|
|
|
|
function getZoneName () {
|
|
return this._isUTC ? 'Coordinated Universal Time' : '';
|
|
}
|
|
|
|
var momentPrototype__proto = Moment.prototype;
|
|
|
|
momentPrototype__proto.add = add_subtract__add;
|
|
momentPrototype__proto.calendar = moment_calendar__calendar;
|
|
momentPrototype__proto.clone = clone;
|
|
momentPrototype__proto.diff = diff;
|
|
momentPrototype__proto.endOf = endOf;
|
|
momentPrototype__proto.format = format;
|
|
momentPrototype__proto.from = from;
|
|
momentPrototype__proto.fromNow = fromNow;
|
|
momentPrototype__proto.to = to;
|
|
momentPrototype__proto.toNow = toNow;
|
|
momentPrototype__proto.get = getSet;
|
|
momentPrototype__proto.invalidAt = invalidAt;
|
|
momentPrototype__proto.isAfter = isAfter;
|
|
momentPrototype__proto.isBefore = isBefore;
|
|
momentPrototype__proto.isBetween = isBetween;
|
|
momentPrototype__proto.isSame = isSame;
|
|
momentPrototype__proto.isSameOrAfter = isSameOrAfter;
|
|
momentPrototype__proto.isSameOrBefore = isSameOrBefore;
|
|
momentPrototype__proto.isValid = moment_valid__isValid;
|
|
momentPrototype__proto.lang = lang;
|
|
momentPrototype__proto.locale = locale;
|
|
momentPrototype__proto.localeData = localeData;
|
|
momentPrototype__proto.max = prototypeMax;
|
|
momentPrototype__proto.min = prototypeMin;
|
|
momentPrototype__proto.parsingFlags = parsingFlags;
|
|
momentPrototype__proto.set = getSet;
|
|
momentPrototype__proto.startOf = startOf;
|
|
momentPrototype__proto.subtract = add_subtract__subtract;
|
|
momentPrototype__proto.toArray = toArray;
|
|
momentPrototype__proto.toObject = toObject;
|
|
momentPrototype__proto.toDate = toDate;
|
|
momentPrototype__proto.toISOString = moment_format__toISOString;
|
|
momentPrototype__proto.toJSON = toJSON;
|
|
momentPrototype__proto.toString = toString;
|
|
momentPrototype__proto.unix = unix;
|
|
momentPrototype__proto.valueOf = to_type__valueOf;
|
|
momentPrototype__proto.creationData = creationData;
|
|
|
|
// Year
|
|
momentPrototype__proto.year = getSetYear;
|
|
momentPrototype__proto.isLeapYear = getIsLeapYear;
|
|
|
|
// Week Year
|
|
momentPrototype__proto.weekYear = getSetWeekYear;
|
|
momentPrototype__proto.isoWeekYear = getSetISOWeekYear;
|
|
|
|
// Quarter
|
|
momentPrototype__proto.quarter = momentPrototype__proto.quarters = getSetQuarter;
|
|
|
|
// Month
|
|
momentPrototype__proto.month = getSetMonth;
|
|
momentPrototype__proto.daysInMonth = getDaysInMonth;
|
|
|
|
// Week
|
|
momentPrototype__proto.week = momentPrototype__proto.weeks = getSetWeek;
|
|
momentPrototype__proto.isoWeek = momentPrototype__proto.isoWeeks = getSetISOWeek;
|
|
momentPrototype__proto.weeksInYear = getWeeksInYear;
|
|
momentPrototype__proto.isoWeeksInYear = getISOWeeksInYear;
|
|
|
|
// Day
|
|
momentPrototype__proto.date = getSetDayOfMonth;
|
|
momentPrototype__proto.day = momentPrototype__proto.days = getSetDayOfWeek;
|
|
momentPrototype__proto.weekday = getSetLocaleDayOfWeek;
|
|
momentPrototype__proto.isoWeekday = getSetISODayOfWeek;
|
|
momentPrototype__proto.dayOfYear = getSetDayOfYear;
|
|
|
|
// Hour
|
|
momentPrototype__proto.hour = momentPrototype__proto.hours = getSetHour;
|
|
|
|
// Minute
|
|
momentPrototype__proto.minute = momentPrototype__proto.minutes = getSetMinute;
|
|
|
|
// Second
|
|
momentPrototype__proto.second = momentPrototype__proto.seconds = getSetSecond;
|
|
|
|
// Millisecond
|
|
momentPrototype__proto.millisecond = momentPrototype__proto.milliseconds = getSetMillisecond;
|
|
|
|
// Offset
|
|
momentPrototype__proto.utcOffset = getSetOffset;
|
|
momentPrototype__proto.utc = setOffsetToUTC;
|
|
momentPrototype__proto.local = setOffsetToLocal;
|
|
momentPrototype__proto.parseZone = setOffsetToParsedOffset;
|
|
momentPrototype__proto.hasAlignedHourOffset = hasAlignedHourOffset;
|
|
momentPrototype__proto.isDST = isDaylightSavingTime;
|
|
momentPrototype__proto.isDSTShifted = isDaylightSavingTimeShifted;
|
|
momentPrototype__proto.isLocal = isLocal;
|
|
momentPrototype__proto.isUtcOffset = isUtcOffset;
|
|
momentPrototype__proto.isUtc = isUtc;
|
|
momentPrototype__proto.isUTC = isUtc;
|
|
|
|
// Timezone
|
|
momentPrototype__proto.zoneAbbr = getZoneAbbr;
|
|
momentPrototype__proto.zoneName = getZoneName;
|
|
|
|
// Deprecations
|
|
momentPrototype__proto.dates = deprecate('dates accessor is deprecated. Use date instead.', getSetDayOfMonth);
|
|
momentPrototype__proto.months = deprecate('months accessor is deprecated. Use month instead', getSetMonth);
|
|
momentPrototype__proto.years = deprecate('years accessor is deprecated. Use year instead', getSetYear);
|
|
momentPrototype__proto.zone = deprecate('moment().zone is deprecated, use moment().utcOffset instead. https://github.com/moment/moment/issues/1779', getSetZone);
|
|
|
|
var momentPrototype = momentPrototype__proto;
|
|
|
|
function moment__createUnix (input) {
|
|
return local__createLocal(input * 1000);
|
|
}
|
|
|
|
function moment__createInZone () {
|
|
return local__createLocal.apply(null, arguments).parseZone();
|
|
}
|
|
|
|
var defaultCalendar = {
|
|
sameDay : '[Today at] LT',
|
|
nextDay : '[Tomorrow at] LT',
|
|
nextWeek : 'dddd [at] LT',
|
|
lastDay : '[Yesterday at] LT',
|
|
lastWeek : '[Last] dddd [at] LT',
|
|
sameElse : 'L'
|
|
};
|
|
|
|
function locale_calendar__calendar (key, mom, now) {
|
|
var output = this._calendar[key];
|
|
return isFunction(output) ? output.call(mom, now) : output;
|
|
}
|
|
|
|
var defaultLongDateFormat = {
|
|
LTS : 'h:mm:ss A',
|
|
LT : 'h:mm A',
|
|
L : 'MM/DD/YYYY',
|
|
LL : 'MMMM D, YYYY',
|
|
LLL : 'MMMM D, YYYY h:mm A',
|
|
LLLL : 'dddd, MMMM D, YYYY h:mm A'
|
|
};
|
|
|
|
function longDateFormat (key) {
|
|
var format = this._longDateFormat[key],
|
|
formatUpper = this._longDateFormat[key.toUpperCase()];
|
|
|
|
if (format || !formatUpper) {
|
|
return format;
|
|
}
|
|
|
|
this._longDateFormat[key] = formatUpper.replace(/MMMM|MM|DD|dddd/g, function (val) {
|
|
return val.slice(1);
|
|
});
|
|
|
|
return this._longDateFormat[key];
|
|
}
|
|
|
|
var defaultInvalidDate = 'Invalid date';
|
|
|
|
function invalidDate () {
|
|
return this._invalidDate;
|
|
}
|
|
|
|
var defaultOrdinal = '%d';
|
|
var defaultOrdinalParse = /\d{1,2}/;
|
|
|
|
function ordinal (number) {
|
|
return this._ordinal.replace('%d', number);
|
|
}
|
|
|
|
function preParsePostFormat (string) {
|
|
return string;
|
|
}
|
|
|
|
var defaultRelativeTime = {
|
|
future : 'in %s',
|
|
past : '%s ago',
|
|
s : 'a few seconds',
|
|
m : 'a minute',
|
|
mm : '%d minutes',
|
|
h : 'an hour',
|
|
hh : '%d hours',
|
|
d : 'a day',
|
|
dd : '%d days',
|
|
M : 'a month',
|
|
MM : '%d months',
|
|
y : 'a year',
|
|
yy : '%d years'
|
|
};
|
|
|
|
function relative__relativeTime (number, withoutSuffix, string, isFuture) {
|
|
var output = this._relativeTime[string];
|
|
return (isFunction(output)) ?
|
|
output(number, withoutSuffix, string, isFuture) :
|
|
output.replace(/%d/i, number);
|
|
}
|
|
|
|
function pastFuture (diff, output) {
|
|
var format = this._relativeTime[diff > 0 ? 'future' : 'past'];
|
|
return isFunction(format) ? format(output) : format.replace(/%s/i, output);
|
|
}
|
|
|
|
var prototype__proto = Locale.prototype;
|
|
|
|
prototype__proto._calendar = defaultCalendar;
|
|
prototype__proto.calendar = locale_calendar__calendar;
|
|
prototype__proto._longDateFormat = defaultLongDateFormat;
|
|
prototype__proto.longDateFormat = longDateFormat;
|
|
prototype__proto._invalidDate = defaultInvalidDate;
|
|
prototype__proto.invalidDate = invalidDate;
|
|
prototype__proto._ordinal = defaultOrdinal;
|
|
prototype__proto.ordinal = ordinal;
|
|
prototype__proto._ordinalParse = defaultOrdinalParse;
|
|
prototype__proto.preparse = preParsePostFormat;
|
|
prototype__proto.postformat = preParsePostFormat;
|
|
prototype__proto._relativeTime = defaultRelativeTime;
|
|
prototype__proto.relativeTime = relative__relativeTime;
|
|
prototype__proto.pastFuture = pastFuture;
|
|
prototype__proto.set = locale_set__set;
|
|
|
|
// Month
|
|
prototype__proto.months = localeMonths;
|
|
prototype__proto._months = defaultLocaleMonths;
|
|
prototype__proto.monthsShort = localeMonthsShort;
|
|
prototype__proto._monthsShort = defaultLocaleMonthsShort;
|
|
prototype__proto.monthsParse = localeMonthsParse;
|
|
prototype__proto._monthsRegex = defaultMonthsRegex;
|
|
prototype__proto.monthsRegex = monthsRegex;
|
|
prototype__proto._monthsShortRegex = defaultMonthsShortRegex;
|
|
prototype__proto.monthsShortRegex = monthsShortRegex;
|
|
|
|
// Week
|
|
prototype__proto.week = localeWeek;
|
|
prototype__proto._week = defaultLocaleWeek;
|
|
prototype__proto.firstDayOfYear = localeFirstDayOfYear;
|
|
prototype__proto.firstDayOfWeek = localeFirstDayOfWeek;
|
|
|
|
// Day of Week
|
|
prototype__proto.weekdays = localeWeekdays;
|
|
prototype__proto._weekdays = defaultLocaleWeekdays;
|
|
prototype__proto.weekdaysMin = localeWeekdaysMin;
|
|
prototype__proto._weekdaysMin = defaultLocaleWeekdaysMin;
|
|
prototype__proto.weekdaysShort = localeWeekdaysShort;
|
|
prototype__proto._weekdaysShort = defaultLocaleWeekdaysShort;
|
|
prototype__proto.weekdaysParse = localeWeekdaysParse;
|
|
|
|
// Hours
|
|
prototype__proto.isPM = localeIsPM;
|
|
prototype__proto._meridiemParse = defaultLocaleMeridiemParse;
|
|
prototype__proto.meridiem = localeMeridiem;
|
|
|
|
function lists__get (format, index, field, setter) {
|
|
var locale = locale_locales__getLocale();
|
|
var utc = create_utc__createUTC().set(setter, index);
|
|
return locale[field](utc, format);
|
|
}
|
|
|
|
function list (format, index, field, count, setter) {
|
|
if (typeof format === 'number') {
|
|
index = format;
|
|
format = undefined;
|
|
}
|
|
|
|
format = format || '';
|
|
|
|
if (index != null) {
|
|
return lists__get(format, index, field, setter);
|
|
}
|
|
|
|
var i;
|
|
var out = [];
|
|
for (i = 0; i < count; i++) {
|
|
out[i] = lists__get(format, i, field, setter);
|
|
}
|
|
return out;
|
|
}
|
|
|
|
function lists__listMonths (format, index) {
|
|
return list(format, index, 'months', 12, 'month');
|
|
}
|
|
|
|
function lists__listMonthsShort (format, index) {
|
|
return list(format, index, 'monthsShort', 12, 'month');
|
|
}
|
|
|
|
function lists__listWeekdays (format, index) {
|
|
return list(format, index, 'weekdays', 7, 'day');
|
|
}
|
|
|
|
function lists__listWeekdaysShort (format, index) {
|
|
return list(format, index, 'weekdaysShort', 7, 'day');
|
|
}
|
|
|
|
function lists__listWeekdaysMin (format, index) {
|
|
return list(format, index, 'weekdaysMin', 7, 'day');
|
|
}
|
|
|
|
locale_locales__getSetGlobalLocale('en', {
|
|
ordinalParse: /\d{1,2}(th|st|nd|rd)/,
|
|
ordinal : function (number) {
|
|
var b = number % 10,
|
|
output = (toInt(number % 100 / 10) === 1) ? 'th' :
|
|
(b === 1) ? 'st' :
|
|
(b === 2) ? 'nd' :
|
|
(b === 3) ? 'rd' : 'th';
|
|
return number + output;
|
|
}
|
|
});
|
|
|
|
// Side effect imports
|
|
utils_hooks__hooks.lang = deprecate('moment.lang is deprecated. Use moment.locale instead.', locale_locales__getSetGlobalLocale);
|
|
utils_hooks__hooks.langData = deprecate('moment.langData is deprecated. Use moment.localeData instead.', locale_locales__getLocale);
|
|
|
|
var mathAbs = Math.abs;
|
|
|
|
function duration_abs__abs () {
|
|
var data = this._data;
|
|
|
|
this._milliseconds = mathAbs(this._milliseconds);
|
|
this._days = mathAbs(this._days);
|
|
this._months = mathAbs(this._months);
|
|
|
|
data.milliseconds = mathAbs(data.milliseconds);
|
|
data.seconds = mathAbs(data.seconds);
|
|
data.minutes = mathAbs(data.minutes);
|
|
data.hours = mathAbs(data.hours);
|
|
data.months = mathAbs(data.months);
|
|
data.years = mathAbs(data.years);
|
|
|
|
return this;
|
|
}
|
|
|
|
function duration_add_subtract__addSubtract (duration, input, value, direction) {
|
|
var other = create__createDuration(input, value);
|
|
|
|
duration._milliseconds += direction * other._milliseconds;
|
|
duration._days += direction * other._days;
|
|
duration._months += direction * other._months;
|
|
|
|
return duration._bubble();
|
|
}
|
|
|
|
// supports only 2.0-style add(1, 's') or add(duration)
|
|
function duration_add_subtract__add (input, value) {
|
|
return duration_add_subtract__addSubtract(this, input, value, 1);
|
|
}
|
|
|
|
// supports only 2.0-style subtract(1, 's') or subtract(duration)
|
|
function duration_add_subtract__subtract (input, value) {
|
|
return duration_add_subtract__addSubtract(this, input, value, -1);
|
|
}
|
|
|
|
function absCeil (number) {
|
|
if (number < 0) {
|
|
return Math.floor(number);
|
|
} else {
|
|
return Math.ceil(number);
|
|
}
|
|
}
|
|
|
|
function bubble () {
|
|
var milliseconds = this._milliseconds;
|
|
var days = this._days;
|
|
var months = this._months;
|
|
var data = this._data;
|
|
var seconds, minutes, hours, years, monthsFromDays;
|
|
|
|
// if we have a mix of positive and negative values, bubble down first
|
|
// check: https://github.com/moment/moment/issues/2166
|
|
if (!((milliseconds >= 0 && days >= 0 && months >= 0) ||
|
|
(milliseconds <= 0 && days <= 0 && months <= 0))) {
|
|
milliseconds += absCeil(monthsToDays(months) + days) * 864e5;
|
|
days = 0;
|
|
months = 0;
|
|
}
|
|
|
|
// The following code bubbles up values, see the tests for
|
|
// examples of what that means.
|
|
data.milliseconds = milliseconds % 1000;
|
|
|
|
seconds = absFloor(milliseconds / 1000);
|
|
data.seconds = seconds % 60;
|
|
|
|
minutes = absFloor(seconds / 60);
|
|
data.minutes = minutes % 60;
|
|
|
|
hours = absFloor(minutes / 60);
|
|
data.hours = hours % 24;
|
|
|
|
days += absFloor(hours / 24);
|
|
|
|
// convert days to months
|
|
monthsFromDays = absFloor(daysToMonths(days));
|
|
months += monthsFromDays;
|
|
days -= absCeil(monthsToDays(monthsFromDays));
|
|
|
|
// 12 months -> 1 year
|
|
years = absFloor(months / 12);
|
|
months %= 12;
|
|
|
|
data.days = days;
|
|
data.months = months;
|
|
data.years = years;
|
|
|
|
return this;
|
|
}
|
|
|
|
function daysToMonths (days) {
|
|
// 400 years have 146097 days (taking into account leap year rules)
|
|
// 400 years have 12 months === 4800
|
|
return days * 4800 / 146097;
|
|
}
|
|
|
|
function monthsToDays (months) {
|
|
// the reverse of daysToMonths
|
|
return months * 146097 / 4800;
|
|
}
|
|
|
|
function as (units) {
|
|
var days;
|
|
var months;
|
|
var milliseconds = this._milliseconds;
|
|
|
|
units = normalizeUnits(units);
|
|
|
|
if (units === 'month' || units === 'year') {
|
|
days = this._days + milliseconds / 864e5;
|
|
months = this._months + daysToMonths(days);
|
|
return units === 'month' ? months : months / 12;
|
|
} else {
|
|
// handle milliseconds separately because of floating point math errors (issue #1867)
|
|
days = this._days + Math.round(monthsToDays(this._months));
|
|
switch (units) {
|
|
case 'week' : return days / 7 + milliseconds / 6048e5;
|
|
case 'day' : return days + milliseconds / 864e5;
|
|
case 'hour' : return days * 24 + milliseconds / 36e5;
|
|
case 'minute' : return days * 1440 + milliseconds / 6e4;
|
|
case 'second' : return days * 86400 + milliseconds / 1000;
|
|
// Math.floor prevents floating point math errors here
|
|
case 'millisecond': return Math.floor(days * 864e5) + milliseconds;
|
|
default: throw new Error('Unknown unit ' + units);
|
|
}
|
|
}
|
|
}
|
|
|
|
// TODO: Use this.as('ms')?
|
|
function duration_as__valueOf () {
|
|
return (
|
|
this._milliseconds +
|
|
this._days * 864e5 +
|
|
(this._months % 12) * 2592e6 +
|
|
toInt(this._months / 12) * 31536e6
|
|
);
|
|
}
|
|
|
|
function makeAs (alias) {
|
|
return function () {
|
|
return this.as(alias);
|
|
};
|
|
}
|
|
|
|
var asMilliseconds = makeAs('ms');
|
|
var asSeconds = makeAs('s');
|
|
var asMinutes = makeAs('m');
|
|
var asHours = makeAs('h');
|
|
var asDays = makeAs('d');
|
|
var asWeeks = makeAs('w');
|
|
var asMonths = makeAs('M');
|
|
var asYears = makeAs('y');
|
|
|
|
function duration_get__get (units) {
|
|
units = normalizeUnits(units);
|
|
return this[units + 's']();
|
|
}
|
|
|
|
function makeGetter(name) {
|
|
return function () {
|
|
return this._data[name];
|
|
};
|
|
}
|
|
|
|
var milliseconds = makeGetter('milliseconds');
|
|
var seconds = makeGetter('seconds');
|
|
var minutes = makeGetter('minutes');
|
|
var hours = makeGetter('hours');
|
|
var days = makeGetter('days');
|
|
var months = makeGetter('months');
|
|
var years = makeGetter('years');
|
|
|
|
function weeks () {
|
|
return absFloor(this.days() / 7);
|
|
}
|
|
|
|
var round = Math.round;
|
|
var thresholds = {
|
|
s: 45, // seconds to minute
|
|
m: 45, // minutes to hour
|
|
h: 22, // hours to day
|
|
d: 26, // days to month
|
|
M: 11 // months to year
|
|
};
|
|
|
|
// helper function for moment.fn.from, moment.fn.fromNow, and moment.duration.fn.humanize
|
|
function substituteTimeAgo(string, number, withoutSuffix, isFuture, locale) {
|
|
return locale.relativeTime(number || 1, !!withoutSuffix, string, isFuture);
|
|
}
|
|
|
|
function duration_humanize__relativeTime (posNegDuration, withoutSuffix, locale) {
|
|
var duration = create__createDuration(posNegDuration).abs();
|
|
var seconds = round(duration.as('s'));
|
|
var minutes = round(duration.as('m'));
|
|
var hours = round(duration.as('h'));
|
|
var days = round(duration.as('d'));
|
|
var months = round(duration.as('M'));
|
|
var years = round(duration.as('y'));
|
|
|
|
var a = seconds < thresholds.s && ['s', seconds] ||
|
|
minutes <= 1 && ['m'] ||
|
|
minutes < thresholds.m && ['mm', minutes] ||
|
|
hours <= 1 && ['h'] ||
|
|
hours < thresholds.h && ['hh', hours] ||
|
|
days <= 1 && ['d'] ||
|
|
days < thresholds.d && ['dd', days] ||
|
|
months <= 1 && ['M'] ||
|
|
months < thresholds.M && ['MM', months] ||
|
|
years <= 1 && ['y'] || ['yy', years];
|
|
|
|
a[2] = withoutSuffix;
|
|
a[3] = +posNegDuration > 0;
|
|
a[4] = locale;
|
|
return substituteTimeAgo.apply(null, a);
|
|
}
|
|
|
|
// This function allows you to set a threshold for relative time strings
|
|
function duration_humanize__getSetRelativeTimeThreshold (threshold, limit) {
|
|
if (thresholds[threshold] === undefined) {
|
|
return false;
|
|
}
|
|
if (limit === undefined) {
|
|
return thresholds[threshold];
|
|
}
|
|
thresholds[threshold] = limit;
|
|
return true;
|
|
}
|
|
|
|
function humanize (withSuffix) {
|
|
var locale = this.localeData();
|
|
var output = duration_humanize__relativeTime(this, !withSuffix, locale);
|
|
|
|
if (withSuffix) {
|
|
output = locale.pastFuture(+this, output);
|
|
}
|
|
|
|
return locale.postformat(output);
|
|
}
|
|
|
|
var iso_string__abs = Math.abs;
|
|
|
|
function iso_string__toISOString() {
|
|
// for ISO strings we do not use the normal bubbling rules:
|
|
// * milliseconds bubble up until they become hours
|
|
// * days do not bubble at all
|
|
// * months bubble up until they become years
|
|
// This is because there is no context-free conversion between hours and days
|
|
// (think of clock changes)
|
|
// and also not between days and months (28-31 days per month)
|
|
var seconds = iso_string__abs(this._milliseconds) / 1000;
|
|
var days = iso_string__abs(this._days);
|
|
var months = iso_string__abs(this._months);
|
|
var minutes, hours, years;
|
|
|
|
// 3600 seconds -> 60 minutes -> 1 hour
|
|
minutes = absFloor(seconds / 60);
|
|
hours = absFloor(minutes / 60);
|
|
seconds %= 60;
|
|
minutes %= 60;
|
|
|
|
// 12 months -> 1 year
|
|
years = absFloor(months / 12);
|
|
months %= 12;
|
|
|
|
|
|
// inspired by https://github.com/dordille/moment-isoduration/blob/master/moment.isoduration.js
|
|
var Y = years;
|
|
var M = months;
|
|
var D = days;
|
|
var h = hours;
|
|
var m = minutes;
|
|
var s = seconds;
|
|
var total = this.asSeconds();
|
|
|
|
if (!total) {
|
|
// this is the same as C#'s (Noda) and python (isodate)...
|
|
// but not other JS (goog.date)
|
|
return 'P0D';
|
|
}
|
|
|
|
return (total < 0 ? '-' : '') +
|
|
'P' +
|
|
(Y ? Y + 'Y' : '') +
|
|
(M ? M + 'M' : '') +
|
|
(D ? D + 'D' : '') +
|
|
((h || m || s) ? 'T' : '') +
|
|
(h ? h + 'H' : '') +
|
|
(m ? m + 'M' : '') +
|
|
(s ? s + 'S' : '');
|
|
}
|
|
|
|
var duration_prototype__proto = Duration.prototype;
|
|
|
|
duration_prototype__proto.abs = duration_abs__abs;
|
|
duration_prototype__proto.add = duration_add_subtract__add;
|
|
duration_prototype__proto.subtract = duration_add_subtract__subtract;
|
|
duration_prototype__proto.as = as;
|
|
duration_prototype__proto.asMilliseconds = asMilliseconds;
|
|
duration_prototype__proto.asSeconds = asSeconds;
|
|
duration_prototype__proto.asMinutes = asMinutes;
|
|
duration_prototype__proto.asHours = asHours;
|
|
duration_prototype__proto.asDays = asDays;
|
|
duration_prototype__proto.asWeeks = asWeeks;
|
|
duration_prototype__proto.asMonths = asMonths;
|
|
duration_prototype__proto.asYears = asYears;
|
|
duration_prototype__proto.valueOf = duration_as__valueOf;
|
|
duration_prototype__proto._bubble = bubble;
|
|
duration_prototype__proto.get = duration_get__get;
|
|
duration_prototype__proto.milliseconds = milliseconds;
|
|
duration_prototype__proto.seconds = seconds;
|
|
duration_prototype__proto.minutes = minutes;
|
|
duration_prototype__proto.hours = hours;
|
|
duration_prototype__proto.days = days;
|
|
duration_prototype__proto.weeks = weeks;
|
|
duration_prototype__proto.months = months;
|
|
duration_prototype__proto.years = years;
|
|
duration_prototype__proto.humanize = humanize;
|
|
duration_prototype__proto.toISOString = iso_string__toISOString;
|
|
duration_prototype__proto.toString = iso_string__toISOString;
|
|
duration_prototype__proto.toJSON = iso_string__toISOString;
|
|
duration_prototype__proto.locale = locale;
|
|
duration_prototype__proto.localeData = localeData;
|
|
|
|
// Deprecations
|
|
duration_prototype__proto.toIsoString = deprecate('toIsoString() is deprecated. Please use toISOString() instead (notice the capitals)', iso_string__toISOString);
|
|
duration_prototype__proto.lang = lang;
|
|
|
|
// Side effect imports
|
|
|
|
// FORMATTING
|
|
|
|
addFormatToken('X', 0, 0, 'unix');
|
|
addFormatToken('x', 0, 0, 'valueOf');
|
|
|
|
// PARSING
|
|
|
|
addRegexToken('x', matchSigned);
|
|
addRegexToken('X', matchTimestamp);
|
|
addParseToken('X', function (input, array, config) {
|
|
config._d = new Date(parseFloat(input, 10) * 1000);
|
|
});
|
|
addParseToken('x', function (input, array, config) {
|
|
config._d = new Date(toInt(input));
|
|
});
|
|
|
|
// Side effect imports
|
|
|
|
|
|
utils_hooks__hooks.version = '2.12.0';
|
|
|
|
setHookCallback(local__createLocal);
|
|
|
|
utils_hooks__hooks.fn = momentPrototype;
|
|
utils_hooks__hooks.min = min;
|
|
utils_hooks__hooks.max = max;
|
|
utils_hooks__hooks.now = now;
|
|
utils_hooks__hooks.utc = create_utc__createUTC;
|
|
utils_hooks__hooks.unix = moment__createUnix;
|
|
utils_hooks__hooks.months = lists__listMonths;
|
|
utils_hooks__hooks.isDate = isDate;
|
|
utils_hooks__hooks.locale = locale_locales__getSetGlobalLocale;
|
|
utils_hooks__hooks.invalid = valid__createInvalid;
|
|
utils_hooks__hooks.duration = create__createDuration;
|
|
utils_hooks__hooks.isMoment = isMoment;
|
|
utils_hooks__hooks.weekdays = lists__listWeekdays;
|
|
utils_hooks__hooks.parseZone = moment__createInZone;
|
|
utils_hooks__hooks.localeData = locale_locales__getLocale;
|
|
utils_hooks__hooks.isDuration = isDuration;
|
|
utils_hooks__hooks.monthsShort = lists__listMonthsShort;
|
|
utils_hooks__hooks.weekdaysMin = lists__listWeekdaysMin;
|
|
utils_hooks__hooks.defineLocale = defineLocale;
|
|
utils_hooks__hooks.updateLocale = updateLocale;
|
|
utils_hooks__hooks.locales = locale_locales__listLocales;
|
|
utils_hooks__hooks.weekdaysShort = lists__listWeekdaysShort;
|
|
utils_hooks__hooks.normalizeUnits = normalizeUnits;
|
|
utils_hooks__hooks.relativeTimeThreshold = duration_humanize__getSetRelativeTimeThreshold;
|
|
utils_hooks__hooks.prototype = momentPrototype;
|
|
|
|
var _moment = utils_hooks__hooks;
|
|
|
|
return _moment;
|
|
|
|
}));
|
|
/**
|
|
* @version: 2.1.19
|
|
* @author: Dan Grossman http://www.dangrossman.info/
|
|
* @copyright: Copyright (c) 2012-2015 Dan Grossman. All rights reserved.
|
|
* @license: Licensed under the MIT license. See http://www.opensource.org/licenses/mit-license.php
|
|
* @website: https://www.improvely.com/
|
|
*/
|
|
|
|
(function(root, factory) {
|
|
|
|
if (typeof define === 'function' && define.amd) {
|
|
define(['moment', 'jquery', 'exports'], function(momentjs, $, exports) {
|
|
root.daterangepicker = factory(root, exports, momentjs, $);
|
|
});
|
|
|
|
} else if (typeof exports !== 'undefined') {
|
|
var momentjs = require('moment');
|
|
var jQuery = (typeof window != 'undefined') ? window.jQuery : undefined; //isomorphic issue
|
|
if (!jQuery) {
|
|
try {
|
|
jQuery = require('jquery');
|
|
if (!jQuery.fn) jQuery.fn = {}; //isomorphic issue
|
|
} catch (err) {
|
|
if (!jQuery) throw new Error('jQuery dependency not found');
|
|
}
|
|
}
|
|
|
|
factory(root, exports, momentjs, jQuery);
|
|
|
|
// Finally, as a browser global.
|
|
} else {
|
|
root.daterangepicker = factory(root, {}, root.moment || moment, (root.jQuery || root.Zepto || root.ender || root.$));
|
|
}
|
|
|
|
}(this || {}, function(root, daterangepicker, moment, $) { // 'this' doesn't exist on a server
|
|
|
|
var DateRangePicker = function(element, options, cb) {
|
|
|
|
//default settings for options
|
|
this.parentEl = 'body';
|
|
this.element = $(element);
|
|
this.startDate = moment().startOf('day');
|
|
this.endDate = moment().endOf('day');
|
|
this.minDate = false;
|
|
this.maxDate = false;
|
|
this.dateLimit = false;
|
|
this.autoApply = false;
|
|
this.singleDatePicker = false;
|
|
this.showDropdowns = false;
|
|
this.showWeekNumbers = false;
|
|
this.showISOWeekNumbers = false;
|
|
this.timePicker = false;
|
|
this.timePicker24Hour = false;
|
|
this.timePickerIncrement = 1;
|
|
this.timePickerSeconds = false;
|
|
this.linkedCalendars = true;
|
|
this.autoUpdateInput = true;
|
|
this.alwaysShowCalendars = false;
|
|
this.ranges = {};
|
|
|
|
this.opens = 'right';
|
|
if (this.element.hasClass('pull-right'))
|
|
this.opens = 'left';
|
|
|
|
this.drops = 'down';
|
|
if (this.element.hasClass('dropup'))
|
|
this.drops = 'up';
|
|
|
|
this.buttonClasses = 'btn btn-sm';
|
|
this.applyClass = 'btn-success';
|
|
this.cancelClass = 'btn-default';
|
|
|
|
this.locale = {
|
|
format: 'MM/DD/YYYY',
|
|
separator: ' - ',
|
|
applyLabel: 'Apply',
|
|
cancelLabel: 'Cancel',
|
|
weekLabel: 'W',
|
|
customRangeLabel: 'Custom Range',
|
|
daysOfWeek: moment.weekdaysMin(),
|
|
monthNames: moment.monthsShort(),
|
|
firstDay: moment.localeData().firstDayOfWeek()
|
|
};
|
|
|
|
this.callback = function() { };
|
|
|
|
//some state information
|
|
this.isShowing = false;
|
|
this.leftCalendar = {};
|
|
this.rightCalendar = {};
|
|
|
|
//custom options from user
|
|
if (typeof options !== 'object' || options === null)
|
|
options = {};
|
|
|
|
//allow setting options with data attributes
|
|
//data-api options will be overwritten with custom javascript options
|
|
options = $.extend(this.element.data(), options);
|
|
|
|
//html template for the picker UI
|
|
if (typeof options.template !== 'string' && !(options.template instanceof $))
|
|
options.template = '<div class="daterangepicker dropdown-menu">' +
|
|
'<div class="calendar left">' +
|
|
'<div class="daterangepicker_input">' +
|
|
'<input class="input-mini" type="text" name="daterangepicker_start" value="" />' +
|
|
'<i class="fa fa-calendar glyphicon glyphicon-calendar"></i>' +
|
|
'<div class="calendar-time">' +
|
|
'<div></div>' +
|
|
'<i class="fa fa-clock-o glyphicon glyphicon-time"></i>' +
|
|
'</div>' +
|
|
'</div>' +
|
|
'<div class="calendar-table"></div>' +
|
|
'</div>' +
|
|
'<div class="calendar right">' +
|
|
'<div class="daterangepicker_input">' +
|
|
'<input class="input-mini" type="text" name="daterangepicker_end" value="" />' +
|
|
'<i class="fa fa-calendar glyphicon glyphicon-calendar"></i>' +
|
|
'<div class="calendar-time">' +
|
|
'<div></div>' +
|
|
'<i class="fa fa-clock-o glyphicon glyphicon-time"></i>' +
|
|
'</div>' +
|
|
'</div>' +
|
|
'<div class="calendar-table"></div>' +
|
|
'</div>' +
|
|
'<div class="ranges">' +
|
|
'<div class="range_inputs">' +
|
|
'<button class="applyBtn" disabled="disabled" type="button"></button> ' +
|
|
'<button class="cancelBtn" type="button"></button>' +
|
|
'</div>' +
|
|
'</div>' +
|
|
'</div>';
|
|
|
|
this.parentEl = (options.parentEl && $(options.parentEl).length) ? $(options.parentEl) : $(this.parentEl);
|
|
this.container = $(options.template).appendTo(this.parentEl);
|
|
|
|
//
|
|
// handle all the possible options overriding defaults
|
|
//
|
|
|
|
if (typeof options.locale === 'object') {
|
|
|
|
if (typeof options.locale.format === 'string')
|
|
this.locale.format = options.locale.format;
|
|
|
|
if (typeof options.locale.separator === 'string')
|
|
this.locale.separator = options.locale.separator;
|
|
|
|
if (typeof options.locale.daysOfWeek === 'object')
|
|
this.locale.daysOfWeek = options.locale.daysOfWeek.slice();
|
|
|
|
if (typeof options.locale.monthNames === 'object')
|
|
this.locale.monthNames = options.locale.monthNames.slice();
|
|
|
|
if (typeof options.locale.firstDay === 'number')
|
|
this.locale.firstDay = options.locale.firstDay;
|
|
|
|
if (typeof options.locale.applyLabel === 'string')
|
|
this.locale.applyLabel = options.locale.applyLabel;
|
|
|
|
if (typeof options.locale.cancelLabel === 'string')
|
|
this.locale.cancelLabel = options.locale.cancelLabel;
|
|
|
|
if (typeof options.locale.weekLabel === 'string')
|
|
this.locale.weekLabel = options.locale.weekLabel;
|
|
|
|
if (typeof options.locale.customRangeLabel === 'string')
|
|
this.locale.customRangeLabel = options.locale.customRangeLabel;
|
|
|
|
}
|
|
|
|
if (typeof options.startDate === 'string')
|
|
this.startDate = moment(options.startDate, this.locale.format);
|
|
|
|
if (typeof options.endDate === 'string')
|
|
this.endDate = moment(options.endDate, this.locale.format);
|
|
|
|
if (typeof options.minDate === 'string')
|
|
this.minDate = moment(options.minDate, this.locale.format);
|
|
|
|
if (typeof options.maxDate === 'string')
|
|
this.maxDate = moment(options.maxDate, this.locale.format);
|
|
|
|
if (typeof options.startDate === 'object')
|
|
this.startDate = moment(options.startDate);
|
|
|
|
if (typeof options.endDate === 'object')
|
|
this.endDate = moment(options.endDate);
|
|
|
|
if (typeof options.minDate === 'object')
|
|
this.minDate = moment(options.minDate);
|
|
|
|
if (typeof options.maxDate === 'object')
|
|
this.maxDate = moment(options.maxDate);
|
|
|
|
// sanity check for bad options
|
|
if (this.minDate && this.startDate.isBefore(this.minDate))
|
|
this.startDate = this.minDate.clone();
|
|
|
|
// sanity check for bad options
|
|
if (this.maxDate && this.endDate.isAfter(this.maxDate))
|
|
this.endDate = this.maxDate.clone();
|
|
|
|
if (typeof options.applyClass === 'string')
|
|
this.applyClass = options.applyClass;
|
|
|
|
if (typeof options.cancelClass === 'string')
|
|
this.cancelClass = options.cancelClass;
|
|
|
|
if (typeof options.dateLimit === 'object')
|
|
this.dateLimit = options.dateLimit;
|
|
|
|
if (typeof options.opens === 'string')
|
|
this.opens = options.opens;
|
|
|
|
if (typeof options.drops === 'string')
|
|
this.drops = options.drops;
|
|
|
|
if (typeof options.showWeekNumbers === 'boolean')
|
|
this.showWeekNumbers = options.showWeekNumbers;
|
|
|
|
if (typeof options.showISOWeekNumbers === 'boolean')
|
|
this.showISOWeekNumbers = options.showISOWeekNumbers;
|
|
|
|
if (typeof options.buttonClasses === 'string')
|
|
this.buttonClasses = options.buttonClasses;
|
|
|
|
if (typeof options.buttonClasses === 'object')
|
|
this.buttonClasses = options.buttonClasses.join(' ');
|
|
|
|
if (typeof options.showDropdowns === 'boolean')
|
|
this.showDropdowns = options.showDropdowns;
|
|
|
|
if (typeof options.singleDatePicker === 'boolean') {
|
|
this.singleDatePicker = options.singleDatePicker;
|
|
if (this.singleDatePicker)
|
|
this.endDate = this.startDate.clone();
|
|
}
|
|
|
|
if (typeof options.timePicker === 'boolean')
|
|
this.timePicker = options.timePicker;
|
|
|
|
if (typeof options.timePickerSeconds === 'boolean')
|
|
this.timePickerSeconds = options.timePickerSeconds;
|
|
|
|
if (typeof options.timePickerIncrement === 'number')
|
|
this.timePickerIncrement = options.timePickerIncrement;
|
|
|
|
if (typeof options.timePicker24Hour === 'boolean')
|
|
this.timePicker24Hour = options.timePicker24Hour;
|
|
|
|
if (typeof options.autoApply === 'boolean')
|
|
this.autoApply = options.autoApply;
|
|
|
|
if (typeof options.autoUpdateInput === 'boolean')
|
|
this.autoUpdateInput = options.autoUpdateInput;
|
|
|
|
if (typeof options.linkedCalendars === 'boolean')
|
|
this.linkedCalendars = options.linkedCalendars;
|
|
|
|
if (typeof options.isInvalidDate === 'function')
|
|
this.isInvalidDate = options.isInvalidDate;
|
|
|
|
if (typeof options.alwaysShowCalendars === 'boolean')
|
|
this.alwaysShowCalendars = options.alwaysShowCalendars;
|
|
|
|
// update day names order to firstDay
|
|
if (this.locale.firstDay != 0) {
|
|
var iterator = this.locale.firstDay;
|
|
while (iterator > 0) {
|
|
this.locale.daysOfWeek.push(this.locale.daysOfWeek.shift());
|
|
iterator--;
|
|
}
|
|
}
|
|
|
|
var start, end, range;
|
|
|
|
//if no start/end dates set, check if an input element contains initial values
|
|
if (typeof options.startDate === 'undefined' && typeof options.endDate === 'undefined') {
|
|
if ($(this.element).is('input[type=text]')) {
|
|
var val = $(this.element).val(),
|
|
split = val.split(this.locale.separator);
|
|
|
|
start = end = null;
|
|
|
|
if (split.length == 2) {
|
|
start = moment(split[0], this.locale.format);
|
|
end = moment(split[1], this.locale.format);
|
|
} else if (this.singleDatePicker && val !== "") {
|
|
start = moment(val, this.locale.format);
|
|
end = moment(val, this.locale.format);
|
|
}
|
|
if (start !== null && end !== null) {
|
|
this.setStartDate(start);
|
|
this.setEndDate(end);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (typeof options.ranges === 'object') {
|
|
for (range in options.ranges) {
|
|
|
|
if (typeof options.ranges[range][0] === 'string')
|
|
start = moment(options.ranges[range][0], this.locale.format);
|
|
else
|
|
start = moment(options.ranges[range][0]);
|
|
|
|
if (typeof options.ranges[range][1] === 'string')
|
|
end = moment(options.ranges[range][1], this.locale.format);
|
|
else
|
|
end = moment(options.ranges[range][1]);
|
|
|
|
// If the start or end date exceed those allowed by the minDate or dateLimit
|
|
// options, shorten the range to the allowable period.
|
|
if (this.minDate && start.isBefore(this.minDate))
|
|
start = this.minDate.clone();
|
|
|
|
var maxDate = this.maxDate;
|
|
if (this.dateLimit && start.clone().add(this.dateLimit).isAfter(maxDate))
|
|
maxDate = start.clone().add(this.dateLimit);
|
|
if (maxDate && end.isAfter(maxDate))
|
|
end = maxDate.clone();
|
|
|
|
// If the end of the range is before the minimum or the start of the range is
|
|
// after the maximum, don't display this range option at all.
|
|
if ((this.minDate && end.isBefore(this.minDate)) || (maxDate && start.isAfter(maxDate)))
|
|
continue;
|
|
|
|
//Support unicode chars in the range names.
|
|
var elem = document.createElement('textarea');
|
|
elem.innerHTML = range;
|
|
var rangeHtml = elem.value;
|
|
|
|
this.ranges[rangeHtml] = [start, end];
|
|
}
|
|
|
|
var list = '<ul>';
|
|
for (range in this.ranges) {
|
|
list += '<li>' + range + '</li>';
|
|
}
|
|
list += '<li>' + this.locale.customRangeLabel + '</li>';
|
|
list += '</ul>';
|
|
this.container.find('.ranges').prepend(list);
|
|
}
|
|
|
|
if (typeof cb === 'function') {
|
|
this.callback = cb;
|
|
}
|
|
|
|
if (!this.timePicker) {
|
|
this.startDate = this.startDate.startOf('day');
|
|
this.endDate = this.endDate.endOf('day');
|
|
this.container.find('.calendar-time').hide();
|
|
}
|
|
|
|
//can't be used together for now
|
|
if (this.timePicker && this.autoApply)
|
|
this.autoApply = false;
|
|
|
|
if (this.autoApply && typeof options.ranges !== 'object') {
|
|
this.container.find('.ranges').hide();
|
|
} else if (this.autoApply) {
|
|
this.container.find('.applyBtn, .cancelBtn').addClass('hide');
|
|
}
|
|
|
|
if (this.singleDatePicker) {
|
|
this.container.addClass('single');
|
|
this.container.find('.calendar.left').addClass('single');
|
|
this.container.find('.calendar.left').show();
|
|
this.container.find('.calendar.right').hide();
|
|
this.container.find('.daterangepicker_input input, .daterangepicker_input i').hide();
|
|
if (!this.timePicker) {
|
|
this.container.find('.ranges').hide();
|
|
}
|
|
}
|
|
|
|
if ((typeof options.ranges === 'undefined' && !this.singleDatePicker) || this.alwaysShowCalendars) {
|
|
this.container.addClass('show-calendar');
|
|
}
|
|
|
|
this.container.addClass('opens' + this.opens);
|
|
|
|
//swap the position of the predefined ranges if opens right
|
|
if (typeof options.ranges !== 'undefined' && this.opens == 'right') {
|
|
var ranges = this.container.find('.ranges');
|
|
var html = ranges.clone();
|
|
ranges.remove();
|
|
this.container.find('.calendar.left').parent().prepend(html);
|
|
}
|
|
|
|
//apply CSS classes and labels to buttons
|
|
this.container.find('.applyBtn, .cancelBtn').addClass(this.buttonClasses);
|
|
if (this.applyClass.length)
|
|
this.container.find('.applyBtn').addClass(this.applyClass);
|
|
if (this.cancelClass.length)
|
|
this.container.find('.cancelBtn').addClass(this.cancelClass);
|
|
this.container.find('.applyBtn').html(this.locale.applyLabel);
|
|
this.container.find('.cancelBtn').html(this.locale.cancelLabel);
|
|
|
|
//
|
|
// event listeners
|
|
//
|
|
|
|
this.container.find('.calendar')
|
|
.on('click.daterangepicker', '.prev', $.proxy(this.clickPrev, this))
|
|
.on('click.daterangepicker', '.next', $.proxy(this.clickNext, this))
|
|
.on('click.daterangepicker', 'td.available', $.proxy(this.clickDate, this))
|
|
.on('mouseenter.daterangepicker', 'td.available', $.proxy(this.hoverDate, this))
|
|
.on('mouseleave.daterangepicker', 'td.available', $.proxy(this.updateFormInputs, this))
|
|
.on('change.daterangepicker', 'select.yearselect', $.proxy(this.monthOrYearChanged, this))
|
|
.on('change.daterangepicker', 'select.monthselect', $.proxy(this.monthOrYearChanged, this))
|
|
.on('change.daterangepicker', 'select.hourselect,select.minuteselect,select.secondselect,select.ampmselect', $.proxy(this.timeChanged, this))
|
|
.on('click.daterangepicker', '.daterangepicker_input input', $.proxy(this.showCalendars, this))
|
|
//.on('keyup.daterangepicker', '.daterangepicker_input input', $.proxy(this.formInputsChanged, this))
|
|
.on('change.daterangepicker', '.daterangepicker_input input', $.proxy(this.formInputsChanged, this));
|
|
|
|
this.container.find('.ranges')
|
|
.on('click.daterangepicker', 'button.applyBtn', $.proxy(this.clickApply, this))
|
|
.on('click.daterangepicker', 'button.cancelBtn', $.proxy(this.clickCancel, this))
|
|
.on('click.daterangepicker', 'li', $.proxy(this.clickRange, this))
|
|
.on('mouseenter.daterangepicker', 'li', $.proxy(this.hoverRange, this))
|
|
.on('mouseleave.daterangepicker', 'li', $.proxy(this.updateFormInputs, this));
|
|
|
|
if (this.element.is('input')) {
|
|
this.element.on({
|
|
'click.daterangepicker': $.proxy(this.show, this),
|
|
'focus.daterangepicker': $.proxy(this.show, this),
|
|
'keyup.daterangepicker': $.proxy(this.elementChanged, this),
|
|
'keydown.daterangepicker': $.proxy(this.keydown, this)
|
|
});
|
|
} else {
|
|
this.element.on('click.daterangepicker', $.proxy(this.toggle, this));
|
|
}
|
|
|
|
//
|
|
// if attached to a text input, set the initial value
|
|
//
|
|
|
|
if (this.element.is('input') && !this.singleDatePicker && this.autoUpdateInput) {
|
|
this.element.val(this.startDate.format(this.locale.format) + this.locale.separator + this.endDate.format(this.locale.format));
|
|
this.element.trigger('change');
|
|
} else if (this.element.is('input') && this.autoUpdateInput) {
|
|
this.element.val(this.startDate.format(this.locale.format));
|
|
this.element.trigger('change');
|
|
}
|
|
|
|
};
|
|
|
|
DateRangePicker.prototype = {
|
|
|
|
constructor: DateRangePicker,
|
|
|
|
setStartDate: function(startDate) {
|
|
if (typeof startDate === 'string')
|
|
this.startDate = moment(startDate, this.locale.format);
|
|
|
|
if (typeof startDate === 'object')
|
|
this.startDate = moment(startDate);
|
|
|
|
if (!this.timePicker)
|
|
this.startDate = this.startDate.startOf('day');
|
|
|
|
if (this.timePicker && this.timePickerIncrement)
|
|
this.startDate.minute(Math.round(this.startDate.minute() / this.timePickerIncrement) * this.timePickerIncrement);
|
|
|
|
if (this.minDate && this.startDate.isBefore(this.minDate))
|
|
this.startDate = this.minDate;
|
|
|
|
if (this.maxDate && this.startDate.isAfter(this.maxDate))
|
|
this.startDate = this.maxDate;
|
|
|
|
if (!this.isShowing)
|
|
this.updateElement();
|
|
|
|
this.updateMonthsInView();
|
|
},
|
|
|
|
setEndDate: function(endDate) {
|
|
if (typeof endDate === 'string')
|
|
this.endDate = moment(endDate, this.locale.format);
|
|
|
|
if (typeof endDate === 'object')
|
|
this.endDate = moment(endDate);
|
|
|
|
if (!this.timePicker)
|
|
this.endDate = this.endDate.endOf('day');
|
|
|
|
if (this.timePicker && this.timePickerIncrement)
|
|
this.endDate.minute(Math.round(this.endDate.minute() / this.timePickerIncrement) * this.timePickerIncrement);
|
|
|
|
if (this.endDate.isBefore(this.startDate))
|
|
this.endDate = this.startDate.clone();
|
|
|
|
if (this.maxDate && this.endDate.isAfter(this.maxDate))
|
|
this.endDate = this.maxDate;
|
|
|
|
if (this.dateLimit && this.startDate.clone().add(this.dateLimit).isBefore(this.endDate))
|
|
this.endDate = this.startDate.clone().add(this.dateLimit);
|
|
|
|
this.previousRightTime = this.endDate.clone();
|
|
|
|
if (!this.isShowing)
|
|
this.updateElement();
|
|
|
|
this.updateMonthsInView();
|
|
},
|
|
|
|
isInvalidDate: function() {
|
|
return false;
|
|
},
|
|
|
|
updateView: function() {
|
|
if (this.timePicker) {
|
|
this.renderTimePicker('left');
|
|
this.renderTimePicker('right');
|
|
if (!this.endDate) {
|
|
this.container.find('.right .calendar-time select').attr('disabled', 'disabled').addClass('disabled');
|
|
} else {
|
|
this.container.find('.right .calendar-time select').removeAttr('disabled').removeClass('disabled');
|
|
}
|
|
}
|
|
if (this.endDate) {
|
|
this.container.find('input[name="daterangepicker_end"]').removeClass('active');
|
|
this.container.find('input[name="daterangepicker_start"]').addClass('active');
|
|
} else {
|
|
this.container.find('input[name="daterangepicker_end"]').addClass('active');
|
|
this.container.find('input[name="daterangepicker_start"]').removeClass('active');
|
|
}
|
|
this.updateMonthsInView();
|
|
this.updateCalendars();
|
|
this.updateFormInputs();
|
|
},
|
|
|
|
updateMonthsInView: function() {
|
|
if (this.endDate) {
|
|
|
|
//if both dates are visible already, do nothing
|
|
if (!this.singleDatePicker && this.leftCalendar.month && this.rightCalendar.month &&
|
|
(this.startDate.format('YYYY-MM') == this.leftCalendar.month.format('YYYY-MM') || this.startDate.format('YYYY-MM') == this.rightCalendar.month.format('YYYY-MM'))
|
|
&&
|
|
(this.endDate.format('YYYY-MM') == this.leftCalendar.month.format('YYYY-MM') || this.endDate.format('YYYY-MM') == this.rightCalendar.month.format('YYYY-MM'))
|
|
) {
|
|
return;
|
|
}
|
|
|
|
this.leftCalendar.month = this.startDate.clone().date(2);
|
|
if (!this.linkedCalendars && (this.endDate.month() != this.startDate.month() || this.endDate.year() != this.startDate.year())) {
|
|
this.rightCalendar.month = this.endDate.clone().date(2);
|
|
} else {
|
|
this.rightCalendar.month = this.startDate.clone().date(2).add(1, 'month');
|
|
}
|
|
|
|
} else {
|
|
if (this.leftCalendar.month.format('YYYY-MM') != this.startDate.format('YYYY-MM') && this.rightCalendar.month.format('YYYY-MM') != this.startDate.format('YYYY-MM')) {
|
|
this.leftCalendar.month = this.startDate.clone().date(2);
|
|
this.rightCalendar.month = this.startDate.clone().date(2).add(1, 'month');
|
|
}
|
|
}
|
|
},
|
|
|
|
updateCalendars: function() {
|
|
|
|
if (this.timePicker) {
|
|
var hour, minute, second;
|
|
if (this.endDate) {
|
|
hour = parseInt(this.container.find('.left .hourselect').val(), 10);
|
|
minute = parseInt(this.container.find('.left .minuteselect').val(), 10);
|
|
second = this.timePickerSeconds ? parseInt(this.container.find('.left .secondselect').val(), 10) : 0;
|
|
if (!this.timePicker24Hour) {
|
|
var ampm = this.container.find('.left .ampmselect').val();
|
|
if (ampm === 'PM' && hour < 12)
|
|
hour += 12;
|
|
if (ampm === 'AM' && hour === 12)
|
|
hour = 0;
|
|
}
|
|
} else {
|
|
hour = parseInt(this.container.find('.right .hourselect').val(), 10);
|
|
minute = parseInt(this.container.find('.right .minuteselect').val(), 10);
|
|
second = this.timePickerSeconds ? parseInt(this.container.find('.right .secondselect').val(), 10) : 0;
|
|
if (!this.timePicker24Hour) {
|
|
var ampm = this.container.find('.right .ampmselect').val();
|
|
if (ampm === 'PM' && hour < 12)
|
|
hour += 12;
|
|
if (ampm === 'AM' && hour === 12)
|
|
hour = 0;
|
|
}
|
|
}
|
|
this.leftCalendar.month.hour(hour).minute(minute).second(second);
|
|
this.rightCalendar.month.hour(hour).minute(minute).second(second);
|
|
}
|
|
|
|
this.renderCalendar('left');
|
|
this.renderCalendar('right');
|
|
|
|
//highlight any predefined range matching the current start and end dates
|
|
this.container.find('.ranges li').removeClass('active');
|
|
if (this.endDate == null) return;
|
|
|
|
this.calculateChosenLabel();
|
|
},
|
|
|
|
renderCalendar: function(side) {
|
|
|
|
//
|
|
// Build the matrix of dates that will populate the calendar
|
|
//
|
|
|
|
var calendar = side == 'left' ? this.leftCalendar : this.rightCalendar;
|
|
var month = calendar.month.month();
|
|
var year = calendar.month.year();
|
|
var hour = calendar.month.hour();
|
|
var minute = calendar.month.minute();
|
|
var second = calendar.month.second();
|
|
var daysInMonth = moment([year, month]).daysInMonth();
|
|
var firstDay = moment([year, month, 1]);
|
|
var lastDay = moment([year, month, daysInMonth]);
|
|
var lastMonth = moment(firstDay).subtract(1, 'month').month();
|
|
var lastYear = moment(firstDay).subtract(1, 'month').year();
|
|
var daysInLastMonth = moment([lastYear, lastMonth]).daysInMonth();
|
|
var dayOfWeek = firstDay.day();
|
|
|
|
//initialize a 6 rows x 7 columns array for the calendar
|
|
var calendar = [];
|
|
calendar.firstDay = firstDay;
|
|
calendar.lastDay = lastDay;
|
|
|
|
for (var i = 0; i < 6; i++) {
|
|
calendar[i] = [];
|
|
}
|
|
|
|
//populate the calendar with date objects
|
|
var startDay = daysInLastMonth - dayOfWeek + this.locale.firstDay + 1;
|
|
if (startDay > daysInLastMonth)
|
|
startDay -= 7;
|
|
|
|
if (dayOfWeek == this.locale.firstDay)
|
|
startDay = daysInLastMonth - 6;
|
|
|
|
var curDate = moment([lastYear, lastMonth, startDay, 12, minute, second]);
|
|
|
|
var col, row;
|
|
for (var i = 0, col = 0, row = 0; i < 42; i++, col++, curDate = moment(curDate).add(24, 'hour')) {
|
|
if (i > 0 && col % 7 === 0) {
|
|
col = 0;
|
|
row++;
|
|
}
|
|
calendar[row][col] = curDate.clone().hour(hour).minute(minute).second(second);
|
|
curDate.hour(12);
|
|
|
|
if (this.minDate && calendar[row][col].format('YYYY-MM-DD') == this.minDate.format('YYYY-MM-DD') && calendar[row][col].isBefore(this.minDate) && side == 'left') {
|
|
calendar[row][col] = this.minDate.clone();
|
|
}
|
|
|
|
if (this.maxDate && calendar[row][col].format('YYYY-MM-DD') == this.maxDate.format('YYYY-MM-DD') && calendar[row][col].isAfter(this.maxDate) && side == 'right') {
|
|
calendar[row][col] = this.maxDate.clone();
|
|
}
|
|
|
|
}
|
|
|
|
//make the calendar object available to hoverDate/clickDate
|
|
if (side == 'left') {
|
|
this.leftCalendar.calendar = calendar;
|
|
} else {
|
|
this.rightCalendar.calendar = calendar;
|
|
}
|
|
|
|
//
|
|
// Display the calendar
|
|
//
|
|
|
|
var minDate = side == 'left' ? this.minDate : this.startDate;
|
|
var maxDate = this.maxDate;
|
|
var selected = side == 'left' ? this.startDate : this.endDate;
|
|
|
|
var html = '<table class="table-condensed">';
|
|
html += '<thead>';
|
|
html += '<tr>';
|
|
|
|
// add empty cell for week number
|
|
if (this.showWeekNumbers || this.showISOWeekNumbers)
|
|
html += '<th></th>';
|
|
|
|
if ((!minDate || minDate.isBefore(calendar.firstDay)) && (!this.linkedCalendars || side == 'left')) {
|
|
html += '<th class="prev available"><i class="fa fa-chevron-left glyphicon glyphicon-chevron-left"></i></th>';
|
|
} else {
|
|
html += '<th></th>';
|
|
}
|
|
|
|
var dateHtml = this.locale.monthNames[calendar[1][1].month()] + calendar[1][1].format(" YYYY");
|
|
|
|
if (this.showDropdowns) {
|
|
var currentMonth = calendar[1][1].month();
|
|
var currentYear = calendar[1][1].year();
|
|
var maxYear = (maxDate && maxDate.year()) || (currentYear + 5);
|
|
var minYear = (minDate && minDate.year()) || (currentYear - 50);
|
|
var inMinYear = currentYear == minYear;
|
|
var inMaxYear = currentYear == maxYear;
|
|
|
|
var monthHtml = '<select class="monthselect">';
|
|
for (var m = 0; m < 12; m++) {
|
|
if ((!inMinYear || m >= minDate.month()) && (!inMaxYear || m <= maxDate.month())) {
|
|
monthHtml += "<option value='" + m + "'" +
|
|
(m === currentMonth ? " selected='selected'" : "") +
|
|
">" + this.locale.monthNames[m] + "</option>";
|
|
} else {
|
|
monthHtml += "<option value='" + m + "'" +
|
|
(m === currentMonth ? " selected='selected'" : "") +
|
|
" disabled='disabled'>" + this.locale.monthNames[m] + "</option>";
|
|
}
|
|
}
|
|
monthHtml += "</select>";
|
|
|
|
var yearHtml = '<select class="yearselect">';
|
|
for (var y = minYear; y <= maxYear; y++) {
|
|
yearHtml += '<option value="' + y + '"' +
|
|
(y === currentYear ? ' selected="selected"' : '') +
|
|
'>' + y + '</option>';
|
|
}
|
|
yearHtml += '</select>';
|
|
|
|
dateHtml = monthHtml + yearHtml;
|
|
}
|
|
|
|
html += '<th colspan="5" class="month">' + dateHtml + '</th>';
|
|
if ((!maxDate || maxDate.isAfter(calendar.lastDay)) && (!this.linkedCalendars || side == 'right' || this.singleDatePicker)) {
|
|
html += '<th class="next available"><i class="fa fa-chevron-right glyphicon glyphicon-chevron-right"></i></th>';
|
|
} else {
|
|
html += '<th></th>';
|
|
}
|
|
|
|
html += '</tr>';
|
|
html += '<tr>';
|
|
|
|
// add week number label
|
|
if (this.showWeekNumbers || this.showISOWeekNumbers)
|
|
html += '<th class="week">' + this.locale.weekLabel + '</th>';
|
|
|
|
$.each(this.locale.daysOfWeek, function(index, dayOfWeek) {
|
|
html += '<th>' + dayOfWeek + '</th>';
|
|
});
|
|
|
|
html += '</tr>';
|
|
html += '</thead>';
|
|
html += '<tbody>';
|
|
|
|
//adjust maxDate to reflect the dateLimit setting in order to
|
|
//grey out end dates beyond the dateLimit
|
|
if (this.endDate == null && this.dateLimit) {
|
|
var maxLimit = this.startDate.clone().add(this.dateLimit).endOf('day');
|
|
if (!maxDate || maxLimit.isBefore(maxDate)) {
|
|
maxDate = maxLimit;
|
|
}
|
|
}
|
|
|
|
for (var row = 0; row < 6; row++) {
|
|
html += '<tr>';
|
|
|
|
// add week number
|
|
if (this.showWeekNumbers)
|
|
html += '<td class="week">' + calendar[row][0].week() + '</td>';
|
|
else if (this.showISOWeekNumbers)
|
|
html += '<td class="week">' + calendar[row][0].isoWeek() + '</td>';
|
|
|
|
for (var col = 0; col < 7; col++) {
|
|
|
|
var classes = [];
|
|
|
|
//highlight today's date
|
|
if (calendar[row][col].isSame(new Date(), "day"))
|
|
classes.push('today');
|
|
|
|
//highlight weekends
|
|
if (calendar[row][col].isoWeekday() > 5)
|
|
classes.push('weekend');
|
|
|
|
//grey out the dates in other months displayed at beginning and end of this calendar
|
|
if (calendar[row][col].month() != calendar[1][1].month())
|
|
classes.push('off');
|
|
|
|
//don't allow selection of dates before the minimum date
|
|
if (this.minDate && calendar[row][col].isBefore(this.minDate, 'day'))
|
|
classes.push('off', 'disabled');
|
|
|
|
//don't allow selection of dates after the maximum date
|
|
if (maxDate && calendar[row][col].isAfter(maxDate, 'day'))
|
|
classes.push('off', 'disabled');
|
|
|
|
//don't allow selection of date if a custom function decides it's invalid
|
|
if (this.isInvalidDate(calendar[row][col]))
|
|
classes.push('off', 'disabled');
|
|
|
|
//highlight the currently selected start date
|
|
if (calendar[row][col].format('YYYY-MM-DD') == this.startDate.format('YYYY-MM-DD'))
|
|
classes.push('active', 'start-date');
|
|
|
|
//highlight the currently selected end date
|
|
if (this.endDate != null && calendar[row][col].format('YYYY-MM-DD') == this.endDate.format('YYYY-MM-DD'))
|
|
classes.push('active', 'end-date');
|
|
|
|
//highlight dates in-between the selected dates
|
|
if (this.endDate != null && calendar[row][col] > this.startDate && calendar[row][col] < this.endDate)
|
|
classes.push('in-range');
|
|
|
|
var cname = '', disabled = false;
|
|
for (var i = 0; i < classes.length; i++) {
|
|
cname += classes[i] + ' ';
|
|
if (classes[i] == 'disabled')
|
|
disabled = true;
|
|
}
|
|
if (!disabled)
|
|
cname += 'available';
|
|
|
|
html += '<td class="' + cname.replace(/^\s+|\s+$/g, '') + '" data-title="' + 'r' + row + 'c' + col + '">' + calendar[row][col].date() + '</td>';
|
|
|
|
}
|
|
html += '</tr>';
|
|
}
|
|
|
|
html += '</tbody>';
|
|
html += '</table>';
|
|
|
|
this.container.find('.calendar.' + side + ' .calendar-table').html(html);
|
|
|
|
},
|
|
|
|
renderTimePicker: function(side) {
|
|
|
|
var html, selected, minDate, maxDate = this.maxDate;
|
|
|
|
if (this.dateLimit && (!this.maxDate || this.startDate.clone().add(this.dateLimit).isAfter(this.maxDate)))
|
|
maxDate = this.startDate.clone().add(this.dateLimit);
|
|
|
|
if (side == 'left') {
|
|
selected = this.startDate.clone();
|
|
minDate = this.minDate;
|
|
} else if (side == 'right') {
|
|
selected = this.endDate ? this.endDate.clone() : this.previousRightTime.clone();
|
|
minDate = this.startDate;
|
|
|
|
//Preserve the time already selected
|
|
var timeSelector = this.container.find('.calendar.right .calendar-time div');
|
|
if (timeSelector.html() != '') {
|
|
|
|
selected.hour(timeSelector.find('.hourselect option:selected').val() || selected.hour());
|
|
selected.minute(timeSelector.find('.minuteselect option:selected').val() || selected.minute());
|
|
selected.second(timeSelector.find('.secondselect option:selected').val() || selected.second());
|
|
|
|
if (!this.timePicker24Hour) {
|
|
var ampm = timeSelector.find('.ampmselect option:selected').val();
|
|
if (ampm === 'PM' && selected.hour() < 12)
|
|
selected.hour(selected.hour() + 12);
|
|
if (ampm === 'AM' && selected.hour() === 12)
|
|
selected.hour(0);
|
|
}
|
|
|
|
if (selected.isBefore(this.startDate))
|
|
selected = this.startDate.clone();
|
|
|
|
if (selected.isAfter(maxDate))
|
|
selected = maxDate.clone();
|
|
|
|
}
|
|
}
|
|
|
|
//
|
|
// hours
|
|
//
|
|
|
|
html = '<select class="hourselect">';
|
|
|
|
var start = this.timePicker24Hour ? 0 : 1;
|
|
var end = this.timePicker24Hour ? 23 : 12;
|
|
|
|
for (var i = start; i <= end; i++) {
|
|
var i_in_24 = i;
|
|
if (!this.timePicker24Hour)
|
|
i_in_24 = selected.hour() >= 12 ? (i == 12 ? 12 : i + 12) : (i == 12 ? 0 : i);
|
|
|
|
var time = selected.clone().hour(i_in_24);
|
|
var disabled = false;
|
|
if (minDate && time.minute(59).isBefore(minDate))
|
|
disabled = true;
|
|
if (maxDate && time.minute(0).isAfter(maxDate))
|
|
disabled = true;
|
|
|
|
if (i_in_24 == selected.hour() && !disabled) {
|
|
html += '<option value="' + i + '" selected="selected">' + i + '</option>';
|
|
} else if (disabled) {
|
|
html += '<option value="' + i + '" disabled="disabled" class="disabled">' + i + '</option>';
|
|
} else {
|
|
html += '<option value="' + i + '">' + i + '</option>';
|
|
}
|
|
}
|
|
|
|
html += '</select> ';
|
|
|
|
//
|
|
// minutes
|
|
//
|
|
|
|
html += ': <select class="minuteselect">';
|
|
|
|
for (var i = 0; i < 60; i += this.timePickerIncrement) {
|
|
var padded = i < 10 ? '0' + i : i;
|
|
var time = selected.clone().minute(i);
|
|
|
|
var disabled = false;
|
|
if (minDate && time.second(59).isBefore(minDate))
|
|
disabled = true;
|
|
if (maxDate && time.second(0).isAfter(maxDate))
|
|
disabled = true;
|
|
|
|
if (selected.minute() == i && !disabled) {
|
|
html += '<option value="' + i + '" selected="selected">' + padded + '</option>';
|
|
} else if (disabled) {
|
|
html += '<option value="' + i + '" disabled="disabled" class="disabled">' + padded + '</option>';
|
|
} else {
|
|
html += '<option value="' + i + '">' + padded + '</option>';
|
|
}
|
|
}
|
|
|
|
html += '</select> ';
|
|
|
|
//
|
|
// seconds
|
|
//
|
|
|
|
if (this.timePickerSeconds) {
|
|
html += ': <select class="secondselect">';
|
|
|
|
for (var i = 0; i < 60; i++) {
|
|
var padded = i < 10 ? '0' + i : i;
|
|
var time = selected.clone().second(i);
|
|
|
|
var disabled = false;
|
|
if (minDate && time.isBefore(minDate))
|
|
disabled = true;
|
|
if (maxDate && time.isAfter(maxDate))
|
|
disabled = true;
|
|
|
|
if (selected.second() == i && !disabled) {
|
|
html += '<option value="' + i + '" selected="selected">' + padded + '</option>';
|
|
} else if (disabled) {
|
|
html += '<option value="' + i + '" disabled="disabled" class="disabled">' + padded + '</option>';
|
|
} else {
|
|
html += '<option value="' + i + '">' + padded + '</option>';
|
|
}
|
|
}
|
|
|
|
html += '</select> ';
|
|
}
|
|
|
|
//
|
|
// AM/PM
|
|
//
|
|
|
|
if (!this.timePicker24Hour) {
|
|
html += '<select class="ampmselect">';
|
|
|
|
var am_html = '';
|
|
var pm_html = '';
|
|
|
|
if (minDate && selected.clone().hour(12).minute(0).second(0).isBefore(minDate))
|
|
am_html = ' disabled="disabled" class="disabled"';
|
|
|
|
if (maxDate && selected.clone().hour(0).minute(0).second(0).isAfter(maxDate))
|
|
pm_html = ' disabled="disabled" class="disabled"';
|
|
|
|
if (selected.hour() >= 12) {
|
|
html += '<option value="AM"' + am_html + '>AM</option><option value="PM" selected="selected"' + pm_html + '>PM</option>';
|
|
} else {
|
|
html += '<option value="AM" selected="selected"' + am_html + '>AM</option><option value="PM"' + pm_html + '>PM</option>';
|
|
}
|
|
|
|
html += '</select>';
|
|
}
|
|
|
|
this.container.find('.calendar.' + side + ' .calendar-time div').html(html);
|
|
|
|
},
|
|
|
|
updateFormInputs: function() {
|
|
|
|
//ignore mouse movements while an above-calendar text input has focus
|
|
if (this.container.find('input[name=daterangepicker_start]').is(":focus") || this.container.find('input[name=daterangepicker_end]').is(":focus"))
|
|
return;
|
|
|
|
this.container.find('input[name=daterangepicker_start]').val(this.startDate.format(this.locale.format));
|
|
if (this.endDate)
|
|
this.container.find('input[name=daterangepicker_end]').val(this.endDate.format(this.locale.format));
|
|
|
|
if (this.singleDatePicker || (this.endDate && (this.startDate.isBefore(this.endDate) || this.startDate.isSame(this.endDate)))) {
|
|
this.container.find('button.applyBtn').removeAttr('disabled');
|
|
} else {
|
|
this.container.find('button.applyBtn').attr('disabled', 'disabled');
|
|
}
|
|
|
|
},
|
|
|
|
move: function() {
|
|
var parentOffset = { top: 0, left: 0 },
|
|
containerTop;
|
|
var parentRightEdge = $(window).width();
|
|
if (!this.parentEl.is('body')) {
|
|
parentOffset = {
|
|
top: this.parentEl.offset().top - this.parentEl.scrollTop(),
|
|
left: this.parentEl.offset().left - this.parentEl.scrollLeft()
|
|
};
|
|
parentRightEdge = this.parentEl[0].clientWidth + this.parentEl.offset().left;
|
|
}
|
|
|
|
if (this.drops == 'up')
|
|
containerTop = this.element.offset().top - this.container.outerHeight() - parentOffset.top;
|
|
else
|
|
containerTop = this.element.offset().top + this.element.outerHeight() - parentOffset.top;
|
|
this.container[this.drops == 'up' ? 'addClass' : 'removeClass']('dropup');
|
|
|
|
if (this.opens == 'left') {
|
|
this.container.css({
|
|
top: containerTop,
|
|
right: parentRightEdge - this.element.offset().left - this.element.outerWidth(),
|
|
left: 'auto'
|
|
});
|
|
if (this.container.offset().left < 0) {
|
|
this.container.css({
|
|
right: 'auto',
|
|
left: 9
|
|
});
|
|
}
|
|
} else if (this.opens == 'center') {
|
|
this.container.css({
|
|
top: containerTop,
|
|
left: this.element.offset().left - parentOffset.left + this.element.outerWidth() / 2
|
|
- this.container.outerWidth() / 2,
|
|
right: 'auto'
|
|
});
|
|
if (this.container.offset().left < 0) {
|
|
this.container.css({
|
|
right: 'auto',
|
|
left: 9
|
|
});
|
|
}
|
|
} else {
|
|
this.container.css({
|
|
top: containerTop,
|
|
left: this.element.offset().left - parentOffset.left,
|
|
right: 'auto'
|
|
});
|
|
if (this.container.offset().left + this.container.outerWidth() > $(window).width()) {
|
|
this.container.css({
|
|
left: 'auto',
|
|
right: 0
|
|
});
|
|
}
|
|
}
|
|
},
|
|
|
|
show: function(e) {
|
|
if (this.isShowing) return;
|
|
|
|
// Create a click proxy that is private to this instance of datepicker, for unbinding
|
|
this._outsideClickProxy = $.proxy(function(e) { this.outsideClick(e); }, this);
|
|
|
|
// Bind global datepicker mousedown for hiding and
|
|
$(document)
|
|
.on('mousedown.daterangepicker', this._outsideClickProxy)
|
|
// also support mobile devices
|
|
.on('touchend.daterangepicker', this._outsideClickProxy)
|
|
// also explicitly play nice with Bootstrap dropdowns, which stopPropagation when clicking them
|
|
.on('click.daterangepicker', '[data-toggle=dropdown]', this._outsideClickProxy)
|
|
// and also close when focus changes to outside the picker (eg. tabbing between controls)
|
|
.on('focusin.daterangepicker', this._outsideClickProxy);
|
|
|
|
// Reposition the picker if the window is resized while it's open
|
|
$(window).on('resize.daterangepicker', $.proxy(function(e) { this.move(e); }, this));
|
|
|
|
this.oldStartDate = this.startDate.clone();
|
|
this.oldEndDate = this.endDate.clone();
|
|
this.previousRightTime = this.endDate.clone();
|
|
|
|
this.updateView();
|
|
this.container.show();
|
|
this.move();
|
|
this.element.trigger('show.daterangepicker', this);
|
|
this.isShowing = true;
|
|
},
|
|
|
|
hide: function(e) {
|
|
if (!this.isShowing) return;
|
|
|
|
//incomplete date selection, revert to last values
|
|
if (!this.endDate) {
|
|
this.startDate = this.oldStartDate.clone();
|
|
this.endDate = this.oldEndDate.clone();
|
|
}
|
|
|
|
//if a new date range was selected, invoke the user callback function
|
|
if (!this.startDate.isSame(this.oldStartDate) || !this.endDate.isSame(this.oldEndDate))
|
|
this.callback(this.startDate, this.endDate, this.chosenLabel);
|
|
|
|
//if picker is attached to a text input, update it
|
|
this.updateElement();
|
|
|
|
$(document).off('.daterangepicker');
|
|
$(window).off('.daterangepicker');
|
|
this.container.hide();
|
|
this.element.trigger('hide.daterangepicker', this);
|
|
this.isShowing = false;
|
|
},
|
|
|
|
toggle: function(e) {
|
|
if (this.isShowing) {
|
|
this.hide();
|
|
} else {
|
|
this.show();
|
|
}
|
|
},
|
|
|
|
outsideClick: function(e) {
|
|
var target = $(e.target);
|
|
// if the page is clicked anywhere except within the daterangerpicker/button
|
|
// itself then call this.hide()
|
|
if (
|
|
// ie modal dialog fix
|
|
e.type == "focusin" ||
|
|
target.closest(this.element).length ||
|
|
target.closest(this.container).length ||
|
|
target.closest('.calendar-table').length
|
|
) return;
|
|
this.hide();
|
|
},
|
|
|
|
showCalendars: function() {
|
|
this.container.addClass('show-calendar');
|
|
this.move();
|
|
this.element.trigger('showCalendar.daterangepicker', this);
|
|
},
|
|
|
|
hideCalendars: function() {
|
|
this.container.removeClass('show-calendar');
|
|
this.element.trigger('hideCalendar.daterangepicker', this);
|
|
},
|
|
|
|
hoverRange: function(e) {
|
|
|
|
//ignore mouse movements while an above-calendar text input has focus
|
|
if (this.container.find('input[name=daterangepicker_start]').is(":focus") || this.container.find('input[name=daterangepicker_end]').is(":focus"))
|
|
return;
|
|
|
|
var label = e.target.innerHTML;
|
|
if (label == this.locale.customRangeLabel) {
|
|
this.updateView();
|
|
} else {
|
|
var dates = this.ranges[label];
|
|
this.container.find('input[name=daterangepicker_start]').val(dates[0].format(this.locale.format));
|
|
this.container.find('input[name=daterangepicker_end]').val(dates[1].format(this.locale.format));
|
|
}
|
|
|
|
},
|
|
|
|
clickRange: function(e) {
|
|
var label = e.target.innerHTML;
|
|
this.chosenLabel = label;
|
|
if (label == this.locale.customRangeLabel) {
|
|
this.showCalendars();
|
|
} else {
|
|
var dates = this.ranges[label];
|
|
this.startDate = dates[0];
|
|
this.endDate = dates[1];
|
|
|
|
if (!this.timePicker) {
|
|
this.startDate.startOf('day');
|
|
this.endDate.endOf('day');
|
|
}
|
|
|
|
if (!this.alwaysShowCalendars)
|
|
this.hideCalendars();
|
|
this.clickApply();
|
|
}
|
|
},
|
|
|
|
clickPrev: function(e) {
|
|
var cal = $(e.target).parents('.calendar');
|
|
if (cal.hasClass('left')) {
|
|
this.leftCalendar.month.subtract(1, 'month');
|
|
if (this.linkedCalendars)
|
|
this.rightCalendar.month.subtract(1, 'month');
|
|
} else {
|
|
this.rightCalendar.month.subtract(1, 'month');
|
|
}
|
|
this.updateCalendars();
|
|
},
|
|
|
|
clickNext: function(e) {
|
|
var cal = $(e.target).parents('.calendar');
|
|
if (cal.hasClass('left')) {
|
|
this.leftCalendar.month.add(1, 'month');
|
|
} else {
|
|
this.rightCalendar.month.add(1, 'month');
|
|
if (this.linkedCalendars)
|
|
this.leftCalendar.month.add(1, 'month');
|
|
}
|
|
this.updateCalendars();
|
|
},
|
|
|
|
hoverDate: function(e) {
|
|
|
|
//ignore mouse movements while an above-calendar text input has focus
|
|
if (this.container.find('input[name=daterangepicker_start]').is(":focus") || this.container.find('input[name=daterangepicker_end]').is(":focus"))
|
|
return;
|
|
|
|
//ignore dates that can't be selected
|
|
if (!$(e.target).hasClass('available')) return;
|
|
|
|
//have the text inputs above calendars reflect the date being hovered over
|
|
var title = $(e.target).attr('data-title');
|
|
var row = title.substr(1, 1);
|
|
var col = title.substr(3, 1);
|
|
var cal = $(e.target).parents('.calendar');
|
|
var date = cal.hasClass('left') ? this.leftCalendar.calendar[row][col] : this.rightCalendar.calendar[row][col];
|
|
|
|
if (this.endDate) {
|
|
this.container.find('input[name=daterangepicker_start]').val(date.format(this.locale.format));
|
|
} else {
|
|
this.container.find('input[name=daterangepicker_end]').val(date.format(this.locale.format));
|
|
}
|
|
|
|
//highlight the dates between the start date and the date being hovered as a potential end date
|
|
var leftCalendar = this.leftCalendar;
|
|
var rightCalendar = this.rightCalendar;
|
|
var startDate = this.startDate;
|
|
if (!this.endDate) {
|
|
this.container.find('.calendar td').each(function(index, el) {
|
|
|
|
//skip week numbers, only look at dates
|
|
if ($(el).hasClass('week')) return;
|
|
|
|
var title = $(el).attr('data-title');
|
|
var row = title.substr(1, 1);
|
|
var col = title.substr(3, 1);
|
|
var cal = $(el).parents('.calendar');
|
|
var dt = cal.hasClass('left') ? leftCalendar.calendar[row][col] : rightCalendar.calendar[row][col];
|
|
|
|
if (dt.isAfter(startDate) && dt.isBefore(date)) {
|
|
$(el).addClass('in-range');
|
|
} else {
|
|
$(el).removeClass('in-range');
|
|
}
|
|
|
|
});
|
|
}
|
|
|
|
},
|
|
|
|
clickDate: function(e) {
|
|
|
|
if (!$(e.target).hasClass('available')) return;
|
|
|
|
var title = $(e.target).attr('data-title');
|
|
var row = title.substr(1, 1);
|
|
var col = title.substr(3, 1);
|
|
var cal = $(e.target).parents('.calendar');
|
|
var date = cal.hasClass('left') ? this.leftCalendar.calendar[row][col] : this.rightCalendar.calendar[row][col];
|
|
|
|
//
|
|
// this function needs to do a few things:
|
|
// * alternate between selecting a start and end date for the range,
|
|
// * if the time picker is enabled, apply the hour/minute/second from the select boxes to the clicked date
|
|
// * if autoapply is enabled, and an end date was chosen, apply the selection
|
|
// * if single date picker mode, and time picker isn't enabled, apply the selection immediately
|
|
//
|
|
|
|
if (this.endDate || date.isBefore(this.startDate, 'day')) {
|
|
if (this.timePicker) {
|
|
var hour = parseInt(this.container.find('.left .hourselect').val(), 10);
|
|
if (!this.timePicker24Hour) {
|
|
var ampm = this.container.find('.left .ampmselect').val();
|
|
if (ampm === 'PM' && hour < 12)
|
|
hour += 12;
|
|
if (ampm === 'AM' && hour === 12)
|
|
hour = 0;
|
|
}
|
|
var minute = parseInt(this.container.find('.left .minuteselect').val(), 10);
|
|
var second = this.timePickerSeconds ? parseInt(this.container.find('.left .secondselect').val(), 10) : 0;
|
|
date = date.clone().hour(hour).minute(minute).second(second);
|
|
}
|
|
this.endDate = null;
|
|
this.setStartDate(date.clone());
|
|
} else if (!this.endDate && date.isBefore(this.startDate)) {
|
|
//special case: clicking the same date for start/end,
|
|
//but the time of the end date is before the start date
|
|
this.setEndDate(this.startDate.clone());
|
|
} else {
|
|
if (this.timePicker) {
|
|
var hour = parseInt(this.container.find('.right .hourselect').val(), 10);
|
|
if (!this.timePicker24Hour) {
|
|
var ampm = this.container.find('.right .ampmselect').val();
|
|
if (ampm === 'PM' && hour < 12)
|
|
hour += 12;
|
|
if (ampm === 'AM' && hour === 12)
|
|
hour = 0;
|
|
}
|
|
var minute = parseInt(this.container.find('.right .minuteselect').val(), 10);
|
|
var second = this.timePickerSeconds ? parseInt(this.container.find('.right .secondselect').val(), 10) : 0;
|
|
date = date.clone().hour(hour).minute(minute).second(second);
|
|
}
|
|
this.setEndDate(date.clone());
|
|
if (this.autoApply) {
|
|
this.calculateChosenLabel();
|
|
this.clickApply();
|
|
}
|
|
}
|
|
|
|
if (this.singleDatePicker) {
|
|
this.setEndDate(this.startDate);
|
|
if (!this.timePicker)
|
|
this.clickApply();
|
|
}
|
|
|
|
this.updateView();
|
|
|
|
},
|
|
|
|
calculateChosenLabel: function() {
|
|
var customRange = true;
|
|
var i = 0;
|
|
for (var range in this.ranges) {
|
|
if (this.timePicker) {
|
|
if (this.startDate.isSame(this.ranges[range][0]) && this.endDate.isSame(this.ranges[range][1])) {
|
|
customRange = false;
|
|
this.chosenLabel = this.container.find('.ranges li:eq(' + i + ')').addClass('active').html();
|
|
break;
|
|
}
|
|
} else {
|
|
//ignore times when comparing dates if time picker is not enabled
|
|
if (this.startDate.format('YYYY-MM-DD') == this.ranges[range][0].format('YYYY-MM-DD') && this.endDate.format('YYYY-MM-DD') == this.ranges[range][1].format('YYYY-MM-DD')) {
|
|
customRange = false;
|
|
this.chosenLabel = this.container.find('.ranges li:eq(' + i + ')').addClass('active').html();
|
|
break;
|
|
}
|
|
}
|
|
i++;
|
|
}
|
|
if (customRange) {
|
|
this.chosenLabel = this.container.find('.ranges li:last').addClass('active').html();
|
|
this.showCalendars();
|
|
}
|
|
},
|
|
|
|
clickApply: function(e) {
|
|
this.hide();
|
|
this.element.trigger('apply.daterangepicker', this);
|
|
},
|
|
|
|
clickCancel: function(e) {
|
|
this.startDate = this.oldStartDate;
|
|
this.endDate = this.oldEndDate;
|
|
this.hide();
|
|
this.element.trigger('cancel.daterangepicker', this);
|
|
},
|
|
|
|
monthOrYearChanged: function(e) {
|
|
var isLeft = $(e.target).closest('.calendar').hasClass('left'),
|
|
leftOrRight = isLeft ? 'left' : 'right',
|
|
cal = this.container.find('.calendar.'+leftOrRight);
|
|
|
|
// Month must be Number for new moment versions
|
|
var month = parseInt(cal.find('.monthselect').val(), 10);
|
|
var year = cal.find('.yearselect').val();
|
|
|
|
if (!isLeft) {
|
|
if (year < this.startDate.year() || (year == this.startDate.year() && month < this.startDate.month())) {
|
|
month = this.startDate.month();
|
|
year = this.startDate.year();
|
|
}
|
|
}
|
|
|
|
if (this.minDate) {
|
|
if (year < this.minDate.year() || (year == this.minDate.year() && month < this.minDate.month())) {
|
|
month = this.minDate.month();
|
|
year = this.minDate.year();
|
|
}
|
|
}
|
|
|
|
if (this.maxDate) {
|
|
if (year > this.maxDate.year() || (year == this.maxDate.year() && month > this.maxDate.month())) {
|
|
month = this.maxDate.month();
|
|
year = this.maxDate.year();
|
|
}
|
|
}
|
|
|
|
if (isLeft) {
|
|
this.leftCalendar.month.month(month).year(year);
|
|
if (this.linkedCalendars)
|
|
this.rightCalendar.month = this.leftCalendar.month.clone().add(1, 'month');
|
|
} else {
|
|
this.rightCalendar.month.month(month).year(year);
|
|
if (this.linkedCalendars)
|
|
this.leftCalendar.month = this.rightCalendar.month.clone().subtract(1, 'month');
|
|
}
|
|
this.updateCalendars();
|
|
},
|
|
|
|
timeChanged: function(e) {
|
|
|
|
var cal = $(e.target).closest('.calendar'),
|
|
isLeft = cal.hasClass('left');
|
|
|
|
var hour = parseInt(cal.find('.hourselect').val(), 10);
|
|
var minute = parseInt(cal.find('.minuteselect').val(), 10);
|
|
var second = this.timePickerSeconds ? parseInt(cal.find('.secondselect').val(), 10) : 0;
|
|
|
|
if (!this.timePicker24Hour) {
|
|
var ampm = cal.find('.ampmselect').val();
|
|
if (ampm === 'PM' && hour < 12)
|
|
hour += 12;
|
|
if (ampm === 'AM' && hour === 12)
|
|
hour = 0;
|
|
}
|
|
|
|
if (isLeft) {
|
|
var start = this.startDate.clone();
|
|
start.hour(hour);
|
|
start.minute(minute);
|
|
start.second(second);
|
|
this.setStartDate(start);
|
|
if (this.singleDatePicker) {
|
|
this.endDate = this.startDate.clone();
|
|
} else if (this.endDate && this.endDate.format('YYYY-MM-DD') == start.format('YYYY-MM-DD') && this.endDate.isBefore(start)) {
|
|
this.setEndDate(start.clone());
|
|
}
|
|
} else if (this.endDate) {
|
|
var end = this.endDate.clone();
|
|
end.hour(hour);
|
|
end.minute(minute);
|
|
end.second(second);
|
|
this.setEndDate(end);
|
|
}
|
|
|
|
//update the calendars so all clickable dates reflect the new time component
|
|
this.updateCalendars();
|
|
|
|
//update the form inputs above the calendars with the new time
|
|
this.updateFormInputs();
|
|
|
|
//re-render the time pickers because changing one selection can affect what's enabled in another
|
|
this.renderTimePicker('left');
|
|
this.renderTimePicker('right');
|
|
|
|
},
|
|
|
|
formInputsChanged: function(e) {
|
|
var isRight = $(e.target).closest('.calendar').hasClass('right');
|
|
var start = moment(this.container.find('input[name="daterangepicker_start"]').val(), this.locale.format);
|
|
var end = moment(this.container.find('input[name="daterangepicker_end"]').val(), this.locale.format);
|
|
|
|
if (start.isValid() && end.isValid()) {
|
|
|
|
if (isRight && end.isBefore(start))
|
|
start = end.clone();
|
|
|
|
this.setStartDate(start);
|
|
this.setEndDate(end);
|
|
|
|
if (isRight) {
|
|
this.container.find('input[name="daterangepicker_start"]').val(this.startDate.format(this.locale.format));
|
|
} else {
|
|
this.container.find('input[name="daterangepicker_end"]').val(this.endDate.format(this.locale.format));
|
|
}
|
|
|
|
}
|
|
|
|
this.updateCalendars();
|
|
if (this.timePicker) {
|
|
this.renderTimePicker('left');
|
|
this.renderTimePicker('right');
|
|
}
|
|
},
|
|
|
|
elementChanged: function() {
|
|
if (!this.element.is('input')) return;
|
|
if (!this.element.val().length) return;
|
|
if (this.element.val().length < this.locale.format.length) return;
|
|
|
|
var dateString = this.element.val().split(this.locale.separator),
|
|
start = null,
|
|
end = null;
|
|
|
|
if (dateString.length === 2) {
|
|
start = moment(dateString[0], this.locale.format);
|
|
end = moment(dateString[1], this.locale.format);
|
|
}
|
|
|
|
if (this.singleDatePicker || start === null || end === null) {
|
|
start = moment(this.element.val(), this.locale.format);
|
|
end = start;
|
|
}
|
|
|
|
if (!start.isValid() || !end.isValid()) return;
|
|
|
|
this.setStartDate(start);
|
|
this.setEndDate(end);
|
|
this.updateView();
|
|
},
|
|
|
|
keydown: function(e) {
|
|
//hide on tab or enter
|
|
if ((e.keyCode === 9) || (e.keyCode === 13)) {
|
|
this.hide();
|
|
}
|
|
},
|
|
|
|
updateElement: function() {
|
|
if (this.element.is('input') && !this.singleDatePicker && this.autoUpdateInput) {
|
|
this.element.val(this.startDate.format(this.locale.format) + this.locale.separator + this.endDate.format(this.locale.format));
|
|
this.element.trigger('change');
|
|
} else if (this.element.is('input') && this.autoUpdateInput) {
|
|
this.element.val(this.startDate.format(this.locale.format));
|
|
this.element.trigger('change');
|
|
}
|
|
},
|
|
|
|
remove: function() {
|
|
this.container.remove();
|
|
this.element.off('.daterangepicker');
|
|
this.element.removeData();
|
|
}
|
|
|
|
};
|
|
|
|
$.fn.daterangepicker = function(options, callback) {
|
|
this.each(function() {
|
|
var el = $(this);
|
|
if (el.data('daterangepicker'))
|
|
el.data('daterangepicker').remove();
|
|
el.data('daterangepicker', new DateRangePicker(el, options, callback));
|
|
});
|
|
return this;
|
|
};
|
|
|
|
return DateRangePicker;
|
|
|
|
}));
|
|
|
|
/*!
|
|
* Bootstrap-select v1.10.0 (http://silviomoreto.github.io/bootstrap-select)
|
|
*
|
|
* Copyright 2013-2016 bootstrap-select
|
|
* Licensed under MIT (https://github.com/silviomoreto/bootstrap-select/blob/master/LICENSE)
|
|
*/
|
|
|
|
(function (root, factory) {
|
|
if (typeof define === 'function' && define.amd) {
|
|
// AMD. Register as an anonymous module unless amdModuleId is set
|
|
define(["jquery"], function (a0) {
|
|
return (factory(a0));
|
|
});
|
|
} else if (typeof exports === 'object') {
|
|
// Node. Does not work with strict CommonJS, but
|
|
// only CommonJS-like environments that support module.exports,
|
|
// like Node.
|
|
module.exports = factory(require("jquery"));
|
|
} else {
|
|
factory(jQuery);
|
|
}
|
|
}(this, function (jQuery) {
|
|
|
|
(function ($) {
|
|
'use strict';
|
|
|
|
//<editor-fold desc="Shims">
|
|
if (!String.prototype.includes) {
|
|
(function () {
|
|
'use strict'; // needed to support `apply`/`call` with `undefined`/`null`
|
|
var toString = {}.toString;
|
|
var defineProperty = (function () {
|
|
// IE 8 only supports `Object.defineProperty` on DOM elements
|
|
try {
|
|
var object = {};
|
|
var $defineProperty = Object.defineProperty;
|
|
var result = $defineProperty(object, object, object) && $defineProperty;
|
|
} catch (error) {
|
|
}
|
|
return result;
|
|
}());
|
|
var indexOf = ''.indexOf;
|
|
var includes = function (search) {
|
|
if (this == null) {
|
|
throw new TypeError();
|
|
}
|
|
var string = String(this);
|
|
if (search && toString.call(search) == '[object RegExp]') {
|
|
throw new TypeError();
|
|
}
|
|
var stringLength = string.length;
|
|
var searchString = String(search);
|
|
var searchLength = searchString.length;
|
|
var position = arguments.length > 1 ? arguments[1] : undefined;
|
|
// `ToInteger`
|
|
var pos = position ? Number(position) : 0;
|
|
if (pos != pos) { // better `isNaN`
|
|
pos = 0;
|
|
}
|
|
var start = Math.min(Math.max(pos, 0), stringLength);
|
|
// Avoid the `indexOf` call if no match is possible
|
|
if (searchLength + start > stringLength) {
|
|
return false;
|
|
}
|
|
return indexOf.call(string, searchString, pos) != -1;
|
|
};
|
|
if (defineProperty) {
|
|
defineProperty(String.prototype, 'includes', {
|
|
'value': includes,
|
|
'configurable': true,
|
|
'writable': true
|
|
});
|
|
} else {
|
|
String.prototype.includes = includes;
|
|
}
|
|
}());
|
|
}
|
|
|
|
if (!String.prototype.startsWith) {
|
|
(function () {
|
|
'use strict'; // needed to support `apply`/`call` with `undefined`/`null`
|
|
var defineProperty = (function () {
|
|
// IE 8 only supports `Object.defineProperty` on DOM elements
|
|
try {
|
|
var object = {};
|
|
var $defineProperty = Object.defineProperty;
|
|
var result = $defineProperty(object, object, object) && $defineProperty;
|
|
} catch (error) {
|
|
}
|
|
return result;
|
|
}());
|
|
var toString = {}.toString;
|
|
var startsWith = function (search) {
|
|
if (this == null) {
|
|
throw new TypeError();
|
|
}
|
|
var string = String(this);
|
|
if (search && toString.call(search) == '[object RegExp]') {
|
|
throw new TypeError();
|
|
}
|
|
var stringLength = string.length;
|
|
var searchString = String(search);
|
|
var searchLength = searchString.length;
|
|
var position = arguments.length > 1 ? arguments[1] : undefined;
|
|
// `ToInteger`
|
|
var pos = position ? Number(position) : 0;
|
|
if (pos != pos) { // better `isNaN`
|
|
pos = 0;
|
|
}
|
|
var start = Math.min(Math.max(pos, 0), stringLength);
|
|
// Avoid the `indexOf` call if no match is possible
|
|
if (searchLength + start > stringLength) {
|
|
return false;
|
|
}
|
|
var index = -1;
|
|
while (++index < searchLength) {
|
|
if (string.charCodeAt(start + index) != searchString.charCodeAt(index)) {
|
|
return false;
|
|
}
|
|
}
|
|
return true;
|
|
};
|
|
if (defineProperty) {
|
|
defineProperty(String.prototype, 'startsWith', {
|
|
'value': startsWith,
|
|
'configurable': true,
|
|
'writable': true
|
|
});
|
|
} else {
|
|
String.prototype.startsWith = startsWith;
|
|
}
|
|
}());
|
|
}
|
|
|
|
if (!Object.keys) {
|
|
Object.keys = function (
|
|
o, // object
|
|
k, // key
|
|
r // result array
|
|
){
|
|
// initialize object and result
|
|
r=[];
|
|
// iterate over object keys
|
|
for (k in o)
|
|
// fill result array with non-prototypical keys
|
|
r.hasOwnProperty.call(o, k) && r.push(k);
|
|
// return result
|
|
return r;
|
|
};
|
|
}
|
|
|
|
$.fn.triggerNative = function (eventName) {
|
|
var el = this[0],
|
|
event;
|
|
|
|
if (el.dispatchEvent) {
|
|
if (typeof Event === 'function') {
|
|
// For modern browsers
|
|
event = new Event(eventName, {
|
|
bubbles: true
|
|
});
|
|
} else {
|
|
// For IE since it doesn't support Event constructor
|
|
event = document.createEvent('Event');
|
|
event.initEvent(eventName, true, false);
|
|
}
|
|
|
|
el.dispatchEvent(event);
|
|
} else {
|
|
if (el.fireEvent) {
|
|
event = document.createEventObject();
|
|
event.eventType = eventName;
|
|
el.fireEvent('on' + eventName, event);
|
|
}
|
|
|
|
this.trigger(eventName);
|
|
}
|
|
};
|
|
//</editor-fold>
|
|
|
|
// Case insensitive contains search
|
|
$.expr[':'].icontains = function (obj, index, meta) {
|
|
var $obj = $(obj);
|
|
var haystack = ($obj.data('tokens') || $obj.text()).toUpperCase();
|
|
return haystack.includes(meta[3].toUpperCase());
|
|
};
|
|
|
|
// Case insensitive begins search
|
|
$.expr[':'].ibegins = function (obj, index, meta) {
|
|
var $obj = $(obj);
|
|
var haystack = ($obj.data('tokens') || $obj.text()).toUpperCase();
|
|
return haystack.startsWith(meta[3].toUpperCase());
|
|
};
|
|
|
|
// Case and accent insensitive contains search
|
|
$.expr[':'].aicontains = function (obj, index, meta) {
|
|
var $obj = $(obj);
|
|
var haystack = ($obj.data('tokens') || $obj.data('normalizedText') || $obj.text()).toUpperCase();
|
|
return haystack.includes(meta[3].toUpperCase());
|
|
};
|
|
|
|
// Case and accent insensitive begins search
|
|
$.expr[':'].aibegins = function (obj, index, meta) {
|
|
var $obj = $(obj);
|
|
var haystack = ($obj.data('tokens') || $obj.data('normalizedText') || $obj.text()).toUpperCase();
|
|
return haystack.startsWith(meta[3].toUpperCase());
|
|
};
|
|
|
|
/**
|
|
* Remove all diatrics from the given text.
|
|
* @access private
|
|
* @param {String} text
|
|
* @returns {String}
|
|
*/
|
|
function normalizeToBase(text) {
|
|
var rExps = [
|
|
{re: /[\xC0-\xC6]/g, ch: "A"},
|
|
{re: /[\xE0-\xE6]/g, ch: "a"},
|
|
{re: /[\xC8-\xCB]/g, ch: "E"},
|
|
{re: /[\xE8-\xEB]/g, ch: "e"},
|
|
{re: /[\xCC-\xCF]/g, ch: "I"},
|
|
{re: /[\xEC-\xEF]/g, ch: "i"},
|
|
{re: /[\xD2-\xD6]/g, ch: "O"},
|
|
{re: /[\xF2-\xF6]/g, ch: "o"},
|
|
{re: /[\xD9-\xDC]/g, ch: "U"},
|
|
{re: /[\xF9-\xFC]/g, ch: "u"},
|
|
{re: /[\xC7-\xE7]/g, ch: "c"},
|
|
{re: /[\xD1]/g, ch: "N"},
|
|
{re: /[\xF1]/g, ch: "n"}
|
|
];
|
|
$.each(rExps, function () {
|
|
text = text.replace(this.re, this.ch);
|
|
});
|
|
return text;
|
|
}
|
|
|
|
|
|
function htmlEscape(html) {
|
|
var escapeMap = {
|
|
'&': '&',
|
|
'<': '<',
|
|
'>': '>',
|
|
'"': '"',
|
|
"'": ''',
|
|
'`': '`'
|
|
};
|
|
var source = '(?:' + Object.keys(escapeMap).join('|') + ')',
|
|
testRegexp = new RegExp(source),
|
|
replaceRegexp = new RegExp(source, 'g'),
|
|
string = html == null ? '' : '' + html;
|
|
return testRegexp.test(string) ? string.replace(replaceRegexp, function (match) {
|
|
return escapeMap[match];
|
|
}) : string;
|
|
}
|
|
|
|
var Selectpicker = function (element, options, e) {
|
|
if (e) {
|
|
e.stopPropagation();
|
|
e.preventDefault();
|
|
}
|
|
|
|
this.$element = $(element);
|
|
this.$newElement = null;
|
|
this.$button = null;
|
|
this.$menu = null;
|
|
this.$lis = null;
|
|
this.options = options;
|
|
|
|
// If we have no title yet, try to pull it from the html title attribute (jQuery doesnt' pick it up as it's not a
|
|
// data-attribute)
|
|
if (this.options.title === null) {
|
|
this.options.title = this.$element.attr('title');
|
|
}
|
|
|
|
//Expose public methods
|
|
this.val = Selectpicker.prototype.val;
|
|
this.render = Selectpicker.prototype.render;
|
|
this.refresh = Selectpicker.prototype.refresh;
|
|
this.setStyle = Selectpicker.prototype.setStyle;
|
|
this.selectAll = Selectpicker.prototype.selectAll;
|
|
this.deselectAll = Selectpicker.prototype.deselectAll;
|
|
this.destroy = Selectpicker.prototype.destroy;
|
|
this.remove = Selectpicker.prototype.remove;
|
|
this.show = Selectpicker.prototype.show;
|
|
this.hide = Selectpicker.prototype.hide;
|
|
|
|
this.init();
|
|
};
|
|
|
|
Selectpicker.VERSION = '1.10.0';
|
|
|
|
// part of this is duplicated in i18n/defaults-en_US.js. Make sure to update both.
|
|
Selectpicker.DEFAULTS = {
|
|
noneSelectedText: 'Nothing selected',
|
|
noneResultsText: 'No results matched {0}',
|
|
countSelectedText: function (numSelected, numTotal) {
|
|
return (numSelected == 1) ? "{0} item selected" : "{0} items selected";
|
|
},
|
|
maxOptionsText: function (numAll, numGroup) {
|
|
return [
|
|
(numAll == 1) ? 'Limit reached ({n} item max)' : 'Limit reached ({n} items max)',
|
|
(numGroup == 1) ? 'Group limit reached ({n} item max)' : 'Group limit reached ({n} items max)'
|
|
];
|
|
},
|
|
selectAllText: 'Select All',
|
|
deselectAllText: 'Deselect All',
|
|
doneButton: false,
|
|
doneButtonText: 'Close',
|
|
multipleSeparator: ', ',
|
|
styleBase: 'btn',
|
|
style: 'btn-default',
|
|
size: 'auto',
|
|
title: null,
|
|
selectedTextFormat: 'values',
|
|
width: false,
|
|
container: false,
|
|
hideDisabled: false,
|
|
showSubtext: false,
|
|
showIcon: true,
|
|
showContent: true,
|
|
dropupAuto: true,
|
|
header: false,
|
|
liveSearch: false,
|
|
liveSearchPlaceholder: null,
|
|
liveSearchNormalize: false,
|
|
liveSearchStyle: 'contains',
|
|
actionsBox: false,
|
|
iconBase: 'glyphicon',
|
|
tickIcon: 'glyphicon-ok',
|
|
showTick: false,
|
|
template: {
|
|
caret: '<span class="caret"></span>'
|
|
},
|
|
maxOptions: false,
|
|
mobile: false,
|
|
selectOnTab: false,
|
|
dropdownAlignRight: false
|
|
};
|
|
|
|
Selectpicker.prototype = {
|
|
|
|
constructor: Selectpicker,
|
|
|
|
init: function () {
|
|
var that = this,
|
|
id = this.$element.attr('id');
|
|
|
|
this.$element.addClass('bs-select-hidden');
|
|
|
|
// store originalIndex (key) and newIndex (value) in this.liObj for fast accessibility
|
|
// allows us to do this.$lis.eq(that.liObj[index]) instead of this.$lis.filter('[data-original-index="' + index + '"]')
|
|
this.liObj = {};
|
|
this.multiple = this.$element.prop('multiple');
|
|
this.autofocus = this.$element.prop('autofocus');
|
|
this.$newElement = this.createView();
|
|
this.$element
|
|
.after(this.$newElement)
|
|
.appendTo(this.$newElement);
|
|
this.$button = this.$newElement.children('button');
|
|
this.$menu = this.$newElement.children('.dropdown-menu');
|
|
this.$menuInner = this.$menu.children('.inner');
|
|
this.$searchbox = this.$menu.find('input');
|
|
|
|
this.$element.removeClass('bs-select-hidden');
|
|
|
|
if (this.options.dropdownAlignRight)
|
|
this.$menu.addClass('dropdown-menu-right');
|
|
|
|
if (typeof id !== 'undefined') {
|
|
this.$button.attr('data-id', id);
|
|
$('label[for="' + id + '"]').click(function (e) {
|
|
e.preventDefault();
|
|
that.$button.focus();
|
|
});
|
|
}
|
|
|
|
this.checkDisabled();
|
|
this.clickListener();
|
|
if (this.options.liveSearch) this.liveSearchListener();
|
|
this.render();
|
|
this.setStyle();
|
|
this.setWidth();
|
|
if (this.options.container) this.selectPosition();
|
|
this.$menu.data('this', this);
|
|
this.$newElement.data('this', this);
|
|
if (this.options.mobile) this.mobile();
|
|
|
|
this.$newElement.on({
|
|
'hide.bs.dropdown': function (e) {
|
|
that.$element.trigger('hide.bs.select', e);
|
|
},
|
|
'hidden.bs.dropdown': function (e) {
|
|
that.$element.trigger('hidden.bs.select', e);
|
|
},
|
|
'show.bs.dropdown': function (e) {
|
|
that.$element.trigger('show.bs.select', e);
|
|
},
|
|
'shown.bs.dropdown': function (e) {
|
|
that.$element.trigger('shown.bs.select', e);
|
|
}
|
|
});
|
|
|
|
if (that.$element[0].hasAttribute('required')) {
|
|
this.$element.on('invalid', function () {
|
|
that.$button
|
|
.addClass('bs-invalid')
|
|
.focus();
|
|
|
|
that.$element.on({
|
|
'focus.bs.select': function () {
|
|
that.$button.focus();
|
|
that.$element.off('focus.bs.select');
|
|
},
|
|
'shown.bs.select': function () {
|
|
that.$element
|
|
.val(that.$element.val()) // set the value to hide the validation message in Chrome when menu is opened
|
|
.off('shown.bs.select');
|
|
},
|
|
'rendered.bs.select': function () {
|
|
// if select is no longer invalid, remove the bs-invalid class
|
|
if (this.validity.valid) that.$button.removeClass('bs-invalid');
|
|
that.$element.off('rendered.bs.select');
|
|
}
|
|
});
|
|
|
|
});
|
|
}
|
|
|
|
setTimeout(function () {
|
|
that.$element.trigger('loaded.bs.select');
|
|
});
|
|
},
|
|
|
|
createDropdown: function () {
|
|
// Options
|
|
// If we are multiple or showTick option is set, then add the show-tick class
|
|
var showTick = (this.multiple || this.options.showTick) ? ' show-tick' : '',
|
|
inputGroup = this.$element.parent().hasClass('input-group') ? ' input-group-btn' : '',
|
|
autofocus = this.autofocus ? ' autofocus' : '';
|
|
// Elements
|
|
var header = this.options.header ? '<div class="popover-title"><button type="button" class="close" aria-hidden="true">×</button>' + this.options.header + '</div>' : '';
|
|
var searchbox = this.options.liveSearch ?
|
|
'<div class="bs-searchbox">' +
|
|
'<input type="text" class="form-control" autocomplete="off"' +
|
|
(null === this.options.liveSearchPlaceholder ? '' : ' placeholder="' + htmlEscape(this.options.liveSearchPlaceholder) + '"') + '>' +
|
|
'</div>'
|
|
: '';
|
|
var actionsbox = this.multiple && this.options.actionsBox ?
|
|
'<div class="bs-actionsbox">' +
|
|
'<div class="btn-group btn-group-sm btn-block">' +
|
|
'<button type="button" class="actions-btn bs-select-all btn btn-default">' +
|
|
this.options.selectAllText +
|
|
'</button>' +
|
|
'<button type="button" class="actions-btn bs-deselect-all btn btn-default">' +
|
|
this.options.deselectAllText +
|
|
'</button>' +
|
|
'</div>' +
|
|
'</div>'
|
|
: '';
|
|
var donebutton = this.multiple && this.options.doneButton ?
|
|
'<div class="bs-donebutton">' +
|
|
'<div class="btn-group btn-block">' +
|
|
'<button type="button" class="btn btn-sm btn-default">' +
|
|
this.options.doneButtonText +
|
|
'</button>' +
|
|
'</div>' +
|
|
'</div>'
|
|
: '';
|
|
var drop =
|
|
'<div class="btn-group bootstrap-select' + showTick + inputGroup + '">' +
|
|
'<button type="button" class="' + this.options.styleBase + ' dropdown-toggle" data-toggle="dropdown"' + autofocus + '>' +
|
|
'<span class="filter-option pull-left"></span> ' +
|
|
'<span class="bs-caret">' +
|
|
this.options.template.caret +
|
|
'</span>' +
|
|
'</button>' +
|
|
'<div class="dropdown-menu open">' +
|
|
header +
|
|
searchbox +
|
|
actionsbox +
|
|
'<ul class="dropdown-menu inner" role="menu">' +
|
|
'</ul>' +
|
|
donebutton +
|
|
'</div>' +
|
|
'</div>';
|
|
|
|
return $(drop);
|
|
},
|
|
|
|
createView: function () {
|
|
var $drop = this.createDropdown(),
|
|
li = this.createLi();
|
|
|
|
$drop.find('ul')[0].innerHTML = li;
|
|
return $drop;
|
|
},
|
|
|
|
reloadLi: function () {
|
|
//Remove all children.
|
|
this.destroyLi();
|
|
//Re build
|
|
var li = this.createLi();
|
|
this.$menuInner[0].innerHTML = li;
|
|
},
|
|
|
|
destroyLi: function () {
|
|
this.$menu.find('li').remove();
|
|
},
|
|
|
|
createLi: function () {
|
|
var that = this,
|
|
_li = [],
|
|
optID = 0,
|
|
titleOption = document.createElement('option'),
|
|
liIndex = -1; // increment liIndex whenever a new <li> element is created to ensure liObj is correct
|
|
|
|
// Helper functions
|
|
/**
|
|
* @param content
|
|
* @param [index]
|
|
* @param [classes]
|
|
* @param [optgroup]
|
|
* @returns {string}
|
|
*/
|
|
var generateLI = function (content, index, classes, optgroup) {
|
|
return '<li' +
|
|
((typeof classes !== 'undefined' & '' !== classes) ? ' class="' + classes + '"' : '') +
|
|
((typeof index !== 'undefined' & null !== index) ? ' data-original-index="' + index + '"' : '') +
|
|
((typeof optgroup !== 'undefined' & null !== optgroup) ? 'data-optgroup="' + optgroup + '"' : '') +
|
|
'>' + content + '</li>';
|
|
};
|
|
|
|
/**
|
|
* @param text
|
|
* @param [classes]
|
|
* @param [inline]
|
|
* @param [tokens]
|
|
* @returns {string}
|
|
*/
|
|
var generateA = function (text, classes, inline, tokens) {
|
|
return '<a tabindex="0"' +
|
|
(typeof classes !== 'undefined' ? ' class="' + classes + '"' : '') +
|
|
(typeof inline !== 'undefined' ? ' style="' + inline + '"' : '') +
|
|
(that.options.liveSearchNormalize ? ' data-normalized-text="' + normalizeToBase(htmlEscape(text)) + '"' : '') +
|
|
(typeof tokens !== 'undefined' || tokens !== null ? ' data-tokens="' + tokens + '"' : '') +
|
|
'>' + text +
|
|
'<span class="' + that.options.iconBase + ' ' + that.options.tickIcon + ' check-mark"></span>' +
|
|
'</a>';
|
|
};
|
|
|
|
if (this.options.title && !this.multiple) {
|
|
// this option doesn't create a new <li> element, but does add a new option, so liIndex is decreased
|
|
// since liObj is recalculated on every refresh, liIndex needs to be decreased even if the titleOption is already appended
|
|
liIndex--;
|
|
|
|
if (!this.$element.find('.bs-title-option').length) {
|
|
// Use native JS to prepend option (faster)
|
|
var element = this.$element[0];
|
|
titleOption.className = 'bs-title-option';
|
|
titleOption.appendChild(document.createTextNode(this.options.title));
|
|
titleOption.value = '';
|
|
element.insertBefore(titleOption, element.firstChild);
|
|
// Check if selected attribute is already set on an option. If not, select the titleOption option.
|
|
if ($(element.options[element.selectedIndex]).attr('selected') === undefined) titleOption.selected = true;
|
|
}
|
|
}
|
|
|
|
this.$element.find('option').each(function (index) {
|
|
var $this = $(this);
|
|
|
|
liIndex++;
|
|
|
|
if ($this.hasClass('bs-title-option')) return;
|
|
|
|
// Get the class and text for the option
|
|
var optionClass = this.className || '',
|
|
inline = this.style.cssText,
|
|
text = $this.data('content') ? $this.data('content') : $this.html(),
|
|
tokens = $this.data('tokens') ? $this.data('tokens') : null,
|
|
subtext = typeof $this.data('subtext') !== 'undefined' ? '<small class="text-muted">' + $this.data('subtext') + '</small>' : '',
|
|
icon = typeof $this.data('icon') !== 'undefined' ? '<span class="' + that.options.iconBase + ' ' + $this.data('icon') + '"></span> ' : '',
|
|
isOptgroup = this.parentNode.tagName === 'OPTGROUP',
|
|
isDisabled = this.disabled || (isOptgroup && this.parentNode.disabled);
|
|
|
|
if (icon !== '' && isDisabled) {
|
|
icon = '<span>' + icon + '</span>';
|
|
}
|
|
|
|
if (that.options.hideDisabled && isDisabled && !isOptgroup) {
|
|
liIndex--;
|
|
return;
|
|
}
|
|
|
|
if (!$this.data('content')) {
|
|
// Prepend any icon and append any subtext to the main text.
|
|
text = icon + '<span class="text">' + text + subtext + '</span>';
|
|
}
|
|
|
|
if (isOptgroup && $this.data('divider') !== true) {
|
|
var optGroupClass = ' ' + this.parentNode.className || '';
|
|
|
|
if ($this.index() === 0) { // Is it the first option of the optgroup?
|
|
optID += 1;
|
|
|
|
// Get the opt group label
|
|
var label = this.parentNode.label,
|
|
labelSubtext = typeof $this.parent().data('subtext') !== 'undefined' ? '<small class="text-muted">' + $this.parent().data('subtext') + '</small>' : '',
|
|
labelIcon = $this.parent().data('icon') ? '<span class="' + that.options.iconBase + ' ' + $this.parent().data('icon') + '"></span> ' : '';
|
|
|
|
label = labelIcon + '<span class="text">' + label + labelSubtext + '</span>';
|
|
|
|
if (index !== 0 && _li.length > 0) { // Is it NOT the first option of the select && are there elements in the dropdown?
|
|
liIndex++;
|
|
_li.push(generateLI('', null, 'divider', optID + 'div'));
|
|
}
|
|
liIndex++;
|
|
_li.push(generateLI(label, null, 'dropdown-header' + optGroupClass, optID));
|
|
}
|
|
|
|
if (that.options.hideDisabled && isDisabled) {
|
|
liIndex--;
|
|
return;
|
|
}
|
|
|
|
_li.push(generateLI(generateA(text, 'opt ' + optionClass + optGroupClass, inline, tokens), index, '', optID));
|
|
} else if ($this.data('divider') === true) {
|
|
_li.push(generateLI('', index, 'divider'));
|
|
} else if ($this.data('hidden') === true) {
|
|
_li.push(generateLI(generateA(text, optionClass, inline, tokens), index, 'hidden is-hidden'));
|
|
} else {
|
|
if (this.previousElementSibling && this.previousElementSibling.tagName === 'OPTGROUP') {
|
|
liIndex++;
|
|
_li.push(generateLI('', null, 'divider', optID + 'div'));
|
|
}
|
|
_li.push(generateLI(generateA(text, optionClass, inline, tokens), index));
|
|
}
|
|
|
|
that.liObj[index] = liIndex;
|
|
});
|
|
|
|
//If we are not multiple, we don't have a selected item, and we don't have a title, select the first element so something is set in the button
|
|
if (!this.multiple && this.$element.find('option:selected').length === 0 && !this.options.title) {
|
|
this.$element.find('option').eq(0).prop('selected', true).attr('selected', 'selected');
|
|
}
|
|
|
|
return _li.join('');
|
|
},
|
|
|
|
findLis: function () {
|
|
if (this.$lis == null) this.$lis = this.$menu.find('li');
|
|
return this.$lis;
|
|
},
|
|
|
|
/**
|
|
* @param [updateLi] defaults to true
|
|
*/
|
|
render: function (updateLi) {
|
|
var that = this,
|
|
notDisabled;
|
|
|
|
//Update the LI to match the SELECT
|
|
if (updateLi !== false) {
|
|
this.$element.find('option').each(function (index) {
|
|
var $lis = that.findLis().eq(that.liObj[index]);
|
|
|
|
that.setDisabled(index, this.disabled || this.parentNode.tagName === 'OPTGROUP' && this.parentNode.disabled, $lis);
|
|
that.setSelected(index, this.selected, $lis);
|
|
});
|
|
}
|
|
|
|
this.tabIndex();
|
|
|
|
var selectedItems = this.$element.find('option').map(function () {
|
|
if (this.selected) {
|
|
if (that.options.hideDisabled && (this.disabled || this.parentNode.tagName === 'OPTGROUP' && this.parentNode.disabled)) return;
|
|
|
|
var $this = $(this),
|
|
icon = $this.data('icon') && that.options.showIcon ? '<i class="' + that.options.iconBase + ' ' + $this.data('icon') + '"></i> ' : '',
|
|
subtext;
|
|
|
|
if (that.options.showSubtext && $this.data('subtext') && !that.multiple) {
|
|
subtext = ' <small class="text-muted">' + $this.data('subtext') + '</small>';
|
|
} else {
|
|
subtext = '';
|
|
}
|
|
if (typeof $this.attr('title') !== 'undefined') {
|
|
return $this.attr('title');
|
|
} else if ($this.data('content') && that.options.showContent) {
|
|
return $this.data('content');
|
|
} else {
|
|
return icon + $this.html() + subtext;
|
|
}
|
|
}
|
|
}).toArray();
|
|
|
|
//Fixes issue in IE10 occurring when no default option is selected and at least one option is disabled
|
|
//Convert all the values into a comma delimited string
|
|
var title = !this.multiple ? selectedItems[0] : selectedItems.join(this.options.multipleSeparator);
|
|
|
|
//If this is multi select, and the selectText type is count, the show 1 of 2 selected etc..
|
|
if (this.multiple && this.options.selectedTextFormat.indexOf('count') > -1) {
|
|
var max = this.options.selectedTextFormat.split('>');
|
|
if ((max.length > 1 && selectedItems.length > max[1]) || (max.length == 1 && selectedItems.length >= 2)) {
|
|
notDisabled = this.options.hideDisabled ? ', [disabled]' : '';
|
|
var totalCount = this.$element.find('option').not('[data-divider="true"], [data-hidden="true"]' + notDisabled).length,
|
|
tr8nText = (typeof this.options.countSelectedText === 'function') ? this.options.countSelectedText(selectedItems.length, totalCount) : this.options.countSelectedText;
|
|
title = tr8nText.replace('{0}', selectedItems.length.toString()).replace('{1}', totalCount.toString());
|
|
}
|
|
}
|
|
|
|
if (this.options.title == undefined) {
|
|
this.options.title = this.$element.attr('title');
|
|
}
|
|
|
|
if (this.options.selectedTextFormat == 'static') {
|
|
title = this.options.title;
|
|
}
|
|
|
|
//If we dont have a title, then use the default, or if nothing is set at all, use the not selected text
|
|
if (!title) {
|
|
title = typeof this.options.title !== 'undefined' ? this.options.title : this.options.noneSelectedText;
|
|
}
|
|
|
|
//strip all html-tags and trim the result
|
|
this.$button.attr('title', $.trim(title.replace(/<[^>]*>?/g, '')));
|
|
this.$button.children('.filter-option').html(title);
|
|
|
|
this.$element.trigger('rendered.bs.select');
|
|
},
|
|
|
|
/**
|
|
* @param [style]
|
|
* @param [status]
|
|
*/
|
|
setStyle: function (style, status) {
|
|
if (this.$element.attr('class')) {
|
|
this.$newElement.addClass(this.$element.attr('class').replace(/selectpicker|mobile-device|bs-select-hidden|validate\[.*\]/gi, ''));
|
|
}
|
|
|
|
var buttonClass = style ? style : this.options.style;
|
|
|
|
if (status == 'add') {
|
|
this.$button.addClass(buttonClass);
|
|
} else if (status == 'remove') {
|
|
this.$button.removeClass(buttonClass);
|
|
} else {
|
|
this.$button.removeClass(this.options.style);
|
|
this.$button.addClass(buttonClass);
|
|
}
|
|
},
|
|
|
|
liHeight: function (refresh) {
|
|
if (!refresh && (this.options.size === false || this.sizeInfo)) return;
|
|
|
|
var newElement = document.createElement('div'),
|
|
menu = document.createElement('div'),
|
|
menuInner = document.createElement('ul'),
|
|
divider = document.createElement('li'),
|
|
li = document.createElement('li'),
|
|
a = document.createElement('a'),
|
|
text = document.createElement('span'),
|
|
header = this.options.header && this.$menu.find('.popover-title').length > 0 ? this.$menu.find('.popover-title')[0].cloneNode(true) : null,
|
|
search = this.options.liveSearch ? document.createElement('div') : null,
|
|
actions = this.options.actionsBox && this.multiple && this.$menu.find('.bs-actionsbox').length > 0 ? this.$menu.find('.bs-actionsbox')[0].cloneNode(true) : null,
|
|
doneButton = this.options.doneButton && this.multiple && this.$menu.find('.bs-donebutton').length > 0 ? this.$menu.find('.bs-donebutton')[0].cloneNode(true) : null;
|
|
|
|
text.className = 'text';
|
|
newElement.className = this.$menu[0].parentNode.className + ' open';
|
|
menu.className = 'dropdown-menu open';
|
|
menuInner.className = 'dropdown-menu inner';
|
|
divider.className = 'divider';
|
|
|
|
text.appendChild(document.createTextNode('Inner text'));
|
|
a.appendChild(text);
|
|
li.appendChild(a);
|
|
menuInner.appendChild(li);
|
|
menuInner.appendChild(divider);
|
|
if (header) menu.appendChild(header);
|
|
if (search) {
|
|
// create a span instead of input as creating an input element is slower
|
|
var input = document.createElement('span');
|
|
search.className = 'bs-searchbox';
|
|
input.className = 'form-control';
|
|
search.appendChild(input);
|
|
menu.appendChild(search);
|
|
}
|
|
if (actions) menu.appendChild(actions);
|
|
menu.appendChild(menuInner);
|
|
if (doneButton) menu.appendChild(doneButton);
|
|
newElement.appendChild(menu);
|
|
|
|
document.body.appendChild(newElement);
|
|
|
|
var liHeight = a.offsetHeight,
|
|
headerHeight = header ? header.offsetHeight : 0,
|
|
searchHeight = search ? search.offsetHeight : 0,
|
|
actionsHeight = actions ? actions.offsetHeight : 0,
|
|
doneButtonHeight = doneButton ? doneButton.offsetHeight : 0,
|
|
dividerHeight = $(divider).outerHeight(true),
|
|
// fall back to jQuery if getComputedStyle is not supported
|
|
menuStyle = typeof getComputedStyle === 'function' ? getComputedStyle(menu) : false,
|
|
$menu = menuStyle ? null : $(menu),
|
|
menuPadding = parseInt(menuStyle ? menuStyle.paddingTop : $menu.css('paddingTop')) +
|
|
parseInt(menuStyle ? menuStyle.paddingBottom : $menu.css('paddingBottom')) +
|
|
parseInt(menuStyle ? menuStyle.borderTopWidth : $menu.css('borderTopWidth')) +
|
|
parseInt(menuStyle ? menuStyle.borderBottomWidth : $menu.css('borderBottomWidth')),
|
|
menuExtras = menuPadding +
|
|
parseInt(menuStyle ? menuStyle.marginTop : $menu.css('marginTop')) +
|
|
parseInt(menuStyle ? menuStyle.marginBottom : $menu.css('marginBottom')) + 2;
|
|
|
|
document.body.removeChild(newElement);
|
|
|
|
this.sizeInfo = {
|
|
liHeight: liHeight,
|
|
headerHeight: headerHeight,
|
|
searchHeight: searchHeight,
|
|
actionsHeight: actionsHeight,
|
|
doneButtonHeight: doneButtonHeight,
|
|
dividerHeight: dividerHeight,
|
|
menuPadding: menuPadding,
|
|
menuExtras: menuExtras
|
|
};
|
|
},
|
|
|
|
setSize: function () {
|
|
this.findLis();
|
|
this.liHeight();
|
|
|
|
if (this.options.header) this.$menu.css('padding-top', 0);
|
|
if (this.options.size === false) return;
|
|
|
|
var that = this,
|
|
$menu = this.$menu,
|
|
$menuInner = this.$menuInner,
|
|
$window = $(window),
|
|
selectHeight = this.$newElement[0].offsetHeight,
|
|
liHeight = this.sizeInfo['liHeight'],
|
|
headerHeight = this.sizeInfo['headerHeight'],
|
|
searchHeight = this.sizeInfo['searchHeight'],
|
|
actionsHeight = this.sizeInfo['actionsHeight'],
|
|
doneButtonHeight = this.sizeInfo['doneButtonHeight'],
|
|
divHeight = this.sizeInfo['dividerHeight'],
|
|
menuPadding = this.sizeInfo['menuPadding'],
|
|
menuExtras = this.sizeInfo['menuExtras'],
|
|
notDisabled = this.options.hideDisabled ? '.disabled' : '',
|
|
menuHeight,
|
|
getHeight,
|
|
selectOffsetTop,
|
|
selectOffsetBot,
|
|
posVert = function () {
|
|
selectOffsetTop = that.$newElement.offset().top - $window.scrollTop();
|
|
selectOffsetBot = $window.height() - selectOffsetTop - selectHeight;
|
|
};
|
|
|
|
posVert();
|
|
|
|
if (this.options.size === 'auto') {
|
|
var getSize = function () {
|
|
var minHeight,
|
|
hasClass = function (className, include) {
|
|
return function (element) {
|
|
if (include) {
|
|
return (element.classList ? element.classList.contains(className) : $(element).hasClass(className));
|
|
} else {
|
|
return !(element.classList ? element.classList.contains(className) : $(element).hasClass(className));
|
|
}
|
|
};
|
|
},
|
|
lis = that.$menuInner[0].getElementsByTagName('li'),
|
|
lisVisible = Array.prototype.filter ? Array.prototype.filter.call(lis, hasClass('hidden', false)) : that.$lis.not('.hidden'),
|
|
optGroup = Array.prototype.filter ? Array.prototype.filter.call(lisVisible, hasClass('dropdown-header', true)) : lisVisible.filter('.dropdown-header');
|
|
|
|
posVert();
|
|
menuHeight = selectOffsetBot - menuExtras;
|
|
|
|
if (that.options.container) {
|
|
if (!$menu.data('height')) $menu.data('height', $menu.height());
|
|
getHeight = $menu.data('height');
|
|
} else {
|
|
getHeight = $menu.height();
|
|
}
|
|
|
|
if (that.options.dropupAuto) {
|
|
that.$newElement.toggleClass('dropup', selectOffsetTop > selectOffsetBot && (menuHeight - menuExtras) < getHeight);
|
|
}
|
|
if (that.$newElement.hasClass('dropup')) {
|
|
menuHeight = selectOffsetTop - menuExtras;
|
|
}
|
|
|
|
if ((lisVisible.length + optGroup.length) > 3) {
|
|
minHeight = liHeight * 3 + menuExtras - 2;
|
|
} else {
|
|
minHeight = 0;
|
|
}
|
|
|
|
$menu.css({
|
|
'max-height': menuHeight + 'px',
|
|
'overflow': 'hidden',
|
|
'min-height': minHeight + headerHeight + searchHeight + actionsHeight + doneButtonHeight + 'px'
|
|
});
|
|
$menuInner.css({
|
|
'max-height': menuHeight - headerHeight - searchHeight - actionsHeight - doneButtonHeight - menuPadding + 'px',
|
|
'overflow-y': 'auto',
|
|
'min-height': Math.max(minHeight - menuPadding, 0) + 'px'
|
|
});
|
|
};
|
|
getSize();
|
|
this.$searchbox.off('input.getSize propertychange.getSize').on('input.getSize propertychange.getSize', getSize);
|
|
$window.off('resize.getSize scroll.getSize').on('resize.getSize scroll.getSize', getSize);
|
|
} else if (this.options.size && this.options.size != 'auto' && this.$lis.not(notDisabled).length > this.options.size) {
|
|
var optIndex = this.$lis.not('.divider').not(notDisabled).children().slice(0, this.options.size).last().parent().index(),
|
|
divLength = this.$lis.slice(0, optIndex + 1).filter('.divider').length;
|
|
menuHeight = liHeight * this.options.size + divLength * divHeight + menuPadding;
|
|
|
|
if (that.options.container) {
|
|
if (!$menu.data('height')) $menu.data('height', $menu.height());
|
|
getHeight = $menu.data('height');
|
|
} else {
|
|
getHeight = $menu.height();
|
|
}
|
|
|
|
if (that.options.dropupAuto) {
|
|
//noinspection JSUnusedAssignment
|
|
this.$newElement.toggleClass('dropup', selectOffsetTop > selectOffsetBot && (menuHeight - menuExtras) < getHeight);
|
|
}
|
|
$menu.css({
|
|
'max-height': menuHeight + headerHeight + searchHeight + actionsHeight + doneButtonHeight + 'px',
|
|
'overflow': 'hidden',
|
|
'min-height': ''
|
|
});
|
|
$menuInner.css({
|
|
'max-height': menuHeight - menuPadding + 'px',
|
|
'overflow-y': 'auto',
|
|
'min-height': ''
|
|
});
|
|
}
|
|
},
|
|
|
|
setWidth: function () {
|
|
if (this.options.width === 'auto') {
|
|
this.$menu.css('min-width', '0');
|
|
|
|
// Get correct width if element is hidden
|
|
var $selectClone = this.$menu.parent().clone().appendTo('body'),
|
|
$selectClone2 = this.options.container ? this.$newElement.clone().appendTo('body') : $selectClone,
|
|
ulWidth = $selectClone.children('.dropdown-menu').outerWidth(),
|
|
btnWidth = $selectClone2.css('width', 'auto').children('button').outerWidth();
|
|
|
|
$selectClone.remove();
|
|
$selectClone2.remove();
|
|
|
|
// Set width to whatever's larger, button title or longest option
|
|
this.$newElement.css('width', Math.max(ulWidth, btnWidth) + 'px');
|
|
} else if (this.options.width === 'fit') {
|
|
// Remove inline min-width so width can be changed from 'auto'
|
|
this.$menu.css('min-width', '');
|
|
this.$newElement.css('width', '').addClass('fit-width');
|
|
} else if (this.options.width) {
|
|
// Remove inline min-width so width can be changed from 'auto'
|
|
this.$menu.css('min-width', '');
|
|
this.$newElement.css('width', this.options.width);
|
|
} else {
|
|
// Remove inline min-width/width so width can be changed
|
|
this.$menu.css('min-width', '');
|
|
this.$newElement.css('width', '');
|
|
}
|
|
// Remove fit-width class if width is changed programmatically
|
|
if (this.$newElement.hasClass('fit-width') && this.options.width !== 'fit') {
|
|
this.$newElement.removeClass('fit-width');
|
|
}
|
|
},
|
|
|
|
selectPosition: function () {
|
|
this.$bsContainer = $('<div class="bs-container" />');
|
|
|
|
var that = this,
|
|
pos,
|
|
actualHeight,
|
|
getPlacement = function ($element) {
|
|
that.$bsContainer.addClass($element.attr('class').replace(/form-control|fit-width/gi, '')).toggleClass('dropup', $element.hasClass('dropup'));
|
|
pos = $element.offset();
|
|
actualHeight = $element.hasClass('dropup') ? 0 : $element[0].offsetHeight;
|
|
that.$bsContainer.css({
|
|
'top': pos.top + actualHeight,
|
|
'left': pos.left,
|
|
'width': $element[0].offsetWidth
|
|
});
|
|
};
|
|
|
|
this.$button.on('click', function () {
|
|
var $this = $(this);
|
|
|
|
if (that.isDisabled()) {
|
|
return;
|
|
}
|
|
|
|
getPlacement(that.$newElement);
|
|
|
|
that.$bsContainer
|
|
.appendTo(that.options.container)
|
|
.toggleClass('open', !$this.hasClass('open'))
|
|
.append(that.$menu);
|
|
});
|
|
|
|
$(window).on('resize scroll', function () {
|
|
getPlacement(that.$newElement);
|
|
});
|
|
|
|
this.$element.on('hide.bs.select', function () {
|
|
that.$menu.data('height', that.$menu.height());
|
|
that.$bsContainer.detach();
|
|
});
|
|
},
|
|
|
|
setSelected: function (index, selected, $lis) {
|
|
if (!$lis) {
|
|
$lis = this.findLis().eq(this.liObj[index]);
|
|
}
|
|
|
|
$lis.toggleClass('selected', selected);
|
|
},
|
|
|
|
setDisabled: function (index, disabled, $lis) {
|
|
if (!$lis) {
|
|
$lis = this.findLis().eq(this.liObj[index]);
|
|
}
|
|
|
|
if (disabled) {
|
|
$lis.addClass('disabled').children('a').attr('href', '#').attr('tabindex', -1);
|
|
} else {
|
|
$lis.removeClass('disabled').children('a').removeAttr('href').attr('tabindex', 0);
|
|
}
|
|
},
|
|
|
|
isDisabled: function () {
|
|
return this.$element[0].disabled;
|
|
},
|
|
|
|
checkDisabled: function () {
|
|
var that = this;
|
|
|
|
if (this.isDisabled()) {
|
|
this.$newElement.addClass('disabled');
|
|
this.$button.addClass('disabled').attr('tabindex', -1);
|
|
} else {
|
|
if (this.$button.hasClass('disabled')) {
|
|
this.$newElement.removeClass('disabled');
|
|
this.$button.removeClass('disabled');
|
|
}
|
|
|
|
if (this.$button.attr('tabindex') == -1 && !this.$element.data('tabindex')) {
|
|
this.$button.removeAttr('tabindex');
|
|
}
|
|
}
|
|
|
|
this.$button.click(function () {
|
|
return !that.isDisabled();
|
|
});
|
|
},
|
|
|
|
tabIndex: function () {
|
|
if (this.$element.data('tabindex') !== this.$element.attr('tabindex') &&
|
|
(this.$element.attr('tabindex') !== -98 && this.$element.attr('tabindex') !== '-98')) {
|
|
this.$element.data('tabindex', this.$element.attr('tabindex'));
|
|
this.$button.attr('tabindex', this.$element.data('tabindex'));
|
|
}
|
|
|
|
this.$element.attr('tabindex', -98);
|
|
},
|
|
|
|
clickListener: function () {
|
|
var that = this,
|
|
$document = $(document);
|
|
|
|
this.$newElement.on('touchstart.dropdown', '.dropdown-menu', function (e) {
|
|
e.stopPropagation();
|
|
});
|
|
|
|
$document.data('spaceSelect', false);
|
|
|
|
this.$button.on('keyup', function (e) {
|
|
if (/(32)/.test(e.keyCode.toString(10)) && $document.data('spaceSelect')) {
|
|
e.preventDefault();
|
|
$document.data('spaceSelect', false);
|
|
}
|
|
});
|
|
|
|
this.$button.on('click', function () {
|
|
that.setSize();
|
|
});
|
|
|
|
this.$element.on('shown.bs.select', function () {
|
|
if (!that.options.liveSearch && !that.multiple) {
|
|
that.$menuInner.find('.selected a').focus();
|
|
} else if (!that.multiple) {
|
|
var selectedIndex = that.liObj[that.$element[0].selectedIndex];
|
|
|
|
if (typeof selectedIndex !== 'number' || that.options.size === false) return;
|
|
|
|
// scroll to selected option
|
|
var offset = that.$lis.eq(selectedIndex)[0].offsetTop - that.$menuInner[0].offsetTop;
|
|
offset = offset - that.$menuInner[0].offsetHeight/2 + that.sizeInfo.liHeight/2;
|
|
that.$menuInner[0].scrollTop = offset;
|
|
}
|
|
});
|
|
|
|
this.$menuInner.on('click', 'li a', function (e) {
|
|
var $this = $(this),
|
|
clickedIndex = $this.parent().data('originalIndex'),
|
|
prevValue = that.$element.val(),
|
|
prevIndex = that.$element.prop('selectedIndex');
|
|
|
|
// Don't close on multi choice menu
|
|
if (that.multiple) {
|
|
e.stopPropagation();
|
|
}
|
|
|
|
e.preventDefault();
|
|
|
|
//Don't run if we have been disabled
|
|
if (!that.isDisabled() && !$this.parent().hasClass('disabled')) {
|
|
var $options = that.$element.find('option'),
|
|
$option = $options.eq(clickedIndex),
|
|
state = $option.prop('selected'),
|
|
$optgroup = $option.parent('optgroup'),
|
|
maxOptions = that.options.maxOptions,
|
|
maxOptionsGrp = $optgroup.data('maxOptions') || false;
|
|
|
|
if (!that.multiple) { // Deselect all others if not multi select box
|
|
$options.prop('selected', false);
|
|
$option.prop('selected', true);
|
|
that.$menuInner.find('.selected').removeClass('selected');
|
|
that.setSelected(clickedIndex, true);
|
|
} else { // Toggle the one we have chosen if we are multi select.
|
|
$option.prop('selected', !state);
|
|
that.setSelected(clickedIndex, !state);
|
|
$this.blur();
|
|
|
|
if (maxOptions !== false || maxOptionsGrp !== false) {
|
|
var maxReached = maxOptions < $options.filter(':selected').length,
|
|
maxReachedGrp = maxOptionsGrp < $optgroup.find('option:selected').length;
|
|
|
|
if ((maxOptions && maxReached) || (maxOptionsGrp && maxReachedGrp)) {
|
|
if (maxOptions && maxOptions == 1) {
|
|
$options.prop('selected', false);
|
|
$option.prop('selected', true);
|
|
that.$menuInner.find('.selected').removeClass('selected');
|
|
that.setSelected(clickedIndex, true);
|
|
} else if (maxOptionsGrp && maxOptionsGrp == 1) {
|
|
$optgroup.find('option:selected').prop('selected', false);
|
|
$option.prop('selected', true);
|
|
var optgroupID = $this.parent().data('optgroup');
|
|
that.$menuInner.find('[data-optgroup="' + optgroupID + '"]').removeClass('selected');
|
|
that.setSelected(clickedIndex, true);
|
|
} else {
|
|
var maxOptionsArr = (typeof that.options.maxOptionsText === 'function') ?
|
|
that.options.maxOptionsText(maxOptions, maxOptionsGrp) : that.options.maxOptionsText,
|
|
maxTxt = maxOptionsArr[0].replace('{n}', maxOptions),
|
|
maxTxtGrp = maxOptionsArr[1].replace('{n}', maxOptionsGrp),
|
|
$notify = $('<div class="notify"></div>');
|
|
// If {var} is set in array, replace it
|
|
/** @deprecated */
|
|
if (maxOptionsArr[2]) {
|
|
maxTxt = maxTxt.replace('{var}', maxOptionsArr[2][maxOptions > 1 ? 0 : 1]);
|
|
maxTxtGrp = maxTxtGrp.replace('{var}', maxOptionsArr[2][maxOptionsGrp > 1 ? 0 : 1]);
|
|
}
|
|
|
|
$option.prop('selected', false);
|
|
|
|
that.$menu.append($notify);
|
|
|
|
if (maxOptions && maxReached) {
|
|
$notify.append($('<div>' + maxTxt + '</div>'));
|
|
that.$element.trigger('maxReached.bs.select');
|
|
}
|
|
|
|
if (maxOptionsGrp && maxReachedGrp) {
|
|
$notify.append($('<div>' + maxTxtGrp + '</div>'));
|
|
that.$element.trigger('maxReachedGrp.bs.select');
|
|
}
|
|
|
|
setTimeout(function () {
|
|
that.setSelected(clickedIndex, false);
|
|
}, 10);
|
|
|
|
$notify.delay(750).fadeOut(300, function () {
|
|
$(this).remove();
|
|
});
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if (!that.multiple) {
|
|
that.$button.focus();
|
|
} else if (that.options.liveSearch) {
|
|
that.$searchbox.focus();
|
|
}
|
|
|
|
// Trigger select 'change'
|
|
if ((prevValue != that.$element.val() && that.multiple) || (prevIndex != that.$element.prop('selectedIndex') && !that.multiple)) {
|
|
// $option.prop('selected') is current option state (selected/unselected). state is previous option state.
|
|
that.$element
|
|
.trigger('changed.bs.select', [clickedIndex, $option.prop('selected'), state])
|
|
.triggerNative('change');
|
|
}
|
|
}
|
|
});
|
|
|
|
this.$menu.on('click', 'li.disabled a, .popover-title, .popover-title :not(.close)', function (e) {
|
|
if (e.currentTarget == this) {
|
|
e.preventDefault();
|
|
e.stopPropagation();
|
|
if (that.options.liveSearch && !$(e.target).hasClass('close')) {
|
|
that.$searchbox.focus();
|
|
} else {
|
|
that.$button.focus();
|
|
}
|
|
}
|
|
});
|
|
|
|
this.$menuInner.on('click', '.divider, .dropdown-header', function (e) {
|
|
e.preventDefault();
|
|
e.stopPropagation();
|
|
if (that.options.liveSearch) {
|
|
that.$searchbox.focus();
|
|
} else {
|
|
that.$button.focus();
|
|
}
|
|
});
|
|
|
|
this.$menu.on('click', '.popover-title .close', function () {
|
|
that.$button.click();
|
|
});
|
|
|
|
this.$searchbox.on('click', function (e) {
|
|
e.stopPropagation();
|
|
});
|
|
|
|
this.$menu.on('click', '.actions-btn', function (e) {
|
|
if (that.options.liveSearch) {
|
|
that.$searchbox.focus();
|
|
} else {
|
|
that.$button.focus();
|
|
}
|
|
|
|
e.preventDefault();
|
|
e.stopPropagation();
|
|
|
|
if ($(this).hasClass('bs-select-all')) {
|
|
that.selectAll();
|
|
} else {
|
|
that.deselectAll();
|
|
}
|
|
});
|
|
|
|
this.$element.change(function () {
|
|
that.render(false);
|
|
});
|
|
},
|
|
|
|
liveSearchListener: function () {
|
|
var that = this,
|
|
$no_results = $('<li class="no-results"></li>');
|
|
|
|
this.$button.on('click.dropdown.data-api touchstart.dropdown.data-api', function () {
|
|
that.$menuInner.find('.active').removeClass('active');
|
|
if (!!that.$searchbox.val()) {
|
|
that.$searchbox.val('');
|
|
that.$lis.not('.is-hidden').removeClass('hidden');
|
|
if (!!$no_results.parent().length) $no_results.remove();
|
|
}
|
|
if (!that.multiple) that.$menuInner.find('.selected').addClass('active');
|
|
setTimeout(function () {
|
|
that.$searchbox.focus();
|
|
}, 10);
|
|
});
|
|
|
|
this.$searchbox.on('click.dropdown.data-api focus.dropdown.data-api touchend.dropdown.data-api', function (e) {
|
|
e.stopPropagation();
|
|
});
|
|
|
|
this.$searchbox.on('input propertychange', function () {
|
|
if (that.$searchbox.val()) {
|
|
var $searchBase = that.$lis.not('.is-hidden').removeClass('hidden').children('a');
|
|
if (that.options.liveSearchNormalize) {
|
|
$searchBase = $searchBase.not(':a' + that._searchStyle() + '("' + normalizeToBase(that.$searchbox.val()) + '")');
|
|
} else {
|
|
$searchBase = $searchBase.not(':' + that._searchStyle() + '("' + that.$searchbox.val() + '")');
|
|
}
|
|
$searchBase.parent().addClass('hidden');
|
|
|
|
that.$lis.filter('.dropdown-header').each(function () {
|
|
var $this = $(this),
|
|
optgroup = $this.data('optgroup');
|
|
|
|
if (that.$lis.filter('[data-optgroup=' + optgroup + ']').not($this).not('.hidden').length === 0) {
|
|
$this.addClass('hidden');
|
|
that.$lis.filter('[data-optgroup=' + optgroup + 'div]').addClass('hidden');
|
|
}
|
|
});
|
|
|
|
var $lisVisible = that.$lis.not('.hidden');
|
|
|
|
// hide divider if first or last visible, or if followed by another divider
|
|
$lisVisible.each(function (index) {
|
|
var $this = $(this);
|
|
|
|
if ($this.hasClass('divider') && (
|
|
$this.index() === $lisVisible.first().index() ||
|
|
$this.index() === $lisVisible.last().index() ||
|
|
$lisVisible.eq(index + 1).hasClass('divider'))) {
|
|
$this.addClass('hidden');
|
|
}
|
|
});
|
|
|
|
if (!that.$lis.not('.hidden, .no-results').length) {
|
|
if (!!$no_results.parent().length) {
|
|
$no_results.remove();
|
|
}
|
|
$no_results.html(that.options.noneResultsText.replace('{0}', '"' + htmlEscape(that.$searchbox.val()) + '"')).show();
|
|
that.$menuInner.append($no_results);
|
|
} else if (!!$no_results.parent().length) {
|
|
$no_results.remove();
|
|
}
|
|
} else {
|
|
that.$lis.not('.is-hidden').removeClass('hidden');
|
|
if (!!$no_results.parent().length) {
|
|
$no_results.remove();
|
|
}
|
|
}
|
|
|
|
that.$lis.filter('.active').removeClass('active');
|
|
if (that.$searchbox.val()) that.$lis.not('.hidden, .divider, .dropdown-header').eq(0).addClass('active').children('a').focus();
|
|
$(this).focus();
|
|
});
|
|
},
|
|
|
|
_searchStyle: function () {
|
|
var styles = {
|
|
begins: 'ibegins',
|
|
startsWith: 'ibegins'
|
|
};
|
|
|
|
return styles[this.options.liveSearchStyle] || 'icontains';
|
|
},
|
|
|
|
val: function (value) {
|
|
if (typeof value !== 'undefined') {
|
|
this.$element.val(value);
|
|
this.render();
|
|
|
|
return this.$element;
|
|
} else {
|
|
return this.$element.val();
|
|
}
|
|
},
|
|
|
|
changeAll: function (status) {
|
|
if (typeof status === 'undefined') status = true;
|
|
|
|
this.findLis();
|
|
|
|
var $options = this.$element.find('option'),
|
|
$lisVisible = this.$lis.not('.divider, .dropdown-header, .disabled, .hidden').toggleClass('selected', status),
|
|
lisVisLen = $lisVisible.length,
|
|
selectedOptions = [];
|
|
|
|
for (var i = 0; i < lisVisLen; i++) {
|
|
var origIndex = $lisVisible[i].getAttribute('data-original-index');
|
|
selectedOptions[selectedOptions.length] = $options.eq(origIndex)[0];
|
|
}
|
|
|
|
$(selectedOptions).prop('selected', status);
|
|
|
|
this.render(false);
|
|
|
|
this.$element
|
|
.trigger('changed.bs.select')
|
|
.triggerNative('change');
|
|
},
|
|
|
|
selectAll: function () {
|
|
return this.changeAll(true);
|
|
},
|
|
|
|
deselectAll: function () {
|
|
return this.changeAll(false);
|
|
},
|
|
|
|
toggle: function (e) {
|
|
e = e || window.event;
|
|
|
|
if (e) e.stopPropagation();
|
|
|
|
this.$button.trigger('click');
|
|
},
|
|
|
|
keydown: function (e) {
|
|
var $this = $(this),
|
|
$parent = $this.is('input') ? $this.parent().parent() : $this.parent(),
|
|
$items,
|
|
that = $parent.data('this'),
|
|
index,
|
|
next,
|
|
first,
|
|
last,
|
|
prev,
|
|
nextPrev,
|
|
prevIndex,
|
|
isActive,
|
|
selector = ':not(.disabled, .hidden, .dropdown-header, .divider)',
|
|
keyCodeMap = {
|
|
32: ' ',
|
|
48: '0',
|
|
49: '1',
|
|
50: '2',
|
|
51: '3',
|
|
52: '4',
|
|
53: '5',
|
|
54: '6',
|
|
55: '7',
|
|
56: '8',
|
|
57: '9',
|
|
59: ';',
|
|
65: 'a',
|
|
66: 'b',
|
|
67: 'c',
|
|
68: 'd',
|
|
69: 'e',
|
|
70: 'f',
|
|
71: 'g',
|
|
72: 'h',
|
|
73: 'i',
|
|
74: 'j',
|
|
75: 'k',
|
|
76: 'l',
|
|
77: 'm',
|
|
78: 'n',
|
|
79: 'o',
|
|
80: 'p',
|
|
81: 'q',
|
|
82: 'r',
|
|
83: 's',
|
|
84: 't',
|
|
85: 'u',
|
|
86: 'v',
|
|
87: 'w',
|
|
88: 'x',
|
|
89: 'y',
|
|
90: 'z',
|
|
96: '0',
|
|
97: '1',
|
|
98: '2',
|
|
99: '3',
|
|
100: '4',
|
|
101: '5',
|
|
102: '6',
|
|
103: '7',
|
|
104: '8',
|
|
105: '9'
|
|
};
|
|
|
|
if (that.options.liveSearch) $parent = $this.parent().parent();
|
|
|
|
if (that.options.container) $parent = that.$menu;
|
|
|
|
$items = $('[role=menu] li', $parent);
|
|
|
|
isActive = that.$newElement.hasClass('open');
|
|
|
|
if (!isActive && (e.keyCode >= 48 && e.keyCode <= 57 || e.keyCode >= 96 && e.keyCode <= 105 || e.keyCode >= 65 && e.keyCode <= 90)) {
|
|
if (!that.options.container) {
|
|
that.setSize();
|
|
that.$menu.parent().addClass('open');
|
|
isActive = true;
|
|
} else {
|
|
that.$button.trigger('click');
|
|
}
|
|
that.$searchbox.focus();
|
|
}
|
|
|
|
if (that.options.liveSearch) {
|
|
if (/(^9$|27)/.test(e.keyCode.toString(10)) && isActive && that.$menu.find('.active').length === 0) {
|
|
e.preventDefault();
|
|
that.$menu.parent().removeClass('open');
|
|
if (that.options.container) that.$newElement.removeClass('open');
|
|
that.$button.focus();
|
|
}
|
|
// $items contains li elements when liveSearch is enabled
|
|
$items = $('[role=menu] li' + selector, $parent);
|
|
if (!$this.val() && !/(38|40)/.test(e.keyCode.toString(10))) {
|
|
if ($items.filter('.active').length === 0) {
|
|
$items = that.$menuInner.find('li');
|
|
if (that.options.liveSearchNormalize) {
|
|
$items = $items.filter(':a' + that._searchStyle() + '(' + normalizeToBase(keyCodeMap[e.keyCode]) + ')');
|
|
} else {
|
|
$items = $items.filter(':' + that._searchStyle() + '(' + keyCodeMap[e.keyCode] + ')');
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if (!$items.length) return;
|
|
|
|
if (/(38|40)/.test(e.keyCode.toString(10))) {
|
|
index = $items.index($items.find('a').filter(':focus').parent());
|
|
first = $items.filter(selector).first().index();
|
|
last = $items.filter(selector).last().index();
|
|
next = $items.eq(index).nextAll(selector).eq(0).index();
|
|
prev = $items.eq(index).prevAll(selector).eq(0).index();
|
|
nextPrev = $items.eq(next).prevAll(selector).eq(0).index();
|
|
|
|
if (that.options.liveSearch) {
|
|
$items.each(function (i) {
|
|
if (!$(this).hasClass('disabled')) {
|
|
$(this).data('index', i);
|
|
}
|
|
});
|
|
index = $items.index($items.filter('.active'));
|
|
first = $items.first().data('index');
|
|
last = $items.last().data('index');
|
|
next = $items.eq(index).nextAll().eq(0).data('index');
|
|
prev = $items.eq(index).prevAll().eq(0).data('index');
|
|
nextPrev = $items.eq(next).prevAll().eq(0).data('index');
|
|
}
|
|
|
|
prevIndex = $this.data('prevIndex');
|
|
|
|
if (e.keyCode == 38) {
|
|
if (that.options.liveSearch) index--;
|
|
if (index != nextPrev && index > prev) index = prev;
|
|
if (index < first) index = first;
|
|
if (index == prevIndex) index = last;
|
|
} else if (e.keyCode == 40) {
|
|
if (that.options.liveSearch) index++;
|
|
if (index == -1) index = 0;
|
|
if (index != nextPrev && index < next) index = next;
|
|
if (index > last) index = last;
|
|
if (index == prevIndex) index = first;
|
|
}
|
|
|
|
$this.data('prevIndex', index);
|
|
|
|
if (!that.options.liveSearch) {
|
|
$items.eq(index).children('a').focus();
|
|
} else {
|
|
e.preventDefault();
|
|
if (!$this.hasClass('dropdown-toggle')) {
|
|
$items.removeClass('active').eq(index).addClass('active').children('a').focus();
|
|
$this.focus();
|
|
}
|
|
}
|
|
|
|
} else if (!$this.is('input')) {
|
|
var keyIndex = [],
|
|
count,
|
|
prevKey;
|
|
|
|
$items.each(function () {
|
|
if (!$(this).hasClass('disabled')) {
|
|
if ($.trim($(this).children('a').text().toLowerCase()).substring(0, 1) == keyCodeMap[e.keyCode]) {
|
|
keyIndex.push($(this).index());
|
|
}
|
|
}
|
|
});
|
|
|
|
count = $(document).data('keycount');
|
|
count++;
|
|
$(document).data('keycount', count);
|
|
|
|
prevKey = $.trim($(':focus').text().toLowerCase()).substring(0, 1);
|
|
|
|
if (prevKey != keyCodeMap[e.keyCode]) {
|
|
count = 1;
|
|
$(document).data('keycount', count);
|
|
} else if (count >= keyIndex.length) {
|
|
$(document).data('keycount', 0);
|
|
if (count > keyIndex.length) count = 1;
|
|
}
|
|
|
|
$items.eq(keyIndex[count - 1]).children('a').focus();
|
|
}
|
|
|
|
// Select focused option if "Enter", "Spacebar" or "Tab" (when selectOnTab is true) are pressed inside the menu.
|
|
if ((/(13|32)/.test(e.keyCode.toString(10)) || (/(^9$)/.test(e.keyCode.toString(10)) && that.options.selectOnTab)) && isActive) {
|
|
if (!/(32)/.test(e.keyCode.toString(10))) e.preventDefault();
|
|
if (!that.options.liveSearch) {
|
|
var elem = $(':focus');
|
|
elem.click();
|
|
// Bring back focus for multiselects
|
|
elem.focus();
|
|
// Prevent screen from scrolling if the user hit the spacebar
|
|
e.preventDefault();
|
|
// Fixes spacebar selection of dropdown items in FF & IE
|
|
$(document).data('spaceSelect', true);
|
|
} else if (!/(32)/.test(e.keyCode.toString(10))) {
|
|
that.$menuInner.find('.active a').click();
|
|
$this.focus();
|
|
}
|
|
$(document).data('keycount', 0);
|
|
}
|
|
|
|
if ((/(^9$|27)/.test(e.keyCode.toString(10)) && isActive && (that.multiple || that.options.liveSearch)) || (/(27)/.test(e.keyCode.toString(10)) && !isActive)) {
|
|
that.$menu.parent().removeClass('open');
|
|
if (that.options.container) that.$newElement.removeClass('open');
|
|
that.$button.focus();
|
|
}
|
|
},
|
|
|
|
mobile: function () {
|
|
this.$element.addClass('mobile-device');
|
|
},
|
|
|
|
refresh: function () {
|
|
this.$lis = null;
|
|
this.liObj = {};
|
|
this.reloadLi();
|
|
this.render();
|
|
this.checkDisabled();
|
|
this.liHeight(true);
|
|
this.setStyle();
|
|
this.setWidth();
|
|
if (this.$lis) this.$searchbox.trigger('propertychange');
|
|
|
|
this.$element.trigger('refreshed.bs.select');
|
|
},
|
|
|
|
hide: function () {
|
|
this.$newElement.hide();
|
|
},
|
|
|
|
show: function () {
|
|
this.$newElement.show();
|
|
},
|
|
|
|
remove: function () {
|
|
this.$newElement.remove();
|
|
this.$element.remove();
|
|
},
|
|
|
|
destroy: function () {
|
|
this.$newElement.before(this.$element).remove();
|
|
|
|
if (this.$bsContainer) {
|
|
this.$bsContainer.remove();
|
|
} else {
|
|
this.$menu.remove();
|
|
}
|
|
|
|
this.$element
|
|
.off('.bs.select')
|
|
.removeData('selectpicker')
|
|
.removeClass('bs-select-hidden selectpicker');
|
|
}
|
|
};
|
|
|
|
// SELECTPICKER PLUGIN DEFINITION
|
|
// ==============================
|
|
function Plugin(option, event) {
|
|
// get the args of the outer function..
|
|
var args = arguments;
|
|
// The arguments of the function are explicitly re-defined from the argument list, because the shift causes them
|
|
// to get lost/corrupted in android 2.3 and IE9 #715 #775
|
|
var _option = option,
|
|
_event = event;
|
|
[].shift.apply(args);
|
|
|
|
var value;
|
|
var chain = this.each(function () {
|
|
var $this = $(this);
|
|
if ($this.is('select')) {
|
|
var data = $this.data('selectpicker'),
|
|
options = typeof _option == 'object' && _option;
|
|
|
|
if (!data) {
|
|
var config = $.extend({}, Selectpicker.DEFAULTS, $.fn.selectpicker.defaults || {}, $this.data(), options);
|
|
config.template = $.extend({}, Selectpicker.DEFAULTS.template, ($.fn.selectpicker.defaults ? $.fn.selectpicker.defaults.template : {}), $this.data().template, options.template);
|
|
$this.data('selectpicker', (data = new Selectpicker(this, config, _event)));
|
|
} else if (options) {
|
|
for (var i in options) {
|
|
if (options.hasOwnProperty(i)) {
|
|
data.options[i] = options[i];
|
|
}
|
|
}
|
|
}
|
|
|
|
if (typeof _option == 'string') {
|
|
if (data[_option] instanceof Function) {
|
|
value = data[_option].apply(data, args);
|
|
} else {
|
|
value = data.options[_option];
|
|
}
|
|
}
|
|
}
|
|
});
|
|
|
|
if (typeof value !== 'undefined') {
|
|
//noinspection JSUnusedAssignment
|
|
return value;
|
|
} else {
|
|
return chain;
|
|
}
|
|
}
|
|
|
|
var old = $.fn.selectpicker;
|
|
$.fn.selectpicker = Plugin;
|
|
$.fn.selectpicker.Constructor = Selectpicker;
|
|
|
|
// SELECTPICKER NO CONFLICT
|
|
// ========================
|
|
$.fn.selectpicker.noConflict = function () {
|
|
$.fn.selectpicker = old;
|
|
return this;
|
|
};
|
|
|
|
$(document)
|
|
.data('keycount', 0)
|
|
.on('keydown.bs.select', '.bootstrap-select [data-toggle=dropdown], .bootstrap-select [role="menu"], .bs-searchbox input', Selectpicker.prototype.keydown)
|
|
.on('focusin.modal', '.bootstrap-select [data-toggle=dropdown], .bootstrap-select [role="menu"], .bs-searchbox input', function (e) {
|
|
e.stopPropagation();
|
|
});
|
|
|
|
// SELECTPICKER DATA-API
|
|
// =====================
|
|
$(window).on('load.bs.select.data-api', function () {
|
|
$('.selectpicker').each(function () {
|
|
var $selectpicker = $(this);
|
|
Plugin.call($selectpicker, $selectpicker.data());
|
|
})
|
|
});
|
|
})(jQuery);
|
|
|
|
|
|
}));
|
|
|
|
/**
|
|
* @author zhixin wen <wenzhixin2010@gmail.com>
|
|
* version: 1.10.1
|
|
* https://github.com/wenzhixin/bootstrap-table/
|
|
*/
|
|
|
|
!function ($) {
|
|
'use strict';
|
|
|
|
// TOOLS DEFINITION
|
|
// ======================
|
|
|
|
var cachedWidth = null;
|
|
|
|
// it only does '%s', and return '' when arguments are undefined
|
|
var sprintf = function (str) {
|
|
var args = arguments,
|
|
flag = true,
|
|
i = 1;
|
|
|
|
str = str.replace(/%s/g, function () {
|
|
var arg = args[i++];
|
|
|
|
if (typeof arg === 'undefined') {
|
|
flag = false;
|
|
return '';
|
|
}
|
|
return arg;
|
|
});
|
|
return flag ? str : '';
|
|
};
|
|
|
|
var getPropertyFromOther = function (list, from, to, value) {
|
|
var result = '';
|
|
$.each(list, function (i, item) {
|
|
if (item[from] === value) {
|
|
result = item[to];
|
|
return false;
|
|
}
|
|
return true;
|
|
});
|
|
return result;
|
|
};
|
|
|
|
var getFieldIndex = function (columns, field) {
|
|
var index = -1;
|
|
|
|
$.each(columns, function (i, column) {
|
|
if (column.field === field) {
|
|
index = i;
|
|
return false;
|
|
}
|
|
return true;
|
|
});
|
|
return index;
|
|
};
|
|
|
|
// http://jsfiddle.net/wenyi/47nz7ez9/3/
|
|
var setFieldIndex = function (columns) {
|
|
var i, j, k,
|
|
totalCol = 0,
|
|
flag = [];
|
|
|
|
for (i = 0; i < columns[0].length; i++) {
|
|
totalCol += columns[0][i].colspan || 1;
|
|
}
|
|
|
|
for (i = 0; i < columns.length; i++) {
|
|
flag[i] = [];
|
|
for (j = 0; j < totalCol; j++) {
|
|
flag[i][j] = false;
|
|
}
|
|
}
|
|
|
|
for (i = 0; i < columns.length; i++) {
|
|
for (j = 0; j < columns[i].length; j++) {
|
|
var r = columns[i][j],
|
|
rowspan = r.rowspan || 1,
|
|
colspan = r.colspan || 1,
|
|
index = $.inArray(false, flag[i]);
|
|
|
|
if (colspan === 1) {
|
|
r.fieldIndex = index;
|
|
// when field is undefined, use index instead
|
|
if (typeof r.field === 'undefined') {
|
|
r.field = index;
|
|
}
|
|
}
|
|
|
|
for (k = 0; k < rowspan; k++) {
|
|
flag[i + k][index] = true;
|
|
}
|
|
for (k = 0; k < colspan; k++) {
|
|
flag[i][index + k] = true;
|
|
}
|
|
}
|
|
}
|
|
};
|
|
|
|
var getScrollBarWidth = function () {
|
|
if (cachedWidth === null) {
|
|
var inner = $('<p/>').addClass('fixed-table-scroll-inner'),
|
|
outer = $('<div/>').addClass('fixed-table-scroll-outer'),
|
|
w1, w2;
|
|
|
|
outer.append(inner);
|
|
$('body').append(outer);
|
|
|
|
w1 = inner[0].offsetWidth;
|
|
outer.css('overflow', 'scroll');
|
|
w2 = inner[0].offsetWidth;
|
|
|
|
if (w1 === w2) {
|
|
w2 = outer[0].clientWidth;
|
|
}
|
|
|
|
outer.remove();
|
|
cachedWidth = w1 - w2;
|
|
}
|
|
return cachedWidth;
|
|
};
|
|
|
|
var calculateObjectValue = function (self, name, args, defaultValue) {
|
|
var func = name;
|
|
|
|
if (typeof name === 'string') {
|
|
// support obj.func1.func2
|
|
var names = name.split('.');
|
|
|
|
if (names.length > 1) {
|
|
func = window;
|
|
$.each(names, function (i, f) {
|
|
func = func[f];
|
|
});
|
|
} else {
|
|
func = window[name];
|
|
}
|
|
}
|
|
if (typeof func === 'object') {
|
|
return func;
|
|
}
|
|
if (typeof func === 'function') {
|
|
return func.apply(self, args);
|
|
}
|
|
if (!func && typeof name === 'string' && sprintf.apply(this, [name].concat(args))) {
|
|
return sprintf.apply(this, [name].concat(args));
|
|
}
|
|
return defaultValue;
|
|
};
|
|
|
|
var compareObjects = function (objectA, objectB, compareLength) {
|
|
// Create arrays of property names
|
|
var objectAProperties = Object.getOwnPropertyNames(objectA),
|
|
objectBProperties = Object.getOwnPropertyNames(objectB),
|
|
propName = '';
|
|
|
|
if (compareLength) {
|
|
// If number of properties is different, objects are not equivalent
|
|
if (objectAProperties.length !== objectBProperties.length) {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
for (var i = 0; i < objectAProperties.length; i++) {
|
|
propName = objectAProperties[i];
|
|
|
|
// If the property is not in the object B properties, continue with the next property
|
|
if ($.inArray(propName, objectBProperties) > -1) {
|
|
// If values of same property are not equal, objects are not equivalent
|
|
if (objectA[propName] !== objectB[propName]) {
|
|
return false;
|
|
}
|
|
}
|
|
}
|
|
|
|
// If we made it this far, objects are considered equivalent
|
|
return true;
|
|
};
|
|
|
|
var escapeHTML = function (text) {
|
|
if (typeof text === 'string') {
|
|
return text
|
|
.replace(/&/g, '&')
|
|
.replace(/</g, '<')
|
|
.replace(/>/g, '>')
|
|
.replace(/"/g, '"')
|
|
.replace(/'/g, ''')
|
|
.replace(/`/g, '`');
|
|
}
|
|
return text;
|
|
};
|
|
|
|
var getRealHeight = function ($el) {
|
|
var height = 0;
|
|
$el.children().each(function () {
|
|
if (height < $(this).outerHeight(true)) {
|
|
height = $(this).outerHeight(true);
|
|
}
|
|
});
|
|
return height;
|
|
};
|
|
|
|
var getRealDataAttr = function (dataAttr) {
|
|
for (var attr in dataAttr) {
|
|
var auxAttr = attr.split(/(?=[A-Z])/).join('-').toLowerCase();
|
|
if (auxAttr !== attr) {
|
|
dataAttr[auxAttr] = dataAttr[attr];
|
|
delete dataAttr[attr];
|
|
}
|
|
}
|
|
|
|
return dataAttr;
|
|
};
|
|
|
|
var getItemField = function (item, field, escape) {
|
|
var value = item;
|
|
|
|
if (typeof field !== 'string' || item.hasOwnProperty(field)) {
|
|
return escape ? escapeHTML(item[field]) : item[field];
|
|
}
|
|
var props = field.split('.');
|
|
for (var p in props) {
|
|
value = value && value[props[p]];
|
|
}
|
|
return escape ? escapeHTML(value) : value;
|
|
};
|
|
|
|
var isIEBrowser = function () {
|
|
return !!(navigator.userAgent.indexOf("MSIE ") > 0 || !!navigator.userAgent.match(/Trident.*rv\:11\./));
|
|
};
|
|
|
|
// BOOTSTRAP TABLE CLASS DEFINITION
|
|
// ======================
|
|
|
|
var BootstrapTable = function (el, options) {
|
|
this.options = options;
|
|
this.$el = $(el);
|
|
this.$el_ = this.$el.clone();
|
|
this.timeoutId_ = 0;
|
|
this.timeoutFooter_ = 0;
|
|
|
|
this.init();
|
|
};
|
|
|
|
BootstrapTable.DEFAULTS = {
|
|
classes: 'table table-hover',
|
|
locale: undefined,
|
|
height: undefined,
|
|
undefinedText: '-',
|
|
sortName: undefined,
|
|
sortOrder: 'asc',
|
|
striped: false,
|
|
columns: [[]],
|
|
data: [],
|
|
dataField: 'rows',
|
|
method: 'get',
|
|
url: undefined,
|
|
ajax: undefined,
|
|
cache: true,
|
|
contentType: 'application/json',
|
|
dataType: 'json',
|
|
ajaxOptions: {},
|
|
queryParams: function (params) {
|
|
return params;
|
|
},
|
|
queryParamsType: 'limit', // undefined
|
|
responseHandler: function (res) {
|
|
return res;
|
|
},
|
|
pagination: false,
|
|
onlyInfoPagination: false,
|
|
sidePagination: 'client', // client or server
|
|
totalRows: 0, // server side need to set
|
|
pageNumber: 1,
|
|
pageSize: 10,
|
|
pageList: [10, 25, 50, 100],
|
|
paginationHAlign: 'right', //right, left
|
|
paginationVAlign: 'bottom', //bottom, top, both
|
|
paginationDetailHAlign: 'left', //right, left
|
|
paginationPreText: '‹',
|
|
paginationNextText: '›',
|
|
search: false,
|
|
searchOnEnterKey: false,
|
|
strictSearch: false,
|
|
searchAlign: 'right',
|
|
selectItemName: 'btSelectItem',
|
|
showHeader: true,
|
|
showFooter: false,
|
|
showColumns: false,
|
|
showPaginationSwitch: false,
|
|
showRefresh: false,
|
|
showToggle: false,
|
|
buttonsAlign: 'right',
|
|
smartDisplay: true,
|
|
escape: false,
|
|
minimumCountColumns: 1,
|
|
idField: undefined,
|
|
uniqueId: undefined,
|
|
cardView: false,
|
|
detailView: false,
|
|
detailFormatter: function (index, row) {
|
|
return '';
|
|
},
|
|
trimOnSearch: true,
|
|
clickToSelect: false,
|
|
singleSelect: false,
|
|
toolbar: undefined,
|
|
toolbarAlign: 'left',
|
|
checkboxHeader: true,
|
|
sortable: true,
|
|
silentSort: true,
|
|
maintainSelected: false,
|
|
searchTimeOut: 500,
|
|
searchText: '',
|
|
iconSize: undefined,
|
|
iconsPrefix: 'glyphicon', // glyphicon of fa (font awesome)
|
|
icons: {
|
|
paginationSwitchDown: 'glyphicon-collapse-down icon-chevron-down',
|
|
paginationSwitchUp: 'glyphicon-collapse-up icon-chevron-up',
|
|
refresh: 'glyphicon-refresh icon-refresh',
|
|
toggle: 'glyphicon-list-alt icon-list-alt',
|
|
columns: 'glyphicon-th icon-th',
|
|
detailOpen: 'glyphicon-plus icon-plus',
|
|
detailClose: 'glyphicon-minus icon-minus'
|
|
},
|
|
|
|
rowStyle: function (row, index) {
|
|
return {};
|
|
},
|
|
|
|
rowAttributes: function (row, index) {
|
|
return {};
|
|
},
|
|
|
|
onAll: function (name, args) {
|
|
return false;
|
|
},
|
|
onClickCell: function (field, value, row, $element) {
|
|
return false;
|
|
},
|
|
onDblClickCell: function (field, value, row, $element) {
|
|
return false;
|
|
},
|
|
onClickRow: function (item, $element) {
|
|
return false;
|
|
},
|
|
onDblClickRow: function (item, $element) {
|
|
return false;
|
|
},
|
|
onSort: function (name, order) {
|
|
return false;
|
|
},
|
|
onCheck: function (row) {
|
|
return false;
|
|
},
|
|
onUncheck: function (row) {
|
|
return false;
|
|
},
|
|
onCheckAll: function (rows) {
|
|
return false;
|
|
},
|
|
onUncheckAll: function (rows) {
|
|
return false;
|
|
},
|
|
onCheckSome: function (rows) {
|
|
return false;
|
|
},
|
|
onUncheckSome: function (rows) {
|
|
return false;
|
|
},
|
|
onLoadSuccess: function (data) {
|
|
return false;
|
|
},
|
|
onLoadError: function (status) {
|
|
return false;
|
|
},
|
|
onColumnSwitch: function (field, checked) {
|
|
return false;
|
|
},
|
|
onPageChange: function (number, size) {
|
|
return false;
|
|
},
|
|
onSearch: function (text) {
|
|
return false;
|
|
},
|
|
onToggle: function (cardView) {
|
|
return false;
|
|
},
|
|
onPreBody: function (data) {
|
|
return false;
|
|
},
|
|
onPostBody: function () {
|
|
return false;
|
|
},
|
|
onPostHeader: function () {
|
|
return false;
|
|
},
|
|
onExpandRow: function (index, row, $detail) {
|
|
return false;
|
|
},
|
|
onCollapseRow: function (index, row) {
|
|
return false;
|
|
},
|
|
onRefreshOptions: function (options) {
|
|
return false;
|
|
},
|
|
onResetView: function () {
|
|
return false;
|
|
}
|
|
};
|
|
|
|
BootstrapTable.LOCALES = [];
|
|
|
|
BootstrapTable.LOCALES['en-US'] = BootstrapTable.LOCALES['en'] = {
|
|
formatLoadingMessage: function () {
|
|
return 'Loading, please wait...';
|
|
},
|
|
formatRecordsPerPage: function (pageNumber) {
|
|
return sprintf('%s records per page', pageNumber);
|
|
},
|
|
formatShowingRows: function (pageFrom, pageTo, totalRows) {
|
|
return sprintf('Showing %s to %s of %s rows', pageFrom, pageTo, totalRows);
|
|
},
|
|
formatDetailPagination: function (totalRows) {
|
|
return sprintf('Showing %s rows', totalRows);
|
|
},
|
|
formatSearch: function () {
|
|
return 'Search';
|
|
},
|
|
formatNoMatches: function () {
|
|
return 'No matching records found';
|
|
},
|
|
formatPaginationSwitch: function () {
|
|
return 'Hide/Show pagination';
|
|
},
|
|
formatRefresh: function () {
|
|
return 'Refresh';
|
|
},
|
|
formatToggle: function () {
|
|
return 'Toggle';
|
|
},
|
|
formatColumns: function () {
|
|
return 'Columns';
|
|
},
|
|
formatAllRows: function () {
|
|
return 'All';
|
|
}
|
|
};
|
|
|
|
$.extend(BootstrapTable.DEFAULTS, BootstrapTable.LOCALES['en-US']);
|
|
|
|
BootstrapTable.COLUMN_DEFAULTS = {
|
|
radio: false,
|
|
checkbox: false,
|
|
checkboxEnabled: true,
|
|
field: undefined,
|
|
title: undefined,
|
|
titleTooltip: undefined,
|
|
'class': undefined,
|
|
align: undefined, // left, right, center
|
|
halign: undefined, // left, right, center
|
|
falign: undefined, // left, right, center
|
|
valign: undefined, // top, middle, bottom
|
|
width: undefined,
|
|
sortable: false,
|
|
order: 'asc', // asc, desc
|
|
visible: true,
|
|
switchable: true,
|
|
clickToSelect: true,
|
|
formatter: undefined,
|
|
footerFormatter: undefined,
|
|
events: undefined,
|
|
sorter: undefined,
|
|
sortName: undefined,
|
|
cellStyle: undefined,
|
|
searchable: true,
|
|
searchFormatter: true,
|
|
cardVisible: true
|
|
};
|
|
|
|
BootstrapTable.EVENTS = {
|
|
'all.bs.table': 'onAll',
|
|
'click-cell.bs.table': 'onClickCell',
|
|
'dbl-click-cell.bs.table': 'onDblClickCell',
|
|
'click-row.bs.table': 'onClickRow',
|
|
'dbl-click-row.bs.table': 'onDblClickRow',
|
|
'sort.bs.table': 'onSort',
|
|
'check.bs.table': 'onCheck',
|
|
'uncheck.bs.table': 'onUncheck',
|
|
'check-all.bs.table': 'onCheckAll',
|
|
'uncheck-all.bs.table': 'onUncheckAll',
|
|
'check-some.bs.table': 'onCheckSome',
|
|
'uncheck-some.bs.table': 'onUncheckSome',
|
|
'load-success.bs.table': 'onLoadSuccess',
|
|
'load-error.bs.table': 'onLoadError',
|
|
'column-switch.bs.table': 'onColumnSwitch',
|
|
'page-change.bs.table': 'onPageChange',
|
|
'search.bs.table': 'onSearch',
|
|
'toggle.bs.table': 'onToggle',
|
|
'pre-body.bs.table': 'onPreBody',
|
|
'post-body.bs.table': 'onPostBody',
|
|
'post-header.bs.table': 'onPostHeader',
|
|
'expand-row.bs.table': 'onExpandRow',
|
|
'collapse-row.bs.table': 'onCollapseRow',
|
|
'refresh-options.bs.table': 'onRefreshOptions',
|
|
'reset-view.bs.table': 'onResetView'
|
|
};
|
|
|
|
BootstrapTable.prototype.init = function () {
|
|
this.initLocale();
|
|
this.initContainer();
|
|
this.initTable();
|
|
this.initHeader();
|
|
this.initData();
|
|
this.initFooter();
|
|
this.initToolbar();
|
|
this.initPagination();
|
|
this.initBody();
|
|
this.initSearchText();
|
|
this.initServer();
|
|
};
|
|
|
|
BootstrapTable.prototype.initLocale = function () {
|
|
if (this.options.locale) {
|
|
var parts = this.options.locale.split(/-|_/);
|
|
parts[0].toLowerCase();
|
|
parts[1] && parts[1].toUpperCase();
|
|
if ($.fn.bootstrapTable.locales[this.options.locale]) {
|
|
// locale as requested
|
|
$.extend(this.options, $.fn.bootstrapTable.locales[this.options.locale]);
|
|
} else if ($.fn.bootstrapTable.locales[parts.join('-')]) {
|
|
// locale with sep set to - (in case original was specified with _)
|
|
$.extend(this.options, $.fn.bootstrapTable.locales[parts.join('-')]);
|
|
} else if ($.fn.bootstrapTable.locales[parts[0]]) {
|
|
// short locale language code (i.e. 'en')
|
|
$.extend(this.options, $.fn.bootstrapTable.locales[parts[0]]);
|
|
}
|
|
}
|
|
};
|
|
|
|
BootstrapTable.prototype.initContainer = function () {
|
|
this.$container = $([
|
|
'<div class="bootstrap-table">',
|
|
'<div class="fixed-table-toolbar"></div>',
|
|
this.options.paginationVAlign === 'top' || this.options.paginationVAlign === 'both' ?
|
|
'<div class="fixed-table-pagination" style="clear: both;"></div>' :
|
|
'',
|
|
'<div class="fixed-table-container">',
|
|
'<div class="fixed-table-header"><table></table></div>',
|
|
'<div class="fixed-table-body">',
|
|
'<div class="fixed-table-loading">',
|
|
this.options.formatLoadingMessage(),
|
|
'</div>',
|
|
'</div>',
|
|
'<div class="fixed-table-footer"><table><tr></tr></table></div>',
|
|
this.options.paginationVAlign === 'bottom' || this.options.paginationVAlign === 'both' ?
|
|
'<div class="fixed-table-pagination"></div>' :
|
|
'',
|
|
'</div>',
|
|
'</div>'
|
|
].join(''));
|
|
|
|
this.$container.insertAfter(this.$el);
|
|
this.$tableContainer = this.$container.find('.fixed-table-container');
|
|
this.$tableHeader = this.$container.find('.fixed-table-header');
|
|
this.$tableBody = this.$container.find('.fixed-table-body');
|
|
this.$tableLoading = this.$container.find('.fixed-table-loading');
|
|
this.$tableFooter = this.$container.find('.fixed-table-footer');
|
|
this.$toolbar = this.$container.find('.fixed-table-toolbar');
|
|
this.$pagination = this.$container.find('.fixed-table-pagination');
|
|
|
|
this.$tableBody.append(this.$el);
|
|
this.$container.after('<div class="clearfix"></div>');
|
|
|
|
this.$el.addClass(this.options.classes);
|
|
if (this.options.striped) {
|
|
this.$el.addClass('table-striped');
|
|
}
|
|
if ($.inArray('table-no-bordered', this.options.classes.split(' ')) !== -1) {
|
|
this.$tableContainer.addClass('table-no-bordered');
|
|
}
|
|
};
|
|
|
|
BootstrapTable.prototype.initTable = function () {
|
|
var that = this,
|
|
columns = [],
|
|
data = [];
|
|
|
|
this.$header = this.$el.find('>thead');
|
|
if (!this.$header.length) {
|
|
this.$header = $('<thead></thead>').appendTo(this.$el);
|
|
}
|
|
this.$header.find('tr').each(function () {
|
|
var column = [];
|
|
|
|
$(this).find('th').each(function () {
|
|
column.push($.extend({}, {
|
|
title: $(this).html(),
|
|
'class': $(this).attr('class'),
|
|
titleTooltip: $(this).attr('title'),
|
|
rowspan: $(this).attr('rowspan') ? +$(this).attr('rowspan') : undefined,
|
|
colspan: $(this).attr('colspan') ? +$(this).attr('colspan') : undefined
|
|
}, $(this).data()));
|
|
});
|
|
columns.push(column);
|
|
});
|
|
if (!$.isArray(this.options.columns[0])) {
|
|
this.options.columns = [this.options.columns];
|
|
}
|
|
this.options.columns = $.extend(true, [], columns, this.options.columns);
|
|
this.columns = [];
|
|
|
|
setFieldIndex(this.options.columns);
|
|
$.each(this.options.columns, function (i, columns) {
|
|
$.each(columns, function (j, column) {
|
|
column = $.extend({}, BootstrapTable.COLUMN_DEFAULTS, column);
|
|
|
|
if (typeof column.fieldIndex !== 'undefined') {
|
|
that.columns[column.fieldIndex] = column;
|
|
}
|
|
|
|
that.options.columns[i][j] = column;
|
|
});
|
|
});
|
|
|
|
// if options.data is setting, do not process tbody data
|
|
if (this.options.data.length) {
|
|
return;
|
|
}
|
|
|
|
this.$el.find('>tbody>tr').each(function () {
|
|
var row = {};
|
|
|
|
// save tr's id, class and data-* attributes
|
|
row._id = $(this).attr('id');
|
|
row._class = $(this).attr('class');
|
|
row._data = getRealDataAttr($(this).data());
|
|
|
|
$(this).find('td').each(function (i) {
|
|
var field = that.columns[i].field;
|
|
|
|
row[field] = $(this).html();
|
|
// save td's id, class and data-* attributes
|
|
row['_' + field + '_id'] = $(this).attr('id');
|
|
row['_' + field + '_class'] = $(this).attr('class');
|
|
row['_' + field + '_rowspan'] = $(this).attr('rowspan');
|
|
row['_' + field + '_title'] = $(this).attr('title');
|
|
row['_' + field + '_data'] = getRealDataAttr($(this).data());
|
|
});
|
|
data.push(row);
|
|
});
|
|
this.options.data = data;
|
|
};
|
|
|
|
BootstrapTable.prototype.initHeader = function () {
|
|
var that = this,
|
|
visibleColumns = {},
|
|
html = [];
|
|
|
|
this.header = {
|
|
fields: [],
|
|
styles: [],
|
|
classes: [],
|
|
formatters: [],
|
|
events: [],
|
|
sorters: [],
|
|
sortNames: [],
|
|
cellStyles: [],
|
|
searchables: []
|
|
};
|
|
|
|
$.each(this.options.columns, function (i, columns) {
|
|
html.push('<tr>');
|
|
|
|
if (i == 0 && !that.options.cardView && that.options.detailView) {
|
|
html.push(sprintf('<th class="detail" rowspan="%s"><div class="fht-cell"></div></th>',
|
|
that.options.columns.length));
|
|
}
|
|
|
|
$.each(columns, function (j, column) {
|
|
var text = '',
|
|
halign = '', // header align style
|
|
align = '', // body align style
|
|
style = '',
|
|
class_ = sprintf(' class="%s"', column['class']),
|
|
order = that.options.sortOrder || column.order,
|
|
unitWidth = 'px',
|
|
width = column.width;
|
|
|
|
if (column.width !== undefined && (!that.options.cardView)) {
|
|
if (typeof column.width === 'string') {
|
|
if (column.width.indexOf('%') !== -1) {
|
|
unitWidth = '%';
|
|
}
|
|
}
|
|
}
|
|
if (column.width && typeof column.width === 'string') {
|
|
width = column.width.replace('%', '').replace('px', '');
|
|
}
|
|
|
|
halign = sprintf('text-align: %s; ', column.halign ? column.halign : column.align);
|
|
align = sprintf('text-align: %s; ', column.align);
|
|
style = sprintf('vertical-align: %s; ', column.valign);
|
|
style += sprintf('width: %s; ', (column.checkbox || column.radio) && !width ?
|
|
'36px' : (width ? width + unitWidth : undefined));
|
|
|
|
if (typeof column.fieldIndex !== 'undefined') {
|
|
that.header.fields[column.fieldIndex] = column.field;
|
|
that.header.styles[column.fieldIndex] = align + style;
|
|
that.header.classes[column.fieldIndex] = class_;
|
|
that.header.formatters[column.fieldIndex] = column.formatter;
|
|
that.header.events[column.fieldIndex] = column.events;
|
|
that.header.sorters[column.fieldIndex] = column.sorter;
|
|
that.header.sortNames[column.fieldIndex] = column.sortName;
|
|
that.header.cellStyles[column.fieldIndex] = column.cellStyle;
|
|
that.header.searchables[column.fieldIndex] = column.searchable;
|
|
|
|
if (!column.visible) {
|
|
return;
|
|
}
|
|
|
|
if (that.options.cardView && (!column.cardVisible)) {
|
|
return;
|
|
}
|
|
|
|
visibleColumns[column.field] = column;
|
|
}
|
|
|
|
html.push('<th' + sprintf(' title="%s"', column.titleTooltip),
|
|
column.checkbox || column.radio ?
|
|
sprintf(' class="bs-checkbox %s"', column['class'] || '') :
|
|
class_,
|
|
sprintf(' style="%s"', halign + style),
|
|
sprintf(' rowspan="%s"', column.rowspan),
|
|
sprintf(' colspan="%s"', column.colspan),
|
|
sprintf(' data-field="%s"', column.field),
|
|
"tabindex='0'",
|
|
'>');
|
|
|
|
html.push(sprintf('<div class="th-inner %s">', that.options.sortable && column.sortable ?
|
|
'sortable both' : ''));
|
|
|
|
text = column.title;
|
|
|
|
if (column.checkbox) {
|
|
if (!that.options.singleSelect && that.options.checkboxHeader) {
|
|
text = '<input name="btSelectAll" type="checkbox" />';
|
|
}
|
|
that.header.stateField = column.field;
|
|
}
|
|
if (column.radio) {
|
|
text = '';
|
|
that.header.stateField = column.field;
|
|
that.options.singleSelect = true;
|
|
}
|
|
|
|
html.push(text);
|
|
html.push('</div>');
|
|
html.push('<div class="fht-cell"></div>');
|
|
html.push('</div>');
|
|
html.push('</th>');
|
|
});
|
|
html.push('</tr>');
|
|
});
|
|
|
|
this.$header.html(html.join(''));
|
|
this.$header.find('th[data-field]').each(function (i) {
|
|
$(this).data(visibleColumns[$(this).data('field')]);
|
|
});
|
|
this.$container.off('click', '.th-inner').on('click', '.th-inner', function (event) {
|
|
var target = $(this);
|
|
if (target.closest('.bootstrap-table')[0] !== that.$container[0])
|
|
return false;
|
|
|
|
if (that.options.sortable && target.parent().data().sortable) {
|
|
that.onSort(event);
|
|
}
|
|
});
|
|
|
|
this.$header.children().children().off('keypress').on('keypress', function (event) {
|
|
if (that.options.sortable && $(this).data().sortable) {
|
|
var code = event.keyCode || event.which;
|
|
if (code == 13) { //Enter keycode
|
|
that.onSort(event);
|
|
}
|
|
}
|
|
});
|
|
|
|
if (!this.options.showHeader || this.options.cardView) {
|
|
this.$header.hide();
|
|
this.$tableHeader.hide();
|
|
this.$tableLoading.css('top', 0);
|
|
} else {
|
|
this.$header.show();
|
|
this.$tableHeader.show();
|
|
this.$tableLoading.css('top', this.$header.outerHeight() + 1);
|
|
// Assign the correct sortable arrow
|
|
this.getCaret();
|
|
}
|
|
|
|
this.$selectAll = this.$header.find('[name="btSelectAll"]');
|
|
this.$selectAll.off('click').on('click', function () {
|
|
var checked = $(this).prop('checked');
|
|
that[checked ? 'checkAll' : 'uncheckAll']();
|
|
that.updateSelected();
|
|
});
|
|
};
|
|
|
|
BootstrapTable.prototype.initFooter = function () {
|
|
if (!this.options.showFooter || this.options.cardView) {
|
|
this.$tableFooter.hide();
|
|
} else {
|
|
this.$tableFooter.show();
|
|
}
|
|
};
|
|
|
|
/**
|
|
* @param data
|
|
* @param type: append / prepend
|
|
*/
|
|
BootstrapTable.prototype.initData = function (data, type) {
|
|
if (type === 'append') {
|
|
this.data = this.data.concat(data);
|
|
} else if (type === 'prepend') {
|
|
this.data = [].concat(data).concat(this.data);
|
|
} else {
|
|
this.data = data || this.options.data;
|
|
}
|
|
|
|
// Fix #839 Records deleted when adding new row on filtered table
|
|
if (type === 'append') {
|
|
this.options.data = this.options.data.concat(data);
|
|
} else if (type === 'prepend') {
|
|
this.options.data = [].concat(data).concat(this.options.data);
|
|
} else {
|
|
this.options.data = this.data;
|
|
}
|
|
|
|
if (this.options.sidePagination === 'server') {
|
|
return;
|
|
}
|
|
this.initSort();
|
|
};
|
|
|
|
BootstrapTable.prototype.initSort = function () {
|
|
var that = this,
|
|
name = this.options.sortName,
|
|
order = this.options.sortOrder === 'desc' ? -1 : 1,
|
|
index = $.inArray(this.options.sortName, this.header.fields);
|
|
|
|
if (index !== -1) {
|
|
this.data.sort(function (a, b) {
|
|
if (that.header.sortNames[index]) {
|
|
name = that.header.sortNames[index];
|
|
}
|
|
var aa = getItemField(a, name, that.options.escape),
|
|
bb = getItemField(b, name, that.options.escape),
|
|
value = calculateObjectValue(that.header, that.header.sorters[index], [aa, bb]);
|
|
|
|
if (value !== undefined) {
|
|
return order * value;
|
|
}
|
|
|
|
// Fix #161: undefined or null string sort bug.
|
|
if (aa === undefined || aa === null) {
|
|
aa = '';
|
|
}
|
|
if (bb === undefined || bb === null) {
|
|
bb = '';
|
|
}
|
|
|
|
// IF both values are numeric, do a numeric comparison
|
|
if ($.isNumeric(aa) && $.isNumeric(bb)) {
|
|
// Convert numerical values form string to float.
|
|
aa = parseFloat(aa);
|
|
bb = parseFloat(bb);
|
|
if (aa < bb) {
|
|
return order * -1;
|
|
}
|
|
return order;
|
|
}
|
|
|
|
if (aa === bb) {
|
|
return 0;
|
|
}
|
|
|
|
// If value is not a string, convert to string
|
|
if (typeof aa !== 'string') {
|
|
aa = aa.toString();
|
|
}
|
|
|
|
if (aa.localeCompare(bb) === -1) {
|
|
return order * -1;
|
|
}
|
|
|
|
return order;
|
|
});
|
|
}
|
|
};
|
|
|
|
BootstrapTable.prototype.onSort = function (event) {
|
|
var $this = event.type === "keypress" ? $(event.currentTarget) : $(event.currentTarget).parent(),
|
|
$this_ = this.$header.find('th').eq($this.index());
|
|
|
|
this.$header.add(this.$header_).find('span.order').remove();
|
|
|
|
if (this.options.sortName === $this.data('field')) {
|
|
this.options.sortOrder = this.options.sortOrder === 'asc' ? 'desc' : 'asc';
|
|
} else {
|
|
this.options.sortName = $this.data('field');
|
|
this.options.sortOrder = $this.data('order') === 'asc' ? 'desc' : 'asc';
|
|
}
|
|
this.trigger('sort', this.options.sortName, this.options.sortOrder);
|
|
|
|
$this.add($this_).data('order', this.options.sortOrder);
|
|
|
|
// Assign the correct sortable arrow
|
|
this.getCaret();
|
|
|
|
if (this.options.sidePagination === 'server') {
|
|
this.initServer(this.options.silentSort);
|
|
return;
|
|
}
|
|
|
|
this.initSort();
|
|
this.initBody();
|
|
};
|
|
|
|
BootstrapTable.prototype.initToolbar = function () {
|
|
var that = this,
|
|
html = [],
|
|
timeoutId = 0,
|
|
$keepOpen,
|
|
$search,
|
|
switchableCount = 0;
|
|
|
|
if (this.$toolbar.find('.bars').children().length) {
|
|
$('body').append($(this.options.toolbar));
|
|
}
|
|
this.$toolbar.html('');
|
|
|
|
if (typeof this.options.toolbar === 'string' || typeof this.options.toolbar === 'object') {
|
|
$(sprintf('<div class="bars pull-%s"></div>', this.options.toolbarAlign))
|
|
.appendTo(this.$toolbar)
|
|
.append($(this.options.toolbar));
|
|
}
|
|
|
|
// showColumns, showToggle, showRefresh
|
|
html = [sprintf('<div class="columns columns-%s btn-group pull-%s">',
|
|
this.options.buttonsAlign, this.options.buttonsAlign)];
|
|
|
|
if (typeof this.options.icons === 'string') {
|
|
this.options.icons = calculateObjectValue(null, this.options.icons);
|
|
}
|
|
|
|
if (this.options.showPaginationSwitch) {
|
|
html.push(sprintf('<button class="btn btn-default" type="button" name="paginationSwitch" title="%s">',
|
|
this.options.formatPaginationSwitch()),
|
|
sprintf('<i class="%s %s"></i>', this.options.iconsPrefix, this.options.icons.paginationSwitchDown),
|
|
'</button>');
|
|
}
|
|
|
|
if (this.options.showRefresh) {
|
|
html.push(sprintf('<button class="btn btn-default' +
|
|
sprintf(' btn-%s', this.options.iconSize) +
|
|
'" type="button" name="refresh" title="%s">',
|
|
this.options.formatRefresh()),
|
|
sprintf('<i class="%s %s"></i>', this.options.iconsPrefix, this.options.icons.refresh),
|
|
'</button>');
|
|
}
|
|
|
|
if (this.options.showToggle) {
|
|
html.push(sprintf('<button class="btn btn-default' +
|
|
sprintf(' btn-%s', this.options.iconSize) +
|
|
'" type="button" name="toggle" title="%s">',
|
|
this.options.formatToggle()),
|
|
sprintf('<i class="%s %s"></i>', this.options.iconsPrefix, this.options.icons.toggle),
|
|
'</button>');
|
|
}
|
|
|
|
if (this.options.showColumns) {
|
|
html.push(sprintf('<div class="keep-open btn-group" title="%s">',
|
|
this.options.formatColumns()),
|
|
'<button type="button" class="btn btn-default' +
|
|
sprintf(' btn-%s', this.options.iconSize) +
|
|
' dropdown-toggle" data-toggle="dropdown">',
|
|
sprintf('<i class="%s %s"></i>', this.options.iconsPrefix, this.options.icons.columns),
|
|
' <span class="caret"></span>',
|
|
'</button>',
|
|
'<ul class="dropdown-menu" role="menu">');
|
|
|
|
$.each(this.columns, function (i, column) {
|
|
if (column.radio || column.checkbox) {
|
|
return;
|
|
}
|
|
|
|
if (that.options.cardView && (!column.cardVisible)) {
|
|
return;
|
|
}
|
|
|
|
var checked = column.visible ? ' checked="checked"' : '';
|
|
|
|
if (column.switchable) {
|
|
html.push(sprintf('<li>' +
|
|
'<label><input type="checkbox" data-field="%s" value="%s"%s> %s</label>' +
|
|
'</li>', column.field, i, checked, column.title));
|
|
switchableCount++;
|
|
}
|
|
});
|
|
html.push('</ul>',
|
|
'</div>');
|
|
}
|
|
|
|
html.push('</div>');
|
|
|
|
// Fix #188: this.showToolbar is for extensions
|
|
if (this.showToolbar || html.length > 2) {
|
|
this.$toolbar.append(html.join(''));
|
|
}
|
|
|
|
if (this.options.showPaginationSwitch) {
|
|
this.$toolbar.find('button[name="paginationSwitch"]')
|
|
.off('click').on('click', $.proxy(this.togglePagination, this));
|
|
}
|
|
|
|
if (this.options.showRefresh) {
|
|
this.$toolbar.find('button[name="refresh"]')
|
|
.off('click').on('click', $.proxy(this.refresh, this));
|
|
}
|
|
|
|
if (this.options.showToggle) {
|
|
this.$toolbar.find('button[name="toggle"]')
|
|
.off('click').on('click', function () {
|
|
that.toggleView();
|
|
});
|
|
}
|
|
|
|
if (this.options.showColumns) {
|
|
$keepOpen = this.$toolbar.find('.keep-open');
|
|
|
|
if (switchableCount <= this.options.minimumCountColumns) {
|
|
$keepOpen.find('input').prop('disabled', true);
|
|
}
|
|
|
|
$keepOpen.find('li').off('click').on('click', function (event) {
|
|
event.stopImmediatePropagation();
|
|
});
|
|
$keepOpen.find('input').off('click').on('click', function () {
|
|
var $this = $(this);
|
|
|
|
that.toggleColumn(getFieldIndex(that.columns,
|
|
$(this).data('field')), $this.prop('checked'), false);
|
|
that.trigger('column-switch', $(this).data('field'), $this.prop('checked'));
|
|
});
|
|
}
|
|
|
|
if (this.options.search) {
|
|
html = [];
|
|
html.push(
|
|
'<div class="pull-' + this.options.searchAlign + ' search">',
|
|
sprintf('<input class="form-control' +
|
|
sprintf(' input-%s', this.options.iconSize) +
|
|
'" type="text" placeholder="%s">',
|
|
this.options.formatSearch()),
|
|
'</div>');
|
|
|
|
this.$toolbar.append(html.join(''));
|
|
$search = this.$toolbar.find('.search input');
|
|
$search.off('keyup drop').on('keyup drop', function (event) {
|
|
if (that.options.searchOnEnterKey) {
|
|
if (event.keyCode !== 13) {
|
|
return;
|
|
}
|
|
}
|
|
|
|
clearTimeout(timeoutId); // doesn't matter if it's 0
|
|
timeoutId = setTimeout(function () {
|
|
that.onSearch(event);
|
|
}, that.options.searchTimeOut);
|
|
});
|
|
|
|
if (isIEBrowser()) {
|
|
$search.off('mouseup').on('mouseup', function (event) {
|
|
clearTimeout(timeoutId); // doesn't matter if it's 0
|
|
timeoutId = setTimeout(function () {
|
|
that.onSearch(event);
|
|
}, that.options.searchTimeOut);
|
|
});
|
|
}
|
|
}
|
|
};
|
|
|
|
BootstrapTable.prototype.onSearch = function (event) {
|
|
var text = $.trim($(event.currentTarget).val());
|
|
|
|
// trim search input
|
|
if (this.options.trimOnSearch && $(event.currentTarget).val() !== text) {
|
|
$(event.currentTarget).val(text);
|
|
}
|
|
|
|
if (text === this.searchText) {
|
|
return;
|
|
}
|
|
this.searchText = text;
|
|
this.options.searchText = text;
|
|
|
|
this.options.pageNumber = 1;
|
|
this.initSearch();
|
|
this.updatePagination();
|
|
this.trigger('search', text);
|
|
};
|
|
|
|
BootstrapTable.prototype.initSearch = function () {
|
|
var that = this;
|
|
|
|
if (this.options.sidePagination !== 'server') {
|
|
var s = this.searchText && this.searchText.toLowerCase();
|
|
var f = $.isEmptyObject(this.filterColumns) ? null : this.filterColumns;
|
|
|
|
// Check filter
|
|
this.data = f ? $.grep(this.options.data, function (item, i) {
|
|
for (var key in f) {
|
|
if ($.isArray(f[key])) {
|
|
if ($.inArray(item[key], f[key]) === -1) {
|
|
return false;
|
|
}
|
|
} else if (item[key] !== f[key]) {
|
|
return false;
|
|
}
|
|
}
|
|
return true;
|
|
}) : this.options.data;
|
|
|
|
this.data = s ? $.grep(this.data, function (item, i) {
|
|
for (var key in item) {
|
|
key = $.isNumeric(key) ? parseInt(key, 10) : key;
|
|
var value = item[key],
|
|
column = that.columns[getFieldIndex(that.columns, key)],
|
|
j = $.inArray(key, that.header.fields);
|
|
|
|
// Fix #142: search use formatted data
|
|
if (column && column.searchFormatter) {
|
|
value = calculateObjectValue(column,
|
|
that.header.formatters[j], [value, item, i], value);
|
|
}
|
|
|
|
var index = $.inArray(key, that.header.fields);
|
|
if (index !== -1 && that.header.searchables[index] && (typeof value === 'string' || typeof value === 'number')) {
|
|
if (that.options.strictSearch) {
|
|
if ((value + '').toLowerCase() === s) {
|
|
return true;
|
|
}
|
|
} else {
|
|
if ((value + '').toLowerCase().indexOf(s) !== -1) {
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return false;
|
|
}) : this.data;
|
|
}
|
|
};
|
|
|
|
BootstrapTable.prototype.initPagination = function () {
|
|
if (!this.options.pagination) {
|
|
this.$pagination.hide();
|
|
return;
|
|
} else {
|
|
this.$pagination.show();
|
|
}
|
|
|
|
var that = this,
|
|
html = [],
|
|
$allSelected = false,
|
|
i, from, to,
|
|
$pageList,
|
|
$first, $pre,
|
|
$next, $last,
|
|
$number,
|
|
data = this.getData();
|
|
|
|
if (this.options.sidePagination !== 'server') {
|
|
this.options.totalRows = data.length;
|
|
}
|
|
|
|
this.totalPages = 0;
|
|
if (this.options.totalRows) {
|
|
if (this.options.pageSize === this.options.formatAllRows()) {
|
|
this.options.pageSize = this.options.totalRows;
|
|
$allSelected = true;
|
|
} else if (this.options.pageSize === this.options.totalRows) {
|
|
// Fix #667 Table with pagination,
|
|
// multiple pages and a search that matches to one page throws exception
|
|
var pageLst = typeof this.options.pageList === 'string' ?
|
|
this.options.pageList.replace('[', '').replace(']', '')
|
|
.replace(/ /g, '').toLowerCase().split(',') : this.options.pageList;
|
|
if ($.inArray(this.options.formatAllRows().toLowerCase(), pageLst) > -1) {
|
|
$allSelected = true;
|
|
}
|
|
}
|
|
|
|
this.totalPages = ~~((this.options.totalRows - 1) / this.options.pageSize) + 1;
|
|
|
|
this.options.totalPages = this.totalPages;
|
|
}
|
|
if (this.totalPages > 0 && this.options.pageNumber > this.totalPages) {
|
|
this.options.pageNumber = this.totalPages;
|
|
}
|
|
|
|
this.pageFrom = (this.options.pageNumber - 1) * this.options.pageSize + 1;
|
|
this.pageTo = this.options.pageNumber * this.options.pageSize;
|
|
if (this.pageTo > this.options.totalRows) {
|
|
this.pageTo = this.options.totalRows;
|
|
}
|
|
|
|
html.push(
|
|
'<div class="pull-' + this.options.paginationDetailHAlign + ' pagination-detail">',
|
|
'<span class="pagination-info">',
|
|
this.options.onlyInfoPagination ? this.options.formatDetailPagination(this.options.totalRows) :
|
|
this.options.formatShowingRows(this.pageFrom, this.pageTo, this.options.totalRows),
|
|
'</span>');
|
|
|
|
if (!this.options.onlyInfoPagination) {
|
|
html.push('<span class="page-list">');
|
|
|
|
var pageNumber = [
|
|
sprintf('<span class="btn-group %s">',
|
|
this.options.paginationVAlign === 'top' || this.options.paginationVAlign === 'both' ?
|
|
'dropdown' : 'dropup'),
|
|
'<button type="button" class="btn btn-default ' +
|
|
sprintf(' btn-%s', this.options.iconSize) +
|
|
' dropdown-toggle" data-toggle="dropdown">',
|
|
'<span class="page-size">',
|
|
$allSelected ? this.options.formatAllRows() : this.options.pageSize,
|
|
'</span>',
|
|
' <span class="caret"></span>',
|
|
'</button>',
|
|
'<ul class="dropdown-menu" role="menu">'
|
|
],
|
|
pageList = this.options.pageList;
|
|
|
|
if (typeof this.options.pageList === 'string') {
|
|
var list = this.options.pageList.replace('[', '').replace(']', '')
|
|
.replace(/ /g, '').split(',');
|
|
|
|
pageList = [];
|
|
$.each(list, function (i, value) {
|
|
pageList.push(value.toUpperCase() === that.options.formatAllRows().toUpperCase() ?
|
|
that.options.formatAllRows() : +value);
|
|
});
|
|
}
|
|
|
|
$.each(pageList, function (i, page) {
|
|
if (!that.options.smartDisplay || i === 0 || pageList[i - 1] <= that.options.totalRows) {
|
|
var active;
|
|
if ($allSelected) {
|
|
active = page === that.options.formatAllRows() ? ' class="active"' : '';
|
|
} else {
|
|
active = page === that.options.pageSize ? ' class="active"' : '';
|
|
}
|
|
pageNumber.push(sprintf('<li%s><a href="javascript:void(0)">%s</a></li>', active, page));
|
|
}
|
|
});
|
|
pageNumber.push('</ul></span>');
|
|
|
|
html.push(this.options.formatRecordsPerPage(pageNumber.join('')));
|
|
html.push('</span>');
|
|
|
|
html.push('</div>',
|
|
'<div class="pull-' + this.options.paginationHAlign + ' pagination">',
|
|
'<ul class="pagination' + sprintf(' pagination-%s', this.options.iconSize) + '">',
|
|
'<li class="page-pre"><a href="javascript:void(0)">' + this.options.paginationPreText + '</a></li>');
|
|
|
|
if (this.totalPages < 5) {
|
|
from = 1;
|
|
to = this.totalPages;
|
|
} else {
|
|
from = this.options.pageNumber - 2;
|
|
to = from + 4;
|
|
if (from < 1) {
|
|
from = 1;
|
|
to = 5;
|
|
}
|
|
if (to > this.totalPages) {
|
|
to = this.totalPages;
|
|
from = to - 4;
|
|
}
|
|
}
|
|
|
|
if (this.totalPages >= 6) {
|
|
if (this.options.pageNumber >= 3) {
|
|
html.push('<li class="page-first' + (1 === this.options.pageNumber ? ' active' : '') + '">',
|
|
'<a href="javascript:void(0)">', 1, '</a>',
|
|
'</li>');
|
|
|
|
from++;
|
|
}
|
|
|
|
if (this.options.pageNumber >= 4) {
|
|
if (this.options.pageNumber == 4 || this.totalPages == 6 || this.totalPages == 7) {
|
|
from--;
|
|
} else {
|
|
html.push('<li class="page-first-separator disabled">',
|
|
'<a href="javascript:void(0)">...</a>',
|
|
'</li>');
|
|
}
|
|
|
|
to--;
|
|
}
|
|
}
|
|
|
|
if (this.totalPages >= 7) {
|
|
if (this.options.pageNumber >= (this.totalPages - 2)) {
|
|
from--;
|
|
}
|
|
}
|
|
|
|
if (this.totalPages == 6) {
|
|
if (this.options.pageNumber >= (this.totalPages - 2)) {
|
|
to++;
|
|
}
|
|
} else if (this.totalPages >= 7) {
|
|
if (this.totalPages == 7 || this.options.pageNumber >= (this.totalPages - 3)) {
|
|
to++;
|
|
}
|
|
}
|
|
|
|
for (i = from; i <= to; i++) {
|
|
html.push('<li class="page-number' + (i === this.options.pageNumber ? ' active' : '') + '">',
|
|
'<a href="javascript:void(0)">', i, '</a>',
|
|
'</li>');
|
|
}
|
|
|
|
if (this.totalPages >= 8) {
|
|
if (this.options.pageNumber <= (this.totalPages - 4)) {
|
|
html.push('<li class="page-last-separator disabled">',
|
|
'<a href="javascript:void(0)">...</a>',
|
|
'</li>');
|
|
}
|
|
}
|
|
|
|
if (this.totalPages >= 6) {
|
|
if (this.options.pageNumber <= (this.totalPages - 3)) {
|
|
html.push('<li class="page-last' + (this.totalPages === this.options.pageNumber ? ' active' : '') + '">',
|
|
'<a href="javascript:void(0)">', this.totalPages, '</a>',
|
|
'</li>');
|
|
}
|
|
}
|
|
|
|
html.push(
|
|
'<li class="page-next"><a href="javascript:void(0)">' + this.options.paginationNextText + '</a></li>',
|
|
'</ul>',
|
|
'</div>');
|
|
}
|
|
this.$pagination.html(html.join(''));
|
|
|
|
if (!this.options.onlyInfoPagination) {
|
|
$pageList = this.$pagination.find('.page-list a');
|
|
$first = this.$pagination.find('.page-first');
|
|
$pre = this.$pagination.find('.page-pre');
|
|
$next = this.$pagination.find('.page-next');
|
|
$last = this.$pagination.find('.page-last');
|
|
$number = this.$pagination.find('.page-number');
|
|
|
|
if (this.options.smartDisplay) {
|
|
if (this.totalPages <= 1) {
|
|
this.$pagination.find('div.pagination').hide();
|
|
}
|
|
if (pageList.length < 2 || this.options.totalRows <= pageList[0]) {
|
|
this.$pagination.find('span.page-list').hide();
|
|
}
|
|
|
|
// when data is empty, hide the pagination
|
|
this.$pagination[this.getData().length ? 'show' : 'hide']();
|
|
}
|
|
if ($allSelected) {
|
|
this.options.pageSize = this.options.formatAllRows();
|
|
}
|
|
$pageList.off('click').on('click', $.proxy(this.onPageListChange, this));
|
|
$first.off('click').on('click', $.proxy(this.onPageFirst, this));
|
|
$pre.off('click').on('click', $.proxy(this.onPagePre, this));
|
|
$next.off('click').on('click', $.proxy(this.onPageNext, this));
|
|
$last.off('click').on('click', $.proxy(this.onPageLast, this));
|
|
$number.off('click').on('click', $.proxy(this.onPageNumber, this));
|
|
}
|
|
};
|
|
|
|
BootstrapTable.prototype.updatePagination = function (event) {
|
|
// Fix #171: IE disabled button can be clicked bug.
|
|
if (event && $(event.currentTarget).hasClass('disabled')) {
|
|
return;
|
|
}
|
|
|
|
if (!this.options.maintainSelected) {
|
|
this.resetRows();
|
|
}
|
|
|
|
this.initPagination();
|
|
if (this.options.sidePagination === 'server') {
|
|
this.initServer();
|
|
} else {
|
|
this.initBody();
|
|
}
|
|
|
|
this.trigger('page-change', this.options.pageNumber, this.options.pageSize);
|
|
};
|
|
|
|
BootstrapTable.prototype.onPageListChange = function (event) {
|
|
var $this = $(event.currentTarget);
|
|
|
|
$this.parent().addClass('active').siblings().removeClass('active');
|
|
this.options.pageSize = $this.text().toUpperCase() === this.options.formatAllRows().toUpperCase() ?
|
|
this.options.formatAllRows() : +$this.text();
|
|
this.$toolbar.find('.page-size').text(this.options.pageSize);
|
|
|
|
this.updatePagination(event);
|
|
};
|
|
|
|
BootstrapTable.prototype.onPageFirst = function (event) {
|
|
this.options.pageNumber = 1;
|
|
this.updatePagination(event);
|
|
};
|
|
|
|
BootstrapTable.prototype.onPagePre = function (event) {
|
|
if ((this.options.pageNumber - 1) == 0) {
|
|
this.options.pageNumber = this.options.totalPages;
|
|
} else {
|
|
this.options.pageNumber--;
|
|
}
|
|
this.updatePagination(event);
|
|
};
|
|
|
|
BootstrapTable.prototype.onPageNext = function (event) {
|
|
if ((this.options.pageNumber + 1) > this.options.totalPages) {
|
|
this.options.pageNumber = 1;
|
|
} else {
|
|
this.options.pageNumber++;
|
|
}
|
|
this.updatePagination(event);
|
|
};
|
|
|
|
BootstrapTable.prototype.onPageLast = function (event) {
|
|
this.options.pageNumber = this.totalPages;
|
|
this.updatePagination(event);
|
|
};
|
|
|
|
BootstrapTable.prototype.onPageNumber = function (event) {
|
|
if (this.options.pageNumber === +$(event.currentTarget).text()) {
|
|
return;
|
|
}
|
|
this.options.pageNumber = +$(event.currentTarget).text();
|
|
this.updatePagination(event);
|
|
};
|
|
|
|
BootstrapTable.prototype.initBody = function (fixedScroll) {
|
|
var that = this,
|
|
html = [],
|
|
data = this.getData();
|
|
|
|
this.trigger('pre-body', data);
|
|
|
|
this.$body = this.$el.find('>tbody');
|
|
if (!this.$body.length) {
|
|
this.$body = $('<tbody></tbody>').appendTo(this.$el);
|
|
}
|
|
|
|
//Fix #389 Bootstrap-table-flatJSON is not working
|
|
|
|
if (!this.options.pagination || this.options.sidePagination === 'server') {
|
|
this.pageFrom = 1;
|
|
this.pageTo = data.length;
|
|
}
|
|
|
|
for (var i = this.pageFrom - 1; i < this.pageTo; i++) {
|
|
var key,
|
|
item = data[i],
|
|
style = {},
|
|
csses = [],
|
|
data_ = '',
|
|
attributes = {},
|
|
htmlAttributes = [];
|
|
|
|
style = calculateObjectValue(this.options, this.options.rowStyle, [item, i], style);
|
|
|
|
if (style && style.css) {
|
|
for (key in style.css) {
|
|
csses.push(key + ': ' + style.css[key]);
|
|
}
|
|
}
|
|
|
|
attributes = calculateObjectValue(this.options,
|
|
this.options.rowAttributes, [item, i], attributes);
|
|
|
|
if (attributes) {
|
|
for (key in attributes) {
|
|
htmlAttributes.push(sprintf('%s="%s"', key, escapeHTML(attributes[key])));
|
|
}
|
|
}
|
|
|
|
if (item._data && !$.isEmptyObject(item._data)) {
|
|
$.each(item._data, function (k, v) {
|
|
// ignore data-index
|
|
if (k === 'index') {
|
|
return;
|
|
}
|
|
data_ += sprintf(' data-%s="%s"', k, v);
|
|
});
|
|
}
|
|
|
|
html.push('<tr',
|
|
sprintf(' %s', htmlAttributes.join(' ')),
|
|
sprintf(' id="%s"', $.isArray(item) ? undefined : item._id),
|
|
sprintf(' class="%s"', style.classes || ($.isArray(item) ? undefined : item._class)),
|
|
sprintf(' data-index="%s"', i),
|
|
sprintf(' data-uniqueid="%s"', item[this.options.uniqueId]),
|
|
sprintf('%s', data_),
|
|
'>'
|
|
);
|
|
|
|
if (this.options.cardView) {
|
|
html.push(sprintf('<td colspan="%s">', this.header.fields.length));
|
|
}
|
|
|
|
if (!this.options.cardView && this.options.detailView) {
|
|
html.push('<td>',
|
|
'<a class="detail-icon" href="javascript:">',
|
|
sprintf('<i class="%s %s"></i>', this.options.iconsPrefix, this.options.icons.detailOpen),
|
|
'</a>',
|
|
'</td>');
|
|
}
|
|
|
|
$.each(this.header.fields, function (j, field) {
|
|
var text = '',
|
|
value = getItemField(item, field, that.options.escape),
|
|
type = '',
|
|
cellStyle = {},
|
|
id_ = '',
|
|
class_ = that.header.classes[j],
|
|
data_ = '',
|
|
rowspan_ = '',
|
|
title_ = '',
|
|
column = that.columns[getFieldIndex(that.columns, field)];
|
|
|
|
if (!column.visible) {
|
|
return;
|
|
}
|
|
|
|
style = sprintf('style="%s"', csses.concat(that.header.styles[j]).join('; '));
|
|
|
|
value = calculateObjectValue(column,
|
|
that.header.formatters[j], [value, item, i], value);
|
|
|
|
// handle td's id and class
|
|
if (item['_' + field + '_id']) {
|
|
id_ = sprintf(' id="%s"', item['_' + field + '_id']);
|
|
}
|
|
if (item['_' + field + '_class']) {
|
|
class_ = sprintf(' class="%s"', item['_' + field + '_class']);
|
|
}
|
|
if (item['_' + field + '_rowspan']) {
|
|
rowspan_ = sprintf(' rowspan="%s"', item['_' + field + '_rowspan']);
|
|
}
|
|
if (item['_' + field + '_title']) {
|
|
title_ = sprintf(' title="%s"', item['_' + field + '_title']);
|
|
}
|
|
cellStyle = calculateObjectValue(that.header,
|
|
that.header.cellStyles[j], [value, item, i], cellStyle);
|
|
if (cellStyle.classes) {
|
|
class_ = sprintf(' class="%s"', cellStyle.classes);
|
|
}
|
|
if (cellStyle.css) {
|
|
var csses_ = [];
|
|
for (var key in cellStyle.css) {
|
|
csses_.push(key + ': ' + cellStyle.css[key]);
|
|
}
|
|
style = sprintf('style="%s"', csses_.concat(that.header.styles[j]).join('; '));
|
|
}
|
|
|
|
if (item['_' + field + '_data'] && !$.isEmptyObject(item['_' + field + '_data'])) {
|
|
$.each(item['_' + field + '_data'], function (k, v) {
|
|
// ignore data-index
|
|
if (k === 'index') {
|
|
return;
|
|
}
|
|
data_ += sprintf(' data-%s="%s"', k, v);
|
|
});
|
|
}
|
|
|
|
if (column.checkbox || column.radio) {
|
|
type = column.checkbox ? 'checkbox' : type;
|
|
type = column.radio ? 'radio' : type;
|
|
|
|
text = [sprintf(that.options.cardView ?
|
|
'<div class="card-view %s">' : '<td class="bs-checkbox %s">', column['class'] || ''),
|
|
'<input' +
|
|
sprintf(' data-index="%s"', i) +
|
|
sprintf(' name="%s"', that.options.selectItemName) +
|
|
sprintf(' type="%s"', type) +
|
|
sprintf(' value="%s"', item[that.options.idField]) +
|
|
sprintf(' checked="%s"', value === true ||
|
|
(value && value.checked) ? 'checked' : undefined) +
|
|
sprintf(' disabled="%s"', !column.checkboxEnabled ||
|
|
(value && value.disabled) ? 'disabled' : undefined) +
|
|
' />',
|
|
that.header.formatters[j] && typeof value === 'string' ? value : '',
|
|
that.options.cardView ? '</div>' : '</td>'
|
|
].join('');
|
|
|
|
item[that.header.stateField] = value === true || (value && value.checked);
|
|
} else {
|
|
value = typeof value === 'undefined' || value === null ?
|
|
that.options.undefinedText : value;
|
|
|
|
text = that.options.cardView ? ['<div class="card-view">',
|
|
that.options.showHeader ? sprintf('<span class="title" %s>%s</span>', style,
|
|
getPropertyFromOther(that.columns, 'field', 'title', field)) : '',
|
|
sprintf('<span class="value">%s</span>', value),
|
|
'</div>'
|
|
].join('') : [sprintf('<td%s %s %s %s %s %s>', id_, class_, style, data_, rowspan_, title_),
|
|
value,
|
|
'</td>'
|
|
].join('');
|
|
|
|
// Hide empty data on Card view when smartDisplay is set to true.
|
|
if (that.options.cardView && that.options.smartDisplay && value === '') {
|
|
// Should set a placeholder for event binding correct fieldIndex
|
|
text = '<div class="card-view"></div>';
|
|
}
|
|
}
|
|
|
|
html.push(text);
|
|
});
|
|
|
|
if (this.options.cardView) {
|
|
html.push('</td>');
|
|
}
|
|
|
|
html.push('</tr>');
|
|
}
|
|
|
|
// show no records
|
|
if (!html.length) {
|
|
html.push('<tr class="no-records-found">',
|
|
sprintf('<td colspan="%s">%s</td>',
|
|
this.$header.find('th').length, this.options.formatNoMatches()),
|
|
'</tr>');
|
|
}
|
|
|
|
this.$body.html(html.join(''));
|
|
|
|
if (!fixedScroll) {
|
|
this.scrollTo(0);
|
|
}
|
|
|
|
// click to select by column
|
|
this.$body.find('> tr[data-index] > td').off('click dblclick').on('click dblclick', function (e) {
|
|
var $td = $(this),
|
|
$tr = $td.parent(),
|
|
item = that.data[$tr.data('index')],
|
|
index = $td[0].cellIndex,
|
|
field = that.header.fields[that.options.detailView && !that.options.cardView ? index - 1 : index],
|
|
column = that.columns[getFieldIndex(that.columns, field)],
|
|
value = getItemField(item, field, that.options.escape);
|
|
|
|
if ($td.find('.detail-icon').length) {
|
|
return;
|
|
}
|
|
|
|
that.trigger(e.type === 'click' ? 'click-cell' : 'dbl-click-cell', field, value, item, $td);
|
|
that.trigger(e.type === 'click' ? 'click-row' : 'dbl-click-row', item, $tr);
|
|
|
|
// if click to select - then trigger the checkbox/radio click
|
|
if (e.type === 'click' && that.options.clickToSelect && column.clickToSelect) {
|
|
var $selectItem = $tr.find(sprintf('[name="%s"]', that.options.selectItemName));
|
|
if ($selectItem.length) {
|
|
$selectItem[0].click(); // #144: .trigger('click') bug
|
|
}
|
|
}
|
|
});
|
|
|
|
this.$body.find('> tr[data-index] > td > .detail-icon').off('click').on('click', function () {
|
|
var $this = $(this),
|
|
$tr = $this.parent().parent(),
|
|
index = $tr.data('index'),
|
|
row = data[index]; // Fix #980 Detail view, when searching, returns wrong row
|
|
|
|
// remove and update
|
|
if ($tr.next().is('tr.detail-view')) {
|
|
$this.find('i').attr('class', sprintf('%s %s', that.options.iconsPrefix, that.options.icons.detailOpen));
|
|
$tr.next().remove();
|
|
that.trigger('collapse-row', index, row);
|
|
} else {
|
|
$this.find('i').attr('class', sprintf('%s %s', that.options.iconsPrefix, that.options.icons.detailClose));
|
|
$tr.after(sprintf('<tr class="detail-view"><td colspan="%s"></td></tr>', $tr.find('td').length));
|
|
var $element = $tr.next().find('td');
|
|
var content = calculateObjectValue(that.options, that.options.detailFormatter, [index, row, $element], '');
|
|
if($element.length === 1) {
|
|
$element.append(content);
|
|
}
|
|
that.trigger('expand-row', index, row, $element);
|
|
}
|
|
that.resetView();
|
|
});
|
|
|
|
this.$selectItem = this.$body.find(sprintf('[name="%s"]', this.options.selectItemName));
|
|
this.$selectItem.off('click').on('click', function (event) {
|
|
event.stopImmediatePropagation();
|
|
|
|
var $this = $(this),
|
|
checked = $this.prop('checked'),
|
|
row = that.data[$this.data('index')];
|
|
|
|
if (that.options.maintainSelected && $(this).is(':radio')) {
|
|
$.each(that.options.data, function (i, row) {
|
|
row[that.header.stateField] = false;
|
|
});
|
|
}
|
|
|
|
row[that.header.stateField] = checked;
|
|
|
|
if (that.options.singleSelect) {
|
|
that.$selectItem.not(this).each(function () {
|
|
that.data[$(this).data('index')][that.header.stateField] = false;
|
|
});
|
|
that.$selectItem.filter(':checked').not(this).prop('checked', false);
|
|
}
|
|
|
|
that.updateSelected();
|
|
that.trigger(checked ? 'check' : 'uncheck', row, $this);
|
|
});
|
|
|
|
$.each(this.header.events, function (i, events) {
|
|
if (!events) {
|
|
return;
|
|
}
|
|
// fix bug, if events is defined with namespace
|
|
if (typeof events === 'string') {
|
|
events = calculateObjectValue(null, events);
|
|
}
|
|
|
|
var field = that.header.fields[i],
|
|
fieldIndex = $.inArray(field, that.getVisibleFields());
|
|
|
|
if (that.options.detailView && !that.options.cardView) {
|
|
fieldIndex += 1;
|
|
}
|
|
|
|
for (var key in events) {
|
|
that.$body.find('>tr:not(.no-records-found)').each(function () {
|
|
var $tr = $(this),
|
|
$td = $tr.find(that.options.cardView ? '.card-view' : 'td').eq(fieldIndex),
|
|
index = key.indexOf(' '),
|
|
name = key.substring(0, index),
|
|
el = key.substring(index + 1),
|
|
func = events[key];
|
|
|
|
$td.find(el).off(name).on(name, function (e) {
|
|
var index = $tr.data('index'),
|
|
row = that.data[index],
|
|
value = row[field];
|
|
|
|
func.apply(this, [e, value, row, index]);
|
|
});
|
|
});
|
|
}
|
|
});
|
|
|
|
this.updateSelected();
|
|
this.resetView();
|
|
|
|
this.trigger('post-body');
|
|
};
|
|
|
|
BootstrapTable.prototype.initServer = function (silent, query) {
|
|
var that = this,
|
|
data = {},
|
|
params = {
|
|
searchText: this.searchText,
|
|
sortName: this.options.sortName,
|
|
sortOrder: this.options.sortOrder
|
|
},
|
|
request;
|
|
|
|
if(this.options.pagination) {
|
|
params.pageSize = this.options.pageSize === this.options.formatAllRows() ?
|
|
this.options.totalRows : this.options.pageSize;
|
|
params.pageNumber = this.options.pageNumber;
|
|
}
|
|
|
|
if (!this.options.url && !this.options.ajax) {
|
|
return;
|
|
}
|
|
|
|
if (this.options.queryParamsType === 'limit') {
|
|
params = {
|
|
search: params.searchText,
|
|
sort: params.sortName,
|
|
order: params.sortOrder
|
|
};
|
|
if (this.options.pagination) {
|
|
params.limit = this.options.pageSize === this.options.formatAllRows() ?
|
|
this.options.totalRows : this.options.pageSize;
|
|
params.offset = this.options.pageSize === this.options.formatAllRows() ?
|
|
0 : this.options.pageSize * (this.options.pageNumber - 1);
|
|
}
|
|
}
|
|
|
|
if (!($.isEmptyObject(this.filterColumnsPartial))) {
|
|
params['filter'] = JSON.stringify(this.filterColumnsPartial, null);
|
|
}
|
|
|
|
data = calculateObjectValue(this.options, this.options.queryParams, [params], data);
|
|
|
|
$.extend(data, query || {});
|
|
|
|
// false to stop request
|
|
if (data === false) {
|
|
return;
|
|
}
|
|
|
|
if (!silent) {
|
|
this.$tableLoading.show();
|
|
}
|
|
request = $.extend({}, calculateObjectValue(null, this.options.ajaxOptions), {
|
|
type: this.options.method,
|
|
url: this.options.url,
|
|
data: this.options.contentType === 'application/json' && this.options.method === 'post' ?
|
|
JSON.stringify(data) : data,
|
|
cache: this.options.cache,
|
|
contentType: this.options.contentType,
|
|
dataType: this.options.dataType,
|
|
success: function (res) {
|
|
res = calculateObjectValue(that.options, that.options.responseHandler, [res], res);
|
|
|
|
that.load(res);
|
|
that.trigger('load-success', res);
|
|
if (!silent) that.$tableLoading.hide();
|
|
},
|
|
error: function (res) {
|
|
that.trigger('load-error', res.status, res);
|
|
if (!silent) that.$tableLoading.hide();
|
|
}
|
|
});
|
|
|
|
if (this.options.ajax) {
|
|
calculateObjectValue(this, this.options.ajax, [request], null);
|
|
} else {
|
|
$.ajax(request);
|
|
}
|
|
};
|
|
|
|
BootstrapTable.prototype.initSearchText = function () {
|
|
if (this.options.search) {
|
|
if (this.options.searchText !== '') {
|
|
var $search = this.$toolbar.find('.search input');
|
|
$search.val(this.options.searchText);
|
|
this.onSearch({currentTarget: $search});
|
|
}
|
|
}
|
|
};
|
|
|
|
BootstrapTable.prototype.getCaret = function () {
|
|
var that = this;
|
|
|
|
$.each(this.$header.find('th'), function (i, th) {
|
|
$(th).find('.sortable').removeClass('desc asc').addClass($(th).data('field') === that.options.sortName ? that.options.sortOrder : 'both');
|
|
});
|
|
};
|
|
|
|
BootstrapTable.prototype.updateSelected = function () {
|
|
var checkAll = this.$selectItem.filter(':enabled').length &&
|
|
this.$selectItem.filter(':enabled').length ===
|
|
this.$selectItem.filter(':enabled').filter(':checked').length;
|
|
|
|
this.$selectAll.add(this.$selectAll_).prop('checked', checkAll);
|
|
|
|
this.$selectItem.each(function () {
|
|
$(this).closest('tr')[$(this).prop('checked') ? 'addClass' : 'removeClass']('selected');
|
|
});
|
|
};
|
|
|
|
BootstrapTable.prototype.updateRows = function () {
|
|
var that = this;
|
|
|
|
this.$selectItem.each(function () {
|
|
that.data[$(this).data('index')][that.header.stateField] = $(this).prop('checked');
|
|
});
|
|
};
|
|
|
|
BootstrapTable.prototype.resetRows = function () {
|
|
var that = this;
|
|
|
|
$.each(this.data, function (i, row) {
|
|
that.$selectAll.prop('checked', false);
|
|
that.$selectItem.prop('checked', false);
|
|
if (that.header.stateField) {
|
|
row[that.header.stateField] = false;
|
|
}
|
|
});
|
|
};
|
|
|
|
BootstrapTable.prototype.trigger = function (name) {
|
|
var args = Array.prototype.slice.call(arguments, 1);
|
|
|
|
name += '.bs.table';
|
|
this.options[BootstrapTable.EVENTS[name]].apply(this.options, args);
|
|
this.$el.trigger($.Event(name), args);
|
|
|
|
this.options.onAll(name, args);
|
|
this.$el.trigger($.Event('all.bs.table'), [name, args]);
|
|
};
|
|
|
|
BootstrapTable.prototype.resetHeader = function () {
|
|
// fix #61: the hidden table reset header bug.
|
|
// fix bug: get $el.css('width') error sometime (height = 500)
|
|
clearTimeout(this.timeoutId_);
|
|
this.timeoutId_ = setTimeout($.proxy(this.fitHeader, this), this.$el.is(':hidden') ? 100 : 0);
|
|
};
|
|
|
|
BootstrapTable.prototype.fitHeader = function () {
|
|
var that = this,
|
|
fixedBody,
|
|
scrollWidth,
|
|
focused,
|
|
focusedTemp;
|
|
|
|
if (that.$el.is(':hidden')) {
|
|
that.timeoutId_ = setTimeout($.proxy(that.fitHeader, that), 100);
|
|
return;
|
|
}
|
|
fixedBody = this.$tableBody.get(0);
|
|
|
|
scrollWidth = fixedBody.scrollWidth > fixedBody.clientWidth &&
|
|
fixedBody.scrollHeight > fixedBody.clientHeight + this.$header.outerHeight() ?
|
|
getScrollBarWidth() : 0;
|
|
|
|
this.$el.css('margin-top', -this.$header.outerHeight());
|
|
|
|
focused = $(':focus');
|
|
if (focused.length > 0) {
|
|
var $th = focused.parents('th');
|
|
if ($th.length > 0) {
|
|
var dataField = $th.attr('data-field');
|
|
if (dataField !== undefined) {
|
|
var $headerTh = this.$header.find("[data-field='" + dataField + "']");
|
|
if ($headerTh.length > 0) {
|
|
$headerTh.find(":input").addClass("focus-temp");
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
this.$header_ = this.$header.clone(true, true);
|
|
this.$selectAll_ = this.$header_.find('[name="btSelectAll"]');
|
|
this.$tableHeader.css({
|
|
'margin-right': scrollWidth
|
|
}).find('table').css('width', this.$el.outerWidth())
|
|
.html('').attr('class', this.$el.attr('class'))
|
|
.append(this.$header_);
|
|
|
|
|
|
focusedTemp = $('.focus-temp:visible:eq(0)');
|
|
if (focusedTemp.length > 0) {
|
|
focusedTemp.focus();
|
|
this.$header.find('.focus-temp').removeClass('focus-temp');
|
|
}
|
|
|
|
// fix bug: $.data() is not working as expected after $.append()
|
|
this.$header.find('th[data-field]').each(function (i) {
|
|
that.$header_.find(sprintf('th[data-field="%s"]', $(this).data('field'))).data($(this).data());
|
|
});
|
|
|
|
var visibleFields = this.getVisibleFields();
|
|
|
|
this.$body.find('>tr:first-child:not(.no-records-found) > *').each(function (i) {
|
|
var $this = $(this),
|
|
index = i;
|
|
|
|
if (that.options.detailView && !that.options.cardView) {
|
|
if (i === 0) {
|
|
that.$header_.find('th.detail').find('.fht-cell').width($this.innerWidth());
|
|
}
|
|
index = i - 1;
|
|
}
|
|
|
|
that.$header_.find(sprintf('th[data-field="%s"]', visibleFields[index]))
|
|
.find('.fht-cell').width($this.innerWidth());
|
|
});
|
|
// horizontal scroll event
|
|
// TODO: it's probably better improving the layout than binding to scroll event
|
|
this.$tableBody.off('scroll').on('scroll', function () {
|
|
that.$tableHeader.scrollLeft($(this).scrollLeft());
|
|
|
|
if (that.options.showFooter && !that.options.cardView) {
|
|
that.$tableFooter.scrollLeft($(this).scrollLeft());
|
|
}
|
|
});
|
|
that.trigger('post-header');
|
|
};
|
|
|
|
BootstrapTable.prototype.resetFooter = function () {
|
|
var that = this,
|
|
data = that.getData(),
|
|
html = [];
|
|
|
|
if (!this.options.showFooter || this.options.cardView) { //do nothing
|
|
return;
|
|
}
|
|
|
|
if (!this.options.cardView && this.options.detailView) {
|
|
html.push('<td><div class="th-inner"> </div><div class="fht-cell"></div></td>');
|
|
}
|
|
|
|
$.each(this.columns, function (i, column) {
|
|
var falign = '', // footer align style
|
|
style = '',
|
|
class_ = sprintf(' class="%s"', column['class']);
|
|
|
|
if (!column.visible) {
|
|
return;
|
|
}
|
|
|
|
if (that.options.cardView && (!column.cardVisible)) {
|
|
return;
|
|
}
|
|
|
|
falign = sprintf('text-align: %s; ', column.falign ? column.falign : column.align);
|
|
style = sprintf('vertical-align: %s; ', column.valign);
|
|
|
|
html.push('<td', class_, sprintf(' style="%s"', falign + style), '>');
|
|
html.push('<div class="th-inner">');
|
|
|
|
html.push(calculateObjectValue(column, column.footerFormatter, [data], ' ') || ' ');
|
|
|
|
html.push('</div>');
|
|
html.push('<div class="fht-cell"></div>');
|
|
html.push('</div>');
|
|
html.push('</td>');
|
|
});
|
|
|
|
this.$tableFooter.find('tr').html(html.join(''));
|
|
clearTimeout(this.timeoutFooter_);
|
|
this.timeoutFooter_ = setTimeout($.proxy(this.fitFooter, this),
|
|
this.$el.is(':hidden') ? 100 : 0);
|
|
};
|
|
|
|
BootstrapTable.prototype.fitFooter = function () {
|
|
var that = this,
|
|
$footerTd,
|
|
elWidth,
|
|
scrollWidth;
|
|
|
|
clearTimeout(this.timeoutFooter_);
|
|
if (this.$el.is(':hidden')) {
|
|
this.timeoutFooter_ = setTimeout($.proxy(this.fitFooter, this), 100);
|
|
return;
|
|
}
|
|
|
|
elWidth = this.$el.css('width');
|
|
scrollWidth = elWidth > this.$tableBody.width() ? getScrollBarWidth() : 0;
|
|
|
|
this.$tableFooter.css({
|
|
'margin-right': scrollWidth
|
|
}).find('table').css('width', elWidth)
|
|
.attr('class', this.$el.attr('class'));
|
|
|
|
$footerTd = this.$tableFooter.find('td');
|
|
|
|
this.$body.find('>tr:first-child:not(.no-records-found) > *').each(function (i) {
|
|
var $this = $(this);
|
|
|
|
$footerTd.eq(i).find('.fht-cell').width($this.innerWidth());
|
|
});
|
|
};
|
|
|
|
BootstrapTable.prototype.toggleColumn = function (index, checked, needUpdate) {
|
|
if (index === -1) {
|
|
return;
|
|
}
|
|
this.columns[index].visible = checked;
|
|
this.initHeader();
|
|
this.initSearch();
|
|
this.initPagination();
|
|
this.initBody();
|
|
|
|
if (this.options.showColumns) {
|
|
var $items = this.$toolbar.find('.keep-open input').prop('disabled', false);
|
|
|
|
if (needUpdate) {
|
|
$items.filter(sprintf('[value="%s"]', index)).prop('checked', checked);
|
|
}
|
|
|
|
if ($items.filter(':checked').length <= this.options.minimumCountColumns) {
|
|
$items.filter(':checked').prop('disabled', true);
|
|
}
|
|
}
|
|
};
|
|
|
|
BootstrapTable.prototype.toggleRow = function (index, uniqueId, visible) {
|
|
if (index === -1) {
|
|
return;
|
|
}
|
|
|
|
this.$body.find(typeof index !== 'undefined' ?
|
|
sprintf('tr[data-index="%s"]', index) :
|
|
sprintf('tr[data-uniqueid="%s"]', uniqueId))
|
|
[visible ? 'show' : 'hide']();
|
|
};
|
|
|
|
BootstrapTable.prototype.getVisibleFields = function () {
|
|
var that = this,
|
|
visibleFields = [];
|
|
|
|
$.each(this.header.fields, function (j, field) {
|
|
var column = that.columns[getFieldIndex(that.columns, field)];
|
|
|
|
if (!column.visible) {
|
|
return;
|
|
}
|
|
visibleFields.push(field);
|
|
});
|
|
return visibleFields;
|
|
};
|
|
|
|
// PUBLIC FUNCTION DEFINITION
|
|
// =======================
|
|
|
|
BootstrapTable.prototype.resetView = function (params) {
|
|
var padding = 0;
|
|
|
|
if (params && params.height) {
|
|
this.options.height = params.height;
|
|
}
|
|
|
|
this.$selectAll.prop('checked', this.$selectItem.length > 0 &&
|
|
this.$selectItem.length === this.$selectItem.filter(':checked').length);
|
|
|
|
if (this.options.height) {
|
|
var toolbarHeight = getRealHeight(this.$toolbar),
|
|
paginationHeight = getRealHeight(this.$pagination),
|
|
height = this.options.height - toolbarHeight - paginationHeight;
|
|
|
|
this.$tableContainer.css('height', height + 'px');
|
|
}
|
|
|
|
if (this.options.cardView) {
|
|
// remove the element css
|
|
this.$el.css('margin-top', '0');
|
|
this.$tableContainer.css('padding-bottom', '0');
|
|
return;
|
|
}
|
|
|
|
if (this.options.showHeader && this.options.height) {
|
|
this.$tableHeader.show();
|
|
this.resetHeader();
|
|
padding += this.$header.outerHeight();
|
|
} else {
|
|
this.$tableHeader.hide();
|
|
this.trigger('post-header');
|
|
}
|
|
|
|
if (this.options.showFooter) {
|
|
this.resetFooter();
|
|
if (this.options.height) {
|
|
padding += this.$tableFooter.outerHeight() + 1;
|
|
}
|
|
}
|
|
|
|
// Assign the correct sortable arrow
|
|
this.getCaret();
|
|
this.$tableContainer.css('padding-bottom', padding + 'px');
|
|
this.trigger('reset-view');
|
|
};
|
|
|
|
BootstrapTable.prototype.getData = function (useCurrentPage) {
|
|
return (this.searchText || !$.isEmptyObject(this.filterColumns) || !$.isEmptyObject(this.filterColumnsPartial)) ?
|
|
(useCurrentPage ? this.data.slice(this.pageFrom - 1, this.pageTo) : this.data) :
|
|
(useCurrentPage ? this.options.data.slice(this.pageFrom - 1, this.pageTo) : this.options.data);
|
|
};
|
|
|
|
BootstrapTable.prototype.load = function (data) {
|
|
var fixedScroll = false;
|
|
|
|
// #431: support pagination
|
|
if (this.options.sidePagination === 'server') {
|
|
this.options.totalRows = data.total;
|
|
fixedScroll = data.fixedScroll;
|
|
data = data[this.options.dataField];
|
|
} else if (!$.isArray(data)) { // support fixedScroll
|
|
fixedScroll = data.fixedScroll;
|
|
data = data.data;
|
|
}
|
|
|
|
this.initData(data);
|
|
this.initSearch();
|
|
this.initPagination();
|
|
this.initBody(fixedScroll);
|
|
};
|
|
|
|
BootstrapTable.prototype.append = function (data) {
|
|
this.initData(data, 'append');
|
|
this.initSearch();
|
|
this.initPagination();
|
|
this.initSort();
|
|
this.initBody(true);
|
|
};
|
|
|
|
BootstrapTable.prototype.prepend = function (data) {
|
|
this.initData(data, 'prepend');
|
|
this.initSearch();
|
|
this.initPagination();
|
|
this.initSort();
|
|
this.initBody(true);
|
|
};
|
|
|
|
BootstrapTable.prototype.remove = function (params) {
|
|
var len = this.options.data.length,
|
|
i, row;
|
|
|
|
if (!params.hasOwnProperty('field') || !params.hasOwnProperty('values')) {
|
|
return;
|
|
}
|
|
|
|
for (i = len - 1; i >= 0; i--) {
|
|
row = this.options.data[i];
|
|
|
|
if (!row.hasOwnProperty(params.field)) {
|
|
continue;
|
|
}
|
|
if ($.inArray(row[params.field], params.values) !== -1) {
|
|
this.options.data.splice(i, 1);
|
|
}
|
|
}
|
|
|
|
if (len === this.options.data.length) {
|
|
return;
|
|
}
|
|
|
|
this.initSearch();
|
|
this.initPagination();
|
|
this.initSort();
|
|
this.initBody(true);
|
|
};
|
|
|
|
BootstrapTable.prototype.removeAll = function () {
|
|
if (this.options.data.length > 0) {
|
|
this.options.data.splice(0, this.options.data.length);
|
|
this.initSearch();
|
|
this.initPagination();
|
|
this.initBody(true);
|
|
}
|
|
};
|
|
|
|
BootstrapTable.prototype.getRowByUniqueId = function (id) {
|
|
var uniqueId = this.options.uniqueId,
|
|
len = this.options.data.length,
|
|
dataRow = null,
|
|
i, row, rowUniqueId;
|
|
|
|
for (i = len - 1; i >= 0; i--) {
|
|
row = this.options.data[i];
|
|
|
|
if (row.hasOwnProperty(uniqueId)) { // uniqueId is a column
|
|
rowUniqueId = row[uniqueId];
|
|
} else if(row._data.hasOwnProperty(uniqueId)) { // uniqueId is a row data property
|
|
rowUniqueId = row._data[uniqueId];
|
|
} else {
|
|
continue;
|
|
}
|
|
|
|
if (typeof rowUniqueId === 'string') {
|
|
id = id.toString();
|
|
} else if (typeof rowUniqueId === 'number') {
|
|
if ((Number(rowUniqueId) === rowUniqueId) && (rowUniqueId % 1 === 0)) {
|
|
id = parseInt(id);
|
|
} else if ((rowUniqueId === Number(rowUniqueId)) && (rowUniqueId !== 0)) {
|
|
id = parseFloat(id);
|
|
}
|
|
}
|
|
|
|
if (rowUniqueId === id) {
|
|
dataRow = row;
|
|
break;
|
|
}
|
|
}
|
|
|
|
return dataRow;
|
|
};
|
|
|
|
BootstrapTable.prototype.removeByUniqueId = function (id) {
|
|
var len = this.options.data.length,
|
|
row = this.getRowByUniqueId(id);
|
|
|
|
if (row) {
|
|
this.options.data.splice(this.options.data.indexOf(row), 1);
|
|
}
|
|
|
|
if (len === this.options.data.length) {
|
|
return;
|
|
}
|
|
|
|
this.initSearch();
|
|
this.initPagination();
|
|
this.initBody(true);
|
|
};
|
|
|
|
BootstrapTable.prototype.updateByUniqueId = function (params) {
|
|
var rowId;
|
|
|
|
if (!params.hasOwnProperty('id') || !params.hasOwnProperty('row')) {
|
|
return;
|
|
}
|
|
|
|
rowId = $.inArray(this.getRowByUniqueId(params.id), this.options.data);
|
|
|
|
if (rowId === -1) {
|
|
return;
|
|
}
|
|
|
|
$.extend(this.data[rowId], params.row);
|
|
this.initSort();
|
|
this.initBody(true);
|
|
};
|
|
|
|
BootstrapTable.prototype.insertRow = function (params) {
|
|
if (!params.hasOwnProperty('index') || !params.hasOwnProperty('row')) {
|
|
return;
|
|
}
|
|
this.data.splice(params.index, 0, params.row);
|
|
this.initSearch();
|
|
this.initPagination();
|
|
this.initSort();
|
|
this.initBody(true);
|
|
};
|
|
|
|
BootstrapTable.prototype.updateRow = function (params) {
|
|
if (!params.hasOwnProperty('index') || !params.hasOwnProperty('row')) {
|
|
return;
|
|
}
|
|
$.extend(this.data[params.index], params.row);
|
|
this.initSort();
|
|
this.initBody(true);
|
|
};
|
|
|
|
BootstrapTable.prototype.showRow = function (params) {
|
|
if (!params.hasOwnProperty('index') && !params.hasOwnProperty('uniqueId')) {
|
|
return;
|
|
}
|
|
this.toggleRow(params.index, params.uniqueId, true);
|
|
};
|
|
|
|
BootstrapTable.prototype.hideRow = function (params) {
|
|
if (!params.hasOwnProperty('index') && !params.hasOwnProperty('uniqueId')) {
|
|
return;
|
|
}
|
|
this.toggleRow(params.index, params.uniqueId, false);
|
|
};
|
|
|
|
BootstrapTable.prototype.getRowsHidden = function (show) {
|
|
var rows = $(this.$body[0]).children().filter(':hidden'),
|
|
i = 0;
|
|
if (show) {
|
|
for (; i < rows.length; i++) {
|
|
$(rows[i]).show();
|
|
}
|
|
}
|
|
return rows;
|
|
};
|
|
|
|
BootstrapTable.prototype.mergeCells = function (options) {
|
|
var row = options.index,
|
|
col = $.inArray(options.field, this.getVisibleFields()),
|
|
rowspan = options.rowspan || 1,
|
|
colspan = options.colspan || 1,
|
|
i, j,
|
|
$tr = this.$body.find('>tr'),
|
|
$td;
|
|
|
|
if (this.options.detailView && !this.options.cardView) {
|
|
col += 1;
|
|
}
|
|
|
|
$td = $tr.eq(row).find('>td').eq(col);
|
|
|
|
if (row < 0 || col < 0 || row >= this.data.length) {
|
|
return;
|
|
}
|
|
|
|
for (i = row; i < row + rowspan; i++) {
|
|
for (j = col; j < col + colspan; j++) {
|
|
$tr.eq(i).find('>td').eq(j).hide();
|
|
}
|
|
}
|
|
|
|
$td.attr('rowspan', rowspan).attr('colspan', colspan).show();
|
|
};
|
|
|
|
BootstrapTable.prototype.updateCell = function (params) {
|
|
if (!params.hasOwnProperty('index') ||
|
|
!params.hasOwnProperty('field') ||
|
|
!params.hasOwnProperty('value')) {
|
|
return;
|
|
}
|
|
this.data[params.index][params.field] = params.value;
|
|
|
|
if (params.reinit === false) {
|
|
return;
|
|
}
|
|
this.initSort();
|
|
this.initBody(true);
|
|
};
|
|
|
|
BootstrapTable.prototype.getOptions = function () {
|
|
return this.options;
|
|
};
|
|
|
|
BootstrapTable.prototype.getSelections = function () {
|
|
var that = this;
|
|
|
|
return $.grep(this.data, function (row) {
|
|
return row[that.header.stateField];
|
|
});
|
|
};
|
|
|
|
BootstrapTable.prototype.getAllSelections = function () {
|
|
var that = this;
|
|
|
|
return $.grep(this.options.data, function (row) {
|
|
return row[that.header.stateField];
|
|
});
|
|
};
|
|
|
|
BootstrapTable.prototype.checkAll = function () {
|
|
this.checkAll_(true);
|
|
};
|
|
|
|
BootstrapTable.prototype.uncheckAll = function () {
|
|
this.checkAll_(false);
|
|
};
|
|
|
|
BootstrapTable.prototype.checkInvert = function () {
|
|
var that = this;
|
|
var rows = that.$selectItem.filter(':enabled');
|
|
var checked = rows.filter(':checked');
|
|
rows.each(function() {
|
|
$(this).prop('checked', !$(this).prop('checked'));
|
|
});
|
|
that.updateRows();
|
|
that.updateSelected();
|
|
that.trigger('uncheck-some', checked);
|
|
checked = that.getSelections();
|
|
that.trigger('check-some', checked);
|
|
};
|
|
|
|
BootstrapTable.prototype.checkAll_ = function (checked) {
|
|
var rows;
|
|
if (!checked) {
|
|
rows = this.getSelections();
|
|
}
|
|
this.$selectAll.add(this.$selectAll_).prop('checked', checked);
|
|
this.$selectItem.filter(':enabled').prop('checked', checked);
|
|
this.updateRows();
|
|
if (checked) {
|
|
rows = this.getSelections();
|
|
}
|
|
this.trigger(checked ? 'check-all' : 'uncheck-all', rows);
|
|
};
|
|
|
|
BootstrapTable.prototype.check = function (index) {
|
|
this.check_(true, index);
|
|
};
|
|
|
|
BootstrapTable.prototype.uncheck = function (index) {
|
|
this.check_(false, index);
|
|
};
|
|
|
|
BootstrapTable.prototype.check_ = function (checked, index) {
|
|
var $el = this.$selectItem.filter(sprintf('[data-index="%s"]', index)).prop('checked', checked);
|
|
this.data[index][this.header.stateField] = checked;
|
|
this.updateSelected();
|
|
this.trigger(checked ? 'check' : 'uncheck', this.data[index], $el);
|
|
};
|
|
|
|
BootstrapTable.prototype.checkBy = function (obj) {
|
|
this.checkBy_(true, obj);
|
|
};
|
|
|
|
BootstrapTable.prototype.uncheckBy = function (obj) {
|
|
this.checkBy_(false, obj);
|
|
};
|
|
|
|
BootstrapTable.prototype.checkBy_ = function (checked, obj) {
|
|
if (!obj.hasOwnProperty('field') || !obj.hasOwnProperty('values')) {
|
|
return;
|
|
}
|
|
|
|
var that = this,
|
|
rows = [];
|
|
$.each(this.options.data, function (index, row) {
|
|
if (!row.hasOwnProperty(obj.field)) {
|
|
return false;
|
|
}
|
|
if ($.inArray(row[obj.field], obj.values) !== -1) {
|
|
var $el = that.$selectItem.filter(':enabled')
|
|
.filter(sprintf('[data-index="%s"]', index)).prop('checked', checked);
|
|
row[that.header.stateField] = checked;
|
|
rows.push(row);
|
|
that.trigger(checked ? 'check' : 'uncheck', row, $el);
|
|
}
|
|
});
|
|
this.updateSelected();
|
|
this.trigger(checked ? 'check-some' : 'uncheck-some', rows);
|
|
};
|
|
|
|
BootstrapTable.prototype.destroy = function () {
|
|
this.$el.insertBefore(this.$container);
|
|
$(this.options.toolbar).insertBefore(this.$el);
|
|
this.$container.next().remove();
|
|
this.$container.remove();
|
|
this.$el.html(this.$el_.html())
|
|
.css('margin-top', '0')
|
|
.attr('class', this.$el_.attr('class') || ''); // reset the class
|
|
};
|
|
|
|
BootstrapTable.prototype.showLoading = function () {
|
|
this.$tableLoading.show();
|
|
};
|
|
|
|
BootstrapTable.prototype.hideLoading = function () {
|
|
this.$tableLoading.hide();
|
|
};
|
|
|
|
BootstrapTable.prototype.togglePagination = function () {
|
|
this.options.pagination = !this.options.pagination;
|
|
var button = this.$toolbar.find('button[name="paginationSwitch"] i');
|
|
if (this.options.pagination) {
|
|
button.attr("class", this.options.iconsPrefix + " " + this.options.icons.paginationSwitchDown);
|
|
} else {
|
|
button.attr("class", this.options.iconsPrefix + " " + this.options.icons.paginationSwitchUp);
|
|
}
|
|
this.updatePagination();
|
|
};
|
|
|
|
BootstrapTable.prototype.refresh = function (params) {
|
|
if (params && params.url) {
|
|
this.options.url = params.url;
|
|
this.options.pageNumber = 1;
|
|
}
|
|
this.initServer(params && params.silent, params && params.query);
|
|
};
|
|
|
|
BootstrapTable.prototype.resetWidth = function () {
|
|
if (this.options.showHeader && this.options.height) {
|
|
this.fitHeader();
|
|
}
|
|
if (this.options.showFooter) {
|
|
this.fitFooter();
|
|
}
|
|
};
|
|
|
|
BootstrapTable.prototype.showColumn = function (field) {
|
|
this.toggleColumn(getFieldIndex(this.columns, field), true, true);
|
|
};
|
|
|
|
BootstrapTable.prototype.hideColumn = function (field) {
|
|
this.toggleColumn(getFieldIndex(this.columns, field), false, true);
|
|
};
|
|
|
|
BootstrapTable.prototype.getHiddenColumns = function () {
|
|
return $.grep(this.columns, function (column) {
|
|
return !column.visible;
|
|
});
|
|
};
|
|
|
|
BootstrapTable.prototype.filterBy = function (columns) {
|
|
this.filterColumns = $.isEmptyObject(columns) ? {} : columns;
|
|
this.options.pageNumber = 1;
|
|
this.initSearch();
|
|
this.updatePagination();
|
|
};
|
|
|
|
BootstrapTable.prototype.scrollTo = function (value) {
|
|
if (typeof value === 'string') {
|
|
value = value === 'bottom' ? this.$tableBody[0].scrollHeight : 0;
|
|
}
|
|
if (typeof value === 'number') {
|
|
this.$tableBody.scrollTop(value);
|
|
}
|
|
if (typeof value === 'undefined') {
|
|
return this.$tableBody.scrollTop();
|
|
}
|
|
};
|
|
|
|
BootstrapTable.prototype.getScrollPosition = function () {
|
|
return this.scrollTo();
|
|
};
|
|
|
|
BootstrapTable.prototype.selectPage = function (page) {
|
|
if (page > 0 && page <= this.options.totalPages) {
|
|
this.options.pageNumber = page;
|
|
this.updatePagination();
|
|
}
|
|
};
|
|
|
|
BootstrapTable.prototype.prevPage = function () {
|
|
if (this.options.pageNumber > 1) {
|
|
this.options.pageNumber--;
|
|
this.updatePagination();
|
|
}
|
|
};
|
|
|
|
BootstrapTable.prototype.nextPage = function () {
|
|
if (this.options.pageNumber < this.options.totalPages) {
|
|
this.options.pageNumber++;
|
|
this.updatePagination();
|
|
}
|
|
};
|
|
|
|
BootstrapTable.prototype.toggleView = function () {
|
|
this.options.cardView = !this.options.cardView;
|
|
this.initHeader();
|
|
// Fixed remove toolbar when click cardView button.
|
|
//that.initToolbar();
|
|
this.initBody();
|
|
this.trigger('toggle', this.options.cardView);
|
|
};
|
|
|
|
BootstrapTable.prototype.refreshOptions = function (options) {
|
|
//If the objects are equivalent then avoid the call of destroy / init methods
|
|
if (compareObjects(this.options, options, true)) {
|
|
return;
|
|
}
|
|
this.options = $.extend(this.options, options);
|
|
this.trigger('refresh-options', this.options);
|
|
this.destroy();
|
|
this.init();
|
|
};
|
|
|
|
BootstrapTable.prototype.resetSearch = function (text) {
|
|
var $search = this.$toolbar.find('.search input');
|
|
$search.val(text || '');
|
|
this.onSearch({currentTarget: $search});
|
|
};
|
|
|
|
BootstrapTable.prototype.expandRow_ = function (expand, index) {
|
|
var $tr = this.$body.find(sprintf('> tr[data-index="%s"]', index));
|
|
if ($tr.next().is('tr.detail-view') === (expand ? false : true)) {
|
|
$tr.find('> td > .detail-icon').click();
|
|
}
|
|
};
|
|
|
|
BootstrapTable.prototype.expandRow = function (index) {
|
|
this.expandRow_(true, index);
|
|
};
|
|
|
|
BootstrapTable.prototype.collapseRow = function (index) {
|
|
this.expandRow_(false, index);
|
|
};
|
|
|
|
BootstrapTable.prototype.expandAllRows = function (isSubTable) {
|
|
if (isSubTable) {
|
|
var $tr = this.$body.find(sprintf('> tr[data-index="%s"]', 0)),
|
|
that = this,
|
|
detailIcon = null,
|
|
executeInterval = false,
|
|
idInterval = -1;
|
|
|
|
if (!$tr.next().is('tr.detail-view')) {
|
|
$tr.find('> td > .detail-icon').click();
|
|
executeInterval = true;
|
|
} else if (!$tr.next().next().is('tr.detail-view')) {
|
|
$tr.next().find(".detail-icon").click();
|
|
executeInterval = true;
|
|
}
|
|
|
|
if (executeInterval) {
|
|
try {
|
|
idInterval = setInterval(function () {
|
|
detailIcon = that.$body.find("tr.detail-view").last().find(".detail-icon");
|
|
if (detailIcon.length > 0) {
|
|
detailIcon.click();
|
|
} else {
|
|
clearInterval(idInterval);
|
|
}
|
|
}, 1);
|
|
} catch (ex) {
|
|
clearInterval(idInterval);
|
|
}
|
|
}
|
|
} else {
|
|
var trs = this.$body.children();
|
|
for (var i = 0; i < trs.length; i++) {
|
|
this.expandRow_(true, $(trs[i]).data("index"));
|
|
}
|
|
}
|
|
};
|
|
|
|
BootstrapTable.prototype.collapseAllRows = function (isSubTable) {
|
|
if (isSubTable) {
|
|
this.expandRow_(false, 0);
|
|
} else {
|
|
var trs = this.$body.children();
|
|
for (var i = 0; i < trs.length; i++) {
|
|
this.expandRow_(false, $(trs[i]).data("index"));
|
|
}
|
|
}
|
|
};
|
|
|
|
BootstrapTable.prototype.updateFormatText = function (name, text) {
|
|
if (this.options[sprintf('format%s', name)]) {
|
|
if (typeof text === 'string') {
|
|
this.options[sprintf('format%s', name)] = function () {
|
|
return text;
|
|
};
|
|
} else if (typeof text === 'function') {
|
|
this.options[sprintf('format%s', name)] = text;
|
|
}
|
|
}
|
|
this.initToolbar();
|
|
this.initPagination();
|
|
this.initBody();
|
|
};
|
|
|
|
// BOOTSTRAP TABLE PLUGIN DEFINITION
|
|
// =======================
|
|
|
|
var allowedMethods = [
|
|
'getOptions',
|
|
'getSelections', 'getAllSelections', 'getData',
|
|
'load', 'append', 'prepend', 'remove', 'removeAll',
|
|
'insertRow', 'updateRow', 'updateCell', 'updateByUniqueId', 'removeByUniqueId',
|
|
'getRowByUniqueId', 'showRow', 'hideRow', 'getRowsHidden',
|
|
'mergeCells',
|
|
'checkAll', 'uncheckAll', 'checkInvert',
|
|
'check', 'uncheck',
|
|
'checkBy', 'uncheckBy',
|
|
'refresh',
|
|
'resetView',
|
|
'resetWidth',
|
|
'destroy',
|
|
'showLoading', 'hideLoading',
|
|
'showColumn', 'hideColumn', 'getHiddenColumns',
|
|
'filterBy',
|
|
'scrollTo',
|
|
'getScrollPosition',
|
|
'selectPage', 'prevPage', 'nextPage',
|
|
'togglePagination',
|
|
'toggleView',
|
|
'refreshOptions',
|
|
'resetSearch',
|
|
'expandRow', 'collapseRow', 'expandAllRows', 'collapseAllRows',
|
|
'updateFormatText'
|
|
];
|
|
|
|
$.fn.bootstrapTable = function (option) {
|
|
var value,
|
|
args = Array.prototype.slice.call(arguments, 1);
|
|
|
|
this.each(function () {
|
|
var $this = $(this),
|
|
data = $this.data('bootstrap.table'),
|
|
options = $.extend({}, BootstrapTable.DEFAULTS, $this.data(),
|
|
typeof option === 'object' && option);
|
|
|
|
if (typeof option === 'string') {
|
|
if ($.inArray(option, allowedMethods) < 0) {
|
|
throw new Error("Unknown method: " + option);
|
|
}
|
|
|
|
if (!data) {
|
|
return;
|
|
}
|
|
|
|
value = data[option].apply(data, args);
|
|
|
|
if (option === 'destroy') {
|
|
$this.removeData('bootstrap.table');
|
|
}
|
|
}
|
|
|
|
if (!data) {
|
|
$this.data('bootstrap.table', (data = new BootstrapTable(this, options)));
|
|
}
|
|
});
|
|
|
|
return typeof value === 'undefined' ? this : value;
|
|
};
|
|
|
|
$.fn.bootstrapTable.Constructor = BootstrapTable;
|
|
$.fn.bootstrapTable.defaults = BootstrapTable.DEFAULTS;
|
|
$.fn.bootstrapTable.columnDefaults = BootstrapTable.COLUMN_DEFAULTS;
|
|
$.fn.bootstrapTable.locales = BootstrapTable.LOCALES;
|
|
$.fn.bootstrapTable.methods = allowedMethods;
|
|
$.fn.bootstrapTable.utils = {
|
|
sprintf: sprintf,
|
|
getFieldIndex: getFieldIndex,
|
|
compareObjects: compareObjects,
|
|
calculateObjectValue: calculateObjectValue
|
|
};
|
|
|
|
// BOOTSTRAP TABLE INIT
|
|
// =======================
|
|
|
|
$(function () {
|
|
$('[data-toggle="table"]').bootstrapTable();
|
|
});
|
|
}(jQuery);
|
|
|
|
/*!
|
|
* Bootstrap v3.3.6 (http://getbootstrap.com)
|
|
* Copyright 2011-2015 Twitter, Inc.
|
|
* Licensed under the MIT license
|
|
*/
|
|
|
|
if (typeof jQuery === 'undefined') {
|
|
throw new Error('Bootstrap\'s JavaScript requires jQuery')
|
|
}
|
|
|
|
+function ($) {
|
|
'use strict';
|
|
var version = $.fn.jquery.split(' ')[0].split('.')
|
|
if ((version[0] < 2 && version[1] < 9) || (version[0] == 1 && version[1] == 9 && version[2] < 1) || (version[0] > 2)) {
|
|
throw new Error('Bootstrap\'s JavaScript requires jQuery version 1.9.1 or higher, but lower than version 3')
|
|
}
|
|
}(jQuery);
|
|
|
|
/* ========================================================================
|
|
* Bootstrap: transition.js v3.3.6
|
|
* http://getbootstrap.com/javascript/#transitions
|
|
* ========================================================================
|
|
* Copyright 2011-2015 Twitter, Inc.
|
|
* Licensed under MIT (https://github.com/twbs/bootstrap/blob/master/LICENSE)
|
|
* ======================================================================== */
|
|
|
|
|
|
+function ($) {
|
|
'use strict';
|
|
|
|
// CSS TRANSITION SUPPORT (Shoutout: http://www.modernizr.com/)
|
|
// ============================================================
|
|
|
|
function transitionEnd() {
|
|
var el = document.createElement('bootstrap')
|
|
|
|
var transEndEventNames = {
|
|
WebkitTransition : 'webkitTransitionEnd',
|
|
MozTransition : 'transitionend',
|
|
OTransition : 'oTransitionEnd otransitionend',
|
|
transition : 'transitionend'
|
|
}
|
|
|
|
for (var name in transEndEventNames) {
|
|
if (el.style[name] !== undefined) {
|
|
return { end: transEndEventNames[name] }
|
|
}
|
|
}
|
|
|
|
return false // explicit for ie8 ( ._.)
|
|
}
|
|
|
|
// http://blog.alexmaccaw.com/css-transitions
|
|
$.fn.emulateTransitionEnd = function (duration) {
|
|
var called = false
|
|
var $el = this
|
|
$(this).one('bsTransitionEnd', function () { called = true })
|
|
var callback = function () { if (!called) $($el).trigger($.support.transition.end) }
|
|
setTimeout(callback, duration)
|
|
return this
|
|
}
|
|
|
|
$(function () {
|
|
$.support.transition = transitionEnd()
|
|
|
|
if (!$.support.transition) return
|
|
|
|
$.event.special.bsTransitionEnd = {
|
|
bindType: $.support.transition.end,
|
|
delegateType: $.support.transition.end,
|
|
handle: function (e) {
|
|
if ($(e.target).is(this)) return e.handleObj.handler.apply(this, arguments)
|
|
}
|
|
}
|
|
})
|
|
|
|
}(jQuery);
|
|
|
|
/* ========================================================================
|
|
* Bootstrap: alert.js v3.3.6
|
|
* http://getbootstrap.com/javascript/#alerts
|
|
* ========================================================================
|
|
* Copyright 2011-2015 Twitter, Inc.
|
|
* Licensed under MIT (https://github.com/twbs/bootstrap/blob/master/LICENSE)
|
|
* ======================================================================== */
|
|
|
|
|
|
+function ($) {
|
|
'use strict';
|
|
|
|
// ALERT CLASS DEFINITION
|
|
// ======================
|
|
|
|
var dismiss = '[data-dismiss="alert"]'
|
|
var Alert = function (el) {
|
|
$(el).on('click', dismiss, this.close)
|
|
}
|
|
|
|
Alert.VERSION = '3.3.6'
|
|
|
|
Alert.TRANSITION_DURATION = 150
|
|
|
|
Alert.prototype.close = function (e) {
|
|
var $this = $(this)
|
|
var selector = $this.attr('data-target')
|
|
|
|
if (!selector) {
|
|
selector = $this.attr('href')
|
|
selector = selector && selector.replace(/.*(?=#[^\s]*$)/, '') // strip for ie7
|
|
}
|
|
|
|
var $parent = $(selector)
|
|
|
|
if (e) e.preventDefault()
|
|
|
|
if (!$parent.length) {
|
|
$parent = $this.closest('.alert')
|
|
}
|
|
|
|
$parent.trigger(e = $.Event('close.bs.alert'))
|
|
|
|
if (e.isDefaultPrevented()) return
|
|
|
|
$parent.removeClass('in')
|
|
|
|
function removeElement() {
|
|
// detach from parent, fire event then clean up data
|
|
$parent.detach().trigger('closed.bs.alert').remove()
|
|
}
|
|
|
|
$.support.transition && $parent.hasClass('fade') ?
|
|
$parent
|
|
.one('bsTransitionEnd', removeElement)
|
|
.emulateTransitionEnd(Alert.TRANSITION_DURATION) :
|
|
removeElement()
|
|
}
|
|
|
|
|
|
// ALERT PLUGIN DEFINITION
|
|
// =======================
|
|
|
|
function Plugin(option) {
|
|
return this.each(function () {
|
|
var $this = $(this)
|
|
var data = $this.data('bs.alert')
|
|
|
|
if (!data) $this.data('bs.alert', (data = new Alert(this)))
|
|
if (typeof option == 'string') data[option].call($this)
|
|
})
|
|
}
|
|
|
|
var old = $.fn.alert
|
|
|
|
$.fn.alert = Plugin
|
|
$.fn.alert.Constructor = Alert
|
|
|
|
|
|
// ALERT NO CONFLICT
|
|
// =================
|
|
|
|
$.fn.alert.noConflict = function () {
|
|
$.fn.alert = old
|
|
return this
|
|
}
|
|
|
|
|
|
// ALERT DATA-API
|
|
// ==============
|
|
|
|
$(document).on('click.bs.alert.data-api', dismiss, Alert.prototype.close)
|
|
|
|
}(jQuery);
|
|
|
|
/* ========================================================================
|
|
* Bootstrap: button.js v3.3.6
|
|
* http://getbootstrap.com/javascript/#buttons
|
|
* ========================================================================
|
|
* Copyright 2011-2015 Twitter, Inc.
|
|
* Licensed under MIT (https://github.com/twbs/bootstrap/blob/master/LICENSE)
|
|
* ======================================================================== */
|
|
|
|
|
|
+function ($) {
|
|
'use strict';
|
|
|
|
// BUTTON PUBLIC CLASS DEFINITION
|
|
// ==============================
|
|
|
|
var Button = function (element, options) {
|
|
this.$element = $(element)
|
|
this.options = $.extend({}, Button.DEFAULTS, options)
|
|
this.isLoading = false
|
|
}
|
|
|
|
Button.VERSION = '3.3.6'
|
|
|
|
Button.DEFAULTS = {
|
|
loadingText: 'loading...'
|
|
}
|
|
|
|
Button.prototype.setState = function (state) {
|
|
var d = 'disabled'
|
|
var $el = this.$element
|
|
var val = $el.is('input') ? 'val' : 'html'
|
|
var data = $el.data()
|
|
|
|
state += 'Text'
|
|
|
|
if (data.resetText == null) $el.data('resetText', $el[val]())
|
|
|
|
// push to event loop to allow forms to submit
|
|
setTimeout($.proxy(function () {
|
|
$el[val](data[state] == null ? this.options[state] : data[state])
|
|
|
|
if (state == 'loadingText') {
|
|
this.isLoading = true
|
|
$el.addClass(d).attr(d, d)
|
|
} else if (this.isLoading) {
|
|
this.isLoading = false
|
|
$el.removeClass(d).removeAttr(d)
|
|
}
|
|
}, this), 0)
|
|
}
|
|
|
|
Button.prototype.toggle = function () {
|
|
var changed = true
|
|
var $parent = this.$element.closest('[data-toggle="buttons"]')
|
|
|
|
if ($parent.length) {
|
|
var $input = this.$element.find('input')
|
|
if ($input.prop('type') == 'radio') {
|
|
if ($input.prop('checked')) changed = false
|
|
$parent.find('.active').removeClass('active')
|
|
this.$element.addClass('active')
|
|
} else if ($input.prop('type') == 'checkbox') {
|
|
if (($input.prop('checked')) !== this.$element.hasClass('active')) changed = false
|
|
this.$element.toggleClass('active')
|
|
}
|
|
$input.prop('checked', this.$element.hasClass('active'))
|
|
if (changed) $input.trigger('change')
|
|
} else {
|
|
this.$element.attr('aria-pressed', !this.$element.hasClass('active'))
|
|
this.$element.toggleClass('active')
|
|
}
|
|
}
|
|
|
|
|
|
// BUTTON PLUGIN DEFINITION
|
|
// ========================
|
|
|
|
function Plugin(option) {
|
|
return this.each(function () {
|
|
var $this = $(this)
|
|
var data = $this.data('bs.button')
|
|
var options = typeof option == 'object' && option
|
|
|
|
if (!data) $this.data('bs.button', (data = new Button(this, options)))
|
|
|
|
if (option == 'toggle') data.toggle()
|
|
else if (option) data.setState(option)
|
|
})
|
|
}
|
|
|
|
var old = $.fn.button
|
|
|
|
$.fn.button = Plugin
|
|
$.fn.button.Constructor = Button
|
|
|
|
|
|
// BUTTON NO CONFLICT
|
|
// ==================
|
|
|
|
$.fn.button.noConflict = function () {
|
|
$.fn.button = old
|
|
return this
|
|
}
|
|
|
|
|
|
// BUTTON DATA-API
|
|
// ===============
|
|
|
|
$(document)
|
|
.on('click.bs.button.data-api', '[data-toggle^="button"]', function (e) {
|
|
var $btn = $(e.target)
|
|
if (!$btn.hasClass('btn')) $btn = $btn.closest('.btn')
|
|
Plugin.call($btn, 'toggle')
|
|
if (!($(e.target).is('input[type="radio"]') || $(e.target).is('input[type="checkbox"]'))) e.preventDefault()
|
|
})
|
|
.on('focus.bs.button.data-api blur.bs.button.data-api', '[data-toggle^="button"]', function (e) {
|
|
$(e.target).closest('.btn').toggleClass('focus', /^focus(in)?$/.test(e.type))
|
|
})
|
|
|
|
}(jQuery);
|
|
|
|
/* ========================================================================
|
|
* Bootstrap: carousel.js v3.3.6
|
|
* http://getbootstrap.com/javascript/#carousel
|
|
* ========================================================================
|
|
* Copyright 2011-2015 Twitter, Inc.
|
|
* Licensed under MIT (https://github.com/twbs/bootstrap/blob/master/LICENSE)
|
|
* ======================================================================== */
|
|
|
|
|
|
+function ($) {
|
|
'use strict';
|
|
|
|
// CAROUSEL CLASS DEFINITION
|
|
// =========================
|
|
|
|
var Carousel = function (element, options) {
|
|
this.$element = $(element)
|
|
this.$indicators = this.$element.find('.carousel-indicators')
|
|
this.options = options
|
|
this.paused = null
|
|
this.sliding = null
|
|
this.interval = null
|
|
this.$active = null
|
|
this.$items = null
|
|
|
|
this.options.keyboard && this.$element.on('keydown.bs.carousel', $.proxy(this.keydown, this))
|
|
|
|
this.options.pause == 'hover' && !('ontouchstart' in document.documentElement) && this.$element
|
|
.on('mouseenter.bs.carousel', $.proxy(this.pause, this))
|
|
.on('mouseleave.bs.carousel', $.proxy(this.cycle, this))
|
|
}
|
|
|
|
Carousel.VERSION = '3.3.6'
|
|
|
|
Carousel.TRANSITION_DURATION = 600
|
|
|
|
Carousel.DEFAULTS = {
|
|
interval: 5000,
|
|
pause: 'hover',
|
|
wrap: true,
|
|
keyboard: true
|
|
}
|
|
|
|
Carousel.prototype.keydown = function (e) {
|
|
if (/input|textarea/i.test(e.target.tagName)) return
|
|
switch (e.which) {
|
|
case 37: this.prev(); break
|
|
case 39: this.next(); break
|
|
default: return
|
|
}
|
|
|
|
e.preventDefault()
|
|
}
|
|
|
|
Carousel.prototype.cycle = function (e) {
|
|
e || (this.paused = false)
|
|
|
|
this.interval && clearInterval(this.interval)
|
|
|
|
this.options.interval
|
|
&& !this.paused
|
|
&& (this.interval = setInterval($.proxy(this.next, this), this.options.interval))
|
|
|
|
return this
|
|
}
|
|
|
|
Carousel.prototype.getItemIndex = function (item) {
|
|
this.$items = item.parent().children('.item')
|
|
return this.$items.index(item || this.$active)
|
|
}
|
|
|
|
Carousel.prototype.getItemForDirection = function (direction, active) {
|
|
var activeIndex = this.getItemIndex(active)
|
|
var willWrap = (direction == 'prev' && activeIndex === 0)
|
|
|| (direction == 'next' && activeIndex == (this.$items.length - 1))
|
|
if (willWrap && !this.options.wrap) return active
|
|
var delta = direction == 'prev' ? -1 : 1
|
|
var itemIndex = (activeIndex + delta) % this.$items.length
|
|
return this.$items.eq(itemIndex)
|
|
}
|
|
|
|
Carousel.prototype.to = function (pos) {
|
|
var that = this
|
|
var activeIndex = this.getItemIndex(this.$active = this.$element.find('.item.active'))
|
|
|
|
if (pos > (this.$items.length - 1) || pos < 0) return
|
|
|
|
if (this.sliding) return this.$element.one('slid.bs.carousel', function () { that.to(pos) }) // yes, "slid"
|
|
if (activeIndex == pos) return this.pause().cycle()
|
|
|
|
return this.slide(pos > activeIndex ? 'next' : 'prev', this.$items.eq(pos))
|
|
}
|
|
|
|
Carousel.prototype.pause = function (e) {
|
|
e || (this.paused = true)
|
|
|
|
if (this.$element.find('.next, .prev').length && $.support.transition) {
|
|
this.$element.trigger($.support.transition.end)
|
|
this.cycle(true)
|
|
}
|
|
|
|
this.interval = clearInterval(this.interval)
|
|
|
|
return this
|
|
}
|
|
|
|
Carousel.prototype.next = function () {
|
|
if (this.sliding) return
|
|
return this.slide('next')
|
|
}
|
|
|
|
Carousel.prototype.prev = function () {
|
|
if (this.sliding) return
|
|
return this.slide('prev')
|
|
}
|
|
|
|
Carousel.prototype.slide = function (type, next) {
|
|
var $active = this.$element.find('.item.active')
|
|
var $next = next || this.getItemForDirection(type, $active)
|
|
var isCycling = this.interval
|
|
var direction = type == 'next' ? 'left' : 'right'
|
|
var that = this
|
|
|
|
if ($next.hasClass('active')) return (this.sliding = false)
|
|
|
|
var relatedTarget = $next[0]
|
|
var slideEvent = $.Event('slide.bs.carousel', {
|
|
relatedTarget: relatedTarget,
|
|
direction: direction
|
|
})
|
|
this.$element.trigger(slideEvent)
|
|
if (slideEvent.isDefaultPrevented()) return
|
|
|
|
this.sliding = true
|
|
|
|
isCycling && this.pause()
|
|
|
|
if (this.$indicators.length) {
|
|
this.$indicators.find('.active').removeClass('active')
|
|
var $nextIndicator = $(this.$indicators.children()[this.getItemIndex($next)])
|
|
$nextIndicator && $nextIndicator.addClass('active')
|
|
}
|
|
|
|
var slidEvent = $.Event('slid.bs.carousel', { relatedTarget: relatedTarget, direction: direction }) // yes, "slid"
|
|
if ($.support.transition && this.$element.hasClass('slide')) {
|
|
$next.addClass(type)
|
|
$next[0].offsetWidth // force reflow
|
|
$active.addClass(direction)
|
|
$next.addClass(direction)
|
|
$active
|
|
.one('bsTransitionEnd', function () {
|
|
$next.removeClass([type, direction].join(' ')).addClass('active')
|
|
$active.removeClass(['active', direction].join(' '))
|
|
that.sliding = false
|
|
setTimeout(function () {
|
|
that.$element.trigger(slidEvent)
|
|
}, 0)
|
|
})
|
|
.emulateTransitionEnd(Carousel.TRANSITION_DURATION)
|
|
} else {
|
|
$active.removeClass('active')
|
|
$next.addClass('active')
|
|
this.sliding = false
|
|
this.$element.trigger(slidEvent)
|
|
}
|
|
|
|
isCycling && this.cycle()
|
|
|
|
return this
|
|
}
|
|
|
|
|
|
// CAROUSEL PLUGIN DEFINITION
|
|
// ==========================
|
|
|
|
function Plugin(option) {
|
|
return this.each(function () {
|
|
var $this = $(this)
|
|
var data = $this.data('bs.carousel')
|
|
var options = $.extend({}, Carousel.DEFAULTS, $this.data(), typeof option == 'object' && option)
|
|
var action = typeof option == 'string' ? option : options.slide
|
|
|
|
if (!data) $this.data('bs.carousel', (data = new Carousel(this, options)))
|
|
if (typeof option == 'number') data.to(option)
|
|
else if (action) data[action]()
|
|
else if (options.interval) data.pause().cycle()
|
|
})
|
|
}
|
|
|
|
var old = $.fn.carousel
|
|
|
|
$.fn.carousel = Plugin
|
|
$.fn.carousel.Constructor = Carousel
|
|
|
|
|
|
// CAROUSEL NO CONFLICT
|
|
// ====================
|
|
|
|
$.fn.carousel.noConflict = function () {
|
|
$.fn.carousel = old
|
|
return this
|
|
}
|
|
|
|
|
|
// CAROUSEL DATA-API
|
|
// =================
|
|
|
|
var clickHandler = function (e) {
|
|
var href
|
|
var $this = $(this)
|
|
var $target = $($this.attr('data-target') || (href = $this.attr('href')) && href.replace(/.*(?=#[^\s]+$)/, '')) // strip for ie7
|
|
if (!$target.hasClass('carousel')) return
|
|
var options = $.extend({}, $target.data(), $this.data())
|
|
var slideIndex = $this.attr('data-slide-to')
|
|
if (slideIndex) options.interval = false
|
|
|
|
Plugin.call($target, options)
|
|
|
|
if (slideIndex) {
|
|
$target.data('bs.carousel').to(slideIndex)
|
|
}
|
|
|
|
e.preventDefault()
|
|
}
|
|
|
|
$(document)
|
|
.on('click.bs.carousel.data-api', '[data-slide]', clickHandler)
|
|
.on('click.bs.carousel.data-api', '[data-slide-to]', clickHandler)
|
|
|
|
$(window).on('load', function () {
|
|
$('[data-ride="carousel"]').each(function () {
|
|
var $carousel = $(this)
|
|
Plugin.call($carousel, $carousel.data())
|
|
})
|
|
})
|
|
|
|
}(jQuery);
|
|
|
|
/* ========================================================================
|
|
* Bootstrap: collapse.js v3.3.6
|
|
* http://getbootstrap.com/javascript/#collapse
|
|
* ========================================================================
|
|
* Copyright 2011-2015 Twitter, Inc.
|
|
* Licensed under MIT (https://github.com/twbs/bootstrap/blob/master/LICENSE)
|
|
* ======================================================================== */
|
|
|
|
|
|
+function ($) {
|
|
'use strict';
|
|
|
|
// COLLAPSE PUBLIC CLASS DEFINITION
|
|
// ================================
|
|
|
|
var Collapse = function (element, options) {
|
|
this.$element = $(element)
|
|
this.options = $.extend({}, Collapse.DEFAULTS, options)
|
|
this.$trigger = $('[data-toggle="collapse"][href="#' + element.id + '"],' +
|
|
'[data-toggle="collapse"][data-target="#' + element.id + '"]')
|
|
this.transitioning = null
|
|
|
|
if (this.options.parent) {
|
|
this.$parent = this.getParent()
|
|
} else {
|
|
this.addAriaAndCollapsedClass(this.$element, this.$trigger)
|
|
}
|
|
|
|
if (this.options.toggle) this.toggle()
|
|
}
|
|
|
|
Collapse.VERSION = '3.3.6'
|
|
|
|
Collapse.TRANSITION_DURATION = 350
|
|
|
|
Collapse.DEFAULTS = {
|
|
toggle: true
|
|
}
|
|
|
|
Collapse.prototype.dimension = function () {
|
|
var hasWidth = this.$element.hasClass('width')
|
|
return hasWidth ? 'width' : 'height'
|
|
}
|
|
|
|
Collapse.prototype.show = function () {
|
|
if (this.transitioning || this.$element.hasClass('in')) return
|
|
|
|
var activesData
|
|
var actives = this.$parent && this.$parent.children('.panel').children('.in, .collapsing')
|
|
|
|
if (actives && actives.length) {
|
|
activesData = actives.data('bs.collapse')
|
|
if (activesData && activesData.transitioning) return
|
|
}
|
|
|
|
var startEvent = $.Event('show.bs.collapse')
|
|
this.$element.trigger(startEvent)
|
|
if (startEvent.isDefaultPrevented()) return
|
|
|
|
if (actives && actives.length) {
|
|
Plugin.call(actives, 'hide')
|
|
activesData || actives.data('bs.collapse', null)
|
|
}
|
|
|
|
var dimension = this.dimension()
|
|
|
|
this.$element
|
|
.removeClass('collapse')
|
|
.addClass('collapsing')[dimension](0)
|
|
.attr('aria-expanded', true)
|
|
|
|
this.$trigger
|
|
.removeClass('collapsed')
|
|
.attr('aria-expanded', true)
|
|
|
|
this.transitioning = 1
|
|
|
|
var complete = function () {
|
|
this.$element
|
|
.removeClass('collapsing')
|
|
.addClass('collapse in')[dimension]('')
|
|
this.transitioning = 0
|
|
this.$element
|
|
.trigger('shown.bs.collapse')
|
|
}
|
|
|
|
if (!$.support.transition) return complete.call(this)
|
|
|
|
var scrollSize = $.camelCase(['scroll', dimension].join('-'))
|
|
|
|
this.$element
|
|
.one('bsTransitionEnd', $.proxy(complete, this))
|
|
.emulateTransitionEnd(Collapse.TRANSITION_DURATION)[dimension](this.$element[0][scrollSize])
|
|
}
|
|
|
|
Collapse.prototype.hide = function () {
|
|
if (this.transitioning || !this.$element.hasClass('in')) return
|
|
|
|
var startEvent = $.Event('hide.bs.collapse')
|
|
this.$element.trigger(startEvent)
|
|
if (startEvent.isDefaultPrevented()) return
|
|
|
|
var dimension = this.dimension()
|
|
|
|
this.$element[dimension](this.$element[dimension]())[0].offsetHeight
|
|
|
|
this.$element
|
|
.addClass('collapsing')
|
|
.removeClass('collapse in')
|
|
.attr('aria-expanded', false)
|
|
|
|
this.$trigger
|
|
.addClass('collapsed')
|
|
.attr('aria-expanded', false)
|
|
|
|
this.transitioning = 1
|
|
|
|
var complete = function () {
|
|
this.transitioning = 0
|
|
this.$element
|
|
.removeClass('collapsing')
|
|
.addClass('collapse')
|
|
.trigger('hidden.bs.collapse')
|
|
}
|
|
|
|
if (!$.support.transition) return complete.call(this)
|
|
|
|
this.$element
|
|
[dimension](0)
|
|
.one('bsTransitionEnd', $.proxy(complete, this))
|
|
.emulateTransitionEnd(Collapse.TRANSITION_DURATION)
|
|
}
|
|
|
|
Collapse.prototype.toggle = function () {
|
|
this[this.$element.hasClass('in') ? 'hide' : 'show']()
|
|
}
|
|
|
|
Collapse.prototype.getParent = function () {
|
|
return $(this.options.parent)
|
|
.find('[data-toggle="collapse"][data-parent="' + this.options.parent + '"]')
|
|
.each($.proxy(function (i, element) {
|
|
var $element = $(element)
|
|
this.addAriaAndCollapsedClass(getTargetFromTrigger($element), $element)
|
|
}, this))
|
|
.end()
|
|
}
|
|
|
|
Collapse.prototype.addAriaAndCollapsedClass = function ($element, $trigger) {
|
|
var isOpen = $element.hasClass('in')
|
|
|
|
$element.attr('aria-expanded', isOpen)
|
|
$trigger
|
|
.toggleClass('collapsed', !isOpen)
|
|
.attr('aria-expanded', isOpen)
|
|
}
|
|
|
|
function getTargetFromTrigger($trigger) {
|
|
var href
|
|
var target = $trigger.attr('data-target')
|
|
|| (href = $trigger.attr('href')) && href.replace(/.*(?=#[^\s]+$)/, '') // strip for ie7
|
|
|
|
return $(target)
|
|
}
|
|
|
|
|
|
// COLLAPSE PLUGIN DEFINITION
|
|
// ==========================
|
|
|
|
function Plugin(option) {
|
|
return this.each(function () {
|
|
var $this = $(this)
|
|
var data = $this.data('bs.collapse')
|
|
var options = $.extend({}, Collapse.DEFAULTS, $this.data(), typeof option == 'object' && option)
|
|
|
|
if (!data && options.toggle && /show|hide/.test(option)) options.toggle = false
|
|
if (!data) $this.data('bs.collapse', (data = new Collapse(this, options)))
|
|
if (typeof option == 'string') data[option]()
|
|
})
|
|
}
|
|
|
|
var old = $.fn.collapse
|
|
|
|
$.fn.collapse = Plugin
|
|
$.fn.collapse.Constructor = Collapse
|
|
|
|
|
|
// COLLAPSE NO CONFLICT
|
|
// ====================
|
|
|
|
$.fn.collapse.noConflict = function () {
|
|
$.fn.collapse = old
|
|
return this
|
|
}
|
|
|
|
|
|
// COLLAPSE DATA-API
|
|
// =================
|
|
|
|
$(document).on('click.bs.collapse.data-api', '[data-toggle="collapse"]', function (e) {
|
|
var $this = $(this)
|
|
|
|
if (!$this.attr('data-target')) e.preventDefault()
|
|
|
|
var $target = getTargetFromTrigger($this)
|
|
var data = $target.data('bs.collapse')
|
|
var option = data ? 'toggle' : $this.data()
|
|
|
|
Plugin.call($target, option)
|
|
})
|
|
|
|
}(jQuery);
|
|
|
|
/* ========================================================================
|
|
* Bootstrap: dropdown.js v3.3.6
|
|
* http://getbootstrap.com/javascript/#dropdowns
|
|
* ========================================================================
|
|
* Copyright 2011-2015 Twitter, Inc.
|
|
* Licensed under MIT (https://github.com/twbs/bootstrap/blob/master/LICENSE)
|
|
* ======================================================================== */
|
|
|
|
|
|
+function ($) {
|
|
'use strict';
|
|
|
|
// DROPDOWN CLASS DEFINITION
|
|
// =========================
|
|
|
|
var backdrop = '.dropdown-backdrop'
|
|
var toggle = '[data-toggle="dropdown"]'
|
|
var Dropdown = function (element) {
|
|
$(element).on('click.bs.dropdown', this.toggle)
|
|
}
|
|
|
|
Dropdown.VERSION = '3.3.6'
|
|
|
|
function getParent($this) {
|
|
var selector = $this.attr('data-target')
|
|
|
|
if (!selector) {
|
|
selector = $this.attr('href')
|
|
selector = selector && /#[A-Za-z]/.test(selector) && selector.replace(/.*(?=#[^\s]*$)/, '') // strip for ie7
|
|
}
|
|
|
|
var $parent = selector && $(selector)
|
|
|
|
return $parent && $parent.length ? $parent : $this.parent()
|
|
}
|
|
|
|
function clearMenus(e) {
|
|
if (e && e.which === 3) return
|
|
$(backdrop).remove()
|
|
$(toggle).each(function () {
|
|
var $this = $(this)
|
|
var $parent = getParent($this)
|
|
var relatedTarget = { relatedTarget: this }
|
|
|
|
if (!$parent.hasClass('open')) return
|
|
|
|
if (e && e.type == 'click' && /input|textarea/i.test(e.target.tagName) && $.contains($parent[0], e.target)) return
|
|
|
|
$parent.trigger(e = $.Event('hide.bs.dropdown', relatedTarget))
|
|
|
|
if (e.isDefaultPrevented()) return
|
|
|
|
$this.attr('aria-expanded', 'false')
|
|
$parent.removeClass('open').trigger($.Event('hidden.bs.dropdown', relatedTarget))
|
|
})
|
|
}
|
|
|
|
Dropdown.prototype.toggle = function (e) {
|
|
var $this = $(this)
|
|
|
|
if ($this.is('.disabled, :disabled')) return
|
|
|
|
var $parent = getParent($this)
|
|
var isActive = $parent.hasClass('open')
|
|
|
|
clearMenus()
|
|
|
|
if (!isActive) {
|
|
if ('ontouchstart' in document.documentElement && !$parent.closest('.navbar-nav').length) {
|
|
// if mobile we use a backdrop because click events don't delegate
|
|
$(document.createElement('div'))
|
|
.addClass('dropdown-backdrop')
|
|
.insertAfter($(this))
|
|
.on('click', clearMenus)
|
|
}
|
|
|
|
var relatedTarget = { relatedTarget: this }
|
|
$parent.trigger(e = $.Event('show.bs.dropdown', relatedTarget))
|
|
|
|
if (e.isDefaultPrevented()) return
|
|
|
|
$this
|
|
.trigger('focus')
|
|
.attr('aria-expanded', 'true')
|
|
|
|
$parent
|
|
.toggleClass('open')
|
|
.trigger($.Event('shown.bs.dropdown', relatedTarget))
|
|
}
|
|
|
|
return false
|
|
}
|
|
|
|
Dropdown.prototype.keydown = function (e) {
|
|
if (!/(38|40|27|32)/.test(e.which) || /input|textarea/i.test(e.target.tagName)) return
|
|
|
|
var $this = $(this)
|
|
|
|
e.preventDefault()
|
|
e.stopPropagation()
|
|
|
|
if ($this.is('.disabled, :disabled')) return
|
|
|
|
var $parent = getParent($this)
|
|
var isActive = $parent.hasClass('open')
|
|
|
|
if (!isActive && e.which != 27 || isActive && e.which == 27) {
|
|
if (e.which == 27) $parent.find(toggle).trigger('focus')
|
|
return $this.trigger('click')
|
|
}
|
|
|
|
var desc = ' li:not(.disabled):visible a'
|
|
var $items = $parent.find('.dropdown-menu' + desc)
|
|
|
|
if (!$items.length) return
|
|
|
|
var index = $items.index(e.target)
|
|
|
|
if (e.which == 38 && index > 0) index-- // up
|
|
if (e.which == 40 && index < $items.length - 1) index++ // down
|
|
if (!~index) index = 0
|
|
|
|
$items.eq(index).trigger('focus')
|
|
}
|
|
|
|
|
|
// DROPDOWN PLUGIN DEFINITION
|
|
// ==========================
|
|
|
|
function Plugin(option) {
|
|
return this.each(function () {
|
|
var $this = $(this)
|
|
var data = $this.data('bs.dropdown')
|
|
|
|
if (!data) $this.data('bs.dropdown', (data = new Dropdown(this)))
|
|
if (typeof option == 'string') data[option].call($this)
|
|
})
|
|
}
|
|
|
|
var old = $.fn.dropdown
|
|
|
|
$.fn.dropdown = Plugin
|
|
$.fn.dropdown.Constructor = Dropdown
|
|
|
|
|
|
// DROPDOWN NO CONFLICT
|
|
// ====================
|
|
|
|
$.fn.dropdown.noConflict = function () {
|
|
$.fn.dropdown = old
|
|
return this
|
|
}
|
|
|
|
|
|
// APPLY TO STANDARD DROPDOWN ELEMENTS
|
|
// ===================================
|
|
|
|
$(document)
|
|
.on('click.bs.dropdown.data-api', clearMenus)
|
|
.on('click.bs.dropdown.data-api', '.dropdown form', function (e) { e.stopPropagation() })
|
|
.on('click.bs.dropdown.data-api', toggle, Dropdown.prototype.toggle)
|
|
.on('keydown.bs.dropdown.data-api', toggle, Dropdown.prototype.keydown)
|
|
.on('keydown.bs.dropdown.data-api', '.dropdown-menu', Dropdown.prototype.keydown)
|
|
|
|
}(jQuery);
|
|
|
|
/* ========================================================================
|
|
* Bootstrap: modal.js v3.3.6
|
|
* http://getbootstrap.com/javascript/#modals
|
|
* ========================================================================
|
|
* Copyright 2011-2015 Twitter, Inc.
|
|
* Licensed under MIT (https://github.com/twbs/bootstrap/blob/master/LICENSE)
|
|
* ======================================================================== */
|
|
|
|
|
|
+function ($) {
|
|
'use strict';
|
|
|
|
// MODAL CLASS DEFINITION
|
|
// ======================
|
|
|
|
var Modal = function (element, options) {
|
|
this.options = options
|
|
this.$body = $(document.body)
|
|
this.$element = $(element)
|
|
this.$dialog = this.$element.find('.modal-dialog')
|
|
this.$backdrop = null
|
|
this.isShown = null
|
|
this.originalBodyPad = null
|
|
this.scrollbarWidth = 0
|
|
this.ignoreBackdropClick = false
|
|
|
|
if (this.options.remote) {
|
|
this.$element
|
|
.find('.modal-content')
|
|
.load(this.options.remote, $.proxy(function () {
|
|
this.$element.trigger('loaded.bs.modal')
|
|
}, this))
|
|
}
|
|
}
|
|
|
|
Modal.VERSION = '3.3.6'
|
|
|
|
Modal.TRANSITION_DURATION = 300
|
|
Modal.BACKDROP_TRANSITION_DURATION = 150
|
|
|
|
Modal.DEFAULTS = {
|
|
backdrop: true,
|
|
keyboard: true,
|
|
show: true
|
|
}
|
|
|
|
Modal.prototype.toggle = function (_relatedTarget) {
|
|
return this.isShown ? this.hide() : this.show(_relatedTarget)
|
|
}
|
|
|
|
Modal.prototype.show = function (_relatedTarget) {
|
|
var that = this
|
|
var e = $.Event('show.bs.modal', { relatedTarget: _relatedTarget })
|
|
|
|
this.$element.trigger(e)
|
|
|
|
if (this.isShown || e.isDefaultPrevented()) return
|
|
|
|
this.isShown = true
|
|
|
|
this.checkScrollbar()
|
|
this.setScrollbar()
|
|
this.$body.addClass('modal-open')
|
|
|
|
this.escape()
|
|
this.resize()
|
|
|
|
this.$element.on('click.dismiss.bs.modal', '[data-dismiss="modal"]', $.proxy(this.hide, this))
|
|
|
|
this.$dialog.on('mousedown.dismiss.bs.modal', function () {
|
|
that.$element.one('mouseup.dismiss.bs.modal', function (e) {
|
|
if ($(e.target).is(that.$element)) that.ignoreBackdropClick = true
|
|
})
|
|
})
|
|
|
|
this.backdrop(function () {
|
|
var transition = $.support.transition && that.$element.hasClass('fade')
|
|
|
|
if (!that.$element.parent().length) {
|
|
that.$element.appendTo(that.$body) // don't move modals dom position
|
|
}
|
|
|
|
that.$element
|
|
.show()
|
|
.scrollTop(0)
|
|
|
|
that.adjustDialog()
|
|
|
|
if (transition) {
|
|
that.$element[0].offsetWidth // force reflow
|
|
}
|
|
|
|
that.$element.addClass('in')
|
|
|
|
that.enforceFocus()
|
|
|
|
var e = $.Event('shown.bs.modal', { relatedTarget: _relatedTarget })
|
|
|
|
transition ?
|
|
that.$dialog // wait for modal to slide in
|
|
.one('bsTransitionEnd', function () {
|
|
that.$element.trigger('focus').trigger(e)
|
|
})
|
|
.emulateTransitionEnd(Modal.TRANSITION_DURATION) :
|
|
that.$element.trigger('focus').trigger(e)
|
|
})
|
|
}
|
|
|
|
Modal.prototype.hide = function (e) {
|
|
if (e) e.preventDefault()
|
|
|
|
e = $.Event('hide.bs.modal')
|
|
|
|
this.$element.trigger(e)
|
|
|
|
if (!this.isShown || e.isDefaultPrevented()) return
|
|
|
|
this.isShown = false
|
|
|
|
this.escape()
|
|
this.resize()
|
|
|
|
$(document).off('focusin.bs.modal')
|
|
|
|
this.$element
|
|
.removeClass('in')
|
|
.off('click.dismiss.bs.modal')
|
|
.off('mouseup.dismiss.bs.modal')
|
|
|
|
this.$dialog.off('mousedown.dismiss.bs.modal')
|
|
|
|
$.support.transition && this.$element.hasClass('fade') ?
|
|
this.$element
|
|
.one('bsTransitionEnd', $.proxy(this.hideModal, this))
|
|
.emulateTransitionEnd(Modal.TRANSITION_DURATION) :
|
|
this.hideModal()
|
|
}
|
|
|
|
Modal.prototype.enforceFocus = function () {
|
|
$(document)
|
|
.off('focusin.bs.modal') // guard against infinite focus loop
|
|
.on('focusin.bs.modal', $.proxy(function (e) {
|
|
if (this.$element[0] !== e.target && !this.$element.has(e.target).length) {
|
|
this.$element.trigger('focus')
|
|
}
|
|
}, this))
|
|
}
|
|
|
|
Modal.prototype.escape = function () {
|
|
if (this.isShown && this.options.keyboard) {
|
|
this.$element.on('keydown.dismiss.bs.modal', $.proxy(function (e) {
|
|
e.which == 27 && this.hide()
|
|
}, this))
|
|
} else if (!this.isShown) {
|
|
this.$element.off('keydown.dismiss.bs.modal')
|
|
}
|
|
}
|
|
|
|
Modal.prototype.resize = function () {
|
|
if (this.isShown) {
|
|
$(window).on('resize.bs.modal', $.proxy(this.handleUpdate, this))
|
|
} else {
|
|
$(window).off('resize.bs.modal')
|
|
}
|
|
}
|
|
|
|
Modal.prototype.hideModal = function () {
|
|
var that = this
|
|
this.$element.hide()
|
|
this.backdrop(function () {
|
|
that.$body.removeClass('modal-open')
|
|
that.resetAdjustments()
|
|
that.resetScrollbar()
|
|
that.$element.trigger('hidden.bs.modal')
|
|
})
|
|
}
|
|
|
|
Modal.prototype.removeBackdrop = function () {
|
|
this.$backdrop && this.$backdrop.remove()
|
|
this.$backdrop = null
|
|
}
|
|
|
|
Modal.prototype.backdrop = function (callback) {
|
|
var that = this
|
|
var animate = this.$element.hasClass('fade') ? 'fade' : ''
|
|
|
|
if (this.isShown && this.options.backdrop) {
|
|
var doAnimate = $.support.transition && animate
|
|
|
|
this.$backdrop = $(document.createElement('div'))
|
|
.addClass('modal-backdrop ' + animate)
|
|
.appendTo(this.$body)
|
|
|
|
this.$element.on('click.dismiss.bs.modal', $.proxy(function (e) {
|
|
if (this.ignoreBackdropClick) {
|
|
this.ignoreBackdropClick = false
|
|
return
|
|
}
|
|
if (e.target !== e.currentTarget) return
|
|
this.options.backdrop == 'static'
|
|
? this.$element[0].focus()
|
|
: this.hide()
|
|
}, this))
|
|
|
|
if (doAnimate) this.$backdrop[0].offsetWidth // force reflow
|
|
|
|
this.$backdrop.addClass('in')
|
|
|
|
if (!callback) return
|
|
|
|
doAnimate ?
|
|
this.$backdrop
|
|
.one('bsTransitionEnd', callback)
|
|
.emulateTransitionEnd(Modal.BACKDROP_TRANSITION_DURATION) :
|
|
callback()
|
|
|
|
} else if (!this.isShown && this.$backdrop) {
|
|
this.$backdrop.removeClass('in')
|
|
|
|
var callbackRemove = function () {
|
|
that.removeBackdrop()
|
|
callback && callback()
|
|
}
|
|
$.support.transition && this.$element.hasClass('fade') ?
|
|
this.$backdrop
|
|
.one('bsTransitionEnd', callbackRemove)
|
|
.emulateTransitionEnd(Modal.BACKDROP_TRANSITION_DURATION) :
|
|
callbackRemove()
|
|
|
|
} else if (callback) {
|
|
callback()
|
|
}
|
|
}
|
|
|
|
// these following methods are used to handle overflowing modals
|
|
|
|
Modal.prototype.handleUpdate = function () {
|
|
this.adjustDialog()
|
|
}
|
|
|
|
Modal.prototype.adjustDialog = function () {
|
|
var modalIsOverflowing = this.$element[0].scrollHeight > document.documentElement.clientHeight
|
|
|
|
this.$element.css({
|
|
paddingLeft: !this.bodyIsOverflowing && modalIsOverflowing ? this.scrollbarWidth : '',
|
|
paddingRight: this.bodyIsOverflowing && !modalIsOverflowing ? this.scrollbarWidth : ''
|
|
})
|
|
}
|
|
|
|
Modal.prototype.resetAdjustments = function () {
|
|
this.$element.css({
|
|
paddingLeft: '',
|
|
paddingRight: ''
|
|
})
|
|
}
|
|
|
|
Modal.prototype.checkScrollbar = function () {
|
|
var fullWindowWidth = window.innerWidth
|
|
if (!fullWindowWidth) { // workaround for missing window.innerWidth in IE8
|
|
var documentElementRect = document.documentElement.getBoundingClientRect()
|
|
fullWindowWidth = documentElementRect.right - Math.abs(documentElementRect.left)
|
|
}
|
|
this.bodyIsOverflowing = document.body.clientWidth < fullWindowWidth
|
|
this.scrollbarWidth = this.measureScrollbar()
|
|
}
|
|
|
|
Modal.prototype.setScrollbar = function () {
|
|
var bodyPad = parseInt((this.$body.css('padding-right') || 0), 10)
|
|
this.originalBodyPad = document.body.style.paddingRight || ''
|
|
if (this.bodyIsOverflowing) this.$body.css('padding-right', bodyPad + this.scrollbarWidth)
|
|
}
|
|
|
|
Modal.prototype.resetScrollbar = function () {
|
|
this.$body.css('padding-right', this.originalBodyPad)
|
|
}
|
|
|
|
Modal.prototype.measureScrollbar = function () { // thx walsh
|
|
var scrollDiv = document.createElement('div')
|
|
scrollDiv.className = 'modal-scrollbar-measure'
|
|
this.$body.append(scrollDiv)
|
|
var scrollbarWidth = scrollDiv.offsetWidth - scrollDiv.clientWidth
|
|
this.$body[0].removeChild(scrollDiv)
|
|
return scrollbarWidth
|
|
}
|
|
|
|
|
|
// MODAL PLUGIN DEFINITION
|
|
// =======================
|
|
|
|
function Plugin(option, _relatedTarget) {
|
|
return this.each(function () {
|
|
var $this = $(this)
|
|
var data = $this.data('bs.modal')
|
|
var options = $.extend({}, Modal.DEFAULTS, $this.data(), typeof option == 'object' && option)
|
|
|
|
if (!data) $this.data('bs.modal', (data = new Modal(this, options)))
|
|
if (typeof option == 'string') data[option](_relatedTarget)
|
|
else if (options.show) data.show(_relatedTarget)
|
|
})
|
|
}
|
|
|
|
var old = $.fn.modal
|
|
|
|
$.fn.modal = Plugin
|
|
$.fn.modal.Constructor = Modal
|
|
|
|
|
|
// MODAL NO CONFLICT
|
|
// =================
|
|
|
|
$.fn.modal.noConflict = function () {
|
|
$.fn.modal = old
|
|
return this
|
|
}
|
|
|
|
|
|
// MODAL DATA-API
|
|
// ==============
|
|
|
|
$(document).on('click.bs.modal.data-api', '[data-toggle="modal"]', function (e) {
|
|
var $this = $(this)
|
|
var href = $this.attr('href')
|
|
var $target = $($this.attr('data-target') || (href && href.replace(/.*(?=#[^\s]+$)/, ''))) // strip for ie7
|
|
var option = $target.data('bs.modal') ? 'toggle' : $.extend({ remote: !/#/.test(href) && href }, $target.data(), $this.data())
|
|
|
|
if ($this.is('a')) e.preventDefault()
|
|
|
|
$target.one('show.bs.modal', function (showEvent) {
|
|
if (showEvent.isDefaultPrevented()) return // only register focus restorer if modal will actually get shown
|
|
$target.one('hidden.bs.modal', function () {
|
|
$this.is(':visible') && $this.trigger('focus')
|
|
})
|
|
})
|
|
Plugin.call($target, option, this)
|
|
})
|
|
|
|
}(jQuery);
|
|
|
|
/* ========================================================================
|
|
* Bootstrap: tooltip.js v3.3.6
|
|
* http://getbootstrap.com/javascript/#tooltip
|
|
* Inspired by the original jQuery.tipsy by Jason Frame
|
|
* ========================================================================
|
|
* Copyright 2011-2015 Twitter, Inc.
|
|
* Licensed under MIT (https://github.com/twbs/bootstrap/blob/master/LICENSE)
|
|
* ======================================================================== */
|
|
|
|
|
|
+function ($) {
|
|
'use strict';
|
|
|
|
// TOOLTIP PUBLIC CLASS DEFINITION
|
|
// ===============================
|
|
|
|
var Tooltip = function (element, options) {
|
|
this.type = null
|
|
this.options = null
|
|
this.enabled = null
|
|
this.timeout = null
|
|
this.hoverState = null
|
|
this.$element = null
|
|
this.inState = null
|
|
|
|
this.init('tooltip', element, options)
|
|
}
|
|
|
|
Tooltip.VERSION = '3.3.6'
|
|
|
|
Tooltip.TRANSITION_DURATION = 150
|
|
|
|
Tooltip.DEFAULTS = {
|
|
animation: true,
|
|
placement: 'top',
|
|
selector: false,
|
|
template: '<div class="tooltip" role="tooltip"><div class="tooltip-arrow"></div><div class="tooltip-inner"></div></div>',
|
|
trigger: 'hover focus',
|
|
title: '',
|
|
delay: 0,
|
|
html: false,
|
|
container: false,
|
|
viewport: {
|
|
selector: 'body',
|
|
padding: 0
|
|
}
|
|
}
|
|
|
|
Tooltip.prototype.init = function (type, element, options) {
|
|
this.enabled = true
|
|
this.type = type
|
|
this.$element = $(element)
|
|
this.options = this.getOptions(options)
|
|
this.$viewport = this.options.viewport && $($.isFunction(this.options.viewport) ? this.options.viewport.call(this, this.$element) : (this.options.viewport.selector || this.options.viewport))
|
|
this.inState = { click: false, hover: false, focus: false }
|
|
|
|
if (this.$element[0] instanceof document.constructor && !this.options.selector) {
|
|
throw new Error('`selector` option must be specified when initializing ' + this.type + ' on the window.document object!')
|
|
}
|
|
|
|
var triggers = this.options.trigger.split(' ')
|
|
|
|
for (var i = triggers.length; i--;) {
|
|
var trigger = triggers[i]
|
|
|
|
if (trigger == 'click') {
|
|
this.$element.on('click.' + this.type, this.options.selector, $.proxy(this.toggle, this))
|
|
} else if (trigger != 'manual') {
|
|
var eventIn = trigger == 'hover' ? 'mouseenter' : 'focusin'
|
|
var eventOut = trigger == 'hover' ? 'mouseleave' : 'focusout'
|
|
|
|
this.$element.on(eventIn + '.' + this.type, this.options.selector, $.proxy(this.enter, this))
|
|
this.$element.on(eventOut + '.' + this.type, this.options.selector, $.proxy(this.leave, this))
|
|
}
|
|
}
|
|
|
|
this.options.selector ?
|
|
(this._options = $.extend({}, this.options, { trigger: 'manual', selector: '' })) :
|
|
this.fixTitle()
|
|
}
|
|
|
|
Tooltip.prototype.getDefaults = function () {
|
|
return Tooltip.DEFAULTS
|
|
}
|
|
|
|
Tooltip.prototype.getOptions = function (options) {
|
|
options = $.extend({}, this.getDefaults(), this.$element.data(), options)
|
|
|
|
if (options.delay && typeof options.delay == 'number') {
|
|
options.delay = {
|
|
show: options.delay,
|
|
hide: options.delay
|
|
}
|
|
}
|
|
|
|
return options
|
|
}
|
|
|
|
Tooltip.prototype.getDelegateOptions = function () {
|
|
var options = {}
|
|
var defaults = this.getDefaults()
|
|
|
|
this._options && $.each(this._options, function (key, value) {
|
|
if (defaults[key] != value) options[key] = value
|
|
})
|
|
|
|
return options
|
|
}
|
|
|
|
Tooltip.prototype.enter = function (obj) {
|
|
var self = obj instanceof this.constructor ?
|
|
obj : $(obj.currentTarget).data('bs.' + this.type)
|
|
|
|
if (!self) {
|
|
self = new this.constructor(obj.currentTarget, this.getDelegateOptions())
|
|
$(obj.currentTarget).data('bs.' + this.type, self)
|
|
}
|
|
|
|
if (obj instanceof $.Event) {
|
|
self.inState[obj.type == 'focusin' ? 'focus' : 'hover'] = true
|
|
}
|
|
|
|
if (self.tip().hasClass('in') || self.hoverState == 'in') {
|
|
self.hoverState = 'in'
|
|
return
|
|
}
|
|
|
|
clearTimeout(self.timeout)
|
|
|
|
self.hoverState = 'in'
|
|
|
|
if (!self.options.delay || !self.options.delay.show) return self.show()
|
|
|
|
self.timeout = setTimeout(function () {
|
|
if (self.hoverState == 'in') self.show()
|
|
}, self.options.delay.show)
|
|
}
|
|
|
|
Tooltip.prototype.isInStateTrue = function () {
|
|
for (var key in this.inState) {
|
|
if (this.inState[key]) return true
|
|
}
|
|
|
|
return false
|
|
}
|
|
|
|
Tooltip.prototype.leave = function (obj) {
|
|
var self = obj instanceof this.constructor ?
|
|
obj : $(obj.currentTarget).data('bs.' + this.type)
|
|
|
|
if (!self) {
|
|
self = new this.constructor(obj.currentTarget, this.getDelegateOptions())
|
|
$(obj.currentTarget).data('bs.' + this.type, self)
|
|
}
|
|
|
|
if (obj instanceof $.Event) {
|
|
self.inState[obj.type == 'focusout' ? 'focus' : 'hover'] = false
|
|
}
|
|
|
|
if (self.isInStateTrue()) return
|
|
|
|
clearTimeout(self.timeout)
|
|
|
|
self.hoverState = 'out'
|
|
|
|
if (!self.options.delay || !self.options.delay.hide) return self.hide()
|
|
|
|
self.timeout = setTimeout(function () {
|
|
if (self.hoverState == 'out') self.hide()
|
|
}, self.options.delay.hide)
|
|
}
|
|
|
|
Tooltip.prototype.show = function () {
|
|
var e = $.Event('show.bs.' + this.type)
|
|
|
|
if (this.hasContent() && this.enabled) {
|
|
this.$element.trigger(e)
|
|
|
|
var inDom = $.contains(this.$element[0].ownerDocument.documentElement, this.$element[0])
|
|
if (e.isDefaultPrevented() || !inDom) return
|
|
var that = this
|
|
|
|
var $tip = this.tip()
|
|
|
|
var tipId = this.getUID(this.type)
|
|
|
|
this.setContent()
|
|
$tip.attr('id', tipId)
|
|
this.$element.attr('aria-describedby', tipId)
|
|
|
|
if (this.options.animation) $tip.addClass('fade')
|
|
|
|
var placement = typeof this.options.placement == 'function' ?
|
|
this.options.placement.call(this, $tip[0], this.$element[0]) :
|
|
this.options.placement
|
|
|
|
var autoToken = /\s?auto?\s?/i
|
|
var autoPlace = autoToken.test(placement)
|
|
if (autoPlace) placement = placement.replace(autoToken, '') || 'top'
|
|
|
|
$tip
|
|
.detach()
|
|
.css({ top: 0, left: 0, display: 'block' })
|
|
.addClass(placement)
|
|
.data('bs.' + this.type, this)
|
|
|
|
this.options.container ? $tip.appendTo(this.options.container) : $tip.insertAfter(this.$element)
|
|
this.$element.trigger('inserted.bs.' + this.type)
|
|
|
|
var pos = this.getPosition()
|
|
var actualWidth = $tip[0].offsetWidth
|
|
var actualHeight = $tip[0].offsetHeight
|
|
|
|
if (autoPlace) {
|
|
var orgPlacement = placement
|
|
var viewportDim = this.getPosition(this.$viewport)
|
|
|
|
placement = placement == 'bottom' && pos.bottom + actualHeight > viewportDim.bottom ? 'top' :
|
|
placement == 'top' && pos.top - actualHeight < viewportDim.top ? 'bottom' :
|
|
placement == 'right' && pos.right + actualWidth > viewportDim.width ? 'left' :
|
|
placement == 'left' && pos.left - actualWidth < viewportDim.left ? 'right' :
|
|
placement
|
|
|
|
$tip
|
|
.removeClass(orgPlacement)
|
|
.addClass(placement)
|
|
}
|
|
|
|
var calculatedOffset = this.getCalculatedOffset(placement, pos, actualWidth, actualHeight)
|
|
|
|
this.applyPlacement(calculatedOffset, placement)
|
|
|
|
var complete = function () {
|
|
var prevHoverState = that.hoverState
|
|
that.$element.trigger('shown.bs.' + that.type)
|
|
that.hoverState = null
|
|
|
|
if (prevHoverState == 'out') that.leave(that)
|
|
}
|
|
|
|
$.support.transition && this.$tip.hasClass('fade') ?
|
|
$tip
|
|
.one('bsTransitionEnd', complete)
|
|
.emulateTransitionEnd(Tooltip.TRANSITION_DURATION) :
|
|
complete()
|
|
}
|
|
}
|
|
|
|
Tooltip.prototype.applyPlacement = function (offset, placement) {
|
|
var $tip = this.tip()
|
|
var width = $tip[0].offsetWidth
|
|
var height = $tip[0].offsetHeight
|
|
|
|
// manually read margins because getBoundingClientRect includes difference
|
|
var marginTop = parseInt($tip.css('margin-top'), 10)
|
|
var marginLeft = parseInt($tip.css('margin-left'), 10)
|
|
|
|
// we must check for NaN for ie 8/9
|
|
if (isNaN(marginTop)) marginTop = 0
|
|
if (isNaN(marginLeft)) marginLeft = 0
|
|
|
|
offset.top += marginTop
|
|
offset.left += marginLeft
|
|
|
|
// $.fn.offset doesn't round pixel values
|
|
// so we use setOffset directly with our own function B-0
|
|
$.offset.setOffset($tip[0], $.extend({
|
|
using: function (props) {
|
|
$tip.css({
|
|
top: Math.round(props.top),
|
|
left: Math.round(props.left)
|
|
})
|
|
}
|
|
}, offset), 0)
|
|
|
|
$tip.addClass('in')
|
|
|
|
// check to see if placing tip in new offset caused the tip to resize itself
|
|
var actualWidth = $tip[0].offsetWidth
|
|
var actualHeight = $tip[0].offsetHeight
|
|
|
|
if (placement == 'top' && actualHeight != height) {
|
|
offset.top = offset.top + height - actualHeight
|
|
}
|
|
|
|
var delta = this.getViewportAdjustedDelta(placement, offset, actualWidth, actualHeight)
|
|
|
|
if (delta.left) offset.left += delta.left
|
|
else offset.top += delta.top
|
|
|
|
var isVertical = /top|bottom/.test(placement)
|
|
var arrowDelta = isVertical ? delta.left * 2 - width + actualWidth : delta.top * 2 - height + actualHeight
|
|
var arrowOffsetPosition = isVertical ? 'offsetWidth' : 'offsetHeight'
|
|
|
|
$tip.offset(offset)
|
|
this.replaceArrow(arrowDelta, $tip[0][arrowOffsetPosition], isVertical)
|
|
}
|
|
|
|
Tooltip.prototype.replaceArrow = function (delta, dimension, isVertical) {
|
|
this.arrow()
|
|
.css(isVertical ? 'left' : 'top', 50 * (1 - delta / dimension) + '%')
|
|
.css(isVertical ? 'top' : 'left', '')
|
|
}
|
|
|
|
Tooltip.prototype.setContent = function () {
|
|
var $tip = this.tip()
|
|
var title = this.getTitle()
|
|
|
|
$tip.find('.tooltip-inner')[this.options.html ? 'html' : 'text'](title)
|
|
$tip.removeClass('fade in top bottom left right')
|
|
}
|
|
|
|
Tooltip.prototype.hide = function (callback) {
|
|
var that = this
|
|
var $tip = $(this.$tip)
|
|
var e = $.Event('hide.bs.' + this.type)
|
|
|
|
function complete() {
|
|
if (that.hoverState != 'in') $tip.detach()
|
|
that.$element
|
|
.removeAttr('aria-describedby')
|
|
.trigger('hidden.bs.' + that.type)
|
|
callback && callback()
|
|
}
|
|
|
|
this.$element.trigger(e)
|
|
|
|
if (e.isDefaultPrevented()) return
|
|
|
|
$tip.removeClass('in')
|
|
|
|
$.support.transition && $tip.hasClass('fade') ?
|
|
$tip
|
|
.one('bsTransitionEnd', complete)
|
|
.emulateTransitionEnd(Tooltip.TRANSITION_DURATION) :
|
|
complete()
|
|
|
|
this.hoverState = null
|
|
|
|
return this
|
|
}
|
|
|
|
Tooltip.prototype.fixTitle = function () {
|
|
var $e = this.$element
|
|
if ($e.attr('title') || typeof $e.attr('data-original-title') != 'string') {
|
|
$e.attr('data-original-title', $e.attr('title') || '').attr('title', '')
|
|
}
|
|
}
|
|
|
|
Tooltip.prototype.hasContent = function () {
|
|
return this.getTitle()
|
|
}
|
|
|
|
Tooltip.prototype.getPosition = function ($element) {
|
|
$element = $element || this.$element
|
|
|
|
var el = $element[0]
|
|
var isBody = el.tagName == 'BODY'
|
|
|
|
var elRect = el.getBoundingClientRect()
|
|
if (elRect.width == null) {
|
|
// width and height are missing in IE8, so compute them manually; see https://github.com/twbs/bootstrap/issues/14093
|
|
elRect = $.extend({}, elRect, { width: elRect.right - elRect.left, height: elRect.bottom - elRect.top })
|
|
}
|
|
var elOffset = isBody ? { top: 0, left: 0 } : $element.offset()
|
|
var scroll = { scroll: isBody ? document.documentElement.scrollTop || document.body.scrollTop : $element.scrollTop() }
|
|
var outerDims = isBody ? { width: $(window).width(), height: $(window).height() } : null
|
|
|
|
return $.extend({}, elRect, scroll, outerDims, elOffset)
|
|
}
|
|
|
|
Tooltip.prototype.getCalculatedOffset = function (placement, pos, actualWidth, actualHeight) {
|
|
return placement == 'bottom' ? { top: pos.top + pos.height, left: pos.left + pos.width / 2 - actualWidth / 2 } :
|
|
placement == 'top' ? { top: pos.top - actualHeight, left: pos.left + pos.width / 2 - actualWidth / 2 } :
|
|
placement == 'left' ? { top: pos.top + pos.height / 2 - actualHeight / 2, left: pos.left - actualWidth } :
|
|
/* placement == 'right' */ { top: pos.top + pos.height / 2 - actualHeight / 2, left: pos.left + pos.width }
|
|
|
|
}
|
|
|
|
Tooltip.prototype.getViewportAdjustedDelta = function (placement, pos, actualWidth, actualHeight) {
|
|
var delta = { top: 0, left: 0 }
|
|
if (!this.$viewport) return delta
|
|
|
|
var viewportPadding = this.options.viewport && this.options.viewport.padding || 0
|
|
var viewportDimensions = this.getPosition(this.$viewport)
|
|
|
|
if (/right|left/.test(placement)) {
|
|
var topEdgeOffset = pos.top - viewportPadding - viewportDimensions.scroll
|
|
var bottomEdgeOffset = pos.top + viewportPadding - viewportDimensions.scroll + actualHeight
|
|
if (topEdgeOffset < viewportDimensions.top) { // top overflow
|
|
delta.top = viewportDimensions.top - topEdgeOffset
|
|
} else if (bottomEdgeOffset > viewportDimensions.top + viewportDimensions.height) { // bottom overflow
|
|
delta.top = viewportDimensions.top + viewportDimensions.height - bottomEdgeOffset
|
|
}
|
|
} else {
|
|
var leftEdgeOffset = pos.left - viewportPadding
|
|
var rightEdgeOffset = pos.left + viewportPadding + actualWidth
|
|
if (leftEdgeOffset < viewportDimensions.left) { // left overflow
|
|
delta.left = viewportDimensions.left - leftEdgeOffset
|
|
} else if (rightEdgeOffset > viewportDimensions.right) { // right overflow
|
|
delta.left = viewportDimensions.left + viewportDimensions.width - rightEdgeOffset
|
|
}
|
|
}
|
|
|
|
return delta
|
|
}
|
|
|
|
Tooltip.prototype.getTitle = function () {
|
|
var title
|
|
var $e = this.$element
|
|
var o = this.options
|
|
|
|
title = $e.attr('data-original-title')
|
|
|| (typeof o.title == 'function' ? o.title.call($e[0]) : o.title)
|
|
|
|
return title
|
|
}
|
|
|
|
Tooltip.prototype.getUID = function (prefix) {
|
|
do prefix += ~~(Math.random() * 1000000)
|
|
while (document.getElementById(prefix))
|
|
return prefix
|
|
}
|
|
|
|
Tooltip.prototype.tip = function () {
|
|
if (!this.$tip) {
|
|
this.$tip = $(this.options.template)
|
|
if (this.$tip.length != 1) {
|
|
throw new Error(this.type + ' `template` option must consist of exactly 1 top-level element!')
|
|
}
|
|
}
|
|
return this.$tip
|
|
}
|
|
|
|
Tooltip.prototype.arrow = function () {
|
|
return (this.$arrow = this.$arrow || this.tip().find('.tooltip-arrow'))
|
|
}
|
|
|
|
Tooltip.prototype.enable = function () {
|
|
this.enabled = true
|
|
}
|
|
|
|
Tooltip.prototype.disable = function () {
|
|
this.enabled = false
|
|
}
|
|
|
|
Tooltip.prototype.toggleEnabled = function () {
|
|
this.enabled = !this.enabled
|
|
}
|
|
|
|
Tooltip.prototype.toggle = function (e) {
|
|
var self = this
|
|
if (e) {
|
|
self = $(e.currentTarget).data('bs.' + this.type)
|
|
if (!self) {
|
|
self = new this.constructor(e.currentTarget, this.getDelegateOptions())
|
|
$(e.currentTarget).data('bs.' + this.type, self)
|
|
}
|
|
}
|
|
|
|
if (e) {
|
|
self.inState.click = !self.inState.click
|
|
if (self.isInStateTrue()) self.enter(self)
|
|
else self.leave(self)
|
|
} else {
|
|
self.tip().hasClass('in') ? self.leave(self) : self.enter(self)
|
|
}
|
|
}
|
|
|
|
Tooltip.prototype.destroy = function () {
|
|
var that = this
|
|
clearTimeout(this.timeout)
|
|
this.hide(function () {
|
|
that.$element.off('.' + that.type).removeData('bs.' + that.type)
|
|
if (that.$tip) {
|
|
that.$tip.detach()
|
|
}
|
|
that.$tip = null
|
|
that.$arrow = null
|
|
that.$viewport = null
|
|
})
|
|
}
|
|
|
|
|
|
// TOOLTIP PLUGIN DEFINITION
|
|
// =========================
|
|
|
|
function Plugin(option) {
|
|
return this.each(function () {
|
|
var $this = $(this)
|
|
var data = $this.data('bs.tooltip')
|
|
var options = typeof option == 'object' && option
|
|
|
|
if (!data && /destroy|hide/.test(option)) return
|
|
if (!data) $this.data('bs.tooltip', (data = new Tooltip(this, options)))
|
|
if (typeof option == 'string') data[option]()
|
|
})
|
|
}
|
|
|
|
var old = $.fn.tooltip
|
|
|
|
$.fn.tooltip = Plugin
|
|
$.fn.tooltip.Constructor = Tooltip
|
|
|
|
|
|
// TOOLTIP NO CONFLICT
|
|
// ===================
|
|
|
|
$.fn.tooltip.noConflict = function () {
|
|
$.fn.tooltip = old
|
|
return this
|
|
}
|
|
|
|
}(jQuery);
|
|
|
|
/* ========================================================================
|
|
* Bootstrap: popover.js v3.3.6
|
|
* http://getbootstrap.com/javascript/#popovers
|
|
* ========================================================================
|
|
* Copyright 2011-2015 Twitter, Inc.
|
|
* Licensed under MIT (https://github.com/twbs/bootstrap/blob/master/LICENSE)
|
|
* ======================================================================== */
|
|
|
|
|
|
+function ($) {
|
|
'use strict';
|
|
|
|
// POPOVER PUBLIC CLASS DEFINITION
|
|
// ===============================
|
|
|
|
var Popover = function (element, options) {
|
|
this.init('popover', element, options)
|
|
}
|
|
|
|
if (!$.fn.tooltip) throw new Error('Popover requires tooltip.js')
|
|
|
|
Popover.VERSION = '3.3.6'
|
|
|
|
Popover.DEFAULTS = $.extend({}, $.fn.tooltip.Constructor.DEFAULTS, {
|
|
placement: 'right',
|
|
trigger: 'click',
|
|
content: '',
|
|
template: '<div class="popover" role="tooltip"><div class="arrow"></div><h3 class="popover-title"></h3><div class="popover-content"></div></div>'
|
|
})
|
|
|
|
|
|
// NOTE: POPOVER EXTENDS tooltip.js
|
|
// ================================
|
|
|
|
Popover.prototype = $.extend({}, $.fn.tooltip.Constructor.prototype)
|
|
|
|
Popover.prototype.constructor = Popover
|
|
|
|
Popover.prototype.getDefaults = function () {
|
|
return Popover.DEFAULTS
|
|
}
|
|
|
|
Popover.prototype.setContent = function () {
|
|
var $tip = this.tip()
|
|
var title = this.getTitle()
|
|
var content = this.getContent()
|
|
|
|
$tip.find('.popover-title')[this.options.html ? 'html' : 'text'](title)
|
|
$tip.find('.popover-content').children().detach().end()[ // we use append for html objects to maintain js events
|
|
this.options.html ? (typeof content == 'string' ? 'html' : 'append') : 'text'
|
|
](content)
|
|
|
|
$tip.removeClass('fade top bottom left right in')
|
|
|
|
// IE8 doesn't accept hiding via the `:empty` pseudo selector, we have to do
|
|
// this manually by checking the contents.
|
|
if (!$tip.find('.popover-title').html()) $tip.find('.popover-title').hide()
|
|
}
|
|
|
|
Popover.prototype.hasContent = function () {
|
|
return this.getTitle() || this.getContent()
|
|
}
|
|
|
|
Popover.prototype.getContent = function () {
|
|
var $e = this.$element
|
|
var o = this.options
|
|
|
|
return $e.attr('data-content')
|
|
|| (typeof o.content == 'function' ?
|
|
o.content.call($e[0]) :
|
|
o.content)
|
|
}
|
|
|
|
Popover.prototype.arrow = function () {
|
|
return (this.$arrow = this.$arrow || this.tip().find('.arrow'))
|
|
}
|
|
|
|
|
|
// POPOVER PLUGIN DEFINITION
|
|
// =========================
|
|
|
|
function Plugin(option) {
|
|
return this.each(function () {
|
|
var $this = $(this)
|
|
var data = $this.data('bs.popover')
|
|
var options = typeof option == 'object' && option
|
|
|
|
if (!data && /destroy|hide/.test(option)) return
|
|
if (!data) $this.data('bs.popover', (data = new Popover(this, options)))
|
|
if (typeof option == 'string') data[option]()
|
|
})
|
|
}
|
|
|
|
var old = $.fn.popover
|
|
|
|
$.fn.popover = Plugin
|
|
$.fn.popover.Constructor = Popover
|
|
|
|
|
|
// POPOVER NO CONFLICT
|
|
// ===================
|
|
|
|
$.fn.popover.noConflict = function () {
|
|
$.fn.popover = old
|
|
return this
|
|
}
|
|
|
|
}(jQuery);
|
|
|
|
/* ========================================================================
|
|
* Bootstrap: scrollspy.js v3.3.6
|
|
* http://getbootstrap.com/javascript/#scrollspy
|
|
* ========================================================================
|
|
* Copyright 2011-2015 Twitter, Inc.
|
|
* Licensed under MIT (https://github.com/twbs/bootstrap/blob/master/LICENSE)
|
|
* ======================================================================== */
|
|
|
|
|
|
+function ($) {
|
|
'use strict';
|
|
|
|
// SCROLLSPY CLASS DEFINITION
|
|
// ==========================
|
|
|
|
function ScrollSpy(element, options) {
|
|
this.$body = $(document.body)
|
|
this.$scrollElement = $(element).is(document.body) ? $(window) : $(element)
|
|
this.options = $.extend({}, ScrollSpy.DEFAULTS, options)
|
|
this.selector = (this.options.target || '') + ' .nav li > a'
|
|
this.offsets = []
|
|
this.targets = []
|
|
this.activeTarget = null
|
|
this.scrollHeight = 0
|
|
|
|
this.$scrollElement.on('scroll.bs.scrollspy', $.proxy(this.process, this))
|
|
this.refresh()
|
|
this.process()
|
|
}
|
|
|
|
ScrollSpy.VERSION = '3.3.6'
|
|
|
|
ScrollSpy.DEFAULTS = {
|
|
offset: 10
|
|
}
|
|
|
|
ScrollSpy.prototype.getScrollHeight = function () {
|
|
return this.$scrollElement[0].scrollHeight || Math.max(this.$body[0].scrollHeight, document.documentElement.scrollHeight)
|
|
}
|
|
|
|
ScrollSpy.prototype.refresh = function () {
|
|
var that = this
|
|
var offsetMethod = 'offset'
|
|
var offsetBase = 0
|
|
|
|
this.offsets = []
|
|
this.targets = []
|
|
this.scrollHeight = this.getScrollHeight()
|
|
|
|
if (!$.isWindow(this.$scrollElement[0])) {
|
|
offsetMethod = 'position'
|
|
offsetBase = this.$scrollElement.scrollTop()
|
|
}
|
|
|
|
this.$body
|
|
.find(this.selector)
|
|
.map(function () {
|
|
var $el = $(this)
|
|
var href = $el.data('target') || $el.attr('href')
|
|
var $href = /^#./.test(href) && $(href)
|
|
|
|
return ($href
|
|
&& $href.length
|
|
&& $href.is(':visible')
|
|
&& [[$href[offsetMethod]().top + offsetBase, href]]) || null
|
|
})
|
|
.sort(function (a, b) { return a[0] - b[0] })
|
|
.each(function () {
|
|
that.offsets.push(this[0])
|
|
that.targets.push(this[1])
|
|
})
|
|
}
|
|
|
|
ScrollSpy.prototype.process = function () {
|
|
var scrollTop = this.$scrollElement.scrollTop() + this.options.offset
|
|
var scrollHeight = this.getScrollHeight()
|
|
var maxScroll = this.options.offset + scrollHeight - this.$scrollElement.height()
|
|
var offsets = this.offsets
|
|
var targets = this.targets
|
|
var activeTarget = this.activeTarget
|
|
var i
|
|
|
|
if (this.scrollHeight != scrollHeight) {
|
|
this.refresh()
|
|
}
|
|
|
|
if (scrollTop >= maxScroll) {
|
|
return activeTarget != (i = targets[targets.length - 1]) && this.activate(i)
|
|
}
|
|
|
|
if (activeTarget && scrollTop < offsets[0]) {
|
|
this.activeTarget = null
|
|
return this.clear()
|
|
}
|
|
|
|
for (i = offsets.length; i--;) {
|
|
activeTarget != targets[i]
|
|
&& scrollTop >= offsets[i]
|
|
&& (offsets[i + 1] === undefined || scrollTop < offsets[i + 1])
|
|
&& this.activate(targets[i])
|
|
}
|
|
}
|
|
|
|
ScrollSpy.prototype.activate = function (target) {
|
|
this.activeTarget = target
|
|
|
|
this.clear()
|
|
|
|
var selector = this.selector +
|
|
'[data-target="' + target + '"],' +
|
|
this.selector + '[href="' + target + '"]'
|
|
|
|
var active = $(selector)
|
|
.parents('li')
|
|
.addClass('active')
|
|
|
|
if (active.parent('.dropdown-menu').length) {
|
|
active = active
|
|
.closest('li.dropdown')
|
|
.addClass('active')
|
|
}
|
|
|
|
active.trigger('activate.bs.scrollspy')
|
|
}
|
|
|
|
ScrollSpy.prototype.clear = function () {
|
|
$(this.selector)
|
|
.parentsUntil(this.options.target, '.active')
|
|
.removeClass('active')
|
|
}
|
|
|
|
|
|
// SCROLLSPY PLUGIN DEFINITION
|
|
// ===========================
|
|
|
|
function Plugin(option) {
|
|
return this.each(function () {
|
|
var $this = $(this)
|
|
var data = $this.data('bs.scrollspy')
|
|
var options = typeof option == 'object' && option
|
|
|
|
if (!data) $this.data('bs.scrollspy', (data = new ScrollSpy(this, options)))
|
|
if (typeof option == 'string') data[option]()
|
|
})
|
|
}
|
|
|
|
var old = $.fn.scrollspy
|
|
|
|
$.fn.scrollspy = Plugin
|
|
$.fn.scrollspy.Constructor = ScrollSpy
|
|
|
|
|
|
// SCROLLSPY NO CONFLICT
|
|
// =====================
|
|
|
|
$.fn.scrollspy.noConflict = function () {
|
|
$.fn.scrollspy = old
|
|
return this
|
|
}
|
|
|
|
|
|
// SCROLLSPY DATA-API
|
|
// ==================
|
|
|
|
$(window).on('load.bs.scrollspy.data-api', function () {
|
|
$('[data-spy="scroll"]').each(function () {
|
|
var $spy = $(this)
|
|
Plugin.call($spy, $spy.data())
|
|
})
|
|
})
|
|
|
|
}(jQuery);
|
|
|
|
/* ========================================================================
|
|
* Bootstrap: tab.js v3.3.6
|
|
* http://getbootstrap.com/javascript/#tabs
|
|
* ========================================================================
|
|
* Copyright 2011-2015 Twitter, Inc.
|
|
* Licensed under MIT (https://github.com/twbs/bootstrap/blob/master/LICENSE)
|
|
* ======================================================================== */
|
|
|
|
|
|
+function ($) {
|
|
'use strict';
|
|
|
|
// TAB CLASS DEFINITION
|
|
// ====================
|
|
|
|
var Tab = function (element) {
|
|
// jscs:disable requireDollarBeforejQueryAssignment
|
|
this.element = $(element)
|
|
// jscs:enable requireDollarBeforejQueryAssignment
|
|
}
|
|
|
|
Tab.VERSION = '3.3.6'
|
|
|
|
Tab.TRANSITION_DURATION = 150
|
|
|
|
Tab.prototype.show = function () {
|
|
var $this = this.element
|
|
var $ul = $this.closest('ul:not(.dropdown-menu)')
|
|
var selector = $this.data('target')
|
|
|
|
if (!selector) {
|
|
selector = $this.attr('href')
|
|
selector = selector && selector.replace(/.*(?=#[^\s]*$)/, '') // strip for ie7
|
|
}
|
|
|
|
if ($this.parent('li').hasClass('active')) return
|
|
|
|
var $previous = $ul.find('.active:last a')
|
|
var hideEvent = $.Event('hide.bs.tab', {
|
|
relatedTarget: $this[0]
|
|
})
|
|
var showEvent = $.Event('show.bs.tab', {
|
|
relatedTarget: $previous[0]
|
|
})
|
|
|
|
$previous.trigger(hideEvent)
|
|
$this.trigger(showEvent)
|
|
|
|
if (showEvent.isDefaultPrevented() || hideEvent.isDefaultPrevented()) return
|
|
|
|
var $target = $(selector)
|
|
|
|
this.activate($this.closest('li'), $ul)
|
|
this.activate($target, $target.parent(), function () {
|
|
$previous.trigger({
|
|
type: 'hidden.bs.tab',
|
|
relatedTarget: $this[0]
|
|
})
|
|
$this.trigger({
|
|
type: 'shown.bs.tab',
|
|
relatedTarget: $previous[0]
|
|
})
|
|
})
|
|
}
|
|
|
|
Tab.prototype.activate = function (element, container, callback) {
|
|
var $active = container.find('> .active')
|
|
var transition = callback
|
|
&& $.support.transition
|
|
&& ($active.length && $active.hasClass('fade') || !!container.find('> .fade').length)
|
|
|
|
function next() {
|
|
$active
|
|
.removeClass('active')
|
|
.find('> .dropdown-menu > .active')
|
|
.removeClass('active')
|
|
.end()
|
|
.find('[data-toggle="tab"]')
|
|
.attr('aria-expanded', false)
|
|
|
|
element
|
|
.addClass('active')
|
|
.find('[data-toggle="tab"]')
|
|
.attr('aria-expanded', true)
|
|
|
|
if (transition) {
|
|
element[0].offsetWidth // reflow for transition
|
|
element.addClass('in')
|
|
} else {
|
|
element.removeClass('fade')
|
|
}
|
|
|
|
if (element.parent('.dropdown-menu').length) {
|
|
element
|
|
.closest('li.dropdown')
|
|
.addClass('active')
|
|
.end()
|
|
.find('[data-toggle="tab"]')
|
|
.attr('aria-expanded', true)
|
|
}
|
|
|
|
callback && callback()
|
|
}
|
|
|
|
$active.length && transition ?
|
|
$active
|
|
.one('bsTransitionEnd', next)
|
|
.emulateTransitionEnd(Tab.TRANSITION_DURATION) :
|
|
next()
|
|
|
|
$active.removeClass('in')
|
|
}
|
|
|
|
|
|
// TAB PLUGIN DEFINITION
|
|
// =====================
|
|
|
|
function Plugin(option) {
|
|
return this.each(function () {
|
|
var $this = $(this)
|
|
var data = $this.data('bs.tab')
|
|
|
|
if (!data) $this.data('bs.tab', (data = new Tab(this)))
|
|
if (typeof option == 'string') data[option]()
|
|
})
|
|
}
|
|
|
|
var old = $.fn.tab
|
|
|
|
$.fn.tab = Plugin
|
|
$.fn.tab.Constructor = Tab
|
|
|
|
|
|
// TAB NO CONFLICT
|
|
// ===============
|
|
|
|
$.fn.tab.noConflict = function () {
|
|
$.fn.tab = old
|
|
return this
|
|
}
|
|
|
|
|
|
// TAB DATA-API
|
|
// ============
|
|
|
|
var clickHandler = function (e) {
|
|
e.preventDefault()
|
|
Plugin.call($(this), 'show')
|
|
}
|
|
|
|
$(document)
|
|
.on('click.bs.tab.data-api', '[data-toggle="tab"]', clickHandler)
|
|
.on('click.bs.tab.data-api', '[data-toggle="pill"]', clickHandler)
|
|
|
|
}(jQuery);
|
|
|
|
/* ========================================================================
|
|
* Bootstrap: affix.js v3.3.6
|
|
* http://getbootstrap.com/javascript/#affix
|
|
* ========================================================================
|
|
* Copyright 2011-2015 Twitter, Inc.
|
|
* Licensed under MIT (https://github.com/twbs/bootstrap/blob/master/LICENSE)
|
|
* ======================================================================== */
|
|
|
|
|
|
+function ($) {
|
|
'use strict';
|
|
|
|
// AFFIX CLASS DEFINITION
|
|
// ======================
|
|
|
|
var Affix = function (element, options) {
|
|
this.options = $.extend({}, Affix.DEFAULTS, options)
|
|
|
|
this.$target = $(this.options.target)
|
|
.on('scroll.bs.affix.data-api', $.proxy(this.checkPosition, this))
|
|
.on('click.bs.affix.data-api', $.proxy(this.checkPositionWithEventLoop, this))
|
|
|
|
this.$element = $(element)
|
|
this.affixed = null
|
|
this.unpin = null
|
|
this.pinnedOffset = null
|
|
|
|
this.checkPosition()
|
|
}
|
|
|
|
Affix.VERSION = '3.3.6'
|
|
|
|
Affix.RESET = 'affix affix-top affix-bottom'
|
|
|
|
Affix.DEFAULTS = {
|
|
offset: 0,
|
|
target: window
|
|
}
|
|
|
|
Affix.prototype.getState = function (scrollHeight, height, offsetTop, offsetBottom) {
|
|
var scrollTop = this.$target.scrollTop()
|
|
var position = this.$element.offset()
|
|
var targetHeight = this.$target.height()
|
|
|
|
if (offsetTop != null && this.affixed == 'top') return scrollTop < offsetTop ? 'top' : false
|
|
|
|
if (this.affixed == 'bottom') {
|
|
if (offsetTop != null) return (scrollTop + this.unpin <= position.top) ? false : 'bottom'
|
|
return (scrollTop + targetHeight <= scrollHeight - offsetBottom) ? false : 'bottom'
|
|
}
|
|
|
|
var initializing = this.affixed == null
|
|
var colliderTop = initializing ? scrollTop : position.top
|
|
var colliderHeight = initializing ? targetHeight : height
|
|
|
|
if (offsetTop != null && scrollTop <= offsetTop) return 'top'
|
|
if (offsetBottom != null && (colliderTop + colliderHeight >= scrollHeight - offsetBottom)) return 'bottom'
|
|
|
|
return false
|
|
}
|
|
|
|
Affix.prototype.getPinnedOffset = function () {
|
|
if (this.pinnedOffset) return this.pinnedOffset
|
|
this.$element.removeClass(Affix.RESET).addClass('affix')
|
|
var scrollTop = this.$target.scrollTop()
|
|
var position = this.$element.offset()
|
|
return (this.pinnedOffset = position.top - scrollTop)
|
|
}
|
|
|
|
Affix.prototype.checkPositionWithEventLoop = function () {
|
|
setTimeout($.proxy(this.checkPosition, this), 1)
|
|
}
|
|
|
|
Affix.prototype.checkPosition = function () {
|
|
if (!this.$element.is(':visible')) return
|
|
|
|
var height = this.$element.height()
|
|
var offset = this.options.offset
|
|
var offsetTop = offset.top
|
|
var offsetBottom = offset.bottom
|
|
var scrollHeight = Math.max($(document).height(), $(document.body).height())
|
|
|
|
if (typeof offset != 'object') offsetBottom = offsetTop = offset
|
|
if (typeof offsetTop == 'function') offsetTop = offset.top(this.$element)
|
|
if (typeof offsetBottom == 'function') offsetBottom = offset.bottom(this.$element)
|
|
|
|
var affix = this.getState(scrollHeight, height, offsetTop, offsetBottom)
|
|
|
|
if (this.affixed != affix) {
|
|
if (this.unpin != null) this.$element.css('top', '')
|
|
|
|
var affixType = 'affix' + (affix ? '-' + affix : '')
|
|
var e = $.Event(affixType + '.bs.affix')
|
|
|
|
this.$element.trigger(e)
|
|
|
|
if (e.isDefaultPrevented()) return
|
|
|
|
this.affixed = affix
|
|
this.unpin = affix == 'bottom' ? this.getPinnedOffset() : null
|
|
|
|
this.$element
|
|
.removeClass(Affix.RESET)
|
|
.addClass(affixType)
|
|
.trigger(affixType.replace('affix', 'affixed') + '.bs.affix')
|
|
}
|
|
|
|
if (affix == 'bottom') {
|
|
this.$element.offset({
|
|
top: scrollHeight - height - offsetBottom
|
|
})
|
|
}
|
|
}
|
|
|
|
|
|
// AFFIX PLUGIN DEFINITION
|
|
// =======================
|
|
|
|
function Plugin(option) {
|
|
return this.each(function () {
|
|
var $this = $(this)
|
|
var data = $this.data('bs.affix')
|
|
var options = typeof option == 'object' && option
|
|
|
|
if (!data) $this.data('bs.affix', (data = new Affix(this, options)))
|
|
if (typeof option == 'string') data[option]()
|
|
})
|
|
}
|
|
|
|
var old = $.fn.affix
|
|
|
|
$.fn.affix = Plugin
|
|
$.fn.affix.Constructor = Affix
|
|
|
|
|
|
// AFFIX NO CONFLICT
|
|
// =================
|
|
|
|
$.fn.affix.noConflict = function () {
|
|
$.fn.affix = old
|
|
return this
|
|
}
|
|
|
|
|
|
// AFFIX DATA-API
|
|
// ==============
|
|
|
|
$(window).on('load', function () {
|
|
$('[data-spy="affix"]').each(function () {
|
|
var $spy = $(this)
|
|
var data = $spy.data()
|
|
|
|
data.offset = data.offset || {}
|
|
|
|
if (data.offsetBottom != null) data.offset.bottom = data.offsetBottom
|
|
if (data.offsetTop != null) data.offset.top = data.offsetTop
|
|
|
|
Plugin.call($spy, data)
|
|
})
|
|
})
|
|
|
|
}(jQuery);
|
|
|
|
!function(t,e){"use strict";if("undefined"!=typeof module&&module.exports){var n="undefined"!=typeof process,o=n&&"electron"in process.versions;o?t.BootstrapDialog=e(t.jQuery):module.exports=e(require("jquery"),require("bootstrap"))}else"function"==typeof define&&define.amd?define("bootstrap-dialog",["jquery","bootstrap"],function(t){return e(t)}):t.BootstrapDialog=e(t.jQuery)}(this,function(t){"use strict";var e=t.fn.modal.Constructor,n=function(t,n){e.call(this,t,n)};n.getModalVersion=function(){var e=null;return e="undefined"==typeof t.fn.modal.Constructor.VERSION?"v3.1":/3\.2\.\d+/.test(t.fn.modal.Constructor.VERSION)?"v3.2":/3\.3\.[1,2]/.test(t.fn.modal.Constructor.VERSION)?"v3.3":"v3.3.4"},n.ORIGINAL_BODY_PADDING=parseInt(t("body").css("padding-right")||0,10),n.METHODS_TO_OVERRIDE={},n.METHODS_TO_OVERRIDE["v3.1"]={},n.METHODS_TO_OVERRIDE["v3.2"]={hide:function(e){if(e&&e.preventDefault(),e=t.Event("hide.bs.modal"),this.$element.trigger(e),this.isShown&&!e.isDefaultPrevented()){this.isShown=!1;var n=this.getGlobalOpenedDialogs();0===n.length&&this.$body.removeClass("modal-open"),this.resetScrollbar(),this.escape(),t(document).off("focusin.bs.modal"),this.$element.removeClass("in").attr("aria-hidden",!0).off("click.dismiss.bs.modal"),t.support.transition&&this.$element.hasClass("fade")?this.$element.one("bsTransitionEnd",t.proxy(this.hideModal,this)).emulateTransitionEnd(300):this.hideModal()}}},n.METHODS_TO_OVERRIDE["v3.3"]={setScrollbar:function(){var t=n.ORIGINAL_BODY_PADDING;this.bodyIsOverflowing&&this.$body.css("padding-right",t+this.scrollbarWidth)},resetScrollbar:function(){var t=this.getGlobalOpenedDialogs();0===t.length&&this.$body.css("padding-right",n.ORIGINAL_BODY_PADDING)},hideModal:function(){this.$element.hide(),this.backdrop(t.proxy(function(){var t=this.getGlobalOpenedDialogs();0===t.length&&this.$body.removeClass("modal-open"),this.resetAdjustments(),this.resetScrollbar(),this.$element.trigger("hidden.bs.modal")},this))}},n.METHODS_TO_OVERRIDE["v3.3.4"]=t.extend({},n.METHODS_TO_OVERRIDE["v3.3"]),n.prototype={constructor:n,getGlobalOpenedDialogs:function(){var e=[];return t.each(o.dialogs,function(t,n){n.isRealized()&&n.isOpened()&&e.push(n)}),e}},n.prototype=t.extend(n.prototype,e.prototype,n.METHODS_TO_OVERRIDE[n.getModalVersion()]);var o=function(e){this.defaultOptions=t.extend(!0,{id:o.newGuid(),buttons:[],data:{},onshow:null,onshown:null,onhide:null,onhidden:null},o.defaultOptions),this.indexedButtons={},this.registeredButtonHotkeys={},this.draggableData={isMouseDown:!1,mouseOffset:{}},this.realized=!1,this.opened=!1,this.initOptions(e),this.holdThisInstance()};return o.BootstrapDialogModal=n,o.NAMESPACE="bootstrap-dialog",o.TYPE_DEFAULT="type-default",o.TYPE_INFO="type-info",o.TYPE_PRIMARY="type-primary",o.TYPE_SUCCESS="type-success",o.TYPE_WARNING="type-warning",o.TYPE_DANGER="type-danger",o.DEFAULT_TEXTS={},o.DEFAULT_TEXTS[o.TYPE_DEFAULT]="Information",o.DEFAULT_TEXTS[o.TYPE_INFO]="Information",o.DEFAULT_TEXTS[o.TYPE_PRIMARY]="Information",o.DEFAULT_TEXTS[o.TYPE_SUCCESS]="Success",o.DEFAULT_TEXTS[o.TYPE_WARNING]="Warning",o.DEFAULT_TEXTS[o.TYPE_DANGER]="Danger",o.DEFAULT_TEXTS.OK="OK",o.DEFAULT_TEXTS.CANCEL="Cancel",o.DEFAULT_TEXTS.CONFIRM="Confirmation",o.SIZE_NORMAL="size-normal",o.SIZE_SMALL="size-small",o.SIZE_WIDE="size-wide",o.SIZE_LARGE="size-large",o.BUTTON_SIZES={},o.BUTTON_SIZES[o.SIZE_NORMAL]="",o.BUTTON_SIZES[o.SIZE_SMALL]="",o.BUTTON_SIZES[o.SIZE_WIDE]="",o.BUTTON_SIZES[o.SIZE_LARGE]="btn-lg",o.ICON_SPINNER="glyphicon glyphicon-asterisk",o.defaultOptions={type:o.TYPE_PRIMARY,size:o.SIZE_NORMAL,cssClass:"",title:null,message:null,nl2br:!0,closable:!0,closeByBackdrop:!0,closeByKeyboard:!0,spinicon:o.ICON_SPINNER,autodestroy:!0,draggable:!1,animate:!0,description:"",tabindex:-1},o.configDefaultOptions=function(e){o.defaultOptions=t.extend(!0,o.defaultOptions,e)},o.dialogs={},o.openAll=function(){t.each(o.dialogs,function(t,e){e.open()})},o.closeAll=function(){t.each(o.dialogs,function(t,e){e.close()})},o.getDialog=function(t){var e=null;return"undefined"!=typeof o.dialogs[t]&&(e=o.dialogs[t]),e},o.setDialog=function(t){return o.dialogs[t.getId()]=t,t},o.addDialog=function(t){return o.setDialog(t)},o.moveFocus=function(){var e=null;t.each(o.dialogs,function(t,n){n.isRealized()&&n.isOpened()&&(e=n)}),null!==e&&e.getModal().focus()},o.METHODS_TO_OVERRIDE={},o.METHODS_TO_OVERRIDE["v3.1"]={handleModalBackdropEvent:function(){return this.getModal().on("click",{dialog:this},function(t){t.target===this&&t.data.dialog.isClosable()&&t.data.dialog.canCloseByBackdrop()&&t.data.dialog.close()}),this},updateZIndex:function(){if(this.isOpened()){var e=1040,n=1050,i=0;t.each(o.dialogs,function(t,e){e.isRealized()&&e.isOpened()&&i++});var s=this.getModal(),a=s.data("bs.modal").$backdrop;s.css("z-index",n+20*(i-1)),a.css("z-index",e+20*(i-1))}return this},open:function(){return!this.isRealized()&&this.realize(),this.getModal().modal("show"),this.updateZIndex(),this}},o.METHODS_TO_OVERRIDE["v3.2"]={handleModalBackdropEvent:o.METHODS_TO_OVERRIDE["v3.1"].handleModalBackdropEvent,updateZIndex:o.METHODS_TO_OVERRIDE["v3.1"].updateZIndex,open:o.METHODS_TO_OVERRIDE["v3.1"].open},o.METHODS_TO_OVERRIDE["v3.3"]={},o.METHODS_TO_OVERRIDE["v3.3.4"]=t.extend({},o.METHODS_TO_OVERRIDE["v3.1"]),o.prototype={constructor:o,initOptions:function(e){return this.options=t.extend(!0,this.defaultOptions,e),this},holdThisInstance:function(){return o.addDialog(this),this},initModalStuff:function(){return this.setModal(this.createModal()).setModalDialog(this.createModalDialog()).setModalContent(this.createModalContent()).setModalHeader(this.createModalHeader()).setModalBody(this.createModalBody()).setModalFooter(this.createModalFooter()),this.getModal().append(this.getModalDialog()),this.getModalDialog().append(this.getModalContent()),this.getModalContent().append(this.getModalHeader()).append(this.getModalBody()).append(this.getModalFooter()),this},createModal:function(){var e=t('<div class="modal" role="dialog" aria-hidden="true"></div>');return e.prop("id",this.getId()),e.attr("aria-labelledby",this.getId()+"_title"),e},getModal:function(){return this.$modal},setModal:function(t){return this.$modal=t,this},createModalDialog:function(){return t('<div class="modal-dialog"></div>')},getModalDialog:function(){return this.$modalDialog},setModalDialog:function(t){return this.$modalDialog=t,this},createModalContent:function(){return t('<div class="modal-content"></div>')},getModalContent:function(){return this.$modalContent},setModalContent:function(t){return this.$modalContent=t,this},createModalHeader:function(){return t('<div class="modal-header"></div>')},getModalHeader:function(){return this.$modalHeader},setModalHeader:function(t){return this.$modalHeader=t,this},createModalBody:function(){return t('<div class="modal-body"></div>')},getModalBody:function(){return this.$modalBody},setModalBody:function(t){return this.$modalBody=t,this},createModalFooter:function(){return t('<div class="modal-footer"></div>')},getModalFooter:function(){return this.$modalFooter},setModalFooter:function(t){return this.$modalFooter=t,this},createDynamicContent:function(t){var e=null;return e="function"==typeof t?t.call(t,this):t,"string"==typeof e&&(e=this.formatStringContent(e)),e},formatStringContent:function(t){return this.options.nl2br?t.replace(/\r\n/g,"<br />").replace(/[\r\n]/g,"<br />"):t},setData:function(t,e){return this.options.data[t]=e,this},getData:function(t){return this.options.data[t]},setId:function(t){return this.options.id=t,this},getId:function(){return this.options.id},getType:function(){return this.options.type},setType:function(t){return this.options.type=t,this.updateType(),this},updateType:function(){if(this.isRealized()){var t=[o.TYPE_DEFAULT,o.TYPE_INFO,o.TYPE_PRIMARY,o.TYPE_SUCCESS,o.TYPE_WARNING,o.TYPE_DANGER];this.getModal().removeClass(t.join(" ")).addClass(this.getType())}return this},getSize:function(){return this.options.size},setSize:function(t){return this.options.size=t,this.updateSize(),this},updateSize:function(){if(this.isRealized()){var e=this;this.getModal().removeClass(o.SIZE_NORMAL).removeClass(o.SIZE_SMALL).removeClass(o.SIZE_WIDE).removeClass(o.SIZE_LARGE),this.getModal().addClass(this.getSize()),this.getModalDialog().removeClass("modal-sm"),this.getSize()===o.SIZE_SMALL&&this.getModalDialog().addClass("modal-sm"),this.getModalDialog().removeClass("modal-lg"),this.getSize()===o.SIZE_WIDE&&this.getModalDialog().addClass("modal-lg"),t.each(this.options.buttons,function(n,o){var i=e.getButton(o.id),s=["btn-lg","btn-sm","btn-xs"],a=!1;if("string"==typeof o.cssClass){var d=o.cssClass.split(" ");t.each(d,function(e,n){-1!==t.inArray(n,s)&&(a=!0)})}a||(i.removeClass(s.join(" ")),i.addClass(e.getButtonSize()))})}return this},getCssClass:function(){return this.options.cssClass},setCssClass:function(t){return this.options.cssClass=t,this},getTitle:function(){return this.options.title},setTitle:function(t){return this.options.title=t,this.updateTitle(),this},updateTitle:function(){if(this.isRealized()){var t=null!==this.getTitle()?this.createDynamicContent(this.getTitle()):this.getDefaultText();this.getModalHeader().find("."+this.getNamespace("title")).html("").append(t).prop("id",this.getId()+"_title")}return this},getMessage:function(){return this.options.message},setMessage:function(t){return this.options.message=t,this.updateMessage(),this},updateMessage:function(){if(this.isRealized()){var t=this.createDynamicContent(this.getMessage());this.getModalBody().find("."+this.getNamespace("message")).html("").append(t)}return this},isClosable:function(){return this.options.closable},setClosable:function(t){return this.options.closable=t,this.updateClosable(),this},setCloseByBackdrop:function(t){return this.options.closeByBackdrop=t,this},canCloseByBackdrop:function(){return this.options.closeByBackdrop},setCloseByKeyboard:function(t){return this.options.closeByKeyboard=t,this},canCloseByKeyboard:function(){return this.options.closeByKeyboard},isAnimate:function(){return this.options.animate},setAnimate:function(t){return this.options.animate=t,this},updateAnimate:function(){return this.isRealized()&&this.getModal().toggleClass("fade",this.isAnimate()),this},getSpinicon:function(){return this.options.spinicon},setSpinicon:function(t){return this.options.spinicon=t,this},addButton:function(t){return this.options.buttons.push(t),this},addButtons:function(e){var n=this;return t.each(e,function(t,e){n.addButton(e)}),this},getButtons:function(){return this.options.buttons},setButtons:function(t){return this.options.buttons=t,this.updateButtons(),this},getButton:function(t){return"undefined"!=typeof this.indexedButtons[t]?this.indexedButtons[t]:null},getButtonSize:function(){return"undefined"!=typeof o.BUTTON_SIZES[this.getSize()]?o.BUTTON_SIZES[this.getSize()]:""},updateButtons:function(){return this.isRealized()&&(0===this.getButtons().length?this.getModalFooter().hide():this.getModalFooter().show().find("."+this.getNamespace("footer")).html("").append(this.createFooterButtons())),this},isAutodestroy:function(){return this.options.autodestroy},setAutodestroy:function(t){this.options.autodestroy=t},getDescription:function(){return this.options.description},setDescription:function(t){return this.options.description=t,this},setTabindex:function(t){return this.options.tabindex=t,this},getTabindex:function(){return this.options.tabindex},updateTabindex:function(){return this.isRealized()&&this.getModal().attr("tabindex",this.getTabindex()),this},getDefaultText:function(){return o.DEFAULT_TEXTS[this.getType()]},getNamespace:function(t){return o.NAMESPACE+"-"+t},createHeaderContent:function(){var e=t("<div></div>");return e.addClass(this.getNamespace("header")),e.append(this.createTitleContent()),e.prepend(this.createCloseButton()),e},createTitleContent:function(){var e=t("<div></div>");return e.addClass(this.getNamespace("title")),e},createCloseButton:function(){var e=t("<div></div>");e.addClass(this.getNamespace("close-button"));var n=t('<button class="close">×</button>');return e.append(n),e.on("click",{dialog:this},function(t){t.data.dialog.close()}),e},createBodyContent:function(){var e=t("<div></div>");return e.addClass(this.getNamespace("body")),e.append(this.createMessageContent()),e},createMessageContent:function(){var e=t("<div></div>");return e.addClass(this.getNamespace("message")),e},createFooterContent:function(){var e=t("<div></div>");return e.addClass(this.getNamespace("footer")),e},createFooterButtons:function(){var e=this,n=t("<div></div>");return n.addClass(this.getNamespace("footer-buttons")),this.indexedButtons={},t.each(this.options.buttons,function(t,i){i.id||(i.id=o.newGuid());var s=e.createButton(i);e.indexedButtons[i.id]=s,n.append(s)}),n},createButton:function(e){var n=t('<button class="btn"></button>');return n.prop("id",e.id),n.data("button",e),"undefined"!=typeof e.icon&&""!==t.trim(e.icon)&&n.append(this.createButtonIcon(e.icon)),"undefined"!=typeof e.label&&n.append(e.label),n.addClass("undefined"!=typeof e.cssClass&&""!==t.trim(e.cssClass)?e.cssClass:"btn-default"),"undefined"!=typeof e.hotkey&&(this.registeredButtonHotkeys[e.hotkey]=n),n.on("click",{dialog:this,$button:n,button:e},function(t){var e=t.data.dialog,n=t.data.$button,o=n.data("button");return o.autospin&&n.toggleSpin(!0),"function"==typeof o.action?o.action.call(n,e,t):void 0}),this.enhanceButton(n),"undefined"!=typeof e.enabled&&n.toggleEnable(e.enabled),n},enhanceButton:function(t){return t.dialog=this,t.toggleEnable=function(t){var e=this;return"undefined"!=typeof t?e.prop("disabled",!t).toggleClass("disabled",!t):e.prop("disabled",!e.prop("disabled")),e},t.enable=function(){var t=this;return t.toggleEnable(!0),t},t.disable=function(){var t=this;return t.toggleEnable(!1),t},t.toggleSpin=function(e){var n=this,o=n.dialog,i=n.find("."+o.getNamespace("button-icon"));return"undefined"==typeof e&&(e=!(t.find(".icon-spin").length>0)),e?(i.hide(),t.prepend(o.createButtonIcon(o.getSpinicon()).addClass("icon-spin"))):(i.show(),t.find(".icon-spin").remove()),n},t.spin=function(){var t=this;return t.toggleSpin(!0),t},t.stopSpin=function(){var t=this;return t.toggleSpin(!1),t},this},createButtonIcon:function(e){var n=t("<span></span>");return n.addClass(this.getNamespace("button-icon")).addClass(e),n},enableButtons:function(e){return t.each(this.indexedButtons,function(t,n){n.toggleEnable(e)}),this},updateClosable:function(){return this.isRealized()&&this.getModalHeader().find("."+this.getNamespace("close-button")).toggle(this.isClosable()),this},onShow:function(t){return this.options.onshow=t,this},onShown:function(t){return this.options.onshown=t,this},onHide:function(t){return this.options.onhide=t,this},onHidden:function(t){return this.options.onhidden=t,this},isRealized:function(){return this.realized},setRealized:function(t){return this.realized=t,this},isOpened:function(){return this.opened},setOpened:function(t){return this.opened=t,this},handleModalEvents:function(){return this.getModal().on("show.bs.modal",{dialog:this},function(t){var e=t.data.dialog;if(e.setOpened(!0),e.isModalEvent(t)&&"function"==typeof e.options.onshow){var n=e.options.onshow(e);return n===!1&&e.setOpened(!1),n}}),this.getModal().on("shown.bs.modal",{dialog:this},function(t){var e=t.data.dialog;e.isModalEvent(t)&&"function"==typeof e.options.onshown&&e.options.onshown(e)}),this.getModal().on("hide.bs.modal",{dialog:this},function(t){var e=t.data.dialog;if(e.setOpened(!1),e.isModalEvent(t)&&"function"==typeof e.options.onhide){var n=e.options.onhide(e);return n===!1&&e.setOpened(!0),n}}),this.getModal().on("hidden.bs.modal",{dialog:this},function(e){var n=e.data.dialog;n.isModalEvent(e)&&"function"==typeof n.options.onhidden&&n.options.onhidden(n),n.isAutodestroy()&&(n.setRealized(!1),delete o.dialogs[n.getId()],t(this).remove()),o.moveFocus()}),this.handleModalBackdropEvent(),this.getModal().on("keyup",{dialog:this},function(t){27===t.which&&t.data.dialog.isClosable()&&t.data.dialog.canCloseByKeyboard()&&t.data.dialog.close()}),this.getModal().on("keyup",{dialog:this},function(e){var n=e.data.dialog;if("undefined"!=typeof n.registeredButtonHotkeys[e.which]){var o=t(n.registeredButtonHotkeys[e.which]);!o.prop("disabled")&&o.focus().trigger("click")}}),this},handleModalBackdropEvent:function(){return this.getModal().on("click",{dialog:this},function(e){t(e.target).hasClass("modal-backdrop")&&e.data.dialog.isClosable()&&e.data.dialog.canCloseByBackdrop()&&e.data.dialog.close()}),this},isModalEvent:function(t){return"undefined"!=typeof t.namespace&&"bs.modal"===t.namespace},makeModalDraggable:function(){return this.options.draggable&&(this.getModalHeader().addClass(this.getNamespace("draggable")).on("mousedown",{dialog:this},function(t){var e=t.data.dialog;e.draggableData.isMouseDown=!0;var n=e.getModalDialog().offset();e.draggableData.mouseOffset={top:t.clientY-n.top,left:t.clientX-n.left}}),this.getModal().on("mouseup mouseleave",{dialog:this},function(t){t.data.dialog.draggableData.isMouseDown=!1}),t("body").on("mousemove",{dialog:this},function(t){var e=t.data.dialog;e.draggableData.isMouseDown&&e.getModalDialog().offset({top:t.clientY-e.draggableData.mouseOffset.top,left:t.clientX-e.draggableData.mouseOffset.left})})),this},realize:function(){return this.initModalStuff(),this.getModal().addClass(o.NAMESPACE).addClass(this.getCssClass()),this.updateSize(),this.getDescription()&&this.getModal().attr("aria-describedby",this.getDescription()),this.getModalFooter().append(this.createFooterContent()),this.getModalHeader().append(this.createHeaderContent()),this.getModalBody().append(this.createBodyContent()),this.getModal().data("bs.modal",new n(this.getModal(),{backdrop:"static",keyboard:!1,show:!1})),this.makeModalDraggable(),this.handleModalEvents(),this.setRealized(!0),this.updateButtons(),this.updateType(),this.updateTitle(),this.updateMessage(),this.updateClosable(),this.updateAnimate(),this.updateSize(),this.updateTabindex(),this},open:function(){return!this.isRealized()&&this.realize(),this.getModal().modal("show"),this},close:function(){return!this.isRealized()&&this.realize(),this.getModal().modal("hide"),this}},o.prototype=t.extend(o.prototype,o.METHODS_TO_OVERRIDE[n.getModalVersion()]),o.newGuid=function(){return"xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx".replace(/[xy]/g,function(t){var e=16*Math.random()|0,n="x"===t?e:3&e|8;return n.toString(16)})},o.show=function(t){return new o(t).open()},o.alert=function(){var e={},n={type:o.TYPE_PRIMARY,title:null,message:null,closable:!1,draggable:!1,buttonLabel:o.DEFAULT_TEXTS.OK,callback:null};e="object"==typeof arguments[0]&&arguments[0].constructor==={}.constructor?t.extend(!0,n,arguments[0]):t.extend(!0,n,{message:arguments[0],callback:"undefined"!=typeof arguments[1]?arguments[1]:null});var i=new o(e);return i.setData("callback",e.callback),i.addButton({label:e.buttonLabel,action:function(t){return"function"==typeof t.getData("callback")&&t.getData("callback").call(this,!0)===!1?!1:(t.setData("btnClicked",!0),t.close())}}),i.onHide("function"==typeof i.options.onhide?function(t){var e=!0;return!t.getData("btnClicked")&&t.isClosable()&&"function"==typeof t.getData("callback")&&(e=t.getData("callback")(!1)),e===!1?!1:e=this.onhide(t)}.bind({onhide:i.options.onhide}):function(t){var e=!0;return!t.getData("btnClicked")&&t.isClosable()&&"function"==typeof t.getData("callback")&&(e=t.getData("callback")(!1)),e}),i.open()},o.confirm=function(){var e={},n={type:o.TYPE_PRIMARY,title:null,message:null,closable:!1,draggable:!1,btnCancelLabel:o.DEFAULT_TEXTS.CANCEL,btnCancelClass:null,btnOKLabel:o.DEFAULT_TEXTS.OK,btnOKClass:null,callback:null};e="object"==typeof arguments[0]&&arguments[0].constructor==={}.constructor?t.extend(!0,n,arguments[0]):t.extend(!0,n,{message:arguments[0],callback:"undefined"!=typeof arguments[1]?arguments[1]:null}),null===e.btnOKClass&&(e.btnOKClass=["btn",e.type.split("-")[1]].join("-"));var i=new o(e);return i.setData("callback",e.callback),i.addButton({label:e.btnCancelLabel,cssClass:e.btnCancelClass,action:function(t){return"function"==typeof t.getData("callback")&&t.getData("callback").call(this,!1)===!1?!1:t.close()}}),i.addButton({label:e.btnOKLabel,cssClass:e.btnOKClass,action:function(t){return"function"==typeof t.getData("callback")&&t.getData("callback").call(this,!0)===!1?!1:t.close()}}),i.open()},o.warning=function(t,e){return new o({type:o.TYPE_WARNING,message:t}).open()},o.danger=function(t,e){return new o({type:o.TYPE_DANGER,message:t}).open()},o.success=function(t,e){return new o({type:o.TYPE_SUCCESS,message:t}).open()},o});
|
|
/*!
|
|
* Jasny Bootstrap v3.1.3 (http://jasny.github.io/bootstrap)
|
|
* Copyright 2012-2014 Arnold Daniels
|
|
* Licensed under Apache-2.0 (https://github.com/jasny/bootstrap/blob/master/LICENSE)
|
|
*/
|
|
|
|
if (typeof jQuery === 'undefined') { throw new Error('Jasny Bootstrap\'s JavaScript requires jQuery') }
|
|
|
|
/* ========================================================================
|
|
* Bootstrap: transition.js v3.1.3
|
|
* http://getbootstrap.com/javascript/#transitions
|
|
* ========================================================================
|
|
* Copyright 2011-2014 Twitter, Inc.
|
|
* Licensed under MIT (https://github.com/twbs/bootstrap/blob/master/LICENSE)
|
|
* ======================================================================== */
|
|
|
|
|
|
+function ($) {
|
|
'use strict';
|
|
|
|
// CSS TRANSITION SUPPORT (Shoutout: http://www.modernizr.com/)
|
|
// ============================================================
|
|
|
|
function transitionEnd() {
|
|
var el = document.createElement('bootstrap')
|
|
|
|
var transEndEventNames = {
|
|
WebkitTransition : 'webkitTransitionEnd',
|
|
MozTransition : 'transitionend',
|
|
OTransition : 'oTransitionEnd otransitionend',
|
|
transition : 'transitionend'
|
|
}
|
|
|
|
for (var name in transEndEventNames) {
|
|
if (el.style[name] !== undefined) {
|
|
return { end: transEndEventNames[name] }
|
|
}
|
|
}
|
|
|
|
return false // explicit for ie8 ( ._.)
|
|
}
|
|
|
|
if ($.support.transition !== undefined) return // Prevent conflict with Twitter Bootstrap
|
|
|
|
// http://blog.alexmaccaw.com/css-transitions
|
|
$.fn.emulateTransitionEnd = function (duration) {
|
|
var called = false, $el = this
|
|
$(this).one($.support.transition.end, function () { called = true })
|
|
var callback = function () { if (!called) $($el).trigger($.support.transition.end) }
|
|
setTimeout(callback, duration)
|
|
return this
|
|
}
|
|
|
|
$(function () {
|
|
$.support.transition = transitionEnd()
|
|
})
|
|
|
|
}(window.jQuery);
|
|
|
|
/* ========================================================================
|
|
* Bootstrap: offcanvas.js v3.1.3
|
|
* http://jasny.github.io/bootstrap/javascript/#offcanvas
|
|
* ========================================================================
|
|
* Copyright 2013-2014 Arnold Daniels
|
|
*
|
|
* Licensed under the Apache License, Version 2.0 (the "License")
|
|
* you may not use this file except in compliance with the License.
|
|
* You may obtain a copy of the License at
|
|
*
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
*
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
* See the License for the specific language governing permissions and
|
|
* limitations under the License.
|
|
* ======================================================================== */
|
|
|
|
+function ($) { "use strict";
|
|
|
|
// OFFCANVAS PUBLIC CLASS DEFINITION
|
|
// =================================
|
|
|
|
var OffCanvas = function (element, options) {
|
|
this.$element = $(element)
|
|
this.options = $.extend({}, OffCanvas.DEFAULTS, options)
|
|
this.state = null
|
|
this.placement = null
|
|
|
|
if (this.options.recalc) {
|
|
this.calcClone()
|
|
$(window).on('resize', $.proxy(this.recalc, this))
|
|
}
|
|
|
|
if (this.options.autohide)
|
|
$(document).on('click', $.proxy(this.autohide, this))
|
|
|
|
if (this.options.toggle) this.toggle()
|
|
|
|
if (this.options.disablescrolling) {
|
|
this.options.disableScrolling = this.options.disablescrolling
|
|
delete this.options.disablescrolling
|
|
}
|
|
}
|
|
|
|
OffCanvas.DEFAULTS = {
|
|
toggle: true,
|
|
placement: 'auto',
|
|
autohide: true,
|
|
recalc: true,
|
|
disableScrolling: true
|
|
}
|
|
|
|
OffCanvas.prototype.offset = function () {
|
|
switch (this.placement) {
|
|
case 'left':
|
|
case 'right': return this.$element.outerWidth()
|
|
case 'top':
|
|
case 'bottom': return this.$element.outerHeight()
|
|
}
|
|
}
|
|
|
|
OffCanvas.prototype.calcPlacement = function () {
|
|
if (this.options.placement !== 'auto') {
|
|
this.placement = this.options.placement
|
|
return
|
|
}
|
|
|
|
if (!this.$element.hasClass('in')) {
|
|
this.$element.css('visiblity', 'hidden !important').addClass('in')
|
|
}
|
|
|
|
var horizontal = $(window).width() / this.$element.width()
|
|
var vertical = $(window).height() / this.$element.height()
|
|
|
|
var element = this.$element
|
|
function ab(a, b) {
|
|
if (element.css(b) === 'auto') return a
|
|
if (element.css(a) === 'auto') return b
|
|
|
|
var size_a = parseInt(element.css(a), 10)
|
|
var size_b = parseInt(element.css(b), 10)
|
|
|
|
return size_a > size_b ? b : a
|
|
}
|
|
|
|
this.placement = horizontal >= vertical ? ab('left', 'right') : ab('top', 'bottom')
|
|
|
|
if (this.$element.css('visibility') === 'hidden !important') {
|
|
this.$element.removeClass('in').css('visiblity', '')
|
|
}
|
|
}
|
|
|
|
OffCanvas.prototype.opposite = function (placement) {
|
|
switch (placement) {
|
|
case 'top': return 'bottom'
|
|
case 'left': return 'right'
|
|
case 'bottom': return 'top'
|
|
case 'right': return 'left'
|
|
}
|
|
}
|
|
|
|
OffCanvas.prototype.getCanvasElements = function() {
|
|
// Return a set containing the canvas plus all fixed elements
|
|
var canvas = this.options.canvas ? $(this.options.canvas) : this.$element
|
|
|
|
var fixed_elements = canvas.find('*').filter(function() {
|
|
return $(this).css('position') === 'fixed'
|
|
}).not(this.options.exclude)
|
|
|
|
return canvas.add(fixed_elements)
|
|
}
|
|
|
|
OffCanvas.prototype.slide = function (elements, offset, callback) {
|
|
// Use jQuery animation if CSS transitions aren't supported
|
|
if (!$.support.transition) {
|
|
var anim = {}
|
|
anim[this.placement] = "+=" + offset
|
|
return elements.animate(anim, 350, callback)
|
|
}
|
|
|
|
var placement = this.placement
|
|
var opposite = this.opposite(placement)
|
|
|
|
elements.each(function() {
|
|
if ($(this).css(placement) !== 'auto')
|
|
$(this).css(placement, (parseInt($(this).css(placement), 10) || 0) + offset)
|
|
|
|
if ($(this).css(opposite) !== 'auto')
|
|
$(this).css(opposite, (parseInt($(this).css(opposite), 10) || 0) - offset)
|
|
})
|
|
|
|
this.$element
|
|
.one($.support.transition.end, callback)
|
|
.emulateTransitionEnd(350)
|
|
}
|
|
|
|
OffCanvas.prototype.disableScrolling = function() {
|
|
var bodyWidth = $('body').width()
|
|
var prop = 'padding-' + this.opposite(this.placement)
|
|
|
|
if ($('body').data('offcanvas-style') === undefined) {
|
|
$('body').data('offcanvas-style', $('body').attr('style') || '')
|
|
}
|
|
|
|
$('body').css('overflow', 'hidden')
|
|
|
|
if ($('body').width() > bodyWidth) {
|
|
var padding = parseInt($('body').css(prop), 10) + $('body').width() - bodyWidth
|
|
|
|
setTimeout(function() {
|
|
$('body').css(prop, padding)
|
|
}, 1)
|
|
}
|
|
}
|
|
|
|
OffCanvas.prototype.show = function () {
|
|
if (this.state) return
|
|
|
|
var startEvent = $.Event('show.bs.offcanvas')
|
|
this.$element.trigger(startEvent)
|
|
if (startEvent.isDefaultPrevented()) return
|
|
|
|
this.state = 'slide-in'
|
|
this.calcPlacement();
|
|
|
|
var elements = this.getCanvasElements()
|
|
var placement = this.placement
|
|
var opposite = this.opposite(placement)
|
|
var offset = this.offset()
|
|
|
|
if (elements.index(this.$element) !== -1) {
|
|
$(this.$element).data('offcanvas-style', $(this.$element).attr('style') || '')
|
|
this.$element.css(placement, -1 * offset)
|
|
this.$element.css(placement); // Workaround: Need to get the CSS property for it to be applied before the next line of code
|
|
}
|
|
|
|
elements.addClass('canvas-sliding').each(function() {
|
|
if ($(this).data('offcanvas-style') === undefined) $(this).data('offcanvas-style', $(this).attr('style') || '')
|
|
if ($(this).css('position') === 'static') $(this).css('position', 'relative')
|
|
if (($(this).css(placement) === 'auto' || $(this).css(placement) === '0px') &&
|
|
($(this).css(opposite) === 'auto' || $(this).css(opposite) === '0px')) {
|
|
$(this).css(placement, 0)
|
|
}
|
|
})
|
|
|
|
if (this.options.disableScrolling) this.disableScrolling()
|
|
|
|
var complete = function () {
|
|
if (this.state != 'slide-in') return
|
|
|
|
this.state = 'slid'
|
|
|
|
elements.removeClass('canvas-sliding').addClass('canvas-slid')
|
|
this.$element.trigger('shown.bs.offcanvas')
|
|
}
|
|
|
|
setTimeout($.proxy(function() {
|
|
this.$element.addClass('in')
|
|
this.slide(elements, offset, $.proxy(complete, this))
|
|
}, this), 1)
|
|
}
|
|
|
|
OffCanvas.prototype.hide = function (fast) {
|
|
if (this.state !== 'slid') return
|
|
|
|
var startEvent = $.Event('hide.bs.offcanvas')
|
|
this.$element.trigger(startEvent)
|
|
if (startEvent.isDefaultPrevented()) return
|
|
|
|
this.state = 'slide-out'
|
|
|
|
var elements = $('.canvas-slid')
|
|
var placement = this.placement
|
|
var offset = -1 * this.offset()
|
|
|
|
var complete = function () {
|
|
if (this.state != 'slide-out') return
|
|
|
|
this.state = null
|
|
this.placement = null
|
|
|
|
this.$element.removeClass('in')
|
|
|
|
elements.removeClass('canvas-sliding')
|
|
elements.add(this.$element).add('body').each(function() {
|
|
$(this).attr('style', $(this).data('offcanvas-style')).removeData('offcanvas-style')
|
|
})
|
|
|
|
this.$element.trigger('hidden.bs.offcanvas')
|
|
}
|
|
|
|
elements.removeClass('canvas-slid').addClass('canvas-sliding')
|
|
|
|
setTimeout($.proxy(function() {
|
|
this.slide(elements, offset, $.proxy(complete, this))
|
|
}, this), 1)
|
|
}
|
|
|
|
OffCanvas.prototype.toggle = function () {
|
|
if (this.state === 'slide-in' || this.state === 'slide-out') return
|
|
this[this.state === 'slid' ? 'hide' : 'show']()
|
|
}
|
|
|
|
OffCanvas.prototype.calcClone = function() {
|
|
this.$calcClone = this.$element.clone()
|
|
.html('')
|
|
.addClass('offcanvas-clone').removeClass('in')
|
|
.appendTo($('body'))
|
|
}
|
|
|
|
OffCanvas.prototype.recalc = function () {
|
|
if (this.$calcClone.css('display') === 'none' || (this.state !== 'slid' && this.state !== 'slide-in')) return
|
|
|
|
this.state = null
|
|
this.placement = null
|
|
var elements = this.getCanvasElements()
|
|
|
|
this.$element.removeClass('in')
|
|
|
|
elements.removeClass('canvas-slid')
|
|
elements.add(this.$element).add('body').each(function() {
|
|
$(this).attr('style', $(this).data('offcanvas-style')).removeData('offcanvas-style')
|
|
})
|
|
}
|
|
|
|
OffCanvas.prototype.autohide = function (e) {
|
|
if ($(e.target).closest(this.$element).length === 0) this.hide()
|
|
}
|
|
|
|
// OFFCANVAS PLUGIN DEFINITION
|
|
// ==========================
|
|
|
|
var old = $.fn.offcanvas
|
|
|
|
$.fn.offcanvas = function (option) {
|
|
return this.each(function () {
|
|
var $this = $(this)
|
|
var data = $this.data('bs.offcanvas')
|
|
var options = $.extend({}, OffCanvas.DEFAULTS, $this.data(), typeof option === 'object' && option)
|
|
|
|
if (!data) $this.data('bs.offcanvas', (data = new OffCanvas(this, options)))
|
|
if (typeof option === 'string') data[option]()
|
|
})
|
|
}
|
|
|
|
$.fn.offcanvas.Constructor = OffCanvas
|
|
|
|
|
|
// OFFCANVAS NO CONFLICT
|
|
// ====================
|
|
|
|
$.fn.offcanvas.noConflict = function () {
|
|
$.fn.offcanvas = old
|
|
return this
|
|
}
|
|
|
|
|
|
// OFFCANVAS DATA-API
|
|
// =================
|
|
|
|
$(document).on('click.bs.offcanvas.data-api', '[data-toggle=offcanvas]', function (e) {
|
|
var $this = $(this), href
|
|
var target = $this.attr('data-target')
|
|
|| e.preventDefault()
|
|
|| (href = $this.attr('href')) && href.replace(/.*(?=#[^\s]+$)/, '') //strip for ie7
|
|
var $canvas = $(target)
|
|
var data = $canvas.data('bs.offcanvas')
|
|
var option = data ? 'toggle' : $this.data()
|
|
|
|
e.stopPropagation()
|
|
|
|
if (data) data.toggle()
|
|
else $canvas.offcanvas(option)
|
|
})
|
|
|
|
}(window.jQuery);
|
|
|
|
/* ============================================================
|
|
* Bootstrap: rowlink.js v3.1.3
|
|
* http://jasny.github.io/bootstrap/javascript/#rowlink
|
|
* ============================================================
|
|
* Copyright 2012-2014 Arnold Daniels
|
|
*
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
* you may not use this file except in compliance with the License.
|
|
* You may obtain a copy of the License at
|
|
*
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
*
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
* See the License for the specific language governing permissions and
|
|
* limitations under the License.
|
|
* ============================================================ */
|
|
|
|
+function ($) { "use strict";
|
|
|
|
var Rowlink = function (element, options) {
|
|
this.$element = $(element)
|
|
this.options = $.extend({}, Rowlink.DEFAULTS, options)
|
|
|
|
this.$element.on('click.bs.rowlink', 'td:not(.rowlink-skip)', $.proxy(this.click, this))
|
|
}
|
|
|
|
Rowlink.DEFAULTS = {
|
|
target: "a"
|
|
}
|
|
|
|
Rowlink.prototype.click = function(e) {
|
|
var target = $(e.currentTarget).closest('tr').find(this.options.target)[0]
|
|
if ($(e.target)[0] === target) return
|
|
|
|
e.preventDefault();
|
|
|
|
if (target.click) {
|
|
target.click()
|
|
} else if (document.createEvent) {
|
|
var evt = document.createEvent("MouseEvents");
|
|
evt.initMouseEvent("click", true, true, window, 0, 0, 0, 0, 0, false, false, false, false, 0, null);
|
|
target.dispatchEvent(evt);
|
|
}
|
|
}
|
|
|
|
|
|
// ROWLINK PLUGIN DEFINITION
|
|
// ===========================
|
|
|
|
var old = $.fn.rowlink
|
|
|
|
$.fn.rowlink = function (options) {
|
|
return this.each(function () {
|
|
var $this = $(this)
|
|
var data = $this.data('bs.rowlink')
|
|
if (!data) $this.data('bs.rowlink', (data = new Rowlink(this, options)))
|
|
})
|
|
}
|
|
|
|
$.fn.rowlink.Constructor = Rowlink
|
|
|
|
|
|
// ROWLINK NO CONFLICT
|
|
// ====================
|
|
|
|
$.fn.rowlink.noConflict = function () {
|
|
$.fn.rowlink = old
|
|
return this
|
|
}
|
|
|
|
|
|
// ROWLINK DATA-API
|
|
// ==================
|
|
|
|
$(document).on('click.bs.rowlink.data-api', '[data-link="row"]', function (e) {
|
|
if ($(e.target).closest('.rowlink-skip').length !== 0) return
|
|
|
|
var $this = $(this)
|
|
if ($this.data('bs.rowlink')) return
|
|
$this.rowlink($this.data())
|
|
$(e.target).trigger('click.bs.rowlink')
|
|
})
|
|
|
|
}(window.jQuery);
|
|
|
|
/* ===========================================================
|
|
* Bootstrap: inputmask.js v3.1.0
|
|
* http://jasny.github.io/bootstrap/javascript/#inputmask
|
|
*
|
|
* Based on Masked Input plugin by Josh Bush (digitalbush.com)
|
|
* ===========================================================
|
|
* Copyright 2012-2014 Arnold Daniels
|
|
*
|
|
* Licensed under the Apache License, Version 2.0 (the "License")
|
|
* you may not use this file except in compliance with the License.
|
|
* You may obtain a copy of the License at
|
|
*
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
*
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
* See the License for the specific language governing permissions and
|
|
* limitations under the License.
|
|
* ========================================================== */
|
|
|
|
+function ($) { "use strict";
|
|
|
|
var isIphone = (window.orientation !== undefined)
|
|
var isAndroid = navigator.userAgent.toLowerCase().indexOf("android") > -1
|
|
var isIE = window.navigator.appName == 'Microsoft Internet Explorer'
|
|
|
|
// INPUTMASK PUBLIC CLASS DEFINITION
|
|
// =================================
|
|
|
|
var Inputmask = function (element, options) {
|
|
if (isAndroid) return // No support because caret positioning doesn't work on Android
|
|
|
|
this.$element = $(element)
|
|
this.options = $.extend({}, Inputmask.DEFAULTS, options)
|
|
this.mask = String(this.options.mask)
|
|
|
|
this.init()
|
|
this.listen()
|
|
|
|
this.checkVal() //Perform initial check for existing values
|
|
}
|
|
|
|
Inputmask.DEFAULTS = {
|
|
mask: "",
|
|
placeholder: "_",
|
|
definitions: {
|
|
'9': "[0-9]",
|
|
'a': "[A-Za-z]",
|
|
'w': "[A-Za-z0-9]",
|
|
'*': "."
|
|
}
|
|
}
|
|
|
|
Inputmask.prototype.init = function() {
|
|
var defs = this.options.definitions
|
|
var len = this.mask.length
|
|
|
|
this.tests = []
|
|
this.partialPosition = this.mask.length
|
|
this.firstNonMaskPos = null
|
|
|
|
$.each(this.mask.split(""), $.proxy(function(i, c) {
|
|
if (c == '?') {
|
|
len--
|
|
this.partialPosition = i
|
|
} else if (defs[c]) {
|
|
this.tests.push(new RegExp(defs[c]))
|
|
if (this.firstNonMaskPos === null)
|
|
this.firstNonMaskPos = this.tests.length - 1
|
|
} else {
|
|
this.tests.push(null)
|
|
}
|
|
}, this))
|
|
|
|
this.buffer = $.map(this.mask.split(""), $.proxy(function(c, i) {
|
|
if (c != '?') return defs[c] ? this.options.placeholder : c
|
|
}, this))
|
|
|
|
this.focusText = this.$element.val()
|
|
|
|
this.$element.data("rawMaskFn", $.proxy(function() {
|
|
return $.map(this.buffer, function(c, i) {
|
|
return this.tests[i] && c != this.options.placeholder ? c : null
|
|
}).join('')
|
|
}, this))
|
|
}
|
|
|
|
Inputmask.prototype.listen = function() {
|
|
if (this.$element.attr("readonly")) return
|
|
|
|
var pasteEventName = (isIE ? 'paste' : 'input') + ".mask"
|
|
|
|
this.$element
|
|
.on("unmask.bs.inputmask", $.proxy(this.unmask, this))
|
|
|
|
.on("focus.bs.inputmask", $.proxy(this.focusEvent, this))
|
|
.on("blur.bs.inputmask", $.proxy(this.blurEvent, this))
|
|
|
|
.on("keydown.bs.inputmask", $.proxy(this.keydownEvent, this))
|
|
.on("keypress.bs.inputmask", $.proxy(this.keypressEvent, this))
|
|
|
|
.on(pasteEventName, $.proxy(this.pasteEvent, this))
|
|
}
|
|
|
|
//Helper Function for Caret positioning
|
|
Inputmask.prototype.caret = function(begin, end) {
|
|
if (this.$element.length === 0) return
|
|
if (typeof begin == 'number') {
|
|
end = (typeof end == 'number') ? end : begin
|
|
return this.$element.each(function() {
|
|
if (this.setSelectionRange) {
|
|
this.setSelectionRange(begin, end)
|
|
} else if (this.createTextRange) {
|
|
var range = this.createTextRange()
|
|
range.collapse(true)
|
|
range.moveEnd('character', end)
|
|
range.moveStart('character', begin)
|
|
range.select()
|
|
}
|
|
})
|
|
} else {
|
|
if (this.$element[0].setSelectionRange) {
|
|
begin = this.$element[0].selectionStart
|
|
end = this.$element[0].selectionEnd
|
|
} else if (document.selection && document.selection.createRange) {
|
|
var range = document.selection.createRange()
|
|
begin = 0 - range.duplicate().moveStart('character', -100000)
|
|
end = begin + range.text.length
|
|
}
|
|
return {
|
|
begin: begin,
|
|
end: end
|
|
}
|
|
}
|
|
}
|
|
|
|
Inputmask.prototype.seekNext = function(pos) {
|
|
var len = this.mask.length
|
|
while (++pos <= len && !this.tests[pos]);
|
|
|
|
return pos
|
|
}
|
|
|
|
Inputmask.prototype.seekPrev = function(pos) {
|
|
while (--pos >= 0 && !this.tests[pos]);
|
|
|
|
return pos
|
|
}
|
|
|
|
Inputmask.prototype.shiftL = function(begin,end) {
|
|
var len = this.mask.length
|
|
|
|
if (begin < 0) return
|
|
|
|
for (var i = begin, j = this.seekNext(end); i < len; i++) {
|
|
if (this.tests[i]) {
|
|
if (j < len && this.tests[i].test(this.buffer[j])) {
|
|
this.buffer[i] = this.buffer[j]
|
|
this.buffer[j] = this.options.placeholder
|
|
} else
|
|
break
|
|
j = this.seekNext(j)
|
|
}
|
|
}
|
|
this.writeBuffer()
|
|
this.caret(Math.max(this.firstNonMaskPos, begin))
|
|
}
|
|
|
|
Inputmask.prototype.shiftR = function(pos) {
|
|
var len = this.mask.length
|
|
|
|
for (var i = pos, c = this.options.placeholder; i < len; i++) {
|
|
if (this.tests[i]) {
|
|
var j = this.seekNext(i)
|
|
var t = this.buffer[i]
|
|
this.buffer[i] = c
|
|
if (j < len && this.tests[j].test(t))
|
|
c = t
|
|
else
|
|
break
|
|
}
|
|
}
|
|
},
|
|
|
|
Inputmask.prototype.unmask = function() {
|
|
this.$element
|
|
.unbind(".mask")
|
|
.removeData("inputmask")
|
|
}
|
|
|
|
Inputmask.prototype.focusEvent = function() {
|
|
this.focusText = this.$element.val()
|
|
var len = this.mask.length
|
|
var pos = this.checkVal()
|
|
this.writeBuffer()
|
|
|
|
var that = this
|
|
var moveCaret = function() {
|
|
if (pos == len)
|
|
that.caret(0, pos)
|
|
else
|
|
that.caret(pos)
|
|
}
|
|
|
|
moveCaret()
|
|
setTimeout(moveCaret, 50)
|
|
}
|
|
|
|
Inputmask.prototype.blurEvent = function() {
|
|
this.checkVal()
|
|
if (this.$element.val() !== this.focusText)
|
|
this.$element.trigger('change')
|
|
}
|
|
|
|
Inputmask.prototype.keydownEvent = function(e) {
|
|
var k = e.which
|
|
|
|
//backspace, delete, and escape get special treatment
|
|
if (k == 8 || k == 46 || (isIphone && k == 127)) {
|
|
var pos = this.caret(),
|
|
begin = pos.begin,
|
|
end = pos.end
|
|
|
|
if (end - begin === 0) {
|
|
begin = k != 46 ? this.seekPrev(begin) : (end = this.seekNext(begin - 1))
|
|
end = k == 46 ? this.seekNext(end) : end
|
|
}
|
|
this.clearBuffer(begin, end)
|
|
this.shiftL(begin, end - 1)
|
|
|
|
return false
|
|
} else if (k == 27) {//escape
|
|
this.$element.val(this.focusText)
|
|
this.caret(0, this.checkVal())
|
|
return false
|
|
}
|
|
}
|
|
|
|
Inputmask.prototype.keypressEvent = function(e) {
|
|
var len = this.mask.length
|
|
|
|
var k = e.which,
|
|
pos = this.caret()
|
|
|
|
if (e.ctrlKey || e.altKey || e.metaKey || k < 32) {//Ignore
|
|
return true
|
|
} else if (k) {
|
|
if (pos.end - pos.begin !== 0) {
|
|
this.clearBuffer(pos.begin, pos.end)
|
|
this.shiftL(pos.begin, pos.end - 1)
|
|
}
|
|
|
|
var p = this.seekNext(pos.begin - 1)
|
|
if (p < len) {
|
|
var c = String.fromCharCode(k)
|
|
if (this.tests[p].test(c)) {
|
|
this.shiftR(p)
|
|
this.buffer[p] = c
|
|
this.writeBuffer()
|
|
var next = this.seekNext(p)
|
|
this.caret(next)
|
|
}
|
|
}
|
|
return false
|
|
}
|
|
}
|
|
|
|
Inputmask.prototype.pasteEvent = function() {
|
|
var that = this
|
|
|
|
setTimeout(function() {
|
|
that.caret(that.checkVal(true))
|
|
}, 0)
|
|
}
|
|
|
|
Inputmask.prototype.clearBuffer = function(start, end) {
|
|
var len = this.mask.length
|
|
|
|
for (var i = start; i < end && i < len; i++) {
|
|
if (this.tests[i])
|
|
this.buffer[i] = this.options.placeholder
|
|
}
|
|
}
|
|
|
|
Inputmask.prototype.writeBuffer = function() {
|
|
return this.$element.val(this.buffer.join('')).val()
|
|
}
|
|
|
|
Inputmask.prototype.checkVal = function(allow) {
|
|
var len = this.mask.length
|
|
//try to place characters where they belong
|
|
var test = this.$element.val()
|
|
var lastMatch = -1
|
|
|
|
for (var i = 0, pos = 0; i < len; i++) {
|
|
if (this.tests[i]) {
|
|
this.buffer[i] = this.options.placeholder
|
|
while (pos++ < test.length) {
|
|
var c = test.charAt(pos - 1)
|
|
if (this.tests[i].test(c)) {
|
|
this.buffer[i] = c
|
|
lastMatch = i
|
|
break
|
|
}
|
|
}
|
|
if (pos > test.length)
|
|
break
|
|
} else if (this.buffer[i] == test.charAt(pos) && i != this.partialPosition) {
|
|
pos++
|
|
lastMatch = i
|
|
}
|
|
}
|
|
if (!allow && lastMatch + 1 < this.partialPosition) {
|
|
this.$element.val("")
|
|
this.clearBuffer(0, len)
|
|
} else if (allow || lastMatch + 1 >= this.partialPosition) {
|
|
this.writeBuffer()
|
|
if (!allow) this.$element.val(this.$element.val().substring(0, lastMatch + 1))
|
|
}
|
|
return (this.partialPosition ? i : this.firstNonMaskPos)
|
|
}
|
|
|
|
|
|
// INPUTMASK PLUGIN DEFINITION
|
|
// ===========================
|
|
|
|
var old = $.fn.inputmask
|
|
|
|
$.fn.inputmask = function (options) {
|
|
return this.each(function () {
|
|
var $this = $(this)
|
|
var data = $this.data('bs.inputmask')
|
|
|
|
if (!data) $this.data('bs.inputmask', (data = new Inputmask(this, options)))
|
|
})
|
|
}
|
|
|
|
$.fn.inputmask.Constructor = Inputmask
|
|
|
|
|
|
// INPUTMASK NO CONFLICT
|
|
// ====================
|
|
|
|
$.fn.inputmask.noConflict = function () {
|
|
$.fn.inputmask = old
|
|
return this
|
|
}
|
|
|
|
|
|
// INPUTMASK DATA-API
|
|
// ==================
|
|
|
|
$(document).on('focus.bs.inputmask.data-api', '[data-mask]', function (e) {
|
|
var $this = $(this)
|
|
if ($this.data('bs.inputmask')) return
|
|
$this.inputmask($this.data())
|
|
})
|
|
|
|
}(window.jQuery);
|
|
|
|
/* ===========================================================
|
|
* Bootstrap: fileinput.js v3.1.3
|
|
* http://jasny.github.com/bootstrap/javascript/#fileinput
|
|
* ===========================================================
|
|
* Copyright 2012-2014 Arnold Daniels
|
|
*
|
|
* Licensed under the Apache License, Version 2.0 (the "License")
|
|
* you may not use this file except in compliance with the License.
|
|
* You may obtain a copy of the License at
|
|
*
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
*
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
* See the License for the specific language governing permissions and
|
|
* limitations under the License.
|
|
* ========================================================== */
|
|
|
|
+function ($) { "use strict";
|
|
|
|
var isIE = window.navigator.appName == 'Microsoft Internet Explorer'
|
|
|
|
// FILEUPLOAD PUBLIC CLASS DEFINITION
|
|
// =================================
|
|
|
|
var Fileinput = function (element, options) {
|
|
this.$element = $(element)
|
|
|
|
this.$input = this.$element.find(':file')
|
|
if (this.$input.length === 0) return
|
|
|
|
this.name = this.$input.attr('name') || options.name
|
|
|
|
this.$hidden = this.$element.find('input[type=hidden][name="' + this.name + '"]')
|
|
if (this.$hidden.length === 0) {
|
|
this.$hidden = $('<input type="hidden">').insertBefore(this.$input)
|
|
}
|
|
|
|
this.$preview = this.$element.find('.fileinput-preview')
|
|
var height = this.$preview.css('height')
|
|
if (this.$preview.css('display') !== 'inline' && height !== '0px' && height !== 'none') {
|
|
this.$preview.css('line-height', height)
|
|
}
|
|
|
|
this.original = {
|
|
exists: this.$element.hasClass('fileinput-exists'),
|
|
preview: this.$preview.html(),
|
|
hiddenVal: this.$hidden.val()
|
|
}
|
|
|
|
this.listen()
|
|
}
|
|
|
|
Fileinput.prototype.listen = function() {
|
|
this.$input.on('change.bs.fileinput', $.proxy(this.change, this))
|
|
$(this.$input[0].form).on('reset.bs.fileinput', $.proxy(this.reset, this))
|
|
|
|
this.$element.find('[data-trigger="fileinput"]').on('click.bs.fileinput', $.proxy(this.trigger, this))
|
|
this.$element.find('[data-dismiss="fileinput"]').on('click.bs.fileinput', $.proxy(this.clear, this))
|
|
},
|
|
|
|
Fileinput.prototype.change = function(e) {
|
|
var files = e.target.files === undefined ? (e.target && e.target.value ? [{ name: e.target.value.replace(/^.+\\/, '')}] : []) : e.target.files
|
|
|
|
e.stopPropagation()
|
|
|
|
if (files.length === 0) {
|
|
this.clear()
|
|
return
|
|
}
|
|
|
|
this.$hidden.val('')
|
|
this.$hidden.attr('name', '')
|
|
this.$input.attr('name', this.name)
|
|
|
|
var file = files[0]
|
|
|
|
if (this.$preview.length > 0 && (typeof file.type !== "undefined" ? file.type.match(/^image\/(gif|png|jpeg)$/) : file.name.match(/\.(gif|png|jpe?g)$/i)) && typeof FileReader !== "undefined") {
|
|
var reader = new FileReader()
|
|
var preview = this.$preview
|
|
var element = this.$element
|
|
|
|
reader.onload = function(re) {
|
|
var $img = $('<img>')
|
|
$img[0].src = re.target.result
|
|
files[0].result = re.target.result
|
|
|
|
element.find('.fileinput-filename').text(file.name)
|
|
|
|
// if parent has max-height, using `(max-)height: 100%` on child doesn't take padding and border into account
|
|
if (preview.css('max-height') != 'none') $img.css('max-height', parseInt(preview.css('max-height'), 10) - parseInt(preview.css('padding-top'), 10) - parseInt(preview.css('padding-bottom'), 10) - parseInt(preview.css('border-top'), 10) - parseInt(preview.css('border-bottom'), 10))
|
|
|
|
preview.html($img)
|
|
element.addClass('fileinput-exists').removeClass('fileinput-new')
|
|
|
|
element.trigger('change.bs.fileinput', files)
|
|
}
|
|
|
|
reader.readAsDataURL(file)
|
|
} else {
|
|
this.$element.find('.fileinput-filename').text(file.name)
|
|
this.$preview.text(file.name)
|
|
|
|
this.$element.addClass('fileinput-exists').removeClass('fileinput-new')
|
|
|
|
this.$element.trigger('change.bs.fileinput')
|
|
}
|
|
},
|
|
|
|
Fileinput.prototype.clear = function(e) {
|
|
if (e) e.preventDefault()
|
|
|
|
this.$hidden.val('')
|
|
this.$hidden.attr('name', this.name)
|
|
this.$input.attr('name', '')
|
|
|
|
//ie8+ doesn't support changing the value of input with type=file so clone instead
|
|
if (isIE) {
|
|
var inputClone = this.$input.clone(true);
|
|
this.$input.after(inputClone);
|
|
this.$input.remove();
|
|
this.$input = inputClone;
|
|
} else {
|
|
this.$input.val('')
|
|
}
|
|
|
|
this.$preview.html('')
|
|
this.$element.find('.fileinput-filename').text('')
|
|
this.$element.addClass('fileinput-new').removeClass('fileinput-exists')
|
|
|
|
if (e !== undefined) {
|
|
this.$input.trigger('change')
|
|
this.$element.trigger('clear.bs.fileinput')
|
|
}
|
|
},
|
|
|
|
Fileinput.prototype.reset = function() {
|
|
this.clear()
|
|
|
|
this.$hidden.val(this.original.hiddenVal)
|
|
this.$preview.html(this.original.preview)
|
|
this.$element.find('.fileinput-filename').text('')
|
|
|
|
if (this.original.exists) this.$element.addClass('fileinput-exists').removeClass('fileinput-new')
|
|
else this.$element.addClass('fileinput-new').removeClass('fileinput-exists')
|
|
|
|
this.$element.trigger('reset.bs.fileinput')
|
|
},
|
|
|
|
Fileinput.prototype.trigger = function(e) {
|
|
this.$input.trigger('click')
|
|
e.preventDefault()
|
|
}
|
|
|
|
|
|
// FILEUPLOAD PLUGIN DEFINITION
|
|
// ===========================
|
|
|
|
var old = $.fn.fileinput
|
|
|
|
$.fn.fileinput = function (options) {
|
|
return this.each(function () {
|
|
var $this = $(this),
|
|
data = $this.data('bs.fileinput')
|
|
if (!data) $this.data('bs.fileinput', (data = new Fileinput(this, options)))
|
|
if (typeof options == 'string') data[options]()
|
|
})
|
|
}
|
|
|
|
$.fn.fileinput.Constructor = Fileinput
|
|
|
|
|
|
// FILEINPUT NO CONFLICT
|
|
// ====================
|
|
|
|
$.fn.fileinput.noConflict = function () {
|
|
$.fn.fileinput = old
|
|
return this
|
|
}
|
|
|
|
|
|
// FILEUPLOAD DATA-API
|
|
// ==================
|
|
|
|
$(document).on('click.fileinput.data-api', '[data-provides="fileinput"]', function (e) {
|
|
var $this = $(this)
|
|
if ($this.data('bs.fileinput')) return
|
|
$this.fileinput($this.data())
|
|
|
|
var $target = $(e.target).closest('[data-dismiss="fileinput"],[data-trigger="fileinput"]');
|
|
if ($target.length > 0) {
|
|
e.preventDefault()
|
|
$target.trigger('click.bs.fileinput')
|
|
}
|
|
})
|
|
|
|
}(window.jQuery);
|
|
|
|
/*!
|
|
* jQuery Form Plugin
|
|
* version: 3.46.0-2013.11.21
|
|
* Requires jQuery v1.5 or later
|
|
* Copyright (c) 2013 M. Alsup
|
|
* Examples and documentation at: http://malsup.com/jquery/form/
|
|
* Project repository: https://github.com/malsup/form
|
|
* Dual licensed under the MIT and GPL licenses.
|
|
* https://github.com/malsup/form#copyright-and-license
|
|
*/
|
|
/*global ActiveXObject */
|
|
|
|
// AMD support
|
|
(function (factory) {
|
|
if (typeof define === 'function' && define.amd) {
|
|
// using AMD; register as anon module
|
|
define(['jquery'], factory);
|
|
} else {
|
|
// no AMD; invoke directly
|
|
factory( (typeof(jQuery) != 'undefined') ? jQuery : window.Zepto );
|
|
}
|
|
}
|
|
|
|
(function($) {
|
|
"use strict";
|
|
|
|
/*
|
|
Usage Note:
|
|
-----------
|
|
Do not use both ajaxSubmit and ajaxForm on the same form. These
|
|
functions are mutually exclusive. Use ajaxSubmit if you want
|
|
to bind your own submit handler to the form. For example,
|
|
|
|
$(document).ready(function() {
|
|
$('#myForm').on('submit', function(e) {
|
|
e.preventDefault(); // <-- important
|
|
$(this).ajaxSubmit({
|
|
target: '#output'
|
|
});
|
|
});
|
|
});
|
|
|
|
Use ajaxForm when you want the plugin to manage all the event binding
|
|
for you. For example,
|
|
|
|
$(document).ready(function() {
|
|
$('#myForm').ajaxForm({
|
|
target: '#output'
|
|
});
|
|
});
|
|
|
|
You can also use ajaxForm with delegation (requires jQuery v1.7+), so the
|
|
form does not have to exist when you invoke ajaxForm:
|
|
|
|
$('#myForm').ajaxForm({
|
|
delegation: true,
|
|
target: '#output'
|
|
});
|
|
|
|
When using ajaxForm, the ajaxSubmit function will be invoked for you
|
|
at the appropriate time.
|
|
*/
|
|
|
|
/**
|
|
* Feature detection
|
|
*/
|
|
var feature = {};
|
|
feature.fileapi = $("<input type='file'/>").get(0).files !== undefined;
|
|
feature.formdata = window.FormData !== undefined;
|
|
|
|
var hasProp = !!$.fn.prop;
|
|
|
|
// attr2 uses prop when it can but checks the return type for
|
|
// an expected string. this accounts for the case where a form
|
|
// contains inputs with names like "action" or "method"; in those
|
|
// cases "prop" returns the element
|
|
$.fn.attr2 = function() {
|
|
if ( ! hasProp )
|
|
return this.attr.apply(this, arguments);
|
|
var val = this.prop.apply(this, arguments);
|
|
if ( ( val && val.jquery ) || typeof val === 'string' )
|
|
return val;
|
|
return this.attr.apply(this, arguments);
|
|
};
|
|
|
|
/**
|
|
* ajaxSubmit() provides a mechanism for immediately submitting
|
|
* an HTML form using AJAX.
|
|
*/
|
|
$.fn.ajaxSubmit = function(options) {
|
|
/*jshint scripturl:true */
|
|
|
|
// fast fail if nothing selected (http://dev.jquery.com/ticket/2752)
|
|
if (!this.length) {
|
|
log('ajaxSubmit: skipping submit process - no element selected');
|
|
return this;
|
|
}
|
|
|
|
var method, action, url, $form = this;
|
|
|
|
if (typeof options == 'function') {
|
|
options = { success: options };
|
|
}
|
|
else if ( options === undefined ) {
|
|
options = {};
|
|
}
|
|
|
|
method = options.type || this.attr2('method');
|
|
action = options.url || this.attr2('action');
|
|
|
|
url = (typeof action === 'string') ? $.trim(action) : '';
|
|
url = url || window.location.href || '';
|
|
if (url) {
|
|
// clean url (don't include hash vaue)
|
|
url = (url.match(/^([^#]+)/)||[])[1];
|
|
}
|
|
|
|
options = $.extend(true, {
|
|
url: url,
|
|
success: $.ajaxSettings.success,
|
|
type: method || $.ajaxSettings.type,
|
|
iframeSrc: /^https/i.test(window.location.href || '') ? 'javascript:false' : 'about:blank'
|
|
}, options);
|
|
|
|
// hook for manipulating the form data before it is extracted;
|
|
// convenient for use with rich editors like tinyMCE or FCKEditor
|
|
var veto = {};
|
|
this.trigger('form-pre-serialize', [this, options, veto]);
|
|
if (veto.veto) {
|
|
log('ajaxSubmit: submit vetoed via form-pre-serialize trigger');
|
|
return this;
|
|
}
|
|
|
|
// provide opportunity to alter form data before it is serialized
|
|
if (options.beforeSerialize && options.beforeSerialize(this, options) === false) {
|
|
log('ajaxSubmit: submit aborted via beforeSerialize callback');
|
|
return this;
|
|
}
|
|
|
|
var traditional = options.traditional;
|
|
if ( traditional === undefined ) {
|
|
traditional = $.ajaxSettings.traditional;
|
|
}
|
|
|
|
var elements = [];
|
|
var qx, a = this.formToArray(options.semantic, elements);
|
|
if (options.data) {
|
|
options.extraData = options.data;
|
|
qx = $.param(options.data, traditional);
|
|
}
|
|
|
|
// give pre-submit callback an opportunity to abort the submit
|
|
if (options.beforeSubmit && options.beforeSubmit(a, this, options) === false) {
|
|
log('ajaxSubmit: submit aborted via beforeSubmit callback');
|
|
return this;
|
|
}
|
|
|
|
// fire vetoable 'validate' event
|
|
this.trigger('form-submit-validate', [a, this, options, veto]);
|
|
if (veto.veto) {
|
|
log('ajaxSubmit: submit vetoed via form-submit-validate trigger');
|
|
return this;
|
|
}
|
|
|
|
var q = $.param(a, traditional);
|
|
if (qx) {
|
|
q = ( q ? (q + '&' + qx) : qx );
|
|
}
|
|
if (options.type.toUpperCase() == 'GET') {
|
|
options.url += (options.url.indexOf('?') >= 0 ? '&' : '?') + q;
|
|
options.data = null; // data is null for 'get'
|
|
}
|
|
else {
|
|
options.data = q; // data is the query string for 'post'
|
|
}
|
|
|
|
var callbacks = [];
|
|
if (options.resetForm) {
|
|
callbacks.push(function() { $form.resetForm(); });
|
|
}
|
|
if (options.clearForm) {
|
|
callbacks.push(function() { $form.clearForm(options.includeHidden); });
|
|
}
|
|
|
|
// perform a load on the target only if dataType is not provided
|
|
if (!options.dataType && options.target) {
|
|
var oldSuccess = options.success || function(){};
|
|
callbacks.push(function(data) {
|
|
var fn = options.replaceTarget ? 'replaceWith' : 'html';
|
|
$(options.target)[fn](data).each(oldSuccess, arguments);
|
|
});
|
|
}
|
|
else if (options.success) {
|
|
callbacks.push(options.success);
|
|
}
|
|
|
|
options.success = function(data, status, xhr) { // jQuery 1.4+ passes xhr as 3rd arg
|
|
var context = options.context || this ; // jQuery 1.4+ supports scope context
|
|
for (var i=0, max=callbacks.length; i < max; i++) {
|
|
callbacks[i].apply(context, [data, status, xhr || $form, $form]);
|
|
}
|
|
};
|
|
|
|
if (options.error) {
|
|
var oldError = options.error;
|
|
options.error = function(xhr, status, error) {
|
|
var context = options.context || this;
|
|
oldError.apply(context, [xhr, status, error, $form]);
|
|
};
|
|
}
|
|
|
|
if (options.complete) {
|
|
var oldComplete = options.complete;
|
|
options.complete = function(xhr, status) {
|
|
var context = options.context || this;
|
|
oldComplete.apply(context, [xhr, status, $form]);
|
|
};
|
|
}
|
|
|
|
// are there files to upload?
|
|
|
|
// [value] (issue #113), also see comment:
|
|
// https://github.com/malsup/form/commit/588306aedba1de01388032d5f42a60159eea9228#commitcomment-2180219
|
|
var fileInputs = $('input[type=file]:enabled', this).filter(function() { return $(this).val() !== ''; });
|
|
|
|
var hasFileInputs = fileInputs.length > 0;
|
|
var mp = 'multipart/form-data';
|
|
var multipart = ($form.attr('enctype') == mp || $form.attr('encoding') == mp);
|
|
|
|
var fileAPI = feature.fileapi && feature.formdata;
|
|
log("fileAPI :" + fileAPI);
|
|
var shouldUseFrame = (hasFileInputs || multipart) && !fileAPI;
|
|
|
|
var jqxhr;
|
|
|
|
// options.iframe allows user to force iframe mode
|
|
// 06-NOV-09: now defaulting to iframe mode if file input is detected
|
|
if (options.iframe !== false && (options.iframe || shouldUseFrame)) {
|
|
// hack to fix Safari hang (thanks to Tim Molendijk for this)
|
|
// see: http://groups.google.com/group/jquery-dev/browse_thread/thread/36395b7ab510dd5d
|
|
if (options.closeKeepAlive) {
|
|
$.get(options.closeKeepAlive, function() {
|
|
jqxhr = fileUploadIframe(a);
|
|
});
|
|
}
|
|
else {
|
|
jqxhr = fileUploadIframe(a);
|
|
}
|
|
}
|
|
else if ((hasFileInputs || multipart) && fileAPI) {
|
|
jqxhr = fileUploadXhr(a);
|
|
}
|
|
else {
|
|
jqxhr = $.ajax(options);
|
|
}
|
|
|
|
$form.removeData('jqxhr').data('jqxhr', jqxhr);
|
|
|
|
// clear element array
|
|
for (var k=0; k < elements.length; k++)
|
|
elements[k] = null;
|
|
|
|
// fire 'notify' event
|
|
this.trigger('form-submit-notify', [this, options]);
|
|
return this;
|
|
|
|
// utility fn for deep serialization
|
|
function deepSerialize(extraData){
|
|
var serialized = $.param(extraData, options.traditional).split('&');
|
|
var len = serialized.length;
|
|
var result = [];
|
|
var i, part;
|
|
for (i=0; i < len; i++) {
|
|
// #252; undo param space replacement
|
|
serialized[i] = serialized[i].replace(/\+/g,' ');
|
|
part = serialized[i].split('=');
|
|
// #278; use array instead of object storage, favoring array serializations
|
|
result.push([decodeURIComponent(part[0]), decodeURIComponent(part[1])]);
|
|
}
|
|
return result;
|
|
}
|
|
|
|
// XMLHttpRequest Level 2 file uploads (big hat tip to francois2metz)
|
|
function fileUploadXhr(a) {
|
|
var formdata = new FormData();
|
|
|
|
for (var i=0; i < a.length; i++) {
|
|
formdata.append(a[i].name, a[i].value);
|
|
}
|
|
|
|
if (options.extraData) {
|
|
var serializedData = deepSerialize(options.extraData);
|
|
for (i=0; i < serializedData.length; i++)
|
|
if (serializedData[i])
|
|
formdata.append(serializedData[i][0], serializedData[i][1]);
|
|
}
|
|
|
|
options.data = null;
|
|
|
|
var s = $.extend(true, {}, $.ajaxSettings, options, {
|
|
contentType: false,
|
|
processData: false,
|
|
cache: false,
|
|
type: method || 'POST'
|
|
});
|
|
|
|
if (options.uploadProgress) {
|
|
// workaround because jqXHR does not expose upload property
|
|
s.xhr = function() {
|
|
var xhr = $.ajaxSettings.xhr();
|
|
if (xhr.upload) {
|
|
xhr.upload.addEventListener('progress', function(event) {
|
|
var percent = 0;
|
|
var position = event.loaded || event.position; /*event.position is deprecated*/
|
|
var total = event.total;
|
|
if (event.lengthComputable) {
|
|
percent = Math.ceil(position / total * 100);
|
|
}
|
|
options.uploadProgress(event, position, total, percent);
|
|
}, false);
|
|
}
|
|
return xhr;
|
|
};
|
|
}
|
|
|
|
s.data = null;
|
|
var beforeSend = s.beforeSend;
|
|
s.beforeSend = function(xhr, o) {
|
|
//Send FormData() provided by user
|
|
if (options.formData)
|
|
o.data = options.formData;
|
|
else
|
|
o.data = formdata;
|
|
if(beforeSend)
|
|
beforeSend.call(this, xhr, o);
|
|
};
|
|
return $.ajax(s);
|
|
}
|
|
|
|
// private function for handling file uploads (hat tip to YAHOO!)
|
|
function fileUploadIframe(a) {
|
|
var form = $form[0], el, i, s, g, id, $io, io, xhr, sub, n, timedOut, timeoutHandle;
|
|
var deferred = $.Deferred();
|
|
|
|
// #341
|
|
deferred.abort = function(status) {
|
|
xhr.abort(status);
|
|
};
|
|
|
|
if (a) {
|
|
// ensure that every serialized input is still enabled
|
|
for (i=0; i < elements.length; i++) {
|
|
el = $(elements[i]);
|
|
if ( hasProp )
|
|
el.prop('disabled', false);
|
|
else
|
|
el.removeAttr('disabled');
|
|
}
|
|
}
|
|
|
|
s = $.extend(true, {}, $.ajaxSettings, options);
|
|
s.context = s.context || s;
|
|
id = 'jqFormIO' + (new Date().getTime());
|
|
if (s.iframeTarget) {
|
|
$io = $(s.iframeTarget);
|
|
n = $io.attr2('name');
|
|
if (!n)
|
|
$io.attr2('name', id);
|
|
else
|
|
id = n;
|
|
}
|
|
else {
|
|
$io = $('<iframe name="' + id + '" src="'+ s.iframeSrc +'" />');
|
|
$io.css({ position: 'absolute', top: '-1000px', left: '-1000px' });
|
|
}
|
|
io = $io[0];
|
|
|
|
|
|
xhr = { // mock object
|
|
aborted: 0,
|
|
responseText: null,
|
|
responseXML: null,
|
|
status: 0,
|
|
statusText: 'n/a',
|
|
getAllResponseHeaders: function() {},
|
|
getResponseHeader: function() {},
|
|
setRequestHeader: function() {},
|
|
abort: function(status) {
|
|
var e = (status === 'timeout' ? 'timeout' : 'aborted');
|
|
log('aborting upload... ' + e);
|
|
this.aborted = 1;
|
|
|
|
try { // #214, #257
|
|
if (io.contentWindow.document.execCommand) {
|
|
io.contentWindow.document.execCommand('Stop');
|
|
}
|
|
}
|
|
catch(ignore) {}
|
|
|
|
$io.attr('src', s.iframeSrc); // abort op in progress
|
|
xhr.error = e;
|
|
if (s.error)
|
|
s.error.call(s.context, xhr, e, status);
|
|
if (g)
|
|
$.event.trigger("ajaxError", [xhr, s, e]);
|
|
if (s.complete)
|
|
s.complete.call(s.context, xhr, e);
|
|
}
|
|
};
|
|
|
|
g = s.global;
|
|
// trigger ajax global events so that activity/block indicators work like normal
|
|
if (g && 0 === $.active++) {
|
|
$.event.trigger("ajaxStart");
|
|
}
|
|
if (g) {
|
|
$.event.trigger("ajaxSend", [xhr, s]);
|
|
}
|
|
|
|
if (s.beforeSend && s.beforeSend.call(s.context, xhr, s) === false) {
|
|
if (s.global) {
|
|
$.active--;
|
|
}
|
|
deferred.reject();
|
|
return deferred;
|
|
}
|
|
if (xhr.aborted) {
|
|
deferred.reject();
|
|
return deferred;
|
|
}
|
|
|
|
// add submitting element to data if we know it
|
|
sub = form.clk;
|
|
if (sub) {
|
|
n = sub.name;
|
|
if (n && !sub.disabled) {
|
|
s.extraData = s.extraData || {};
|
|
s.extraData[n] = sub.value;
|
|
if (sub.type == "image") {
|
|
s.extraData[n+'.x'] = form.clk_x;
|
|
s.extraData[n+'.y'] = form.clk_y;
|
|
}
|
|
}
|
|
}
|
|
|
|
var CLIENT_TIMEOUT_ABORT = 1;
|
|
var SERVER_ABORT = 2;
|
|
|
|
function getDoc(frame) {
|
|
/* it looks like contentWindow or contentDocument do not
|
|
* carry the protocol property in ie8, when running under ssl
|
|
* frame.document is the only valid response document, since
|
|
* the protocol is know but not on the other two objects. strange?
|
|
* "Same origin policy" http://en.wikipedia.org/wiki/Same_origin_policy
|
|
*/
|
|
|
|
var doc = null;
|
|
|
|
// IE8 cascading access check
|
|
try {
|
|
if (frame.contentWindow) {
|
|
doc = frame.contentWindow.document;
|
|
}
|
|
} catch(err) {
|
|
// IE8 access denied under ssl & missing protocol
|
|
log('cannot get iframe.contentWindow document: ' + err);
|
|
}
|
|
|
|
if (doc) { // successful getting content
|
|
return doc;
|
|
}
|
|
|
|
try { // simply checking may throw in ie8 under ssl or mismatched protocol
|
|
doc = frame.contentDocument ? frame.contentDocument : frame.document;
|
|
} catch(err) {
|
|
// last attempt
|
|
log('cannot get iframe.contentDocument: ' + err);
|
|
doc = frame.document;
|
|
}
|
|
return doc;
|
|
}
|
|
|
|
// Rails CSRF hack (thanks to Yvan Barthelemy)
|
|
var csrf_token = $('meta[name=csrf-token]').attr('content');
|
|
var csrf_param = $('meta[name=csrf-param]').attr('content');
|
|
if (csrf_param && csrf_token) {
|
|
s.extraData = s.extraData || {};
|
|
s.extraData[csrf_param] = csrf_token;
|
|
}
|
|
|
|
// take a breath so that pending repaints get some cpu time before the upload starts
|
|
function doSubmit() {
|
|
// make sure form attrs are set
|
|
var t = $form.attr2('target'), a = $form.attr2('action');
|
|
|
|
// update form attrs in IE friendly way
|
|
form.setAttribute('target',id);
|
|
if (!method || /post/i.test(method) ) {
|
|
form.setAttribute('method', 'POST');
|
|
}
|
|
if (a != s.url) {
|
|
form.setAttribute('action', s.url);
|
|
}
|
|
|
|
// ie borks in some cases when setting encoding
|
|
if (! s.skipEncodingOverride && (!method || /post/i.test(method))) {
|
|
$form.attr({
|
|
encoding: 'multipart/form-data',
|
|
enctype: 'multipart/form-data'
|
|
});
|
|
}
|
|
|
|
// support timout
|
|
if (s.timeout) {
|
|
timeoutHandle = setTimeout(function() { timedOut = true; cb(CLIENT_TIMEOUT_ABORT); }, s.timeout);
|
|
}
|
|
|
|
// look for server aborts
|
|
function checkState() {
|
|
try {
|
|
var state = getDoc(io).readyState;
|
|
log('state = ' + state);
|
|
if (state && state.toLowerCase() == 'uninitialized')
|
|
setTimeout(checkState,50);
|
|
}
|
|
catch(e) {
|
|
log('Server abort: ' , e, ' (', e.name, ')');
|
|
cb(SERVER_ABORT);
|
|
if (timeoutHandle)
|
|
clearTimeout(timeoutHandle);
|
|
timeoutHandle = undefined;
|
|
}
|
|
}
|
|
|
|
// add "extra" data to form if provided in options
|
|
var extraInputs = [];
|
|
try {
|
|
if (s.extraData) {
|
|
for (var n in s.extraData) {
|
|
if (s.extraData.hasOwnProperty(n)) {
|
|
// if using the $.param format that allows for multiple values with the same name
|
|
if($.isPlainObject(s.extraData[n]) && s.extraData[n].hasOwnProperty('name') && s.extraData[n].hasOwnProperty('value')) {
|
|
extraInputs.push(
|
|
$('<input type="hidden" name="'+s.extraData[n].name+'">').val(s.extraData[n].value)
|
|
.appendTo(form)[0]);
|
|
} else {
|
|
extraInputs.push(
|
|
$('<input type="hidden" name="'+n+'">').val(s.extraData[n])
|
|
.appendTo(form)[0]);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if (!s.iframeTarget) {
|
|
// add iframe to doc and submit the form
|
|
$io.appendTo('body');
|
|
}
|
|
if (io.attachEvent)
|
|
io.attachEvent('onload', cb);
|
|
else
|
|
io.addEventListener('load', cb, false);
|
|
setTimeout(checkState,15);
|
|
|
|
try {
|
|
form.submit();
|
|
} catch(err) {
|
|
// just in case form has element with name/id of 'submit'
|
|
var submitFn = document.createElement('form').submit;
|
|
submitFn.apply(form);
|
|
}
|
|
}
|
|
finally {
|
|
// reset attrs and remove "extra" input elements
|
|
form.setAttribute('action',a);
|
|
if(t) {
|
|
form.setAttribute('target', t);
|
|
} else {
|
|
$form.removeAttr('target');
|
|
}
|
|
$(extraInputs).remove();
|
|
}
|
|
}
|
|
|
|
if (s.forceSync) {
|
|
doSubmit();
|
|
}
|
|
else {
|
|
setTimeout(doSubmit, 10); // this lets dom updates render
|
|
}
|
|
|
|
var data, doc, domCheckCount = 50, callbackProcessed;
|
|
|
|
function cb(e) {
|
|
if (xhr.aborted || callbackProcessed) {
|
|
return;
|
|
}
|
|
|
|
doc = getDoc(io);
|
|
if(!doc) {
|
|
log('cannot access response document');
|
|
e = SERVER_ABORT;
|
|
}
|
|
if (e === CLIENT_TIMEOUT_ABORT && xhr) {
|
|
xhr.abort('timeout');
|
|
deferred.reject(xhr, 'timeout');
|
|
return;
|
|
}
|
|
else if (e == SERVER_ABORT && xhr) {
|
|
xhr.abort('server abort');
|
|
deferred.reject(xhr, 'error', 'server abort');
|
|
return;
|
|
}
|
|
|
|
if (!doc || doc.location.href == s.iframeSrc) {
|
|
// response not received yet
|
|
if (!timedOut)
|
|
return;
|
|
}
|
|
if (io.detachEvent)
|
|
io.detachEvent('onload', cb);
|
|
else
|
|
io.removeEventListener('load', cb, false);
|
|
|
|
var status = 'success', errMsg;
|
|
try {
|
|
if (timedOut) {
|
|
throw 'timeout';
|
|
}
|
|
|
|
var isXml = s.dataType == 'xml' || doc.XMLDocument || $.isXMLDoc(doc);
|
|
log('isXml='+isXml);
|
|
if (!isXml && window.opera && (doc.body === null || !doc.body.innerHTML)) {
|
|
if (--domCheckCount) {
|
|
// in some browsers (Opera) the iframe DOM is not always traversable when
|
|
// the onload callback fires, so we loop a bit to accommodate
|
|
log('requeing onLoad callback, DOM not available');
|
|
setTimeout(cb, 250);
|
|
return;
|
|
}
|
|
// let this fall through because server response could be an empty document
|
|
//log('Could not access iframe DOM after mutiple tries.');
|
|
//throw 'DOMException: not available';
|
|
}
|
|
|
|
//log('response detected');
|
|
var docRoot = doc.body ? doc.body : doc.documentElement;
|
|
xhr.responseText = docRoot ? docRoot.innerHTML : null;
|
|
xhr.responseXML = doc.XMLDocument ? doc.XMLDocument : doc;
|
|
if (isXml)
|
|
s.dataType = 'xml';
|
|
xhr.getResponseHeader = function(header){
|
|
var headers = {'content-type': s.dataType};
|
|
return headers[header.toLowerCase()];
|
|
};
|
|
// support for XHR 'status' & 'statusText' emulation :
|
|
if (docRoot) {
|
|
xhr.status = Number( docRoot.getAttribute('status') ) || xhr.status;
|
|
xhr.statusText = docRoot.getAttribute('statusText') || xhr.statusText;
|
|
}
|
|
|
|
var dt = (s.dataType || '').toLowerCase();
|
|
var scr = /(json|script|text)/.test(dt);
|
|
if (scr || s.textarea) {
|
|
// see if user embedded response in textarea
|
|
var ta = doc.getElementsByTagName('textarea')[0];
|
|
if (ta) {
|
|
xhr.responseText = ta.value;
|
|
// support for XHR 'status' & 'statusText' emulation :
|
|
xhr.status = Number( ta.getAttribute('status') ) || xhr.status;
|
|
xhr.statusText = ta.getAttribute('statusText') || xhr.statusText;
|
|
}
|
|
else if (scr) {
|
|
// account for browsers injecting pre around json response
|
|
var pre = doc.getElementsByTagName('pre')[0];
|
|
var b = doc.getElementsByTagName('body')[0];
|
|
if (pre) {
|
|
xhr.responseText = pre.textContent ? pre.textContent : pre.innerText;
|
|
}
|
|
else if (b) {
|
|
xhr.responseText = b.textContent ? b.textContent : b.innerText;
|
|
}
|
|
}
|
|
}
|
|
else if (dt == 'xml' && !xhr.responseXML && xhr.responseText) {
|
|
xhr.responseXML = toXml(xhr.responseText);
|
|
}
|
|
|
|
try {
|
|
data = httpData(xhr, dt, s);
|
|
}
|
|
catch (err) {
|
|
status = 'parsererror';
|
|
xhr.error = errMsg = (err || status);
|
|
}
|
|
}
|
|
catch (err) {
|
|
log('error caught: ',err);
|
|
status = 'error';
|
|
xhr.error = errMsg = (err || status);
|
|
}
|
|
|
|
if (xhr.aborted) {
|
|
log('upload aborted');
|
|
status = null;
|
|
}
|
|
|
|
if (xhr.status) { // we've set xhr.status
|
|
status = (xhr.status >= 200 && xhr.status < 300 || xhr.status === 304) ? 'success' : 'error';
|
|
}
|
|
|
|
// ordering of these callbacks/triggers is odd, but that's how $.ajax does it
|
|
if (status === 'success') {
|
|
if (s.success)
|
|
s.success.call(s.context, data, 'success', xhr);
|
|
deferred.resolve(xhr.responseText, 'success', xhr);
|
|
if (g)
|
|
$.event.trigger("ajaxSuccess", [xhr, s]);
|
|
}
|
|
else if (status) {
|
|
if (errMsg === undefined)
|
|
errMsg = xhr.statusText;
|
|
if (s.error)
|
|
s.error.call(s.context, xhr, status, errMsg);
|
|
deferred.reject(xhr, 'error', errMsg);
|
|
if (g)
|
|
$.event.trigger("ajaxError", [xhr, s, errMsg]);
|
|
}
|
|
|
|
if (g)
|
|
$.event.trigger("ajaxComplete", [xhr, s]);
|
|
|
|
if (g && ! --$.active) {
|
|
$.event.trigger("ajaxStop");
|
|
}
|
|
|
|
if (s.complete)
|
|
s.complete.call(s.context, xhr, status);
|
|
|
|
callbackProcessed = true;
|
|
if (s.timeout)
|
|
clearTimeout(timeoutHandle);
|
|
|
|
// clean up
|
|
setTimeout(function() {
|
|
if (!s.iframeTarget)
|
|
$io.remove();
|
|
else //adding else to clean up existing iframe response.
|
|
$io.attr('src', s.iframeSrc);
|
|
xhr.responseXML = null;
|
|
}, 100);
|
|
}
|
|
|
|
var toXml = $.parseXML || function(s, doc) { // use parseXML if available (jQuery 1.5+)
|
|
if (window.ActiveXObject) {
|
|
doc = new ActiveXObject('Microsoft.XMLDOM');
|
|
doc.async = 'false';
|
|
doc.loadXML(s);
|
|
}
|
|
else {
|
|
doc = (new DOMParser()).parseFromString(s, 'text/xml');
|
|
}
|
|
return (doc && doc.documentElement && doc.documentElement.nodeName != 'parsererror') ? doc : null;
|
|
};
|
|
var parseJSON = $.parseJSON || function(s) {
|
|
/*jslint evil:true */
|
|
return window['eval']('(' + s + ')');
|
|
};
|
|
|
|
var httpData = function( xhr, type, s ) { // mostly lifted from jq1.4.4
|
|
|
|
var ct = xhr.getResponseHeader('content-type') || '',
|
|
xml = type === 'xml' || !type && ct.indexOf('xml') >= 0,
|
|
data = xml ? xhr.responseXML : xhr.responseText;
|
|
|
|
if (xml && data.documentElement.nodeName === 'parsererror') {
|
|
if ($.error)
|
|
$.error('parsererror');
|
|
}
|
|
if (s && s.dataFilter) {
|
|
data = s.dataFilter(data, type);
|
|
}
|
|
if (typeof data === 'string') {
|
|
if (type === 'json' || !type && ct.indexOf('json') >= 0) {
|
|
data = parseJSON(data);
|
|
} else if (type === "script" || !type && ct.indexOf("javascript") >= 0) {
|
|
$.globalEval(data);
|
|
}
|
|
}
|
|
return data;
|
|
};
|
|
|
|
return deferred;
|
|
}
|
|
};
|
|
|
|
/**
|
|
* ajaxForm() provides a mechanism for fully automating form submission.
|
|
*
|
|
* The advantages of using this method instead of ajaxSubmit() are:
|
|
*
|
|
* 1: This method will include coordinates for <input type="image" /> elements (if the element
|
|
* is used to submit the form).
|
|
* 2. This method will include the submit element's name/value data (for the element that was
|
|
* used to submit the form).
|
|
* 3. This method binds the submit() method to the form for you.
|
|
*
|
|
* The options argument for ajaxForm works exactly as it does for ajaxSubmit. ajaxForm merely
|
|
* passes the options argument along after properly binding events for submit elements and
|
|
* the form itself.
|
|
*/
|
|
$.fn.ajaxForm = function(options) {
|
|
options = options || {};
|
|
options.delegation = options.delegation && $.isFunction($.fn.on);
|
|
|
|
// in jQuery 1.3+ we can fix mistakes with the ready state
|
|
if (!options.delegation && this.length === 0) {
|
|
var o = { s: this.selector, c: this.context };
|
|
if (!$.isReady && o.s) {
|
|
log('DOM not ready, queuing ajaxForm');
|
|
$(function() {
|
|
$(o.s,o.c).ajaxForm(options);
|
|
});
|
|
return this;
|
|
}
|
|
// is your DOM ready? http://docs.jquery.com/Tutorials:Introducing_$(document).ready()
|
|
log('terminating; zero elements found by selector' + ($.isReady ? '' : ' (DOM not ready)'));
|
|
return this;
|
|
}
|
|
|
|
if ( options.delegation ) {
|
|
$(document)
|
|
.off('submit.form-plugin', this.selector, doAjaxSubmit)
|
|
.off('click.form-plugin', this.selector, captureSubmittingElement)
|
|
.on('submit.form-plugin', this.selector, options, doAjaxSubmit)
|
|
.on('click.form-plugin', this.selector, options, captureSubmittingElement);
|
|
return this;
|
|
}
|
|
|
|
return this.ajaxFormUnbind()
|
|
.bind('submit.form-plugin', options, doAjaxSubmit)
|
|
.bind('click.form-plugin', options, captureSubmittingElement);
|
|
};
|
|
|
|
// private event handlers
|
|
function doAjaxSubmit(e) {
|
|
/*jshint validthis:true */
|
|
var options = e.data;
|
|
if (!e.isDefaultPrevented()) { // if event has been canceled, don't proceed
|
|
e.preventDefault();
|
|
$(e.target).ajaxSubmit(options); // #365
|
|
}
|
|
}
|
|
|
|
function captureSubmittingElement(e) {
|
|
/*jshint validthis:true */
|
|
var target = e.target;
|
|
var $el = $(target);
|
|
if (!($el.is("[type=submit],[type=image]"))) {
|
|
// is this a child element of the submit el? (ex: a span within a button)
|
|
var t = $el.closest('[type=submit]');
|
|
if (t.length === 0) {
|
|
return;
|
|
}
|
|
target = t[0];
|
|
}
|
|
var form = this;
|
|
form.clk = target;
|
|
if (target.type == 'image') {
|
|
if (e.offsetX !== undefined) {
|
|
form.clk_x = e.offsetX;
|
|
form.clk_y = e.offsetY;
|
|
} else if (typeof $.fn.offset == 'function') {
|
|
var offset = $el.offset();
|
|
form.clk_x = e.pageX - offset.left;
|
|
form.clk_y = e.pageY - offset.top;
|
|
} else {
|
|
form.clk_x = e.pageX - target.offsetLeft;
|
|
form.clk_y = e.pageY - target.offsetTop;
|
|
}
|
|
}
|
|
// clear form vars
|
|
setTimeout(function() { form.clk = form.clk_x = form.clk_y = null; }, 100);
|
|
}
|
|
|
|
|
|
// ajaxFormUnbind unbinds the event handlers that were bound by ajaxForm
|
|
$.fn.ajaxFormUnbind = function() {
|
|
return this.unbind('submit.form-plugin click.form-plugin');
|
|
};
|
|
|
|
/**
|
|
* formToArray() gathers form element data into an array of objects that can
|
|
* be passed to any of the following ajax functions: $.get, $.post, or load.
|
|
* Each object in the array has both a 'name' and 'value' property. An example of
|
|
* an array for a simple login form might be:
|
|
*
|
|
* [ { name: 'username', value: 'jresig' }, { name: 'password', value: 'secret' } ]
|
|
*
|
|
* It is this array that is passed to pre-submit callback functions provided to the
|
|
* ajaxSubmit() and ajaxForm() methods.
|
|
*/
|
|
$.fn.formToArray = function(semantic, elements) {
|
|
var a = [];
|
|
if (this.length === 0) {
|
|
return a;
|
|
}
|
|
|
|
var form = this[0];
|
|
var els = semantic ? form.getElementsByTagName('*') : form.elements;
|
|
if (!els) {
|
|
return a;
|
|
}
|
|
|
|
var i,j,n,v,el,max,jmax;
|
|
for(i=0, max=els.length; i < max; i++) {
|
|
el = els[i];
|
|
n = el.name;
|
|
if (!n || el.disabled) {
|
|
continue;
|
|
}
|
|
|
|
if (semantic && form.clk && el.type == "image") {
|
|
// handle image inputs on the fly when semantic == true
|
|
if(form.clk == el) {
|
|
a.push({name: n, value: $(el).val(), type: el.type });
|
|
a.push({name: n+'.x', value: form.clk_x}, {name: n+'.y', value: form.clk_y});
|
|
}
|
|
continue;
|
|
}
|
|
|
|
v = $.fieldValue(el, true);
|
|
if (v && v.constructor == Array) {
|
|
if (elements)
|
|
elements.push(el);
|
|
for(j=0, jmax=v.length; j < jmax; j++) {
|
|
a.push({name: n, value: v[j]});
|
|
}
|
|
}
|
|
else if (feature.fileapi && el.type == 'file') {
|
|
if (elements)
|
|
elements.push(el);
|
|
var files = el.files;
|
|
if (files.length) {
|
|
for (j=0; j < files.length; j++) {
|
|
a.push({name: n, value: files[j], type: el.type});
|
|
}
|
|
}
|
|
else {
|
|
// #180
|
|
a.push({ name: n, value: '', type: el.type });
|
|
}
|
|
}
|
|
else if (v !== null && typeof v != 'undefined') {
|
|
if (elements)
|
|
elements.push(el);
|
|
a.push({name: n, value: v, type: el.type, required: el.required});
|
|
}
|
|
}
|
|
|
|
if (!semantic && form.clk) {
|
|
// input type=='image' are not found in elements array! handle it here
|
|
var $input = $(form.clk), input = $input[0];
|
|
n = input.name;
|
|
if (n && !input.disabled && input.type == 'image') {
|
|
a.push({name: n, value: $input.val()});
|
|
a.push({name: n+'.x', value: form.clk_x}, {name: n+'.y', value: form.clk_y});
|
|
}
|
|
}
|
|
return a;
|
|
};
|
|
|
|
/**
|
|
* Serializes form data into a 'submittable' string. This method will return a string
|
|
* in the format: name1=value1&name2=value2
|
|
*/
|
|
$.fn.formSerialize = function(semantic) {
|
|
//hand off to jQuery.param for proper encoding
|
|
return $.param(this.formToArray(semantic));
|
|
};
|
|
|
|
/**
|
|
* Serializes all field elements in the jQuery object into a query string.
|
|
* This method will return a string in the format: name1=value1&name2=value2
|
|
*/
|
|
$.fn.fieldSerialize = function(successful) {
|
|
var a = [];
|
|
this.each(function() {
|
|
var n = this.name;
|
|
if (!n) {
|
|
return;
|
|
}
|
|
var v = $.fieldValue(this, successful);
|
|
if (v && v.constructor == Array) {
|
|
for (var i=0,max=v.length; i < max; i++) {
|
|
a.push({name: n, value: v[i]});
|
|
}
|
|
}
|
|
else if (v !== null && typeof v != 'undefined') {
|
|
a.push({name: this.name, value: v});
|
|
}
|
|
});
|
|
//hand off to jQuery.param for proper encoding
|
|
return $.param(a);
|
|
};
|
|
|
|
/**
|
|
* Returns the value(s) of the element in the matched set. For example, consider the following form:
|
|
*
|
|
* <form><fieldset>
|
|
* <input name="A" type="text" />
|
|
* <input name="A" type="text" />
|
|
* <input name="B" type="checkbox" value="B1" />
|
|
* <input name="B" type="checkbox" value="B2"/>
|
|
* <input name="C" type="radio" value="C1" />
|
|
* <input name="C" type="radio" value="C2" />
|
|
* </fieldset></form>
|
|
*
|
|
* var v = $('input[type=text]').fieldValue();
|
|
* // if no values are entered into the text inputs
|
|
* v == ['','']
|
|
* // if values entered into the text inputs are 'foo' and 'bar'
|
|
* v == ['foo','bar']
|
|
*
|
|
* var v = $('input[type=checkbox]').fieldValue();
|
|
* // if neither checkbox is checked
|
|
* v === undefined
|
|
* // if both checkboxes are checked
|
|
* v == ['B1', 'B2']
|
|
*
|
|
* var v = $('input[type=radio]').fieldValue();
|
|
* // if neither radio is checked
|
|
* v === undefined
|
|
* // if first radio is checked
|
|
* v == ['C1']
|
|
*
|
|
* The successful argument controls whether or not the field element must be 'successful'
|
|
* (per http://www.w3.org/TR/html4/interact/forms.html#successful-controls).
|
|
* The default value of the successful argument is true. If this value is false the value(s)
|
|
* for each element is returned.
|
|
*
|
|
* Note: This method *always* returns an array. If no valid value can be determined the
|
|
* array will be empty, otherwise it will contain one or more values.
|
|
*/
|
|
$.fn.fieldValue = function(successful) {
|
|
for (var val=[], i=0, max=this.length; i < max; i++) {
|
|
var el = this[i];
|
|
var v = $.fieldValue(el, successful);
|
|
if (v === null || typeof v == 'undefined' || (v.constructor == Array && !v.length)) {
|
|
continue;
|
|
}
|
|
if (v.constructor == Array)
|
|
$.merge(val, v);
|
|
else
|
|
val.push(v);
|
|
}
|
|
return val;
|
|
};
|
|
|
|
/**
|
|
* Returns the value of the field element.
|
|
*/
|
|
$.fieldValue = function(el, successful) {
|
|
var n = el.name, t = el.type, tag = el.tagName.toLowerCase();
|
|
if (successful === undefined) {
|
|
successful = true;
|
|
}
|
|
|
|
if (successful && (!n || el.disabled || t == 'reset' || t == 'button' ||
|
|
(t == 'checkbox' || t == 'radio') && !el.checked ||
|
|
(t == 'submit' || t == 'image') && el.form && el.form.clk != el ||
|
|
tag == 'select' && el.selectedIndex == -1)) {
|
|
return null;
|
|
}
|
|
|
|
if (tag == 'select') {
|
|
var index = el.selectedIndex;
|
|
if (index < 0) {
|
|
return null;
|
|
}
|
|
var a = [], ops = el.options;
|
|
var one = (t == 'select-one');
|
|
var max = (one ? index+1 : ops.length);
|
|
for(var i=(one ? index : 0); i < max; i++) {
|
|
var op = ops[i];
|
|
if (op.selected) {
|
|
var v = op.value;
|
|
if (!v) { // extra pain for IE...
|
|
v = (op.attributes && op.attributes['value'] && !(op.attributes['value'].specified)) ? op.text : op.value;
|
|
}
|
|
if (one) {
|
|
return v;
|
|
}
|
|
a.push(v);
|
|
}
|
|
}
|
|
return a;
|
|
}
|
|
return $(el).val();
|
|
};
|
|
|
|
/**
|
|
* Clears the form data. Takes the following actions on the form's input fields:
|
|
* - input text fields will have their 'value' property set to the empty string
|
|
* - select elements will have their 'selectedIndex' property set to -1
|
|
* - checkbox and radio inputs will have their 'checked' property set to false
|
|
* - inputs of type submit, button, reset, and hidden will *not* be effected
|
|
* - button elements will *not* be effected
|
|
*/
|
|
$.fn.clearForm = function(includeHidden) {
|
|
return this.each(function() {
|
|
$('input,select,textarea', this).clearFields(includeHidden);
|
|
});
|
|
};
|
|
|
|
/**
|
|
* Clears the selected form elements.
|
|
*/
|
|
$.fn.clearFields = $.fn.clearInputs = function(includeHidden) {
|
|
var re = /^(?:color|date|datetime|email|month|number|password|range|search|tel|text|time|url|week)$/i; // 'hidden' is not in this list
|
|
return this.each(function() {
|
|
var t = this.type, tag = this.tagName.toLowerCase();
|
|
if (re.test(t) || tag == 'textarea') {
|
|
this.value = '';
|
|
}
|
|
else if (t == 'checkbox' || t == 'radio') {
|
|
this.checked = false;
|
|
}
|
|
else if (tag == 'select') {
|
|
this.selectedIndex = -1;
|
|
}
|
|
else if (t == "file") {
|
|
if (/MSIE/.test(navigator.userAgent)) {
|
|
$(this).replaceWith($(this).clone(true));
|
|
} else {
|
|
$(this).val('');
|
|
}
|
|
}
|
|
else if (includeHidden) {
|
|
// includeHidden can be the value true, or it can be a selector string
|
|
// indicating a special test; for example:
|
|
// $('#myForm').clearForm('.special:hidden')
|
|
// the above would clean hidden inputs that have the class of 'special'
|
|
if ( (includeHidden === true && /hidden/.test(t)) ||
|
|
(typeof includeHidden == 'string' && $(this).is(includeHidden)) )
|
|
this.value = '';
|
|
}
|
|
});
|
|
};
|
|
|
|
/**
|
|
* Resets the form data. Causes all form elements to be reset to their original value.
|
|
*/
|
|
$.fn.resetForm = function() {
|
|
return this.each(function() {
|
|
// guard against an input with the name of 'reset'
|
|
// note that IE reports the reset function as an 'object'
|
|
if (typeof this.reset == 'function' || (typeof this.reset == 'object' && !this.reset.nodeType)) {
|
|
this.reset();
|
|
}
|
|
});
|
|
};
|
|
|
|
/**
|
|
* Enables or disables any matching elements.
|
|
*/
|
|
$.fn.enable = function(b) {
|
|
if (b === undefined) {
|
|
b = true;
|
|
}
|
|
return this.each(function() {
|
|
this.disabled = !b;
|
|
});
|
|
};
|
|
|
|
/**
|
|
* Checks/unchecks any matching checkboxes or radio buttons and
|
|
* selects/deselects and matching option elements.
|
|
*/
|
|
$.fn.selected = function(select) {
|
|
if (select === undefined) {
|
|
select = true;
|
|
}
|
|
return this.each(function() {
|
|
var t = this.type;
|
|
if (t == 'checkbox' || t == 'radio') {
|
|
this.checked = select;
|
|
}
|
|
else if (this.tagName.toLowerCase() == 'option') {
|
|
var $sel = $(this).parent('select');
|
|
if (select && $sel[0] && $sel[0].type == 'select-one') {
|
|
// deselect all other options
|
|
$sel.find('option').selected(false);
|
|
}
|
|
this.selected = select;
|
|
}
|
|
});
|
|
};
|
|
|
|
// expose debug var
|
|
$.fn.ajaxSubmit.debug = false;
|
|
|
|
// helper fn for console logging
|
|
function log() {
|
|
if (!$.fn.ajaxSubmit.debug)
|
|
return;
|
|
var msg = '[jquery.form] ' + Array.prototype.join.call(arguments,'');
|
|
if (window.console && window.console.log) {
|
|
window.console.log(msg);
|
|
}
|
|
else if (window.opera && window.opera.postError) {
|
|
window.opera.postError(msg);
|
|
}
|
|
}
|
|
|
|
}));
|
|
|
|
|
|
/*! jQuery UI - v1.11.4 - 2015-03-11
|
|
* http://jqueryui.com
|
|
* Includes: core.js, widget.js, mouse.js, position.js, accordion.js, autocomplete.js, button.js, datepicker.js, dialog.js, draggable.js, droppable.js, effect.js, effect-blind.js, effect-bounce.js, effect-clip.js, effect-drop.js, effect-explode.js, effect-fade.js, effect-fold.js, effect-highlight.js, effect-puff.js, effect-pulsate.js, effect-scale.js, effect-shake.js, effect-size.js, effect-slide.js, effect-transfer.js, menu.js, progressbar.js, resizable.js, selectable.js, selectmenu.js, slider.js, sortable.js, spinner.js, tabs.js, tooltip.js
|
|
* Copyright 2015 jQuery Foundation and other contributors; Licensed MIT */
|
|
|
|
(function( factory ) {
|
|
if ( typeof define === "function" && define.amd ) {
|
|
|
|
// AMD. Register as an anonymous module.
|
|
define([ "jquery" ], factory );
|
|
} else {
|
|
|
|
// Browser globals
|
|
factory( jQuery );
|
|
}
|
|
}(function( $ ) {
|
|
/*!
|
|
* jQuery UI Core 1.11.4
|
|
* http://jqueryui.com
|
|
*
|
|
* Copyright jQuery Foundation and other contributors
|
|
* Released under the MIT license.
|
|
* http://jquery.org/license
|
|
*
|
|
* http://api.jqueryui.com/category/ui-core/
|
|
*/
|
|
|
|
|
|
// $.ui might exist from components with no dependencies, e.g., $.ui.position
|
|
$.ui = $.ui || {};
|
|
|
|
$.extend( $.ui, {
|
|
version: "1.11.4",
|
|
|
|
keyCode: {
|
|
BACKSPACE: 8,
|
|
COMMA: 188,
|
|
DELETE: 46,
|
|
DOWN: 40,
|
|
END: 35,
|
|
ENTER: 13,
|
|
ESCAPE: 27,
|
|
HOME: 36,
|
|
LEFT: 37,
|
|
PAGE_DOWN: 34,
|
|
PAGE_UP: 33,
|
|
PERIOD: 190,
|
|
RIGHT: 39,
|
|
SPACE: 32,
|
|
TAB: 9,
|
|
UP: 38
|
|
}
|
|
});
|
|
|
|
// plugins
|
|
$.fn.extend({
|
|
scrollParent: function( includeHidden ) {
|
|
var position = this.css( "position" ),
|
|
excludeStaticParent = position === "absolute",
|
|
overflowRegex = includeHidden ? /(auto|scroll|hidden)/ : /(auto|scroll)/,
|
|
scrollParent = this.parents().filter( function() {
|
|
var parent = $( this );
|
|
if ( excludeStaticParent && parent.css( "position" ) === "static" ) {
|
|
return false;
|
|
}
|
|
return overflowRegex.test( parent.css( "overflow" ) + parent.css( "overflow-y" ) + parent.css( "overflow-x" ) );
|
|
}).eq( 0 );
|
|
|
|
return position === "fixed" || !scrollParent.length ? $( this[ 0 ].ownerDocument || document ) : scrollParent;
|
|
},
|
|
|
|
uniqueId: (function() {
|
|
var uuid = 0;
|
|
|
|
return function() {
|
|
return this.each(function() {
|
|
if ( !this.id ) {
|
|
this.id = "ui-id-" + ( ++uuid );
|
|
}
|
|
});
|
|
};
|
|
})(),
|
|
|
|
removeUniqueId: function() {
|
|
return this.each(function() {
|
|
if ( /^ui-id-\d+$/.test( this.id ) ) {
|
|
$( this ).removeAttr( "id" );
|
|
}
|
|
});
|
|
}
|
|
});
|
|
|
|
// selectors
|
|
function focusable( element, isTabIndexNotNaN ) {
|
|
var map, mapName, img,
|
|
nodeName = element.nodeName.toLowerCase();
|
|
if ( "area" === nodeName ) {
|
|
map = element.parentNode;
|
|
mapName = map.name;
|
|
if ( !element.href || !mapName || map.nodeName.toLowerCase() !== "map" ) {
|
|
return false;
|
|
}
|
|
img = $( "img[usemap='#" + mapName + "']" )[ 0 ];
|
|
return !!img && visible( img );
|
|
}
|
|
return ( /^(input|select|textarea|button|object)$/.test( nodeName ) ?
|
|
!element.disabled :
|
|
"a" === nodeName ?
|
|
element.href || isTabIndexNotNaN :
|
|
isTabIndexNotNaN) &&
|
|
// the element and all of its ancestors must be visible
|
|
visible( element );
|
|
}
|
|
|
|
function visible( element ) {
|
|
return $.expr.filters.visible( element ) &&
|
|
!$( element ).parents().addBack().filter(function() {
|
|
return $.css( this, "visibility" ) === "hidden";
|
|
}).length;
|
|
}
|
|
|
|
$.extend( $.expr[ ":" ], {
|
|
data: $.expr.createPseudo ?
|
|
$.expr.createPseudo(function( dataName ) {
|
|
return function( elem ) {
|
|
return !!$.data( elem, dataName );
|
|
};
|
|
}) :
|
|
// support: jQuery <1.8
|
|
function( elem, i, match ) {
|
|
return !!$.data( elem, match[ 3 ] );
|
|
},
|
|
|
|
focusable: function( element ) {
|
|
return focusable( element, !isNaN( $.attr( element, "tabindex" ) ) );
|
|
},
|
|
|
|
tabbable: function( element ) {
|
|
var tabIndex = $.attr( element, "tabindex" ),
|
|
isTabIndexNaN = isNaN( tabIndex );
|
|
return ( isTabIndexNaN || tabIndex >= 0 ) && focusable( element, !isTabIndexNaN );
|
|
}
|
|
});
|
|
|
|
// support: jQuery <1.8
|
|
if ( !$( "<a>" ).outerWidth( 1 ).jquery ) {
|
|
$.each( [ "Width", "Height" ], function( i, name ) {
|
|
var side = name === "Width" ? [ "Left", "Right" ] : [ "Top", "Bottom" ],
|
|
type = name.toLowerCase(),
|
|
orig = {
|
|
innerWidth: $.fn.innerWidth,
|
|
innerHeight: $.fn.innerHeight,
|
|
outerWidth: $.fn.outerWidth,
|
|
outerHeight: $.fn.outerHeight
|
|
};
|
|
|
|
function reduce( elem, size, border, margin ) {
|
|
$.each( side, function() {
|
|
size -= parseFloat( $.css( elem, "padding" + this ) ) || 0;
|
|
if ( border ) {
|
|
size -= parseFloat( $.css( elem, "border" + this + "Width" ) ) || 0;
|
|
}
|
|
if ( margin ) {
|
|
size -= parseFloat( $.css( elem, "margin" + this ) ) || 0;
|
|
}
|
|
});
|
|
return size;
|
|
}
|
|
|
|
$.fn[ "inner" + name ] = function( size ) {
|
|
if ( size === undefined ) {
|
|
return orig[ "inner" + name ].call( this );
|
|
}
|
|
|
|
return this.each(function() {
|
|
$( this ).css( type, reduce( this, size ) + "px" );
|
|
});
|
|
};
|
|
|
|
$.fn[ "outer" + name] = function( size, margin ) {
|
|
if ( typeof size !== "number" ) {
|
|
return orig[ "outer" + name ].call( this, size );
|
|
}
|
|
|
|
return this.each(function() {
|
|
$( this).css( type, reduce( this, size, true, margin ) + "px" );
|
|
});
|
|
};
|
|
});
|
|
}
|
|
|
|
// support: jQuery <1.8
|
|
if ( !$.fn.addBack ) {
|
|
$.fn.addBack = function( selector ) {
|
|
return this.add( selector == null ?
|
|
this.prevObject : this.prevObject.filter( selector )
|
|
);
|
|
};
|
|
}
|
|
|
|
// support: jQuery 1.6.1, 1.6.2 (http://bugs.jquery.com/ticket/9413)
|
|
if ( $( "<a>" ).data( "a-b", "a" ).removeData( "a-b" ).data( "a-b" ) ) {
|
|
$.fn.removeData = (function( removeData ) {
|
|
return function( key ) {
|
|
if ( arguments.length ) {
|
|
return removeData.call( this, $.camelCase( key ) );
|
|
} else {
|
|
return removeData.call( this );
|
|
}
|
|
};
|
|
})( $.fn.removeData );
|
|
}
|
|
|
|
// deprecated
|
|
$.ui.ie = !!/msie [\w.]+/.exec( navigator.userAgent.toLowerCase() );
|
|
|
|
$.fn.extend({
|
|
focus: (function( orig ) {
|
|
return function( delay, fn ) {
|
|
return typeof delay === "number" ?
|
|
this.each(function() {
|
|
var elem = this;
|
|
setTimeout(function() {
|
|
$( elem ).focus();
|
|
if ( fn ) {
|
|
fn.call( elem );
|
|
}
|
|
}, delay );
|
|
}) :
|
|
orig.apply( this, arguments );
|
|
};
|
|
})( $.fn.focus ),
|
|
|
|
disableSelection: (function() {
|
|
var eventType = "onselectstart" in document.createElement( "div" ) ?
|
|
"selectstart" :
|
|
"mousedown";
|
|
|
|
return function() {
|
|
return this.bind( eventType + ".ui-disableSelection", function( event ) {
|
|
event.preventDefault();
|
|
});
|
|
};
|
|
})(),
|
|
|
|
enableSelection: function() {
|
|
return this.unbind( ".ui-disableSelection" );
|
|
},
|
|
|
|
zIndex: function( zIndex ) {
|
|
if ( zIndex !== undefined ) {
|
|
return this.css( "zIndex", zIndex );
|
|
}
|
|
|
|
if ( this.length ) {
|
|
var elem = $( this[ 0 ] ), position, value;
|
|
while ( elem.length && elem[ 0 ] !== document ) {
|
|
// Ignore z-index if position is set to a value where z-index is ignored by the browser
|
|
// This makes behavior of this function consistent across browsers
|
|
// WebKit always returns auto if the element is positioned
|
|
position = elem.css( "position" );
|
|
if ( position === "absolute" || position === "relative" || position === "fixed" ) {
|
|
// IE returns 0 when zIndex is not specified
|
|
// other browsers return a string
|
|
// we ignore the case of nested elements with an explicit value of 0
|
|
// <div style="z-index: -10;"><div style="z-index: 0;"></div></div>
|
|
value = parseInt( elem.css( "zIndex" ), 10 );
|
|
if ( !isNaN( value ) && value !== 0 ) {
|
|
return value;
|
|
}
|
|
}
|
|
elem = elem.parent();
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
});
|
|
|
|
// $.ui.plugin is deprecated. Use $.widget() extensions instead.
|
|
$.ui.plugin = {
|
|
add: function( module, option, set ) {
|
|
var i,
|
|
proto = $.ui[ module ].prototype;
|
|
for ( i in set ) {
|
|
proto.plugins[ i ] = proto.plugins[ i ] || [];
|
|
proto.plugins[ i ].push( [ option, set[ i ] ] );
|
|
}
|
|
},
|
|
call: function( instance, name, args, allowDisconnected ) {
|
|
var i,
|
|
set = instance.plugins[ name ];
|
|
|
|
if ( !set ) {
|
|
return;
|
|
}
|
|
|
|
if ( !allowDisconnected && ( !instance.element[ 0 ].parentNode || instance.element[ 0 ].parentNode.nodeType === 11 ) ) {
|
|
return;
|
|
}
|
|
|
|
for ( i = 0; i < set.length; i++ ) {
|
|
if ( instance.options[ set[ i ][ 0 ] ] ) {
|
|
set[ i ][ 1 ].apply( instance.element, args );
|
|
}
|
|
}
|
|
}
|
|
};
|
|
|
|
|
|
/*!
|
|
* jQuery UI Widget 1.11.4
|
|
* http://jqueryui.com
|
|
*
|
|
* Copyright jQuery Foundation and other contributors
|
|
* Released under the MIT license.
|
|
* http://jquery.org/license
|
|
*
|
|
* http://api.jqueryui.com/jQuery.widget/
|
|
*/
|
|
|
|
|
|
var widget_uuid = 0,
|
|
widget_slice = Array.prototype.slice;
|
|
|
|
$.cleanData = (function( orig ) {
|
|
return function( elems ) {
|
|
var events, elem, i;
|
|
for ( i = 0; (elem = elems[i]) != null; i++ ) {
|
|
try {
|
|
|
|
// Only trigger remove when necessary to save time
|
|
events = $._data( elem, "events" );
|
|
if ( events && events.remove ) {
|
|
$( elem ).triggerHandler( "remove" );
|
|
}
|
|
|
|
// http://bugs.jquery.com/ticket/8235
|
|
} catch ( e ) {}
|
|
}
|
|
orig( elems );
|
|
};
|
|
})( $.cleanData );
|
|
|
|
$.widget = function( name, base, prototype ) {
|
|
var fullName, existingConstructor, constructor, basePrototype,
|
|
// proxiedPrototype allows the provided prototype to remain unmodified
|
|
// so that it can be used as a mixin for multiple widgets (#8876)
|
|
proxiedPrototype = {},
|
|
namespace = name.split( "." )[ 0 ];
|
|
|
|
name = name.split( "." )[ 1 ];
|
|
fullName = namespace + "-" + name;
|
|
|
|
if ( !prototype ) {
|
|
prototype = base;
|
|
base = $.Widget;
|
|
}
|
|
|
|
// create selector for plugin
|
|
$.expr[ ":" ][ fullName.toLowerCase() ] = function( elem ) {
|
|
return !!$.data( elem, fullName );
|
|
};
|
|
|
|
$[ namespace ] = $[ namespace ] || {};
|
|
existingConstructor = $[ namespace ][ name ];
|
|
constructor = $[ namespace ][ name ] = function( options, element ) {
|
|
// allow instantiation without "new" keyword
|
|
if ( !this._createWidget ) {
|
|
return new constructor( options, element );
|
|
}
|
|
|
|
// allow instantiation without initializing for simple inheritance
|
|
// must use "new" keyword (the code above always passes args)
|
|
if ( arguments.length ) {
|
|
this._createWidget( options, element );
|
|
}
|
|
};
|
|
// extend with the existing constructor to carry over any static properties
|
|
$.extend( constructor, existingConstructor, {
|
|
version: prototype.version,
|
|
// copy the object used to create the prototype in case we need to
|
|
// redefine the widget later
|
|
_proto: $.extend( {}, prototype ),
|
|
// track widgets that inherit from this widget in case this widget is
|
|
// redefined after a widget inherits from it
|
|
_childConstructors: []
|
|
});
|
|
|
|
basePrototype = new base();
|
|
// we need to make the options hash a property directly on the new instance
|
|
// otherwise we'll modify the options hash on the prototype that we're
|
|
// inheriting from
|
|
basePrototype.options = $.widget.extend( {}, basePrototype.options );
|
|
$.each( prototype, function( prop, value ) {
|
|
if ( !$.isFunction( value ) ) {
|
|
proxiedPrototype[ prop ] = value;
|
|
return;
|
|
}
|
|
proxiedPrototype[ prop ] = (function() {
|
|
var _super = function() {
|
|
return base.prototype[ prop ].apply( this, arguments );
|
|
},
|
|
_superApply = function( args ) {
|
|
return base.prototype[ prop ].apply( this, args );
|
|
};
|
|
return function() {
|
|
var __super = this._super,
|
|
__superApply = this._superApply,
|
|
returnValue;
|
|
|
|
this._super = _super;
|
|
this._superApply = _superApply;
|
|
|
|
returnValue = value.apply( this, arguments );
|
|
|
|
this._super = __super;
|
|
this._superApply = __superApply;
|
|
|
|
return returnValue;
|
|
};
|
|
})();
|
|
});
|
|
constructor.prototype = $.widget.extend( basePrototype, {
|
|
// TODO: remove support for widgetEventPrefix
|
|
// always use the name + a colon as the prefix, e.g., draggable:start
|
|
// don't prefix for widgets that aren't DOM-based
|
|
widgetEventPrefix: existingConstructor ? (basePrototype.widgetEventPrefix || name) : name
|
|
}, proxiedPrototype, {
|
|
constructor: constructor,
|
|
namespace: namespace,
|
|
widgetName: name,
|
|
widgetFullName: fullName
|
|
});
|
|
|
|
// If this widget is being redefined then we need to find all widgets that
|
|
// are inheriting from it and redefine all of them so that they inherit from
|
|
// the new version of this widget. We're essentially trying to replace one
|
|
// level in the prototype chain.
|
|
if ( existingConstructor ) {
|
|
$.each( existingConstructor._childConstructors, function( i, child ) {
|
|
var childPrototype = child.prototype;
|
|
|
|
// redefine the child widget using the same prototype that was
|
|
// originally used, but inherit from the new version of the base
|
|
$.widget( childPrototype.namespace + "." + childPrototype.widgetName, constructor, child._proto );
|
|
});
|
|
// remove the list of existing child constructors from the old constructor
|
|
// so the old child constructors can be garbage collected
|
|
delete existingConstructor._childConstructors;
|
|
} else {
|
|
base._childConstructors.push( constructor );
|
|
}
|
|
|
|
$.widget.bridge( name, constructor );
|
|
|
|
return constructor;
|
|
};
|
|
|
|
$.widget.extend = function( target ) {
|
|
var input = widget_slice.call( arguments, 1 ),
|
|
inputIndex = 0,
|
|
inputLength = input.length,
|
|
key,
|
|
value;
|
|
for ( ; inputIndex < inputLength; inputIndex++ ) {
|
|
for ( key in input[ inputIndex ] ) {
|
|
value = input[ inputIndex ][ key ];
|
|
if ( input[ inputIndex ].hasOwnProperty( key ) && value !== undefined ) {
|
|
// Clone objects
|
|
if ( $.isPlainObject( value ) ) {
|
|
target[ key ] = $.isPlainObject( target[ key ] ) ?
|
|
$.widget.extend( {}, target[ key ], value ) :
|
|
// Don't extend strings, arrays, etc. with objects
|
|
$.widget.extend( {}, value );
|
|
// Copy everything else by reference
|
|
} else {
|
|
target[ key ] = value;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return target;
|
|
};
|
|
|
|
$.widget.bridge = function( name, object ) {
|
|
var fullName = object.prototype.widgetFullName || name;
|
|
$.fn[ name ] = function( options ) {
|
|
var isMethodCall = typeof options === "string",
|
|
args = widget_slice.call( arguments, 1 ),
|
|
returnValue = this;
|
|
|
|
if ( isMethodCall ) {
|
|
this.each(function() {
|
|
var methodValue,
|
|
instance = $.data( this, fullName );
|
|
if ( options === "instance" ) {
|
|
returnValue = instance;
|
|
return false;
|
|
}
|
|
if ( !instance ) {
|
|
return $.error( "cannot call methods on " + name + " prior to initialization; " +
|
|
"attempted to call method '" + options + "'" );
|
|
}
|
|
if ( !$.isFunction( instance[options] ) || options.charAt( 0 ) === "_" ) {
|
|
return $.error( "no such method '" + options + "' for " + name + " widget instance" );
|
|
}
|
|
methodValue = instance[ options ].apply( instance, args );
|
|
if ( methodValue !== instance && methodValue !== undefined ) {
|
|
returnValue = methodValue && methodValue.jquery ?
|
|
returnValue.pushStack( methodValue.get() ) :
|
|
methodValue;
|
|
return false;
|
|
}
|
|
});
|
|
} else {
|
|
|
|
// Allow multiple hashes to be passed on init
|
|
if ( args.length ) {
|
|
options = $.widget.extend.apply( null, [ options ].concat(args) );
|
|
}
|
|
|
|
this.each(function() {
|
|
var instance = $.data( this, fullName );
|
|
if ( instance ) {
|
|
instance.option( options || {} );
|
|
if ( instance._init ) {
|
|
instance._init();
|
|
}
|
|
} else {
|
|
$.data( this, fullName, new object( options, this ) );
|
|
}
|
|
});
|
|
}
|
|
|
|
return returnValue;
|
|
};
|
|
};
|
|
|
|
$.Widget = function( /* options, element */ ) {};
|
|
$.Widget._childConstructors = [];
|
|
|
|
$.Widget.prototype = {
|
|
widgetName: "widget",
|
|
widgetEventPrefix: "",
|
|
defaultElement: "<div>",
|
|
options: {
|
|
disabled: false,
|
|
|
|
// callbacks
|
|
create: null
|
|
},
|
|
_createWidget: function( options, element ) {
|
|
element = $( element || this.defaultElement || this )[ 0 ];
|
|
this.element = $( element );
|
|
this.uuid = widget_uuid++;
|
|
this.eventNamespace = "." + this.widgetName + this.uuid;
|
|
|
|
this.bindings = $();
|
|
this.hoverable = $();
|
|
this.focusable = $();
|
|
|
|
if ( element !== this ) {
|
|
$.data( element, this.widgetFullName, this );
|
|
this._on( true, this.element, {
|
|
remove: function( event ) {
|
|
if ( event.target === element ) {
|
|
this.destroy();
|
|
}
|
|
}
|
|
});
|
|
this.document = $( element.style ?
|
|
// element within the document
|
|
element.ownerDocument :
|
|
// element is window or document
|
|
element.document || element );
|
|
this.window = $( this.document[0].defaultView || this.document[0].parentWindow );
|
|
}
|
|
|
|
this.options = $.widget.extend( {},
|
|
this.options,
|
|
this._getCreateOptions(),
|
|
options );
|
|
|
|
this._create();
|
|
this._trigger( "create", null, this._getCreateEventData() );
|
|
this._init();
|
|
},
|
|
_getCreateOptions: $.noop,
|
|
_getCreateEventData: $.noop,
|
|
_create: $.noop,
|
|
_init: $.noop,
|
|
|
|
destroy: function() {
|
|
this._destroy();
|
|
// we can probably remove the unbind calls in 2.0
|
|
// all event bindings should go through this._on()
|
|
this.element
|
|
.unbind( this.eventNamespace )
|
|
.removeData( this.widgetFullName )
|
|
// support: jquery <1.6.3
|
|
// http://bugs.jquery.com/ticket/9413
|
|
.removeData( $.camelCase( this.widgetFullName ) );
|
|
this.widget()
|
|
.unbind( this.eventNamespace )
|
|
.removeAttr( "aria-disabled" )
|
|
.removeClass(
|
|
this.widgetFullName + "-disabled " +
|
|
"ui-state-disabled" );
|
|
|
|
// clean up events and states
|
|
this.bindings.unbind( this.eventNamespace );
|
|
this.hoverable.removeClass( "ui-state-hover" );
|
|
this.focusable.removeClass( "ui-state-focus" );
|
|
},
|
|
_destroy: $.noop,
|
|
|
|
widget: function() {
|
|
return this.element;
|
|
},
|
|
|
|
option: function( key, value ) {
|
|
var options = key,
|
|
parts,
|
|
curOption,
|
|
i;
|
|
|
|
if ( arguments.length === 0 ) {
|
|
// don't return a reference to the internal hash
|
|
return $.widget.extend( {}, this.options );
|
|
}
|
|
|
|
if ( typeof key === "string" ) {
|
|
// handle nested keys, e.g., "foo.bar" => { foo: { bar: ___ } }
|
|
options = {};
|
|
parts = key.split( "." );
|
|
key = parts.shift();
|
|
if ( parts.length ) {
|
|
curOption = options[ key ] = $.widget.extend( {}, this.options[ key ] );
|
|
for ( i = 0; i < parts.length - 1; i++ ) {
|
|
curOption[ parts[ i ] ] = curOption[ parts[ i ] ] || {};
|
|
curOption = curOption[ parts[ i ] ];
|
|
}
|
|
key = parts.pop();
|
|
if ( arguments.length === 1 ) {
|
|
return curOption[ key ] === undefined ? null : curOption[ key ];
|
|
}
|
|
curOption[ key ] = value;
|
|
} else {
|
|
if ( arguments.length === 1 ) {
|
|
return this.options[ key ] === undefined ? null : this.options[ key ];
|
|
}
|
|
options[ key ] = value;
|
|
}
|
|
}
|
|
|
|
this._setOptions( options );
|
|
|
|
return this;
|
|
},
|
|
_setOptions: function( options ) {
|
|
var key;
|
|
|
|
for ( key in options ) {
|
|
this._setOption( key, options[ key ] );
|
|
}
|
|
|
|
return this;
|
|
},
|
|
_setOption: function( key, value ) {
|
|
this.options[ key ] = value;
|
|
|
|
if ( key === "disabled" ) {
|
|
this.widget()
|
|
.toggleClass( this.widgetFullName + "-disabled", !!value );
|
|
|
|
// If the widget is becoming disabled, then nothing is interactive
|
|
if ( value ) {
|
|
this.hoverable.removeClass( "ui-state-hover" );
|
|
this.focusable.removeClass( "ui-state-focus" );
|
|
}
|
|
}
|
|
|
|
return this;
|
|
},
|
|
|
|
enable: function() {
|
|
return this._setOptions({ disabled: false });
|
|
},
|
|
disable: function() {
|
|
return this._setOptions({ disabled: true });
|
|
},
|
|
|
|
_on: function( suppressDisabledCheck, element, handlers ) {
|
|
var delegateElement,
|
|
instance = this;
|
|
|
|
// no suppressDisabledCheck flag, shuffle arguments
|
|
if ( typeof suppressDisabledCheck !== "boolean" ) {
|
|
handlers = element;
|
|
element = suppressDisabledCheck;
|
|
suppressDisabledCheck = false;
|
|
}
|
|
|
|
// no element argument, shuffle and use this.element
|
|
if ( !handlers ) {
|
|
handlers = element;
|
|
element = this.element;
|
|
delegateElement = this.widget();
|
|
} else {
|
|
element = delegateElement = $( element );
|
|
this.bindings = this.bindings.add( element );
|
|
}
|
|
|
|
$.each( handlers, function( event, handler ) {
|
|
function handlerProxy() {
|
|
// allow widgets to customize the disabled handling
|
|
// - disabled as an array instead of boolean
|
|
// - disabled class as method for disabling individual parts
|
|
if ( !suppressDisabledCheck &&
|
|
( instance.options.disabled === true ||
|
|
$( this ).hasClass( "ui-state-disabled" ) ) ) {
|
|
return;
|
|
}
|
|
return ( typeof handler === "string" ? instance[ handler ] : handler )
|
|
.apply( instance, arguments );
|
|
}
|
|
|
|
// copy the guid so direct unbinding works
|
|
if ( typeof handler !== "string" ) {
|
|
handlerProxy.guid = handler.guid =
|
|
handler.guid || handlerProxy.guid || $.guid++;
|
|
}
|
|
|
|
var match = event.match( /^([\w:-]*)\s*(.*)$/ ),
|
|
eventName = match[1] + instance.eventNamespace,
|
|
selector = match[2];
|
|
if ( selector ) {
|
|
delegateElement.delegate( selector, eventName, handlerProxy );
|
|
} else {
|
|
element.bind( eventName, handlerProxy );
|
|
}
|
|
});
|
|
},
|
|
|
|
_off: function( element, eventName ) {
|
|
eventName = (eventName || "").split( " " ).join( this.eventNamespace + " " ) +
|
|
this.eventNamespace;
|
|
element.unbind( eventName ).undelegate( eventName );
|
|
|
|
// Clear the stack to avoid memory leaks (#10056)
|
|
this.bindings = $( this.bindings.not( element ).get() );
|
|
this.focusable = $( this.focusable.not( element ).get() );
|
|
this.hoverable = $( this.hoverable.not( element ).get() );
|
|
},
|
|
|
|
_delay: function( handler, delay ) {
|
|
function handlerProxy() {
|
|
return ( typeof handler === "string" ? instance[ handler ] : handler )
|
|
.apply( instance, arguments );
|
|
}
|
|
var instance = this;
|
|
return setTimeout( handlerProxy, delay || 0 );
|
|
},
|
|
|
|
_hoverable: function( element ) {
|
|
this.hoverable = this.hoverable.add( element );
|
|
this._on( element, {
|
|
mouseenter: function( event ) {
|
|
$( event.currentTarget ).addClass( "ui-state-hover" );
|
|
},
|
|
mouseleave: function( event ) {
|
|
$( event.currentTarget ).removeClass( "ui-state-hover" );
|
|
}
|
|
});
|
|
},
|
|
|
|
_focusable: function( element ) {
|
|
this.focusable = this.focusable.add( element );
|
|
this._on( element, {
|
|
focusin: function( event ) {
|
|
$( event.currentTarget ).addClass( "ui-state-focus" );
|
|
},
|
|
focusout: function( event ) {
|
|
$( event.currentTarget ).removeClass( "ui-state-focus" );
|
|
}
|
|
});
|
|
},
|
|
|
|
_trigger: function( type, event, data ) {
|
|
var prop, orig,
|
|
callback = this.options[ type ];
|
|
|
|
data = data || {};
|
|
event = $.Event( event );
|
|
event.type = ( type === this.widgetEventPrefix ?
|
|
type :
|
|
this.widgetEventPrefix + type ).toLowerCase();
|
|
// the original event may come from any element
|
|
// so we need to reset the target on the new event
|
|
event.target = this.element[ 0 ];
|
|
|
|
// copy original event properties over to the new event
|
|
orig = event.originalEvent;
|
|
if ( orig ) {
|
|
for ( prop in orig ) {
|
|
if ( !( prop in event ) ) {
|
|
event[ prop ] = orig[ prop ];
|
|
}
|
|
}
|
|
}
|
|
|
|
this.element.trigger( event, data );
|
|
return !( $.isFunction( callback ) &&
|
|
callback.apply( this.element[0], [ event ].concat( data ) ) === false ||
|
|
event.isDefaultPrevented() );
|
|
}
|
|
};
|
|
|
|
$.each( { show: "fadeIn", hide: "fadeOut" }, function( method, defaultEffect ) {
|
|
$.Widget.prototype[ "_" + method ] = function( element, options, callback ) {
|
|
if ( typeof options === "string" ) {
|
|
options = { effect: options };
|
|
}
|
|
var hasOptions,
|
|
effectName = !options ?
|
|
method :
|
|
options === true || typeof options === "number" ?
|
|
defaultEffect :
|
|
options.effect || defaultEffect;
|
|
options = options || {};
|
|
if ( typeof options === "number" ) {
|
|
options = { duration: options };
|
|
}
|
|
hasOptions = !$.isEmptyObject( options );
|
|
options.complete = callback;
|
|
if ( options.delay ) {
|
|
element.delay( options.delay );
|
|
}
|
|
if ( hasOptions && $.effects && $.effects.effect[ effectName ] ) {
|
|
element[ method ]( options );
|
|
} else if ( effectName !== method && element[ effectName ] ) {
|
|
element[ effectName ]( options.duration, options.easing, callback );
|
|
} else {
|
|
element.queue(function( next ) {
|
|
$( this )[ method ]();
|
|
if ( callback ) {
|
|
callback.call( element[ 0 ] );
|
|
}
|
|
next();
|
|
});
|
|
}
|
|
};
|
|
});
|
|
|
|
var widget = $.widget;
|
|
|
|
|
|
/*!
|
|
* jQuery UI Mouse 1.11.4
|
|
* http://jqueryui.com
|
|
*
|
|
* Copyright jQuery Foundation and other contributors
|
|
* Released under the MIT license.
|
|
* http://jquery.org/license
|
|
*
|
|
* http://api.jqueryui.com/mouse/
|
|
*/
|
|
|
|
|
|
var mouseHandled = false;
|
|
$( document ).mouseup( function() {
|
|
mouseHandled = false;
|
|
});
|
|
|
|
var mouse = $.widget("ui.mouse", {
|
|
version: "1.11.4",
|
|
options: {
|
|
cancel: "input,textarea,button,select,option",
|
|
distance: 1,
|
|
delay: 0
|
|
},
|
|
_mouseInit: function() {
|
|
var that = this;
|
|
|
|
this.element
|
|
.bind("mousedown." + this.widgetName, function(event) {
|
|
return that._mouseDown(event);
|
|
})
|
|
.bind("click." + this.widgetName, function(event) {
|
|
if (true === $.data(event.target, that.widgetName + ".preventClickEvent")) {
|
|
$.removeData(event.target, that.widgetName + ".preventClickEvent");
|
|
event.stopImmediatePropagation();
|
|
return false;
|
|
}
|
|
});
|
|
|
|
this.started = false;
|
|
},
|
|
|
|
// TODO: make sure destroying one instance of mouse doesn't mess with
|
|
// other instances of mouse
|
|
_mouseDestroy: function() {
|
|
this.element.unbind("." + this.widgetName);
|
|
if ( this._mouseMoveDelegate ) {
|
|
this.document
|
|
.unbind("mousemove." + this.widgetName, this._mouseMoveDelegate)
|
|
.unbind("mouseup." + this.widgetName, this._mouseUpDelegate);
|
|
}
|
|
},
|
|
|
|
_mouseDown: function(event) {
|
|
// don't let more than one widget handle mouseStart
|
|
if ( mouseHandled ) {
|
|
return;
|
|
}
|
|
|
|
this._mouseMoved = false;
|
|
|
|
// we may have missed mouseup (out of window)
|
|
(this._mouseStarted && this._mouseUp(event));
|
|
|
|
this._mouseDownEvent = event;
|
|
|
|
var that = this,
|
|
btnIsLeft = (event.which === 1),
|
|
// event.target.nodeName works around a bug in IE 8 with
|
|
// disabled inputs (#7620)
|
|
elIsCancel = (typeof this.options.cancel === "string" && event.target.nodeName ? $(event.target).closest(this.options.cancel).length : false);
|
|
if (!btnIsLeft || elIsCancel || !this._mouseCapture(event)) {
|
|
return true;
|
|
}
|
|
|
|
this.mouseDelayMet = !this.options.delay;
|
|
if (!this.mouseDelayMet) {
|
|
this._mouseDelayTimer = setTimeout(function() {
|
|
that.mouseDelayMet = true;
|
|
}, this.options.delay);
|
|
}
|
|
|
|
if (this._mouseDistanceMet(event) && this._mouseDelayMet(event)) {
|
|
this._mouseStarted = (this._mouseStart(event) !== false);
|
|
if (!this._mouseStarted) {
|
|
event.preventDefault();
|
|
return true;
|
|
}
|
|
}
|
|
|
|
// Click event may never have fired (Gecko & Opera)
|
|
if (true === $.data(event.target, this.widgetName + ".preventClickEvent")) {
|
|
$.removeData(event.target, this.widgetName + ".preventClickEvent");
|
|
}
|
|
|
|
// these delegates are required to keep context
|
|
this._mouseMoveDelegate = function(event) {
|
|
return that._mouseMove(event);
|
|
};
|
|
this._mouseUpDelegate = function(event) {
|
|
return that._mouseUp(event);
|
|
};
|
|
|
|
this.document
|
|
.bind( "mousemove." + this.widgetName, this._mouseMoveDelegate )
|
|
.bind( "mouseup." + this.widgetName, this._mouseUpDelegate );
|
|
|
|
event.preventDefault();
|
|
|
|
mouseHandled = true;
|
|
return true;
|
|
},
|
|
|
|
_mouseMove: function(event) {
|
|
// Only check for mouseups outside the document if you've moved inside the document
|
|
// at least once. This prevents the firing of mouseup in the case of IE<9, which will
|
|
// fire a mousemove event if content is placed under the cursor. See #7778
|
|
// Support: IE <9
|
|
if ( this._mouseMoved ) {
|
|
// IE mouseup check - mouseup happened when mouse was out of window
|
|
if ($.ui.ie && ( !document.documentMode || document.documentMode < 9 ) && !event.button) {
|
|
return this._mouseUp(event);
|
|
|
|
// Iframe mouseup check - mouseup occurred in another document
|
|
} else if ( !event.which ) {
|
|
return this._mouseUp( event );
|
|
}
|
|
}
|
|
|
|
if ( event.which || event.button ) {
|
|
this._mouseMoved = true;
|
|
}
|
|
|
|
if (this._mouseStarted) {
|
|
this._mouseDrag(event);
|
|
return event.preventDefault();
|
|
}
|
|
|
|
if (this._mouseDistanceMet(event) && this._mouseDelayMet(event)) {
|
|
this._mouseStarted =
|
|
(this._mouseStart(this._mouseDownEvent, event) !== false);
|
|
(this._mouseStarted ? this._mouseDrag(event) : this._mouseUp(event));
|
|
}
|
|
|
|
return !this._mouseStarted;
|
|
},
|
|
|
|
_mouseUp: function(event) {
|
|
this.document
|
|
.unbind( "mousemove." + this.widgetName, this._mouseMoveDelegate )
|
|
.unbind( "mouseup." + this.widgetName, this._mouseUpDelegate );
|
|
|
|
if (this._mouseStarted) {
|
|
this._mouseStarted = false;
|
|
|
|
if (event.target === this._mouseDownEvent.target) {
|
|
$.data(event.target, this.widgetName + ".preventClickEvent", true);
|
|
}
|
|
|
|
this._mouseStop(event);
|
|
}
|
|
|
|
mouseHandled = false;
|
|
return false;
|
|
},
|
|
|
|
_mouseDistanceMet: function(event) {
|
|
return (Math.max(
|
|
Math.abs(this._mouseDownEvent.pageX - event.pageX),
|
|
Math.abs(this._mouseDownEvent.pageY - event.pageY)
|
|
) >= this.options.distance
|
|
);
|
|
},
|
|
|
|
_mouseDelayMet: function(/* event */) {
|
|
return this.mouseDelayMet;
|
|
},
|
|
|
|
// These are placeholder methods, to be overriden by extending plugin
|
|
_mouseStart: function(/* event */) {},
|
|
_mouseDrag: function(/* event */) {},
|
|
_mouseStop: function(/* event */) {},
|
|
_mouseCapture: function(/* event */) { return true; }
|
|
});
|
|
|
|
|
|
/*!
|
|
* jQuery UI Position 1.11.4
|
|
* http://jqueryui.com
|
|
*
|
|
* Copyright jQuery Foundation and other contributors
|
|
* Released under the MIT license.
|
|
* http://jquery.org/license
|
|
*
|
|
* http://api.jqueryui.com/position/
|
|
*/
|
|
|
|
(function() {
|
|
|
|
$.ui = $.ui || {};
|
|
|
|
var cachedScrollbarWidth, supportsOffsetFractions,
|
|
max = Math.max,
|
|
abs = Math.abs,
|
|
round = Math.round,
|
|
rhorizontal = /left|center|right/,
|
|
rvertical = /top|center|bottom/,
|
|
roffset = /[\+\-]\d+(\.[\d]+)?%?/,
|
|
rposition = /^\w+/,
|
|
rpercent = /%$/,
|
|
_position = $.fn.position;
|
|
|
|
function getOffsets( offsets, width, height ) {
|
|
return [
|
|
parseFloat( offsets[ 0 ] ) * ( rpercent.test( offsets[ 0 ] ) ? width / 100 : 1 ),
|
|
parseFloat( offsets[ 1 ] ) * ( rpercent.test( offsets[ 1 ] ) ? height / 100 : 1 )
|
|
];
|
|
}
|
|
|
|
function parseCss( element, property ) {
|
|
return parseInt( $.css( element, property ), 10 ) || 0;
|
|
}
|
|
|
|
function getDimensions( elem ) {
|
|
var raw = elem[0];
|
|
if ( raw.nodeType === 9 ) {
|
|
return {
|
|
width: elem.width(),
|
|
height: elem.height(),
|
|
offset: { top: 0, left: 0 }
|
|
};
|
|
}
|
|
if ( $.isWindow( raw ) ) {
|
|
return {
|
|
width: elem.width(),
|
|
height: elem.height(),
|
|
offset: { top: elem.scrollTop(), left: elem.scrollLeft() }
|
|
};
|
|
}
|
|
if ( raw.preventDefault ) {
|
|
return {
|
|
width: 0,
|
|
height: 0,
|
|
offset: { top: raw.pageY, left: raw.pageX }
|
|
};
|
|
}
|
|
return {
|
|
width: elem.outerWidth(),
|
|
height: elem.outerHeight(),
|
|
offset: elem.offset()
|
|
};
|
|
}
|
|
|
|
$.position = {
|
|
scrollbarWidth: function() {
|
|
if ( cachedScrollbarWidth !== undefined ) {
|
|
return cachedScrollbarWidth;
|
|
}
|
|
var w1, w2,
|
|
div = $( "<div style='display:block;position:absolute;width:50px;height:50px;overflow:hidden;'><div style='height:100px;width:auto;'></div></div>" ),
|
|
innerDiv = div.children()[0];
|
|
|
|
$( "body" ).append( div );
|
|
w1 = innerDiv.offsetWidth;
|
|
div.css( "overflow", "scroll" );
|
|
|
|
w2 = innerDiv.offsetWidth;
|
|
|
|
if ( w1 === w2 ) {
|
|
w2 = div[0].clientWidth;
|
|
}
|
|
|
|
div.remove();
|
|
|
|
return (cachedScrollbarWidth = w1 - w2);
|
|
},
|
|
getScrollInfo: function( within ) {
|
|
var overflowX = within.isWindow || within.isDocument ? "" :
|
|
within.element.css( "overflow-x" ),
|
|
overflowY = within.isWindow || within.isDocument ? "" :
|
|
within.element.css( "overflow-y" ),
|
|
hasOverflowX = overflowX === "scroll" ||
|
|
( overflowX === "auto" && within.width < within.element[0].scrollWidth ),
|
|
hasOverflowY = overflowY === "scroll" ||
|
|
( overflowY === "auto" && within.height < within.element[0].scrollHeight );
|
|
return {
|
|
width: hasOverflowY ? $.position.scrollbarWidth() : 0,
|
|
height: hasOverflowX ? $.position.scrollbarWidth() : 0
|
|
};
|
|
},
|
|
getWithinInfo: function( element ) {
|
|
var withinElement = $( element || window ),
|
|
isWindow = $.isWindow( withinElement[0] ),
|
|
isDocument = !!withinElement[ 0 ] && withinElement[ 0 ].nodeType === 9;
|
|
return {
|
|
element: withinElement,
|
|
isWindow: isWindow,
|
|
isDocument: isDocument,
|
|
offset: withinElement.offset() || { left: 0, top: 0 },
|
|
scrollLeft: withinElement.scrollLeft(),
|
|
scrollTop: withinElement.scrollTop(),
|
|
|
|
// support: jQuery 1.6.x
|
|
// jQuery 1.6 doesn't support .outerWidth/Height() on documents or windows
|
|
width: isWindow || isDocument ? withinElement.width() : withinElement.outerWidth(),
|
|
height: isWindow || isDocument ? withinElement.height() : withinElement.outerHeight()
|
|
};
|
|
}
|
|
};
|
|
|
|
$.fn.position = function( options ) {
|
|
if ( !options || !options.of ) {
|
|
return _position.apply( this, arguments );
|
|
}
|
|
|
|
// make a copy, we don't want to modify arguments
|
|
options = $.extend( {}, options );
|
|
|
|
var atOffset, targetWidth, targetHeight, targetOffset, basePosition, dimensions,
|
|
target = $( options.of ),
|
|
within = $.position.getWithinInfo( options.within ),
|
|
scrollInfo = $.position.getScrollInfo( within ),
|
|
collision = ( options.collision || "flip" ).split( " " ),
|
|
offsets = {};
|
|
|
|
dimensions = getDimensions( target );
|
|
if ( target[0].preventDefault ) {
|
|
// force left top to allow flipping
|
|
options.at = "left top";
|
|
}
|
|
targetWidth = dimensions.width;
|
|
targetHeight = dimensions.height;
|
|
targetOffset = dimensions.offset;
|
|
// clone to reuse original targetOffset later
|
|
basePosition = $.extend( {}, targetOffset );
|
|
|
|
// force my and at to have valid horizontal and vertical positions
|
|
// if a value is missing or invalid, it will be converted to center
|
|
$.each( [ "my", "at" ], function() {
|
|
var pos = ( options[ this ] || "" ).split( " " ),
|
|
horizontalOffset,
|
|
verticalOffset;
|
|
|
|
if ( pos.length === 1) {
|
|
pos = rhorizontal.test( pos[ 0 ] ) ?
|
|
pos.concat( [ "center" ] ) :
|
|
rvertical.test( pos[ 0 ] ) ?
|
|
[ "center" ].concat( pos ) :
|
|
[ "center", "center" ];
|
|
}
|
|
pos[ 0 ] = rhorizontal.test( pos[ 0 ] ) ? pos[ 0 ] : "center";
|
|
pos[ 1 ] = rvertical.test( pos[ 1 ] ) ? pos[ 1 ] : "center";
|
|
|
|
// calculate offsets
|
|
horizontalOffset = roffset.exec( pos[ 0 ] );
|
|
verticalOffset = roffset.exec( pos[ 1 ] );
|
|
offsets[ this ] = [
|
|
horizontalOffset ? horizontalOffset[ 0 ] : 0,
|
|
verticalOffset ? verticalOffset[ 0 ] : 0
|
|
];
|
|
|
|
// reduce to just the positions without the offsets
|
|
options[ this ] = [
|
|
rposition.exec( pos[ 0 ] )[ 0 ],
|
|
rposition.exec( pos[ 1 ] )[ 0 ]
|
|
];
|
|
});
|
|
|
|
// normalize collision option
|
|
if ( collision.length === 1 ) {
|
|
collision[ 1 ] = collision[ 0 ];
|
|
}
|
|
|
|
if ( options.at[ 0 ] === "right" ) {
|
|
basePosition.left += targetWidth;
|
|
} else if ( options.at[ 0 ] === "center" ) {
|
|
basePosition.left += targetWidth / 2;
|
|
}
|
|
|
|
if ( options.at[ 1 ] === "bottom" ) {
|
|
basePosition.top += targetHeight;
|
|
} else if ( options.at[ 1 ] === "center" ) {
|
|
basePosition.top += targetHeight / 2;
|
|
}
|
|
|
|
atOffset = getOffsets( offsets.at, targetWidth, targetHeight );
|
|
basePosition.left += atOffset[ 0 ];
|
|
basePosition.top += atOffset[ 1 ];
|
|
|
|
return this.each(function() {
|
|
var collisionPosition, using,
|
|
elem = $( this ),
|
|
elemWidth = elem.outerWidth(),
|
|
elemHeight = elem.outerHeight(),
|
|
marginLeft = parseCss( this, "marginLeft" ),
|
|
marginTop = parseCss( this, "marginTop" ),
|
|
collisionWidth = elemWidth + marginLeft + parseCss( this, "marginRight" ) + scrollInfo.width,
|
|
collisionHeight = elemHeight + marginTop + parseCss( this, "marginBottom" ) + scrollInfo.height,
|
|
position = $.extend( {}, basePosition ),
|
|
myOffset = getOffsets( offsets.my, elem.outerWidth(), elem.outerHeight() );
|
|
|
|
if ( options.my[ 0 ] === "right" ) {
|
|
position.left -= elemWidth;
|
|
} else if ( options.my[ 0 ] === "center" ) {
|
|
position.left -= elemWidth / 2;
|
|
}
|
|
|
|
if ( options.my[ 1 ] === "bottom" ) {
|
|
position.top -= elemHeight;
|
|
} else if ( options.my[ 1 ] === "center" ) {
|
|
position.top -= elemHeight / 2;
|
|
}
|
|
|
|
position.left += myOffset[ 0 ];
|
|
position.top += myOffset[ 1 ];
|
|
|
|
// if the browser doesn't support fractions, then round for consistent results
|
|
if ( !supportsOffsetFractions ) {
|
|
position.left = round( position.left );
|
|
position.top = round( position.top );
|
|
}
|
|
|
|
collisionPosition = {
|
|
marginLeft: marginLeft,
|
|
marginTop: marginTop
|
|
};
|
|
|
|
$.each( [ "left", "top" ], function( i, dir ) {
|
|
if ( $.ui.position[ collision[ i ] ] ) {
|
|
$.ui.position[ collision[ i ] ][ dir ]( position, {
|
|
targetWidth: targetWidth,
|
|
targetHeight: targetHeight,
|
|
elemWidth: elemWidth,
|
|
elemHeight: elemHeight,
|
|
collisionPosition: collisionPosition,
|
|
collisionWidth: collisionWidth,
|
|
collisionHeight: collisionHeight,
|
|
offset: [ atOffset[ 0 ] + myOffset[ 0 ], atOffset [ 1 ] + myOffset[ 1 ] ],
|
|
my: options.my,
|
|
at: options.at,
|
|
within: within,
|
|
elem: elem
|
|
});
|
|
}
|
|
});
|
|
|
|
if ( options.using ) {
|
|
// adds feedback as second argument to using callback, if present
|
|
using = function( props ) {
|
|
var left = targetOffset.left - position.left,
|
|
right = left + targetWidth - elemWidth,
|
|
top = targetOffset.top - position.top,
|
|
bottom = top + targetHeight - elemHeight,
|
|
feedback = {
|
|
target: {
|
|
element: target,
|
|
left: targetOffset.left,
|
|
top: targetOffset.top,
|
|
width: targetWidth,
|
|
height: targetHeight
|
|
},
|
|
element: {
|
|
element: elem,
|
|
left: position.left,
|
|
top: position.top,
|
|
width: elemWidth,
|
|
height: elemHeight
|
|
},
|
|
horizontal: right < 0 ? "left" : left > 0 ? "right" : "center",
|
|
vertical: bottom < 0 ? "top" : top > 0 ? "bottom" : "middle"
|
|
};
|
|
if ( targetWidth < elemWidth && abs( left + right ) < targetWidth ) {
|
|
feedback.horizontal = "center";
|
|
}
|
|
if ( targetHeight < elemHeight && abs( top + bottom ) < targetHeight ) {
|
|
feedback.vertical = "middle";
|
|
}
|
|
if ( max( abs( left ), abs( right ) ) > max( abs( top ), abs( bottom ) ) ) {
|
|
feedback.important = "horizontal";
|
|
} else {
|
|
feedback.important = "vertical";
|
|
}
|
|
options.using.call( this, props, feedback );
|
|
};
|
|
}
|
|
|
|
elem.offset( $.extend( position, { using: using } ) );
|
|
});
|
|
};
|
|
|
|
$.ui.position = {
|
|
fit: {
|
|
left: function( position, data ) {
|
|
var within = data.within,
|
|
withinOffset = within.isWindow ? within.scrollLeft : within.offset.left,
|
|
outerWidth = within.width,
|
|
collisionPosLeft = position.left - data.collisionPosition.marginLeft,
|
|
overLeft = withinOffset - collisionPosLeft,
|
|
overRight = collisionPosLeft + data.collisionWidth - outerWidth - withinOffset,
|
|
newOverRight;
|
|
|
|
// element is wider than within
|
|
if ( data.collisionWidth > outerWidth ) {
|
|
// element is initially over the left side of within
|
|
if ( overLeft > 0 && overRight <= 0 ) {
|
|
newOverRight = position.left + overLeft + data.collisionWidth - outerWidth - withinOffset;
|
|
position.left += overLeft - newOverRight;
|
|
// element is initially over right side of within
|
|
} else if ( overRight > 0 && overLeft <= 0 ) {
|
|
position.left = withinOffset;
|
|
// element is initially over both left and right sides of within
|
|
} else {
|
|
if ( overLeft > overRight ) {
|
|
position.left = withinOffset + outerWidth - data.collisionWidth;
|
|
} else {
|
|
position.left = withinOffset;
|
|
}
|
|
}
|
|
// too far left -> align with left edge
|
|
} else if ( overLeft > 0 ) {
|
|
position.left += overLeft;
|
|
// too far right -> align with right edge
|
|
} else if ( overRight > 0 ) {
|
|
position.left -= overRight;
|
|
// adjust based on position and margin
|
|
} else {
|
|
position.left = max( position.left - collisionPosLeft, position.left );
|
|
}
|
|
},
|
|
top: function( position, data ) {
|
|
var within = data.within,
|
|
withinOffset = within.isWindow ? within.scrollTop : within.offset.top,
|
|
outerHeight = data.within.height,
|
|
collisionPosTop = position.top - data.collisionPosition.marginTop,
|
|
overTop = withinOffset - collisionPosTop,
|
|
overBottom = collisionPosTop + data.collisionHeight - outerHeight - withinOffset,
|
|
newOverBottom;
|
|
|
|
// element is taller than within
|
|
if ( data.collisionHeight > outerHeight ) {
|
|
// element is initially over the top of within
|
|
if ( overTop > 0 && overBottom <= 0 ) {
|
|
newOverBottom = position.top + overTop + data.collisionHeight - outerHeight - withinOffset;
|
|
position.top += overTop - newOverBottom;
|
|
// element is initially over bottom of within
|
|
} else if ( overBottom > 0 && overTop <= 0 ) {
|
|
position.top = withinOffset;
|
|
// element is initially over both top and bottom of within
|
|
} else {
|
|
if ( overTop > overBottom ) {
|
|
position.top = withinOffset + outerHeight - data.collisionHeight;
|
|
} else {
|
|
position.top = withinOffset;
|
|
}
|
|
}
|
|
// too far up -> align with top
|
|
} else if ( overTop > 0 ) {
|
|
position.top += overTop;
|
|
// too far down -> align with bottom edge
|
|
} else if ( overBottom > 0 ) {
|
|
position.top -= overBottom;
|
|
// adjust based on position and margin
|
|
} else {
|
|
position.top = max( position.top - collisionPosTop, position.top );
|
|
}
|
|
}
|
|
},
|
|
flip: {
|
|
left: function( position, data ) {
|
|
var within = data.within,
|
|
withinOffset = within.offset.left + within.scrollLeft,
|
|
outerWidth = within.width,
|
|
offsetLeft = within.isWindow ? within.scrollLeft : within.offset.left,
|
|
collisionPosLeft = position.left - data.collisionPosition.marginLeft,
|
|
overLeft = collisionPosLeft - offsetLeft,
|
|
overRight = collisionPosLeft + data.collisionWidth - outerWidth - offsetLeft,
|
|
myOffset = data.my[ 0 ] === "left" ?
|
|
-data.elemWidth :
|
|
data.my[ 0 ] === "right" ?
|
|
data.elemWidth :
|
|
0,
|
|
atOffset = data.at[ 0 ] === "left" ?
|
|
data.targetWidth :
|
|
data.at[ 0 ] === "right" ?
|
|
-data.targetWidth :
|
|
0,
|
|
offset = -2 * data.offset[ 0 ],
|
|
newOverRight,
|
|
newOverLeft;
|
|
|
|
if ( overLeft < 0 ) {
|
|
newOverRight = position.left + myOffset + atOffset + offset + data.collisionWidth - outerWidth - withinOffset;
|
|
if ( newOverRight < 0 || newOverRight < abs( overLeft ) ) {
|
|
position.left += myOffset + atOffset + offset;
|
|
}
|
|
} else if ( overRight > 0 ) {
|
|
newOverLeft = position.left - data.collisionPosition.marginLeft + myOffset + atOffset + offset - offsetLeft;
|
|
if ( newOverLeft > 0 || abs( newOverLeft ) < overRight ) {
|
|
position.left += myOffset + atOffset + offset;
|
|
}
|
|
}
|
|
},
|
|
top: function( position, data ) {
|
|
var within = data.within,
|
|
withinOffset = within.offset.top + within.scrollTop,
|
|
outerHeight = within.height,
|
|
offsetTop = within.isWindow ? within.scrollTop : within.offset.top,
|
|
collisionPosTop = position.top - data.collisionPosition.marginTop,
|
|
overTop = collisionPosTop - offsetTop,
|
|
overBottom = collisionPosTop + data.collisionHeight - outerHeight - offsetTop,
|
|
top = data.my[ 1 ] === "top",
|
|
myOffset = top ?
|
|
-data.elemHeight :
|
|
data.my[ 1 ] === "bottom" ?
|
|
data.elemHeight :
|
|
0,
|
|
atOffset = data.at[ 1 ] === "top" ?
|
|
data.targetHeight :
|
|
data.at[ 1 ] === "bottom" ?
|
|
-data.targetHeight :
|
|
0,
|
|
offset = -2 * data.offset[ 1 ],
|
|
newOverTop,
|
|
newOverBottom;
|
|
if ( overTop < 0 ) {
|
|
newOverBottom = position.top + myOffset + atOffset + offset + data.collisionHeight - outerHeight - withinOffset;
|
|
if ( newOverBottom < 0 || newOverBottom < abs( overTop ) ) {
|
|
position.top += myOffset + atOffset + offset;
|
|
}
|
|
} else if ( overBottom > 0 ) {
|
|
newOverTop = position.top - data.collisionPosition.marginTop + myOffset + atOffset + offset - offsetTop;
|
|
if ( newOverTop > 0 || abs( newOverTop ) < overBottom ) {
|
|
position.top += myOffset + atOffset + offset;
|
|
}
|
|
}
|
|
}
|
|
},
|
|
flipfit: {
|
|
left: function() {
|
|
$.ui.position.flip.left.apply( this, arguments );
|
|
$.ui.position.fit.left.apply( this, arguments );
|
|
},
|
|
top: function() {
|
|
$.ui.position.flip.top.apply( this, arguments );
|
|
$.ui.position.fit.top.apply( this, arguments );
|
|
}
|
|
}
|
|
};
|
|
|
|
// fraction support test
|
|
(function() {
|
|
var testElement, testElementParent, testElementStyle, offsetLeft, i,
|
|
body = document.getElementsByTagName( "body" )[ 0 ],
|
|
div = document.createElement( "div" );
|
|
|
|
//Create a "fake body" for testing based on method used in jQuery.support
|
|
testElement = document.createElement( body ? "div" : "body" );
|
|
testElementStyle = {
|
|
visibility: "hidden",
|
|
width: 0,
|
|
height: 0,
|
|
border: 0,
|
|
margin: 0,
|
|
background: "none"
|
|
};
|
|
if ( body ) {
|
|
$.extend( testElementStyle, {
|
|
position: "absolute",
|
|
left: "-1000px",
|
|
top: "-1000px"
|
|
});
|
|
}
|
|
for ( i in testElementStyle ) {
|
|
testElement.style[ i ] = testElementStyle[ i ];
|
|
}
|
|
testElement.appendChild( div );
|
|
testElementParent = body || document.documentElement;
|
|
testElementParent.insertBefore( testElement, testElementParent.firstChild );
|
|
|
|
div.style.cssText = "position: absolute; left: 10.7432222px;";
|
|
|
|
offsetLeft = $( div ).offset().left;
|
|
supportsOffsetFractions = offsetLeft > 10 && offsetLeft < 11;
|
|
|
|
testElement.innerHTML = "";
|
|
testElementParent.removeChild( testElement );
|
|
})();
|
|
|
|
})();
|
|
|
|
var position = $.ui.position;
|
|
|
|
|
|
/*!
|
|
* jQuery UI Accordion 1.11.4
|
|
* http://jqueryui.com
|
|
*
|
|
* Copyright jQuery Foundation and other contributors
|
|
* Released under the MIT license.
|
|
* http://jquery.org/license
|
|
*
|
|
* http://api.jqueryui.com/accordion/
|
|
*/
|
|
|
|
|
|
var accordion = $.widget( "ui.accordion", {
|
|
version: "1.11.4",
|
|
options: {
|
|
active: 0,
|
|
animate: {},
|
|
collapsible: false,
|
|
event: "click",
|
|
header: "> li > :first-child,> :not(li):even",
|
|
heightStyle: "auto",
|
|
icons: {
|
|
activeHeader: "ui-icon-triangle-1-s",
|
|
header: "ui-icon-triangle-1-e"
|
|
},
|
|
|
|
// callbacks
|
|
activate: null,
|
|
beforeActivate: null
|
|
},
|
|
|
|
hideProps: {
|
|
borderTopWidth: "hide",
|
|
borderBottomWidth: "hide",
|
|
paddingTop: "hide",
|
|
paddingBottom: "hide",
|
|
height: "hide"
|
|
},
|
|
|
|
showProps: {
|
|
borderTopWidth: "show",
|
|
borderBottomWidth: "show",
|
|
paddingTop: "show",
|
|
paddingBottom: "show",
|
|
height: "show"
|
|
},
|
|
|
|
_create: function() {
|
|
var options = this.options;
|
|
this.prevShow = this.prevHide = $();
|
|
this.element.addClass( "ui-accordion ui-widget ui-helper-reset" )
|
|
// ARIA
|
|
.attr( "role", "tablist" );
|
|
|
|
// don't allow collapsible: false and active: false / null
|
|
if ( !options.collapsible && (options.active === false || options.active == null) ) {
|
|
options.active = 0;
|
|
}
|
|
|
|
this._processPanels();
|
|
// handle negative values
|
|
if ( options.active < 0 ) {
|
|
options.active += this.headers.length;
|
|
}
|
|
this._refresh();
|
|
},
|
|
|
|
_getCreateEventData: function() {
|
|
return {
|
|
header: this.active,
|
|
panel: !this.active.length ? $() : this.active.next()
|
|
};
|
|
},
|
|
|
|
_createIcons: function() {
|
|
var icons = this.options.icons;
|
|
if ( icons ) {
|
|
$( "<span>" )
|
|
.addClass( "ui-accordion-header-icon ui-icon " + icons.header )
|
|
.prependTo( this.headers );
|
|
this.active.children( ".ui-accordion-header-icon" )
|
|
.removeClass( icons.header )
|
|
.addClass( icons.activeHeader );
|
|
this.headers.addClass( "ui-accordion-icons" );
|
|
}
|
|
},
|
|
|
|
_destroyIcons: function() {
|
|
this.headers
|
|
.removeClass( "ui-accordion-icons" )
|
|
.children( ".ui-accordion-header-icon" )
|
|
.remove();
|
|
},
|
|
|
|
_destroy: function() {
|
|
var contents;
|
|
|
|
// clean up main element
|
|
this.element
|
|
.removeClass( "ui-accordion ui-widget ui-helper-reset" )
|
|
.removeAttr( "role" );
|
|
|
|
// clean up headers
|
|
this.headers
|
|
.removeClass( "ui-accordion-header ui-accordion-header-active ui-state-default " +
|
|
"ui-corner-all ui-state-active ui-state-disabled ui-corner-top" )
|
|
.removeAttr( "role" )
|
|
.removeAttr( "aria-expanded" )
|
|
.removeAttr( "aria-selected" )
|
|
.removeAttr( "aria-controls" )
|
|
.removeAttr( "tabIndex" )
|
|
.removeUniqueId();
|
|
|
|
this._destroyIcons();
|
|
|
|
// clean up content panels
|
|
contents = this.headers.next()
|
|
.removeClass( "ui-helper-reset ui-widget-content ui-corner-bottom " +
|
|
"ui-accordion-content ui-accordion-content-active ui-state-disabled" )
|
|
.css( "display", "" )
|
|
.removeAttr( "role" )
|
|
.removeAttr( "aria-hidden" )
|
|
.removeAttr( "aria-labelledby" )
|
|
.removeUniqueId();
|
|
|
|
if ( this.options.heightStyle !== "content" ) {
|
|
contents.css( "height", "" );
|
|
}
|
|
},
|
|
|
|
_setOption: function( key, value ) {
|
|
if ( key === "active" ) {
|
|
// _activate() will handle invalid values and update this.options
|
|
this._activate( value );
|
|
return;
|
|
}
|
|
|
|
if ( key === "event" ) {
|
|
if ( this.options.event ) {
|
|
this._off( this.headers, this.options.event );
|
|
}
|
|
this._setupEvents( value );
|
|
}
|
|
|
|
this._super( key, value );
|
|
|
|
// setting collapsible: false while collapsed; open first panel
|
|
if ( key === "collapsible" && !value && this.options.active === false ) {
|
|
this._activate( 0 );
|
|
}
|
|
|
|
if ( key === "icons" ) {
|
|
this._destroyIcons();
|
|
if ( value ) {
|
|
this._createIcons();
|
|
}
|
|
}
|
|
|
|
// #5332 - opacity doesn't cascade to positioned elements in IE
|
|
// so we need to add the disabled class to the headers and panels
|
|
if ( key === "disabled" ) {
|
|
this.element
|
|
.toggleClass( "ui-state-disabled", !!value )
|
|
.attr( "aria-disabled", value );
|
|
this.headers.add( this.headers.next() )
|
|
.toggleClass( "ui-state-disabled", !!value );
|
|
}
|
|
},
|
|
|
|
_keydown: function( event ) {
|
|
if ( event.altKey || event.ctrlKey ) {
|
|
return;
|
|
}
|
|
|
|
var keyCode = $.ui.keyCode,
|
|
length = this.headers.length,
|
|
currentIndex = this.headers.index( event.target ),
|
|
toFocus = false;
|
|
|
|
switch ( event.keyCode ) {
|
|
case keyCode.RIGHT:
|
|
case keyCode.DOWN:
|
|
toFocus = this.headers[ ( currentIndex + 1 ) % length ];
|
|
break;
|
|
case keyCode.LEFT:
|
|
case keyCode.UP:
|
|
toFocus = this.headers[ ( currentIndex - 1 + length ) % length ];
|
|
break;
|
|
case keyCode.SPACE:
|
|
case keyCode.ENTER:
|
|
this._eventHandler( event );
|
|
break;
|
|
case keyCode.HOME:
|
|
toFocus = this.headers[ 0 ];
|
|
break;
|
|
case keyCode.END:
|
|
toFocus = this.headers[ length - 1 ];
|
|
break;
|
|
}
|
|
|
|
if ( toFocus ) {
|
|
$( event.target ).attr( "tabIndex", -1 );
|
|
$( toFocus ).attr( "tabIndex", 0 );
|
|
toFocus.focus();
|
|
event.preventDefault();
|
|
}
|
|
},
|
|
|
|
_panelKeyDown: function( event ) {
|
|
if ( event.keyCode === $.ui.keyCode.UP && event.ctrlKey ) {
|
|
$( event.currentTarget ).prev().focus();
|
|
}
|
|
},
|
|
|
|
refresh: function() {
|
|
var options = this.options;
|
|
this._processPanels();
|
|
|
|
// was collapsed or no panel
|
|
if ( ( options.active === false && options.collapsible === true ) || !this.headers.length ) {
|
|
options.active = false;
|
|
this.active = $();
|
|
// active false only when collapsible is true
|
|
} else if ( options.active === false ) {
|
|
this._activate( 0 );
|
|
// was active, but active panel is gone
|
|
} else if ( this.active.length && !$.contains( this.element[ 0 ], this.active[ 0 ] ) ) {
|
|
// all remaining panel are disabled
|
|
if ( this.headers.length === this.headers.find(".ui-state-disabled").length ) {
|
|
options.active = false;
|
|
this.active = $();
|
|
// activate previous panel
|
|
} else {
|
|
this._activate( Math.max( 0, options.active - 1 ) );
|
|
}
|
|
// was active, active panel still exists
|
|
} else {
|
|
// make sure active index is correct
|
|
options.active = this.headers.index( this.active );
|
|
}
|
|
|
|
this._destroyIcons();
|
|
|
|
this._refresh();
|
|
},
|
|
|
|
_processPanels: function() {
|
|
var prevHeaders = this.headers,
|
|
prevPanels = this.panels;
|
|
|
|
this.headers = this.element.find( this.options.header )
|
|
.addClass( "ui-accordion-header ui-state-default ui-corner-all" );
|
|
|
|
this.panels = this.headers.next()
|
|
.addClass( "ui-accordion-content ui-helper-reset ui-widget-content ui-corner-bottom" )
|
|
.filter( ":not(.ui-accordion-content-active)" )
|
|
.hide();
|
|
|
|
// Avoid memory leaks (#10056)
|
|
if ( prevPanels ) {
|
|
this._off( prevHeaders.not( this.headers ) );
|
|
this._off( prevPanels.not( this.panels ) );
|
|
}
|
|
},
|
|
|
|
_refresh: function() {
|
|
var maxHeight,
|
|
options = this.options,
|
|
heightStyle = options.heightStyle,
|
|
parent = this.element.parent();
|
|
|
|
this.active = this._findActive( options.active )
|
|
.addClass( "ui-accordion-header-active ui-state-active ui-corner-top" )
|
|
.removeClass( "ui-corner-all" );
|
|
this.active.next()
|
|
.addClass( "ui-accordion-content-active" )
|
|
.show();
|
|
|
|
this.headers
|
|
.attr( "role", "tab" )
|
|
.each(function() {
|
|
var header = $( this ),
|
|
headerId = header.uniqueId().attr( "id" ),
|
|
panel = header.next(),
|
|
panelId = panel.uniqueId().attr( "id" );
|
|
header.attr( "aria-controls", panelId );
|
|
panel.attr( "aria-labelledby", headerId );
|
|
})
|
|
.next()
|
|
.attr( "role", "tabpanel" );
|
|
|
|
this.headers
|
|
.not( this.active )
|
|
.attr({
|
|
"aria-selected": "false",
|
|
"aria-expanded": "false",
|
|
tabIndex: -1
|
|
})
|
|
.next()
|
|
.attr({
|
|
"aria-hidden": "true"
|
|
})
|
|
.hide();
|
|
|
|
// make sure at least one header is in the tab order
|
|
if ( !this.active.length ) {
|
|
this.headers.eq( 0 ).attr( "tabIndex", 0 );
|
|
} else {
|
|
this.active.attr({
|
|
"aria-selected": "true",
|
|
"aria-expanded": "true",
|
|
tabIndex: 0
|
|
})
|
|
.next()
|
|
.attr({
|
|
"aria-hidden": "false"
|
|
});
|
|
}
|
|
|
|
this._createIcons();
|
|
|
|
this._setupEvents( options.event );
|
|
|
|
if ( heightStyle === "fill" ) {
|
|
maxHeight = parent.height();
|
|
this.element.siblings( ":visible" ).each(function() {
|
|
var elem = $( this ),
|
|
position = elem.css( "position" );
|
|
|
|
if ( position === "absolute" || position === "fixed" ) {
|
|
return;
|
|
}
|
|
maxHeight -= elem.outerHeight( true );
|
|
});
|
|
|
|
this.headers.each(function() {
|
|
maxHeight -= $( this ).outerHeight( true );
|
|
});
|
|
|
|
this.headers.next()
|
|
.each(function() {
|
|
$( this ).height( Math.max( 0, maxHeight -
|
|
$( this ).innerHeight() + $( this ).height() ) );
|
|
})
|
|
.css( "overflow", "auto" );
|
|
} else if ( heightStyle === "auto" ) {
|
|
maxHeight = 0;
|
|
this.headers.next()
|
|
.each(function() {
|
|
maxHeight = Math.max( maxHeight, $( this ).css( "height", "" ).height() );
|
|
})
|
|
.height( maxHeight );
|
|
}
|
|
},
|
|
|
|
_activate: function( index ) {
|
|
var active = this._findActive( index )[ 0 ];
|
|
|
|
// trying to activate the already active panel
|
|
if ( active === this.active[ 0 ] ) {
|
|
return;
|
|
}
|
|
|
|
// trying to collapse, simulate a click on the currently active header
|
|
active = active || this.active[ 0 ];
|
|
|
|
this._eventHandler({
|
|
target: active,
|
|
currentTarget: active,
|
|
preventDefault: $.noop
|
|
});
|
|
},
|
|
|
|
_findActive: function( selector ) {
|
|
return typeof selector === "number" ? this.headers.eq( selector ) : $();
|
|
},
|
|
|
|
_setupEvents: function( event ) {
|
|
var events = {
|
|
keydown: "_keydown"
|
|
};
|
|
if ( event ) {
|
|
$.each( event.split( " " ), function( index, eventName ) {
|
|
events[ eventName ] = "_eventHandler";
|
|
});
|
|
}
|
|
|
|
this._off( this.headers.add( this.headers.next() ) );
|
|
this._on( this.headers, events );
|
|
this._on( this.headers.next(), { keydown: "_panelKeyDown" });
|
|
this._hoverable( this.headers );
|
|
this._focusable( this.headers );
|
|
},
|
|
|
|
_eventHandler: function( event ) {
|
|
var options = this.options,
|
|
active = this.active,
|
|
clicked = $( event.currentTarget ),
|
|
clickedIsActive = clicked[ 0 ] === active[ 0 ],
|
|
collapsing = clickedIsActive && options.collapsible,
|
|
toShow = collapsing ? $() : clicked.next(),
|
|
toHide = active.next(),
|
|
eventData = {
|
|
oldHeader: active,
|
|
oldPanel: toHide,
|
|
newHeader: collapsing ? $() : clicked,
|
|
newPanel: toShow
|
|
};
|
|
|
|
event.preventDefault();
|
|
|
|
if (
|
|
// click on active header, but not collapsible
|
|
( clickedIsActive && !options.collapsible ) ||
|
|
// allow canceling activation
|
|
( this._trigger( "beforeActivate", event, eventData ) === false ) ) {
|
|
return;
|
|
}
|
|
|
|
options.active = collapsing ? false : this.headers.index( clicked );
|
|
|
|
// when the call to ._toggle() comes after the class changes
|
|
// it causes a very odd bug in IE 8 (see #6720)
|
|
this.active = clickedIsActive ? $() : clicked;
|
|
this._toggle( eventData );
|
|
|
|
// switch classes
|
|
// corner classes on the previously active header stay after the animation
|
|
active.removeClass( "ui-accordion-header-active ui-state-active" );
|
|
if ( options.icons ) {
|
|
active.children( ".ui-accordion-header-icon" )
|
|
.removeClass( options.icons.activeHeader )
|
|
.addClass( options.icons.header );
|
|
}
|
|
|
|
if ( !clickedIsActive ) {
|
|
clicked
|
|
.removeClass( "ui-corner-all" )
|
|
.addClass( "ui-accordion-header-active ui-state-active ui-corner-top" );
|
|
if ( options.icons ) {
|
|
clicked.children( ".ui-accordion-header-icon" )
|
|
.removeClass( options.icons.header )
|
|
.addClass( options.icons.activeHeader );
|
|
}
|
|
|
|
clicked
|
|
.next()
|
|
.addClass( "ui-accordion-content-active" );
|
|
}
|
|
},
|
|
|
|
_toggle: function( data ) {
|
|
var toShow = data.newPanel,
|
|
toHide = this.prevShow.length ? this.prevShow : data.oldPanel;
|
|
|
|
// handle activating a panel during the animation for another activation
|
|
this.prevShow.add( this.prevHide ).stop( true, true );
|
|
this.prevShow = toShow;
|
|
this.prevHide = toHide;
|
|
|
|
if ( this.options.animate ) {
|
|
this._animate( toShow, toHide, data );
|
|
} else {
|
|
toHide.hide();
|
|
toShow.show();
|
|
this._toggleComplete( data );
|
|
}
|
|
|
|
toHide.attr({
|
|
"aria-hidden": "true"
|
|
});
|
|
toHide.prev().attr({
|
|
"aria-selected": "false",
|
|
"aria-expanded": "false"
|
|
});
|
|
// if we're switching panels, remove the old header from the tab order
|
|
// if we're opening from collapsed state, remove the previous header from the tab order
|
|
// if we're collapsing, then keep the collapsing header in the tab order
|
|
if ( toShow.length && toHide.length ) {
|
|
toHide.prev().attr({
|
|
"tabIndex": -1,
|
|
"aria-expanded": "false"
|
|
});
|
|
} else if ( toShow.length ) {
|
|
this.headers.filter(function() {
|
|
return parseInt( $( this ).attr( "tabIndex" ), 10 ) === 0;
|
|
})
|
|
.attr( "tabIndex", -1 );
|
|
}
|
|
|
|
toShow
|
|
.attr( "aria-hidden", "false" )
|
|
.prev()
|
|
.attr({
|
|
"aria-selected": "true",
|
|
"aria-expanded": "true",
|
|
tabIndex: 0
|
|
});
|
|
},
|
|
|
|
_animate: function( toShow, toHide, data ) {
|
|
var total, easing, duration,
|
|
that = this,
|
|
adjust = 0,
|
|
boxSizing = toShow.css( "box-sizing" ),
|
|
down = toShow.length &&
|
|
( !toHide.length || ( toShow.index() < toHide.index() ) ),
|
|
animate = this.options.animate || {},
|
|
options = down && animate.down || animate,
|
|
complete = function() {
|
|
that._toggleComplete( data );
|
|
};
|
|
|
|
if ( typeof options === "number" ) {
|
|
duration = options;
|
|
}
|
|
if ( typeof options === "string" ) {
|
|
easing = options;
|
|
}
|
|
// fall back from options to animation in case of partial down settings
|
|
easing = easing || options.easing || animate.easing;
|
|
duration = duration || options.duration || animate.duration;
|
|
|
|
if ( !toHide.length ) {
|
|
return toShow.animate( this.showProps, duration, easing, complete );
|
|
}
|
|
if ( !toShow.length ) {
|
|
return toHide.animate( this.hideProps, duration, easing, complete );
|
|
}
|
|
|
|
total = toShow.show().outerHeight();
|
|
toHide.animate( this.hideProps, {
|
|
duration: duration,
|
|
easing: easing,
|
|
step: function( now, fx ) {
|
|
fx.now = Math.round( now );
|
|
}
|
|
});
|
|
toShow
|
|
.hide()
|
|
.animate( this.showProps, {
|
|
duration: duration,
|
|
easing: easing,
|
|
complete: complete,
|
|
step: function( now, fx ) {
|
|
fx.now = Math.round( now );
|
|
if ( fx.prop !== "height" ) {
|
|
if ( boxSizing === "content-box" ) {
|
|
adjust += fx.now;
|
|
}
|
|
} else if ( that.options.heightStyle !== "content" ) {
|
|
fx.now = Math.round( total - toHide.outerHeight() - adjust );
|
|
adjust = 0;
|
|
}
|
|
}
|
|
});
|
|
},
|
|
|
|
_toggleComplete: function( data ) {
|
|
var toHide = data.oldPanel;
|
|
|
|
toHide
|
|
.removeClass( "ui-accordion-content-active" )
|
|
.prev()
|
|
.removeClass( "ui-corner-top" )
|
|
.addClass( "ui-corner-all" );
|
|
|
|
// Work around for rendering bug in IE (#5421)
|
|
if ( toHide.length ) {
|
|
toHide.parent()[ 0 ].className = toHide.parent()[ 0 ].className;
|
|
}
|
|
this._trigger( "activate", null, data );
|
|
}
|
|
});
|
|
|
|
|
|
/*!
|
|
* jQuery UI Menu 1.11.4
|
|
* http://jqueryui.com
|
|
*
|
|
* Copyright jQuery Foundation and other contributors
|
|
* Released under the MIT license.
|
|
* http://jquery.org/license
|
|
*
|
|
* http://api.jqueryui.com/menu/
|
|
*/
|
|
|
|
|
|
var menu = $.widget( "ui.menu", {
|
|
version: "1.11.4",
|
|
defaultElement: "<ul>",
|
|
delay: 300,
|
|
options: {
|
|
icons: {
|
|
submenu: "ui-icon-carat-1-e"
|
|
},
|
|
items: "> *",
|
|
menus: "ul",
|
|
position: {
|
|
my: "left-1 top",
|
|
at: "right top"
|
|
},
|
|
role: "menu",
|
|
|
|
// callbacks
|
|
blur: null,
|
|
focus: null,
|
|
select: null
|
|
},
|
|
|
|
_create: function() {
|
|
this.activeMenu = this.element;
|
|
|
|
// Flag used to prevent firing of the click handler
|
|
// as the event bubbles up through nested menus
|
|
this.mouseHandled = false;
|
|
this.element
|
|
.uniqueId()
|
|
.addClass( "ui-menu ui-widget ui-widget-content" )
|
|
.toggleClass( "ui-menu-icons", !!this.element.find( ".ui-icon" ).length )
|
|
.attr({
|
|
role: this.options.role,
|
|
tabIndex: 0
|
|
});
|
|
|
|
if ( this.options.disabled ) {
|
|
this.element
|
|
.addClass( "ui-state-disabled" )
|
|
.attr( "aria-disabled", "true" );
|
|
}
|
|
|
|
this._on({
|
|
// Prevent focus from sticking to links inside menu after clicking
|
|
// them (focus should always stay on UL during navigation).
|
|
"mousedown .ui-menu-item": function( event ) {
|
|
event.preventDefault();
|
|
},
|
|
"click .ui-menu-item": function( event ) {
|
|
var target = $( event.target );
|
|
if ( !this.mouseHandled && target.not( ".ui-state-disabled" ).length ) {
|
|
this.select( event );
|
|
|
|
// Only set the mouseHandled flag if the event will bubble, see #9469.
|
|
if ( !event.isPropagationStopped() ) {
|
|
this.mouseHandled = true;
|
|
}
|
|
|
|
// Open submenu on click
|
|
if ( target.has( ".ui-menu" ).length ) {
|
|
this.expand( event );
|
|
} else if ( !this.element.is( ":focus" ) && $( this.document[ 0 ].activeElement ).closest( ".ui-menu" ).length ) {
|
|
|
|
// Redirect focus to the menu
|
|
this.element.trigger( "focus", [ true ] );
|
|
|
|
// If the active item is on the top level, let it stay active.
|
|
// Otherwise, blur the active item since it is no longer visible.
|
|
if ( this.active && this.active.parents( ".ui-menu" ).length === 1 ) {
|
|
clearTimeout( this.timer );
|
|
}
|
|
}
|
|
}
|
|
},
|
|
"mouseenter .ui-menu-item": function( event ) {
|
|
// Ignore mouse events while typeahead is active, see #10458.
|
|
// Prevents focusing the wrong item when typeahead causes a scroll while the mouse
|
|
// is over an item in the menu
|
|
if ( this.previousFilter ) {
|
|
return;
|
|
}
|
|
var target = $( event.currentTarget );
|
|
// Remove ui-state-active class from siblings of the newly focused menu item
|
|
// to avoid a jump caused by adjacent elements both having a class with a border
|
|
target.siblings( ".ui-state-active" ).removeClass( "ui-state-active" );
|
|
this.focus( event, target );
|
|
},
|
|
mouseleave: "collapseAll",
|
|
"mouseleave .ui-menu": "collapseAll",
|
|
focus: function( event, keepActiveItem ) {
|
|
// If there's already an active item, keep it active
|
|
// If not, activate the first item
|
|
var item = this.active || this.element.find( this.options.items ).eq( 0 );
|
|
|
|
if ( !keepActiveItem ) {
|
|
this.focus( event, item );
|
|
}
|
|
},
|
|
blur: function( event ) {
|
|
this._delay(function() {
|
|
if ( !$.contains( this.element[0], this.document[0].activeElement ) ) {
|
|
this.collapseAll( event );
|
|
}
|
|
});
|
|
},
|
|
keydown: "_keydown"
|
|
});
|
|
|
|
this.refresh();
|
|
|
|
// Clicks outside of a menu collapse any open menus
|
|
this._on( this.document, {
|
|
click: function( event ) {
|
|
if ( this._closeOnDocumentClick( event ) ) {
|
|
this.collapseAll( event );
|
|
}
|
|
|
|
// Reset the mouseHandled flag
|
|
this.mouseHandled = false;
|
|
}
|
|
});
|
|
},
|
|
|
|
_destroy: function() {
|
|
// Destroy (sub)menus
|
|
this.element
|
|
.removeAttr( "aria-activedescendant" )
|
|
.find( ".ui-menu" ).addBack()
|
|
.removeClass( "ui-menu ui-widget ui-widget-content ui-menu-icons ui-front" )
|
|
.removeAttr( "role" )
|
|
.removeAttr( "tabIndex" )
|
|
.removeAttr( "aria-labelledby" )
|
|
.removeAttr( "aria-expanded" )
|
|
.removeAttr( "aria-hidden" )
|
|
.removeAttr( "aria-disabled" )
|
|
.removeUniqueId()
|
|
.show();
|
|
|
|
// Destroy menu items
|
|
this.element.find( ".ui-menu-item" )
|
|
.removeClass( "ui-menu-item" )
|
|
.removeAttr( "role" )
|
|
.removeAttr( "aria-disabled" )
|
|
.removeUniqueId()
|
|
.removeClass( "ui-state-hover" )
|
|
.removeAttr( "tabIndex" )
|
|
.removeAttr( "role" )
|
|
.removeAttr( "aria-haspopup" )
|
|
.children().each( function() {
|
|
var elem = $( this );
|
|
if ( elem.data( "ui-menu-submenu-carat" ) ) {
|
|
elem.remove();
|
|
}
|
|
});
|
|
|
|
// Destroy menu dividers
|
|
this.element.find( ".ui-menu-divider" ).removeClass( "ui-menu-divider ui-widget-content" );
|
|
},
|
|
|
|
_keydown: function( event ) {
|
|
var match, prev, character, skip,
|
|
preventDefault = true;
|
|
|
|
switch ( event.keyCode ) {
|
|
case $.ui.keyCode.PAGE_UP:
|
|
this.previousPage( event );
|
|
break;
|
|
case $.ui.keyCode.PAGE_DOWN:
|
|
this.nextPage( event );
|
|
break;
|
|
case $.ui.keyCode.HOME:
|
|
this._move( "first", "first", event );
|
|
break;
|
|
case $.ui.keyCode.END:
|
|
this._move( "last", "last", event );
|
|
break;
|
|
case $.ui.keyCode.UP:
|
|
this.previous( event );
|
|
break;
|
|
case $.ui.keyCode.DOWN:
|
|
this.next( event );
|
|
break;
|
|
case $.ui.keyCode.LEFT:
|
|
this.collapse( event );
|
|
break;
|
|
case $.ui.keyCode.RIGHT:
|
|
if ( this.active && !this.active.is( ".ui-state-disabled" ) ) {
|
|
this.expand( event );
|
|
}
|
|
break;
|
|
case $.ui.keyCode.ENTER:
|
|
case $.ui.keyCode.SPACE:
|
|
this._activate( event );
|
|
break;
|
|
case $.ui.keyCode.ESCAPE:
|
|
this.collapse( event );
|
|
break;
|
|
default:
|
|
preventDefault = false;
|
|
prev = this.previousFilter || "";
|
|
character = String.fromCharCode( event.keyCode );
|
|
skip = false;
|
|
|
|
clearTimeout( this.filterTimer );
|
|
|
|
if ( character === prev ) {
|
|
skip = true;
|
|
} else {
|
|
character = prev + character;
|
|
}
|
|
|
|
match = this._filterMenuItems( character );
|
|
match = skip && match.index( this.active.next() ) !== -1 ?
|
|
this.active.nextAll( ".ui-menu-item" ) :
|
|
match;
|
|
|
|
// If no matches on the current filter, reset to the last character pressed
|
|
// to move down the menu to the first item that starts with that character
|
|
if ( !match.length ) {
|
|
character = String.fromCharCode( event.keyCode );
|
|
match = this._filterMenuItems( character );
|
|
}
|
|
|
|
if ( match.length ) {
|
|
this.focus( event, match );
|
|
this.previousFilter = character;
|
|
this.filterTimer = this._delay(function() {
|
|
delete this.previousFilter;
|
|
}, 1000 );
|
|
} else {
|
|
delete this.previousFilter;
|
|
}
|
|
}
|
|
|
|
if ( preventDefault ) {
|
|
event.preventDefault();
|
|
}
|
|
},
|
|
|
|
_activate: function( event ) {
|
|
if ( !this.active.is( ".ui-state-disabled" ) ) {
|
|
if ( this.active.is( "[aria-haspopup='true']" ) ) {
|
|
this.expand( event );
|
|
} else {
|
|
this.select( event );
|
|
}
|
|
}
|
|
},
|
|
|
|
refresh: function() {
|
|
var menus, items,
|
|
that = this,
|
|
icon = this.options.icons.submenu,
|
|
submenus = this.element.find( this.options.menus );
|
|
|
|
this.element.toggleClass( "ui-menu-icons", !!this.element.find( ".ui-icon" ).length );
|
|
|
|
// Initialize nested menus
|
|
submenus.filter( ":not(.ui-menu)" )
|
|
.addClass( "ui-menu ui-widget ui-widget-content ui-front" )
|
|
.hide()
|
|
.attr({
|
|
role: this.options.role,
|
|
"aria-hidden": "true",
|
|
"aria-expanded": "false"
|
|
})
|
|
.each(function() {
|
|
var menu = $( this ),
|
|
item = menu.parent(),
|
|
submenuCarat = $( "<span>" )
|
|
.addClass( "ui-menu-icon ui-icon " + icon )
|
|
.data( "ui-menu-submenu-carat", true );
|
|
|
|
item
|
|
.attr( "aria-haspopup", "true" )
|
|
.prepend( submenuCarat );
|
|
menu.attr( "aria-labelledby", item.attr( "id" ) );
|
|
});
|
|
|
|
menus = submenus.add( this.element );
|
|
items = menus.find( this.options.items );
|
|
|
|
// Initialize menu-items containing spaces and/or dashes only as dividers
|
|
items.not( ".ui-menu-item" ).each(function() {
|
|
var item = $( this );
|
|
if ( that._isDivider( item ) ) {
|
|
item.addClass( "ui-widget-content ui-menu-divider" );
|
|
}
|
|
});
|
|
|
|
// Don't refresh list items that are already adapted
|
|
items.not( ".ui-menu-item, .ui-menu-divider" )
|
|
.addClass( "ui-menu-item" )
|
|
.uniqueId()
|
|
.attr({
|
|
tabIndex: -1,
|
|
role: this._itemRole()
|
|
});
|
|
|
|
// Add aria-disabled attribute to any disabled menu item
|
|
items.filter( ".ui-state-disabled" ).attr( "aria-disabled", "true" );
|
|
|
|
// If the active item has been removed, blur the menu
|
|
if ( this.active && !$.contains( this.element[ 0 ], this.active[ 0 ] ) ) {
|
|
this.blur();
|
|
}
|
|
},
|
|
|
|
_itemRole: function() {
|
|
return {
|
|
menu: "menuitem",
|
|
listbox: "option"
|
|
}[ this.options.role ];
|
|
},
|
|
|
|
_setOption: function( key, value ) {
|
|
if ( key === "icons" ) {
|
|
this.element.find( ".ui-menu-icon" )
|
|
.removeClass( this.options.icons.submenu )
|
|
.addClass( value.submenu );
|
|
}
|
|
if ( key === "disabled" ) {
|
|
this.element
|
|
.toggleClass( "ui-state-disabled", !!value )
|
|
.attr( "aria-disabled", value );
|
|
}
|
|
this._super( key, value );
|
|
},
|
|
|
|
focus: function( event, item ) {
|
|
var nested, focused;
|
|
this.blur( event, event && event.type === "focus" );
|
|
|
|
this._scrollIntoView( item );
|
|
|
|
this.active = item.first();
|
|
focused = this.active.addClass( "ui-state-focus" ).removeClass( "ui-state-active" );
|
|
// Only update aria-activedescendant if there's a role
|
|
// otherwise we assume focus is managed elsewhere
|
|
if ( this.options.role ) {
|
|
this.element.attr( "aria-activedescendant", focused.attr( "id" ) );
|
|
}
|
|
|
|
// Highlight active parent menu item, if any
|
|
this.active
|
|
.parent()
|
|
.closest( ".ui-menu-item" )
|
|
.addClass( "ui-state-active" );
|
|
|
|
if ( event && event.type === "keydown" ) {
|
|
this._close();
|
|
} else {
|
|
this.timer = this._delay(function() {
|
|
this._close();
|
|
}, this.delay );
|
|
}
|
|
|
|
nested = item.children( ".ui-menu" );
|
|
if ( nested.length && event && ( /^mouse/.test( event.type ) ) ) {
|
|
this._startOpening(nested);
|
|
}
|
|
this.activeMenu = item.parent();
|
|
|
|
this._trigger( "focus", event, { item: item } );
|
|
},
|
|
|
|
_scrollIntoView: function( item ) {
|
|
var borderTop, paddingTop, offset, scroll, elementHeight, itemHeight;
|
|
if ( this._hasScroll() ) {
|
|
borderTop = parseFloat( $.css( this.activeMenu[0], "borderTopWidth" ) ) || 0;
|
|
paddingTop = parseFloat( $.css( this.activeMenu[0], "paddingTop" ) ) || 0;
|
|
offset = item.offset().top - this.activeMenu.offset().top - borderTop - paddingTop;
|
|
scroll = this.activeMenu.scrollTop();
|
|
elementHeight = this.activeMenu.height();
|
|
itemHeight = item.outerHeight();
|
|
|
|
if ( offset < 0 ) {
|
|
this.activeMenu.scrollTop( scroll + offset );
|
|
} else if ( offset + itemHeight > elementHeight ) {
|
|
this.activeMenu.scrollTop( scroll + offset - elementHeight + itemHeight );
|
|
}
|
|
}
|
|
},
|
|
|
|
blur: function( event, fromFocus ) {
|
|
if ( !fromFocus ) {
|
|
clearTimeout( this.timer );
|
|
}
|
|
|
|
if ( !this.active ) {
|
|
return;
|
|
}
|
|
|
|
this.active.removeClass( "ui-state-focus" );
|
|
this.active = null;
|
|
|
|
this._trigger( "blur", event, { item: this.active } );
|
|
},
|
|
|
|
_startOpening: function( submenu ) {
|
|
clearTimeout( this.timer );
|
|
|
|
// Don't open if already open fixes a Firefox bug that caused a .5 pixel
|
|
// shift in the submenu position when mousing over the carat icon
|
|
if ( submenu.attr( "aria-hidden" ) !== "true" ) {
|
|
return;
|
|
}
|
|
|
|
this.timer = this._delay(function() {
|
|
this._close();
|
|
this._open( submenu );
|
|
}, this.delay );
|
|
},
|
|
|
|
_open: function( submenu ) {
|
|
var position = $.extend({
|
|
of: this.active
|
|
}, this.options.position );
|
|
|
|
clearTimeout( this.timer );
|
|
this.element.find( ".ui-menu" ).not( submenu.parents( ".ui-menu" ) )
|
|
.hide()
|
|
.attr( "aria-hidden", "true" );
|
|
|
|
submenu
|
|
.show()
|
|
.removeAttr( "aria-hidden" )
|
|
.attr( "aria-expanded", "true" )
|
|
.position( position );
|
|
},
|
|
|
|
collapseAll: function( event, all ) {
|
|
clearTimeout( this.timer );
|
|
this.timer = this._delay(function() {
|
|
// If we were passed an event, look for the submenu that contains the event
|
|
var currentMenu = all ? this.element :
|
|
$( event && event.target ).closest( this.element.find( ".ui-menu" ) );
|
|
|
|
// If we found no valid submenu ancestor, use the main menu to close all sub menus anyway
|
|
if ( !currentMenu.length ) {
|
|
currentMenu = this.element;
|
|
}
|
|
|
|
this._close( currentMenu );
|
|
|
|
this.blur( event );
|
|
this.activeMenu = currentMenu;
|
|
}, this.delay );
|
|
},
|
|
|
|
// With no arguments, closes the currently active menu - if nothing is active
|
|
// it closes all menus. If passed an argument, it will search for menus BELOW
|
|
_close: function( startMenu ) {
|
|
if ( !startMenu ) {
|
|
startMenu = this.active ? this.active.parent() : this.element;
|
|
}
|
|
|
|
startMenu
|
|
.find( ".ui-menu" )
|
|
.hide()
|
|
.attr( "aria-hidden", "true" )
|
|
.attr( "aria-expanded", "false" )
|
|
.end()
|
|
.find( ".ui-state-active" ).not( ".ui-state-focus" )
|
|
.removeClass( "ui-state-active" );
|
|
},
|
|
|
|
_closeOnDocumentClick: function( event ) {
|
|
return !$( event.target ).closest( ".ui-menu" ).length;
|
|
},
|
|
|
|
_isDivider: function( item ) {
|
|
|
|
// Match hyphen, em dash, en dash
|
|
return !/[^\-\u2014\u2013\s]/.test( item.text() );
|
|
},
|
|
|
|
collapse: function( event ) {
|
|
var newItem = this.active &&
|
|
this.active.parent().closest( ".ui-menu-item", this.element );
|
|
if ( newItem && newItem.length ) {
|
|
this._close();
|
|
this.focus( event, newItem );
|
|
}
|
|
},
|
|
|
|
expand: function( event ) {
|
|
var newItem = this.active &&
|
|
this.active
|
|
.children( ".ui-menu " )
|
|
.find( this.options.items )
|
|
.first();
|
|
|
|
if ( newItem && newItem.length ) {
|
|
this._open( newItem.parent() );
|
|
|
|
// Delay so Firefox will not hide activedescendant change in expanding submenu from AT
|
|
this._delay(function() {
|
|
this.focus( event, newItem );
|
|
});
|
|
}
|
|
},
|
|
|
|
next: function( event ) {
|
|
this._move( "next", "first", event );
|
|
},
|
|
|
|
previous: function( event ) {
|
|
this._move( "prev", "last", event );
|
|
},
|
|
|
|
isFirstItem: function() {
|
|
return this.active && !this.active.prevAll( ".ui-menu-item" ).length;
|
|
},
|
|
|
|
isLastItem: function() {
|
|
return this.active && !this.active.nextAll( ".ui-menu-item" ).length;
|
|
},
|
|
|
|
_move: function( direction, filter, event ) {
|
|
var next;
|
|
if ( this.active ) {
|
|
if ( direction === "first" || direction === "last" ) {
|
|
next = this.active
|
|
[ direction === "first" ? "prevAll" : "nextAll" ]( ".ui-menu-item" )
|
|
.eq( -1 );
|
|
} else {
|
|
next = this.active
|
|
[ direction + "All" ]( ".ui-menu-item" )
|
|
.eq( 0 );
|
|
}
|
|
}
|
|
if ( !next || !next.length || !this.active ) {
|
|
next = this.activeMenu.find( this.options.items )[ filter ]();
|
|
}
|
|
|
|
this.focus( event, next );
|
|
},
|
|
|
|
nextPage: function( event ) {
|
|
var item, base, height;
|
|
|
|
if ( !this.active ) {
|
|
this.next( event );
|
|
return;
|
|
}
|
|
if ( this.isLastItem() ) {
|
|
return;
|
|
}
|
|
if ( this._hasScroll() ) {
|
|
base = this.active.offset().top;
|
|
height = this.element.height();
|
|
this.active.nextAll( ".ui-menu-item" ).each(function() {
|
|
item = $( this );
|
|
return item.offset().top - base - height < 0;
|
|
});
|
|
|
|
this.focus( event, item );
|
|
} else {
|
|
this.focus( event, this.activeMenu.find( this.options.items )
|
|
[ !this.active ? "first" : "last" ]() );
|
|
}
|
|
},
|
|
|
|
previousPage: function( event ) {
|
|
var item, base, height;
|
|
if ( !this.active ) {
|
|
this.next( event );
|
|
return;
|
|
}
|
|
if ( this.isFirstItem() ) {
|
|
return;
|
|
}
|
|
if ( this._hasScroll() ) {
|
|
base = this.active.offset().top;
|
|
height = this.element.height();
|
|
this.active.prevAll( ".ui-menu-item" ).each(function() {
|
|
item = $( this );
|
|
return item.offset().top - base + height > 0;
|
|
});
|
|
|
|
this.focus( event, item );
|
|
} else {
|
|
this.focus( event, this.activeMenu.find( this.options.items ).first() );
|
|
}
|
|
},
|
|
|
|
_hasScroll: function() {
|
|
return this.element.outerHeight() < this.element.prop( "scrollHeight" );
|
|
},
|
|
|
|
select: function( event ) {
|
|
// TODO: It should never be possible to not have an active item at this
|
|
// point, but the tests don't trigger mouseenter before click.
|
|
this.active = this.active || $( event.target ).closest( ".ui-menu-item" );
|
|
var ui = { item: this.active };
|
|
if ( !this.active.has( ".ui-menu" ).length ) {
|
|
this.collapseAll( event, true );
|
|
}
|
|
this._trigger( "select", event, ui );
|
|
},
|
|
|
|
_filterMenuItems: function(character) {
|
|
var escapedCharacter = character.replace( /[\-\[\]{}()*+?.,\\\^$|#\s]/g, "\\$&" ),
|
|
regex = new RegExp( "^" + escapedCharacter, "i" );
|
|
|
|
return this.activeMenu
|
|
.find( this.options.items )
|
|
|
|
// Only match on items, not dividers or other content (#10571)
|
|
.filter( ".ui-menu-item" )
|
|
.filter(function() {
|
|
return regex.test( $.trim( $( this ).text() ) );
|
|
});
|
|
}
|
|
});
|
|
|
|
|
|
/*!
|
|
* jQuery UI Autocomplete 1.11.4
|
|
* http://jqueryui.com
|
|
*
|
|
* Copyright jQuery Foundation and other contributors
|
|
* Released under the MIT license.
|
|
* http://jquery.org/license
|
|
*
|
|
* http://api.jqueryui.com/autocomplete/
|
|
*/
|
|
|
|
|
|
$.widget( "ui.autocomplete", {
|
|
version: "1.11.4",
|
|
defaultElement: "<input>",
|
|
options: {
|
|
appendTo: null,
|
|
autoFocus: false,
|
|
delay: 300,
|
|
minLength: 1,
|
|
position: {
|
|
my: "left top",
|
|
at: "left bottom",
|
|
collision: "none"
|
|
},
|
|
source: null,
|
|
|
|
// callbacks
|
|
change: null,
|
|
close: null,
|
|
focus: null,
|
|
open: null,
|
|
response: null,
|
|
search: null,
|
|
select: null
|
|
},
|
|
|
|
requestIndex: 0,
|
|
pending: 0,
|
|
|
|
_create: function() {
|
|
// Some browsers only repeat keydown events, not keypress events,
|
|
// so we use the suppressKeyPress flag to determine if we've already
|
|
// handled the keydown event. #7269
|
|
// Unfortunately the code for & in keypress is the same as the up arrow,
|
|
// so we use the suppressKeyPressRepeat flag to avoid handling keypress
|
|
// events when we know the keydown event was used to modify the
|
|
// search term. #7799
|
|
var suppressKeyPress, suppressKeyPressRepeat, suppressInput,
|
|
nodeName = this.element[ 0 ].nodeName.toLowerCase(),
|
|
isTextarea = nodeName === "textarea",
|
|
isInput = nodeName === "input";
|
|
|
|
this.isMultiLine =
|
|
// Textareas are always multi-line
|
|
isTextarea ? true :
|
|
// Inputs are always single-line, even if inside a contentEditable element
|
|
// IE also treats inputs as contentEditable
|
|
isInput ? false :
|
|
// All other element types are determined by whether or not they're contentEditable
|
|
this.element.prop( "isContentEditable" );
|
|
|
|
this.valueMethod = this.element[ isTextarea || isInput ? "val" : "text" ];
|
|
this.isNewMenu = true;
|
|
|
|
this.element
|
|
.addClass( "ui-autocomplete-input" )
|
|
.attr( "autocomplete", "off" );
|
|
|
|
this._on( this.element, {
|
|
keydown: function( event ) {
|
|
if ( this.element.prop( "readOnly" ) ) {
|
|
suppressKeyPress = true;
|
|
suppressInput = true;
|
|
suppressKeyPressRepeat = true;
|
|
return;
|
|
}
|
|
|
|
suppressKeyPress = false;
|
|
suppressInput = false;
|
|
suppressKeyPressRepeat = false;
|
|
var keyCode = $.ui.keyCode;
|
|
switch ( event.keyCode ) {
|
|
case keyCode.PAGE_UP:
|
|
suppressKeyPress = true;
|
|
this._move( "previousPage", event );
|
|
break;
|
|
case keyCode.PAGE_DOWN:
|
|
suppressKeyPress = true;
|
|
this._move( "nextPage", event );
|
|
break;
|
|
case keyCode.UP:
|
|
suppressKeyPress = true;
|
|
this._keyEvent( "previous", event );
|
|
break;
|
|
case keyCode.DOWN:
|
|
suppressKeyPress = true;
|
|
this._keyEvent( "next", event );
|
|
break;
|
|
case keyCode.ENTER:
|
|
// when menu is open and has focus
|
|
if ( this.menu.active ) {
|
|
// #6055 - Opera still allows the keypress to occur
|
|
// which causes forms to submit
|
|
suppressKeyPress = true;
|
|
event.preventDefault();
|
|
this.menu.select( event );
|
|
}
|
|
break;
|
|
case keyCode.TAB:
|
|
if ( this.menu.active ) {
|
|
this.menu.select( event );
|
|
}
|
|
break;
|
|
case keyCode.ESCAPE:
|
|
if ( this.menu.element.is( ":visible" ) ) {
|
|
if ( !this.isMultiLine ) {
|
|
this._value( this.term );
|
|
}
|
|
this.close( event );
|
|
// Different browsers have different default behavior for escape
|
|
// Single press can mean undo or clear
|
|
// Double press in IE means clear the whole form
|
|
event.preventDefault();
|
|
}
|
|
break;
|
|
default:
|
|
suppressKeyPressRepeat = true;
|
|
// search timeout should be triggered before the input value is changed
|
|
this._searchTimeout( event );
|
|
break;
|
|
}
|
|
},
|
|
keypress: function( event ) {
|
|
if ( suppressKeyPress ) {
|
|
suppressKeyPress = false;
|
|
if ( !this.isMultiLine || this.menu.element.is( ":visible" ) ) {
|
|
event.preventDefault();
|
|
}
|
|
return;
|
|
}
|
|
if ( suppressKeyPressRepeat ) {
|
|
return;
|
|
}
|
|
|
|
// replicate some key handlers to allow them to repeat in Firefox and Opera
|
|
var keyCode = $.ui.keyCode;
|
|
switch ( event.keyCode ) {
|
|
case keyCode.PAGE_UP:
|
|
this._move( "previousPage", event );
|
|
break;
|
|
case keyCode.PAGE_DOWN:
|
|
this._move( "nextPage", event );
|
|
break;
|
|
case keyCode.UP:
|
|
this._keyEvent( "previous", event );
|
|
break;
|
|
case keyCode.DOWN:
|
|
this._keyEvent( "next", event );
|
|
break;
|
|
}
|
|
},
|
|
input: function( event ) {
|
|
if ( suppressInput ) {
|
|
suppressInput = false;
|
|
event.preventDefault();
|
|
return;
|
|
}
|
|
this._searchTimeout( event );
|
|
},
|
|
focus: function() {
|
|
this.selectedItem = null;
|
|
this.previous = this._value();
|
|
},
|
|
blur: function( event ) {
|
|
if ( this.cancelBlur ) {
|
|
delete this.cancelBlur;
|
|
return;
|
|
}
|
|
|
|
clearTimeout( this.searching );
|
|
this.close( event );
|
|
this._change( event );
|
|
}
|
|
});
|
|
|
|
this._initSource();
|
|
this.menu = $( "<ul>" )
|
|
.addClass( "ui-autocomplete ui-front" )
|
|
.appendTo( this._appendTo() )
|
|
.menu({
|
|
// disable ARIA support, the live region takes care of that
|
|
role: null
|
|
})
|
|
.hide()
|
|
.menu( "instance" );
|
|
|
|
this._on( this.menu.element, {
|
|
mousedown: function( event ) {
|
|
// prevent moving focus out of the text field
|
|
event.preventDefault();
|
|
|
|
// IE doesn't prevent moving focus even with event.preventDefault()
|
|
// so we set a flag to know when we should ignore the blur event
|
|
this.cancelBlur = true;
|
|
this._delay(function() {
|
|
delete this.cancelBlur;
|
|
});
|
|
|
|
// clicking on the scrollbar causes focus to shift to the body
|
|
// but we can't detect a mouseup or a click immediately afterward
|
|
// so we have to track the next mousedown and close the menu if
|
|
// the user clicks somewhere outside of the autocomplete
|
|
var menuElement = this.menu.element[ 0 ];
|
|
if ( !$( event.target ).closest( ".ui-menu-item" ).length ) {
|
|
this._delay(function() {
|
|
var that = this;
|
|
this.document.one( "mousedown", function( event ) {
|
|
if ( event.target !== that.element[ 0 ] &&
|
|
event.target !== menuElement &&
|
|
!$.contains( menuElement, event.target ) ) {
|
|
that.close();
|
|
}
|
|
});
|
|
});
|
|
}
|
|
},
|
|
menufocus: function( event, ui ) {
|
|
var label, item;
|
|
// support: Firefox
|
|
// Prevent accidental activation of menu items in Firefox (#7024 #9118)
|
|
if ( this.isNewMenu ) {
|
|
this.isNewMenu = false;
|
|
if ( event.originalEvent && /^mouse/.test( event.originalEvent.type ) ) {
|
|
this.menu.blur();
|
|
|
|
this.document.one( "mousemove", function() {
|
|
$( event.target ).trigger( event.originalEvent );
|
|
});
|
|
|
|
return;
|
|
}
|
|
}
|
|
|
|
item = ui.item.data( "ui-autocomplete-item" );
|
|
if ( false !== this._trigger( "focus", event, { item: item } ) ) {
|
|
// use value to match what will end up in the input, if it was a key event
|
|
if ( event.originalEvent && /^key/.test( event.originalEvent.type ) ) {
|
|
this._value( item.value );
|
|
}
|
|
}
|
|
|
|
// Announce the value in the liveRegion
|
|
label = ui.item.attr( "aria-label" ) || item.value;
|
|
if ( label && $.trim( label ).length ) {
|
|
this.liveRegion.children().hide();
|
|
$( "<div>" ).text( label ).appendTo( this.liveRegion );
|
|
}
|
|
},
|
|
menuselect: function( event, ui ) {
|
|
var item = ui.item.data( "ui-autocomplete-item" ),
|
|
previous = this.previous;
|
|
|
|
// only trigger when focus was lost (click on menu)
|
|
if ( this.element[ 0 ] !== this.document[ 0 ].activeElement ) {
|
|
this.element.focus();
|
|
this.previous = previous;
|
|
// #6109 - IE triggers two focus events and the second
|
|
// is asynchronous, so we need to reset the previous
|
|
// term synchronously and asynchronously :-(
|
|
this._delay(function() {
|
|
this.previous = previous;
|
|
this.selectedItem = item;
|
|
});
|
|
}
|
|
|
|
if ( false !== this._trigger( "select", event, { item: item } ) ) {
|
|
this._value( item.value );
|
|
}
|
|
// reset the term after the select event
|
|
// this allows custom select handling to work properly
|
|
this.term = this._value();
|
|
|
|
this.close( event );
|
|
this.selectedItem = item;
|
|
}
|
|
});
|
|
|
|
this.liveRegion = $( "<span>", {
|
|
role: "status",
|
|
"aria-live": "assertive",
|
|
"aria-relevant": "additions"
|
|
})
|
|
.addClass( "ui-helper-hidden-accessible" )
|
|
.appendTo( this.document[ 0 ].body );
|
|
|
|
// turning off autocomplete prevents the browser from remembering the
|
|
// value when navigating through history, so we re-enable autocomplete
|
|
// if the page is unloaded before the widget is destroyed. #7790
|
|
this._on( this.window, {
|
|
beforeunload: function() {
|
|
this.element.removeAttr( "autocomplete" );
|
|
}
|
|
});
|
|
},
|
|
|
|
_destroy: function() {
|
|
clearTimeout( this.searching );
|
|
this.element
|
|
.removeClass( "ui-autocomplete-input" )
|
|
.removeAttr( "autocomplete" );
|
|
this.menu.element.remove();
|
|
this.liveRegion.remove();
|
|
},
|
|
|
|
_setOption: function( key, value ) {
|
|
this._super( key, value );
|
|
if ( key === "source" ) {
|
|
this._initSource();
|
|
}
|
|
if ( key === "appendTo" ) {
|
|
this.menu.element.appendTo( this._appendTo() );
|
|
}
|
|
if ( key === "disabled" && value && this.xhr ) {
|
|
this.xhr.abort();
|
|
}
|
|
},
|
|
|
|
_appendTo: function() {
|
|
var element = this.options.appendTo;
|
|
|
|
if ( element ) {
|
|
element = element.jquery || element.nodeType ?
|
|
$( element ) :
|
|
this.document.find( element ).eq( 0 );
|
|
}
|
|
|
|
if ( !element || !element[ 0 ] ) {
|
|
element = this.element.closest( ".ui-front" );
|
|
}
|
|
|
|
if ( !element.length ) {
|
|
element = this.document[ 0 ].body;
|
|
}
|
|
|
|
return element;
|
|
},
|
|
|
|
_initSource: function() {
|
|
var array, url,
|
|
that = this;
|
|
if ( $.isArray( this.options.source ) ) {
|
|
array = this.options.source;
|
|
this.source = function( request, response ) {
|
|
response( $.ui.autocomplete.filter( array, request.term ) );
|
|
};
|
|
} else if ( typeof this.options.source === "string" ) {
|
|
url = this.options.source;
|
|
this.source = function( request, response ) {
|
|
if ( that.xhr ) {
|
|
that.xhr.abort();
|
|
}
|
|
that.xhr = $.ajax({
|
|
url: url,
|
|
data: request,
|
|
dataType: "json",
|
|
success: function( data ) {
|
|
response( data );
|
|
},
|
|
error: function() {
|
|
response([]);
|
|
}
|
|
});
|
|
};
|
|
} else {
|
|
this.source = this.options.source;
|
|
}
|
|
},
|
|
|
|
_searchTimeout: function( event ) {
|
|
clearTimeout( this.searching );
|
|
this.searching = this._delay(function() {
|
|
|
|
// Search if the value has changed, or if the user retypes the same value (see #7434)
|
|
var equalValues = this.term === this._value(),
|
|
menuVisible = this.menu.element.is( ":visible" ),
|
|
modifierKey = event.altKey || event.ctrlKey || event.metaKey || event.shiftKey;
|
|
|
|
if ( !equalValues || ( equalValues && !menuVisible && !modifierKey ) ) {
|
|
this.selectedItem = null;
|
|
this.search( null, event );
|
|
}
|
|
}, this.options.delay );
|
|
},
|
|
|
|
search: function( value, event ) {
|
|
value = value != null ? value : this._value();
|
|
|
|
// always save the actual value, not the one passed as an argument
|
|
this.term = this._value();
|
|
|
|
if ( value.length < this.options.minLength ) {
|
|
return this.close( event );
|
|
}
|
|
|
|
if ( this._trigger( "search", event ) === false ) {
|
|
return;
|
|
}
|
|
|
|
return this._search( value );
|
|
},
|
|
|
|
_search: function( value ) {
|
|
this.pending++;
|
|
this.element.addClass( "ui-autocomplete-loading" );
|
|
this.cancelSearch = false;
|
|
|
|
this.source( { term: value }, this._response() );
|
|
},
|
|
|
|
_response: function() {
|
|
var index = ++this.requestIndex;
|
|
|
|
return $.proxy(function( content ) {
|
|
if ( index === this.requestIndex ) {
|
|
this.__response( content );
|
|
}
|
|
|
|
this.pending--;
|
|
if ( !this.pending ) {
|
|
this.element.removeClass( "ui-autocomplete-loading" );
|
|
}
|
|
}, this );
|
|
},
|
|
|
|
__response: function( content ) {
|
|
if ( content ) {
|
|
content = this._normalize( content );
|
|
}
|
|
this._trigger( "response", null, { content: content } );
|
|
if ( !this.options.disabled && content && content.length && !this.cancelSearch ) {
|
|
this._suggest( content );
|
|
this._trigger( "open" );
|
|
} else {
|
|
// use ._close() instead of .close() so we don't cancel future searches
|
|
this._close();
|
|
}
|
|
},
|
|
|
|
close: function( event ) {
|
|
this.cancelSearch = true;
|
|
this._close( event );
|
|
},
|
|
|
|
_close: function( event ) {
|
|
if ( this.menu.element.is( ":visible" ) ) {
|
|
this.menu.element.hide();
|
|
this.menu.blur();
|
|
this.isNewMenu = true;
|
|
this._trigger( "close", event );
|
|
}
|
|
},
|
|
|
|
_change: function( event ) {
|
|
if ( this.previous !== this._value() ) {
|
|
this._trigger( "change", event, { item: this.selectedItem } );
|
|
}
|
|
},
|
|
|
|
_normalize: function( items ) {
|
|
// assume all items have the right format when the first item is complete
|
|
if ( items.length && items[ 0 ].label && items[ 0 ].value ) {
|
|
return items;
|
|
}
|
|
return $.map( items, function( item ) {
|
|
if ( typeof item === "string" ) {
|
|
return {
|
|
label: item,
|
|
value: item
|
|
};
|
|
}
|
|
return $.extend( {}, item, {
|
|
label: item.label || item.value,
|
|
value: item.value || item.label
|
|
});
|
|
});
|
|
},
|
|
|
|
_suggest: function( items ) {
|
|
var ul = this.menu.element.empty();
|
|
this._renderMenu( ul, items );
|
|
this.isNewMenu = true;
|
|
this.menu.refresh();
|
|
|
|
// size and position menu
|
|
ul.show();
|
|
this._resizeMenu();
|
|
ul.position( $.extend({
|
|
of: this.element
|
|
}, this.options.position ) );
|
|
|
|
if ( this.options.autoFocus ) {
|
|
this.menu.next();
|
|
}
|
|
},
|
|
|
|
_resizeMenu: function() {
|
|
var ul = this.menu.element;
|
|
ul.outerWidth( Math.max(
|
|
// Firefox wraps long text (possibly a rounding bug)
|
|
// so we add 1px to avoid the wrapping (#7513)
|
|
ul.width( "" ).outerWidth() + 1,
|
|
this.element.outerWidth()
|
|
) );
|
|
},
|
|
|
|
_renderMenu: function( ul, items ) {
|
|
var that = this;
|
|
$.each( items, function( index, item ) {
|
|
that._renderItemData( ul, item );
|
|
});
|
|
},
|
|
|
|
_renderItemData: function( ul, item ) {
|
|
return this._renderItem( ul, item ).data( "ui-autocomplete-item", item );
|
|
},
|
|
|
|
_renderItem: function( ul, item ) {
|
|
return $( "<li>" ).text( item.label ).appendTo( ul );
|
|
},
|
|
|
|
_move: function( direction, event ) {
|
|
if ( !this.menu.element.is( ":visible" ) ) {
|
|
this.search( null, event );
|
|
return;
|
|
}
|
|
if ( this.menu.isFirstItem() && /^previous/.test( direction ) ||
|
|
this.menu.isLastItem() && /^next/.test( direction ) ) {
|
|
|
|
if ( !this.isMultiLine ) {
|
|
this._value( this.term );
|
|
}
|
|
|
|
this.menu.blur();
|
|
return;
|
|
}
|
|
this.menu[ direction ]( event );
|
|
},
|
|
|
|
widget: function() {
|
|
return this.menu.element;
|
|
},
|
|
|
|
_value: function() {
|
|
return this.valueMethod.apply( this.element, arguments );
|
|
},
|
|
|
|
_keyEvent: function( keyEvent, event ) {
|
|
if ( !this.isMultiLine || this.menu.element.is( ":visible" ) ) {
|
|
this._move( keyEvent, event );
|
|
|
|
// prevents moving cursor to beginning/end of the text field in some browsers
|
|
event.preventDefault();
|
|
}
|
|
}
|
|
});
|
|
|
|
$.extend( $.ui.autocomplete, {
|
|
escapeRegex: function( value ) {
|
|
return value.replace( /[\-\[\]{}()*+?.,\\\^$|#\s]/g, "\\$&" );
|
|
},
|
|
filter: function( array, term ) {
|
|
var matcher = new RegExp( $.ui.autocomplete.escapeRegex( term ), "i" );
|
|
return $.grep( array, function( value ) {
|
|
return matcher.test( value.label || value.value || value );
|
|
});
|
|
}
|
|
});
|
|
|
|
// live region extension, adding a `messages` option
|
|
// NOTE: This is an experimental API. We are still investigating
|
|
// a full solution for string manipulation and internationalization.
|
|
$.widget( "ui.autocomplete", $.ui.autocomplete, {
|
|
options: {
|
|
messages: {
|
|
noResults: "No search results.",
|
|
results: function( amount ) {
|
|
return amount + ( amount > 1 ? " results are" : " result is" ) +
|
|
" available, use up and down arrow keys to navigate.";
|
|
}
|
|
}
|
|
},
|
|
|
|
__response: function( content ) {
|
|
var message;
|
|
this._superApply( arguments );
|
|
if ( this.options.disabled || this.cancelSearch ) {
|
|
return;
|
|
}
|
|
if ( content && content.length ) {
|
|
message = this.options.messages.results( content.length );
|
|
} else {
|
|
message = this.options.messages.noResults;
|
|
}
|
|
this.liveRegion.children().hide();
|
|
$( "<div>" ).text( message ).appendTo( this.liveRegion );
|
|
}
|
|
});
|
|
|
|
var autocomplete = $.ui.autocomplete;
|
|
|
|
|
|
/*!
|
|
* jQuery UI Button 1.11.4
|
|
* http://jqueryui.com
|
|
*
|
|
* Copyright jQuery Foundation and other contributors
|
|
* Released under the MIT license.
|
|
* http://jquery.org/license
|
|
*
|
|
* http://api.jqueryui.com/button/
|
|
*/
|
|
|
|
|
|
var lastActive,
|
|
baseClasses = "ui-button ui-widget ui-state-default ui-corner-all",
|
|
typeClasses = "ui-button-icons-only ui-button-icon-only ui-button-text-icons ui-button-text-icon-primary ui-button-text-icon-secondary ui-button-text-only",
|
|
formResetHandler = function() {
|
|
var form = $( this );
|
|
setTimeout(function() {
|
|
form.find( ":ui-button" ).button( "refresh" );
|
|
}, 1 );
|
|
},
|
|
radioGroup = function( radio ) {
|
|
var name = radio.name,
|
|
form = radio.form,
|
|
radios = $( [] );
|
|
if ( name ) {
|
|
name = name.replace( /'/g, "\\'" );
|
|
if ( form ) {
|
|
radios = $( form ).find( "[name='" + name + "'][type=radio]" );
|
|
} else {
|
|
radios = $( "[name='" + name + "'][type=radio]", radio.ownerDocument )
|
|
.filter(function() {
|
|
return !this.form;
|
|
});
|
|
}
|
|
}
|
|
return radios;
|
|
};
|
|
|
|
$.widget( "ui.button", {
|
|
version: "1.11.4",
|
|
defaultElement: "<button>",
|
|
options: {
|
|
disabled: null,
|
|
text: true,
|
|
label: null,
|
|
icons: {
|
|
primary: null,
|
|
secondary: null
|
|
}
|
|
},
|
|
_create: function() {
|
|
this.element.closest( "form" )
|
|
.unbind( "reset" + this.eventNamespace )
|
|
.bind( "reset" + this.eventNamespace, formResetHandler );
|
|
|
|
if ( typeof this.options.disabled !== "boolean" ) {
|
|
this.options.disabled = !!this.element.prop( "disabled" );
|
|
} else {
|
|
this.element.prop( "disabled", this.options.disabled );
|
|
}
|
|
|
|
this._determineButtonType();
|
|
this.hasTitle = !!this.buttonElement.attr( "title" );
|
|
|
|
var that = this,
|
|
options = this.options,
|
|
toggleButton = this.type === "checkbox" || this.type === "radio",
|
|
activeClass = !toggleButton ? "ui-state-active" : "";
|
|
|
|
if ( options.label === null ) {
|
|
options.label = (this.type === "input" ? this.buttonElement.val() : this.buttonElement.html());
|
|
}
|
|
|
|
this._hoverable( this.buttonElement );
|
|
|
|
this.buttonElement
|
|
.addClass( baseClasses )
|
|
.attr( "role", "button" )
|
|
.bind( "mouseenter" + this.eventNamespace, function() {
|
|
if ( options.disabled ) {
|
|
return;
|
|
}
|
|
if ( this === lastActive ) {
|
|
$( this ).addClass( "ui-state-active" );
|
|
}
|
|
})
|
|
.bind( "mouseleave" + this.eventNamespace, function() {
|
|
if ( options.disabled ) {
|
|
return;
|
|
}
|
|
$( this ).removeClass( activeClass );
|
|
})
|
|
.bind( "click" + this.eventNamespace, function( event ) {
|
|
if ( options.disabled ) {
|
|
event.preventDefault();
|
|
event.stopImmediatePropagation();
|
|
}
|
|
});
|
|
|
|
// Can't use _focusable() because the element that receives focus
|
|
// and the element that gets the ui-state-focus class are different
|
|
this._on({
|
|
focus: function() {
|
|
this.buttonElement.addClass( "ui-state-focus" );
|
|
},
|
|
blur: function() {
|
|
this.buttonElement.removeClass( "ui-state-focus" );
|
|
}
|
|
});
|
|
|
|
if ( toggleButton ) {
|
|
this.element.bind( "change" + this.eventNamespace, function() {
|
|
that.refresh();
|
|
});
|
|
}
|
|
|
|
if ( this.type === "checkbox" ) {
|
|
this.buttonElement.bind( "click" + this.eventNamespace, function() {
|
|
if ( options.disabled ) {
|
|
return false;
|
|
}
|
|
});
|
|
} else if ( this.type === "radio" ) {
|
|
this.buttonElement.bind( "click" + this.eventNamespace, function() {
|
|
if ( options.disabled ) {
|
|
return false;
|
|
}
|
|
$( this ).addClass( "ui-state-active" );
|
|
that.buttonElement.attr( "aria-pressed", "true" );
|
|
|
|
var radio = that.element[ 0 ];
|
|
radioGroup( radio )
|
|
.not( radio )
|
|
.map(function() {
|
|
return $( this ).button( "widget" )[ 0 ];
|
|
})
|
|
.removeClass( "ui-state-active" )
|
|
.attr( "aria-pressed", "false" );
|
|
});
|
|
} else {
|
|
this.buttonElement
|
|
.bind( "mousedown" + this.eventNamespace, function() {
|
|
if ( options.disabled ) {
|
|
return false;
|
|
}
|
|
$( this ).addClass( "ui-state-active" );
|
|
lastActive = this;
|
|
that.document.one( "mouseup", function() {
|
|
lastActive = null;
|
|
});
|
|
})
|
|
.bind( "mouseup" + this.eventNamespace, function() {
|
|
if ( options.disabled ) {
|
|
return false;
|
|
}
|
|
$( this ).removeClass( "ui-state-active" );
|
|
})
|
|
.bind( "keydown" + this.eventNamespace, function(event) {
|
|
if ( options.disabled ) {
|
|
return false;
|
|
}
|
|
if ( event.keyCode === $.ui.keyCode.SPACE || event.keyCode === $.ui.keyCode.ENTER ) {
|
|
$( this ).addClass( "ui-state-active" );
|
|
}
|
|
})
|
|
// see #8559, we bind to blur here in case the button element loses
|
|
// focus between keydown and keyup, it would be left in an "active" state
|
|
.bind( "keyup" + this.eventNamespace + " blur" + this.eventNamespace, function() {
|
|
$( this ).removeClass( "ui-state-active" );
|
|
});
|
|
|
|
if ( this.buttonElement.is("a") ) {
|
|
this.buttonElement.keyup(function(event) {
|
|
if ( event.keyCode === $.ui.keyCode.SPACE ) {
|
|
// TODO pass through original event correctly (just as 2nd argument doesn't work)
|
|
$( this ).click();
|
|
}
|
|
});
|
|
}
|
|
}
|
|
|
|
this._setOption( "disabled", options.disabled );
|
|
this._resetButton();
|
|
},
|
|
|
|
_determineButtonType: function() {
|
|
var ancestor, labelSelector, checked;
|
|
|
|
if ( this.element.is("[type=checkbox]") ) {
|
|
this.type = "checkbox";
|
|
} else if ( this.element.is("[type=radio]") ) {
|
|
this.type = "radio";
|
|
} else if ( this.element.is("input") ) {
|
|
this.type = "input";
|
|
} else {
|
|
this.type = "button";
|
|
}
|
|
|
|
if ( this.type === "checkbox" || this.type === "radio" ) {
|
|
// we don't search against the document in case the element
|
|
// is disconnected from the DOM
|
|
ancestor = this.element.parents().last();
|
|
labelSelector = "label[for='" + this.element.attr("id") + "']";
|
|
this.buttonElement = ancestor.find( labelSelector );
|
|
if ( !this.buttonElement.length ) {
|
|
ancestor = ancestor.length ? ancestor.siblings() : this.element.siblings();
|
|
this.buttonElement = ancestor.filter( labelSelector );
|
|
if ( !this.buttonElement.length ) {
|
|
this.buttonElement = ancestor.find( labelSelector );
|
|
}
|
|
}
|
|
this.element.addClass( "ui-helper-hidden-accessible" );
|
|
|
|
checked = this.element.is( ":checked" );
|
|
if ( checked ) {
|
|
this.buttonElement.addClass( "ui-state-active" );
|
|
}
|
|
this.buttonElement.prop( "aria-pressed", checked );
|
|
} else {
|
|
this.buttonElement = this.element;
|
|
}
|
|
},
|
|
|
|
widget: function() {
|
|
return this.buttonElement;
|
|
},
|
|
|
|
_destroy: function() {
|
|
this.element
|
|
.removeClass( "ui-helper-hidden-accessible" );
|
|
this.buttonElement
|
|
.removeClass( baseClasses + " ui-state-active " + typeClasses )
|
|
.removeAttr( "role" )
|
|
.removeAttr( "aria-pressed" )
|
|
.html( this.buttonElement.find(".ui-button-text").html() );
|
|
|
|
if ( !this.hasTitle ) {
|
|
this.buttonElement.removeAttr( "title" );
|
|
}
|
|
},
|
|
|
|
_setOption: function( key, value ) {
|
|
this._super( key, value );
|
|
if ( key === "disabled" ) {
|
|
this.widget().toggleClass( "ui-state-disabled", !!value );
|
|
this.element.prop( "disabled", !!value );
|
|
if ( value ) {
|
|
if ( this.type === "checkbox" || this.type === "radio" ) {
|
|
this.buttonElement.removeClass( "ui-state-focus" );
|
|
} else {
|
|
this.buttonElement.removeClass( "ui-state-focus ui-state-active" );
|
|
}
|
|
}
|
|
return;
|
|
}
|
|
this._resetButton();
|
|
},
|
|
|
|
refresh: function() {
|
|
//See #8237 & #8828
|
|
var isDisabled = this.element.is( "input, button" ) ? this.element.is( ":disabled" ) : this.element.hasClass( "ui-button-disabled" );
|
|
|
|
if ( isDisabled !== this.options.disabled ) {
|
|
this._setOption( "disabled", isDisabled );
|
|
}
|
|
if ( this.type === "radio" ) {
|
|
radioGroup( this.element[0] ).each(function() {
|
|
if ( $( this ).is( ":checked" ) ) {
|
|
$( this ).button( "widget" )
|
|
.addClass( "ui-state-active" )
|
|
.attr( "aria-pressed", "true" );
|
|
} else {
|
|
$( this ).button( "widget" )
|
|
.removeClass( "ui-state-active" )
|
|
.attr( "aria-pressed", "false" );
|
|
}
|
|
});
|
|
} else if ( this.type === "checkbox" ) {
|
|
if ( this.element.is( ":checked" ) ) {
|
|
this.buttonElement
|
|
.addClass( "ui-state-active" )
|
|
.attr( "aria-pressed", "true" );
|
|
} else {
|
|
this.buttonElement
|
|
.removeClass( "ui-state-active" )
|
|
.attr( "aria-pressed", "false" );
|
|
}
|
|
}
|
|
},
|
|
|
|
_resetButton: function() {
|
|
if ( this.type === "input" ) {
|
|
if ( this.options.label ) {
|
|
this.element.val( this.options.label );
|
|
}
|
|
return;
|
|
}
|
|
var buttonElement = this.buttonElement.removeClass( typeClasses ),
|
|
buttonText = $( "<span></span>", this.document[0] )
|
|
.addClass( "ui-button-text" )
|
|
.html( this.options.label )
|
|
.appendTo( buttonElement.empty() )
|
|
.text(),
|
|
icons = this.options.icons,
|
|
multipleIcons = icons.primary && icons.secondary,
|
|
buttonClasses = [];
|
|
|
|
if ( icons.primary || icons.secondary ) {
|
|
if ( this.options.text ) {
|
|
buttonClasses.push( "ui-button-text-icon" + ( multipleIcons ? "s" : ( icons.primary ? "-primary" : "-secondary" ) ) );
|
|
}
|
|
|
|
if ( icons.primary ) {
|
|
buttonElement.prepend( "<span class='ui-button-icon-primary ui-icon " + icons.primary + "'></span>" );
|
|
}
|
|
|
|
if ( icons.secondary ) {
|
|
buttonElement.append( "<span class='ui-button-icon-secondary ui-icon " + icons.secondary + "'></span>" );
|
|
}
|
|
|
|
if ( !this.options.text ) {
|
|
buttonClasses.push( multipleIcons ? "ui-button-icons-only" : "ui-button-icon-only" );
|
|
|
|
if ( !this.hasTitle ) {
|
|
buttonElement.attr( "title", $.trim( buttonText ) );
|
|
}
|
|
}
|
|
} else {
|
|
buttonClasses.push( "ui-button-text-only" );
|
|
}
|
|
buttonElement.addClass( buttonClasses.join( " " ) );
|
|
}
|
|
});
|
|
|
|
$.widget( "ui.buttonset", {
|
|
version: "1.11.4",
|
|
options: {
|
|
items: "button, input[type=button], input[type=submit], input[type=reset], input[type=checkbox], input[type=radio], a, :data(ui-button)"
|
|
},
|
|
|
|
_create: function() {
|
|
this.element.addClass( "ui-buttonset" );
|
|
},
|
|
|
|
_init: function() {
|
|
this.refresh();
|
|
},
|
|
|
|
_setOption: function( key, value ) {
|
|
if ( key === "disabled" ) {
|
|
this.buttons.button( "option", key, value );
|
|
}
|
|
|
|
this._super( key, value );
|
|
},
|
|
|
|
refresh: function() {
|
|
var rtl = this.element.css( "direction" ) === "rtl",
|
|
allButtons = this.element.find( this.options.items ),
|
|
existingButtons = allButtons.filter( ":ui-button" );
|
|
|
|
// Initialize new buttons
|
|
allButtons.not( ":ui-button" ).button();
|
|
|
|
// Refresh existing buttons
|
|
existingButtons.button( "refresh" );
|
|
|
|
this.buttons = allButtons
|
|
.map(function() {
|
|
return $( this ).button( "widget" )[ 0 ];
|
|
})
|
|
.removeClass( "ui-corner-all ui-corner-left ui-corner-right" )
|
|
.filter( ":first" )
|
|
.addClass( rtl ? "ui-corner-right" : "ui-corner-left" )
|
|
.end()
|
|
.filter( ":last" )
|
|
.addClass( rtl ? "ui-corner-left" : "ui-corner-right" )
|
|
.end()
|
|
.end();
|
|
},
|
|
|
|
_destroy: function() {
|
|
this.element.removeClass( "ui-buttonset" );
|
|
this.buttons
|
|
.map(function() {
|
|
return $( this ).button( "widget" )[ 0 ];
|
|
})
|
|
.removeClass( "ui-corner-left ui-corner-right" )
|
|
.end()
|
|
.button( "destroy" );
|
|
}
|
|
});
|
|
|
|
var button = $.ui.button;
|
|
|
|
|
|
/*!
|
|
* jQuery UI Datepicker 1.11.4
|
|
* http://jqueryui.com
|
|
*
|
|
* Copyright jQuery Foundation and other contributors
|
|
* Released under the MIT license.
|
|
* http://jquery.org/license
|
|
*
|
|
* http://api.jqueryui.com/datepicker/
|
|
*/
|
|
|
|
|
|
$.extend($.ui, { datepicker: { version: "1.11.4" } });
|
|
|
|
var datepicker_instActive;
|
|
|
|
function datepicker_getZindex( elem ) {
|
|
var position, value;
|
|
while ( elem.length && elem[ 0 ] !== document ) {
|
|
// Ignore z-index if position is set to a value where z-index is ignored by the browser
|
|
// This makes behavior of this function consistent across browsers
|
|
// WebKit always returns auto if the element is positioned
|
|
position = elem.css( "position" );
|
|
if ( position === "absolute" || position === "relative" || position === "fixed" ) {
|
|
// IE returns 0 when zIndex is not specified
|
|
// other browsers return a string
|
|
// we ignore the case of nested elements with an explicit value of 0
|
|
// <div style="z-index: -10;"><div style="z-index: 0;"></div></div>
|
|
value = parseInt( elem.css( "zIndex" ), 10 );
|
|
if ( !isNaN( value ) && value !== 0 ) {
|
|
return value;
|
|
}
|
|
}
|
|
elem = elem.parent();
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
/* Date picker manager.
|
|
Use the singleton instance of this class, $.datepicker, to interact with the date picker.
|
|
Settings for (groups of) date pickers are maintained in an instance object,
|
|
allowing multiple different settings on the same page. */
|
|
|
|
function Datepicker() {
|
|
this._curInst = null; // The current instance in use
|
|
this._keyEvent = false; // If the last event was a key event
|
|
this._disabledInputs = []; // List of date picker inputs that have been disabled
|
|
this._datepickerShowing = false; // True if the popup picker is showing , false if not
|
|
this._inDialog = false; // True if showing within a "dialog", false if not
|
|
this._mainDivId = "ui-datepicker-div"; // The ID of the main datepicker division
|
|
this._inlineClass = "ui-datepicker-inline"; // The name of the inline marker class
|
|
this._appendClass = "ui-datepicker-append"; // The name of the append marker class
|
|
this._triggerClass = "ui-datepicker-trigger"; // The name of the trigger marker class
|
|
this._dialogClass = "ui-datepicker-dialog"; // The name of the dialog marker class
|
|
this._disableClass = "ui-datepicker-disabled"; // The name of the disabled covering marker class
|
|
this._unselectableClass = "ui-datepicker-unselectable"; // The name of the unselectable cell marker class
|
|
this._currentClass = "ui-datepicker-current-day"; // The name of the current day marker class
|
|
this._dayOverClass = "ui-datepicker-days-cell-over"; // The name of the day hover marker class
|
|
this.regional = []; // Available regional settings, indexed by language code
|
|
this.regional[""] = { // Default regional settings
|
|
closeText: "Done", // Display text for close link
|
|
prevText: "Prev", // Display text for previous month link
|
|
nextText: "Next", // Display text for next month link
|
|
currentText: "Today", // Display text for current month link
|
|
monthNames: ["January","February","March","April","May","June",
|
|
"July","August","September","October","November","December"], // Names of months for drop-down and formatting
|
|
monthNamesShort: ["Jan", "Feb", "Mar", "Apr", "May", "Jun", "Jul", "Aug", "Sep", "Oct", "Nov", "Dec"], // For formatting
|
|
dayNames: ["Sunday", "Monday", "Tuesday", "Wednesday", "Thursday", "Friday", "Saturday"], // For formatting
|
|
dayNamesShort: ["Sun", "Mon", "Tue", "Wed", "Thu", "Fri", "Sat"], // For formatting
|
|
dayNamesMin: ["Su","Mo","Tu","We","Th","Fr","Sa"], // Column headings for days starting at Sunday
|
|
weekHeader: "Wk", // Column header for week of the year
|
|
dateFormat: "mm/dd/yy", // See format options on parseDate
|
|
firstDay: 0, // The first day of the week, Sun = 0, Mon = 1, ...
|
|
isRTL: false, // True if right-to-left language, false if left-to-right
|
|
showMonthAfterYear: false, // True if the year select precedes month, false for month then year
|
|
yearSuffix: "" // Additional text to append to the year in the month headers
|
|
};
|
|
this._defaults = { // Global defaults for all the date picker instances
|
|
showOn: "focus", // "focus" for popup on focus,
|
|
// "button" for trigger button, or "both" for either
|
|
showAnim: "fadeIn", // Name of jQuery animation for popup
|
|
showOptions: {}, // Options for enhanced animations
|
|
defaultDate: null, // Used when field is blank: actual date,
|
|
// +/-number for offset from today, null for today
|
|
appendText: "", // Display text following the input box, e.g. showing the format
|
|
buttonText: "...", // Text for trigger button
|
|
buttonImage: "", // URL for trigger button image
|
|
buttonImageOnly: false, // True if the image appears alone, false if it appears on a button
|
|
hideIfNoPrevNext: false, // True to hide next/previous month links
|
|
// if not applicable, false to just disable them
|
|
navigationAsDateFormat: false, // True if date formatting applied to prev/today/next links
|
|
gotoCurrent: false, // True if today link goes back to current selection instead
|
|
changeMonth: false, // True if month can be selected directly, false if only prev/next
|
|
changeYear: false, // True if year can be selected directly, false if only prev/next
|
|
yearRange: "c-10:c+10", // Range of years to display in drop-down,
|
|
// either relative to today's year (-nn:+nn), relative to currently displayed year
|
|
// (c-nn:c+nn), absolute (nnnn:nnnn), or a combination of the above (nnnn:-n)
|
|
showOtherMonths: false, // True to show dates in other months, false to leave blank
|
|
selectOtherMonths: false, // True to allow selection of dates in other months, false for unselectable
|
|
showWeek: false, // True to show week of the year, false to not show it
|
|
calculateWeek: this.iso8601Week, // How to calculate the week of the year,
|
|
// takes a Date and returns the number of the week for it
|
|
shortYearCutoff: "+10", // Short year values < this are in the current century,
|
|
// > this are in the previous century,
|
|
// string value starting with "+" for current year + value
|
|
minDate: null, // The earliest selectable date, or null for no limit
|
|
maxDate: null, // The latest selectable date, or null for no limit
|
|
duration: "fast", // Duration of display/closure
|
|
beforeShowDay: null, // Function that takes a date and returns an array with
|
|
// [0] = true if selectable, false if not, [1] = custom CSS class name(s) or "",
|
|
// [2] = cell title (optional), e.g. $.datepicker.noWeekends
|
|
beforeShow: null, // Function that takes an input field and
|
|
// returns a set of custom settings for the date picker
|
|
onSelect: null, // Define a callback function when a date is selected
|
|
onChangeMonthYear: null, // Define a callback function when the month or year is changed
|
|
onClose: null, // Define a callback function when the datepicker is closed
|
|
numberOfMonths: 1, // Number of months to show at a time
|
|
showCurrentAtPos: 0, // The position in multipe months at which to show the current month (starting at 0)
|
|
stepMonths: 1, // Number of months to step back/forward
|
|
stepBigMonths: 12, // Number of months to step back/forward for the big links
|
|
altField: "", // Selector for an alternate field to store selected dates into
|
|
altFormat: "", // The date format to use for the alternate field
|
|
constrainInput: true, // The input is constrained by the current date format
|
|
showButtonPanel: false, // True to show button panel, false to not show it
|
|
autoSize: false, // True to size the input for the date format, false to leave as is
|
|
disabled: false // The initial disabled state
|
|
};
|
|
$.extend(this._defaults, this.regional[""]);
|
|
this.regional.en = $.extend( true, {}, this.regional[ "" ]);
|
|
this.regional[ "en-US" ] = $.extend( true, {}, this.regional.en );
|
|
this.dpDiv = datepicker_bindHover($("<div id='" + this._mainDivId + "' class='ui-datepicker ui-widget ui-widget-content ui-helper-clearfix ui-corner-all'></div>"));
|
|
}
|
|
|
|
$.extend(Datepicker.prototype, {
|
|
/* Class name added to elements to indicate already configured with a date picker. */
|
|
markerClassName: "hasDatepicker",
|
|
|
|
//Keep track of the maximum number of rows displayed (see #7043)
|
|
maxRows: 4,
|
|
|
|
// TODO rename to "widget" when switching to widget factory
|
|
_widgetDatepicker: function() {
|
|
return this.dpDiv;
|
|
},
|
|
|
|
/* Override the default settings for all instances of the date picker.
|
|
* @param settings object - the new settings to use as defaults (anonymous object)
|
|
* @return the manager object
|
|
*/
|
|
setDefaults: function(settings) {
|
|
datepicker_extendRemove(this._defaults, settings || {});
|
|
return this;
|
|
},
|
|
|
|
/* Attach the date picker to a jQuery selection.
|
|
* @param target element - the target input field or division or span
|
|
* @param settings object - the new settings to use for this date picker instance (anonymous)
|
|
*/
|
|
_attachDatepicker: function(target, settings) {
|
|
var nodeName, inline, inst;
|
|
nodeName = target.nodeName.toLowerCase();
|
|
inline = (nodeName === "div" || nodeName === "span");
|
|
if (!target.id) {
|
|
this.uuid += 1;
|
|
target.id = "dp" + this.uuid;
|
|
}
|
|
inst = this._newInst($(target), inline);
|
|
inst.settings = $.extend({}, settings || {});
|
|
if (nodeName === "input") {
|
|
this._connectDatepicker(target, inst);
|
|
} else if (inline) {
|
|
this._inlineDatepicker(target, inst);
|
|
}
|
|
},
|
|
|
|
/* Create a new instance object. */
|
|
_newInst: function(target, inline) {
|
|
var id = target[0].id.replace(/([^A-Za-z0-9_\-])/g, "\\\\$1"); // escape jQuery meta chars
|
|
return {id: id, input: target, // associated target
|
|
selectedDay: 0, selectedMonth: 0, selectedYear: 0, // current selection
|
|
drawMonth: 0, drawYear: 0, // month being drawn
|
|
inline: inline, // is datepicker inline or not
|
|
dpDiv: (!inline ? this.dpDiv : // presentation div
|
|
datepicker_bindHover($("<div class='" + this._inlineClass + " ui-datepicker ui-widget ui-widget-content ui-helper-clearfix ui-corner-all'></div>")))};
|
|
},
|
|
|
|
/* Attach the date picker to an input field. */
|
|
_connectDatepicker: function(target, inst) {
|
|
var input = $(target);
|
|
inst.append = $([]);
|
|
inst.trigger = $([]);
|
|
if (input.hasClass(this.markerClassName)) {
|
|
return;
|
|
}
|
|
this._attachments(input, inst);
|
|
input.addClass(this.markerClassName).keydown(this._doKeyDown).
|
|
keypress(this._doKeyPress).keyup(this._doKeyUp);
|
|
this._autoSize(inst);
|
|
$.data(target, "datepicker", inst);
|
|
//If disabled option is true, disable the datepicker once it has been attached to the input (see ticket #5665)
|
|
if( inst.settings.disabled ) {
|
|
this._disableDatepicker( target );
|
|
}
|
|
},
|
|
|
|
/* Make attachments based on settings. */
|
|
_attachments: function(input, inst) {
|
|
var showOn, buttonText, buttonImage,
|
|
appendText = this._get(inst, "appendText"),
|
|
isRTL = this._get(inst, "isRTL");
|
|
|
|
if (inst.append) {
|
|
inst.append.remove();
|
|
}
|
|
if (appendText) {
|
|
inst.append = $("<span class='" + this._appendClass + "'>" + appendText + "</span>");
|
|
input[isRTL ? "before" : "after"](inst.append);
|
|
}
|
|
|
|
input.unbind("focus", this._showDatepicker);
|
|
|
|
if (inst.trigger) {
|
|
inst.trigger.remove();
|
|
}
|
|
|
|
showOn = this._get(inst, "showOn");
|
|
if (showOn === "focus" || showOn === "both") { // pop-up date picker when in the marked field
|
|
input.focus(this._showDatepicker);
|
|
}
|
|
if (showOn === "button" || showOn === "both") { // pop-up date picker when button clicked
|
|
buttonText = this._get(inst, "buttonText");
|
|
buttonImage = this._get(inst, "buttonImage");
|
|
inst.trigger = $(this._get(inst, "buttonImageOnly") ?
|
|
$("<img/>").addClass(this._triggerClass).
|
|
attr({ src: buttonImage, alt: buttonText, title: buttonText }) :
|
|
$("<button type='button'></button>").addClass(this._triggerClass).
|
|
html(!buttonImage ? buttonText : $("<img/>").attr(
|
|
{ src:buttonImage, alt:buttonText, title:buttonText })));
|
|
input[isRTL ? "before" : "after"](inst.trigger);
|
|
inst.trigger.click(function() {
|
|
if ($.datepicker._datepickerShowing && $.datepicker._lastInput === input[0]) {
|
|
$.datepicker._hideDatepicker();
|
|
} else if ($.datepicker._datepickerShowing && $.datepicker._lastInput !== input[0]) {
|
|
$.datepicker._hideDatepicker();
|
|
$.datepicker._showDatepicker(input[0]);
|
|
} else {
|
|
$.datepicker._showDatepicker(input[0]);
|
|
}
|
|
return false;
|
|
});
|
|
}
|
|
},
|
|
|
|
/* Apply the maximum length for the date format. */
|
|
_autoSize: function(inst) {
|
|
if (this._get(inst, "autoSize") && !inst.inline) {
|
|
var findMax, max, maxI, i,
|
|
date = new Date(2009, 12 - 1, 20), // Ensure double digits
|
|
dateFormat = this._get(inst, "dateFormat");
|
|
|
|
if (dateFormat.match(/[DM]/)) {
|
|
findMax = function(names) {
|
|
max = 0;
|
|
maxI = 0;
|
|
for (i = 0; i < names.length; i++) {
|
|
if (names[i].length > max) {
|
|
max = names[i].length;
|
|
maxI = i;
|
|
}
|
|
}
|
|
return maxI;
|
|
};
|
|
date.setMonth(findMax(this._get(inst, (dateFormat.match(/MM/) ?
|
|
"monthNames" : "monthNamesShort"))));
|
|
date.setDate(findMax(this._get(inst, (dateFormat.match(/DD/) ?
|
|
"dayNames" : "dayNamesShort"))) + 20 - date.getDay());
|
|
}
|
|
inst.input.attr("size", this._formatDate(inst, date).length);
|
|
}
|
|
},
|
|
|
|
/* Attach an inline date picker to a div. */
|
|
_inlineDatepicker: function(target, inst) {
|
|
var divSpan = $(target);
|
|
if (divSpan.hasClass(this.markerClassName)) {
|
|
return;
|
|
}
|
|
divSpan.addClass(this.markerClassName).append(inst.dpDiv);
|
|
$.data(target, "datepicker", inst);
|
|
this._setDate(inst, this._getDefaultDate(inst), true);
|
|
this._updateDatepicker(inst);
|
|
this._updateAlternate(inst);
|
|
//If disabled option is true, disable the datepicker before showing it (see ticket #5665)
|
|
if( inst.settings.disabled ) {
|
|
this._disableDatepicker( target );
|
|
}
|
|
// Set display:block in place of inst.dpDiv.show() which won't work on disconnected elements
|
|
// http://bugs.jqueryui.com/ticket/7552 - A Datepicker created on a detached div has zero height
|
|
inst.dpDiv.css( "display", "block" );
|
|
},
|
|
|
|
/* Pop-up the date picker in a "dialog" box.
|
|
* @param input element - ignored
|
|
* @param date string or Date - the initial date to display
|
|
* @param onSelect function - the function to call when a date is selected
|
|
* @param settings object - update the dialog date picker instance's settings (anonymous object)
|
|
* @param pos int[2] - coordinates for the dialog's position within the screen or
|
|
* event - with x/y coordinates or
|
|
* leave empty for default (screen centre)
|
|
* @return the manager object
|
|
*/
|
|
_dialogDatepicker: function(input, date, onSelect, settings, pos) {
|
|
var id, browserWidth, browserHeight, scrollX, scrollY,
|
|
inst = this._dialogInst; // internal instance
|
|
|
|
if (!inst) {
|
|
this.uuid += 1;
|
|
id = "dp" + this.uuid;
|
|
this._dialogInput = $("<input type='text' id='" + id +
|
|
"' style='position: absolute; top: -100px; width: 0px;'/>");
|
|
this._dialogInput.keydown(this._doKeyDown);
|
|
$("body").append(this._dialogInput);
|
|
inst = this._dialogInst = this._newInst(this._dialogInput, false);
|
|
inst.settings = {};
|
|
$.data(this._dialogInput[0], "datepicker", inst);
|
|
}
|
|
datepicker_extendRemove(inst.settings, settings || {});
|
|
date = (date && date.constructor === Date ? this._formatDate(inst, date) : date);
|
|
this._dialogInput.val(date);
|
|
|
|
this._pos = (pos ? (pos.length ? pos : [pos.pageX, pos.pageY]) : null);
|
|
if (!this._pos) {
|
|
browserWidth = document.documentElement.clientWidth;
|
|
browserHeight = document.documentElement.clientHeight;
|
|
scrollX = document.documentElement.scrollLeft || document.body.scrollLeft;
|
|
scrollY = document.documentElement.scrollTop || document.body.scrollTop;
|
|
this._pos = // should use actual width/height below
|
|
[(browserWidth / 2) - 100 + scrollX, (browserHeight / 2) - 150 + scrollY];
|
|
}
|
|
|
|
// move input on screen for focus, but hidden behind dialog
|
|
this._dialogInput.css("left", (this._pos[0] + 20) + "px").css("top", this._pos[1] + "px");
|
|
inst.settings.onSelect = onSelect;
|
|
this._inDialog = true;
|
|
this.dpDiv.addClass(this._dialogClass);
|
|
this._showDatepicker(this._dialogInput[0]);
|
|
if ($.blockUI) {
|
|
$.blockUI(this.dpDiv);
|
|
}
|
|
$.data(this._dialogInput[0], "datepicker", inst);
|
|
return this;
|
|
},
|
|
|
|
/* Detach a datepicker from its control.
|
|
* @param target element - the target input field or division or span
|
|
*/
|
|
_destroyDatepicker: function(target) {
|
|
var nodeName,
|
|
$target = $(target),
|
|
inst = $.data(target, "datepicker");
|
|
|
|
if (!$target.hasClass(this.markerClassName)) {
|
|
return;
|
|
}
|
|
|
|
nodeName = target.nodeName.toLowerCase();
|
|
$.removeData(target, "datepicker");
|
|
if (nodeName === "input") {
|
|
inst.append.remove();
|
|
inst.trigger.remove();
|
|
$target.removeClass(this.markerClassName).
|
|
unbind("focus", this._showDatepicker).
|
|
unbind("keydown", this._doKeyDown).
|
|
unbind("keypress", this._doKeyPress).
|
|
unbind("keyup", this._doKeyUp);
|
|
} else if (nodeName === "div" || nodeName === "span") {
|
|
$target.removeClass(this.markerClassName).empty();
|
|
}
|
|
|
|
if ( datepicker_instActive === inst ) {
|
|
datepicker_instActive = null;
|
|
}
|
|
},
|
|
|
|
/* Enable the date picker to a jQuery selection.
|
|
* @param target element - the target input field or division or span
|
|
*/
|
|
_enableDatepicker: function(target) {
|
|
var nodeName, inline,
|
|
$target = $(target),
|
|
inst = $.data(target, "datepicker");
|
|
|
|
if (!$target.hasClass(this.markerClassName)) {
|
|
return;
|
|
}
|
|
|
|
nodeName = target.nodeName.toLowerCase();
|
|
if (nodeName === "input") {
|
|
target.disabled = false;
|
|
inst.trigger.filter("button").
|
|
each(function() { this.disabled = false; }).end().
|
|
filter("img").css({opacity: "1.0", cursor: ""});
|
|
} else if (nodeName === "div" || nodeName === "span") {
|
|
inline = $target.children("." + this._inlineClass);
|
|
inline.children().removeClass("ui-state-disabled");
|
|
inline.find("select.ui-datepicker-month, select.ui-datepicker-year").
|
|
prop("disabled", false);
|
|
}
|
|
this._disabledInputs = $.map(this._disabledInputs,
|
|
function(value) { return (value === target ? null : value); }); // delete entry
|
|
},
|
|
|
|
/* Disable the date picker to a jQuery selection.
|
|
* @param target element - the target input field or division or span
|
|
*/
|
|
_disableDatepicker: function(target) {
|
|
var nodeName, inline,
|
|
$target = $(target),
|
|
inst = $.data(target, "datepicker");
|
|
|
|
if (!$target.hasClass(this.markerClassName)) {
|
|
return;
|
|
}
|
|
|
|
nodeName = target.nodeName.toLowerCase();
|
|
if (nodeName === "input") {
|
|
target.disabled = true;
|
|
inst.trigger.filter("button").
|
|
each(function() { this.disabled = true; }).end().
|
|
filter("img").css({opacity: "0.5", cursor: "default"});
|
|
} else if (nodeName === "div" || nodeName === "span") {
|
|
inline = $target.children("." + this._inlineClass);
|
|
inline.children().addClass("ui-state-disabled");
|
|
inline.find("select.ui-datepicker-month, select.ui-datepicker-year").
|
|
prop("disabled", true);
|
|
}
|
|
this._disabledInputs = $.map(this._disabledInputs,
|
|
function(value) { return (value === target ? null : value); }); // delete entry
|
|
this._disabledInputs[this._disabledInputs.length] = target;
|
|
},
|
|
|
|
/* Is the first field in a jQuery collection disabled as a datepicker?
|
|
* @param target element - the target input field or division or span
|
|
* @return boolean - true if disabled, false if enabled
|
|
*/
|
|
_isDisabledDatepicker: function(target) {
|
|
if (!target) {
|
|
return false;
|
|
}
|
|
for (var i = 0; i < this._disabledInputs.length; i++) {
|
|
if (this._disabledInputs[i] === target) {
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
},
|
|
|
|
/* Retrieve the instance data for the target control.
|
|
* @param target element - the target input field or division or span
|
|
* @return object - the associated instance data
|
|
* @throws error if a jQuery problem getting data
|
|
*/
|
|
_getInst: function(target) {
|
|
try {
|
|
return $.data(target, "datepicker");
|
|
}
|
|
catch (err) {
|
|
throw "Missing instance data for this datepicker";
|
|
}
|
|
},
|
|
|
|
/* Update or retrieve the settings for a date picker attached to an input field or division.
|
|
* @param target element - the target input field or division or span
|
|
* @param name object - the new settings to update or
|
|
* string - the name of the setting to change or retrieve,
|
|
* when retrieving also "all" for all instance settings or
|
|
* "defaults" for all global defaults
|
|
* @param value any - the new value for the setting
|
|
* (omit if above is an object or to retrieve a value)
|
|
*/
|
|
_optionDatepicker: function(target, name, value) {
|
|
var settings, date, minDate, maxDate,
|
|
inst = this._getInst(target);
|
|
|
|
if (arguments.length === 2 && typeof name === "string") {
|
|
return (name === "defaults" ? $.extend({}, $.datepicker._defaults) :
|
|
(inst ? (name === "all" ? $.extend({}, inst.settings) :
|
|
this._get(inst, name)) : null));
|
|
}
|
|
|
|
settings = name || {};
|
|
if (typeof name === "string") {
|
|
settings = {};
|
|
settings[name] = value;
|
|
}
|
|
|
|
if (inst) {
|
|
if (this._curInst === inst) {
|
|
this._hideDatepicker();
|
|
}
|
|
|
|
date = this._getDateDatepicker(target, true);
|
|
minDate = this._getMinMaxDate(inst, "min");
|
|
maxDate = this._getMinMaxDate(inst, "max");
|
|
datepicker_extendRemove(inst.settings, settings);
|
|
// reformat the old minDate/maxDate values if dateFormat changes and a new minDate/maxDate isn't provided
|
|
if (minDate !== null && settings.dateFormat !== undefined && settings.minDate === undefined) {
|
|
inst.settings.minDate = this._formatDate(inst, minDate);
|
|
}
|
|
if (maxDate !== null && settings.dateFormat !== undefined && settings.maxDate === undefined) {
|
|
inst.settings.maxDate = this._formatDate(inst, maxDate);
|
|
}
|
|
if ( "disabled" in settings ) {
|
|
if ( settings.disabled ) {
|
|
this._disableDatepicker(target);
|
|
} else {
|
|
this._enableDatepicker(target);
|
|
}
|
|
}
|
|
this._attachments($(target), inst);
|
|
this._autoSize(inst);
|
|
this._setDate(inst, date);
|
|
this._updateAlternate(inst);
|
|
this._updateDatepicker(inst);
|
|
}
|
|
},
|
|
|
|
// change method deprecated
|
|
_changeDatepicker: function(target, name, value) {
|
|
this._optionDatepicker(target, name, value);
|
|
},
|
|
|
|
/* Redraw the date picker attached to an input field or division.
|
|
* @param target element - the target input field or division or span
|
|
*/
|
|
_refreshDatepicker: function(target) {
|
|
var inst = this._getInst(target);
|
|
if (inst) {
|
|
this._updateDatepicker(inst);
|
|
}
|
|
},
|
|
|
|
/* Set the dates for a jQuery selection.
|
|
* @param target element - the target input field or division or span
|
|
* @param date Date - the new date
|
|
*/
|
|
_setDateDatepicker: function(target, date) {
|
|
var inst = this._getInst(target);
|
|
if (inst) {
|
|
this._setDate(inst, date);
|
|
this._updateDatepicker(inst);
|
|
this._updateAlternate(inst);
|
|
}
|
|
},
|
|
|
|
/* Get the date(s) for the first entry in a jQuery selection.
|
|
* @param target element - the target input field or division or span
|
|
* @param noDefault boolean - true if no default date is to be used
|
|
* @return Date - the current date
|
|
*/
|
|
_getDateDatepicker: function(target, noDefault) {
|
|
var inst = this._getInst(target);
|
|
if (inst && !inst.inline) {
|
|
this._setDateFromField(inst, noDefault);
|
|
}
|
|
return (inst ? this._getDate(inst) : null);
|
|
},
|
|
|
|
/* Handle keystrokes. */
|
|
_doKeyDown: function(event) {
|
|
var onSelect, dateStr, sel,
|
|
inst = $.datepicker._getInst(event.target),
|
|
handled = true,
|
|
isRTL = inst.dpDiv.is(".ui-datepicker-rtl");
|
|
|
|
inst._keyEvent = true;
|
|
if ($.datepicker._datepickerShowing) {
|
|
switch (event.keyCode) {
|
|
case 9: $.datepicker._hideDatepicker();
|
|
handled = false;
|
|
break; // hide on tab out
|
|
case 13: sel = $("td." + $.datepicker._dayOverClass + ":not(." +
|
|
$.datepicker._currentClass + ")", inst.dpDiv);
|
|
if (sel[0]) {
|
|
$.datepicker._selectDay(event.target, inst.selectedMonth, inst.selectedYear, sel[0]);
|
|
}
|
|
|
|
onSelect = $.datepicker._get(inst, "onSelect");
|
|
if (onSelect) {
|
|
dateStr = $.datepicker._formatDate(inst);
|
|
|
|
// trigger custom callback
|
|
onSelect.apply((inst.input ? inst.input[0] : null), [dateStr, inst]);
|
|
} else {
|
|
$.datepicker._hideDatepicker();
|
|
}
|
|
|
|
return false; // don't submit the form
|
|
case 27: $.datepicker._hideDatepicker();
|
|
break; // hide on escape
|
|
case 33: $.datepicker._adjustDate(event.target, (event.ctrlKey ?
|
|
-$.datepicker._get(inst, "stepBigMonths") :
|
|
-$.datepicker._get(inst, "stepMonths")), "M");
|
|
break; // previous month/year on page up/+ ctrl
|
|
case 34: $.datepicker._adjustDate(event.target, (event.ctrlKey ?
|
|
+$.datepicker._get(inst, "stepBigMonths") :
|
|
+$.datepicker._get(inst, "stepMonths")), "M");
|
|
break; // next month/year on page down/+ ctrl
|
|
case 35: if (event.ctrlKey || event.metaKey) {
|
|
$.datepicker._clearDate(event.target);
|
|
}
|
|
handled = event.ctrlKey || event.metaKey;
|
|
break; // clear on ctrl or command +end
|
|
case 36: if (event.ctrlKey || event.metaKey) {
|
|
$.datepicker._gotoToday(event.target);
|
|
}
|
|
handled = event.ctrlKey || event.metaKey;
|
|
break; // current on ctrl or command +home
|
|
case 37: if (event.ctrlKey || event.metaKey) {
|
|
$.datepicker._adjustDate(event.target, (isRTL ? +1 : -1), "D");
|
|
}
|
|
handled = event.ctrlKey || event.metaKey;
|
|
// -1 day on ctrl or command +left
|
|
if (event.originalEvent.altKey) {
|
|
$.datepicker._adjustDate(event.target, (event.ctrlKey ?
|
|
-$.datepicker._get(inst, "stepBigMonths") :
|
|
-$.datepicker._get(inst, "stepMonths")), "M");
|
|
}
|
|
// next month/year on alt +left on Mac
|
|
break;
|
|
case 38: if (event.ctrlKey || event.metaKey) {
|
|
$.datepicker._adjustDate(event.target, -7, "D");
|
|
}
|
|
handled = event.ctrlKey || event.metaKey;
|
|
break; // -1 week on ctrl or command +up
|
|
case 39: if (event.ctrlKey || event.metaKey) {
|
|
$.datepicker._adjustDate(event.target, (isRTL ? -1 : +1), "D");
|
|
}
|
|
handled = event.ctrlKey || event.metaKey;
|
|
// +1 day on ctrl or command +right
|
|
if (event.originalEvent.altKey) {
|
|
$.datepicker._adjustDate(event.target, (event.ctrlKey ?
|
|
+$.datepicker._get(inst, "stepBigMonths") :
|
|
+$.datepicker._get(inst, "stepMonths")), "M");
|
|
}
|
|
// next month/year on alt +right
|
|
break;
|
|
case 40: if (event.ctrlKey || event.metaKey) {
|
|
$.datepicker._adjustDate(event.target, +7, "D");
|
|
}
|
|
handled = event.ctrlKey || event.metaKey;
|
|
break; // +1 week on ctrl or command +down
|
|
default: handled = false;
|
|
}
|
|
} else if (event.keyCode === 36 && event.ctrlKey) { // display the date picker on ctrl+home
|
|
$.datepicker._showDatepicker(this);
|
|
} else {
|
|
handled = false;
|
|
}
|
|
|
|
if (handled) {
|
|
event.preventDefault();
|
|
event.stopPropagation();
|
|
}
|
|
},
|
|
|
|
/* Filter entered characters - based on date format. */
|
|
_doKeyPress: function(event) {
|
|
var chars, chr,
|
|
inst = $.datepicker._getInst(event.target);
|
|
|
|
if ($.datepicker._get(inst, "constrainInput")) {
|
|
chars = $.datepicker._possibleChars($.datepicker._get(inst, "dateFormat"));
|
|
chr = String.fromCharCode(event.charCode == null ? event.keyCode : event.charCode);
|
|
return event.ctrlKey || event.metaKey || (chr < " " || !chars || chars.indexOf(chr) > -1);
|
|
}
|
|
},
|
|
|
|
/* Synchronise manual entry and field/alternate field. */
|
|
_doKeyUp: function(event) {
|
|
var date,
|
|
inst = $.datepicker._getInst(event.target);
|
|
|
|
if (inst.input.val() !== inst.lastVal) {
|
|
try {
|
|
date = $.datepicker.parseDate($.datepicker._get(inst, "dateFormat"),
|
|
(inst.input ? inst.input.val() : null),
|
|
$.datepicker._getFormatConfig(inst));
|
|
|
|
if (date) { // only if valid
|
|
$.datepicker._setDateFromField(inst);
|
|
$.datepicker._updateAlternate(inst);
|
|
$.datepicker._updateDatepicker(inst);
|
|
}
|
|
}
|
|
catch (err) {
|
|
}
|
|
}
|
|
return true;
|
|
},
|
|
|
|
/* Pop-up the date picker for a given input field.
|
|
* If false returned from beforeShow event handler do not show.
|
|
* @param input element - the input field attached to the date picker or
|
|
* event - if triggered by focus
|
|
*/
|
|
_showDatepicker: function(input) {
|
|
input = input.target || input;
|
|
if (input.nodeName.toLowerCase() !== "input") { // find from button/image trigger
|
|
input = $("input", input.parentNode)[0];
|
|
}
|
|
|
|
if ($.datepicker._isDisabledDatepicker(input) || $.datepicker._lastInput === input) { // already here
|
|
return;
|
|
}
|
|
|
|
var inst, beforeShow, beforeShowSettings, isFixed,
|
|
offset, showAnim, duration;
|
|
|
|
inst = $.datepicker._getInst(input);
|
|
if ($.datepicker._curInst && $.datepicker._curInst !== inst) {
|
|
$.datepicker._curInst.dpDiv.stop(true, true);
|
|
if ( inst && $.datepicker._datepickerShowing ) {
|
|
$.datepicker._hideDatepicker( $.datepicker._curInst.input[0] );
|
|
}
|
|
}
|
|
|
|
beforeShow = $.datepicker._get(inst, "beforeShow");
|
|
beforeShowSettings = beforeShow ? beforeShow.apply(input, [input, inst]) : {};
|
|
if(beforeShowSettings === false){
|
|
return;
|
|
}
|
|
datepicker_extendRemove(inst.settings, beforeShowSettings);
|
|
|
|
inst.lastVal = null;
|
|
$.datepicker._lastInput = input;
|
|
$.datepicker._setDateFromField(inst);
|
|
|
|
if ($.datepicker._inDialog) { // hide cursor
|
|
input.value = "";
|
|
}
|
|
if (!$.datepicker._pos) { // position below input
|
|
$.datepicker._pos = $.datepicker._findPos(input);
|
|
$.datepicker._pos[1] += input.offsetHeight; // add the height
|
|
}
|
|
|
|
isFixed = false;
|
|
$(input).parents().each(function() {
|
|
isFixed |= $(this).css("position") === "fixed";
|
|
return !isFixed;
|
|
});
|
|
|
|
offset = {left: $.datepicker._pos[0], top: $.datepicker._pos[1]};
|
|
$.datepicker._pos = null;
|
|
//to avoid flashes on Firefox
|
|
inst.dpDiv.empty();
|
|
// determine sizing offscreen
|
|
inst.dpDiv.css({position: "absolute", display: "block", top: "-1000px"});
|
|
$.datepicker._updateDatepicker(inst);
|
|
// fix width for dynamic number of date pickers
|
|
// and adjust position before showing
|
|
offset = $.datepicker._checkOffset(inst, offset, isFixed);
|
|
inst.dpDiv.css({position: ($.datepicker._inDialog && $.blockUI ?
|
|
"static" : (isFixed ? "fixed" : "absolute")), display: "none",
|
|
left: offset.left + "px", top: offset.top + "px"});
|
|
|
|
if (!inst.inline) {
|
|
showAnim = $.datepicker._get(inst, "showAnim");
|
|
duration = $.datepicker._get(inst, "duration");
|
|
inst.dpDiv.css( "z-index", datepicker_getZindex( $( input ) ) + 1 );
|
|
$.datepicker._datepickerShowing = true;
|
|
|
|
if ( $.effects && $.effects.effect[ showAnim ] ) {
|
|
inst.dpDiv.show(showAnim, $.datepicker._get(inst, "showOptions"), duration);
|
|
} else {
|
|
inst.dpDiv[showAnim || "show"](showAnim ? duration : null);
|
|
}
|
|
|
|
if ( $.datepicker._shouldFocusInput( inst ) ) {
|
|
inst.input.focus();
|
|
}
|
|
|
|
$.datepicker._curInst = inst;
|
|
}
|
|
},
|
|
|
|
/* Generate the date picker content. */
|
|
_updateDatepicker: function(inst) {
|
|
this.maxRows = 4; //Reset the max number of rows being displayed (see #7043)
|
|
datepicker_instActive = inst; // for delegate hover events
|
|
inst.dpDiv.empty().append(this._generateHTML(inst));
|
|
this._attachHandlers(inst);
|
|
|
|
var origyearshtml,
|
|
numMonths = this._getNumberOfMonths(inst),
|
|
cols = numMonths[1],
|
|
width = 17,
|
|
activeCell = inst.dpDiv.find( "." + this._dayOverClass + " a" );
|
|
|
|
if ( activeCell.length > 0 ) {
|
|
datepicker_handleMouseover.apply( activeCell.get( 0 ) );
|
|
}
|
|
|
|
inst.dpDiv.removeClass("ui-datepicker-multi-2 ui-datepicker-multi-3 ui-datepicker-multi-4").width("");
|
|
if (cols > 1) {
|
|
inst.dpDiv.addClass("ui-datepicker-multi-" + cols).css("width", (width * cols) + "em");
|
|
}
|
|
inst.dpDiv[(numMonths[0] !== 1 || numMonths[1] !== 1 ? "add" : "remove") +
|
|
"Class"]("ui-datepicker-multi");
|
|
inst.dpDiv[(this._get(inst, "isRTL") ? "add" : "remove") +
|
|
"Class"]("ui-datepicker-rtl");
|
|
|
|
if (inst === $.datepicker._curInst && $.datepicker._datepickerShowing && $.datepicker._shouldFocusInput( inst ) ) {
|
|
inst.input.focus();
|
|
}
|
|
|
|
// deffered render of the years select (to avoid flashes on Firefox)
|
|
if( inst.yearshtml ){
|
|
origyearshtml = inst.yearshtml;
|
|
setTimeout(function(){
|
|
//assure that inst.yearshtml didn't change.
|
|
if( origyearshtml === inst.yearshtml && inst.yearshtml ){
|
|
inst.dpDiv.find("select.ui-datepicker-year:first").replaceWith(inst.yearshtml);
|
|
}
|
|
origyearshtml = inst.yearshtml = null;
|
|
}, 0);
|
|
}
|
|
},
|
|
|
|
// #6694 - don't focus the input if it's already focused
|
|
// this breaks the change event in IE
|
|
// Support: IE and jQuery <1.9
|
|
_shouldFocusInput: function( inst ) {
|
|
return inst.input && inst.input.is( ":visible" ) && !inst.input.is( ":disabled" ) && !inst.input.is( ":focus" );
|
|
},
|
|
|
|
/* Check positioning to remain on screen. */
|
|
_checkOffset: function(inst, offset, isFixed) {
|
|
var dpWidth = inst.dpDiv.outerWidth(),
|
|
dpHeight = inst.dpDiv.outerHeight(),
|
|
inputWidth = inst.input ? inst.input.outerWidth() : 0,
|
|
inputHeight = inst.input ? inst.input.outerHeight() : 0,
|
|
viewWidth = document.documentElement.clientWidth + (isFixed ? 0 : $(document).scrollLeft()),
|
|
viewHeight = document.documentElement.clientHeight + (isFixed ? 0 : $(document).scrollTop());
|
|
|
|
offset.left -= (this._get(inst, "isRTL") ? (dpWidth - inputWidth) : 0);
|
|
offset.left -= (isFixed && offset.left === inst.input.offset().left) ? $(document).scrollLeft() : 0;
|
|
offset.top -= (isFixed && offset.top === (inst.input.offset().top + inputHeight)) ? $(document).scrollTop() : 0;
|
|
|
|
// now check if datepicker is showing outside window viewport - move to a better place if so.
|
|
offset.left -= Math.min(offset.left, (offset.left + dpWidth > viewWidth && viewWidth > dpWidth) ?
|
|
Math.abs(offset.left + dpWidth - viewWidth) : 0);
|
|
offset.top -= Math.min(offset.top, (offset.top + dpHeight > viewHeight && viewHeight > dpHeight) ?
|
|
Math.abs(dpHeight + inputHeight) : 0);
|
|
|
|
return offset;
|
|
},
|
|
|
|
/* Find an object's position on the screen. */
|
|
_findPos: function(obj) {
|
|
var position,
|
|
inst = this._getInst(obj),
|
|
isRTL = this._get(inst, "isRTL");
|
|
|
|
while (obj && (obj.type === "hidden" || obj.nodeType !== 1 || $.expr.filters.hidden(obj))) {
|
|
obj = obj[isRTL ? "previousSibling" : "nextSibling"];
|
|
}
|
|
|
|
position = $(obj).offset();
|
|
return [position.left, position.top];
|
|
},
|
|
|
|
/* Hide the date picker from view.
|
|
* @param input element - the input field attached to the date picker
|
|
*/
|
|
_hideDatepicker: function(input) {
|
|
var showAnim, duration, postProcess, onClose,
|
|
inst = this._curInst;
|
|
|
|
if (!inst || (input && inst !== $.data(input, "datepicker"))) {
|
|
return;
|
|
}
|
|
|
|
if (this._datepickerShowing) {
|
|
showAnim = this._get(inst, "showAnim");
|
|
duration = this._get(inst, "duration");
|
|
postProcess = function() {
|
|
$.datepicker._tidyDialog(inst);
|
|
};
|
|
|
|
// DEPRECATED: after BC for 1.8.x $.effects[ showAnim ] is not needed
|
|
if ( $.effects && ( $.effects.effect[ showAnim ] || $.effects[ showAnim ] ) ) {
|
|
inst.dpDiv.hide(showAnim, $.datepicker._get(inst, "showOptions"), duration, postProcess);
|
|
} else {
|
|
inst.dpDiv[(showAnim === "slideDown" ? "slideUp" :
|
|
(showAnim === "fadeIn" ? "fadeOut" : "hide"))]((showAnim ? duration : null), postProcess);
|
|
}
|
|
|
|
if (!showAnim) {
|
|
postProcess();
|
|
}
|
|
this._datepickerShowing = false;
|
|
|
|
onClose = this._get(inst, "onClose");
|
|
if (onClose) {
|
|
onClose.apply((inst.input ? inst.input[0] : null), [(inst.input ? inst.input.val() : ""), inst]);
|
|
}
|
|
|
|
this._lastInput = null;
|
|
if (this._inDialog) {
|
|
this._dialogInput.css({ position: "absolute", left: "0", top: "-100px" });
|
|
if ($.blockUI) {
|
|
$.unblockUI();
|
|
$("body").append(this.dpDiv);
|
|
}
|
|
}
|
|
this._inDialog = false;
|
|
}
|
|
},
|
|
|
|
/* Tidy up after a dialog display. */
|
|
_tidyDialog: function(inst) {
|
|
inst.dpDiv.removeClass(this._dialogClass).unbind(".ui-datepicker-calendar");
|
|
},
|
|
|
|
/* Close date picker if clicked elsewhere. */
|
|
_checkExternalClick: function(event) {
|
|
if (!$.datepicker._curInst) {
|
|
return;
|
|
}
|
|
|
|
var $target = $(event.target),
|
|
inst = $.datepicker._getInst($target[0]);
|
|
|
|
if ( ( ( $target[0].id !== $.datepicker._mainDivId &&
|
|
$target.parents("#" + $.datepicker._mainDivId).length === 0 &&
|
|
!$target.hasClass($.datepicker.markerClassName) &&
|
|
!$target.closest("." + $.datepicker._triggerClass).length &&
|
|
$.datepicker._datepickerShowing && !($.datepicker._inDialog && $.blockUI) ) ) ||
|
|
( $target.hasClass($.datepicker.markerClassName) && $.datepicker._curInst !== inst ) ) {
|
|
$.datepicker._hideDatepicker();
|
|
}
|
|
},
|
|
|
|
/* Adjust one of the date sub-fields. */
|
|
_adjustDate: function(id, offset, period) {
|
|
var target = $(id),
|
|
inst = this._getInst(target[0]);
|
|
|
|
if (this._isDisabledDatepicker(target[0])) {
|
|
return;
|
|
}
|
|
this._adjustInstDate(inst, offset +
|
|
(period === "M" ? this._get(inst, "showCurrentAtPos") : 0), // undo positioning
|
|
period);
|
|
this._updateDatepicker(inst);
|
|
},
|
|
|
|
/* Action for current link. */
|
|
_gotoToday: function(id) {
|
|
var date,
|
|
target = $(id),
|
|
inst = this._getInst(target[0]);
|
|
|
|
if (this._get(inst, "gotoCurrent") && inst.currentDay) {
|
|
inst.selectedDay = inst.currentDay;
|
|
inst.drawMonth = inst.selectedMonth = inst.currentMonth;
|
|
inst.drawYear = inst.selectedYear = inst.currentYear;
|
|
} else {
|
|
date = new Date();
|
|
inst.selectedDay = date.getDate();
|
|
inst.drawMonth = inst.selectedMonth = date.getMonth();
|
|
inst.drawYear = inst.selectedYear = date.getFullYear();
|
|
}
|
|
this._notifyChange(inst);
|
|
this._adjustDate(target);
|
|
},
|
|
|
|
/* Action for selecting a new month/year. */
|
|
_selectMonthYear: function(id, select, period) {
|
|
var target = $(id),
|
|
inst = this._getInst(target[0]);
|
|
|
|
inst["selected" + (period === "M" ? "Month" : "Year")] =
|
|
inst["draw" + (period === "M" ? "Month" : "Year")] =
|
|
parseInt(select.options[select.selectedIndex].value,10);
|
|
|
|
this._notifyChange(inst);
|
|
this._adjustDate(target);
|
|
},
|
|
|
|
/* Action for selecting a day. */
|
|
_selectDay: function(id, month, year, td) {
|
|
var inst,
|
|
target = $(id);
|
|
|
|
if ($(td).hasClass(this._unselectableClass) || this._isDisabledDatepicker(target[0])) {
|
|
return;
|
|
}
|
|
|
|
inst = this._getInst(target[0]);
|
|
inst.selectedDay = inst.currentDay = $("a", td).html();
|
|
inst.selectedMonth = inst.currentMonth = month;
|
|
inst.selectedYear = inst.currentYear = year;
|
|
this._selectDate(id, this._formatDate(inst,
|
|
inst.currentDay, inst.currentMonth, inst.currentYear));
|
|
},
|
|
|
|
/* Erase the input field and hide the date picker. */
|
|
_clearDate: function(id) {
|
|
var target = $(id);
|
|
this._selectDate(target, "");
|
|
},
|
|
|
|
/* Update the input field with the selected date. */
|
|
_selectDate: function(id, dateStr) {
|
|
var onSelect,
|
|
target = $(id),
|
|
inst = this._getInst(target[0]);
|
|
|
|
dateStr = (dateStr != null ? dateStr : this._formatDate(inst));
|
|
if (inst.input) {
|
|
inst.input.val(dateStr);
|
|
}
|
|
this._updateAlternate(inst);
|
|
|
|
onSelect = this._get(inst, "onSelect");
|
|
if (onSelect) {
|
|
onSelect.apply((inst.input ? inst.input[0] : null), [dateStr, inst]); // trigger custom callback
|
|
} else if (inst.input) {
|
|
inst.input.trigger("change"); // fire the change event
|
|
}
|
|
|
|
if (inst.inline){
|
|
this._updateDatepicker(inst);
|
|
} else {
|
|
this._hideDatepicker();
|
|
this._lastInput = inst.input[0];
|
|
if (typeof(inst.input[0]) !== "object") {
|
|
inst.input.focus(); // restore focus
|
|
}
|
|
this._lastInput = null;
|
|
}
|
|
},
|
|
|
|
/* Update any alternate field to synchronise with the main field. */
|
|
_updateAlternate: function(inst) {
|
|
var altFormat, date, dateStr,
|
|
altField = this._get(inst, "altField");
|
|
|
|
if (altField) { // update alternate field too
|
|
altFormat = this._get(inst, "altFormat") || this._get(inst, "dateFormat");
|
|
date = this._getDate(inst);
|
|
dateStr = this.formatDate(altFormat, date, this._getFormatConfig(inst));
|
|
$(altField).each(function() { $(this).val(dateStr); });
|
|
}
|
|
},
|
|
|
|
/* Set as beforeShowDay function to prevent selection of weekends.
|
|
* @param date Date - the date to customise
|
|
* @return [boolean, string] - is this date selectable?, what is its CSS class?
|
|
*/
|
|
noWeekends: function(date) {
|
|
var day = date.getDay();
|
|
return [(day > 0 && day < 6), ""];
|
|
},
|
|
|
|
/* Set as calculateWeek to determine the week of the year based on the ISO 8601 definition.
|
|
* @param date Date - the date to get the week for
|
|
* @return number - the number of the week within the year that contains this date
|
|
*/
|
|
iso8601Week: function(date) {
|
|
var time,
|
|
checkDate = new Date(date.getTime());
|
|
|
|
// Find Thursday of this week starting on Monday
|
|
checkDate.setDate(checkDate.getDate() + 4 - (checkDate.getDay() || 7));
|
|
|
|
time = checkDate.getTime();
|
|
checkDate.setMonth(0); // Compare with Jan 1
|
|
checkDate.setDate(1);
|
|
return Math.floor(Math.round((time - checkDate) / 86400000) / 7) + 1;
|
|
},
|
|
|
|
/* Parse a string value into a date object.
|
|
* See formatDate below for the possible formats.
|
|
*
|
|
* @param format string - the expected format of the date
|
|
* @param value string - the date in the above format
|
|
* @param settings Object - attributes include:
|
|
* shortYearCutoff number - the cutoff year for determining the century (optional)
|
|
* dayNamesShort string[7] - abbreviated names of the days from Sunday (optional)
|
|
* dayNames string[7] - names of the days from Sunday (optional)
|
|
* monthNamesShort string[12] - abbreviated names of the months (optional)
|
|
* monthNames string[12] - names of the months (optional)
|
|
* @return Date - the extracted date value or null if value is blank
|
|
*/
|
|
parseDate: function (format, value, settings) {
|
|
if (format == null || value == null) {
|
|
throw "Invalid arguments";
|
|
}
|
|
|
|
value = (typeof value === "object" ? value.toString() : value + "");
|
|
if (value === "") {
|
|
return null;
|
|
}
|
|
|
|
var iFormat, dim, extra,
|
|
iValue = 0,
|
|
shortYearCutoffTemp = (settings ? settings.shortYearCutoff : null) || this._defaults.shortYearCutoff,
|
|
shortYearCutoff = (typeof shortYearCutoffTemp !== "string" ? shortYearCutoffTemp :
|
|
new Date().getFullYear() % 100 + parseInt(shortYearCutoffTemp, 10)),
|
|
dayNamesShort = (settings ? settings.dayNamesShort : null) || this._defaults.dayNamesShort,
|
|
dayNames = (settings ? settings.dayNames : null) || this._defaults.dayNames,
|
|
monthNamesShort = (settings ? settings.monthNamesShort : null) || this._defaults.monthNamesShort,
|
|
monthNames = (settings ? settings.monthNames : null) || this._defaults.monthNames,
|
|
year = -1,
|
|
month = -1,
|
|
day = -1,
|
|
doy = -1,
|
|
literal = false,
|
|
date,
|
|
// Check whether a format character is doubled
|
|
lookAhead = function(match) {
|
|
var matches = (iFormat + 1 < format.length && format.charAt(iFormat + 1) === match);
|
|
if (matches) {
|
|
iFormat++;
|
|
}
|
|
return matches;
|
|
},
|
|
// Extract a number from the string value
|
|
getNumber = function(match) {
|
|
var isDoubled = lookAhead(match),
|
|
size = (match === "@" ? 14 : (match === "!" ? 20 :
|
|
(match === "y" && isDoubled ? 4 : (match === "o" ? 3 : 2)))),
|
|
minSize = (match === "y" ? size : 1),
|
|
digits = new RegExp("^\\d{" + minSize + "," + size + "}"),
|
|
num = value.substring(iValue).match(digits);
|
|
if (!num) {
|
|
throw "Missing number at position " + iValue;
|
|
}
|
|
iValue += num[0].length;
|
|
return parseInt(num[0], 10);
|
|
},
|
|
// Extract a name from the string value and convert to an index
|
|
getName = function(match, shortNames, longNames) {
|
|
var index = -1,
|
|
names = $.map(lookAhead(match) ? longNames : shortNames, function (v, k) {
|
|
return [ [k, v] ];
|
|
}).sort(function (a, b) {
|
|
return -(a[1].length - b[1].length);
|
|
});
|
|
|
|
$.each(names, function (i, pair) {
|
|
var name = pair[1];
|
|
if (value.substr(iValue, name.length).toLowerCase() === name.toLowerCase()) {
|
|
index = pair[0];
|
|
iValue += name.length;
|
|
return false;
|
|
}
|
|
});
|
|
if (index !== -1) {
|
|
return index + 1;
|
|
} else {
|
|
throw "Unknown name at position " + iValue;
|
|
}
|
|
},
|
|
// Confirm that a literal character matches the string value
|
|
checkLiteral = function() {
|
|
if (value.charAt(iValue) !== format.charAt(iFormat)) {
|
|
throw "Unexpected literal at position " + iValue;
|
|
}
|
|
iValue++;
|
|
};
|
|
|
|
for (iFormat = 0; iFormat < format.length; iFormat++) {
|
|
if (literal) {
|
|
if (format.charAt(iFormat) === "'" && !lookAhead("'")) {
|
|
literal = false;
|
|
} else {
|
|
checkLiteral();
|
|
}
|
|
} else {
|
|
switch (format.charAt(iFormat)) {
|
|
case "d":
|
|
day = getNumber("d");
|
|
break;
|
|
case "D":
|
|
getName("D", dayNamesShort, dayNames);
|
|
break;
|
|
case "o":
|
|
doy = getNumber("o");
|
|
break;
|
|
case "m":
|
|
month = getNumber("m");
|
|
break;
|
|
case "M":
|
|
month = getName("M", monthNamesShort, monthNames);
|
|
break;
|
|
case "y":
|
|
year = getNumber("y");
|
|
break;
|
|
case "@":
|
|
date = new Date(getNumber("@"));
|
|
year = date.getFullYear();
|
|
month = date.getMonth() + 1;
|
|
day = date.getDate();
|
|
break;
|
|
case "!":
|
|
date = new Date((getNumber("!") - this._ticksTo1970) / 10000);
|
|
year = date.getFullYear();
|
|
month = date.getMonth() + 1;
|
|
day = date.getDate();
|
|
break;
|
|
case "'":
|
|
if (lookAhead("'")){
|
|
checkLiteral();
|
|
} else {
|
|
literal = true;
|
|
}
|
|
break;
|
|
default:
|
|
checkLiteral();
|
|
}
|
|
}
|
|
}
|
|
|
|
if (iValue < value.length){
|
|
extra = value.substr(iValue);
|
|
if (!/^\s+/.test(extra)) {
|
|
throw "Extra/unparsed characters found in date: " + extra;
|
|
}
|
|
}
|
|
|
|
if (year === -1) {
|
|
year = new Date().getFullYear();
|
|
} else if (year < 100) {
|
|
year += new Date().getFullYear() - new Date().getFullYear() % 100 +
|
|
(year <= shortYearCutoff ? 0 : -100);
|
|
}
|
|
|
|
if (doy > -1) {
|
|
month = 1;
|
|
day = doy;
|
|
do {
|
|
dim = this._getDaysInMonth(year, month - 1);
|
|
if (day <= dim) {
|
|
break;
|
|
}
|
|
month++;
|
|
day -= dim;
|
|
} while (true);
|
|
}
|
|
|
|
date = this._daylightSavingAdjust(new Date(year, month - 1, day));
|
|
if (date.getFullYear() !== year || date.getMonth() + 1 !== month || date.getDate() !== day) {
|
|
throw "Invalid date"; // E.g. 31/02/00
|
|
}
|
|
return date;
|
|
},
|
|
|
|
/* Standard date formats. */
|
|
ATOM: "yy-mm-dd", // RFC 3339 (ISO 8601)
|
|
COOKIE: "D, dd M yy",
|
|
ISO_8601: "yy-mm-dd",
|
|
RFC_822: "D, d M y",
|
|
RFC_850: "DD, dd-M-y",
|
|
RFC_1036: "D, d M y",
|
|
RFC_1123: "D, d M yy",
|
|
RFC_2822: "D, d M yy",
|
|
RSS: "D, d M y", // RFC 822
|
|
TICKS: "!",
|
|
TIMESTAMP: "@",
|
|
W3C: "yy-mm-dd", // ISO 8601
|
|
|
|
_ticksTo1970: (((1970 - 1) * 365 + Math.floor(1970 / 4) - Math.floor(1970 / 100) +
|
|
Math.floor(1970 / 400)) * 24 * 60 * 60 * 10000000),
|
|
|
|
/* Format a date object into a string value.
|
|
* The format can be combinations of the following:
|
|
* d - day of month (no leading zero)
|
|
* dd - day of month (two digit)
|
|
* o - day of year (no leading zeros)
|
|
* oo - day of year (three digit)
|
|
* D - day name short
|
|
* DD - day name long
|
|
* m - month of year (no leading zero)
|
|
* mm - month of year (two digit)
|
|
* M - month name short
|
|
* MM - month name long
|
|
* y - year (two digit)
|
|
* yy - year (four digit)
|
|
* @ - Unix timestamp (ms since 01/01/1970)
|
|
* ! - Windows ticks (100ns since 01/01/0001)
|
|
* "..." - literal text
|
|
* '' - single quote
|
|
*
|
|
* @param format string - the desired format of the date
|
|
* @param date Date - the date value to format
|
|
* @param settings Object - attributes include:
|
|
* dayNamesShort string[7] - abbreviated names of the days from Sunday (optional)
|
|
* dayNames string[7] - names of the days from Sunday (optional)
|
|
* monthNamesShort string[12] - abbreviated names of the months (optional)
|
|
* monthNames string[12] - names of the months (optional)
|
|
* @return string - the date in the above format
|
|
*/
|
|
formatDate: function (format, date, settings) {
|
|
if (!date) {
|
|
return "";
|
|
}
|
|
|
|
var iFormat,
|
|
dayNamesShort = (settings ? settings.dayNamesShort : null) || this._defaults.dayNamesShort,
|
|
dayNames = (settings ? settings.dayNames : null) || this._defaults.dayNames,
|
|
monthNamesShort = (settings ? settings.monthNamesShort : null) || this._defaults.monthNamesShort,
|
|
monthNames = (settings ? settings.monthNames : null) || this._defaults.monthNames,
|
|
// Check whether a format character is doubled
|
|
lookAhead = function(match) {
|
|
var matches = (iFormat + 1 < format.length && format.charAt(iFormat + 1) === match);
|
|
if (matches) {
|
|
iFormat++;
|
|
}
|
|
return matches;
|
|
},
|
|
// Format a number, with leading zero if necessary
|
|
formatNumber = function(match, value, len) {
|
|
var num = "" + value;
|
|
if (lookAhead(match)) {
|
|
while (num.length < len) {
|
|
num = "0" + num;
|
|
}
|
|
}
|
|
return num;
|
|
},
|
|
// Format a name, short or long as requested
|
|
formatName = function(match, value, shortNames, longNames) {
|
|
return (lookAhead(match) ? longNames[value] : shortNames[value]);
|
|
},
|
|
output = "",
|
|
literal = false;
|
|
|
|
if (date) {
|
|
for (iFormat = 0; iFormat < format.length; iFormat++) {
|
|
if (literal) {
|
|
if (format.charAt(iFormat) === "'" && !lookAhead("'")) {
|
|
literal = false;
|
|
} else {
|
|
output += format.charAt(iFormat);
|
|
}
|
|
} else {
|
|
switch (format.charAt(iFormat)) {
|
|
case "d":
|
|
output += formatNumber("d", date.getDate(), 2);
|
|
break;
|
|
case "D":
|
|
output += formatName("D", date.getDay(), dayNamesShort, dayNames);
|
|
break;
|
|
case "o":
|
|
output += formatNumber("o",
|
|
Math.round((new Date(date.getFullYear(), date.getMonth(), date.getDate()).getTime() - new Date(date.getFullYear(), 0, 0).getTime()) / 86400000), 3);
|
|
break;
|
|
case "m":
|
|
output += formatNumber("m", date.getMonth() + 1, 2);
|
|
break;
|
|
case "M":
|
|
output += formatName("M", date.getMonth(), monthNamesShort, monthNames);
|
|
break;
|
|
case "y":
|
|
output += (lookAhead("y") ? date.getFullYear() :
|
|
(date.getYear() % 100 < 10 ? "0" : "") + date.getYear() % 100);
|
|
break;
|
|
case "@":
|
|
output += date.getTime();
|
|
break;
|
|
case "!":
|
|
output += date.getTime() * 10000 + this._ticksTo1970;
|
|
break;
|
|
case "'":
|
|
if (lookAhead("'")) {
|
|
output += "'";
|
|
} else {
|
|
literal = true;
|
|
}
|
|
break;
|
|
default:
|
|
output += format.charAt(iFormat);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return output;
|
|
},
|
|
|
|
/* Extract all possible characters from the date format. */
|
|
_possibleChars: function (format) {
|
|
var iFormat,
|
|
chars = "",
|
|
literal = false,
|
|
// Check whether a format character is doubled
|
|
lookAhead = function(match) {
|
|
var matches = (iFormat + 1 < format.length && format.charAt(iFormat + 1) === match);
|
|
if (matches) {
|
|
iFormat++;
|
|
}
|
|
return matches;
|
|
};
|
|
|
|
for (iFormat = 0; iFormat < format.length; iFormat++) {
|
|
if (literal) {
|
|
if (format.charAt(iFormat) === "'" && !lookAhead("'")) {
|
|
literal = false;
|
|
} else {
|
|
chars += format.charAt(iFormat);
|
|
}
|
|
} else {
|
|
switch (format.charAt(iFormat)) {
|
|
case "d": case "m": case "y": case "@":
|
|
chars += "0123456789";
|
|
break;
|
|
case "D": case "M":
|
|
return null; // Accept anything
|
|
case "'":
|
|
if (lookAhead("'")) {
|
|
chars += "'";
|
|
} else {
|
|
literal = true;
|
|
}
|
|
break;
|
|
default:
|
|
chars += format.charAt(iFormat);
|
|
}
|
|
}
|
|
}
|
|
return chars;
|
|
},
|
|
|
|
/* Get a setting value, defaulting if necessary. */
|
|
_get: function(inst, name) {
|
|
return inst.settings[name] !== undefined ?
|
|
inst.settings[name] : this._defaults[name];
|
|
},
|
|
|
|
/* Parse existing date and initialise date picker. */
|
|
_setDateFromField: function(inst, noDefault) {
|
|
if (inst.input.val() === inst.lastVal) {
|
|
return;
|
|
}
|
|
|
|
var dateFormat = this._get(inst, "dateFormat"),
|
|
dates = inst.lastVal = inst.input ? inst.input.val() : null,
|
|
defaultDate = this._getDefaultDate(inst),
|
|
date = defaultDate,
|
|
settings = this._getFormatConfig(inst);
|
|
|
|
try {
|
|
date = this.parseDate(dateFormat, dates, settings) || defaultDate;
|
|
} catch (event) {
|
|
dates = (noDefault ? "" : dates);
|
|
}
|
|
inst.selectedDay = date.getDate();
|
|
inst.drawMonth = inst.selectedMonth = date.getMonth();
|
|
inst.drawYear = inst.selectedYear = date.getFullYear();
|
|
inst.currentDay = (dates ? date.getDate() : 0);
|
|
inst.currentMonth = (dates ? date.getMonth() : 0);
|
|
inst.currentYear = (dates ? date.getFullYear() : 0);
|
|
this._adjustInstDate(inst);
|
|
},
|
|
|
|
/* Retrieve the default date shown on opening. */
|
|
_getDefaultDate: function(inst) {
|
|
return this._restrictMinMax(inst,
|
|
this._determineDate(inst, this._get(inst, "defaultDate"), new Date()));
|
|
},
|
|
|
|
/* A date may be specified as an exact value or a relative one. */
|
|
_determineDate: function(inst, date, defaultDate) {
|
|
var offsetNumeric = function(offset) {
|
|
var date = new Date();
|
|
date.setDate(date.getDate() + offset);
|
|
return date;
|
|
},
|
|
offsetString = function(offset) {
|
|
try {
|
|
return $.datepicker.parseDate($.datepicker._get(inst, "dateFormat"),
|
|
offset, $.datepicker._getFormatConfig(inst));
|
|
}
|
|
catch (e) {
|
|
// Ignore
|
|
}
|
|
|
|
var date = (offset.toLowerCase().match(/^c/) ?
|
|
$.datepicker._getDate(inst) : null) || new Date(),
|
|
year = date.getFullYear(),
|
|
month = date.getMonth(),
|
|
day = date.getDate(),
|
|
pattern = /([+\-]?[0-9]+)\s*(d|D|w|W|m|M|y|Y)?/g,
|
|
matches = pattern.exec(offset);
|
|
|
|
while (matches) {
|
|
switch (matches[2] || "d") {
|
|
case "d" : case "D" :
|
|
day += parseInt(matches[1],10); break;
|
|
case "w" : case "W" :
|
|
day += parseInt(matches[1],10) * 7; break;
|
|
case "m" : case "M" :
|
|
month += parseInt(matches[1],10);
|
|
day = Math.min(day, $.datepicker._getDaysInMonth(year, month));
|
|
break;
|
|
case "y": case "Y" :
|
|
year += parseInt(matches[1],10);
|
|
day = Math.min(day, $.datepicker._getDaysInMonth(year, month));
|
|
break;
|
|
}
|
|
matches = pattern.exec(offset);
|
|
}
|
|
return new Date(year, month, day);
|
|
},
|
|
newDate = (date == null || date === "" ? defaultDate : (typeof date === "string" ? offsetString(date) :
|
|
(typeof date === "number" ? (isNaN(date) ? defaultDate : offsetNumeric(date)) : new Date(date.getTime()))));
|
|
|
|
newDate = (newDate && newDate.toString() === "Invalid Date" ? defaultDate : newDate);
|
|
if (newDate) {
|
|
newDate.setHours(0);
|
|
newDate.setMinutes(0);
|
|
newDate.setSeconds(0);
|
|
newDate.setMilliseconds(0);
|
|
}
|
|
return this._daylightSavingAdjust(newDate);
|
|
},
|
|
|
|
/* Handle switch to/from daylight saving.
|
|
* Hours may be non-zero on daylight saving cut-over:
|
|
* > 12 when midnight changeover, but then cannot generate
|
|
* midnight datetime, so jump to 1AM, otherwise reset.
|
|
* @param date (Date) the date to check
|
|
* @return (Date) the corrected date
|
|
*/
|
|
_daylightSavingAdjust: function(date) {
|
|
if (!date) {
|
|
return null;
|
|
}
|
|
date.setHours(date.getHours() > 12 ? date.getHours() + 2 : 0);
|
|
return date;
|
|
},
|
|
|
|
/* Set the date(s) directly. */
|
|
_setDate: function(inst, date, noChange) {
|
|
var clear = !date,
|
|
origMonth = inst.selectedMonth,
|
|
origYear = inst.selectedYear,
|
|
newDate = this._restrictMinMax(inst, this._determineDate(inst, date, new Date()));
|
|
|
|
inst.selectedDay = inst.currentDay = newDate.getDate();
|
|
inst.drawMonth = inst.selectedMonth = inst.currentMonth = newDate.getMonth();
|
|
inst.drawYear = inst.selectedYear = inst.currentYear = newDate.getFullYear();
|
|
if ((origMonth !== inst.selectedMonth || origYear !== inst.selectedYear) && !noChange) {
|
|
this._notifyChange(inst);
|
|
}
|
|
this._adjustInstDate(inst);
|
|
if (inst.input) {
|
|
inst.input.val(clear ? "" : this._formatDate(inst));
|
|
}
|
|
},
|
|
|
|
/* Retrieve the date(s) directly. */
|
|
_getDate: function(inst) {
|
|
var startDate = (!inst.currentYear || (inst.input && inst.input.val() === "") ? null :
|
|
this._daylightSavingAdjust(new Date(
|
|
inst.currentYear, inst.currentMonth, inst.currentDay)));
|
|
return startDate;
|
|
},
|
|
|
|
/* Attach the onxxx handlers. These are declared statically so
|
|
* they work with static code transformers like Caja.
|
|
*/
|
|
_attachHandlers: function(inst) {
|
|
var stepMonths = this._get(inst, "stepMonths"),
|
|
id = "#" + inst.id.replace( /\\\\/g, "\\" );
|
|
inst.dpDiv.find("[data-handler]").map(function () {
|
|
var handler = {
|
|
prev: function () {
|
|
$.datepicker._adjustDate(id, -stepMonths, "M");
|
|
},
|
|
next: function () {
|
|
$.datepicker._adjustDate(id, +stepMonths, "M");
|
|
},
|
|
hide: function () {
|
|
$.datepicker._hideDatepicker();
|
|
},
|
|
today: function () {
|
|
$.datepicker._gotoToday(id);
|
|
},
|
|
selectDay: function () {
|
|
$.datepicker._selectDay(id, +this.getAttribute("data-month"), +this.getAttribute("data-year"), this);
|
|
return false;
|
|
},
|
|
selectMonth: function () {
|
|
$.datepicker._selectMonthYear(id, this, "M");
|
|
return false;
|
|
},
|
|
selectYear: function () {
|
|
$.datepicker._selectMonthYear(id, this, "Y");
|
|
return false;
|
|
}
|
|
};
|
|
$(this).bind(this.getAttribute("data-event"), handler[this.getAttribute("data-handler")]);
|
|
});
|
|
},
|
|
|
|
/* Generate the HTML for the current state of the date picker. */
|
|
_generateHTML: function(inst) {
|
|
var maxDraw, prevText, prev, nextText, next, currentText, gotoDate,
|
|
controls, buttonPanel, firstDay, showWeek, dayNames, dayNamesMin,
|
|
monthNames, monthNamesShort, beforeShowDay, showOtherMonths,
|
|
selectOtherMonths, defaultDate, html, dow, row, group, col, selectedDate,
|
|
cornerClass, calender, thead, day, daysInMonth, leadDays, curRows, numRows,
|
|
printDate, dRow, tbody, daySettings, otherMonth, unselectable,
|
|
tempDate = new Date(),
|
|
today = this._daylightSavingAdjust(
|
|
new Date(tempDate.getFullYear(), tempDate.getMonth(), tempDate.getDate())), // clear time
|
|
isRTL = this._get(inst, "isRTL"),
|
|
showButtonPanel = this._get(inst, "showButtonPanel"),
|
|
hideIfNoPrevNext = this._get(inst, "hideIfNoPrevNext"),
|
|
navigationAsDateFormat = this._get(inst, "navigationAsDateFormat"),
|
|
numMonths = this._getNumberOfMonths(inst),
|
|
showCurrentAtPos = this._get(inst, "showCurrentAtPos"),
|
|
stepMonths = this._get(inst, "stepMonths"),
|
|
isMultiMonth = (numMonths[0] !== 1 || numMonths[1] !== 1),
|
|
currentDate = this._daylightSavingAdjust((!inst.currentDay ? new Date(9999, 9, 9) :
|
|
new Date(inst.currentYear, inst.currentMonth, inst.currentDay))),
|
|
minDate = this._getMinMaxDate(inst, "min"),
|
|
maxDate = this._getMinMaxDate(inst, "max"),
|
|
drawMonth = inst.drawMonth - showCurrentAtPos,
|
|
drawYear = inst.drawYear;
|
|
|
|
if (drawMonth < 0) {
|
|
drawMonth += 12;
|
|
drawYear--;
|
|
}
|
|
if (maxDate) {
|
|
maxDraw = this._daylightSavingAdjust(new Date(maxDate.getFullYear(),
|
|
maxDate.getMonth() - (numMonths[0] * numMonths[1]) + 1, maxDate.getDate()));
|
|
maxDraw = (minDate && maxDraw < minDate ? minDate : maxDraw);
|
|
while (this._daylightSavingAdjust(new Date(drawYear, drawMonth, 1)) > maxDraw) {
|
|
drawMonth--;
|
|
if (drawMonth < 0) {
|
|
drawMonth = 11;
|
|
drawYear--;
|
|
}
|
|
}
|
|
}
|
|
inst.drawMonth = drawMonth;
|
|
inst.drawYear = drawYear;
|
|
|
|
prevText = this._get(inst, "prevText");
|
|
prevText = (!navigationAsDateFormat ? prevText : this.formatDate(prevText,
|
|
this._daylightSavingAdjust(new Date(drawYear, drawMonth - stepMonths, 1)),
|
|
this._getFormatConfig(inst)));
|
|
|
|
prev = (this._canAdjustMonth(inst, -1, drawYear, drawMonth) ?
|
|
"<a class='ui-datepicker-prev ui-corner-all' data-handler='prev' data-event='click'" +
|
|
" title='" + prevText + "'><span class='ui-icon ui-icon-circle-triangle-" + ( isRTL ? "e" : "w") + "'>" + prevText + "</span></a>" :
|
|
(hideIfNoPrevNext ? "" : "<a class='ui-datepicker-prev ui-corner-all ui-state-disabled' title='"+ prevText +"'><span class='ui-icon ui-icon-circle-triangle-" + ( isRTL ? "e" : "w") + "'>" + prevText + "</span></a>"));
|
|
|
|
nextText = this._get(inst, "nextText");
|
|
nextText = (!navigationAsDateFormat ? nextText : this.formatDate(nextText,
|
|
this._daylightSavingAdjust(new Date(drawYear, drawMonth + stepMonths, 1)),
|
|
this._getFormatConfig(inst)));
|
|
|
|
next = (this._canAdjustMonth(inst, +1, drawYear, drawMonth) ?
|
|
"<a class='ui-datepicker-next ui-corner-all' data-handler='next' data-event='click'" +
|
|
" title='" + nextText + "'><span class='ui-icon ui-icon-circle-triangle-" + ( isRTL ? "w" : "e") + "'>" + nextText + "</span></a>" :
|
|
(hideIfNoPrevNext ? "" : "<a class='ui-datepicker-next ui-corner-all ui-state-disabled' title='"+ nextText + "'><span class='ui-icon ui-icon-circle-triangle-" + ( isRTL ? "w" : "e") + "'>" + nextText + "</span></a>"));
|
|
|
|
currentText = this._get(inst, "currentText");
|
|
gotoDate = (this._get(inst, "gotoCurrent") && inst.currentDay ? currentDate : today);
|
|
currentText = (!navigationAsDateFormat ? currentText :
|
|
this.formatDate(currentText, gotoDate, this._getFormatConfig(inst)));
|
|
|
|
controls = (!inst.inline ? "<button type='button' class='ui-datepicker-close ui-state-default ui-priority-primary ui-corner-all' data-handler='hide' data-event='click'>" +
|
|
this._get(inst, "closeText") + "</button>" : "");
|
|
|
|
buttonPanel = (showButtonPanel) ? "<div class='ui-datepicker-buttonpane ui-widget-content'>" + (isRTL ? controls : "") +
|
|
(this._isInRange(inst, gotoDate) ? "<button type='button' class='ui-datepicker-current ui-state-default ui-priority-secondary ui-corner-all' data-handler='today' data-event='click'" +
|
|
">" + currentText + "</button>" : "") + (isRTL ? "" : controls) + "</div>" : "";
|
|
|
|
firstDay = parseInt(this._get(inst, "firstDay"),10);
|
|
firstDay = (isNaN(firstDay) ? 0 : firstDay);
|
|
|
|
showWeek = this._get(inst, "showWeek");
|
|
dayNames = this._get(inst, "dayNames");
|
|
dayNamesMin = this._get(inst, "dayNamesMin");
|
|
monthNames = this._get(inst, "monthNames");
|
|
monthNamesShort = this._get(inst, "monthNamesShort");
|
|
beforeShowDay = this._get(inst, "beforeShowDay");
|
|
showOtherMonths = this._get(inst, "showOtherMonths");
|
|
selectOtherMonths = this._get(inst, "selectOtherMonths");
|
|
defaultDate = this._getDefaultDate(inst);
|
|
html = "";
|
|
dow;
|
|
for (row = 0; row < numMonths[0]; row++) {
|
|
group = "";
|
|
this.maxRows = 4;
|
|
for (col = 0; col < numMonths[1]; col++) {
|
|
selectedDate = this._daylightSavingAdjust(new Date(drawYear, drawMonth, inst.selectedDay));
|
|
cornerClass = " ui-corner-all";
|
|
calender = "";
|
|
if (isMultiMonth) {
|
|
calender += "<div class='ui-datepicker-group";
|
|
if (numMonths[1] > 1) {
|
|
switch (col) {
|
|
case 0: calender += " ui-datepicker-group-first";
|
|
cornerClass = " ui-corner-" + (isRTL ? "right" : "left"); break;
|
|
case numMonths[1]-1: calender += " ui-datepicker-group-last";
|
|
cornerClass = " ui-corner-" + (isRTL ? "left" : "right"); break;
|
|
default: calender += " ui-datepicker-group-middle"; cornerClass = ""; break;
|
|
}
|
|
}
|
|
calender += "'>";
|
|
}
|
|
calender += "<div class='ui-datepicker-header ui-widget-header ui-helper-clearfix" + cornerClass + "'>" +
|
|
(/all|left/.test(cornerClass) && row === 0 ? (isRTL ? next : prev) : "") +
|
|
(/all|right/.test(cornerClass) && row === 0 ? (isRTL ? prev : next) : "") +
|
|
this._generateMonthYearHeader(inst, drawMonth, drawYear, minDate, maxDate,
|
|
row > 0 || col > 0, monthNames, monthNamesShort) + // draw month headers
|
|
"</div><table class='ui-datepicker-calendar'><thead>" +
|
|
"<tr>";
|
|
thead = (showWeek ? "<th class='ui-datepicker-week-col'>" + this._get(inst, "weekHeader") + "</th>" : "");
|
|
for (dow = 0; dow < 7; dow++) { // days of the week
|
|
day = (dow + firstDay) % 7;
|
|
thead += "<th scope='col'" + ((dow + firstDay + 6) % 7 >= 5 ? " class='ui-datepicker-week-end'" : "") + ">" +
|
|
"<span title='" + dayNames[day] + "'>" + dayNamesMin[day] + "</span></th>";
|
|
}
|
|
calender += thead + "</tr></thead><tbody>";
|
|
daysInMonth = this._getDaysInMonth(drawYear, drawMonth);
|
|
if (drawYear === inst.selectedYear && drawMonth === inst.selectedMonth) {
|
|
inst.selectedDay = Math.min(inst.selectedDay, daysInMonth);
|
|
}
|
|
leadDays = (this._getFirstDayOfMonth(drawYear, drawMonth) - firstDay + 7) % 7;
|
|
curRows = Math.ceil((leadDays + daysInMonth) / 7); // calculate the number of rows to generate
|
|
numRows = (isMultiMonth ? this.maxRows > curRows ? this.maxRows : curRows : curRows); //If multiple months, use the higher number of rows (see #7043)
|
|
this.maxRows = numRows;
|
|
printDate = this._daylightSavingAdjust(new Date(drawYear, drawMonth, 1 - leadDays));
|
|
for (dRow = 0; dRow < numRows; dRow++) { // create date picker rows
|
|
calender += "<tr>";
|
|
tbody = (!showWeek ? "" : "<td class='ui-datepicker-week-col'>" +
|
|
this._get(inst, "calculateWeek")(printDate) + "</td>");
|
|
for (dow = 0; dow < 7; dow++) { // create date picker days
|
|
daySettings = (beforeShowDay ?
|
|
beforeShowDay.apply((inst.input ? inst.input[0] : null), [printDate]) : [true, ""]);
|
|
otherMonth = (printDate.getMonth() !== drawMonth);
|
|
unselectable = (otherMonth && !selectOtherMonths) || !daySettings[0] ||
|
|
(minDate && printDate < minDate) || (maxDate && printDate > maxDate);
|
|
tbody += "<td class='" +
|
|
((dow + firstDay + 6) % 7 >= 5 ? " ui-datepicker-week-end" : "") + // highlight weekends
|
|
(otherMonth ? " ui-datepicker-other-month" : "") + // highlight days from other months
|
|
((printDate.getTime() === selectedDate.getTime() && drawMonth === inst.selectedMonth && inst._keyEvent) || // user pressed key
|
|
(defaultDate.getTime() === printDate.getTime() && defaultDate.getTime() === selectedDate.getTime()) ?
|
|
// or defaultDate is current printedDate and defaultDate is selectedDate
|
|
" " + this._dayOverClass : "") + // highlight selected day
|
|
(unselectable ? " " + this._unselectableClass + " ui-state-disabled": "") + // highlight unselectable days
|
|
(otherMonth && !showOtherMonths ? "" : " " + daySettings[1] + // highlight custom dates
|
|
(printDate.getTime() === currentDate.getTime() ? " " + this._currentClass : "") + // highlight selected day
|
|
(printDate.getTime() === today.getTime() ? " ui-datepicker-today" : "")) + "'" + // highlight today (if different)
|
|
((!otherMonth || showOtherMonths) && daySettings[2] ? " title='" + daySettings[2].replace(/'/g, "'") + "'" : "") + // cell title
|
|
(unselectable ? "" : " data-handler='selectDay' data-event='click' data-month='" + printDate.getMonth() + "' data-year='" + printDate.getFullYear() + "'") + ">" + // actions
|
|
(otherMonth && !showOtherMonths ? " " : // display for other months
|
|
(unselectable ? "<span class='ui-state-default'>" + printDate.getDate() + "</span>" : "<a class='ui-state-default" +
|
|
(printDate.getTime() === today.getTime() ? " ui-state-highlight" : "") +
|
|
(printDate.getTime() === currentDate.getTime() ? " ui-state-active" : "") + // highlight selected day
|
|
(otherMonth ? " ui-priority-secondary" : "") + // distinguish dates from other months
|
|
"' href='#'>" + printDate.getDate() + "</a>")) + "</td>"; // display selectable date
|
|
printDate.setDate(printDate.getDate() + 1);
|
|
printDate = this._daylightSavingAdjust(printDate);
|
|
}
|
|
calender += tbody + "</tr>";
|
|
}
|
|
drawMonth++;
|
|
if (drawMonth > 11) {
|
|
drawMonth = 0;
|
|
drawYear++;
|
|
}
|
|
calender += "</tbody></table>" + (isMultiMonth ? "</div>" +
|
|
((numMonths[0] > 0 && col === numMonths[1]-1) ? "<div class='ui-datepicker-row-break'></div>" : "") : "");
|
|
group += calender;
|
|
}
|
|
html += group;
|
|
}
|
|
html += buttonPanel;
|
|
inst._keyEvent = false;
|
|
return html;
|
|
},
|
|
|
|
/* Generate the month and year header. */
|
|
_generateMonthYearHeader: function(inst, drawMonth, drawYear, minDate, maxDate,
|
|
secondary, monthNames, monthNamesShort) {
|
|
|
|
var inMinYear, inMaxYear, month, years, thisYear, determineYear, year, endYear,
|
|
changeMonth = this._get(inst, "changeMonth"),
|
|
changeYear = this._get(inst, "changeYear"),
|
|
showMonthAfterYear = this._get(inst, "showMonthAfterYear"),
|
|
html = "<div class='ui-datepicker-title'>",
|
|
monthHtml = "";
|
|
|
|
// month selection
|
|
if (secondary || !changeMonth) {
|
|
monthHtml += "<span class='ui-datepicker-month'>" + monthNames[drawMonth] + "</span>";
|
|
} else {
|
|
inMinYear = (minDate && minDate.getFullYear() === drawYear);
|
|
inMaxYear = (maxDate && maxDate.getFullYear() === drawYear);
|
|
monthHtml += "<select class='ui-datepicker-month' data-handler='selectMonth' data-event='change'>";
|
|
for ( month = 0; month < 12; month++) {
|
|
if ((!inMinYear || month >= minDate.getMonth()) && (!inMaxYear || month <= maxDate.getMonth())) {
|
|
monthHtml += "<option value='" + month + "'" +
|
|
(month === drawMonth ? " selected='selected'" : "") +
|
|
">" + monthNamesShort[month] + "</option>";
|
|
}
|
|
}
|
|
monthHtml += "</select>";
|
|
}
|
|
|
|
if (!showMonthAfterYear) {
|
|
html += monthHtml + (secondary || !(changeMonth && changeYear) ? " " : "");
|
|
}
|
|
|
|
// year selection
|
|
if ( !inst.yearshtml ) {
|
|
inst.yearshtml = "";
|
|
if (secondary || !changeYear) {
|
|
html += "<span class='ui-datepicker-year'>" + drawYear + "</span>";
|
|
} else {
|
|
// determine range of years to display
|
|
years = this._get(inst, "yearRange").split(":");
|
|
thisYear = new Date().getFullYear();
|
|
determineYear = function(value) {
|
|
var year = (value.match(/c[+\-].*/) ? drawYear + parseInt(value.substring(1), 10) :
|
|
(value.match(/[+\-].*/) ? thisYear + parseInt(value, 10) :
|
|
parseInt(value, 10)));
|
|
return (isNaN(year) ? thisYear : year);
|
|
};
|
|
year = determineYear(years[0]);
|
|
endYear = Math.max(year, determineYear(years[1] || ""));
|
|
year = (minDate ? Math.max(year, minDate.getFullYear()) : year);
|
|
endYear = (maxDate ? Math.min(endYear, maxDate.getFullYear()) : endYear);
|
|
inst.yearshtml += "<select class='ui-datepicker-year' data-handler='selectYear' data-event='change'>";
|
|
for (; year <= endYear; year++) {
|
|
inst.yearshtml += "<option value='" + year + "'" +
|
|
(year === drawYear ? " selected='selected'" : "") +
|
|
">" + year + "</option>";
|
|
}
|
|
inst.yearshtml += "</select>";
|
|
|
|
html += inst.yearshtml;
|
|
inst.yearshtml = null;
|
|
}
|
|
}
|
|
|
|
html += this._get(inst, "yearSuffix");
|
|
if (showMonthAfterYear) {
|
|
html += (secondary || !(changeMonth && changeYear) ? " " : "") + monthHtml;
|
|
}
|
|
html += "</div>"; // Close datepicker_header
|
|
return html;
|
|
},
|
|
|
|
/* Adjust one of the date sub-fields. */
|
|
_adjustInstDate: function(inst, offset, period) {
|
|
var year = inst.drawYear + (period === "Y" ? offset : 0),
|
|
month = inst.drawMonth + (period === "M" ? offset : 0),
|
|
day = Math.min(inst.selectedDay, this._getDaysInMonth(year, month)) + (period === "D" ? offset : 0),
|
|
date = this._restrictMinMax(inst, this._daylightSavingAdjust(new Date(year, month, day)));
|
|
|
|
inst.selectedDay = date.getDate();
|
|
inst.drawMonth = inst.selectedMonth = date.getMonth();
|
|
inst.drawYear = inst.selectedYear = date.getFullYear();
|
|
if (period === "M" || period === "Y") {
|
|
this._notifyChange(inst);
|
|
}
|
|
},
|
|
|
|
/* Ensure a date is within any min/max bounds. */
|
|
_restrictMinMax: function(inst, date) {
|
|
var minDate = this._getMinMaxDate(inst, "min"),
|
|
maxDate = this._getMinMaxDate(inst, "max"),
|
|
newDate = (minDate && date < minDate ? minDate : date);
|
|
return (maxDate && newDate > maxDate ? maxDate : newDate);
|
|
},
|
|
|
|
/* Notify change of month/year. */
|
|
_notifyChange: function(inst) {
|
|
var onChange = this._get(inst, "onChangeMonthYear");
|
|
if (onChange) {
|
|
onChange.apply((inst.input ? inst.input[0] : null),
|
|
[inst.selectedYear, inst.selectedMonth + 1, inst]);
|
|
}
|
|
},
|
|
|
|
/* Determine the number of months to show. */
|
|
_getNumberOfMonths: function(inst) {
|
|
var numMonths = this._get(inst, "numberOfMonths");
|
|
return (numMonths == null ? [1, 1] : (typeof numMonths === "number" ? [1, numMonths] : numMonths));
|
|
},
|
|
|
|
/* Determine the current maximum date - ensure no time components are set. */
|
|
_getMinMaxDate: function(inst, minMax) {
|
|
return this._determineDate(inst, this._get(inst, minMax + "Date"), null);
|
|
},
|
|
|
|
/* Find the number of days in a given month. */
|
|
_getDaysInMonth: function(year, month) {
|
|
return 32 - this._daylightSavingAdjust(new Date(year, month, 32)).getDate();
|
|
},
|
|
|
|
/* Find the day of the week of the first of a month. */
|
|
_getFirstDayOfMonth: function(year, month) {
|
|
return new Date(year, month, 1).getDay();
|
|
},
|
|
|
|
/* Determines if we should allow a "next/prev" month display change. */
|
|
_canAdjustMonth: function(inst, offset, curYear, curMonth) {
|
|
var numMonths = this._getNumberOfMonths(inst),
|
|
date = this._daylightSavingAdjust(new Date(curYear,
|
|
curMonth + (offset < 0 ? offset : numMonths[0] * numMonths[1]), 1));
|
|
|
|
if (offset < 0) {
|
|
date.setDate(this._getDaysInMonth(date.getFullYear(), date.getMonth()));
|
|
}
|
|
return this._isInRange(inst, date);
|
|
},
|
|
|
|
/* Is the given date in the accepted range? */
|
|
_isInRange: function(inst, date) {
|
|
var yearSplit, currentYear,
|
|
minDate = this._getMinMaxDate(inst, "min"),
|
|
maxDate = this._getMinMaxDate(inst, "max"),
|
|
minYear = null,
|
|
maxYear = null,
|
|
years = this._get(inst, "yearRange");
|
|
if (years){
|
|
yearSplit = years.split(":");
|
|
currentYear = new Date().getFullYear();
|
|
minYear = parseInt(yearSplit[0], 10);
|
|
maxYear = parseInt(yearSplit[1], 10);
|
|
if ( yearSplit[0].match(/[+\-].*/) ) {
|
|
minYear += currentYear;
|
|
}
|
|
if ( yearSplit[1].match(/[+\-].*/) ) {
|
|
maxYear += currentYear;
|
|
}
|
|
}
|
|
|
|
return ((!minDate || date.getTime() >= minDate.getTime()) &&
|
|
(!maxDate || date.getTime() <= maxDate.getTime()) &&
|
|
(!minYear || date.getFullYear() >= minYear) &&
|
|
(!maxYear || date.getFullYear() <= maxYear));
|
|
},
|
|
|
|
/* Provide the configuration settings for formatting/parsing. */
|
|
_getFormatConfig: function(inst) {
|
|
var shortYearCutoff = this._get(inst, "shortYearCutoff");
|
|
shortYearCutoff = (typeof shortYearCutoff !== "string" ? shortYearCutoff :
|
|
new Date().getFullYear() % 100 + parseInt(shortYearCutoff, 10));
|
|
return {shortYearCutoff: shortYearCutoff,
|
|
dayNamesShort: this._get(inst, "dayNamesShort"), dayNames: this._get(inst, "dayNames"),
|
|
monthNamesShort: this._get(inst, "monthNamesShort"), monthNames: this._get(inst, "monthNames")};
|
|
},
|
|
|
|
/* Format the given date for display. */
|
|
_formatDate: function(inst, day, month, year) {
|
|
if (!day) {
|
|
inst.currentDay = inst.selectedDay;
|
|
inst.currentMonth = inst.selectedMonth;
|
|
inst.currentYear = inst.selectedYear;
|
|
}
|
|
var date = (day ? (typeof day === "object" ? day :
|
|
this._daylightSavingAdjust(new Date(year, month, day))) :
|
|
this._daylightSavingAdjust(new Date(inst.currentYear, inst.currentMonth, inst.currentDay)));
|
|
return this.formatDate(this._get(inst, "dateFormat"), date, this._getFormatConfig(inst));
|
|
}
|
|
});
|
|
|
|
/*
|
|
* Bind hover events for datepicker elements.
|
|
* Done via delegate so the binding only occurs once in the lifetime of the parent div.
|
|
* Global datepicker_instActive, set by _updateDatepicker allows the handlers to find their way back to the active picker.
|
|
*/
|
|
function datepicker_bindHover(dpDiv) {
|
|
var selector = "button, .ui-datepicker-prev, .ui-datepicker-next, .ui-datepicker-calendar td a";
|
|
return dpDiv.delegate(selector, "mouseout", function() {
|
|
$(this).removeClass("ui-state-hover");
|
|
if (this.className.indexOf("ui-datepicker-prev") !== -1) {
|
|
$(this).removeClass("ui-datepicker-prev-hover");
|
|
}
|
|
if (this.className.indexOf("ui-datepicker-next") !== -1) {
|
|
$(this).removeClass("ui-datepicker-next-hover");
|
|
}
|
|
})
|
|
.delegate( selector, "mouseover", datepicker_handleMouseover );
|
|
}
|
|
|
|
function datepicker_handleMouseover() {
|
|
if (!$.datepicker._isDisabledDatepicker( datepicker_instActive.inline? datepicker_instActive.dpDiv.parent()[0] : datepicker_instActive.input[0])) {
|
|
$(this).parents(".ui-datepicker-calendar").find("a").removeClass("ui-state-hover");
|
|
$(this).addClass("ui-state-hover");
|
|
if (this.className.indexOf("ui-datepicker-prev") !== -1) {
|
|
$(this).addClass("ui-datepicker-prev-hover");
|
|
}
|
|
if (this.className.indexOf("ui-datepicker-next") !== -1) {
|
|
$(this).addClass("ui-datepicker-next-hover");
|
|
}
|
|
}
|
|
}
|
|
|
|
/* jQuery extend now ignores nulls! */
|
|
function datepicker_extendRemove(target, props) {
|
|
$.extend(target, props);
|
|
for (var name in props) {
|
|
if (props[name] == null) {
|
|
target[name] = props[name];
|
|
}
|
|
}
|
|
return target;
|
|
}
|
|
|
|
/* Invoke the datepicker functionality.
|
|
@param options string - a command, optionally followed by additional parameters or
|
|
Object - settings for attaching new datepicker functionality
|
|
@return jQuery object */
|
|
$.fn.datepicker = function(options){
|
|
|
|
/* Verify an empty collection wasn't passed - Fixes #6976 */
|
|
if ( !this.length ) {
|
|
return this;
|
|
}
|
|
|
|
/* Initialise the date picker. */
|
|
if (!$.datepicker.initialized) {
|
|
$(document).mousedown($.datepicker._checkExternalClick);
|
|
$.datepicker.initialized = true;
|
|
}
|
|
|
|
/* Append datepicker main container to body if not exist. */
|
|
if ($("#"+$.datepicker._mainDivId).length === 0) {
|
|
$("body").append($.datepicker.dpDiv);
|
|
}
|
|
|
|
var otherArgs = Array.prototype.slice.call(arguments, 1);
|
|
if (typeof options === "string" && (options === "isDisabled" || options === "getDate" || options === "widget")) {
|
|
return $.datepicker["_" + options + "Datepicker"].
|
|
apply($.datepicker, [this[0]].concat(otherArgs));
|
|
}
|
|
if (options === "option" && arguments.length === 2 && typeof arguments[1] === "string") {
|
|
return $.datepicker["_" + options + "Datepicker"].
|
|
apply($.datepicker, [this[0]].concat(otherArgs));
|
|
}
|
|
return this.each(function() {
|
|
typeof options === "string" ?
|
|
$.datepicker["_" + options + "Datepicker"].
|
|
apply($.datepicker, [this].concat(otherArgs)) :
|
|
$.datepicker._attachDatepicker(this, options);
|
|
});
|
|
};
|
|
|
|
$.datepicker = new Datepicker(); // singleton instance
|
|
$.datepicker.initialized = false;
|
|
$.datepicker.uuid = new Date().getTime();
|
|
$.datepicker.version = "1.11.4";
|
|
|
|
var datepicker = $.datepicker;
|
|
|
|
|
|
/*!
|
|
* jQuery UI Draggable 1.11.4
|
|
* http://jqueryui.com
|
|
*
|
|
* Copyright jQuery Foundation and other contributors
|
|
* Released under the MIT license.
|
|
* http://jquery.org/license
|
|
*
|
|
* http://api.jqueryui.com/draggable/
|
|
*/
|
|
|
|
|
|
$.widget("ui.draggable", $.ui.mouse, {
|
|
version: "1.11.4",
|
|
widgetEventPrefix: "drag",
|
|
options: {
|
|
addClasses: true,
|
|
appendTo: "parent",
|
|
axis: false,
|
|
connectToSortable: false,
|
|
containment: false,
|
|
cursor: "auto",
|
|
cursorAt: false,
|
|
grid: false,
|
|
handle: false,
|
|
helper: "original",
|
|
iframeFix: false,
|
|
opacity: false,
|
|
refreshPositions: false,
|
|
revert: false,
|
|
revertDuration: 500,
|
|
scope: "default",
|
|
scroll: true,
|
|
scrollSensitivity: 20,
|
|
scrollSpeed: 20,
|
|
snap: false,
|
|
snapMode: "both",
|
|
snapTolerance: 20,
|
|
stack: false,
|
|
zIndex: false,
|
|
|
|
// callbacks
|
|
drag: null,
|
|
start: null,
|
|
stop: null
|
|
},
|
|
_create: function() {
|
|
|
|
if ( this.options.helper === "original" ) {
|
|
this._setPositionRelative();
|
|
}
|
|
if (this.options.addClasses){
|
|
this.element.addClass("ui-draggable");
|
|
}
|
|
if (this.options.disabled){
|
|
this.element.addClass("ui-draggable-disabled");
|
|
}
|
|
this._setHandleClassName();
|
|
|
|
this._mouseInit();
|
|
},
|
|
|
|
_setOption: function( key, value ) {
|
|
this._super( key, value );
|
|
if ( key === "handle" ) {
|
|
this._removeHandleClassName();
|
|
this._setHandleClassName();
|
|
}
|
|
},
|
|
|
|
_destroy: function() {
|
|
if ( ( this.helper || this.element ).is( ".ui-draggable-dragging" ) ) {
|
|
this.destroyOnClear = true;
|
|
return;
|
|
}
|
|
this.element.removeClass( "ui-draggable ui-draggable-dragging ui-draggable-disabled" );
|
|
this._removeHandleClassName();
|
|
this._mouseDestroy();
|
|
},
|
|
|
|
_mouseCapture: function(event) {
|
|
var o = this.options;
|
|
|
|
this._blurActiveElement( event );
|
|
|
|
// among others, prevent a drag on a resizable-handle
|
|
if (this.helper || o.disabled || $(event.target).closest(".ui-resizable-handle").length > 0) {
|
|
return false;
|
|
}
|
|
|
|
//Quit if we're not on a valid handle
|
|
this.handle = this._getHandle(event);
|
|
if (!this.handle) {
|
|
return false;
|
|
}
|
|
|
|
this._blockFrames( o.iframeFix === true ? "iframe" : o.iframeFix );
|
|
|
|
return true;
|
|
|
|
},
|
|
|
|
_blockFrames: function( selector ) {
|
|
this.iframeBlocks = this.document.find( selector ).map(function() {
|
|
var iframe = $( this );
|
|
|
|
return $( "<div>" )
|
|
.css( "position", "absolute" )
|
|
.appendTo( iframe.parent() )
|
|
.outerWidth( iframe.outerWidth() )
|
|
.outerHeight( iframe.outerHeight() )
|
|
.offset( iframe.offset() )[ 0 ];
|
|
});
|
|
},
|
|
|
|
_unblockFrames: function() {
|
|
if ( this.iframeBlocks ) {
|
|
this.iframeBlocks.remove();
|
|
delete this.iframeBlocks;
|
|
}
|
|
},
|
|
|
|
_blurActiveElement: function( event ) {
|
|
var document = this.document[ 0 ];
|
|
|
|
// Only need to blur if the event occurred on the draggable itself, see #10527
|
|
if ( !this.handleElement.is( event.target ) ) {
|
|
return;
|
|
}
|
|
|
|
// support: IE9
|
|
// IE9 throws an "Unspecified error" accessing document.activeElement from an <iframe>
|
|
try {
|
|
|
|
// Support: IE9, IE10
|
|
// If the <body> is blurred, IE will switch windows, see #9520
|
|
if ( document.activeElement && document.activeElement.nodeName.toLowerCase() !== "body" ) {
|
|
|
|
// Blur any element that currently has focus, see #4261
|
|
$( document.activeElement ).blur();
|
|
}
|
|
} catch ( error ) {}
|
|
},
|
|
|
|
_mouseStart: function(event) {
|
|
|
|
var o = this.options;
|
|
|
|
//Create and append the visible helper
|
|
this.helper = this._createHelper(event);
|
|
|
|
this.helper.addClass("ui-draggable-dragging");
|
|
|
|
//Cache the helper size
|
|
this._cacheHelperProportions();
|
|
|
|
//If ddmanager is used for droppables, set the global draggable
|
|
if ($.ui.ddmanager) {
|
|
$.ui.ddmanager.current = this;
|
|
}
|
|
|
|
/*
|
|
* - Position generation -
|
|
* This block generates everything position related - it's the core of draggables.
|
|
*/
|
|
|
|
//Cache the margins of the original element
|
|
this._cacheMargins();
|
|
|
|
//Store the helper's css position
|
|
this.cssPosition = this.helper.css( "position" );
|
|
this.scrollParent = this.helper.scrollParent( true );
|
|
this.offsetParent = this.helper.offsetParent();
|
|
this.hasFixedAncestor = this.helper.parents().filter(function() {
|
|
return $( this ).css( "position" ) === "fixed";
|
|
}).length > 0;
|
|
|
|
//The element's absolute position on the page minus margins
|
|
this.positionAbs = this.element.offset();
|
|
this._refreshOffsets( event );
|
|
|
|
//Generate the original position
|
|
this.originalPosition = this.position = this._generatePosition( event, false );
|
|
this.originalPageX = event.pageX;
|
|
this.originalPageY = event.pageY;
|
|
|
|
//Adjust the mouse offset relative to the helper if "cursorAt" is supplied
|
|
(o.cursorAt && this._adjustOffsetFromHelper(o.cursorAt));
|
|
|
|
//Set a containment if given in the options
|
|
this._setContainment();
|
|
|
|
//Trigger event + callbacks
|
|
if (this._trigger("start", event) === false) {
|
|
this._clear();
|
|
return false;
|
|
}
|
|
|
|
//Recache the helper size
|
|
this._cacheHelperProportions();
|
|
|
|
//Prepare the droppable offsets
|
|
if ($.ui.ddmanager && !o.dropBehaviour) {
|
|
$.ui.ddmanager.prepareOffsets(this, event);
|
|
}
|
|
|
|
// Reset helper's right/bottom css if they're set and set explicit width/height instead
|
|
// as this prevents resizing of elements with right/bottom set (see #7772)
|
|
this._normalizeRightBottom();
|
|
|
|
this._mouseDrag(event, true); //Execute the drag once - this causes the helper not to be visible before getting its correct position
|
|
|
|
//If the ddmanager is used for droppables, inform the manager that dragging has started (see #5003)
|
|
if ( $.ui.ddmanager ) {
|
|
$.ui.ddmanager.dragStart(this, event);
|
|
}
|
|
|
|
return true;
|
|
},
|
|
|
|
_refreshOffsets: function( event ) {
|
|
this.offset = {
|
|
top: this.positionAbs.top - this.margins.top,
|
|
left: this.positionAbs.left - this.margins.left,
|
|
scroll: false,
|
|
parent: this._getParentOffset(),
|
|
relative: this._getRelativeOffset()
|
|
};
|
|
|
|
this.offset.click = {
|
|
left: event.pageX - this.offset.left,
|
|
top: event.pageY - this.offset.top
|
|
};
|
|
},
|
|
|
|
_mouseDrag: function(event, noPropagation) {
|
|
// reset any necessary cached properties (see #5009)
|
|
if ( this.hasFixedAncestor ) {
|
|
this.offset.parent = this._getParentOffset();
|
|
}
|
|
|
|
//Compute the helpers position
|
|
this.position = this._generatePosition( event, true );
|
|
this.positionAbs = this._convertPositionTo("absolute");
|
|
|
|
//Call plugins and callbacks and use the resulting position if something is returned
|
|
if (!noPropagation) {
|
|
var ui = this._uiHash();
|
|
if (this._trigger("drag", event, ui) === false) {
|
|
this._mouseUp({});
|
|
return false;
|
|
}
|
|
this.position = ui.position;
|
|
}
|
|
|
|
this.helper[ 0 ].style.left = this.position.left + "px";
|
|
this.helper[ 0 ].style.top = this.position.top + "px";
|
|
|
|
if ($.ui.ddmanager) {
|
|
$.ui.ddmanager.drag(this, event);
|
|
}
|
|
|
|
return false;
|
|
},
|
|
|
|
_mouseStop: function(event) {
|
|
|
|
//If we are using droppables, inform the manager about the drop
|
|
var that = this,
|
|
dropped = false;
|
|
if ($.ui.ddmanager && !this.options.dropBehaviour) {
|
|
dropped = $.ui.ddmanager.drop(this, event);
|
|
}
|
|
|
|
//if a drop comes from outside (a sortable)
|
|
if (this.dropped) {
|
|
dropped = this.dropped;
|
|
this.dropped = false;
|
|
}
|
|
|
|
if ((this.options.revert === "invalid" && !dropped) || (this.options.revert === "valid" && dropped) || this.options.revert === true || ($.isFunction(this.options.revert) && this.options.revert.call(this.element, dropped))) {
|
|
$(this.helper).animate(this.originalPosition, parseInt(this.options.revertDuration, 10), function() {
|
|
if (that._trigger("stop", event) !== false) {
|
|
that._clear();
|
|
}
|
|
});
|
|
} else {
|
|
if (this._trigger("stop", event) !== false) {
|
|
this._clear();
|
|
}
|
|
}
|
|
|
|
return false;
|
|
},
|
|
|
|
_mouseUp: function( event ) {
|
|
this._unblockFrames();
|
|
|
|
//If the ddmanager is used for droppables, inform the manager that dragging has stopped (see #5003)
|
|
if ( $.ui.ddmanager ) {
|
|
$.ui.ddmanager.dragStop(this, event);
|
|
}
|
|
|
|
// Only need to focus if the event occurred on the draggable itself, see #10527
|
|
if ( this.handleElement.is( event.target ) ) {
|
|
// The interaction is over; whether or not the click resulted in a drag, focus the element
|
|
this.element.focus();
|
|
}
|
|
|
|
return $.ui.mouse.prototype._mouseUp.call(this, event);
|
|
},
|
|
|
|
cancel: function() {
|
|
|
|
if (this.helper.is(".ui-draggable-dragging")) {
|
|
this._mouseUp({});
|
|
} else {
|
|
this._clear();
|
|
}
|
|
|
|
return this;
|
|
|
|
},
|
|
|
|
_getHandle: function(event) {
|
|
return this.options.handle ?
|
|
!!$( event.target ).closest( this.element.find( this.options.handle ) ).length :
|
|
true;
|
|
},
|
|
|
|
_setHandleClassName: function() {
|
|
this.handleElement = this.options.handle ?
|
|
this.element.find( this.options.handle ) : this.element;
|
|
this.handleElement.addClass( "ui-draggable-handle" );
|
|
},
|
|
|
|
_removeHandleClassName: function() {
|
|
this.handleElement.removeClass( "ui-draggable-handle" );
|
|
},
|
|
|
|
_createHelper: function(event) {
|
|
|
|
var o = this.options,
|
|
helperIsFunction = $.isFunction( o.helper ),
|
|
helper = helperIsFunction ?
|
|
$( o.helper.apply( this.element[ 0 ], [ event ] ) ) :
|
|
( o.helper === "clone" ?
|
|
this.element.clone().removeAttr( "id" ) :
|
|
this.element );
|
|
|
|
if (!helper.parents("body").length) {
|
|
helper.appendTo((o.appendTo === "parent" ? this.element[0].parentNode : o.appendTo));
|
|
}
|
|
|
|
// http://bugs.jqueryui.com/ticket/9446
|
|
// a helper function can return the original element
|
|
// which wouldn't have been set to relative in _create
|
|
if ( helperIsFunction && helper[ 0 ] === this.element[ 0 ] ) {
|
|
this._setPositionRelative();
|
|
}
|
|
|
|
if (helper[0] !== this.element[0] && !(/(fixed|absolute)/).test(helper.css("position"))) {
|
|
helper.css("position", "absolute");
|
|
}
|
|
|
|
return helper;
|
|
|
|
},
|
|
|
|
_setPositionRelative: function() {
|
|
if ( !( /^(?:r|a|f)/ ).test( this.element.css( "position" ) ) ) {
|
|
this.element[ 0 ].style.position = "relative";
|
|
}
|
|
},
|
|
|
|
_adjustOffsetFromHelper: function(obj) {
|
|
if (typeof obj === "string") {
|
|
obj = obj.split(" ");
|
|
}
|
|
if ($.isArray(obj)) {
|
|
obj = { left: +obj[0], top: +obj[1] || 0 };
|
|
}
|
|
if ("left" in obj) {
|
|
this.offset.click.left = obj.left + this.margins.left;
|
|
}
|
|
if ("right" in obj) {
|
|
this.offset.click.left = this.helperProportions.width - obj.right + this.margins.left;
|
|
}
|
|
if ("top" in obj) {
|
|
this.offset.click.top = obj.top + this.margins.top;
|
|
}
|
|
if ("bottom" in obj) {
|
|
this.offset.click.top = this.helperProportions.height - obj.bottom + this.margins.top;
|
|
}
|
|
},
|
|
|
|
_isRootNode: function( element ) {
|
|
return ( /(html|body)/i ).test( element.tagName ) || element === this.document[ 0 ];
|
|
},
|
|
|
|
_getParentOffset: function() {
|
|
|
|
//Get the offsetParent and cache its position
|
|
var po = this.offsetParent.offset(),
|
|
document = this.document[ 0 ];
|
|
|
|
// This is a special case where we need to modify a offset calculated on start, since the following happened:
|
|
// 1. The position of the helper is absolute, so it's position is calculated based on the next positioned parent
|
|
// 2. The actual offset parent is a child of the scroll parent, and the scroll parent isn't the document, which means that
|
|
// the scroll is included in the initial calculation of the offset of the parent, and never recalculated upon drag
|
|
if (this.cssPosition === "absolute" && this.scrollParent[0] !== document && $.contains(this.scrollParent[0], this.offsetParent[0])) {
|
|
po.left += this.scrollParent.scrollLeft();
|
|
po.top += this.scrollParent.scrollTop();
|
|
}
|
|
|
|
if ( this._isRootNode( this.offsetParent[ 0 ] ) ) {
|
|
po = { top: 0, left: 0 };
|
|
}
|
|
|
|
return {
|
|
top: po.top + (parseInt(this.offsetParent.css("borderTopWidth"), 10) || 0),
|
|
left: po.left + (parseInt(this.offsetParent.css("borderLeftWidth"), 10) || 0)
|
|
};
|
|
|
|
},
|
|
|
|
_getRelativeOffset: function() {
|
|
if ( this.cssPosition !== "relative" ) {
|
|
return { top: 0, left: 0 };
|
|
}
|
|
|
|
var p = this.element.position(),
|
|
scrollIsRootNode = this._isRootNode( this.scrollParent[ 0 ] );
|
|
|
|
return {
|
|
top: p.top - ( parseInt(this.helper.css( "top" ), 10) || 0 ) + ( !scrollIsRootNode ? this.scrollParent.scrollTop() : 0 ),
|
|
left: p.left - ( parseInt(this.helper.css( "left" ), 10) || 0 ) + ( !scrollIsRootNode ? this.scrollParent.scrollLeft() : 0 )
|
|
};
|
|
|
|
},
|
|
|
|
_cacheMargins: function() {
|
|
this.margins = {
|
|
left: (parseInt(this.element.css("marginLeft"), 10) || 0),
|
|
top: (parseInt(this.element.css("marginTop"), 10) || 0),
|
|
right: (parseInt(this.element.css("marginRight"), 10) || 0),
|
|
bottom: (parseInt(this.element.css("marginBottom"), 10) || 0)
|
|
};
|
|
},
|
|
|
|
_cacheHelperProportions: function() {
|
|
this.helperProportions = {
|
|
width: this.helper.outerWidth(),
|
|
height: this.helper.outerHeight()
|
|
};
|
|
},
|
|
|
|
_setContainment: function() {
|
|
|
|
var isUserScrollable, c, ce,
|
|
o = this.options,
|
|
document = this.document[ 0 ];
|
|
|
|
this.relativeContainer = null;
|
|
|
|
if ( !o.containment ) {
|
|
this.containment = null;
|
|
return;
|
|
}
|
|
|
|
if ( o.containment === "window" ) {
|
|
this.containment = [
|
|
$( window ).scrollLeft() - this.offset.relative.left - this.offset.parent.left,
|
|
$( window ).scrollTop() - this.offset.relative.top - this.offset.parent.top,
|
|
$( window ).scrollLeft() + $( window ).width() - this.helperProportions.width - this.margins.left,
|
|
$( window ).scrollTop() + ( $( window ).height() || document.body.parentNode.scrollHeight ) - this.helperProportions.height - this.margins.top
|
|
];
|
|
return;
|
|
}
|
|
|
|
if ( o.containment === "document") {
|
|
this.containment = [
|
|
0,
|
|
0,
|
|
$( document ).width() - this.helperProportions.width - this.margins.left,
|
|
( $( document ).height() || document.body.parentNode.scrollHeight ) - this.helperProportions.height - this.margins.top
|
|
];
|
|
return;
|
|
}
|
|
|
|
if ( o.containment.constructor === Array ) {
|
|
this.containment = o.containment;
|
|
return;
|
|
}
|
|
|
|
if ( o.containment === "parent" ) {
|
|
o.containment = this.helper[ 0 ].parentNode;
|
|
}
|
|
|
|
c = $( o.containment );
|
|
ce = c[ 0 ];
|
|
|
|
if ( !ce ) {
|
|
return;
|
|
}
|
|
|
|
isUserScrollable = /(scroll|auto)/.test( c.css( "overflow" ) );
|
|
|
|
this.containment = [
|
|
( parseInt( c.css( "borderLeftWidth" ), 10 ) || 0 ) + ( parseInt( c.css( "paddingLeft" ), 10 ) || 0 ),
|
|
( parseInt( c.css( "borderTopWidth" ), 10 ) || 0 ) + ( parseInt( c.css( "paddingTop" ), 10 ) || 0 ),
|
|
( isUserScrollable ? Math.max( ce.scrollWidth, ce.offsetWidth ) : ce.offsetWidth ) -
|
|
( parseInt( c.css( "borderRightWidth" ), 10 ) || 0 ) -
|
|
( parseInt( c.css( "paddingRight" ), 10 ) || 0 ) -
|
|
this.helperProportions.width -
|
|
this.margins.left -
|
|
this.margins.right,
|
|
( isUserScrollable ? Math.max( ce.scrollHeight, ce.offsetHeight ) : ce.offsetHeight ) -
|
|
( parseInt( c.css( "borderBottomWidth" ), 10 ) || 0 ) -
|
|
( parseInt( c.css( "paddingBottom" ), 10 ) || 0 ) -
|
|
this.helperProportions.height -
|
|
this.margins.top -
|
|
this.margins.bottom
|
|
];
|
|
this.relativeContainer = c;
|
|
},
|
|
|
|
_convertPositionTo: function(d, pos) {
|
|
|
|
if (!pos) {
|
|
pos = this.position;
|
|
}
|
|
|
|
var mod = d === "absolute" ? 1 : -1,
|
|
scrollIsRootNode = this._isRootNode( this.scrollParent[ 0 ] );
|
|
|
|
return {
|
|
top: (
|
|
pos.top + // The absolute mouse position
|
|
this.offset.relative.top * mod + // Only for relative positioned nodes: Relative offset from element to offset parent
|
|
this.offset.parent.top * mod - // The offsetParent's offset without borders (offset + border)
|
|
( ( this.cssPosition === "fixed" ? -this.offset.scroll.top : ( scrollIsRootNode ? 0 : this.offset.scroll.top ) ) * mod)
|
|
),
|
|
left: (
|
|
pos.left + // The absolute mouse position
|
|
this.offset.relative.left * mod + // Only for relative positioned nodes: Relative offset from element to offset parent
|
|
this.offset.parent.left * mod - // The offsetParent's offset without borders (offset + border)
|
|
( ( this.cssPosition === "fixed" ? -this.offset.scroll.left : ( scrollIsRootNode ? 0 : this.offset.scroll.left ) ) * mod)
|
|
)
|
|
};
|
|
|
|
},
|
|
|
|
_generatePosition: function( event, constrainPosition ) {
|
|
|
|
var containment, co, top, left,
|
|
o = this.options,
|
|
scrollIsRootNode = this._isRootNode( this.scrollParent[ 0 ] ),
|
|
pageX = event.pageX,
|
|
pageY = event.pageY;
|
|
|
|
// Cache the scroll
|
|
if ( !scrollIsRootNode || !this.offset.scroll ) {
|
|
this.offset.scroll = {
|
|
top: this.scrollParent.scrollTop(),
|
|
left: this.scrollParent.scrollLeft()
|
|
};
|
|
}
|
|
|
|
/*
|
|
* - Position constraining -
|
|
* Constrain the position to a mix of grid, containment.
|
|
*/
|
|
|
|
// If we are not dragging yet, we won't check for options
|
|
if ( constrainPosition ) {
|
|
if ( this.containment ) {
|
|
if ( this.relativeContainer ){
|
|
co = this.relativeContainer.offset();
|
|
containment = [
|
|
this.containment[ 0 ] + co.left,
|
|
this.containment[ 1 ] + co.top,
|
|
this.containment[ 2 ] + co.left,
|
|
this.containment[ 3 ] + co.top
|
|
];
|
|
} else {
|
|
containment = this.containment;
|
|
}
|
|
|
|
if (event.pageX - this.offset.click.left < containment[0]) {
|
|
pageX = containment[0] + this.offset.click.left;
|
|
}
|
|
if (event.pageY - this.offset.click.top < containment[1]) {
|
|
pageY = containment[1] + this.offset.click.top;
|
|
}
|
|
if (event.pageX - this.offset.click.left > containment[2]) {
|
|
pageX = containment[2] + this.offset.click.left;
|
|
}
|
|
if (event.pageY - this.offset.click.top > containment[3]) {
|
|
pageY = containment[3] + this.offset.click.top;
|
|
}
|
|
}
|
|
|
|
if (o.grid) {
|
|
//Check for grid elements set to 0 to prevent divide by 0 error causing invalid argument errors in IE (see ticket #6950)
|
|
top = o.grid[1] ? this.originalPageY + Math.round((pageY - this.originalPageY) / o.grid[1]) * o.grid[1] : this.originalPageY;
|
|
pageY = containment ? ((top - this.offset.click.top >= containment[1] || top - this.offset.click.top > containment[3]) ? top : ((top - this.offset.click.top >= containment[1]) ? top - o.grid[1] : top + o.grid[1])) : top;
|
|
|
|
left = o.grid[0] ? this.originalPageX + Math.round((pageX - this.originalPageX) / o.grid[0]) * o.grid[0] : this.originalPageX;
|
|
pageX = containment ? ((left - this.offset.click.left >= containment[0] || left - this.offset.click.left > containment[2]) ? left : ((left - this.offset.click.left >= containment[0]) ? left - o.grid[0] : left + o.grid[0])) : left;
|
|
}
|
|
|
|
if ( o.axis === "y" ) {
|
|
pageX = this.originalPageX;
|
|
}
|
|
|
|
if ( o.axis === "x" ) {
|
|
pageY = this.originalPageY;
|
|
}
|
|
}
|
|
|
|
return {
|
|
top: (
|
|
pageY - // The absolute mouse position
|
|
this.offset.click.top - // Click offset (relative to the element)
|
|
this.offset.relative.top - // Only for relative positioned nodes: Relative offset from element to offset parent
|
|
this.offset.parent.top + // The offsetParent's offset without borders (offset + border)
|
|
( this.cssPosition === "fixed" ? -this.offset.scroll.top : ( scrollIsRootNode ? 0 : this.offset.scroll.top ) )
|
|
),
|
|
left: (
|
|
pageX - // The absolute mouse position
|
|
this.offset.click.left - // Click offset (relative to the element)
|
|
this.offset.relative.left - // Only for relative positioned nodes: Relative offset from element to offset parent
|
|
this.offset.parent.left + // The offsetParent's offset without borders (offset + border)
|
|
( this.cssPosition === "fixed" ? -this.offset.scroll.left : ( scrollIsRootNode ? 0 : this.offset.scroll.left ) )
|
|
)
|
|
};
|
|
|
|
},
|
|
|
|
_clear: function() {
|
|
this.helper.removeClass("ui-draggable-dragging");
|
|
if (this.helper[0] !== this.element[0] && !this.cancelHelperRemoval) {
|
|
this.helper.remove();
|
|
}
|
|
this.helper = null;
|
|
this.cancelHelperRemoval = false;
|
|
if ( this.destroyOnClear ) {
|
|
this.destroy();
|
|
}
|
|
},
|
|
|
|
_normalizeRightBottom: function() {
|
|
if ( this.options.axis !== "y" && this.helper.css( "right" ) !== "auto" ) {
|
|
this.helper.width( this.helper.width() );
|
|
this.helper.css( "right", "auto" );
|
|
}
|
|
if ( this.options.axis !== "x" && this.helper.css( "bottom" ) !== "auto" ) {
|
|
this.helper.height( this.helper.height() );
|
|
this.helper.css( "bottom", "auto" );
|
|
}
|
|
},
|
|
|
|
// From now on bulk stuff - mainly helpers
|
|
|
|
_trigger: function( type, event, ui ) {
|
|
ui = ui || this._uiHash();
|
|
$.ui.plugin.call( this, type, [ event, ui, this ], true );
|
|
|
|
// Absolute position and offset (see #6884 ) have to be recalculated after plugins
|
|
if ( /^(drag|start|stop)/.test( type ) ) {
|
|
this.positionAbs = this._convertPositionTo( "absolute" );
|
|
ui.offset = this.positionAbs;
|
|
}
|
|
return $.Widget.prototype._trigger.call( this, type, event, ui );
|
|
},
|
|
|
|
plugins: {},
|
|
|
|
_uiHash: function() {
|
|
return {
|
|
helper: this.helper,
|
|
position: this.position,
|
|
originalPosition: this.originalPosition,
|
|
offset: this.positionAbs
|
|
};
|
|
}
|
|
|
|
});
|
|
|
|
$.ui.plugin.add( "draggable", "connectToSortable", {
|
|
start: function( event, ui, draggable ) {
|
|
var uiSortable = $.extend( {}, ui, {
|
|
item: draggable.element
|
|
});
|
|
|
|
draggable.sortables = [];
|
|
$( draggable.options.connectToSortable ).each(function() {
|
|
var sortable = $( this ).sortable( "instance" );
|
|
|
|
if ( sortable && !sortable.options.disabled ) {
|
|
draggable.sortables.push( sortable );
|
|
|
|
// refreshPositions is called at drag start to refresh the containerCache
|
|
// which is used in drag. This ensures it's initialized and synchronized
|
|
// with any changes that might have happened on the page since initialization.
|
|
sortable.refreshPositions();
|
|
sortable._trigger("activate", event, uiSortable);
|
|
}
|
|
});
|
|
},
|
|
stop: function( event, ui, draggable ) {
|
|
var uiSortable = $.extend( {}, ui, {
|
|
item: draggable.element
|
|
});
|
|
|
|
draggable.cancelHelperRemoval = false;
|
|
|
|
$.each( draggable.sortables, function() {
|
|
var sortable = this;
|
|
|
|
if ( sortable.isOver ) {
|
|
sortable.isOver = 0;
|
|
|
|
// Allow this sortable to handle removing the helper
|
|
draggable.cancelHelperRemoval = true;
|
|
sortable.cancelHelperRemoval = false;
|
|
|
|
// Use _storedCSS To restore properties in the sortable,
|
|
// as this also handles revert (#9675) since the draggable
|
|
// may have modified them in unexpected ways (#8809)
|
|
sortable._storedCSS = {
|
|
position: sortable.placeholder.css( "position" ),
|
|
top: sortable.placeholder.css( "top" ),
|
|
left: sortable.placeholder.css( "left" )
|
|
};
|
|
|
|
sortable._mouseStop(event);
|
|
|
|
// Once drag has ended, the sortable should return to using
|
|
// its original helper, not the shared helper from draggable
|
|
sortable.options.helper = sortable.options._helper;
|
|
} else {
|
|
// Prevent this Sortable from removing the helper.
|
|
// However, don't set the draggable to remove the helper
|
|
// either as another connected Sortable may yet handle the removal.
|
|
sortable.cancelHelperRemoval = true;
|
|
|
|
sortable._trigger( "deactivate", event, uiSortable );
|
|
}
|
|
});
|
|
},
|
|
drag: function( event, ui, draggable ) {
|
|
$.each( draggable.sortables, function() {
|
|
var innermostIntersecting = false,
|
|
sortable = this;
|
|
|
|
// Copy over variables that sortable's _intersectsWith uses
|
|
sortable.positionAbs = draggable.positionAbs;
|
|
sortable.helperProportions = draggable.helperProportions;
|
|
sortable.offset.click = draggable.offset.click;
|
|
|
|
if ( sortable._intersectsWith( sortable.containerCache ) ) {
|
|
innermostIntersecting = true;
|
|
|
|
$.each( draggable.sortables, function() {
|
|
// Copy over variables that sortable's _intersectsWith uses
|
|
this.positionAbs = draggable.positionAbs;
|
|
this.helperProportions = draggable.helperProportions;
|
|
this.offset.click = draggable.offset.click;
|
|
|
|
if ( this !== sortable &&
|
|
this._intersectsWith( this.containerCache ) &&
|
|
$.contains( sortable.element[ 0 ], this.element[ 0 ] ) ) {
|
|
innermostIntersecting = false;
|
|
}
|
|
|
|
return innermostIntersecting;
|
|
});
|
|
}
|
|
|
|
if ( innermostIntersecting ) {
|
|
// If it intersects, we use a little isOver variable and set it once,
|
|
// so that the move-in stuff gets fired only once.
|
|
if ( !sortable.isOver ) {
|
|
sortable.isOver = 1;
|
|
|
|
// Store draggable's parent in case we need to reappend to it later.
|
|
draggable._parent = ui.helper.parent();
|
|
|
|
sortable.currentItem = ui.helper
|
|
.appendTo( sortable.element )
|
|
.data( "ui-sortable-item", true );
|
|
|
|
// Store helper option to later restore it
|
|
sortable.options._helper = sortable.options.helper;
|
|
|
|
sortable.options.helper = function() {
|
|
return ui.helper[ 0 ];
|
|
};
|
|
|
|
// Fire the start events of the sortable with our passed browser event,
|
|
// and our own helper (so it doesn't create a new one)
|
|
event.target = sortable.currentItem[ 0 ];
|
|
sortable._mouseCapture( event, true );
|
|
sortable._mouseStart( event, true, true );
|
|
|
|
// Because the browser event is way off the new appended portlet,
|
|
// modify necessary variables to reflect the changes
|
|
sortable.offset.click.top = draggable.offset.click.top;
|
|
sortable.offset.click.left = draggable.offset.click.left;
|
|
sortable.offset.parent.left -= draggable.offset.parent.left -
|
|
sortable.offset.parent.left;
|
|
sortable.offset.parent.top -= draggable.offset.parent.top -
|
|
sortable.offset.parent.top;
|
|
|
|
draggable._trigger( "toSortable", event );
|
|
|
|
// Inform draggable that the helper is in a valid drop zone,
|
|
// used solely in the revert option to handle "valid/invalid".
|
|
draggable.dropped = sortable.element;
|
|
|
|
// Need to refreshPositions of all sortables in the case that
|
|
// adding to one sortable changes the location of the other sortables (#9675)
|
|
$.each( draggable.sortables, function() {
|
|
this.refreshPositions();
|
|
});
|
|
|
|
// hack so receive/update callbacks work (mostly)
|
|
draggable.currentItem = draggable.element;
|
|
sortable.fromOutside = draggable;
|
|
}
|
|
|
|
if ( sortable.currentItem ) {
|
|
sortable._mouseDrag( event );
|
|
// Copy the sortable's position because the draggable's can potentially reflect
|
|
// a relative position, while sortable is always absolute, which the dragged
|
|
// element has now become. (#8809)
|
|
ui.position = sortable.position;
|
|
}
|
|
} else {
|
|
// If it doesn't intersect with the sortable, and it intersected before,
|
|
// we fake the drag stop of the sortable, but make sure it doesn't remove
|
|
// the helper by using cancelHelperRemoval.
|
|
if ( sortable.isOver ) {
|
|
|
|
sortable.isOver = 0;
|
|
sortable.cancelHelperRemoval = true;
|
|
|
|
// Calling sortable's mouseStop would trigger a revert,
|
|
// so revert must be temporarily false until after mouseStop is called.
|
|
sortable.options._revert = sortable.options.revert;
|
|
sortable.options.revert = false;
|
|
|
|
sortable._trigger( "out", event, sortable._uiHash( sortable ) );
|
|
sortable._mouseStop( event, true );
|
|
|
|
// restore sortable behaviors that were modfied
|
|
// when the draggable entered the sortable area (#9481)
|
|
sortable.options.revert = sortable.options._revert;
|
|
sortable.options.helper = sortable.options._helper;
|
|
|
|
if ( sortable.placeholder ) {
|
|
sortable.placeholder.remove();
|
|
}
|
|
|
|
// Restore and recalculate the draggable's offset considering the sortable
|
|
// may have modified them in unexpected ways. (#8809, #10669)
|
|
ui.helper.appendTo( draggable._parent );
|
|
draggable._refreshOffsets( event );
|
|
ui.position = draggable._generatePosition( event, true );
|
|
|
|
draggable._trigger( "fromSortable", event );
|
|
|
|
// Inform draggable that the helper is no longer in a valid drop zone
|
|
draggable.dropped = false;
|
|
|
|
// Need to refreshPositions of all sortables just in case removing
|
|
// from one sortable changes the location of other sortables (#9675)
|
|
$.each( draggable.sortables, function() {
|
|
this.refreshPositions();
|
|
});
|
|
}
|
|
}
|
|
});
|
|
}
|
|
});
|
|
|
|
$.ui.plugin.add("draggable", "cursor", {
|
|
start: function( event, ui, instance ) {
|
|
var t = $( "body" ),
|
|
o = instance.options;
|
|
|
|
if (t.css("cursor")) {
|
|
o._cursor = t.css("cursor");
|
|
}
|
|
t.css("cursor", o.cursor);
|
|
},
|
|
stop: function( event, ui, instance ) {
|
|
var o = instance.options;
|
|
if (o._cursor) {
|
|
$("body").css("cursor", o._cursor);
|
|
}
|
|
}
|
|
});
|
|
|
|
$.ui.plugin.add("draggable", "opacity", {
|
|
start: function( event, ui, instance ) {
|
|
var t = $( ui.helper ),
|
|
o = instance.options;
|
|
if (t.css("opacity")) {
|
|
o._opacity = t.css("opacity");
|
|
}
|
|
t.css("opacity", o.opacity);
|
|
},
|
|
stop: function( event, ui, instance ) {
|
|
var o = instance.options;
|
|
if (o._opacity) {
|
|
$(ui.helper).css("opacity", o._opacity);
|
|
}
|
|
}
|
|
});
|
|
|
|
$.ui.plugin.add("draggable", "scroll", {
|
|
start: function( event, ui, i ) {
|
|
if ( !i.scrollParentNotHidden ) {
|
|
i.scrollParentNotHidden = i.helper.scrollParent( false );
|
|
}
|
|
|
|
if ( i.scrollParentNotHidden[ 0 ] !== i.document[ 0 ] && i.scrollParentNotHidden[ 0 ].tagName !== "HTML" ) {
|
|
i.overflowOffset = i.scrollParentNotHidden.offset();
|
|
}
|
|
},
|
|
drag: function( event, ui, i ) {
|
|
|
|
var o = i.options,
|
|
scrolled = false,
|
|
scrollParent = i.scrollParentNotHidden[ 0 ],
|
|
document = i.document[ 0 ];
|
|
|
|
if ( scrollParent !== document && scrollParent.tagName !== "HTML" ) {
|
|
if ( !o.axis || o.axis !== "x" ) {
|
|
if ( ( i.overflowOffset.top + scrollParent.offsetHeight ) - event.pageY < o.scrollSensitivity ) {
|
|
scrollParent.scrollTop = scrolled = scrollParent.scrollTop + o.scrollSpeed;
|
|
} else if ( event.pageY - i.overflowOffset.top < o.scrollSensitivity ) {
|
|
scrollParent.scrollTop = scrolled = scrollParent.scrollTop - o.scrollSpeed;
|
|
}
|
|
}
|
|
|
|
if ( !o.axis || o.axis !== "y" ) {
|
|
if ( ( i.overflowOffset.left + scrollParent.offsetWidth ) - event.pageX < o.scrollSensitivity ) {
|
|
scrollParent.scrollLeft = scrolled = scrollParent.scrollLeft + o.scrollSpeed;
|
|
} else if ( event.pageX - i.overflowOffset.left < o.scrollSensitivity ) {
|
|
scrollParent.scrollLeft = scrolled = scrollParent.scrollLeft - o.scrollSpeed;
|
|
}
|
|
}
|
|
|
|
} else {
|
|
|
|
if (!o.axis || o.axis !== "x") {
|
|
if (event.pageY - $(document).scrollTop() < o.scrollSensitivity) {
|
|
scrolled = $(document).scrollTop($(document).scrollTop() - o.scrollSpeed);
|
|
} else if ($(window).height() - (event.pageY - $(document).scrollTop()) < o.scrollSensitivity) {
|
|
scrolled = $(document).scrollTop($(document).scrollTop() + o.scrollSpeed);
|
|
}
|
|
}
|
|
|
|
if (!o.axis || o.axis !== "y") {
|
|
if (event.pageX - $(document).scrollLeft() < o.scrollSensitivity) {
|
|
scrolled = $(document).scrollLeft($(document).scrollLeft() - o.scrollSpeed);
|
|
} else if ($(window).width() - (event.pageX - $(document).scrollLeft()) < o.scrollSensitivity) {
|
|
scrolled = $(document).scrollLeft($(document).scrollLeft() + o.scrollSpeed);
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
if (scrolled !== false && $.ui.ddmanager && !o.dropBehaviour) {
|
|
$.ui.ddmanager.prepareOffsets(i, event);
|
|
}
|
|
|
|
}
|
|
});
|
|
|
|
$.ui.plugin.add("draggable", "snap", {
|
|
start: function( event, ui, i ) {
|
|
|
|
var o = i.options;
|
|
|
|
i.snapElements = [];
|
|
|
|
$(o.snap.constructor !== String ? ( o.snap.items || ":data(ui-draggable)" ) : o.snap).each(function() {
|
|
var $t = $(this),
|
|
$o = $t.offset();
|
|
if (this !== i.element[0]) {
|
|
i.snapElements.push({
|
|
item: this,
|
|
width: $t.outerWidth(), height: $t.outerHeight(),
|
|
top: $o.top, left: $o.left
|
|
});
|
|
}
|
|
});
|
|
|
|
},
|
|
drag: function( event, ui, inst ) {
|
|
|
|
var ts, bs, ls, rs, l, r, t, b, i, first,
|
|
o = inst.options,
|
|
d = o.snapTolerance,
|
|
x1 = ui.offset.left, x2 = x1 + inst.helperProportions.width,
|
|
y1 = ui.offset.top, y2 = y1 + inst.helperProportions.height;
|
|
|
|
for (i = inst.snapElements.length - 1; i >= 0; i--){
|
|
|
|
l = inst.snapElements[i].left - inst.margins.left;
|
|
r = l + inst.snapElements[i].width;
|
|
t = inst.snapElements[i].top - inst.margins.top;
|
|
b = t + inst.snapElements[i].height;
|
|
|
|
if ( x2 < l - d || x1 > r + d || y2 < t - d || y1 > b + d || !$.contains( inst.snapElements[ i ].item.ownerDocument, inst.snapElements[ i ].item ) ) {
|
|
if (inst.snapElements[i].snapping) {
|
|
(inst.options.snap.release && inst.options.snap.release.call(inst.element, event, $.extend(inst._uiHash(), { snapItem: inst.snapElements[i].item })));
|
|
}
|
|
inst.snapElements[i].snapping = false;
|
|
continue;
|
|
}
|
|
|
|
if (o.snapMode !== "inner") {
|
|
ts = Math.abs(t - y2) <= d;
|
|
bs = Math.abs(b - y1) <= d;
|
|
ls = Math.abs(l - x2) <= d;
|
|
rs = Math.abs(r - x1) <= d;
|
|
if (ts) {
|
|
ui.position.top = inst._convertPositionTo("relative", { top: t - inst.helperProportions.height, left: 0 }).top;
|
|
}
|
|
if (bs) {
|
|
ui.position.top = inst._convertPositionTo("relative", { top: b, left: 0 }).top;
|
|
}
|
|
if (ls) {
|
|
ui.position.left = inst._convertPositionTo("relative", { top: 0, left: l - inst.helperProportions.width }).left;
|
|
}
|
|
if (rs) {
|
|
ui.position.left = inst._convertPositionTo("relative", { top: 0, left: r }).left;
|
|
}
|
|
}
|
|
|
|
first = (ts || bs || ls || rs);
|
|
|
|
if (o.snapMode !== "outer") {
|
|
ts = Math.abs(t - y1) <= d;
|
|
bs = Math.abs(b - y2) <= d;
|
|
ls = Math.abs(l - x1) <= d;
|
|
rs = Math.abs(r - x2) <= d;
|
|
if (ts) {
|
|
ui.position.top = inst._convertPositionTo("relative", { top: t, left: 0 }).top;
|
|
}
|
|
if (bs) {
|
|
ui.position.top = inst._convertPositionTo("relative", { top: b - inst.helperProportions.height, left: 0 }).top;
|
|
}
|
|
if (ls) {
|
|
ui.position.left = inst._convertPositionTo("relative", { top: 0, left: l }).left;
|
|
}
|
|
if (rs) {
|
|
ui.position.left = inst._convertPositionTo("relative", { top: 0, left: r - inst.helperProportions.width }).left;
|
|
}
|
|
}
|
|
|
|
if (!inst.snapElements[i].snapping && (ts || bs || ls || rs || first)) {
|
|
(inst.options.snap.snap && inst.options.snap.snap.call(inst.element, event, $.extend(inst._uiHash(), { snapItem: inst.snapElements[i].item })));
|
|
}
|
|
inst.snapElements[i].snapping = (ts || bs || ls || rs || first);
|
|
|
|
}
|
|
|
|
}
|
|
});
|
|
|
|
$.ui.plugin.add("draggable", "stack", {
|
|
start: function( event, ui, instance ) {
|
|
var min,
|
|
o = instance.options,
|
|
group = $.makeArray($(o.stack)).sort(function(a, b) {
|
|
return (parseInt($(a).css("zIndex"), 10) || 0) - (parseInt($(b).css("zIndex"), 10) || 0);
|
|
});
|
|
|
|
if (!group.length) { return; }
|
|
|
|
min = parseInt($(group[0]).css("zIndex"), 10) || 0;
|
|
$(group).each(function(i) {
|
|
$(this).css("zIndex", min + i);
|
|
});
|
|
this.css("zIndex", (min + group.length));
|
|
}
|
|
});
|
|
|
|
$.ui.plugin.add("draggable", "zIndex", {
|
|
start: function( event, ui, instance ) {
|
|
var t = $( ui.helper ),
|
|
o = instance.options;
|
|
|
|
if (t.css("zIndex")) {
|
|
o._zIndex = t.css("zIndex");
|
|
}
|
|
t.css("zIndex", o.zIndex);
|
|
},
|
|
stop: function( event, ui, instance ) {
|
|
var o = instance.options;
|
|
|
|
if (o._zIndex) {
|
|
$(ui.helper).css("zIndex", o._zIndex);
|
|
}
|
|
}
|
|
});
|
|
|
|
var draggable = $.ui.draggable;
|
|
|
|
|
|
/*!
|
|
* jQuery UI Resizable 1.11.4
|
|
* http://jqueryui.com
|
|
*
|
|
* Copyright jQuery Foundation and other contributors
|
|
* Released under the MIT license.
|
|
* http://jquery.org/license
|
|
*
|
|
* http://api.jqueryui.com/resizable/
|
|
*/
|
|
|
|
|
|
$.widget("ui.resizable", $.ui.mouse, {
|
|
version: "1.11.4",
|
|
widgetEventPrefix: "resize",
|
|
options: {
|
|
alsoResize: false,
|
|
animate: false,
|
|
animateDuration: "slow",
|
|
animateEasing: "swing",
|
|
aspectRatio: false,
|
|
autoHide: false,
|
|
containment: false,
|
|
ghost: false,
|
|
grid: false,
|
|
handles: "e,s,se",
|
|
helper: false,
|
|
maxHeight: null,
|
|
maxWidth: null,
|
|
minHeight: 10,
|
|
minWidth: 10,
|
|
// See #7960
|
|
zIndex: 90,
|
|
|
|
// callbacks
|
|
resize: null,
|
|
start: null,
|
|
stop: null
|
|
},
|
|
|
|
_num: function( value ) {
|
|
return parseInt( value, 10 ) || 0;
|
|
},
|
|
|
|
_isNumber: function( value ) {
|
|
return !isNaN( parseInt( value, 10 ) );
|
|
},
|
|
|
|
_hasScroll: function( el, a ) {
|
|
|
|
if ( $( el ).css( "overflow" ) === "hidden") {
|
|
return false;
|
|
}
|
|
|
|
var scroll = ( a && a === "left" ) ? "scrollLeft" : "scrollTop",
|
|
has = false;
|
|
|
|
if ( el[ scroll ] > 0 ) {
|
|
return true;
|
|
}
|
|
|
|
// TODO: determine which cases actually cause this to happen
|
|
// if the element doesn't have the scroll set, see if it's possible to
|
|
// set the scroll
|
|
el[ scroll ] = 1;
|
|
has = ( el[ scroll ] > 0 );
|
|
el[ scroll ] = 0;
|
|
return has;
|
|
},
|
|
|
|
_create: function() {
|
|
|
|
var n, i, handle, axis, hname,
|
|
that = this,
|
|
o = this.options;
|
|
this.element.addClass("ui-resizable");
|
|
|
|
$.extend(this, {
|
|
_aspectRatio: !!(o.aspectRatio),
|
|
aspectRatio: o.aspectRatio,
|
|
originalElement: this.element,
|
|
_proportionallyResizeElements: [],
|
|
_helper: o.helper || o.ghost || o.animate ? o.helper || "ui-resizable-helper" : null
|
|
});
|
|
|
|
// Wrap the element if it cannot hold child nodes
|
|
if (this.element[0].nodeName.match(/^(canvas|textarea|input|select|button|img)$/i)) {
|
|
|
|
this.element.wrap(
|
|
$("<div class='ui-wrapper' style='overflow: hidden;'></div>").css({
|
|
position: this.element.css("position"),
|
|
width: this.element.outerWidth(),
|
|
height: this.element.outerHeight(),
|
|
top: this.element.css("top"),
|
|
left: this.element.css("left")
|
|
})
|
|
);
|
|
|
|
this.element = this.element.parent().data(
|
|
"ui-resizable", this.element.resizable( "instance" )
|
|
);
|
|
|
|
this.elementIsWrapper = true;
|
|
|
|
this.element.css({
|
|
marginLeft: this.originalElement.css("marginLeft"),
|
|
marginTop: this.originalElement.css("marginTop"),
|
|
marginRight: this.originalElement.css("marginRight"),
|
|
marginBottom: this.originalElement.css("marginBottom")
|
|
});
|
|
this.originalElement.css({
|
|
marginLeft: 0,
|
|
marginTop: 0,
|
|
marginRight: 0,
|
|
marginBottom: 0
|
|
});
|
|
// support: Safari
|
|
// Prevent Safari textarea resize
|
|
this.originalResizeStyle = this.originalElement.css("resize");
|
|
this.originalElement.css("resize", "none");
|
|
|
|
this._proportionallyResizeElements.push( this.originalElement.css({
|
|
position: "static",
|
|
zoom: 1,
|
|
display: "block"
|
|
}) );
|
|
|
|
// support: IE9
|
|
// avoid IE jump (hard set the margin)
|
|
this.originalElement.css({ margin: this.originalElement.css("margin") });
|
|
|
|
this._proportionallyResize();
|
|
}
|
|
|
|
this.handles = o.handles ||
|
|
( !$(".ui-resizable-handle", this.element).length ?
|
|
"e,s,se" : {
|
|
n: ".ui-resizable-n",
|
|
e: ".ui-resizable-e",
|
|
s: ".ui-resizable-s",
|
|
w: ".ui-resizable-w",
|
|
se: ".ui-resizable-se",
|
|
sw: ".ui-resizable-sw",
|
|
ne: ".ui-resizable-ne",
|
|
nw: ".ui-resizable-nw"
|
|
} );
|
|
|
|
this._handles = $();
|
|
if ( this.handles.constructor === String ) {
|
|
|
|
if ( this.handles === "all") {
|
|
this.handles = "n,e,s,w,se,sw,ne,nw";
|
|
}
|
|
|
|
n = this.handles.split(",");
|
|
this.handles = {};
|
|
|
|
for (i = 0; i < n.length; i++) {
|
|
|
|
handle = $.trim(n[i]);
|
|
hname = "ui-resizable-" + handle;
|
|
axis = $("<div class='ui-resizable-handle " + hname + "'></div>");
|
|
|
|
axis.css({ zIndex: o.zIndex });
|
|
|
|
// TODO : What's going on here?
|
|
if ("se" === handle) {
|
|
axis.addClass("ui-icon ui-icon-gripsmall-diagonal-se");
|
|
}
|
|
|
|
this.handles[handle] = ".ui-resizable-" + handle;
|
|
this.element.append(axis);
|
|
}
|
|
|
|
}
|
|
|
|
this._renderAxis = function(target) {
|
|
|
|
var i, axis, padPos, padWrapper;
|
|
|
|
target = target || this.element;
|
|
|
|
for (i in this.handles) {
|
|
|
|
if (this.handles[i].constructor === String) {
|
|
this.handles[i] = this.element.children( this.handles[ i ] ).first().show();
|
|
} else if ( this.handles[ i ].jquery || this.handles[ i ].nodeType ) {
|
|
this.handles[ i ] = $( this.handles[ i ] );
|
|
this._on( this.handles[ i ], { "mousedown": that._mouseDown });
|
|
}
|
|
|
|
if (this.elementIsWrapper && this.originalElement[0].nodeName.match(/^(textarea|input|select|button)$/i)) {
|
|
|
|
axis = $(this.handles[i], this.element);
|
|
|
|
padWrapper = /sw|ne|nw|se|n|s/.test(i) ? axis.outerHeight() : axis.outerWidth();
|
|
|
|
padPos = [ "padding",
|
|
/ne|nw|n/.test(i) ? "Top" :
|
|
/se|sw|s/.test(i) ? "Bottom" :
|
|
/^e$/.test(i) ? "Right" : "Left" ].join("");
|
|
|
|
target.css(padPos, padWrapper);
|
|
|
|
this._proportionallyResize();
|
|
}
|
|
|
|
this._handles = this._handles.add( this.handles[ i ] );
|
|
}
|
|
};
|
|
|
|
// TODO: make renderAxis a prototype function
|
|
this._renderAxis(this.element);
|
|
|
|
this._handles = this._handles.add( this.element.find( ".ui-resizable-handle" ) );
|
|
this._handles.disableSelection();
|
|
|
|
this._handles.mouseover(function() {
|
|
if (!that.resizing) {
|
|
if (this.className) {
|
|
axis = this.className.match(/ui-resizable-(se|sw|ne|nw|n|e|s|w)/i);
|
|
}
|
|
that.axis = axis && axis[1] ? axis[1] : "se";
|
|
}
|
|
});
|
|
|
|
if (o.autoHide) {
|
|
this._handles.hide();
|
|
$(this.element)
|
|
.addClass("ui-resizable-autohide")
|
|
.mouseenter(function() {
|
|
if (o.disabled) {
|
|
return;
|
|
}
|
|
$(this).removeClass("ui-resizable-autohide");
|
|
that._handles.show();
|
|
})
|
|
.mouseleave(function() {
|
|
if (o.disabled) {
|
|
return;
|
|
}
|
|
if (!that.resizing) {
|
|
$(this).addClass("ui-resizable-autohide");
|
|
that._handles.hide();
|
|
}
|
|
});
|
|
}
|
|
|
|
this._mouseInit();
|
|
},
|
|
|
|
_destroy: function() {
|
|
|
|
this._mouseDestroy();
|
|
|
|
var wrapper,
|
|
_destroy = function(exp) {
|
|
$(exp)
|
|
.removeClass("ui-resizable ui-resizable-disabled ui-resizable-resizing")
|
|
.removeData("resizable")
|
|
.removeData("ui-resizable")
|
|
.unbind(".resizable")
|
|
.find(".ui-resizable-handle")
|
|
.remove();
|
|
};
|
|
|
|
// TODO: Unwrap at same DOM position
|
|
if (this.elementIsWrapper) {
|
|
_destroy(this.element);
|
|
wrapper = this.element;
|
|
this.originalElement.css({
|
|
position: wrapper.css("position"),
|
|
width: wrapper.outerWidth(),
|
|
height: wrapper.outerHeight(),
|
|
top: wrapper.css("top"),
|
|
left: wrapper.css("left")
|
|
}).insertAfter( wrapper );
|
|
wrapper.remove();
|
|
}
|
|
|
|
this.originalElement.css("resize", this.originalResizeStyle);
|
|
_destroy(this.originalElement);
|
|
|
|
return this;
|
|
},
|
|
|
|
_mouseCapture: function(event) {
|
|
var i, handle,
|
|
capture = false;
|
|
|
|
for (i in this.handles) {
|
|
handle = $(this.handles[i])[0];
|
|
if (handle === event.target || $.contains(handle, event.target)) {
|
|
capture = true;
|
|
}
|
|
}
|
|
|
|
return !this.options.disabled && capture;
|
|
},
|
|
|
|
_mouseStart: function(event) {
|
|
|
|
var curleft, curtop, cursor,
|
|
o = this.options,
|
|
el = this.element;
|
|
|
|
this.resizing = true;
|
|
|
|
this._renderProxy();
|
|
|
|
curleft = this._num(this.helper.css("left"));
|
|
curtop = this._num(this.helper.css("top"));
|
|
|
|
if (o.containment) {
|
|
curleft += $(o.containment).scrollLeft() || 0;
|
|
curtop += $(o.containment).scrollTop() || 0;
|
|
}
|
|
|
|
this.offset = this.helper.offset();
|
|
this.position = { left: curleft, top: curtop };
|
|
|
|
this.size = this._helper ? {
|
|
width: this.helper.width(),
|
|
height: this.helper.height()
|
|
} : {
|
|
width: el.width(),
|
|
height: el.height()
|
|
};
|
|
|
|
this.originalSize = this._helper ? {
|
|
width: el.outerWidth(),
|
|
height: el.outerHeight()
|
|
} : {
|
|
width: el.width(),
|
|
height: el.height()
|
|
};
|
|
|
|
this.sizeDiff = {
|
|
width: el.outerWidth() - el.width(),
|
|
height: el.outerHeight() - el.height()
|
|
};
|
|
|
|
this.originalPosition = { left: curleft, top: curtop };
|
|
this.originalMousePosition = { left: event.pageX, top: event.pageY };
|
|
|
|
this.aspectRatio = (typeof o.aspectRatio === "number") ?
|
|
o.aspectRatio :
|
|
((this.originalSize.width / this.originalSize.height) || 1);
|
|
|
|
cursor = $(".ui-resizable-" + this.axis).css("cursor");
|
|
$("body").css("cursor", cursor === "auto" ? this.axis + "-resize" : cursor);
|
|
|
|
el.addClass("ui-resizable-resizing");
|
|
this._propagate("start", event);
|
|
return true;
|
|
},
|
|
|
|
_mouseDrag: function(event) {
|
|
|
|
var data, props,
|
|
smp = this.originalMousePosition,
|
|
a = this.axis,
|
|
dx = (event.pageX - smp.left) || 0,
|
|
dy = (event.pageY - smp.top) || 0,
|
|
trigger = this._change[a];
|
|
|
|
this._updatePrevProperties();
|
|
|
|
if (!trigger) {
|
|
return false;
|
|
}
|
|
|
|
data = trigger.apply(this, [ event, dx, dy ]);
|
|
|
|
this._updateVirtualBoundaries(event.shiftKey);
|
|
if (this._aspectRatio || event.shiftKey) {
|
|
data = this._updateRatio(data, event);
|
|
}
|
|
|
|
data = this._respectSize(data, event);
|
|
|
|
this._updateCache(data);
|
|
|
|
this._propagate("resize", event);
|
|
|
|
props = this._applyChanges();
|
|
|
|
if ( !this._helper && this._proportionallyResizeElements.length ) {
|
|
this._proportionallyResize();
|
|
}
|
|
|
|
if ( !$.isEmptyObject( props ) ) {
|
|
this._updatePrevProperties();
|
|
this._trigger( "resize", event, this.ui() );
|
|
this._applyChanges();
|
|
}
|
|
|
|
return false;
|
|
},
|
|
|
|
_mouseStop: function(event) {
|
|
|
|
this.resizing = false;
|
|
var pr, ista, soffseth, soffsetw, s, left, top,
|
|
o = this.options, that = this;
|
|
|
|
if (this._helper) {
|
|
|
|
pr = this._proportionallyResizeElements;
|
|
ista = pr.length && (/textarea/i).test(pr[0].nodeName);
|
|
soffseth = ista && this._hasScroll(pr[0], "left") ? 0 : that.sizeDiff.height;
|
|
soffsetw = ista ? 0 : that.sizeDiff.width;
|
|
|
|
s = {
|
|
width: (that.helper.width() - soffsetw),
|
|
height: (that.helper.height() - soffseth)
|
|
};
|
|
left = (parseInt(that.element.css("left"), 10) +
|
|
(that.position.left - that.originalPosition.left)) || null;
|
|
top = (parseInt(that.element.css("top"), 10) +
|
|
(that.position.top - that.originalPosition.top)) || null;
|
|
|
|
if (!o.animate) {
|
|
this.element.css($.extend(s, { top: top, left: left }));
|
|
}
|
|
|
|
that.helper.height(that.size.height);
|
|
that.helper.width(that.size.width);
|
|
|
|
if (this._helper && !o.animate) {
|
|
this._proportionallyResize();
|
|
}
|
|
}
|
|
|
|
$("body").css("cursor", "auto");
|
|
|
|
this.element.removeClass("ui-resizable-resizing");
|
|
|
|
this._propagate("stop", event);
|
|
|
|
if (this._helper) {
|
|
this.helper.remove();
|
|
}
|
|
|
|
return false;
|
|
|
|
},
|
|
|
|
_updatePrevProperties: function() {
|
|
this.prevPosition = {
|
|
top: this.position.top,
|
|
left: this.position.left
|
|
};
|
|
this.prevSize = {
|
|
width: this.size.width,
|
|
height: this.size.height
|
|
};
|
|
},
|
|
|
|
_applyChanges: function() {
|
|
var props = {};
|
|
|
|
if ( this.position.top !== this.prevPosition.top ) {
|
|
props.top = this.position.top + "px";
|
|
}
|
|
if ( this.position.left !== this.prevPosition.left ) {
|
|
props.left = this.position.left + "px";
|
|
}
|
|
if ( this.size.width !== this.prevSize.width ) {
|
|
props.width = this.size.width + "px";
|
|
}
|
|
if ( this.size.height !== this.prevSize.height ) {
|
|
props.height = this.size.height + "px";
|
|
}
|
|
|
|
this.helper.css( props );
|
|
|
|
return props;
|
|
},
|
|
|
|
_updateVirtualBoundaries: function(forceAspectRatio) {
|
|
var pMinWidth, pMaxWidth, pMinHeight, pMaxHeight, b,
|
|
o = this.options;
|
|
|
|
b = {
|
|
minWidth: this._isNumber(o.minWidth) ? o.minWidth : 0,
|
|
maxWidth: this._isNumber(o.maxWidth) ? o.maxWidth : Infinity,
|
|
minHeight: this._isNumber(o.minHeight) ? o.minHeight : 0,
|
|
maxHeight: this._isNumber(o.maxHeight) ? o.maxHeight : Infinity
|
|
};
|
|
|
|
if (this._aspectRatio || forceAspectRatio) {
|
|
pMinWidth = b.minHeight * this.aspectRatio;
|
|
pMinHeight = b.minWidth / this.aspectRatio;
|
|
pMaxWidth = b.maxHeight * this.aspectRatio;
|
|
pMaxHeight = b.maxWidth / this.aspectRatio;
|
|
|
|
if (pMinWidth > b.minWidth) {
|
|
b.minWidth = pMinWidth;
|
|
}
|
|
if (pMinHeight > b.minHeight) {
|
|
b.minHeight = pMinHeight;
|
|
}
|
|
if (pMaxWidth < b.maxWidth) {
|
|
b.maxWidth = pMaxWidth;
|
|
}
|
|
if (pMaxHeight < b.maxHeight) {
|
|
b.maxHeight = pMaxHeight;
|
|
}
|
|
}
|
|
this._vBoundaries = b;
|
|
},
|
|
|
|
_updateCache: function(data) {
|
|
this.offset = this.helper.offset();
|
|
if (this._isNumber(data.left)) {
|
|
this.position.left = data.left;
|
|
}
|
|
if (this._isNumber(data.top)) {
|
|
this.position.top = data.top;
|
|
}
|
|
if (this._isNumber(data.height)) {
|
|
this.size.height = data.height;
|
|
}
|
|
if (this._isNumber(data.width)) {
|
|
this.size.width = data.width;
|
|
}
|
|
},
|
|
|
|
_updateRatio: function( data ) {
|
|
|
|
var cpos = this.position,
|
|
csize = this.size,
|
|
a = this.axis;
|
|
|
|
if (this._isNumber(data.height)) {
|
|
data.width = (data.height * this.aspectRatio);
|
|
} else if (this._isNumber(data.width)) {
|
|
data.height = (data.width / this.aspectRatio);
|
|
}
|
|
|
|
if (a === "sw") {
|
|
data.left = cpos.left + (csize.width - data.width);
|
|
data.top = null;
|
|
}
|
|
if (a === "nw") {
|
|
data.top = cpos.top + (csize.height - data.height);
|
|
data.left = cpos.left + (csize.width - data.width);
|
|
}
|
|
|
|
return data;
|
|
},
|
|
|
|
_respectSize: function( data ) {
|
|
|
|
var o = this._vBoundaries,
|
|
a = this.axis,
|
|
ismaxw = this._isNumber(data.width) && o.maxWidth && (o.maxWidth < data.width),
|
|
ismaxh = this._isNumber(data.height) && o.maxHeight && (o.maxHeight < data.height),
|
|
isminw = this._isNumber(data.width) && o.minWidth && (o.minWidth > data.width),
|
|
isminh = this._isNumber(data.height) && o.minHeight && (o.minHeight > data.height),
|
|
dw = this.originalPosition.left + this.originalSize.width,
|
|
dh = this.position.top + this.size.height,
|
|
cw = /sw|nw|w/.test(a), ch = /nw|ne|n/.test(a);
|
|
if (isminw) {
|
|
data.width = o.minWidth;
|
|
}
|
|
if (isminh) {
|
|
data.height = o.minHeight;
|
|
}
|
|
if (ismaxw) {
|
|
data.width = o.maxWidth;
|
|
}
|
|
if (ismaxh) {
|
|
data.height = o.maxHeight;
|
|
}
|
|
|
|
if (isminw && cw) {
|
|
data.left = dw - o.minWidth;
|
|
}
|
|
if (ismaxw && cw) {
|
|
data.left = dw - o.maxWidth;
|
|
}
|
|
if (isminh && ch) {
|
|
data.top = dh - o.minHeight;
|
|
}
|
|
if (ismaxh && ch) {
|
|
data.top = dh - o.maxHeight;
|
|
}
|
|
|
|
// Fixing jump error on top/left - bug #2330
|
|
if (!data.width && !data.height && !data.left && data.top) {
|
|
data.top = null;
|
|
} else if (!data.width && !data.height && !data.top && data.left) {
|
|
data.left = null;
|
|
}
|
|
|
|
return data;
|
|
},
|
|
|
|
_getPaddingPlusBorderDimensions: function( element ) {
|
|
var i = 0,
|
|
widths = [],
|
|
borders = [
|
|
element.css( "borderTopWidth" ),
|
|
element.css( "borderRightWidth" ),
|
|
element.css( "borderBottomWidth" ),
|
|
element.css( "borderLeftWidth" )
|
|
],
|
|
paddings = [
|
|
element.css( "paddingTop" ),
|
|
element.css( "paddingRight" ),
|
|
element.css( "paddingBottom" ),
|
|
element.css( "paddingLeft" )
|
|
];
|
|
|
|
for ( ; i < 4; i++ ) {
|
|
widths[ i ] = ( parseInt( borders[ i ], 10 ) || 0 );
|
|
widths[ i ] += ( parseInt( paddings[ i ], 10 ) || 0 );
|
|
}
|
|
|
|
return {
|
|
height: widths[ 0 ] + widths[ 2 ],
|
|
width: widths[ 1 ] + widths[ 3 ]
|
|
};
|
|
},
|
|
|
|
_proportionallyResize: function() {
|
|
|
|
if (!this._proportionallyResizeElements.length) {
|
|
return;
|
|
}
|
|
|
|
var prel,
|
|
i = 0,
|
|
element = this.helper || this.element;
|
|
|
|
for ( ; i < this._proportionallyResizeElements.length; i++) {
|
|
|
|
prel = this._proportionallyResizeElements[i];
|
|
|
|
// TODO: Seems like a bug to cache this.outerDimensions
|
|
// considering that we are in a loop.
|
|
if (!this.outerDimensions) {
|
|
this.outerDimensions = this._getPaddingPlusBorderDimensions( prel );
|
|
}
|
|
|
|
prel.css({
|
|
height: (element.height() - this.outerDimensions.height) || 0,
|
|
width: (element.width() - this.outerDimensions.width) || 0
|
|
});
|
|
|
|
}
|
|
|
|
},
|
|
|
|
_renderProxy: function() {
|
|
|
|
var el = this.element, o = this.options;
|
|
this.elementOffset = el.offset();
|
|
|
|
if (this._helper) {
|
|
|
|
this.helper = this.helper || $("<div style='overflow:hidden;'></div>");
|
|
|
|
this.helper.addClass(this._helper).css({
|
|
width: this.element.outerWidth() - 1,
|
|
height: this.element.outerHeight() - 1,
|
|
position: "absolute",
|
|
left: this.elementOffset.left + "px",
|
|
top: this.elementOffset.top + "px",
|
|
zIndex: ++o.zIndex //TODO: Don't modify option
|
|
});
|
|
|
|
this.helper
|
|
.appendTo("body")
|
|
.disableSelection();
|
|
|
|
} else {
|
|
this.helper = this.element;
|
|
}
|
|
|
|
},
|
|
|
|
_change: {
|
|
e: function(event, dx) {
|
|
return { width: this.originalSize.width + dx };
|
|
},
|
|
w: function(event, dx) {
|
|
var cs = this.originalSize, sp = this.originalPosition;
|
|
return { left: sp.left + dx, width: cs.width - dx };
|
|
},
|
|
n: function(event, dx, dy) {
|
|
var cs = this.originalSize, sp = this.originalPosition;
|
|
return { top: sp.top + dy, height: cs.height - dy };
|
|
},
|
|
s: function(event, dx, dy) {
|
|
return { height: this.originalSize.height + dy };
|
|
},
|
|
se: function(event, dx, dy) {
|
|
return $.extend(this._change.s.apply(this, arguments),
|
|
this._change.e.apply(this, [ event, dx, dy ]));
|
|
},
|
|
sw: function(event, dx, dy) {
|
|
return $.extend(this._change.s.apply(this, arguments),
|
|
this._change.w.apply(this, [ event, dx, dy ]));
|
|
},
|
|
ne: function(event, dx, dy) {
|
|
return $.extend(this._change.n.apply(this, arguments),
|
|
this._change.e.apply(this, [ event, dx, dy ]));
|
|
},
|
|
nw: function(event, dx, dy) {
|
|
return $.extend(this._change.n.apply(this, arguments),
|
|
this._change.w.apply(this, [ event, dx, dy ]));
|
|
}
|
|
},
|
|
|
|
_propagate: function(n, event) {
|
|
$.ui.plugin.call(this, n, [ event, this.ui() ]);
|
|
(n !== "resize" && this._trigger(n, event, this.ui()));
|
|
},
|
|
|
|
plugins: {},
|
|
|
|
ui: function() {
|
|
return {
|
|
originalElement: this.originalElement,
|
|
element: this.element,
|
|
helper: this.helper,
|
|
position: this.position,
|
|
size: this.size,
|
|
originalSize: this.originalSize,
|
|
originalPosition: this.originalPosition
|
|
};
|
|
}
|
|
|
|
});
|
|
|
|
/*
|
|
* Resizable Extensions
|
|
*/
|
|
|
|
$.ui.plugin.add("resizable", "animate", {
|
|
|
|
stop: function( event ) {
|
|
var that = $(this).resizable( "instance" ),
|
|
o = that.options,
|
|
pr = that._proportionallyResizeElements,
|
|
ista = pr.length && (/textarea/i).test(pr[0].nodeName),
|
|
soffseth = ista && that._hasScroll(pr[0], "left") ? 0 : that.sizeDiff.height,
|
|
soffsetw = ista ? 0 : that.sizeDiff.width,
|
|
style = { width: (that.size.width - soffsetw), height: (that.size.height - soffseth) },
|
|
left = (parseInt(that.element.css("left"), 10) +
|
|
(that.position.left - that.originalPosition.left)) || null,
|
|
top = (parseInt(that.element.css("top"), 10) +
|
|
(that.position.top - that.originalPosition.top)) || null;
|
|
|
|
that.element.animate(
|
|
$.extend(style, top && left ? { top: top, left: left } : {}), {
|
|
duration: o.animateDuration,
|
|
easing: o.animateEasing,
|
|
step: function() {
|
|
|
|
var data = {
|
|
width: parseInt(that.element.css("width"), 10),
|
|
height: parseInt(that.element.css("height"), 10),
|
|
top: parseInt(that.element.css("top"), 10),
|
|
left: parseInt(that.element.css("left"), 10)
|
|
};
|
|
|
|
if (pr && pr.length) {
|
|
$(pr[0]).css({ width: data.width, height: data.height });
|
|
}
|
|
|
|
// propagating resize, and updating values for each animation step
|
|
that._updateCache(data);
|
|
that._propagate("resize", event);
|
|
|
|
}
|
|
}
|
|
);
|
|
}
|
|
|
|
});
|
|
|
|
$.ui.plugin.add( "resizable", "containment", {
|
|
|
|
start: function() {
|
|
var element, p, co, ch, cw, width, height,
|
|
that = $( this ).resizable( "instance" ),
|
|
o = that.options,
|
|
el = that.element,
|
|
oc = o.containment,
|
|
ce = ( oc instanceof $ ) ? oc.get( 0 ) : ( /parent/.test( oc ) ) ? el.parent().get( 0 ) : oc;
|
|
|
|
if ( !ce ) {
|
|
return;
|
|
}
|
|
|
|
that.containerElement = $( ce );
|
|
|
|
if ( /document/.test( oc ) || oc === document ) {
|
|
that.containerOffset = {
|
|
left: 0,
|
|
top: 0
|
|
};
|
|
that.containerPosition = {
|
|
left: 0,
|
|
top: 0
|
|
};
|
|
|
|
that.parentData = {
|
|
element: $( document ),
|
|
left: 0,
|
|
top: 0,
|
|
width: $( document ).width(),
|
|
height: $( document ).height() || document.body.parentNode.scrollHeight
|
|
};
|
|
} else {
|
|
element = $( ce );
|
|
p = [];
|
|
$([ "Top", "Right", "Left", "Bottom" ]).each(function( i, name ) {
|
|
p[ i ] = that._num( element.css( "padding" + name ) );
|
|
});
|
|
|
|
that.containerOffset = element.offset();
|
|
that.containerPosition = element.position();
|
|
that.containerSize = {
|
|
height: ( element.innerHeight() - p[ 3 ] ),
|
|
width: ( element.innerWidth() - p[ 1 ] )
|
|
};
|
|
|
|
co = that.containerOffset;
|
|
ch = that.containerSize.height;
|
|
cw = that.containerSize.width;
|
|
width = ( that._hasScroll ( ce, "left" ) ? ce.scrollWidth : cw );
|
|
height = ( that._hasScroll ( ce ) ? ce.scrollHeight : ch ) ;
|
|
|
|
that.parentData = {
|
|
element: ce,
|
|
left: co.left,
|
|
top: co.top,
|
|
width: width,
|
|
height: height
|
|
};
|
|
}
|
|
},
|
|
|
|
resize: function( event ) {
|
|
var woset, hoset, isParent, isOffsetRelative,
|
|
that = $( this ).resizable( "instance" ),
|
|
o = that.options,
|
|
co = that.containerOffset,
|
|
cp = that.position,
|
|
pRatio = that._aspectRatio || event.shiftKey,
|
|
cop = {
|
|
top: 0,
|
|
left: 0
|
|
},
|
|
ce = that.containerElement,
|
|
continueResize = true;
|
|
|
|
if ( ce[ 0 ] !== document && ( /static/ ).test( ce.css( "position" ) ) ) {
|
|
cop = co;
|
|
}
|
|
|
|
if ( cp.left < ( that._helper ? co.left : 0 ) ) {
|
|
that.size.width = that.size.width +
|
|
( that._helper ?
|
|
( that.position.left - co.left ) :
|
|
( that.position.left - cop.left ) );
|
|
|
|
if ( pRatio ) {
|
|
that.size.height = that.size.width / that.aspectRatio;
|
|
continueResize = false;
|
|
}
|
|
that.position.left = o.helper ? co.left : 0;
|
|
}
|
|
|
|
if ( cp.top < ( that._helper ? co.top : 0 ) ) {
|
|
that.size.height = that.size.height +
|
|
( that._helper ?
|
|
( that.position.top - co.top ) :
|
|
that.position.top );
|
|
|
|
if ( pRatio ) {
|
|
that.size.width = that.size.height * that.aspectRatio;
|
|
continueResize = false;
|
|
}
|
|
that.position.top = that._helper ? co.top : 0;
|
|
}
|
|
|
|
isParent = that.containerElement.get( 0 ) === that.element.parent().get( 0 );
|
|
isOffsetRelative = /relative|absolute/.test( that.containerElement.css( "position" ) );
|
|
|
|
if ( isParent && isOffsetRelative ) {
|
|
that.offset.left = that.parentData.left + that.position.left;
|
|
that.offset.top = that.parentData.top + that.position.top;
|
|
} else {
|
|
that.offset.left = that.element.offset().left;
|
|
that.offset.top = that.element.offset().top;
|
|
}
|
|
|
|
woset = Math.abs( that.sizeDiff.width +
|
|
(that._helper ?
|
|
that.offset.left - cop.left :
|
|
(that.offset.left - co.left)) );
|
|
|
|
hoset = Math.abs( that.sizeDiff.height +
|
|
(that._helper ?
|
|
that.offset.top - cop.top :
|
|
(that.offset.top - co.top)) );
|
|
|
|
if ( woset + that.size.width >= that.parentData.width ) {
|
|
that.size.width = that.parentData.width - woset;
|
|
if ( pRatio ) {
|
|
that.size.height = that.size.width / that.aspectRatio;
|
|
continueResize = false;
|
|
}
|
|
}
|
|
|
|
if ( hoset + that.size.height >= that.parentData.height ) {
|
|
that.size.height = that.parentData.height - hoset;
|
|
if ( pRatio ) {
|
|
that.size.width = that.size.height * that.aspectRatio;
|
|
continueResize = false;
|
|
}
|
|
}
|
|
|
|
if ( !continueResize ) {
|
|
that.position.left = that.prevPosition.left;
|
|
that.position.top = that.prevPosition.top;
|
|
that.size.width = that.prevSize.width;
|
|
that.size.height = that.prevSize.height;
|
|
}
|
|
},
|
|
|
|
stop: function() {
|
|
var that = $( this ).resizable( "instance" ),
|
|
o = that.options,
|
|
co = that.containerOffset,
|
|
cop = that.containerPosition,
|
|
ce = that.containerElement,
|
|
helper = $( that.helper ),
|
|
ho = helper.offset(),
|
|
w = helper.outerWidth() - that.sizeDiff.width,
|
|
h = helper.outerHeight() - that.sizeDiff.height;
|
|
|
|
if ( that._helper && !o.animate && ( /relative/ ).test( ce.css( "position" ) ) ) {
|
|
$( this ).css({
|
|
left: ho.left - cop.left - co.left,
|
|
width: w,
|
|
height: h
|
|
});
|
|
}
|
|
|
|
if ( that._helper && !o.animate && ( /static/ ).test( ce.css( "position" ) ) ) {
|
|
$( this ).css({
|
|
left: ho.left - cop.left - co.left,
|
|
width: w,
|
|
height: h
|
|
});
|
|
}
|
|
}
|
|
});
|
|
|
|
$.ui.plugin.add("resizable", "alsoResize", {
|
|
|
|
start: function() {
|
|
var that = $(this).resizable( "instance" ),
|
|
o = that.options;
|
|
|
|
$(o.alsoResize).each(function() {
|
|
var el = $(this);
|
|
el.data("ui-resizable-alsoresize", {
|
|
width: parseInt(el.width(), 10), height: parseInt(el.height(), 10),
|
|
left: parseInt(el.css("left"), 10), top: parseInt(el.css("top"), 10)
|
|
});
|
|
});
|
|
},
|
|
|
|
resize: function(event, ui) {
|
|
var that = $(this).resizable( "instance" ),
|
|
o = that.options,
|
|
os = that.originalSize,
|
|
op = that.originalPosition,
|
|
delta = {
|
|
height: (that.size.height - os.height) || 0,
|
|
width: (that.size.width - os.width) || 0,
|
|
top: (that.position.top - op.top) || 0,
|
|
left: (that.position.left - op.left) || 0
|
|
};
|
|
|
|
$(o.alsoResize).each(function() {
|
|
var el = $(this), start = $(this).data("ui-resizable-alsoresize"), style = {},
|
|
css = el.parents(ui.originalElement[0]).length ?
|
|
[ "width", "height" ] :
|
|
[ "width", "height", "top", "left" ];
|
|
|
|
$.each(css, function(i, prop) {
|
|
var sum = (start[prop] || 0) + (delta[prop] || 0);
|
|
if (sum && sum >= 0) {
|
|
style[prop] = sum || null;
|
|
}
|
|
});
|
|
|
|
el.css(style);
|
|
});
|
|
},
|
|
|
|
stop: function() {
|
|
$(this).removeData("resizable-alsoresize");
|
|
}
|
|
});
|
|
|
|
$.ui.plugin.add("resizable", "ghost", {
|
|
|
|
start: function() {
|
|
|
|
var that = $(this).resizable( "instance" ), o = that.options, cs = that.size;
|
|
|
|
that.ghost = that.originalElement.clone();
|
|
that.ghost
|
|
.css({
|
|
opacity: 0.25,
|
|
display: "block",
|
|
position: "relative",
|
|
height: cs.height,
|
|
width: cs.width,
|
|
margin: 0,
|
|
left: 0,
|
|
top: 0
|
|
})
|
|
.addClass("ui-resizable-ghost")
|
|
.addClass(typeof o.ghost === "string" ? o.ghost : "");
|
|
|
|
that.ghost.appendTo(that.helper);
|
|
|
|
},
|
|
|
|
resize: function() {
|
|
var that = $(this).resizable( "instance" );
|
|
if (that.ghost) {
|
|
that.ghost.css({
|
|
position: "relative",
|
|
height: that.size.height,
|
|
width: that.size.width
|
|
});
|
|
}
|
|
},
|
|
|
|
stop: function() {
|
|
var that = $(this).resizable( "instance" );
|
|
if (that.ghost && that.helper) {
|
|
that.helper.get(0).removeChild(that.ghost.get(0));
|
|
}
|
|
}
|
|
|
|
});
|
|
|
|
$.ui.plugin.add("resizable", "grid", {
|
|
|
|
resize: function() {
|
|
var outerDimensions,
|
|
that = $(this).resizable( "instance" ),
|
|
o = that.options,
|
|
cs = that.size,
|
|
os = that.originalSize,
|
|
op = that.originalPosition,
|
|
a = that.axis,
|
|
grid = typeof o.grid === "number" ? [ o.grid, o.grid ] : o.grid,
|
|
gridX = (grid[0] || 1),
|
|
gridY = (grid[1] || 1),
|
|
ox = Math.round((cs.width - os.width) / gridX) * gridX,
|
|
oy = Math.round((cs.height - os.height) / gridY) * gridY,
|
|
newWidth = os.width + ox,
|
|
newHeight = os.height + oy,
|
|
isMaxWidth = o.maxWidth && (o.maxWidth < newWidth),
|
|
isMaxHeight = o.maxHeight && (o.maxHeight < newHeight),
|
|
isMinWidth = o.minWidth && (o.minWidth > newWidth),
|
|
isMinHeight = o.minHeight && (o.minHeight > newHeight);
|
|
|
|
o.grid = grid;
|
|
|
|
if (isMinWidth) {
|
|
newWidth += gridX;
|
|
}
|
|
if (isMinHeight) {
|
|
newHeight += gridY;
|
|
}
|
|
if (isMaxWidth) {
|
|
newWidth -= gridX;
|
|
}
|
|
if (isMaxHeight) {
|
|
newHeight -= gridY;
|
|
}
|
|
|
|
if (/^(se|s|e)$/.test(a)) {
|
|
that.size.width = newWidth;
|
|
that.size.height = newHeight;
|
|
} else if (/^(ne)$/.test(a)) {
|
|
that.size.width = newWidth;
|
|
that.size.height = newHeight;
|
|
that.position.top = op.top - oy;
|
|
} else if (/^(sw)$/.test(a)) {
|
|
that.size.width = newWidth;
|
|
that.size.height = newHeight;
|
|
that.position.left = op.left - ox;
|
|
} else {
|
|
if ( newHeight - gridY <= 0 || newWidth - gridX <= 0) {
|
|
outerDimensions = that._getPaddingPlusBorderDimensions( this );
|
|
}
|
|
|
|
if ( newHeight - gridY > 0 ) {
|
|
that.size.height = newHeight;
|
|
that.position.top = op.top - oy;
|
|
} else {
|
|
newHeight = gridY - outerDimensions.height;
|
|
that.size.height = newHeight;
|
|
that.position.top = op.top + os.height - newHeight;
|
|
}
|
|
if ( newWidth - gridX > 0 ) {
|
|
that.size.width = newWidth;
|
|
that.position.left = op.left - ox;
|
|
} else {
|
|
newWidth = gridX - outerDimensions.width;
|
|
that.size.width = newWidth;
|
|
that.position.left = op.left + os.width - newWidth;
|
|
}
|
|
}
|
|
}
|
|
|
|
});
|
|
|
|
var resizable = $.ui.resizable;
|
|
|
|
|
|
/*!
|
|
* jQuery UI Dialog 1.11.4
|
|
* http://jqueryui.com
|
|
*
|
|
* Copyright jQuery Foundation and other contributors
|
|
* Released under the MIT license.
|
|
* http://jquery.org/license
|
|
*
|
|
* http://api.jqueryui.com/dialog/
|
|
*/
|
|
|
|
|
|
var dialog = $.widget( "ui.dialog", {
|
|
version: "1.11.4",
|
|
options: {
|
|
appendTo: "body",
|
|
autoOpen: true,
|
|
buttons: [],
|
|
closeOnEscape: true,
|
|
closeText: "Close",
|
|
dialogClass: "",
|
|
draggable: true,
|
|
hide: null,
|
|
height: "auto",
|
|
maxHeight: null,
|
|
maxWidth: null,
|
|
minHeight: 150,
|
|
minWidth: 150,
|
|
modal: false,
|
|
position: {
|
|
my: "center",
|
|
at: "center",
|
|
of: window,
|
|
collision: "fit",
|
|
// Ensure the titlebar is always visible
|
|
using: function( pos ) {
|
|
var topOffset = $( this ).css( pos ).offset().top;
|
|
if ( topOffset < 0 ) {
|
|
$( this ).css( "top", pos.top - topOffset );
|
|
}
|
|
}
|
|
},
|
|
resizable: true,
|
|
show: null,
|
|
title: null,
|
|
width: 300,
|
|
|
|
// callbacks
|
|
beforeClose: null,
|
|
close: null,
|
|
drag: null,
|
|
dragStart: null,
|
|
dragStop: null,
|
|
focus: null,
|
|
open: null,
|
|
resize: null,
|
|
resizeStart: null,
|
|
resizeStop: null
|
|
},
|
|
|
|
sizeRelatedOptions: {
|
|
buttons: true,
|
|
height: true,
|
|
maxHeight: true,
|
|
maxWidth: true,
|
|
minHeight: true,
|
|
minWidth: true,
|
|
width: true
|
|
},
|
|
|
|
resizableRelatedOptions: {
|
|
maxHeight: true,
|
|
maxWidth: true,
|
|
minHeight: true,
|
|
minWidth: true
|
|
},
|
|
|
|
_create: function() {
|
|
this.originalCss = {
|
|
display: this.element[ 0 ].style.display,
|
|
width: this.element[ 0 ].style.width,
|
|
minHeight: this.element[ 0 ].style.minHeight,
|
|
maxHeight: this.element[ 0 ].style.maxHeight,
|
|
height: this.element[ 0 ].style.height
|
|
};
|
|
this.originalPosition = {
|
|
parent: this.element.parent(),
|
|
index: this.element.parent().children().index( this.element )
|
|
};
|
|
this.originalTitle = this.element.attr( "title" );
|
|
this.options.title = this.options.title || this.originalTitle;
|
|
|
|
this._createWrapper();
|
|
|
|
this.element
|
|
.show()
|
|
.removeAttr( "title" )
|
|
.addClass( "ui-dialog-content ui-widget-content" )
|
|
.appendTo( this.uiDialog );
|
|
|
|
this._createTitlebar();
|
|
this._createButtonPane();
|
|
|
|
if ( this.options.draggable && $.fn.draggable ) {
|
|
this._makeDraggable();
|
|
}
|
|
if ( this.options.resizable && $.fn.resizable ) {
|
|
this._makeResizable();
|
|
}
|
|
|
|
this._isOpen = false;
|
|
|
|
this._trackFocus();
|
|
},
|
|
|
|
_init: function() {
|
|
if ( this.options.autoOpen ) {
|
|
this.open();
|
|
}
|
|
},
|
|
|
|
_appendTo: function() {
|
|
var element = this.options.appendTo;
|
|
if ( element && (element.jquery || element.nodeType) ) {
|
|
return $( element );
|
|
}
|
|
return this.document.find( element || "body" ).eq( 0 );
|
|
},
|
|
|
|
_destroy: function() {
|
|
var next,
|
|
originalPosition = this.originalPosition;
|
|
|
|
this._untrackInstance();
|
|
this._destroyOverlay();
|
|
|
|
this.element
|
|
.removeUniqueId()
|
|
.removeClass( "ui-dialog-content ui-widget-content" )
|
|
.css( this.originalCss )
|
|
// Without detaching first, the following becomes really slow
|
|
.detach();
|
|
|
|
this.uiDialog.stop( true, true ).remove();
|
|
|
|
if ( this.originalTitle ) {
|
|
this.element.attr( "title", this.originalTitle );
|
|
}
|
|
|
|
next = originalPosition.parent.children().eq( originalPosition.index );
|
|
// Don't try to place the dialog next to itself (#8613)
|
|
if ( next.length && next[ 0 ] !== this.element[ 0 ] ) {
|
|
next.before( this.element );
|
|
} else {
|
|
originalPosition.parent.append( this.element );
|
|
}
|
|
},
|
|
|
|
widget: function() {
|
|
return this.uiDialog;
|
|
},
|
|
|
|
disable: $.noop,
|
|
enable: $.noop,
|
|
|
|
close: function( event ) {
|
|
var activeElement,
|
|
that = this;
|
|
|
|
if ( !this._isOpen || this._trigger( "beforeClose", event ) === false ) {
|
|
return;
|
|
}
|
|
|
|
this._isOpen = false;
|
|
this._focusedElement = null;
|
|
this._destroyOverlay();
|
|
this._untrackInstance();
|
|
|
|
if ( !this.opener.filter( ":focusable" ).focus().length ) {
|
|
|
|
// support: IE9
|
|
// IE9 throws an "Unspecified error" accessing document.activeElement from an <iframe>
|
|
try {
|
|
activeElement = this.document[ 0 ].activeElement;
|
|
|
|
// Support: IE9, IE10
|
|
// If the <body> is blurred, IE will switch windows, see #4520
|
|
if ( activeElement && activeElement.nodeName.toLowerCase() !== "body" ) {
|
|
|
|
// Hiding a focused element doesn't trigger blur in WebKit
|
|
// so in case we have nothing to focus on, explicitly blur the active element
|
|
// https://bugs.webkit.org/show_bug.cgi?id=47182
|
|
$( activeElement ).blur();
|
|
}
|
|
} catch ( error ) {}
|
|
}
|
|
|
|
this._hide( this.uiDialog, this.options.hide, function() {
|
|
that._trigger( "close", event );
|
|
});
|
|
},
|
|
|
|
isOpen: function() {
|
|
return this._isOpen;
|
|
},
|
|
|
|
moveToTop: function() {
|
|
this._moveToTop();
|
|
},
|
|
|
|
_moveToTop: function( event, silent ) {
|
|
var moved = false,
|
|
zIndices = this.uiDialog.siblings( ".ui-front:visible" ).map(function() {
|
|
return +$( this ).css( "z-index" );
|
|
}).get(),
|
|
zIndexMax = Math.max.apply( null, zIndices );
|
|
|
|
if ( zIndexMax >= +this.uiDialog.css( "z-index" ) ) {
|
|
this.uiDialog.css( "z-index", zIndexMax + 1 );
|
|
moved = true;
|
|
}
|
|
|
|
if ( moved && !silent ) {
|
|
this._trigger( "focus", event );
|
|
}
|
|
return moved;
|
|
},
|
|
|
|
open: function() {
|
|
var that = this;
|
|
if ( this._isOpen ) {
|
|
if ( this._moveToTop() ) {
|
|
this._focusTabbable();
|
|
}
|
|
return;
|
|
}
|
|
|
|
this._isOpen = true;
|
|
this.opener = $( this.document[ 0 ].activeElement );
|
|
|
|
this._size();
|
|
this._position();
|
|
this._createOverlay();
|
|
this._moveToTop( null, true );
|
|
|
|
// Ensure the overlay is moved to the top with the dialog, but only when
|
|
// opening. The overlay shouldn't move after the dialog is open so that
|
|
// modeless dialogs opened after the modal dialog stack properly.
|
|
if ( this.overlay ) {
|
|
this.overlay.css( "z-index", this.uiDialog.css( "z-index" ) - 1 );
|
|
}
|
|
|
|
this._show( this.uiDialog, this.options.show, function() {
|
|
that._focusTabbable();
|
|
that._trigger( "focus" );
|
|
});
|
|
|
|
// Track the dialog immediately upon openening in case a focus event
|
|
// somehow occurs outside of the dialog before an element inside the
|
|
// dialog is focused (#10152)
|
|
this._makeFocusTarget();
|
|
|
|
this._trigger( "open" );
|
|
},
|
|
|
|
_focusTabbable: function() {
|
|
// Set focus to the first match:
|
|
// 1. An element that was focused previously
|
|
// 2. First element inside the dialog matching [autofocus]
|
|
// 3. Tabbable element inside the content element
|
|
// 4. Tabbable element inside the buttonpane
|
|
// 5. The close button
|
|
// 6. The dialog itself
|
|
var hasFocus = this._focusedElement;
|
|
if ( !hasFocus ) {
|
|
hasFocus = this.element.find( "[autofocus]" );
|
|
}
|
|
if ( !hasFocus.length ) {
|
|
hasFocus = this.element.find( ":tabbable" );
|
|
}
|
|
if ( !hasFocus.length ) {
|
|
hasFocus = this.uiDialogButtonPane.find( ":tabbable" );
|
|
}
|
|
if ( !hasFocus.length ) {
|
|
hasFocus = this.uiDialogTitlebarClose.filter( ":tabbable" );
|
|
}
|
|
if ( !hasFocus.length ) {
|
|
hasFocus = this.uiDialog;
|
|
}
|
|
hasFocus.eq( 0 ).focus();
|
|
},
|
|
|
|
_keepFocus: function( event ) {
|
|
function checkFocus() {
|
|
var activeElement = this.document[0].activeElement,
|
|
isActive = this.uiDialog[0] === activeElement ||
|
|
$.contains( this.uiDialog[0], activeElement );
|
|
if ( !isActive ) {
|
|
this._focusTabbable();
|
|
}
|
|
}
|
|
event.preventDefault();
|
|
checkFocus.call( this );
|
|
// support: IE
|
|
// IE <= 8 doesn't prevent moving focus even with event.preventDefault()
|
|
// so we check again later
|
|
this._delay( checkFocus );
|
|
},
|
|
|
|
_createWrapper: function() {
|
|
this.uiDialog = $("<div>")
|
|
.addClass( "ui-dialog ui-widget ui-widget-content ui-corner-all ui-front " +
|
|
this.options.dialogClass )
|
|
.hide()
|
|
.attr({
|
|
// Setting tabIndex makes the div focusable
|
|
tabIndex: -1,
|
|
role: "dialog"
|
|
})
|
|
.appendTo( this._appendTo() );
|
|
|
|
this._on( this.uiDialog, {
|
|
keydown: function( event ) {
|
|
if ( this.options.closeOnEscape && !event.isDefaultPrevented() && event.keyCode &&
|
|
event.keyCode === $.ui.keyCode.ESCAPE ) {
|
|
event.preventDefault();
|
|
this.close( event );
|
|
return;
|
|
}
|
|
|
|
// prevent tabbing out of dialogs
|
|
if ( event.keyCode !== $.ui.keyCode.TAB || event.isDefaultPrevented() ) {
|
|
return;
|
|
}
|
|
var tabbables = this.uiDialog.find( ":tabbable" ),
|
|
first = tabbables.filter( ":first" ),
|
|
last = tabbables.filter( ":last" );
|
|
|
|
if ( ( event.target === last[0] || event.target === this.uiDialog[0] ) && !event.shiftKey ) {
|
|
this._delay(function() {
|
|
first.focus();
|
|
});
|
|
event.preventDefault();
|
|
} else if ( ( event.target === first[0] || event.target === this.uiDialog[0] ) && event.shiftKey ) {
|
|
this._delay(function() {
|
|
last.focus();
|
|
});
|
|
event.preventDefault();
|
|
}
|
|
},
|
|
mousedown: function( event ) {
|
|
if ( this._moveToTop( event ) ) {
|
|
this._focusTabbable();
|
|
}
|
|
}
|
|
});
|
|
|
|
// We assume that any existing aria-describedby attribute means
|
|
// that the dialog content is marked up properly
|
|
// otherwise we brute force the content as the description
|
|
if ( !this.element.find( "[aria-describedby]" ).length ) {
|
|
this.uiDialog.attr({
|
|
"aria-describedby": this.element.uniqueId().attr( "id" )
|
|
});
|
|
}
|
|
},
|
|
|
|
_createTitlebar: function() {
|
|
var uiDialogTitle;
|
|
|
|
this.uiDialogTitlebar = $( "<div>" )
|
|
.addClass( "ui-dialog-titlebar ui-widget-header ui-corner-all ui-helper-clearfix" )
|
|
.prependTo( this.uiDialog );
|
|
this._on( this.uiDialogTitlebar, {
|
|
mousedown: function( event ) {
|
|
// Don't prevent click on close button (#8838)
|
|
// Focusing a dialog that is partially scrolled out of view
|
|
// causes the browser to scroll it into view, preventing the click event
|
|
if ( !$( event.target ).closest( ".ui-dialog-titlebar-close" ) ) {
|
|
// Dialog isn't getting focus when dragging (#8063)
|
|
this.uiDialog.focus();
|
|
}
|
|
}
|
|
});
|
|
|
|
// support: IE
|
|
// Use type="button" to prevent enter keypresses in textboxes from closing the
|
|
// dialog in IE (#9312)
|
|
this.uiDialogTitlebarClose = $( "<button type='button'></button>" )
|
|
.button({
|
|
label: this.options.closeText,
|
|
icons: {
|
|
primary: "ui-icon-closethick"
|
|
},
|
|
text: false
|
|
})
|
|
.addClass( "ui-dialog-titlebar-close" )
|
|
.appendTo( this.uiDialogTitlebar );
|
|
this._on( this.uiDialogTitlebarClose, {
|
|
click: function( event ) {
|
|
event.preventDefault();
|
|
this.close( event );
|
|
}
|
|
});
|
|
|
|
uiDialogTitle = $( "<span>" )
|
|
.uniqueId()
|
|
.addClass( "ui-dialog-title" )
|
|
.prependTo( this.uiDialogTitlebar );
|
|
this._title( uiDialogTitle );
|
|
|
|
this.uiDialog.attr({
|
|
"aria-labelledby": uiDialogTitle.attr( "id" )
|
|
});
|
|
},
|
|
|
|
_title: function( title ) {
|
|
if ( !this.options.title ) {
|
|
title.html( " " );
|
|
}
|
|
title.text( this.options.title );
|
|
},
|
|
|
|
_createButtonPane: function() {
|
|
this.uiDialogButtonPane = $( "<div>" )
|
|
.addClass( "ui-dialog-buttonpane ui-widget-content ui-helper-clearfix" );
|
|
|
|
this.uiButtonSet = $( "<div>" )
|
|
.addClass( "ui-dialog-buttonset" )
|
|
.appendTo( this.uiDialogButtonPane );
|
|
|
|
this._createButtons();
|
|
},
|
|
|
|
_createButtons: function() {
|
|
var that = this,
|
|
buttons = this.options.buttons;
|
|
|
|
// if we already have a button pane, remove it
|
|
this.uiDialogButtonPane.remove();
|
|
this.uiButtonSet.empty();
|
|
|
|
if ( $.isEmptyObject( buttons ) || ($.isArray( buttons ) && !buttons.length) ) {
|
|
this.uiDialog.removeClass( "ui-dialog-buttons" );
|
|
return;
|
|
}
|
|
|
|
$.each( buttons, function( name, props ) {
|
|
var click, buttonOptions;
|
|
props = $.isFunction( props ) ?
|
|
{ click: props, text: name } :
|
|
props;
|
|
// Default to a non-submitting button
|
|
props = $.extend( { type: "button" }, props );
|
|
// Change the context for the click callback to be the main element
|
|
click = props.click;
|
|
props.click = function() {
|
|
click.apply( that.element[ 0 ], arguments );
|
|
};
|
|
buttonOptions = {
|
|
icons: props.icons,
|
|
text: props.showText
|
|
};
|
|
delete props.icons;
|
|
delete props.showText;
|
|
$( "<button></button>", props )
|
|
.button( buttonOptions )
|
|
.appendTo( that.uiButtonSet );
|
|
});
|
|
this.uiDialog.addClass( "ui-dialog-buttons" );
|
|
this.uiDialogButtonPane.appendTo( this.uiDialog );
|
|
},
|
|
|
|
_makeDraggable: function() {
|
|
var that = this,
|
|
options = this.options;
|
|
|
|
function filteredUi( ui ) {
|
|
return {
|
|
position: ui.position,
|
|
offset: ui.offset
|
|
};
|
|
}
|
|
|
|
this.uiDialog.draggable({
|
|
cancel: ".ui-dialog-content, .ui-dialog-titlebar-close",
|
|
handle: ".ui-dialog-titlebar",
|
|
containment: "document",
|
|
start: function( event, ui ) {
|
|
$( this ).addClass( "ui-dialog-dragging" );
|
|
that._blockFrames();
|
|
that._trigger( "dragStart", event, filteredUi( ui ) );
|
|
},
|
|
drag: function( event, ui ) {
|
|
that._trigger( "drag", event, filteredUi( ui ) );
|
|
},
|
|
stop: function( event, ui ) {
|
|
var left = ui.offset.left - that.document.scrollLeft(),
|
|
top = ui.offset.top - that.document.scrollTop();
|
|
|
|
options.position = {
|
|
my: "left top",
|
|
at: "left" + (left >= 0 ? "+" : "") + left + " " +
|
|
"top" + (top >= 0 ? "+" : "") + top,
|
|
of: that.window
|
|
};
|
|
$( this ).removeClass( "ui-dialog-dragging" );
|
|
that._unblockFrames();
|
|
that._trigger( "dragStop", event, filteredUi( ui ) );
|
|
}
|
|
});
|
|
},
|
|
|
|
_makeResizable: function() {
|
|
var that = this,
|
|
options = this.options,
|
|
handles = options.resizable,
|
|
// .ui-resizable has position: relative defined in the stylesheet
|
|
// but dialogs have to use absolute or fixed positioning
|
|
position = this.uiDialog.css("position"),
|
|
resizeHandles = typeof handles === "string" ?
|
|
handles :
|
|
"n,e,s,w,se,sw,ne,nw";
|
|
|
|
function filteredUi( ui ) {
|
|
return {
|
|
originalPosition: ui.originalPosition,
|
|
originalSize: ui.originalSize,
|
|
position: ui.position,
|
|
size: ui.size
|
|
};
|
|
}
|
|
|
|
this.uiDialog.resizable({
|
|
cancel: ".ui-dialog-content",
|
|
containment: "document",
|
|
alsoResize: this.element,
|
|
maxWidth: options.maxWidth,
|
|
maxHeight: options.maxHeight,
|
|
minWidth: options.minWidth,
|
|
minHeight: this._minHeight(),
|
|
handles: resizeHandles,
|
|
start: function( event, ui ) {
|
|
$( this ).addClass( "ui-dialog-resizing" );
|
|
that._blockFrames();
|
|
that._trigger( "resizeStart", event, filteredUi( ui ) );
|
|
},
|
|
resize: function( event, ui ) {
|
|
that._trigger( "resize", event, filteredUi( ui ) );
|
|
},
|
|
stop: function( event, ui ) {
|
|
var offset = that.uiDialog.offset(),
|
|
left = offset.left - that.document.scrollLeft(),
|
|
top = offset.top - that.document.scrollTop();
|
|
|
|
options.height = that.uiDialog.height();
|
|
options.width = that.uiDialog.width();
|
|
options.position = {
|
|
my: "left top",
|
|
at: "left" + (left >= 0 ? "+" : "") + left + " " +
|
|
"top" + (top >= 0 ? "+" : "") + top,
|
|
of: that.window
|
|
};
|
|
$( this ).removeClass( "ui-dialog-resizing" );
|
|
that._unblockFrames();
|
|
that._trigger( "resizeStop", event, filteredUi( ui ) );
|
|
}
|
|
})
|
|
.css( "position", position );
|
|
},
|
|
|
|
_trackFocus: function() {
|
|
this._on( this.widget(), {
|
|
focusin: function( event ) {
|
|
this._makeFocusTarget();
|
|
this._focusedElement = $( event.target );
|
|
}
|
|
});
|
|
},
|
|
|
|
_makeFocusTarget: function() {
|
|
this._untrackInstance();
|
|
this._trackingInstances().unshift( this );
|
|
},
|
|
|
|
_untrackInstance: function() {
|
|
var instances = this._trackingInstances(),
|
|
exists = $.inArray( this, instances );
|
|
if ( exists !== -1 ) {
|
|
instances.splice( exists, 1 );
|
|
}
|
|
},
|
|
|
|
_trackingInstances: function() {
|
|
var instances = this.document.data( "ui-dialog-instances" );
|
|
if ( !instances ) {
|
|
instances = [];
|
|
this.document.data( "ui-dialog-instances", instances );
|
|
}
|
|
return instances;
|
|
},
|
|
|
|
_minHeight: function() {
|
|
var options = this.options;
|
|
|
|
return options.height === "auto" ?
|
|
options.minHeight :
|
|
Math.min( options.minHeight, options.height );
|
|
},
|
|
|
|
_position: function() {
|
|
// Need to show the dialog to get the actual offset in the position plugin
|
|
var isVisible = this.uiDialog.is( ":visible" );
|
|
if ( !isVisible ) {
|
|
this.uiDialog.show();
|
|
}
|
|
this.uiDialog.position( this.options.position );
|
|
if ( !isVisible ) {
|
|
this.uiDialog.hide();
|
|
}
|
|
},
|
|
|
|
_setOptions: function( options ) {
|
|
var that = this,
|
|
resize = false,
|
|
resizableOptions = {};
|
|
|
|
$.each( options, function( key, value ) {
|
|
that._setOption( key, value );
|
|
|
|
if ( key in that.sizeRelatedOptions ) {
|
|
resize = true;
|
|
}
|
|
if ( key in that.resizableRelatedOptions ) {
|
|
resizableOptions[ key ] = value;
|
|
}
|
|
});
|
|
|
|
if ( resize ) {
|
|
this._size();
|
|
this._position();
|
|
}
|
|
if ( this.uiDialog.is( ":data(ui-resizable)" ) ) {
|
|
this.uiDialog.resizable( "option", resizableOptions );
|
|
}
|
|
},
|
|
|
|
_setOption: function( key, value ) {
|
|
var isDraggable, isResizable,
|
|
uiDialog = this.uiDialog;
|
|
|
|
if ( key === "dialogClass" ) {
|
|
uiDialog
|
|
.removeClass( this.options.dialogClass )
|
|
.addClass( value );
|
|
}
|
|
|
|
if ( key === "disabled" ) {
|
|
return;
|
|
}
|
|
|
|
this._super( key, value );
|
|
|
|
if ( key === "appendTo" ) {
|
|
this.uiDialog.appendTo( this._appendTo() );
|
|
}
|
|
|
|
if ( key === "buttons" ) {
|
|
this._createButtons();
|
|
}
|
|
|
|
if ( key === "closeText" ) {
|
|
this.uiDialogTitlebarClose.button({
|
|
// Ensure that we always pass a string
|
|
label: "" + value
|
|
});
|
|
}
|
|
|
|
if ( key === "draggable" ) {
|
|
isDraggable = uiDialog.is( ":data(ui-draggable)" );
|
|
if ( isDraggable && !value ) {
|
|
uiDialog.draggable( "destroy" );
|
|
}
|
|
|
|
if ( !isDraggable && value ) {
|
|
this._makeDraggable();
|
|
}
|
|
}
|
|
|
|
if ( key === "position" ) {
|
|
this._position();
|
|
}
|
|
|
|
if ( key === "resizable" ) {
|
|
// currently resizable, becoming non-resizable
|
|
isResizable = uiDialog.is( ":data(ui-resizable)" );
|
|
if ( isResizable && !value ) {
|
|
uiDialog.resizable( "destroy" );
|
|
}
|
|
|
|
// currently resizable, changing handles
|
|
if ( isResizable && typeof value === "string" ) {
|
|
uiDialog.resizable( "option", "handles", value );
|
|
}
|
|
|
|
// currently non-resizable, becoming resizable
|
|
if ( !isResizable && value !== false ) {
|
|
this._makeResizable();
|
|
}
|
|
}
|
|
|
|
if ( key === "title" ) {
|
|
this._title( this.uiDialogTitlebar.find( ".ui-dialog-title" ) );
|
|
}
|
|
},
|
|
|
|
_size: function() {
|
|
// If the user has resized the dialog, the .ui-dialog and .ui-dialog-content
|
|
// divs will both have width and height set, so we need to reset them
|
|
var nonContentHeight, minContentHeight, maxContentHeight,
|
|
options = this.options;
|
|
|
|
// Reset content sizing
|
|
this.element.show().css({
|
|
width: "auto",
|
|
minHeight: 0,
|
|
maxHeight: "none",
|
|
height: 0
|
|
});
|
|
|
|
if ( options.minWidth > options.width ) {
|
|
options.width = options.minWidth;
|
|
}
|
|
|
|
// reset wrapper sizing
|
|
// determine the height of all the non-content elements
|
|
nonContentHeight = this.uiDialog.css({
|
|
height: "auto",
|
|
width: options.width
|
|
})
|
|
.outerHeight();
|
|
minContentHeight = Math.max( 0, options.minHeight - nonContentHeight );
|
|
maxContentHeight = typeof options.maxHeight === "number" ?
|
|
Math.max( 0, options.maxHeight - nonContentHeight ) :
|
|
"none";
|
|
|
|
if ( options.height === "auto" ) {
|
|
this.element.css({
|
|
minHeight: minContentHeight,
|
|
maxHeight: maxContentHeight,
|
|
height: "auto"
|
|
});
|
|
} else {
|
|
this.element.height( Math.max( 0, options.height - nonContentHeight ) );
|
|
}
|
|
|
|
if ( this.uiDialog.is( ":data(ui-resizable)" ) ) {
|
|
this.uiDialog.resizable( "option", "minHeight", this._minHeight() );
|
|
}
|
|
},
|
|
|
|
_blockFrames: function() {
|
|
this.iframeBlocks = this.document.find( "iframe" ).map(function() {
|
|
var iframe = $( this );
|
|
|
|
return $( "<div>" )
|
|
.css({
|
|
position: "absolute",
|
|
width: iframe.outerWidth(),
|
|
height: iframe.outerHeight()
|
|
})
|
|
.appendTo( iframe.parent() )
|
|
.offset( iframe.offset() )[0];
|
|
});
|
|
},
|
|
|
|
_unblockFrames: function() {
|
|
if ( this.iframeBlocks ) {
|
|
this.iframeBlocks.remove();
|
|
delete this.iframeBlocks;
|
|
}
|
|
},
|
|
|
|
_allowInteraction: function( event ) {
|
|
if ( $( event.target ).closest( ".ui-dialog" ).length ) {
|
|
return true;
|
|
}
|
|
|
|
// TODO: Remove hack when datepicker implements
|
|
// the .ui-front logic (#8989)
|
|
return !!$( event.target ).closest( ".ui-datepicker" ).length;
|
|
},
|
|
|
|
_createOverlay: function() {
|
|
if ( !this.options.modal ) {
|
|
return;
|
|
}
|
|
|
|
// We use a delay in case the overlay is created from an
|
|
// event that we're going to be cancelling (#2804)
|
|
var isOpening = true;
|
|
this._delay(function() {
|
|
isOpening = false;
|
|
});
|
|
|
|
if ( !this.document.data( "ui-dialog-overlays" ) ) {
|
|
|
|
// Prevent use of anchors and inputs
|
|
// Using _on() for an event handler shared across many instances is
|
|
// safe because the dialogs stack and must be closed in reverse order
|
|
this._on( this.document, {
|
|
focusin: function( event ) {
|
|
if ( isOpening ) {
|
|
return;
|
|
}
|
|
|
|
if ( !this._allowInteraction( event ) ) {
|
|
event.preventDefault();
|
|
this._trackingInstances()[ 0 ]._focusTabbable();
|
|
}
|
|
}
|
|
});
|
|
}
|
|
|
|
this.overlay = $( "<div>" )
|
|
.addClass( "ui-widget-overlay ui-front" )
|
|
.appendTo( this._appendTo() );
|
|
this._on( this.overlay, {
|
|
mousedown: "_keepFocus"
|
|
});
|
|
this.document.data( "ui-dialog-overlays",
|
|
(this.document.data( "ui-dialog-overlays" ) || 0) + 1 );
|
|
},
|
|
|
|
_destroyOverlay: function() {
|
|
if ( !this.options.modal ) {
|
|
return;
|
|
}
|
|
|
|
if ( this.overlay ) {
|
|
var overlays = this.document.data( "ui-dialog-overlays" ) - 1;
|
|
|
|
if ( !overlays ) {
|
|
this.document
|
|
.unbind( "focusin" )
|
|
.removeData( "ui-dialog-overlays" );
|
|
} else {
|
|
this.document.data( "ui-dialog-overlays", overlays );
|
|
}
|
|
|
|
this.overlay.remove();
|
|
this.overlay = null;
|
|
}
|
|
}
|
|
});
|
|
|
|
|
|
/*!
|
|
* jQuery UI Droppable 1.11.4
|
|
* http://jqueryui.com
|
|
*
|
|
* Copyright jQuery Foundation and other contributors
|
|
* Released under the MIT license.
|
|
* http://jquery.org/license
|
|
*
|
|
* http://api.jqueryui.com/droppable/
|
|
*/
|
|
|
|
|
|
$.widget( "ui.droppable", {
|
|
version: "1.11.4",
|
|
widgetEventPrefix: "drop",
|
|
options: {
|
|
accept: "*",
|
|
activeClass: false,
|
|
addClasses: true,
|
|
greedy: false,
|
|
hoverClass: false,
|
|
scope: "default",
|
|
tolerance: "intersect",
|
|
|
|
// callbacks
|
|
activate: null,
|
|
deactivate: null,
|
|
drop: null,
|
|
out: null,
|
|
over: null
|
|
},
|
|
_create: function() {
|
|
|
|
var proportions,
|
|
o = this.options,
|
|
accept = o.accept;
|
|
|
|
this.isover = false;
|
|
this.isout = true;
|
|
|
|
this.accept = $.isFunction( accept ) ? accept : function( d ) {
|
|
return d.is( accept );
|
|
};
|
|
|
|
this.proportions = function( /* valueToWrite */ ) {
|
|
if ( arguments.length ) {
|
|
// Store the droppable's proportions
|
|
proportions = arguments[ 0 ];
|
|
} else {
|
|
// Retrieve or derive the droppable's proportions
|
|
return proportions ?
|
|
proportions :
|
|
proportions = {
|
|
width: this.element[ 0 ].offsetWidth,
|
|
height: this.element[ 0 ].offsetHeight
|
|
};
|
|
}
|
|
};
|
|
|
|
this._addToManager( o.scope );
|
|
|
|
o.addClasses && this.element.addClass( "ui-droppable" );
|
|
|
|
},
|
|
|
|
_addToManager: function( scope ) {
|
|
// Add the reference and positions to the manager
|
|
$.ui.ddmanager.droppables[ scope ] = $.ui.ddmanager.droppables[ scope ] || [];
|
|
$.ui.ddmanager.droppables[ scope ].push( this );
|
|
},
|
|
|
|
_splice: function( drop ) {
|
|
var i = 0;
|
|
for ( ; i < drop.length; i++ ) {
|
|
if ( drop[ i ] === this ) {
|
|
drop.splice( i, 1 );
|
|
}
|
|
}
|
|
},
|
|
|
|
_destroy: function() {
|
|
var drop = $.ui.ddmanager.droppables[ this.options.scope ];
|
|
|
|
this._splice( drop );
|
|
|
|
this.element.removeClass( "ui-droppable ui-droppable-disabled" );
|
|
},
|
|
|
|
_setOption: function( key, value ) {
|
|
|
|
if ( key === "accept" ) {
|
|
this.accept = $.isFunction( value ) ? value : function( d ) {
|
|
return d.is( value );
|
|
};
|
|
} else if ( key === "scope" ) {
|
|
var drop = $.ui.ddmanager.droppables[ this.options.scope ];
|
|
|
|
this._splice( drop );
|
|
this._addToManager( value );
|
|
}
|
|
|
|
this._super( key, value );
|
|
},
|
|
|
|
_activate: function( event ) {
|
|
var draggable = $.ui.ddmanager.current;
|
|
if ( this.options.activeClass ) {
|
|
this.element.addClass( this.options.activeClass );
|
|
}
|
|
if ( draggable ){
|
|
this._trigger( "activate", event, this.ui( draggable ) );
|
|
}
|
|
},
|
|
|
|
_deactivate: function( event ) {
|
|
var draggable = $.ui.ddmanager.current;
|
|
if ( this.options.activeClass ) {
|
|
this.element.removeClass( this.options.activeClass );
|
|
}
|
|
if ( draggable ){
|
|
this._trigger( "deactivate", event, this.ui( draggable ) );
|
|
}
|
|
},
|
|
|
|
_over: function( event ) {
|
|
|
|
var draggable = $.ui.ddmanager.current;
|
|
|
|
// Bail if draggable and droppable are same element
|
|
if ( !draggable || ( draggable.currentItem || draggable.element )[ 0 ] === this.element[ 0 ] ) {
|
|
return;
|
|
}
|
|
|
|
if ( this.accept.call( this.element[ 0 ], ( draggable.currentItem || draggable.element ) ) ) {
|
|
if ( this.options.hoverClass ) {
|
|
this.element.addClass( this.options.hoverClass );
|
|
}
|
|
this._trigger( "over", event, this.ui( draggable ) );
|
|
}
|
|
|
|
},
|
|
|
|
_out: function( event ) {
|
|
|
|
var draggable = $.ui.ddmanager.current;
|
|
|
|
// Bail if draggable and droppable are same element
|
|
if ( !draggable || ( draggable.currentItem || draggable.element )[ 0 ] === this.element[ 0 ] ) {
|
|
return;
|
|
}
|
|
|
|
if ( this.accept.call( this.element[ 0 ], ( draggable.currentItem || draggable.element ) ) ) {
|
|
if ( this.options.hoverClass ) {
|
|
this.element.removeClass( this.options.hoverClass );
|
|
}
|
|
this._trigger( "out", event, this.ui( draggable ) );
|
|
}
|
|
|
|
},
|
|
|
|
_drop: function( event, custom ) {
|
|
|
|
var draggable = custom || $.ui.ddmanager.current,
|
|
childrenIntersection = false;
|
|
|
|
// Bail if draggable and droppable are same element
|
|
if ( !draggable || ( draggable.currentItem || draggable.element )[ 0 ] === this.element[ 0 ] ) {
|
|
return false;
|
|
}
|
|
|
|
this.element.find( ":data(ui-droppable)" ).not( ".ui-draggable-dragging" ).each(function() {
|
|
var inst = $( this ).droppable( "instance" );
|
|
if (
|
|
inst.options.greedy &&
|
|
!inst.options.disabled &&
|
|
inst.options.scope === draggable.options.scope &&
|
|
inst.accept.call( inst.element[ 0 ], ( draggable.currentItem || draggable.element ) ) &&
|
|
$.ui.intersect( draggable, $.extend( inst, { offset: inst.element.offset() } ), inst.options.tolerance, event )
|
|
) { childrenIntersection = true; return false; }
|
|
});
|
|
if ( childrenIntersection ) {
|
|
return false;
|
|
}
|
|
|
|
if ( this.accept.call( this.element[ 0 ], ( draggable.currentItem || draggable.element ) ) ) {
|
|
if ( this.options.activeClass ) {
|
|
this.element.removeClass( this.options.activeClass );
|
|
}
|
|
if ( this.options.hoverClass ) {
|
|
this.element.removeClass( this.options.hoverClass );
|
|
}
|
|
this._trigger( "drop", event, this.ui( draggable ) );
|
|
return this.element;
|
|
}
|
|
|
|
return false;
|
|
|
|
},
|
|
|
|
ui: function( c ) {
|
|
return {
|
|
draggable: ( c.currentItem || c.element ),
|
|
helper: c.helper,
|
|
position: c.position,
|
|
offset: c.positionAbs
|
|
};
|
|
}
|
|
|
|
});
|
|
|
|
$.ui.intersect = (function() {
|
|
function isOverAxis( x, reference, size ) {
|
|
return ( x >= reference ) && ( x < ( reference + size ) );
|
|
}
|
|
|
|
return function( draggable, droppable, toleranceMode, event ) {
|
|
|
|
if ( !droppable.offset ) {
|
|
return false;
|
|
}
|
|
|
|
var x1 = ( draggable.positionAbs || draggable.position.absolute ).left + draggable.margins.left,
|
|
y1 = ( draggable.positionAbs || draggable.position.absolute ).top + draggable.margins.top,
|
|
x2 = x1 + draggable.helperProportions.width,
|
|
y2 = y1 + draggable.helperProportions.height,
|
|
l = droppable.offset.left,
|
|
t = droppable.offset.top,
|
|
r = l + droppable.proportions().width,
|
|
b = t + droppable.proportions().height;
|
|
|
|
switch ( toleranceMode ) {
|
|
case "fit":
|
|
return ( l <= x1 && x2 <= r && t <= y1 && y2 <= b );
|
|
case "intersect":
|
|
return ( l < x1 + ( draggable.helperProportions.width / 2 ) && // Right Half
|
|
x2 - ( draggable.helperProportions.width / 2 ) < r && // Left Half
|
|
t < y1 + ( draggable.helperProportions.height / 2 ) && // Bottom Half
|
|
y2 - ( draggable.helperProportions.height / 2 ) < b ); // Top Half
|
|
case "pointer":
|
|
return isOverAxis( event.pageY, t, droppable.proportions().height ) && isOverAxis( event.pageX, l, droppable.proportions().width );
|
|
case "touch":
|
|
return (
|
|
( y1 >= t && y1 <= b ) || // Top edge touching
|
|
( y2 >= t && y2 <= b ) || // Bottom edge touching
|
|
( y1 < t && y2 > b ) // Surrounded vertically
|
|
) && (
|
|
( x1 >= l && x1 <= r ) || // Left edge touching
|
|
( x2 >= l && x2 <= r ) || // Right edge touching
|
|
( x1 < l && x2 > r ) // Surrounded horizontally
|
|
);
|
|
default:
|
|
return false;
|
|
}
|
|
};
|
|
})();
|
|
|
|
/*
|
|
This manager tracks offsets of draggables and droppables
|
|
*/
|
|
$.ui.ddmanager = {
|
|
current: null,
|
|
droppables: { "default": [] },
|
|
prepareOffsets: function( t, event ) {
|
|
|
|
var i, j,
|
|
m = $.ui.ddmanager.droppables[ t.options.scope ] || [],
|
|
type = event ? event.type : null, // workaround for #2317
|
|
list = ( t.currentItem || t.element ).find( ":data(ui-droppable)" ).addBack();
|
|
|
|
droppablesLoop: for ( i = 0; i < m.length; i++ ) {
|
|
|
|
// No disabled and non-accepted
|
|
if ( m[ i ].options.disabled || ( t && !m[ i ].accept.call( m[ i ].element[ 0 ], ( t.currentItem || t.element ) ) ) ) {
|
|
continue;
|
|
}
|
|
|
|
// Filter out elements in the current dragged item
|
|
for ( j = 0; j < list.length; j++ ) {
|
|
if ( list[ j ] === m[ i ].element[ 0 ] ) {
|
|
m[ i ].proportions().height = 0;
|
|
continue droppablesLoop;
|
|
}
|
|
}
|
|
|
|
m[ i ].visible = m[ i ].element.css( "display" ) !== "none";
|
|
if ( !m[ i ].visible ) {
|
|
continue;
|
|
}
|
|
|
|
// Activate the droppable if used directly from draggables
|
|
if ( type === "mousedown" ) {
|
|
m[ i ]._activate.call( m[ i ], event );
|
|
}
|
|
|
|
m[ i ].offset = m[ i ].element.offset();
|
|
m[ i ].proportions({ width: m[ i ].element[ 0 ].offsetWidth, height: m[ i ].element[ 0 ].offsetHeight });
|
|
|
|
}
|
|
|
|
},
|
|
drop: function( draggable, event ) {
|
|
|
|
var dropped = false;
|
|
// Create a copy of the droppables in case the list changes during the drop (#9116)
|
|
$.each( ( $.ui.ddmanager.droppables[ draggable.options.scope ] || [] ).slice(), function() {
|
|
|
|
if ( !this.options ) {
|
|
return;
|
|
}
|
|
if ( !this.options.disabled && this.visible && $.ui.intersect( draggable, this, this.options.tolerance, event ) ) {
|
|
dropped = this._drop.call( this, event ) || dropped;
|
|
}
|
|
|
|
if ( !this.options.disabled && this.visible && this.accept.call( this.element[ 0 ], ( draggable.currentItem || draggable.element ) ) ) {
|
|
this.isout = true;
|
|
this.isover = false;
|
|
this._deactivate.call( this, event );
|
|
}
|
|
|
|
});
|
|
return dropped;
|
|
|
|
},
|
|
dragStart: function( draggable, event ) {
|
|
// Listen for scrolling so that if the dragging causes scrolling the position of the droppables can be recalculated (see #5003)
|
|
draggable.element.parentsUntil( "body" ).bind( "scroll.droppable", function() {
|
|
if ( !draggable.options.refreshPositions ) {
|
|
$.ui.ddmanager.prepareOffsets( draggable, event );
|
|
}
|
|
});
|
|
},
|
|
drag: function( draggable, event ) {
|
|
|
|
// If you have a highly dynamic page, you might try this option. It renders positions every time you move the mouse.
|
|
if ( draggable.options.refreshPositions ) {
|
|
$.ui.ddmanager.prepareOffsets( draggable, event );
|
|
}
|
|
|
|
// Run through all droppables and check their positions based on specific tolerance options
|
|
$.each( $.ui.ddmanager.droppables[ draggable.options.scope ] || [], function() {
|
|
|
|
if ( this.options.disabled || this.greedyChild || !this.visible ) {
|
|
return;
|
|
}
|
|
|
|
var parentInstance, scope, parent,
|
|
intersects = $.ui.intersect( draggable, this, this.options.tolerance, event ),
|
|
c = !intersects && this.isover ? "isout" : ( intersects && !this.isover ? "isover" : null );
|
|
if ( !c ) {
|
|
return;
|
|
}
|
|
|
|
if ( this.options.greedy ) {
|
|
// find droppable parents with same scope
|
|
scope = this.options.scope;
|
|
parent = this.element.parents( ":data(ui-droppable)" ).filter(function() {
|
|
return $( this ).droppable( "instance" ).options.scope === scope;
|
|
});
|
|
|
|
if ( parent.length ) {
|
|
parentInstance = $( parent[ 0 ] ).droppable( "instance" );
|
|
parentInstance.greedyChild = ( c === "isover" );
|
|
}
|
|
}
|
|
|
|
// we just moved into a greedy child
|
|
if ( parentInstance && c === "isover" ) {
|
|
parentInstance.isover = false;
|
|
parentInstance.isout = true;
|
|
parentInstance._out.call( parentInstance, event );
|
|
}
|
|
|
|
this[ c ] = true;
|
|
this[c === "isout" ? "isover" : "isout"] = false;
|
|
this[c === "isover" ? "_over" : "_out"].call( this, event );
|
|
|
|
// we just moved out of a greedy child
|
|
if ( parentInstance && c === "isout" ) {
|
|
parentInstance.isout = false;
|
|
parentInstance.isover = true;
|
|
parentInstance._over.call( parentInstance, event );
|
|
}
|
|
});
|
|
|
|
},
|
|
dragStop: function( draggable, event ) {
|
|
draggable.element.parentsUntil( "body" ).unbind( "scroll.droppable" );
|
|
// Call prepareOffsets one final time since IE does not fire return scroll events when overflow was caused by drag (see #5003)
|
|
if ( !draggable.options.refreshPositions ) {
|
|
$.ui.ddmanager.prepareOffsets( draggable, event );
|
|
}
|
|
}
|
|
};
|
|
|
|
var droppable = $.ui.droppable;
|
|
|
|
|
|
/*!
|
|
* jQuery UI Effects 1.11.4
|
|
* http://jqueryui.com
|
|
*
|
|
* Copyright jQuery Foundation and other contributors
|
|
* Released under the MIT license.
|
|
* http://jquery.org/license
|
|
*
|
|
* http://api.jqueryui.com/category/effects-core/
|
|
*/
|
|
|
|
|
|
var dataSpace = "ui-effects-",
|
|
|
|
// Create a local jQuery because jQuery Color relies on it and the
|
|
// global may not exist with AMD and a custom build (#10199)
|
|
jQuery = $;
|
|
|
|
$.effects = {
|
|
effect: {}
|
|
};
|
|
|
|
/*!
|
|
* jQuery Color Animations v2.1.2
|
|
* https://github.com/jquery/jquery-color
|
|
*
|
|
* Copyright 2014 jQuery Foundation and other contributors
|
|
* Released under the MIT license.
|
|
* http://jquery.org/license
|
|
*
|
|
* Date: Wed Jan 16 08:47:09 2013 -0600
|
|
*/
|
|
(function( jQuery, undefined ) {
|
|
|
|
var stepHooks = "backgroundColor borderBottomColor borderLeftColor borderRightColor borderTopColor color columnRuleColor outlineColor textDecorationColor textEmphasisColor",
|
|
|
|
// plusequals test for += 100 -= 100
|
|
rplusequals = /^([\-+])=\s*(\d+\.?\d*)/,
|
|
// a set of RE's that can match strings and generate color tuples.
|
|
stringParsers = [ {
|
|
re: /rgba?\(\s*(\d{1,3})\s*,\s*(\d{1,3})\s*,\s*(\d{1,3})\s*(?:,\s*(\d?(?:\.\d+)?)\s*)?\)/,
|
|
parse: function( execResult ) {
|
|
return [
|
|
execResult[ 1 ],
|
|
execResult[ 2 ],
|
|
execResult[ 3 ],
|
|
execResult[ 4 ]
|
|
];
|
|
}
|
|
}, {
|
|
re: /rgba?\(\s*(\d+(?:\.\d+)?)\%\s*,\s*(\d+(?:\.\d+)?)\%\s*,\s*(\d+(?:\.\d+)?)\%\s*(?:,\s*(\d?(?:\.\d+)?)\s*)?\)/,
|
|
parse: function( execResult ) {
|
|
return [
|
|
execResult[ 1 ] * 2.55,
|
|
execResult[ 2 ] * 2.55,
|
|
execResult[ 3 ] * 2.55,
|
|
execResult[ 4 ]
|
|
];
|
|
}
|
|
}, {
|
|
// this regex ignores A-F because it's compared against an already lowercased string
|
|
re: /#([a-f0-9]{2})([a-f0-9]{2})([a-f0-9]{2})/,
|
|
parse: function( execResult ) {
|
|
return [
|
|
parseInt( execResult[ 1 ], 16 ),
|
|
parseInt( execResult[ 2 ], 16 ),
|
|
parseInt( execResult[ 3 ], 16 )
|
|
];
|
|
}
|
|
}, {
|
|
// this regex ignores A-F because it's compared against an already lowercased string
|
|
re: /#([a-f0-9])([a-f0-9])([a-f0-9])/,
|
|
parse: function( execResult ) {
|
|
return [
|
|
parseInt( execResult[ 1 ] + execResult[ 1 ], 16 ),
|
|
parseInt( execResult[ 2 ] + execResult[ 2 ], 16 ),
|
|
parseInt( execResult[ 3 ] + execResult[ 3 ], 16 )
|
|
];
|
|
}
|
|
}, {
|
|
re: /hsla?\(\s*(\d+(?:\.\d+)?)\s*,\s*(\d+(?:\.\d+)?)\%\s*,\s*(\d+(?:\.\d+)?)\%\s*(?:,\s*(\d?(?:\.\d+)?)\s*)?\)/,
|
|
space: "hsla",
|
|
parse: function( execResult ) {
|
|
return [
|
|
execResult[ 1 ],
|
|
execResult[ 2 ] / 100,
|
|
execResult[ 3 ] / 100,
|
|
execResult[ 4 ]
|
|
];
|
|
}
|
|
} ],
|
|
|
|
// jQuery.Color( )
|
|
color = jQuery.Color = function( color, green, blue, alpha ) {
|
|
return new jQuery.Color.fn.parse( color, green, blue, alpha );
|
|
},
|
|
spaces = {
|
|
rgba: {
|
|
props: {
|
|
red: {
|
|
idx: 0,
|
|
type: "byte"
|
|
},
|
|
green: {
|
|
idx: 1,
|
|
type: "byte"
|
|
},
|
|
blue: {
|
|
idx: 2,
|
|
type: "byte"
|
|
}
|
|
}
|
|
},
|
|
|
|
hsla: {
|
|
props: {
|
|
hue: {
|
|
idx: 0,
|
|
type: "degrees"
|
|
},
|
|
saturation: {
|
|
idx: 1,
|
|
type: "percent"
|
|
},
|
|
lightness: {
|
|
idx: 2,
|
|
type: "percent"
|
|
}
|
|
}
|
|
}
|
|
},
|
|
propTypes = {
|
|
"byte": {
|
|
floor: true,
|
|
max: 255
|
|
},
|
|
"percent": {
|
|
max: 1
|
|
},
|
|
"degrees": {
|
|
mod: 360,
|
|
floor: true
|
|
}
|
|
},
|
|
support = color.support = {},
|
|
|
|
// element for support tests
|
|
supportElem = jQuery( "<p>" )[ 0 ],
|
|
|
|
// colors = jQuery.Color.names
|
|
colors,
|
|
|
|
// local aliases of functions called often
|
|
each = jQuery.each;
|
|
|
|
// determine rgba support immediately
|
|
supportElem.style.cssText = "background-color:rgba(1,1,1,.5)";
|
|
support.rgba = supportElem.style.backgroundColor.indexOf( "rgba" ) > -1;
|
|
|
|
// define cache name and alpha properties
|
|
// for rgba and hsla spaces
|
|
each( spaces, function( spaceName, space ) {
|
|
space.cache = "_" + spaceName;
|
|
space.props.alpha = {
|
|
idx: 3,
|
|
type: "percent",
|
|
def: 1
|
|
};
|
|
});
|
|
|
|
function clamp( value, prop, allowEmpty ) {
|
|
var type = propTypes[ prop.type ] || {};
|
|
|
|
if ( value == null ) {
|
|
return (allowEmpty || !prop.def) ? null : prop.def;
|
|
}
|
|
|
|
// ~~ is an short way of doing floor for positive numbers
|
|
value = type.floor ? ~~value : parseFloat( value );
|
|
|
|
// IE will pass in empty strings as value for alpha,
|
|
// which will hit this case
|
|
if ( isNaN( value ) ) {
|
|
return prop.def;
|
|
}
|
|
|
|
if ( type.mod ) {
|
|
// we add mod before modding to make sure that negatives values
|
|
// get converted properly: -10 -> 350
|
|
return (value + type.mod) % type.mod;
|
|
}
|
|
|
|
// for now all property types without mod have min and max
|
|
return 0 > value ? 0 : type.max < value ? type.max : value;
|
|
}
|
|
|
|
function stringParse( string ) {
|
|
var inst = color(),
|
|
rgba = inst._rgba = [];
|
|
|
|
string = string.toLowerCase();
|
|
|
|
each( stringParsers, function( i, parser ) {
|
|
var parsed,
|
|
match = parser.re.exec( string ),
|
|
values = match && parser.parse( match ),
|
|
spaceName = parser.space || "rgba";
|
|
|
|
if ( values ) {
|
|
parsed = inst[ spaceName ]( values );
|
|
|
|
// if this was an rgba parse the assignment might happen twice
|
|
// oh well....
|
|
inst[ spaces[ spaceName ].cache ] = parsed[ spaces[ spaceName ].cache ];
|
|
rgba = inst._rgba = parsed._rgba;
|
|
|
|
// exit each( stringParsers ) here because we matched
|
|
return false;
|
|
}
|
|
});
|
|
|
|
// Found a stringParser that handled it
|
|
if ( rgba.length ) {
|
|
|
|
// if this came from a parsed string, force "transparent" when alpha is 0
|
|
// chrome, (and maybe others) return "transparent" as rgba(0,0,0,0)
|
|
if ( rgba.join() === "0,0,0,0" ) {
|
|
jQuery.extend( rgba, colors.transparent );
|
|
}
|
|
return inst;
|
|
}
|
|
|
|
// named colors
|
|
return colors[ string ];
|
|
}
|
|
|
|
color.fn = jQuery.extend( color.prototype, {
|
|
parse: function( red, green, blue, alpha ) {
|
|
if ( red === undefined ) {
|
|
this._rgba = [ null, null, null, null ];
|
|
return this;
|
|
}
|
|
if ( red.jquery || red.nodeType ) {
|
|
red = jQuery( red ).css( green );
|
|
green = undefined;
|
|
}
|
|
|
|
var inst = this,
|
|
type = jQuery.type( red ),
|
|
rgba = this._rgba = [];
|
|
|
|
// more than 1 argument specified - assume ( red, green, blue, alpha )
|
|
if ( green !== undefined ) {
|
|
red = [ red, green, blue, alpha ];
|
|
type = "array";
|
|
}
|
|
|
|
if ( type === "string" ) {
|
|
return this.parse( stringParse( red ) || colors._default );
|
|
}
|
|
|
|
if ( type === "array" ) {
|
|
each( spaces.rgba.props, function( key, prop ) {
|
|
rgba[ prop.idx ] = clamp( red[ prop.idx ], prop );
|
|
});
|
|
return this;
|
|
}
|
|
|
|
if ( type === "object" ) {
|
|
if ( red instanceof color ) {
|
|
each( spaces, function( spaceName, space ) {
|
|
if ( red[ space.cache ] ) {
|
|
inst[ space.cache ] = red[ space.cache ].slice();
|
|
}
|
|
});
|
|
} else {
|
|
each( spaces, function( spaceName, space ) {
|
|
var cache = space.cache;
|
|
each( space.props, function( key, prop ) {
|
|
|
|
// if the cache doesn't exist, and we know how to convert
|
|
if ( !inst[ cache ] && space.to ) {
|
|
|
|
// if the value was null, we don't need to copy it
|
|
// if the key was alpha, we don't need to copy it either
|
|
if ( key === "alpha" || red[ key ] == null ) {
|
|
return;
|
|
}
|
|
inst[ cache ] = space.to( inst._rgba );
|
|
}
|
|
|
|
// this is the only case where we allow nulls for ALL properties.
|
|
// call clamp with alwaysAllowEmpty
|
|
inst[ cache ][ prop.idx ] = clamp( red[ key ], prop, true );
|
|
});
|
|
|
|
// everything defined but alpha?
|
|
if ( inst[ cache ] && jQuery.inArray( null, inst[ cache ].slice( 0, 3 ) ) < 0 ) {
|
|
// use the default of 1
|
|
inst[ cache ][ 3 ] = 1;
|
|
if ( space.from ) {
|
|
inst._rgba = space.from( inst[ cache ] );
|
|
}
|
|
}
|
|
});
|
|
}
|
|
return this;
|
|
}
|
|
},
|
|
is: function( compare ) {
|
|
var is = color( compare ),
|
|
same = true,
|
|
inst = this;
|
|
|
|
each( spaces, function( _, space ) {
|
|
var localCache,
|
|
isCache = is[ space.cache ];
|
|
if (isCache) {
|
|
localCache = inst[ space.cache ] || space.to && space.to( inst._rgba ) || [];
|
|
each( space.props, function( _, prop ) {
|
|
if ( isCache[ prop.idx ] != null ) {
|
|
same = ( isCache[ prop.idx ] === localCache[ prop.idx ] );
|
|
return same;
|
|
}
|
|
});
|
|
}
|
|
return same;
|
|
});
|
|
return same;
|
|
},
|
|
_space: function() {
|
|
var used = [],
|
|
inst = this;
|
|
each( spaces, function( spaceName, space ) {
|
|
if ( inst[ space.cache ] ) {
|
|
used.push( spaceName );
|
|
}
|
|
});
|
|
return used.pop();
|
|
},
|
|
transition: function( other, distance ) {
|
|
var end = color( other ),
|
|
spaceName = end._space(),
|
|
space = spaces[ spaceName ],
|
|
startColor = this.alpha() === 0 ? color( "transparent" ) : this,
|
|
start = startColor[ space.cache ] || space.to( startColor._rgba ),
|
|
result = start.slice();
|
|
|
|
end = end[ space.cache ];
|
|
each( space.props, function( key, prop ) {
|
|
var index = prop.idx,
|
|
startValue = start[ index ],
|
|
endValue = end[ index ],
|
|
type = propTypes[ prop.type ] || {};
|
|
|
|
// if null, don't override start value
|
|
if ( endValue === null ) {
|
|
return;
|
|
}
|
|
// if null - use end
|
|
if ( startValue === null ) {
|
|
result[ index ] = endValue;
|
|
} else {
|
|
if ( type.mod ) {
|
|
if ( endValue - startValue > type.mod / 2 ) {
|
|
startValue += type.mod;
|
|
} else if ( startValue - endValue > type.mod / 2 ) {
|
|
startValue -= type.mod;
|
|
}
|
|
}
|
|
result[ index ] = clamp( ( endValue - startValue ) * distance + startValue, prop );
|
|
}
|
|
});
|
|
return this[ spaceName ]( result );
|
|
},
|
|
blend: function( opaque ) {
|
|
// if we are already opaque - return ourself
|
|
if ( this._rgba[ 3 ] === 1 ) {
|
|
return this;
|
|
}
|
|
|
|
var rgb = this._rgba.slice(),
|
|
a = rgb.pop(),
|
|
blend = color( opaque )._rgba;
|
|
|
|
return color( jQuery.map( rgb, function( v, i ) {
|
|
return ( 1 - a ) * blend[ i ] + a * v;
|
|
}));
|
|
},
|
|
toRgbaString: function() {
|
|
var prefix = "rgba(",
|
|
rgba = jQuery.map( this._rgba, function( v, i ) {
|
|
return v == null ? ( i > 2 ? 1 : 0 ) : v;
|
|
});
|
|
|
|
if ( rgba[ 3 ] === 1 ) {
|
|
rgba.pop();
|
|
prefix = "rgb(";
|
|
}
|
|
|
|
return prefix + rgba.join() + ")";
|
|
},
|
|
toHslaString: function() {
|
|
var prefix = "hsla(",
|
|
hsla = jQuery.map( this.hsla(), function( v, i ) {
|
|
if ( v == null ) {
|
|
v = i > 2 ? 1 : 0;
|
|
}
|
|
|
|
// catch 1 and 2
|
|
if ( i && i < 3 ) {
|
|
v = Math.round( v * 100 ) + "%";
|
|
}
|
|
return v;
|
|
});
|
|
|
|
if ( hsla[ 3 ] === 1 ) {
|
|
hsla.pop();
|
|
prefix = "hsl(";
|
|
}
|
|
return prefix + hsla.join() + ")";
|
|
},
|
|
toHexString: function( includeAlpha ) {
|
|
var rgba = this._rgba.slice(),
|
|
alpha = rgba.pop();
|
|
|
|
if ( includeAlpha ) {
|
|
rgba.push( ~~( alpha * 255 ) );
|
|
}
|
|
|
|
return "#" + jQuery.map( rgba, function( v ) {
|
|
|
|
// default to 0 when nulls exist
|
|
v = ( v || 0 ).toString( 16 );
|
|
return v.length === 1 ? "0" + v : v;
|
|
}).join("");
|
|
},
|
|
toString: function() {
|
|
return this._rgba[ 3 ] === 0 ? "transparent" : this.toRgbaString();
|
|
}
|
|
});
|
|
color.fn.parse.prototype = color.fn;
|
|
|
|
// hsla conversions adapted from:
|
|
// https://code.google.com/p/maashaack/source/browse/packages/graphics/trunk/src/graphics/colors/HUE2RGB.as?r=5021
|
|
|
|
function hue2rgb( p, q, h ) {
|
|
h = ( h + 1 ) % 1;
|
|
if ( h * 6 < 1 ) {
|
|
return p + ( q - p ) * h * 6;
|
|
}
|
|
if ( h * 2 < 1) {
|
|
return q;
|
|
}
|
|
if ( h * 3 < 2 ) {
|
|
return p + ( q - p ) * ( ( 2 / 3 ) - h ) * 6;
|
|
}
|
|
return p;
|
|
}
|
|
|
|
spaces.hsla.to = function( rgba ) {
|
|
if ( rgba[ 0 ] == null || rgba[ 1 ] == null || rgba[ 2 ] == null ) {
|
|
return [ null, null, null, rgba[ 3 ] ];
|
|
}
|
|
var r = rgba[ 0 ] / 255,
|
|
g = rgba[ 1 ] / 255,
|
|
b = rgba[ 2 ] / 255,
|
|
a = rgba[ 3 ],
|
|
max = Math.max( r, g, b ),
|
|
min = Math.min( r, g, b ),
|
|
diff = max - min,
|
|
add = max + min,
|
|
l = add * 0.5,
|
|
h, s;
|
|
|
|
if ( min === max ) {
|
|
h = 0;
|
|
} else if ( r === max ) {
|
|
h = ( 60 * ( g - b ) / diff ) + 360;
|
|
} else if ( g === max ) {
|
|
h = ( 60 * ( b - r ) / diff ) + 120;
|
|
} else {
|
|
h = ( 60 * ( r - g ) / diff ) + 240;
|
|
}
|
|
|
|
// chroma (diff) == 0 means greyscale which, by definition, saturation = 0%
|
|
// otherwise, saturation is based on the ratio of chroma (diff) to lightness (add)
|
|
if ( diff === 0 ) {
|
|
s = 0;
|
|
} else if ( l <= 0.5 ) {
|
|
s = diff / add;
|
|
} else {
|
|
s = diff / ( 2 - add );
|
|
}
|
|
return [ Math.round(h) % 360, s, l, a == null ? 1 : a ];
|
|
};
|
|
|
|
spaces.hsla.from = function( hsla ) {
|
|
if ( hsla[ 0 ] == null || hsla[ 1 ] == null || hsla[ 2 ] == null ) {
|
|
return [ null, null, null, hsla[ 3 ] ];
|
|
}
|
|
var h = hsla[ 0 ] / 360,
|
|
s = hsla[ 1 ],
|
|
l = hsla[ 2 ],
|
|
a = hsla[ 3 ],
|
|
q = l <= 0.5 ? l * ( 1 + s ) : l + s - l * s,
|
|
p = 2 * l - q;
|
|
|
|
return [
|
|
Math.round( hue2rgb( p, q, h + ( 1 / 3 ) ) * 255 ),
|
|
Math.round( hue2rgb( p, q, h ) * 255 ),
|
|
Math.round( hue2rgb( p, q, h - ( 1 / 3 ) ) * 255 ),
|
|
a
|
|
];
|
|
};
|
|
|
|
each( spaces, function( spaceName, space ) {
|
|
var props = space.props,
|
|
cache = space.cache,
|
|
to = space.to,
|
|
from = space.from;
|
|
|
|
// makes rgba() and hsla()
|
|
color.fn[ spaceName ] = function( value ) {
|
|
|
|
// generate a cache for this space if it doesn't exist
|
|
if ( to && !this[ cache ] ) {
|
|
this[ cache ] = to( this._rgba );
|
|
}
|
|
if ( value === undefined ) {
|
|
return this[ cache ].slice();
|
|
}
|
|
|
|
var ret,
|
|
type = jQuery.type( value ),
|
|
arr = ( type === "array" || type === "object" ) ? value : arguments,
|
|
local = this[ cache ].slice();
|
|
|
|
each( props, function( key, prop ) {
|
|
var val = arr[ type === "object" ? key : prop.idx ];
|
|
if ( val == null ) {
|
|
val = local[ prop.idx ];
|
|
}
|
|
local[ prop.idx ] = clamp( val, prop );
|
|
});
|
|
|
|
if ( from ) {
|
|
ret = color( from( local ) );
|
|
ret[ cache ] = local;
|
|
return ret;
|
|
} else {
|
|
return color( local );
|
|
}
|
|
};
|
|
|
|
// makes red() green() blue() alpha() hue() saturation() lightness()
|
|
each( props, function( key, prop ) {
|
|
// alpha is included in more than one space
|
|
if ( color.fn[ key ] ) {
|
|
return;
|
|
}
|
|
color.fn[ key ] = function( value ) {
|
|
var vtype = jQuery.type( value ),
|
|
fn = ( key === "alpha" ? ( this._hsla ? "hsla" : "rgba" ) : spaceName ),
|
|
local = this[ fn ](),
|
|
cur = local[ prop.idx ],
|
|
match;
|
|
|
|
if ( vtype === "undefined" ) {
|
|
return cur;
|
|
}
|
|
|
|
if ( vtype === "function" ) {
|
|
value = value.call( this, cur );
|
|
vtype = jQuery.type( value );
|
|
}
|
|
if ( value == null && prop.empty ) {
|
|
return this;
|
|
}
|
|
if ( vtype === "string" ) {
|
|
match = rplusequals.exec( value );
|
|
if ( match ) {
|
|
value = cur + parseFloat( match[ 2 ] ) * ( match[ 1 ] === "+" ? 1 : -1 );
|
|
}
|
|
}
|
|
local[ prop.idx ] = value;
|
|
return this[ fn ]( local );
|
|
};
|
|
});
|
|
});
|
|
|
|
// add cssHook and .fx.step function for each named hook.
|
|
// accept a space separated string of properties
|
|
color.hook = function( hook ) {
|
|
var hooks = hook.split( " " );
|
|
each( hooks, function( i, hook ) {
|
|
jQuery.cssHooks[ hook ] = {
|
|
set: function( elem, value ) {
|
|
var parsed, curElem,
|
|
backgroundColor = "";
|
|
|
|
if ( value !== "transparent" && ( jQuery.type( value ) !== "string" || ( parsed = stringParse( value ) ) ) ) {
|
|
value = color( parsed || value );
|
|
if ( !support.rgba && value._rgba[ 3 ] !== 1 ) {
|
|
curElem = hook === "backgroundColor" ? elem.parentNode : elem;
|
|
while (
|
|
(backgroundColor === "" || backgroundColor === "transparent") &&
|
|
curElem && curElem.style
|
|
) {
|
|
try {
|
|
backgroundColor = jQuery.css( curElem, "backgroundColor" );
|
|
curElem = curElem.parentNode;
|
|
} catch ( e ) {
|
|
}
|
|
}
|
|
|
|
value = value.blend( backgroundColor && backgroundColor !== "transparent" ?
|
|
backgroundColor :
|
|
"_default" );
|
|
}
|
|
|
|
value = value.toRgbaString();
|
|
}
|
|
try {
|
|
elem.style[ hook ] = value;
|
|
} catch ( e ) {
|
|
// wrapped to prevent IE from throwing errors on "invalid" values like 'auto' or 'inherit'
|
|
}
|
|
}
|
|
};
|
|
jQuery.fx.step[ hook ] = function( fx ) {
|
|
if ( !fx.colorInit ) {
|
|
fx.start = color( fx.elem, hook );
|
|
fx.end = color( fx.end );
|
|
fx.colorInit = true;
|
|
}
|
|
jQuery.cssHooks[ hook ].set( fx.elem, fx.start.transition( fx.end, fx.pos ) );
|
|
};
|
|
});
|
|
|
|
};
|
|
|
|
color.hook( stepHooks );
|
|
|
|
jQuery.cssHooks.borderColor = {
|
|
expand: function( value ) {
|
|
var expanded = {};
|
|
|
|
each( [ "Top", "Right", "Bottom", "Left" ], function( i, part ) {
|
|
expanded[ "border" + part + "Color" ] = value;
|
|
});
|
|
return expanded;
|
|
}
|
|
};
|
|
|
|
// Basic color names only.
|
|
// Usage of any of the other color names requires adding yourself or including
|
|
// jquery.color.svg-names.js.
|
|
colors = jQuery.Color.names = {
|
|
// 4.1. Basic color keywords
|
|
aqua: "#00ffff",
|
|
black: "#000000",
|
|
blue: "#0000ff",
|
|
fuchsia: "#ff00ff",
|
|
gray: "#808080",
|
|
green: "#008000",
|
|
lime: "#00ff00",
|
|
maroon: "#800000",
|
|
navy: "#000080",
|
|
olive: "#808000",
|
|
purple: "#800080",
|
|
red: "#ff0000",
|
|
silver: "#c0c0c0",
|
|
teal: "#008080",
|
|
white: "#ffffff",
|
|
yellow: "#ffff00",
|
|
|
|
// 4.2.3. "transparent" color keyword
|
|
transparent: [ null, null, null, 0 ],
|
|
|
|
_default: "#ffffff"
|
|
};
|
|
|
|
})( jQuery );
|
|
|
|
/******************************************************************************/
|
|
/****************************** CLASS ANIMATIONS ******************************/
|
|
/******************************************************************************/
|
|
(function() {
|
|
|
|
var classAnimationActions = [ "add", "remove", "toggle" ],
|
|
shorthandStyles = {
|
|
border: 1,
|
|
borderBottom: 1,
|
|
borderColor: 1,
|
|
borderLeft: 1,
|
|
borderRight: 1,
|
|
borderTop: 1,
|
|
borderWidth: 1,
|
|
margin: 1,
|
|
padding: 1
|
|
};
|
|
|
|
$.each([ "borderLeftStyle", "borderRightStyle", "borderBottomStyle", "borderTopStyle" ], function( _, prop ) {
|
|
$.fx.step[ prop ] = function( fx ) {
|
|
if ( fx.end !== "none" && !fx.setAttr || fx.pos === 1 && !fx.setAttr ) {
|
|
jQuery.style( fx.elem, prop, fx.end );
|
|
fx.setAttr = true;
|
|
}
|
|
};
|
|
});
|
|
|
|
function getElementStyles( elem ) {
|
|
var key, len,
|
|
style = elem.ownerDocument.defaultView ?
|
|
elem.ownerDocument.defaultView.getComputedStyle( elem, null ) :
|
|
elem.currentStyle,
|
|
styles = {};
|
|
|
|
if ( style && style.length && style[ 0 ] && style[ style[ 0 ] ] ) {
|
|
len = style.length;
|
|
while ( len-- ) {
|
|
key = style[ len ];
|
|
if ( typeof style[ key ] === "string" ) {
|
|
styles[ $.camelCase( key ) ] = style[ key ];
|
|
}
|
|
}
|
|
// support: Opera, IE <9
|
|
} else {
|
|
for ( key in style ) {
|
|
if ( typeof style[ key ] === "string" ) {
|
|
styles[ key ] = style[ key ];
|
|
}
|
|
}
|
|
}
|
|
|
|
return styles;
|
|
}
|
|
|
|
function styleDifference( oldStyle, newStyle ) {
|
|
var diff = {},
|
|
name, value;
|
|
|
|
for ( name in newStyle ) {
|
|
value = newStyle[ name ];
|
|
if ( oldStyle[ name ] !== value ) {
|
|
if ( !shorthandStyles[ name ] ) {
|
|
if ( $.fx.step[ name ] || !isNaN( parseFloat( value ) ) ) {
|
|
diff[ name ] = value;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return diff;
|
|
}
|
|
|
|
// support: jQuery <1.8
|
|
if ( !$.fn.addBack ) {
|
|
$.fn.addBack = function( selector ) {
|
|
return this.add( selector == null ?
|
|
this.prevObject : this.prevObject.filter( selector )
|
|
);
|
|
};
|
|
}
|
|
|
|
$.effects.animateClass = function( value, duration, easing, callback ) {
|
|
var o = $.speed( duration, easing, callback );
|
|
|
|
return this.queue( function() {
|
|
var animated = $( this ),
|
|
baseClass = animated.attr( "class" ) || "",
|
|
applyClassChange,
|
|
allAnimations = o.children ? animated.find( "*" ).addBack() : animated;
|
|
|
|
// map the animated objects to store the original styles.
|
|
allAnimations = allAnimations.map(function() {
|
|
var el = $( this );
|
|
return {
|
|
el: el,
|
|
start: getElementStyles( this )
|
|
};
|
|
});
|
|
|
|
// apply class change
|
|
applyClassChange = function() {
|
|
$.each( classAnimationActions, function(i, action) {
|
|
if ( value[ action ] ) {
|
|
animated[ action + "Class" ]( value[ action ] );
|
|
}
|
|
});
|
|
};
|
|
applyClassChange();
|
|
|
|
// map all animated objects again - calculate new styles and diff
|
|
allAnimations = allAnimations.map(function() {
|
|
this.end = getElementStyles( this.el[ 0 ] );
|
|
this.diff = styleDifference( this.start, this.end );
|
|
return this;
|
|
});
|
|
|
|
// apply original class
|
|
animated.attr( "class", baseClass );
|
|
|
|
// map all animated objects again - this time collecting a promise
|
|
allAnimations = allAnimations.map(function() {
|
|
var styleInfo = this,
|
|
dfd = $.Deferred(),
|
|
opts = $.extend({}, o, {
|
|
queue: false,
|
|
complete: function() {
|
|
dfd.resolve( styleInfo );
|
|
}
|
|
});
|
|
|
|
this.el.animate( this.diff, opts );
|
|
return dfd.promise();
|
|
});
|
|
|
|
// once all animations have completed:
|
|
$.when.apply( $, allAnimations.get() ).done(function() {
|
|
|
|
// set the final class
|
|
applyClassChange();
|
|
|
|
// for each animated element,
|
|
// clear all css properties that were animated
|
|
$.each( arguments, function() {
|
|
var el = this.el;
|
|
$.each( this.diff, function(key) {
|
|
el.css( key, "" );
|
|
});
|
|
});
|
|
|
|
// this is guarnteed to be there if you use jQuery.speed()
|
|
// it also handles dequeuing the next anim...
|
|
o.complete.call( animated[ 0 ] );
|
|
});
|
|
});
|
|
};
|
|
|
|
$.fn.extend({
|
|
addClass: (function( orig ) {
|
|
return function( classNames, speed, easing, callback ) {
|
|
return speed ?
|
|
$.effects.animateClass.call( this,
|
|
{ add: classNames }, speed, easing, callback ) :
|
|
orig.apply( this, arguments );
|
|
};
|
|
})( $.fn.addClass ),
|
|
|
|
removeClass: (function( orig ) {
|
|
return function( classNames, speed, easing, callback ) {
|
|
return arguments.length > 1 ?
|
|
$.effects.animateClass.call( this,
|
|
{ remove: classNames }, speed, easing, callback ) :
|
|
orig.apply( this, arguments );
|
|
};
|
|
})( $.fn.removeClass ),
|
|
|
|
toggleClass: (function( orig ) {
|
|
return function( classNames, force, speed, easing, callback ) {
|
|
if ( typeof force === "boolean" || force === undefined ) {
|
|
if ( !speed ) {
|
|
// without speed parameter
|
|
return orig.apply( this, arguments );
|
|
} else {
|
|
return $.effects.animateClass.call( this,
|
|
(force ? { add: classNames } : { remove: classNames }),
|
|
speed, easing, callback );
|
|
}
|
|
} else {
|
|
// without force parameter
|
|
return $.effects.animateClass.call( this,
|
|
{ toggle: classNames }, force, speed, easing );
|
|
}
|
|
};
|
|
})( $.fn.toggleClass ),
|
|
|
|
switchClass: function( remove, add, speed, easing, callback) {
|
|
return $.effects.animateClass.call( this, {
|
|
add: add,
|
|
remove: remove
|
|
}, speed, easing, callback );
|
|
}
|
|
});
|
|
|
|
})();
|
|
|
|
/******************************************************************************/
|
|
/*********************************** EFFECTS **********************************/
|
|
/******************************************************************************/
|
|
|
|
(function() {
|
|
|
|
$.extend( $.effects, {
|
|
version: "1.11.4",
|
|
|
|
// Saves a set of properties in a data storage
|
|
save: function( element, set ) {
|
|
for ( var i = 0; i < set.length; i++ ) {
|
|
if ( set[ i ] !== null ) {
|
|
element.data( dataSpace + set[ i ], element[ 0 ].style[ set[ i ] ] );
|
|
}
|
|
}
|
|
},
|
|
|
|
// Restores a set of previously saved properties from a data storage
|
|
restore: function( element, set ) {
|
|
var val, i;
|
|
for ( i = 0; i < set.length; i++ ) {
|
|
if ( set[ i ] !== null ) {
|
|
val = element.data( dataSpace + set[ i ] );
|
|
// support: jQuery 1.6.2
|
|
// http://bugs.jquery.com/ticket/9917
|
|
// jQuery 1.6.2 incorrectly returns undefined for any falsy value.
|
|
// We can't differentiate between "" and 0 here, so we just assume
|
|
// empty string since it's likely to be a more common value...
|
|
if ( val === undefined ) {
|
|
val = "";
|
|
}
|
|
element.css( set[ i ], val );
|
|
}
|
|
}
|
|
},
|
|
|
|
setMode: function( el, mode ) {
|
|
if (mode === "toggle") {
|
|
mode = el.is( ":hidden" ) ? "show" : "hide";
|
|
}
|
|
return mode;
|
|
},
|
|
|
|
// Translates a [top,left] array into a baseline value
|
|
// this should be a little more flexible in the future to handle a string & hash
|
|
getBaseline: function( origin, original ) {
|
|
var y, x;
|
|
switch ( origin[ 0 ] ) {
|
|
case "top": y = 0; break;
|
|
case "middle": y = 0.5; break;
|
|
case "bottom": y = 1; break;
|
|
default: y = origin[ 0 ] / original.height;
|
|
}
|
|
switch ( origin[ 1 ] ) {
|
|
case "left": x = 0; break;
|
|
case "center": x = 0.5; break;
|
|
case "right": x = 1; break;
|
|
default: x = origin[ 1 ] / original.width;
|
|
}
|
|
return {
|
|
x: x,
|
|
y: y
|
|
};
|
|
},
|
|
|
|
// Wraps the element around a wrapper that copies position properties
|
|
createWrapper: function( element ) {
|
|
|
|
// if the element is already wrapped, return it
|
|
if ( element.parent().is( ".ui-effects-wrapper" )) {
|
|
return element.parent();
|
|
}
|
|
|
|
// wrap the element
|
|
var props = {
|
|
width: element.outerWidth(true),
|
|
height: element.outerHeight(true),
|
|
"float": element.css( "float" )
|
|
},
|
|
wrapper = $( "<div></div>" )
|
|
.addClass( "ui-effects-wrapper" )
|
|
.css({
|
|
fontSize: "100%",
|
|
background: "transparent",
|
|
border: "none",
|
|
margin: 0,
|
|
padding: 0
|
|
}),
|
|
// Store the size in case width/height are defined in % - Fixes #5245
|
|
size = {
|
|
width: element.width(),
|
|
height: element.height()
|
|
},
|
|
active = document.activeElement;
|
|
|
|
// support: Firefox
|
|
// Firefox incorrectly exposes anonymous content
|
|
// https://bugzilla.mozilla.org/show_bug.cgi?id=561664
|
|
try {
|
|
active.id;
|
|
} catch ( e ) {
|
|
active = document.body;
|
|
}
|
|
|
|
element.wrap( wrapper );
|
|
|
|
// Fixes #7595 - Elements lose focus when wrapped.
|
|
if ( element[ 0 ] === active || $.contains( element[ 0 ], active ) ) {
|
|
$( active ).focus();
|
|
}
|
|
|
|
wrapper = element.parent(); //Hotfix for jQuery 1.4 since some change in wrap() seems to actually lose the reference to the wrapped element
|
|
|
|
// transfer positioning properties to the wrapper
|
|
if ( element.css( "position" ) === "static" ) {
|
|
wrapper.css({ position: "relative" });
|
|
element.css({ position: "relative" });
|
|
} else {
|
|
$.extend( props, {
|
|
position: element.css( "position" ),
|
|
zIndex: element.css( "z-index" )
|
|
});
|
|
$.each([ "top", "left", "bottom", "right" ], function(i, pos) {
|
|
props[ pos ] = element.css( pos );
|
|
if ( isNaN( parseInt( props[ pos ], 10 ) ) ) {
|
|
props[ pos ] = "auto";
|
|
}
|
|
});
|
|
element.css({
|
|
position: "relative",
|
|
top: 0,
|
|
left: 0,
|
|
right: "auto",
|
|
bottom: "auto"
|
|
});
|
|
}
|
|
element.css(size);
|
|
|
|
return wrapper.css( props ).show();
|
|
},
|
|
|
|
removeWrapper: function( element ) {
|
|
var active = document.activeElement;
|
|
|
|
if ( element.parent().is( ".ui-effects-wrapper" ) ) {
|
|
element.parent().replaceWith( element );
|
|
|
|
// Fixes #7595 - Elements lose focus when wrapped.
|
|
if ( element[ 0 ] === active || $.contains( element[ 0 ], active ) ) {
|
|
$( active ).focus();
|
|
}
|
|
}
|
|
|
|
return element;
|
|
},
|
|
|
|
setTransition: function( element, list, factor, value ) {
|
|
value = value || {};
|
|
$.each( list, function( i, x ) {
|
|
var unit = element.cssUnit( x );
|
|
if ( unit[ 0 ] > 0 ) {
|
|
value[ x ] = unit[ 0 ] * factor + unit[ 1 ];
|
|
}
|
|
});
|
|
return value;
|
|
}
|
|
});
|
|
|
|
// return an effect options object for the given parameters:
|
|
function _normalizeArguments( effect, options, speed, callback ) {
|
|
|
|
// allow passing all options as the first parameter
|
|
if ( $.isPlainObject( effect ) ) {
|
|
options = effect;
|
|
effect = effect.effect;
|
|
}
|
|
|
|
// convert to an object
|
|
effect = { effect: effect };
|
|
|
|
// catch (effect, null, ...)
|
|
if ( options == null ) {
|
|
options = {};
|
|
}
|
|
|
|
// catch (effect, callback)
|
|
if ( $.isFunction( options ) ) {
|
|
callback = options;
|
|
speed = null;
|
|
options = {};
|
|
}
|
|
|
|
// catch (effect, speed, ?)
|
|
if ( typeof options === "number" || $.fx.speeds[ options ] ) {
|
|
callback = speed;
|
|
speed = options;
|
|
options = {};
|
|
}
|
|
|
|
// catch (effect, options, callback)
|
|
if ( $.isFunction( speed ) ) {
|
|
callback = speed;
|
|
speed = null;
|
|
}
|
|
|
|
// add options to effect
|
|
if ( options ) {
|
|
$.extend( effect, options );
|
|
}
|
|
|
|
speed = speed || options.duration;
|
|
effect.duration = $.fx.off ? 0 :
|
|
typeof speed === "number" ? speed :
|
|
speed in $.fx.speeds ? $.fx.speeds[ speed ] :
|
|
$.fx.speeds._default;
|
|
|
|
effect.complete = callback || options.complete;
|
|
|
|
return effect;
|
|
}
|
|
|
|
function standardAnimationOption( option ) {
|
|
// Valid standard speeds (nothing, number, named speed)
|
|
if ( !option || typeof option === "number" || $.fx.speeds[ option ] ) {
|
|
return true;
|
|
}
|
|
|
|
// Invalid strings - treat as "normal" speed
|
|
if ( typeof option === "string" && !$.effects.effect[ option ] ) {
|
|
return true;
|
|
}
|
|
|
|
// Complete callback
|
|
if ( $.isFunction( option ) ) {
|
|
return true;
|
|
}
|
|
|
|
// Options hash (but not naming an effect)
|
|
if ( typeof option === "object" && !option.effect ) {
|
|
return true;
|
|
}
|
|
|
|
// Didn't match any standard API
|
|
return false;
|
|
}
|
|
|
|
$.fn.extend({
|
|
effect: function( /* effect, options, speed, callback */ ) {
|
|
var args = _normalizeArguments.apply( this, arguments ),
|
|
mode = args.mode,
|
|
queue = args.queue,
|
|
effectMethod = $.effects.effect[ args.effect ];
|
|
|
|
if ( $.fx.off || !effectMethod ) {
|
|
// delegate to the original method (e.g., .show()) if possible
|
|
if ( mode ) {
|
|
return this[ mode ]( args.duration, args.complete );
|
|
} else {
|
|
return this.each( function() {
|
|
if ( args.complete ) {
|
|
args.complete.call( this );
|
|
}
|
|
});
|
|
}
|
|
}
|
|
|
|
function run( next ) {
|
|
var elem = $( this ),
|
|
complete = args.complete,
|
|
mode = args.mode;
|
|
|
|
function done() {
|
|
if ( $.isFunction( complete ) ) {
|
|
complete.call( elem[0] );
|
|
}
|
|
if ( $.isFunction( next ) ) {
|
|
next();
|
|
}
|
|
}
|
|
|
|
// If the element already has the correct final state, delegate to
|
|
// the core methods so the internal tracking of "olddisplay" works.
|
|
if ( elem.is( ":hidden" ) ? mode === "hide" : mode === "show" ) {
|
|
elem[ mode ]();
|
|
done();
|
|
} else {
|
|
effectMethod.call( elem[0], args, done );
|
|
}
|
|
}
|
|
|
|
return queue === false ? this.each( run ) : this.queue( queue || "fx", run );
|
|
},
|
|
|
|
show: (function( orig ) {
|
|
return function( option ) {
|
|
if ( standardAnimationOption( option ) ) {
|
|
return orig.apply( this, arguments );
|
|
} else {
|
|
var args = _normalizeArguments.apply( this, arguments );
|
|
args.mode = "show";
|
|
return this.effect.call( this, args );
|
|
}
|
|
};
|
|
})( $.fn.show ),
|
|
|
|
hide: (function( orig ) {
|
|
return function( option ) {
|
|
if ( standardAnimationOption( option ) ) {
|
|
return orig.apply( this, arguments );
|
|
} else {
|
|
var args = _normalizeArguments.apply( this, arguments );
|
|
args.mode = "hide";
|
|
return this.effect.call( this, args );
|
|
}
|
|
};
|
|
})( $.fn.hide ),
|
|
|
|
toggle: (function( orig ) {
|
|
return function( option ) {
|
|
if ( standardAnimationOption( option ) || typeof option === "boolean" ) {
|
|
return orig.apply( this, arguments );
|
|
} else {
|
|
var args = _normalizeArguments.apply( this, arguments );
|
|
args.mode = "toggle";
|
|
return this.effect.call( this, args );
|
|
}
|
|
};
|
|
})( $.fn.toggle ),
|
|
|
|
// helper functions
|
|
cssUnit: function(key) {
|
|
var style = this.css( key ),
|
|
val = [];
|
|
|
|
$.each( [ "em", "px", "%", "pt" ], function( i, unit ) {
|
|
if ( style.indexOf( unit ) > 0 ) {
|
|
val = [ parseFloat( style ), unit ];
|
|
}
|
|
});
|
|
return val;
|
|
}
|
|
});
|
|
|
|
})();
|
|
|
|
/******************************************************************************/
|
|
/*********************************** EASING ***********************************/
|
|
/******************************************************************************/
|
|
|
|
(function() {
|
|
|
|
// based on easing equations from Robert Penner (http://www.robertpenner.com/easing)
|
|
|
|
var baseEasings = {};
|
|
|
|
$.each( [ "Quad", "Cubic", "Quart", "Quint", "Expo" ], function( i, name ) {
|
|
baseEasings[ name ] = function( p ) {
|
|
return Math.pow( p, i + 2 );
|
|
};
|
|
});
|
|
|
|
$.extend( baseEasings, {
|
|
Sine: function( p ) {
|
|
return 1 - Math.cos( p * Math.PI / 2 );
|
|
},
|
|
Circ: function( p ) {
|
|
return 1 - Math.sqrt( 1 - p * p );
|
|
},
|
|
Elastic: function( p ) {
|
|
return p === 0 || p === 1 ? p :
|
|
-Math.pow( 2, 8 * (p - 1) ) * Math.sin( ( (p - 1) * 80 - 7.5 ) * Math.PI / 15 );
|
|
},
|
|
Back: function( p ) {
|
|
return p * p * ( 3 * p - 2 );
|
|
},
|
|
Bounce: function( p ) {
|
|
var pow2,
|
|
bounce = 4;
|
|
|
|
while ( p < ( ( pow2 = Math.pow( 2, --bounce ) ) - 1 ) / 11 ) {}
|
|
return 1 / Math.pow( 4, 3 - bounce ) - 7.5625 * Math.pow( ( pow2 * 3 - 2 ) / 22 - p, 2 );
|
|
}
|
|
});
|
|
|
|
$.each( baseEasings, function( name, easeIn ) {
|
|
$.easing[ "easeIn" + name ] = easeIn;
|
|
$.easing[ "easeOut" + name ] = function( p ) {
|
|
return 1 - easeIn( 1 - p );
|
|
};
|
|
$.easing[ "easeInOut" + name ] = function( p ) {
|
|
return p < 0.5 ?
|
|
easeIn( p * 2 ) / 2 :
|
|
1 - easeIn( p * -2 + 2 ) / 2;
|
|
};
|
|
});
|
|
|
|
})();
|
|
|
|
var effect = $.effects;
|
|
|
|
|
|
/*!
|
|
* jQuery UI Effects Blind 1.11.4
|
|
* http://jqueryui.com
|
|
*
|
|
* Copyright jQuery Foundation and other contributors
|
|
* Released under the MIT license.
|
|
* http://jquery.org/license
|
|
*
|
|
* http://api.jqueryui.com/blind-effect/
|
|
*/
|
|
|
|
|
|
var effectBlind = $.effects.effect.blind = function( o, done ) {
|
|
// Create element
|
|
var el = $( this ),
|
|
rvertical = /up|down|vertical/,
|
|
rpositivemotion = /up|left|vertical|horizontal/,
|
|
props = [ "position", "top", "bottom", "left", "right", "height", "width" ],
|
|
mode = $.effects.setMode( el, o.mode || "hide" ),
|
|
direction = o.direction || "up",
|
|
vertical = rvertical.test( direction ),
|
|
ref = vertical ? "height" : "width",
|
|
ref2 = vertical ? "top" : "left",
|
|
motion = rpositivemotion.test( direction ),
|
|
animation = {},
|
|
show = mode === "show",
|
|
wrapper, distance, margin;
|
|
|
|
// if already wrapped, the wrapper's properties are my property. #6245
|
|
if ( el.parent().is( ".ui-effects-wrapper" ) ) {
|
|
$.effects.save( el.parent(), props );
|
|
} else {
|
|
$.effects.save( el, props );
|
|
}
|
|
el.show();
|
|
wrapper = $.effects.createWrapper( el ).css({
|
|
overflow: "hidden"
|
|
});
|
|
|
|
distance = wrapper[ ref ]();
|
|
margin = parseFloat( wrapper.css( ref2 ) ) || 0;
|
|
|
|
animation[ ref ] = show ? distance : 0;
|
|
if ( !motion ) {
|
|
el
|
|
.css( vertical ? "bottom" : "right", 0 )
|
|
.css( vertical ? "top" : "left", "auto" )
|
|
.css({ position: "absolute" });
|
|
|
|
animation[ ref2 ] = show ? margin : distance + margin;
|
|
}
|
|
|
|
// start at 0 if we are showing
|
|
if ( show ) {
|
|
wrapper.css( ref, 0 );
|
|
if ( !motion ) {
|
|
wrapper.css( ref2, margin + distance );
|
|
}
|
|
}
|
|
|
|
// Animate
|
|
wrapper.animate( animation, {
|
|
duration: o.duration,
|
|
easing: o.easing,
|
|
queue: false,
|
|
complete: function() {
|
|
if ( mode === "hide" ) {
|
|
el.hide();
|
|
}
|
|
$.effects.restore( el, props );
|
|
$.effects.removeWrapper( el );
|
|
done();
|
|
}
|
|
});
|
|
};
|
|
|
|
|
|
/*!
|
|
* jQuery UI Effects Bounce 1.11.4
|
|
* http://jqueryui.com
|
|
*
|
|
* Copyright jQuery Foundation and other contributors
|
|
* Released under the MIT license.
|
|
* http://jquery.org/license
|
|
*
|
|
* http://api.jqueryui.com/bounce-effect/
|
|
*/
|
|
|
|
|
|
var effectBounce = $.effects.effect.bounce = function( o, done ) {
|
|
var el = $( this ),
|
|
props = [ "position", "top", "bottom", "left", "right", "height", "width" ],
|
|
|
|
// defaults:
|
|
mode = $.effects.setMode( el, o.mode || "effect" ),
|
|
hide = mode === "hide",
|
|
show = mode === "show",
|
|
direction = o.direction || "up",
|
|
distance = o.distance,
|
|
times = o.times || 5,
|
|
|
|
// number of internal animations
|
|
anims = times * 2 + ( show || hide ? 1 : 0 ),
|
|
speed = o.duration / anims,
|
|
easing = o.easing,
|
|
|
|
// utility:
|
|
ref = ( direction === "up" || direction === "down" ) ? "top" : "left",
|
|
motion = ( direction === "up" || direction === "left" ),
|
|
i,
|
|
upAnim,
|
|
downAnim,
|
|
|
|
// we will need to re-assemble the queue to stack our animations in place
|
|
queue = el.queue(),
|
|
queuelen = queue.length;
|
|
|
|
// Avoid touching opacity to prevent clearType and PNG issues in IE
|
|
if ( show || hide ) {
|
|
props.push( "opacity" );
|
|
}
|
|
|
|
$.effects.save( el, props );
|
|
el.show();
|
|
$.effects.createWrapper( el ); // Create Wrapper
|
|
|
|
// default distance for the BIGGEST bounce is the outer Distance / 3
|
|
if ( !distance ) {
|
|
distance = el[ ref === "top" ? "outerHeight" : "outerWidth" ]() / 3;
|
|
}
|
|
|
|
if ( show ) {
|
|
downAnim = { opacity: 1 };
|
|
downAnim[ ref ] = 0;
|
|
|
|
// if we are showing, force opacity 0 and set the initial position
|
|
// then do the "first" animation
|
|
el.css( "opacity", 0 )
|
|
.css( ref, motion ? -distance * 2 : distance * 2 )
|
|
.animate( downAnim, speed, easing );
|
|
}
|
|
|
|
// start at the smallest distance if we are hiding
|
|
if ( hide ) {
|
|
distance = distance / Math.pow( 2, times - 1 );
|
|
}
|
|
|
|
downAnim = {};
|
|
downAnim[ ref ] = 0;
|
|
// Bounces up/down/left/right then back to 0 -- times * 2 animations happen here
|
|
for ( i = 0; i < times; i++ ) {
|
|
upAnim = {};
|
|
upAnim[ ref ] = ( motion ? "-=" : "+=" ) + distance;
|
|
|
|
el.animate( upAnim, speed, easing )
|
|
.animate( downAnim, speed, easing );
|
|
|
|
distance = hide ? distance * 2 : distance / 2;
|
|
}
|
|
|
|
// Last Bounce when Hiding
|
|
if ( hide ) {
|
|
upAnim = { opacity: 0 };
|
|
upAnim[ ref ] = ( motion ? "-=" : "+=" ) + distance;
|
|
|
|
el.animate( upAnim, speed, easing );
|
|
}
|
|
|
|
el.queue(function() {
|
|
if ( hide ) {
|
|
el.hide();
|
|
}
|
|
$.effects.restore( el, props );
|
|
$.effects.removeWrapper( el );
|
|
done();
|
|
});
|
|
|
|
// inject all the animations we just queued to be first in line (after "inprogress")
|
|
if ( queuelen > 1) {
|
|
queue.splice.apply( queue,
|
|
[ 1, 0 ].concat( queue.splice( queuelen, anims + 1 ) ) );
|
|
}
|
|
el.dequeue();
|
|
|
|
};
|
|
|
|
|
|
/*!
|
|
* jQuery UI Effects Clip 1.11.4
|
|
* http://jqueryui.com
|
|
*
|
|
* Copyright jQuery Foundation and other contributors
|
|
* Released under the MIT license.
|
|
* http://jquery.org/license
|
|
*
|
|
* http://api.jqueryui.com/clip-effect/
|
|
*/
|
|
|
|
|
|
var effectClip = $.effects.effect.clip = function( o, done ) {
|
|
// Create element
|
|
var el = $( this ),
|
|
props = [ "position", "top", "bottom", "left", "right", "height", "width" ],
|
|
mode = $.effects.setMode( el, o.mode || "hide" ),
|
|
show = mode === "show",
|
|
direction = o.direction || "vertical",
|
|
vert = direction === "vertical",
|
|
size = vert ? "height" : "width",
|
|
position = vert ? "top" : "left",
|
|
animation = {},
|
|
wrapper, animate, distance;
|
|
|
|
// Save & Show
|
|
$.effects.save( el, props );
|
|
el.show();
|
|
|
|
// Create Wrapper
|
|
wrapper = $.effects.createWrapper( el ).css({
|
|
overflow: "hidden"
|
|
});
|
|
animate = ( el[0].tagName === "IMG" ) ? wrapper : el;
|
|
distance = animate[ size ]();
|
|
|
|
// Shift
|
|
if ( show ) {
|
|
animate.css( size, 0 );
|
|
animate.css( position, distance / 2 );
|
|
}
|
|
|
|
// Create Animation Object:
|
|
animation[ size ] = show ? distance : 0;
|
|
animation[ position ] = show ? 0 : distance / 2;
|
|
|
|
// Animate
|
|
animate.animate( animation, {
|
|
queue: false,
|
|
duration: o.duration,
|
|
easing: o.easing,
|
|
complete: function() {
|
|
if ( !show ) {
|
|
el.hide();
|
|
}
|
|
$.effects.restore( el, props );
|
|
$.effects.removeWrapper( el );
|
|
done();
|
|
}
|
|
});
|
|
|
|
};
|
|
|
|
|
|
/*!
|
|
* jQuery UI Effects Drop 1.11.4
|
|
* http://jqueryui.com
|
|
*
|
|
* Copyright jQuery Foundation and other contributors
|
|
* Released under the MIT license.
|
|
* http://jquery.org/license
|
|
*
|
|
* http://api.jqueryui.com/drop-effect/
|
|
*/
|
|
|
|
|
|
var effectDrop = $.effects.effect.drop = function( o, done ) {
|
|
|
|
var el = $( this ),
|
|
props = [ "position", "top", "bottom", "left", "right", "opacity", "height", "width" ],
|
|
mode = $.effects.setMode( el, o.mode || "hide" ),
|
|
show = mode === "show",
|
|
direction = o.direction || "left",
|
|
ref = ( direction === "up" || direction === "down" ) ? "top" : "left",
|
|
motion = ( direction === "up" || direction === "left" ) ? "pos" : "neg",
|
|
animation = {
|
|
opacity: show ? 1 : 0
|
|
},
|
|
distance;
|
|
|
|
// Adjust
|
|
$.effects.save( el, props );
|
|
el.show();
|
|
$.effects.createWrapper( el );
|
|
|
|
distance = o.distance || el[ ref === "top" ? "outerHeight" : "outerWidth" ]( true ) / 2;
|
|
|
|
if ( show ) {
|
|
el
|
|
.css( "opacity", 0 )
|
|
.css( ref, motion === "pos" ? -distance : distance );
|
|
}
|
|
|
|
// Animation
|
|
animation[ ref ] = ( show ?
|
|
( motion === "pos" ? "+=" : "-=" ) :
|
|
( motion === "pos" ? "-=" : "+=" ) ) +
|
|
distance;
|
|
|
|
// Animate
|
|
el.animate( animation, {
|
|
queue: false,
|
|
duration: o.duration,
|
|
easing: o.easing,
|
|
complete: function() {
|
|
if ( mode === "hide" ) {
|
|
el.hide();
|
|
}
|
|
$.effects.restore( el, props );
|
|
$.effects.removeWrapper( el );
|
|
done();
|
|
}
|
|
});
|
|
};
|
|
|
|
|
|
/*!
|
|
* jQuery UI Effects Explode 1.11.4
|
|
* http://jqueryui.com
|
|
*
|
|
* Copyright jQuery Foundation and other contributors
|
|
* Released under the MIT license.
|
|
* http://jquery.org/license
|
|
*
|
|
* http://api.jqueryui.com/explode-effect/
|
|
*/
|
|
|
|
|
|
var effectExplode = $.effects.effect.explode = function( o, done ) {
|
|
|
|
var rows = o.pieces ? Math.round( Math.sqrt( o.pieces ) ) : 3,
|
|
cells = rows,
|
|
el = $( this ),
|
|
mode = $.effects.setMode( el, o.mode || "hide" ),
|
|
show = mode === "show",
|
|
|
|
// show and then visibility:hidden the element before calculating offset
|
|
offset = el.show().css( "visibility", "hidden" ).offset(),
|
|
|
|
// width and height of a piece
|
|
width = Math.ceil( el.outerWidth() / cells ),
|
|
height = Math.ceil( el.outerHeight() / rows ),
|
|
pieces = [],
|
|
|
|
// loop
|
|
i, j, left, top, mx, my;
|
|
|
|
// children animate complete:
|
|
function childComplete() {
|
|
pieces.push( this );
|
|
if ( pieces.length === rows * cells ) {
|
|
animComplete();
|
|
}
|
|
}
|
|
|
|
// clone the element for each row and cell.
|
|
for ( i = 0; i < rows ; i++ ) { // ===>
|
|
top = offset.top + i * height;
|
|
my = i - ( rows - 1 ) / 2 ;
|
|
|
|
for ( j = 0; j < cells ; j++ ) { // |||
|
|
left = offset.left + j * width;
|
|
mx = j - ( cells - 1 ) / 2 ;
|
|
|
|
// Create a clone of the now hidden main element that will be absolute positioned
|
|
// within a wrapper div off the -left and -top equal to size of our pieces
|
|
el
|
|
.clone()
|
|
.appendTo( "body" )
|
|
.wrap( "<div></div>" )
|
|
.css({
|
|
position: "absolute",
|
|
visibility: "visible",
|
|
left: -j * width,
|
|
top: -i * height
|
|
})
|
|
|
|
// select the wrapper - make it overflow: hidden and absolute positioned based on
|
|
// where the original was located +left and +top equal to the size of pieces
|
|
.parent()
|
|
.addClass( "ui-effects-explode" )
|
|
.css({
|
|
position: "absolute",
|
|
overflow: "hidden",
|
|
width: width,
|
|
height: height,
|
|
left: left + ( show ? mx * width : 0 ),
|
|
top: top + ( show ? my * height : 0 ),
|
|
opacity: show ? 0 : 1
|
|
}).animate({
|
|
left: left + ( show ? 0 : mx * width ),
|
|
top: top + ( show ? 0 : my * height ),
|
|
opacity: show ? 1 : 0
|
|
}, o.duration || 500, o.easing, childComplete );
|
|
}
|
|
}
|
|
|
|
function animComplete() {
|
|
el.css({
|
|
visibility: "visible"
|
|
});
|
|
$( pieces ).remove();
|
|
if ( !show ) {
|
|
el.hide();
|
|
}
|
|
done();
|
|
}
|
|
};
|
|
|
|
|
|
/*!
|
|
* jQuery UI Effects Fade 1.11.4
|
|
* http://jqueryui.com
|
|
*
|
|
* Copyright jQuery Foundation and other contributors
|
|
* Released under the MIT license.
|
|
* http://jquery.org/license
|
|
*
|
|
* http://api.jqueryui.com/fade-effect/
|
|
*/
|
|
|
|
|
|
var effectFade = $.effects.effect.fade = function( o, done ) {
|
|
var el = $( this ),
|
|
mode = $.effects.setMode( el, o.mode || "toggle" );
|
|
|
|
el.animate({
|
|
opacity: mode
|
|
}, {
|
|
queue: false,
|
|
duration: o.duration,
|
|
easing: o.easing,
|
|
complete: done
|
|
});
|
|
};
|
|
|
|
|
|
/*!
|
|
* jQuery UI Effects Fold 1.11.4
|
|
* http://jqueryui.com
|
|
*
|
|
* Copyright jQuery Foundation and other contributors
|
|
* Released under the MIT license.
|
|
* http://jquery.org/license
|
|
*
|
|
* http://api.jqueryui.com/fold-effect/
|
|
*/
|
|
|
|
|
|
var effectFold = $.effects.effect.fold = function( o, done ) {
|
|
|
|
// Create element
|
|
var el = $( this ),
|
|
props = [ "position", "top", "bottom", "left", "right", "height", "width" ],
|
|
mode = $.effects.setMode( el, o.mode || "hide" ),
|
|
show = mode === "show",
|
|
hide = mode === "hide",
|
|
size = o.size || 15,
|
|
percent = /([0-9]+)%/.exec( size ),
|
|
horizFirst = !!o.horizFirst,
|
|
widthFirst = show !== horizFirst,
|
|
ref = widthFirst ? [ "width", "height" ] : [ "height", "width" ],
|
|
duration = o.duration / 2,
|
|
wrapper, distance,
|
|
animation1 = {},
|
|
animation2 = {};
|
|
|
|
$.effects.save( el, props );
|
|
el.show();
|
|
|
|
// Create Wrapper
|
|
wrapper = $.effects.createWrapper( el ).css({
|
|
overflow: "hidden"
|
|
});
|
|
distance = widthFirst ?
|
|
[ wrapper.width(), wrapper.height() ] :
|
|
[ wrapper.height(), wrapper.width() ];
|
|
|
|
if ( percent ) {
|
|
size = parseInt( percent[ 1 ], 10 ) / 100 * distance[ hide ? 0 : 1 ];
|
|
}
|
|
if ( show ) {
|
|
wrapper.css( horizFirst ? {
|
|
height: 0,
|
|
width: size
|
|
} : {
|
|
height: size,
|
|
width: 0
|
|
});
|
|
}
|
|
|
|
// Animation
|
|
animation1[ ref[ 0 ] ] = show ? distance[ 0 ] : size;
|
|
animation2[ ref[ 1 ] ] = show ? distance[ 1 ] : 0;
|
|
|
|
// Animate
|
|
wrapper
|
|
.animate( animation1, duration, o.easing )
|
|
.animate( animation2, duration, o.easing, function() {
|
|
if ( hide ) {
|
|
el.hide();
|
|
}
|
|
$.effects.restore( el, props );
|
|
$.effects.removeWrapper( el );
|
|
done();
|
|
});
|
|
|
|
};
|
|
|
|
|
|
/*!
|
|
* jQuery UI Effects Highlight 1.11.4
|
|
* http://jqueryui.com
|
|
*
|
|
* Copyright jQuery Foundation and other contributors
|
|
* Released under the MIT license.
|
|
* http://jquery.org/license
|
|
*
|
|
* http://api.jqueryui.com/highlight-effect/
|
|
*/
|
|
|
|
|
|
var effectHighlight = $.effects.effect.highlight = function( o, done ) {
|
|
var elem = $( this ),
|
|
props = [ "backgroundImage", "backgroundColor", "opacity" ],
|
|
mode = $.effects.setMode( elem, o.mode || "show" ),
|
|
animation = {
|
|
backgroundColor: elem.css( "backgroundColor" )
|
|
};
|
|
|
|
if (mode === "hide") {
|
|
animation.opacity = 0;
|
|
}
|
|
|
|
$.effects.save( elem, props );
|
|
|
|
elem
|
|
.show()
|
|
.css({
|
|
backgroundImage: "none",
|
|
backgroundColor: o.color || "#ffff99"
|
|
})
|
|
.animate( animation, {
|
|
queue: false,
|
|
duration: o.duration,
|
|
easing: o.easing,
|
|
complete: function() {
|
|
if ( mode === "hide" ) {
|
|
elem.hide();
|
|
}
|
|
$.effects.restore( elem, props );
|
|
done();
|
|
}
|
|
});
|
|
};
|
|
|
|
|
|
/*!
|
|
* jQuery UI Effects Size 1.11.4
|
|
* http://jqueryui.com
|
|
*
|
|
* Copyright jQuery Foundation and other contributors
|
|
* Released under the MIT license.
|
|
* http://jquery.org/license
|
|
*
|
|
* http://api.jqueryui.com/size-effect/
|
|
*/
|
|
|
|
|
|
var effectSize = $.effects.effect.size = function( o, done ) {
|
|
|
|
// Create element
|
|
var original, baseline, factor,
|
|
el = $( this ),
|
|
props0 = [ "position", "top", "bottom", "left", "right", "width", "height", "overflow", "opacity" ],
|
|
|
|
// Always restore
|
|
props1 = [ "position", "top", "bottom", "left", "right", "overflow", "opacity" ],
|
|
|
|
// Copy for children
|
|
props2 = [ "width", "height", "overflow" ],
|
|
cProps = [ "fontSize" ],
|
|
vProps = [ "borderTopWidth", "borderBottomWidth", "paddingTop", "paddingBottom" ],
|
|
hProps = [ "borderLeftWidth", "borderRightWidth", "paddingLeft", "paddingRight" ],
|
|
|
|
// Set options
|
|
mode = $.effects.setMode( el, o.mode || "effect" ),
|
|
restore = o.restore || mode !== "effect",
|
|
scale = o.scale || "both",
|
|
origin = o.origin || [ "middle", "center" ],
|
|
position = el.css( "position" ),
|
|
props = restore ? props0 : props1,
|
|
zero = {
|
|
height: 0,
|
|
width: 0,
|
|
outerHeight: 0,
|
|
outerWidth: 0
|
|
};
|
|
|
|
if ( mode === "show" ) {
|
|
el.show();
|
|
}
|
|
original = {
|
|
height: el.height(),
|
|
width: el.width(),
|
|
outerHeight: el.outerHeight(),
|
|
outerWidth: el.outerWidth()
|
|
};
|
|
|
|
if ( o.mode === "toggle" && mode === "show" ) {
|
|
el.from = o.to || zero;
|
|
el.to = o.from || original;
|
|
} else {
|
|
el.from = o.from || ( mode === "show" ? zero : original );
|
|
el.to = o.to || ( mode === "hide" ? zero : original );
|
|
}
|
|
|
|
// Set scaling factor
|
|
factor = {
|
|
from: {
|
|
y: el.from.height / original.height,
|
|
x: el.from.width / original.width
|
|
},
|
|
to: {
|
|
y: el.to.height / original.height,
|
|
x: el.to.width / original.width
|
|
}
|
|
};
|
|
|
|
// Scale the css box
|
|
if ( scale === "box" || scale === "both" ) {
|
|
|
|
// Vertical props scaling
|
|
if ( factor.from.y !== factor.to.y ) {
|
|
props = props.concat( vProps );
|
|
el.from = $.effects.setTransition( el, vProps, factor.from.y, el.from );
|
|
el.to = $.effects.setTransition( el, vProps, factor.to.y, el.to );
|
|
}
|
|
|
|
// Horizontal props scaling
|
|
if ( factor.from.x !== factor.to.x ) {
|
|
props = props.concat( hProps );
|
|
el.from = $.effects.setTransition( el, hProps, factor.from.x, el.from );
|
|
el.to = $.effects.setTransition( el, hProps, factor.to.x, el.to );
|
|
}
|
|
}
|
|
|
|
// Scale the content
|
|
if ( scale === "content" || scale === "both" ) {
|
|
|
|
// Vertical props scaling
|
|
if ( factor.from.y !== factor.to.y ) {
|
|
props = props.concat( cProps ).concat( props2 );
|
|
el.from = $.effects.setTransition( el, cProps, factor.from.y, el.from );
|
|
el.to = $.effects.setTransition( el, cProps, factor.to.y, el.to );
|
|
}
|
|
}
|
|
|
|
$.effects.save( el, props );
|
|
el.show();
|
|
$.effects.createWrapper( el );
|
|
el.css( "overflow", "hidden" ).css( el.from );
|
|
|
|
// Adjust
|
|
if (origin) { // Calculate baseline shifts
|
|
baseline = $.effects.getBaseline( origin, original );
|
|
el.from.top = ( original.outerHeight - el.outerHeight() ) * baseline.y;
|
|
el.from.left = ( original.outerWidth - el.outerWidth() ) * baseline.x;
|
|
el.to.top = ( original.outerHeight - el.to.outerHeight ) * baseline.y;
|
|
el.to.left = ( original.outerWidth - el.to.outerWidth ) * baseline.x;
|
|
}
|
|
el.css( el.from ); // set top & left
|
|
|
|
// Animate
|
|
if ( scale === "content" || scale === "both" ) { // Scale the children
|
|
|
|
// Add margins/font-size
|
|
vProps = vProps.concat([ "marginTop", "marginBottom" ]).concat(cProps);
|
|
hProps = hProps.concat([ "marginLeft", "marginRight" ]);
|
|
props2 = props0.concat(vProps).concat(hProps);
|
|
|
|
el.find( "*[width]" ).each( function() {
|
|
var child = $( this ),
|
|
c_original = {
|
|
height: child.height(),
|
|
width: child.width(),
|
|
outerHeight: child.outerHeight(),
|
|
outerWidth: child.outerWidth()
|
|
};
|
|
if (restore) {
|
|
$.effects.save(child, props2);
|
|
}
|
|
|
|
child.from = {
|
|
height: c_original.height * factor.from.y,
|
|
width: c_original.width * factor.from.x,
|
|
outerHeight: c_original.outerHeight * factor.from.y,
|
|
outerWidth: c_original.outerWidth * factor.from.x
|
|
};
|
|
child.to = {
|
|
height: c_original.height * factor.to.y,
|
|
width: c_original.width * factor.to.x,
|
|
outerHeight: c_original.height * factor.to.y,
|
|
outerWidth: c_original.width * factor.to.x
|
|
};
|
|
|
|
// Vertical props scaling
|
|
if ( factor.from.y !== factor.to.y ) {
|
|
child.from = $.effects.setTransition( child, vProps, factor.from.y, child.from );
|
|
child.to = $.effects.setTransition( child, vProps, factor.to.y, child.to );
|
|
}
|
|
|
|
// Horizontal props scaling
|
|
if ( factor.from.x !== factor.to.x ) {
|
|
child.from = $.effects.setTransition( child, hProps, factor.from.x, child.from );
|
|
child.to = $.effects.setTransition( child, hProps, factor.to.x, child.to );
|
|
}
|
|
|
|
// Animate children
|
|
child.css( child.from );
|
|
child.animate( child.to, o.duration, o.easing, function() {
|
|
|
|
// Restore children
|
|
if ( restore ) {
|
|
$.effects.restore( child, props2 );
|
|
}
|
|
});
|
|
});
|
|
}
|
|
|
|
// Animate
|
|
el.animate( el.to, {
|
|
queue: false,
|
|
duration: o.duration,
|
|
easing: o.easing,
|
|
complete: function() {
|
|
if ( el.to.opacity === 0 ) {
|
|
el.css( "opacity", el.from.opacity );
|
|
}
|
|
if ( mode === "hide" ) {
|
|
el.hide();
|
|
}
|
|
$.effects.restore( el, props );
|
|
if ( !restore ) {
|
|
|
|
// we need to calculate our new positioning based on the scaling
|
|
if ( position === "static" ) {
|
|
el.css({
|
|
position: "relative",
|
|
top: el.to.top,
|
|
left: el.to.left
|
|
});
|
|
} else {
|
|
$.each([ "top", "left" ], function( idx, pos ) {
|
|
el.css( pos, function( _, str ) {
|
|
var val = parseInt( str, 10 ),
|
|
toRef = idx ? el.to.left : el.to.top;
|
|
|
|
// if original was "auto", recalculate the new value from wrapper
|
|
if ( str === "auto" ) {
|
|
return toRef + "px";
|
|
}
|
|
|
|
return val + toRef + "px";
|
|
});
|
|
});
|
|
}
|
|
}
|
|
|
|
$.effects.removeWrapper( el );
|
|
done();
|
|
}
|
|
});
|
|
|
|
};
|
|
|
|
|
|
/*!
|
|
* jQuery UI Effects Scale 1.11.4
|
|
* http://jqueryui.com
|
|
*
|
|
* Copyright jQuery Foundation and other contributors
|
|
* Released under the MIT license.
|
|
* http://jquery.org/license
|
|
*
|
|
* http://api.jqueryui.com/scale-effect/
|
|
*/
|
|
|
|
|
|
var effectScale = $.effects.effect.scale = function( o, done ) {
|
|
|
|
// Create element
|
|
var el = $( this ),
|
|
options = $.extend( true, {}, o ),
|
|
mode = $.effects.setMode( el, o.mode || "effect" ),
|
|
percent = parseInt( o.percent, 10 ) ||
|
|
( parseInt( o.percent, 10 ) === 0 ? 0 : ( mode === "hide" ? 0 : 100 ) ),
|
|
direction = o.direction || "both",
|
|
origin = o.origin,
|
|
original = {
|
|
height: el.height(),
|
|
width: el.width(),
|
|
outerHeight: el.outerHeight(),
|
|
outerWidth: el.outerWidth()
|
|
},
|
|
factor = {
|
|
y: direction !== "horizontal" ? (percent / 100) : 1,
|
|
x: direction !== "vertical" ? (percent / 100) : 1
|
|
};
|
|
|
|
// We are going to pass this effect to the size effect:
|
|
options.effect = "size";
|
|
options.queue = false;
|
|
options.complete = done;
|
|
|
|
// Set default origin and restore for show/hide
|
|
if ( mode !== "effect" ) {
|
|
options.origin = origin || [ "middle", "center" ];
|
|
options.restore = true;
|
|
}
|
|
|
|
options.from = o.from || ( mode === "show" ? {
|
|
height: 0,
|
|
width: 0,
|
|
outerHeight: 0,
|
|
outerWidth: 0
|
|
} : original );
|
|
options.to = {
|
|
height: original.height * factor.y,
|
|
width: original.width * factor.x,
|
|
outerHeight: original.outerHeight * factor.y,
|
|
outerWidth: original.outerWidth * factor.x
|
|
};
|
|
|
|
// Fade option to support puff
|
|
if ( options.fade ) {
|
|
if ( mode === "show" ) {
|
|
options.from.opacity = 0;
|
|
options.to.opacity = 1;
|
|
}
|
|
if ( mode === "hide" ) {
|
|
options.from.opacity = 1;
|
|
options.to.opacity = 0;
|
|
}
|
|
}
|
|
|
|
// Animate
|
|
el.effect( options );
|
|
|
|
};
|
|
|
|
|
|
/*!
|
|
* jQuery UI Effects Puff 1.11.4
|
|
* http://jqueryui.com
|
|
*
|
|
* Copyright jQuery Foundation and other contributors
|
|
* Released under the MIT license.
|
|
* http://jquery.org/license
|
|
*
|
|
* http://api.jqueryui.com/puff-effect/
|
|
*/
|
|
|
|
|
|
var effectPuff = $.effects.effect.puff = function( o, done ) {
|
|
var elem = $( this ),
|
|
mode = $.effects.setMode( elem, o.mode || "hide" ),
|
|
hide = mode === "hide",
|
|
percent = parseInt( o.percent, 10 ) || 150,
|
|
factor = percent / 100,
|
|
original = {
|
|
height: elem.height(),
|
|
width: elem.width(),
|
|
outerHeight: elem.outerHeight(),
|
|
outerWidth: elem.outerWidth()
|
|
};
|
|
|
|
$.extend( o, {
|
|
effect: "scale",
|
|
queue: false,
|
|
fade: true,
|
|
mode: mode,
|
|
complete: done,
|
|
percent: hide ? percent : 100,
|
|
from: hide ?
|
|
original :
|
|
{
|
|
height: original.height * factor,
|
|
width: original.width * factor,
|
|
outerHeight: original.outerHeight * factor,
|
|
outerWidth: original.outerWidth * factor
|
|
}
|
|
});
|
|
|
|
elem.effect( o );
|
|
};
|
|
|
|
|
|
/*!
|
|
* jQuery UI Effects Pulsate 1.11.4
|
|
* http://jqueryui.com
|
|
*
|
|
* Copyright jQuery Foundation and other contributors
|
|
* Released under the MIT license.
|
|
* http://jquery.org/license
|
|
*
|
|
* http://api.jqueryui.com/pulsate-effect/
|
|
*/
|
|
|
|
|
|
var effectPulsate = $.effects.effect.pulsate = function( o, done ) {
|
|
var elem = $( this ),
|
|
mode = $.effects.setMode( elem, o.mode || "show" ),
|
|
show = mode === "show",
|
|
hide = mode === "hide",
|
|
showhide = ( show || mode === "hide" ),
|
|
|
|
// showing or hiding leaves of the "last" animation
|
|
anims = ( ( o.times || 5 ) * 2 ) + ( showhide ? 1 : 0 ),
|
|
duration = o.duration / anims,
|
|
animateTo = 0,
|
|
queue = elem.queue(),
|
|
queuelen = queue.length,
|
|
i;
|
|
|
|
if ( show || !elem.is(":visible")) {
|
|
elem.css( "opacity", 0 ).show();
|
|
animateTo = 1;
|
|
}
|
|
|
|
// anims - 1 opacity "toggles"
|
|
for ( i = 1; i < anims; i++ ) {
|
|
elem.animate({
|
|
opacity: animateTo
|
|
}, duration, o.easing );
|
|
animateTo = 1 - animateTo;
|
|
}
|
|
|
|
elem.animate({
|
|
opacity: animateTo
|
|
}, duration, o.easing);
|
|
|
|
elem.queue(function() {
|
|
if ( hide ) {
|
|
elem.hide();
|
|
}
|
|
done();
|
|
});
|
|
|
|
// We just queued up "anims" animations, we need to put them next in the queue
|
|
if ( queuelen > 1 ) {
|
|
queue.splice.apply( queue,
|
|
[ 1, 0 ].concat( queue.splice( queuelen, anims + 1 ) ) );
|
|
}
|
|
elem.dequeue();
|
|
};
|
|
|
|
|
|
/*!
|
|
* jQuery UI Effects Shake 1.11.4
|
|
* http://jqueryui.com
|
|
*
|
|
* Copyright jQuery Foundation and other contributors
|
|
* Released under the MIT license.
|
|
* http://jquery.org/license
|
|
*
|
|
* http://api.jqueryui.com/shake-effect/
|
|
*/
|
|
|
|
|
|
var effectShake = $.effects.effect.shake = function( o, done ) {
|
|
|
|
var el = $( this ),
|
|
props = [ "position", "top", "bottom", "left", "right", "height", "width" ],
|
|
mode = $.effects.setMode( el, o.mode || "effect" ),
|
|
direction = o.direction || "left",
|
|
distance = o.distance || 20,
|
|
times = o.times || 3,
|
|
anims = times * 2 + 1,
|
|
speed = Math.round( o.duration / anims ),
|
|
ref = (direction === "up" || direction === "down") ? "top" : "left",
|
|
positiveMotion = (direction === "up" || direction === "left"),
|
|
animation = {},
|
|
animation1 = {},
|
|
animation2 = {},
|
|
i,
|
|
|
|
// we will need to re-assemble the queue to stack our animations in place
|
|
queue = el.queue(),
|
|
queuelen = queue.length;
|
|
|
|
$.effects.save( el, props );
|
|
el.show();
|
|
$.effects.createWrapper( el );
|
|
|
|
// Animation
|
|
animation[ ref ] = ( positiveMotion ? "-=" : "+=" ) + distance;
|
|
animation1[ ref ] = ( positiveMotion ? "+=" : "-=" ) + distance * 2;
|
|
animation2[ ref ] = ( positiveMotion ? "-=" : "+=" ) + distance * 2;
|
|
|
|
// Animate
|
|
el.animate( animation, speed, o.easing );
|
|
|
|
// Shakes
|
|
for ( i = 1; i < times; i++ ) {
|
|
el.animate( animation1, speed, o.easing ).animate( animation2, speed, o.easing );
|
|
}
|
|
el
|
|
.animate( animation1, speed, o.easing )
|
|
.animate( animation, speed / 2, o.easing )
|
|
.queue(function() {
|
|
if ( mode === "hide" ) {
|
|
el.hide();
|
|
}
|
|
$.effects.restore( el, props );
|
|
$.effects.removeWrapper( el );
|
|
done();
|
|
});
|
|
|
|
// inject all the animations we just queued to be first in line (after "inprogress")
|
|
if ( queuelen > 1) {
|
|
queue.splice.apply( queue,
|
|
[ 1, 0 ].concat( queue.splice( queuelen, anims + 1 ) ) );
|
|
}
|
|
el.dequeue();
|
|
|
|
};
|
|
|
|
|
|
/*!
|
|
* jQuery UI Effects Slide 1.11.4
|
|
* http://jqueryui.com
|
|
*
|
|
* Copyright jQuery Foundation and other contributors
|
|
* Released under the MIT license.
|
|
* http://jquery.org/license
|
|
*
|
|
* http://api.jqueryui.com/slide-effect/
|
|
*/
|
|
|
|
|
|
var effectSlide = $.effects.effect.slide = function( o, done ) {
|
|
|
|
// Create element
|
|
var el = $( this ),
|
|
props = [ "position", "top", "bottom", "left", "right", "width", "height" ],
|
|
mode = $.effects.setMode( el, o.mode || "show" ),
|
|
show = mode === "show",
|
|
direction = o.direction || "left",
|
|
ref = (direction === "up" || direction === "down") ? "top" : "left",
|
|
positiveMotion = (direction === "up" || direction === "left"),
|
|
distance,
|
|
animation = {};
|
|
|
|
// Adjust
|
|
$.effects.save( el, props );
|
|
el.show();
|
|
distance = o.distance || el[ ref === "top" ? "outerHeight" : "outerWidth" ]( true );
|
|
|
|
$.effects.createWrapper( el ).css({
|
|
overflow: "hidden"
|
|
});
|
|
|
|
if ( show ) {
|
|
el.css( ref, positiveMotion ? (isNaN(distance) ? "-" + distance : -distance) : distance );
|
|
}
|
|
|
|
// Animation
|
|
animation[ ref ] = ( show ?
|
|
( positiveMotion ? "+=" : "-=") :
|
|
( positiveMotion ? "-=" : "+=")) +
|
|
distance;
|
|
|
|
// Animate
|
|
el.animate( animation, {
|
|
queue: false,
|
|
duration: o.duration,
|
|
easing: o.easing,
|
|
complete: function() {
|
|
if ( mode === "hide" ) {
|
|
el.hide();
|
|
}
|
|
$.effects.restore( el, props );
|
|
$.effects.removeWrapper( el );
|
|
done();
|
|
}
|
|
});
|
|
};
|
|
|
|
|
|
/*!
|
|
* jQuery UI Effects Transfer 1.11.4
|
|
* http://jqueryui.com
|
|
*
|
|
* Copyright jQuery Foundation and other contributors
|
|
* Released under the MIT license.
|
|
* http://jquery.org/license
|
|
*
|
|
* http://api.jqueryui.com/transfer-effect/
|
|
*/
|
|
|
|
|
|
var effectTransfer = $.effects.effect.transfer = function( o, done ) {
|
|
var elem = $( this ),
|
|
target = $( o.to ),
|
|
targetFixed = target.css( "position" ) === "fixed",
|
|
body = $("body"),
|
|
fixTop = targetFixed ? body.scrollTop() : 0,
|
|
fixLeft = targetFixed ? body.scrollLeft() : 0,
|
|
endPosition = target.offset(),
|
|
animation = {
|
|
top: endPosition.top - fixTop,
|
|
left: endPosition.left - fixLeft,
|
|
height: target.innerHeight(),
|
|
width: target.innerWidth()
|
|
},
|
|
startPosition = elem.offset(),
|
|
transfer = $( "<div class='ui-effects-transfer'></div>" )
|
|
.appendTo( document.body )
|
|
.addClass( o.className )
|
|
.css({
|
|
top: startPosition.top - fixTop,
|
|
left: startPosition.left - fixLeft,
|
|
height: elem.innerHeight(),
|
|
width: elem.innerWidth(),
|
|
position: targetFixed ? "fixed" : "absolute"
|
|
})
|
|
.animate( animation, o.duration, o.easing, function() {
|
|
transfer.remove();
|
|
done();
|
|
});
|
|
};
|
|
|
|
|
|
/*!
|
|
* jQuery UI Progressbar 1.11.4
|
|
* http://jqueryui.com
|
|
*
|
|
* Copyright jQuery Foundation and other contributors
|
|
* Released under the MIT license.
|
|
* http://jquery.org/license
|
|
*
|
|
* http://api.jqueryui.com/progressbar/
|
|
*/
|
|
|
|
|
|
var progressbar = $.widget( "ui.progressbar", {
|
|
version: "1.11.4",
|
|
options: {
|
|
max: 100,
|
|
value: 0,
|
|
|
|
change: null,
|
|
complete: null
|
|
},
|
|
|
|
min: 0,
|
|
|
|
_create: function() {
|
|
// Constrain initial value
|
|
this.oldValue = this.options.value = this._constrainedValue();
|
|
|
|
this.element
|
|
.addClass( "ui-progressbar ui-widget ui-widget-content ui-corner-all" )
|
|
.attr({
|
|
// Only set static values, aria-valuenow and aria-valuemax are
|
|
// set inside _refreshValue()
|
|
role: "progressbar",
|
|
"aria-valuemin": this.min
|
|
});
|
|
|
|
this.valueDiv = $( "<div class='ui-progressbar-value ui-widget-header ui-corner-left'></div>" )
|
|
.appendTo( this.element );
|
|
|
|
this._refreshValue();
|
|
},
|
|
|
|
_destroy: function() {
|
|
this.element
|
|
.removeClass( "ui-progressbar ui-widget ui-widget-content ui-corner-all" )
|
|
.removeAttr( "role" )
|
|
.removeAttr( "aria-valuemin" )
|
|
.removeAttr( "aria-valuemax" )
|
|
.removeAttr( "aria-valuenow" );
|
|
|
|
this.valueDiv.remove();
|
|
},
|
|
|
|
value: function( newValue ) {
|
|
if ( newValue === undefined ) {
|
|
return this.options.value;
|
|
}
|
|
|
|
this.options.value = this._constrainedValue( newValue );
|
|
this._refreshValue();
|
|
},
|
|
|
|
_constrainedValue: function( newValue ) {
|
|
if ( newValue === undefined ) {
|
|
newValue = this.options.value;
|
|
}
|
|
|
|
this.indeterminate = newValue === false;
|
|
|
|
// sanitize value
|
|
if ( typeof newValue !== "number" ) {
|
|
newValue = 0;
|
|
}
|
|
|
|
return this.indeterminate ? false :
|
|
Math.min( this.options.max, Math.max( this.min, newValue ) );
|
|
},
|
|
|
|
_setOptions: function( options ) {
|
|
// Ensure "value" option is set after other values (like max)
|
|
var value = options.value;
|
|
delete options.value;
|
|
|
|
this._super( options );
|
|
|
|
this.options.value = this._constrainedValue( value );
|
|
this._refreshValue();
|
|
},
|
|
|
|
_setOption: function( key, value ) {
|
|
if ( key === "max" ) {
|
|
// Don't allow a max less than min
|
|
value = Math.max( this.min, value );
|
|
}
|
|
if ( key === "disabled" ) {
|
|
this.element
|
|
.toggleClass( "ui-state-disabled", !!value )
|
|
.attr( "aria-disabled", value );
|
|
}
|
|
this._super( key, value );
|
|
},
|
|
|
|
_percentage: function() {
|
|
return this.indeterminate ? 100 : 100 * ( this.options.value - this.min ) / ( this.options.max - this.min );
|
|
},
|
|
|
|
_refreshValue: function() {
|
|
var value = this.options.value,
|
|
percentage = this._percentage();
|
|
|
|
this.valueDiv
|
|
.toggle( this.indeterminate || value > this.min )
|
|
.toggleClass( "ui-corner-right", value === this.options.max )
|
|
.width( percentage.toFixed(0) + "%" );
|
|
|
|
this.element.toggleClass( "ui-progressbar-indeterminate", this.indeterminate );
|
|
|
|
if ( this.indeterminate ) {
|
|
this.element.removeAttr( "aria-valuenow" );
|
|
if ( !this.overlayDiv ) {
|
|
this.overlayDiv = $( "<div class='ui-progressbar-overlay'></div>" ).appendTo( this.valueDiv );
|
|
}
|
|
} else {
|
|
this.element.attr({
|
|
"aria-valuemax": this.options.max,
|
|
"aria-valuenow": value
|
|
});
|
|
if ( this.overlayDiv ) {
|
|
this.overlayDiv.remove();
|
|
this.overlayDiv = null;
|
|
}
|
|
}
|
|
|
|
if ( this.oldValue !== value ) {
|
|
this.oldValue = value;
|
|
this._trigger( "change" );
|
|
}
|
|
if ( value === this.options.max ) {
|
|
this._trigger( "complete" );
|
|
}
|
|
}
|
|
});
|
|
|
|
|
|
/*!
|
|
* jQuery UI Selectable 1.11.4
|
|
* http://jqueryui.com
|
|
*
|
|
* Copyright jQuery Foundation and other contributors
|
|
* Released under the MIT license.
|
|
* http://jquery.org/license
|
|
*
|
|
* http://api.jqueryui.com/selectable/
|
|
*/
|
|
|
|
|
|
var selectable = $.widget("ui.selectable", $.ui.mouse, {
|
|
version: "1.11.4",
|
|
options: {
|
|
appendTo: "body",
|
|
autoRefresh: true,
|
|
distance: 0,
|
|
filter: "*",
|
|
tolerance: "touch",
|
|
|
|
// callbacks
|
|
selected: null,
|
|
selecting: null,
|
|
start: null,
|
|
stop: null,
|
|
unselected: null,
|
|
unselecting: null
|
|
},
|
|
_create: function() {
|
|
var selectees,
|
|
that = this;
|
|
|
|
this.element.addClass("ui-selectable");
|
|
|
|
this.dragged = false;
|
|
|
|
// cache selectee children based on filter
|
|
this.refresh = function() {
|
|
selectees = $(that.options.filter, that.element[0]);
|
|
selectees.addClass("ui-selectee");
|
|
selectees.each(function() {
|
|
var $this = $(this),
|
|
pos = $this.offset();
|
|
$.data(this, "selectable-item", {
|
|
element: this,
|
|
$element: $this,
|
|
left: pos.left,
|
|
top: pos.top,
|
|
right: pos.left + $this.outerWidth(),
|
|
bottom: pos.top + $this.outerHeight(),
|
|
startselected: false,
|
|
selected: $this.hasClass("ui-selected"),
|
|
selecting: $this.hasClass("ui-selecting"),
|
|
unselecting: $this.hasClass("ui-unselecting")
|
|
});
|
|
});
|
|
};
|
|
this.refresh();
|
|
|
|
this.selectees = selectees.addClass("ui-selectee");
|
|
|
|
this._mouseInit();
|
|
|
|
this.helper = $("<div class='ui-selectable-helper'></div>");
|
|
},
|
|
|
|
_destroy: function() {
|
|
this.selectees
|
|
.removeClass("ui-selectee")
|
|
.removeData("selectable-item");
|
|
this.element
|
|
.removeClass("ui-selectable ui-selectable-disabled");
|
|
this._mouseDestroy();
|
|
},
|
|
|
|
_mouseStart: function(event) {
|
|
var that = this,
|
|
options = this.options;
|
|
|
|
this.opos = [ event.pageX, event.pageY ];
|
|
|
|
if (this.options.disabled) {
|
|
return;
|
|
}
|
|
|
|
this.selectees = $(options.filter, this.element[0]);
|
|
|
|
this._trigger("start", event);
|
|
|
|
$(options.appendTo).append(this.helper);
|
|
// position helper (lasso)
|
|
this.helper.css({
|
|
"left": event.pageX,
|
|
"top": event.pageY,
|
|
"width": 0,
|
|
"height": 0
|
|
});
|
|
|
|
if (options.autoRefresh) {
|
|
this.refresh();
|
|
}
|
|
|
|
this.selectees.filter(".ui-selected").each(function() {
|
|
var selectee = $.data(this, "selectable-item");
|
|
selectee.startselected = true;
|
|
if (!event.metaKey && !event.ctrlKey) {
|
|
selectee.$element.removeClass("ui-selected");
|
|
selectee.selected = false;
|
|
selectee.$element.addClass("ui-unselecting");
|
|
selectee.unselecting = true;
|
|
// selectable UNSELECTING callback
|
|
that._trigger("unselecting", event, {
|
|
unselecting: selectee.element
|
|
});
|
|
}
|
|
});
|
|
|
|
$(event.target).parents().addBack().each(function() {
|
|
var doSelect,
|
|
selectee = $.data(this, "selectable-item");
|
|
if (selectee) {
|
|
doSelect = (!event.metaKey && !event.ctrlKey) || !selectee.$element.hasClass("ui-selected");
|
|
selectee.$element
|
|
.removeClass(doSelect ? "ui-unselecting" : "ui-selected")
|
|
.addClass(doSelect ? "ui-selecting" : "ui-unselecting");
|
|
selectee.unselecting = !doSelect;
|
|
selectee.selecting = doSelect;
|
|
selectee.selected = doSelect;
|
|
// selectable (UN)SELECTING callback
|
|
if (doSelect) {
|
|
that._trigger("selecting", event, {
|
|
selecting: selectee.element
|
|
});
|
|
} else {
|
|
that._trigger("unselecting", event, {
|
|
unselecting: selectee.element
|
|
});
|
|
}
|
|
return false;
|
|
}
|
|
});
|
|
|
|
},
|
|
|
|
_mouseDrag: function(event) {
|
|
|
|
this.dragged = true;
|
|
|
|
if (this.options.disabled) {
|
|
return;
|
|
}
|
|
|
|
var tmp,
|
|
that = this,
|
|
options = this.options,
|
|
x1 = this.opos[0],
|
|
y1 = this.opos[1],
|
|
x2 = event.pageX,
|
|
y2 = event.pageY;
|
|
|
|
if (x1 > x2) { tmp = x2; x2 = x1; x1 = tmp; }
|
|
if (y1 > y2) { tmp = y2; y2 = y1; y1 = tmp; }
|
|
this.helper.css({ left: x1, top: y1, width: x2 - x1, height: y2 - y1 });
|
|
|
|
this.selectees.each(function() {
|
|
var selectee = $.data(this, "selectable-item"),
|
|
hit = false;
|
|
|
|
//prevent helper from being selected if appendTo: selectable
|
|
if (!selectee || selectee.element === that.element[0]) {
|
|
return;
|
|
}
|
|
|
|
if (options.tolerance === "touch") {
|
|
hit = ( !(selectee.left > x2 || selectee.right < x1 || selectee.top > y2 || selectee.bottom < y1) );
|
|
} else if (options.tolerance === "fit") {
|
|
hit = (selectee.left > x1 && selectee.right < x2 && selectee.top > y1 && selectee.bottom < y2);
|
|
}
|
|
|
|
if (hit) {
|
|
// SELECT
|
|
if (selectee.selected) {
|
|
selectee.$element.removeClass("ui-selected");
|
|
selectee.selected = false;
|
|
}
|
|
if (selectee.unselecting) {
|
|
selectee.$element.removeClass("ui-unselecting");
|
|
selectee.unselecting = false;
|
|
}
|
|
if (!selectee.selecting) {
|
|
selectee.$element.addClass("ui-selecting");
|
|
selectee.selecting = true;
|
|
// selectable SELECTING callback
|
|
that._trigger("selecting", event, {
|
|
selecting: selectee.element
|
|
});
|
|
}
|
|
} else {
|
|
// UNSELECT
|
|
if (selectee.selecting) {
|
|
if ((event.metaKey || event.ctrlKey) && selectee.startselected) {
|
|
selectee.$element.removeClass("ui-selecting");
|
|
selectee.selecting = false;
|
|
selectee.$element.addClass("ui-selected");
|
|
selectee.selected = true;
|
|
} else {
|
|
selectee.$element.removeClass("ui-selecting");
|
|
selectee.selecting = false;
|
|
if (selectee.startselected) {
|
|
selectee.$element.addClass("ui-unselecting");
|
|
selectee.unselecting = true;
|
|
}
|
|
// selectable UNSELECTING callback
|
|
that._trigger("unselecting", event, {
|
|
unselecting: selectee.element
|
|
});
|
|
}
|
|
}
|
|
if (selectee.selected) {
|
|
if (!event.metaKey && !event.ctrlKey && !selectee.startselected) {
|
|
selectee.$element.removeClass("ui-selected");
|
|
selectee.selected = false;
|
|
|
|
selectee.$element.addClass("ui-unselecting");
|
|
selectee.unselecting = true;
|
|
// selectable UNSELECTING callback
|
|
that._trigger("unselecting", event, {
|
|
unselecting: selectee.element
|
|
});
|
|
}
|
|
}
|
|
}
|
|
});
|
|
|
|
return false;
|
|
},
|
|
|
|
_mouseStop: function(event) {
|
|
var that = this;
|
|
|
|
this.dragged = false;
|
|
|
|
$(".ui-unselecting", this.element[0]).each(function() {
|
|
var selectee = $.data(this, "selectable-item");
|
|
selectee.$element.removeClass("ui-unselecting");
|
|
selectee.unselecting = false;
|
|
selectee.startselected = false;
|
|
that._trigger("unselected", event, {
|
|
unselected: selectee.element
|
|
});
|
|
});
|
|
$(".ui-selecting", this.element[0]).each(function() {
|
|
var selectee = $.data(this, "selectable-item");
|
|
selectee.$element.removeClass("ui-selecting").addClass("ui-selected");
|
|
selectee.selecting = false;
|
|
selectee.selected = true;
|
|
selectee.startselected = true;
|
|
that._trigger("selected", event, {
|
|
selected: selectee.element
|
|
});
|
|
});
|
|
this._trigger("stop", event);
|
|
|
|
this.helper.remove();
|
|
|
|
return false;
|
|
}
|
|
|
|
});
|
|
|
|
|
|
/*!
|
|
* jQuery UI Selectmenu 1.11.4
|
|
* http://jqueryui.com
|
|
*
|
|
* Copyright jQuery Foundation and other contributors
|
|
* Released under the MIT license.
|
|
* http://jquery.org/license
|
|
*
|
|
* http://api.jqueryui.com/selectmenu
|
|
*/
|
|
|
|
|
|
var selectmenu = $.widget( "ui.selectmenu", {
|
|
version: "1.11.4",
|
|
defaultElement: "<select>",
|
|
options: {
|
|
appendTo: null,
|
|
disabled: null,
|
|
icons: {
|
|
button: "ui-icon-triangle-1-s"
|
|
},
|
|
position: {
|
|
my: "left top",
|
|
at: "left bottom",
|
|
collision: "none"
|
|
},
|
|
width: null,
|
|
|
|
// callbacks
|
|
change: null,
|
|
close: null,
|
|
focus: null,
|
|
open: null,
|
|
select: null
|
|
},
|
|
|
|
_create: function() {
|
|
var selectmenuId = this.element.uniqueId().attr( "id" );
|
|
this.ids = {
|
|
element: selectmenuId,
|
|
button: selectmenuId + "-button",
|
|
menu: selectmenuId + "-menu"
|
|
};
|
|
|
|
this._drawButton();
|
|
this._drawMenu();
|
|
|
|
if ( this.options.disabled ) {
|
|
this.disable();
|
|
}
|
|
},
|
|
|
|
_drawButton: function() {
|
|
var that = this;
|
|
|
|
// Associate existing label with the new button
|
|
this.label = $( "label[for='" + this.ids.element + "']" ).attr( "for", this.ids.button );
|
|
this._on( this.label, {
|
|
click: function( event ) {
|
|
this.button.focus();
|
|
event.preventDefault();
|
|
}
|
|
});
|
|
|
|
// Hide original select element
|
|
this.element.hide();
|
|
|
|
// Create button
|
|
this.button = $( "<span>", {
|
|
"class": "ui-selectmenu-button ui-widget ui-state-default ui-corner-all",
|
|
tabindex: this.options.disabled ? -1 : 0,
|
|
id: this.ids.button,
|
|
role: "combobox",
|
|
"aria-expanded": "false",
|
|
"aria-autocomplete": "list",
|
|
"aria-owns": this.ids.menu,
|
|
"aria-haspopup": "true"
|
|
})
|
|
.insertAfter( this.element );
|
|
|
|
$( "<span>", {
|
|
"class": "ui-icon " + this.options.icons.button
|
|
})
|
|
.prependTo( this.button );
|
|
|
|
this.buttonText = $( "<span>", {
|
|
"class": "ui-selectmenu-text"
|
|
})
|
|
.appendTo( this.button );
|
|
|
|
this._setText( this.buttonText, this.element.find( "option:selected" ).text() );
|
|
this._resizeButton();
|
|
|
|
this._on( this.button, this._buttonEvents );
|
|
this.button.one( "focusin", function() {
|
|
|
|
// Delay rendering the menu items until the button receives focus.
|
|
// The menu may have already been rendered via a programmatic open.
|
|
if ( !that.menuItems ) {
|
|
that._refreshMenu();
|
|
}
|
|
});
|
|
this._hoverable( this.button );
|
|
this._focusable( this.button );
|
|
},
|
|
|
|
_drawMenu: function() {
|
|
var that = this;
|
|
|
|
// Create menu
|
|
this.menu = $( "<ul>", {
|
|
"aria-hidden": "true",
|
|
"aria-labelledby": this.ids.button,
|
|
id: this.ids.menu
|
|
});
|
|
|
|
// Wrap menu
|
|
this.menuWrap = $( "<div>", {
|
|
"class": "ui-selectmenu-menu ui-front"
|
|
})
|
|
.append( this.menu )
|
|
.appendTo( this._appendTo() );
|
|
|
|
// Initialize menu widget
|
|
this.menuInstance = this.menu
|
|
.menu({
|
|
role: "listbox",
|
|
select: function( event, ui ) {
|
|
event.preventDefault();
|
|
|
|
// support: IE8
|
|
// If the item was selected via a click, the text selection
|
|
// will be destroyed in IE
|
|
that._setSelection();
|
|
|
|
that._select( ui.item.data( "ui-selectmenu-item" ), event );
|
|
},
|
|
focus: function( event, ui ) {
|
|
var item = ui.item.data( "ui-selectmenu-item" );
|
|
|
|
// Prevent inital focus from firing and check if its a newly focused item
|
|
if ( that.focusIndex != null && item.index !== that.focusIndex ) {
|
|
that._trigger( "focus", event, { item: item } );
|
|
if ( !that.isOpen ) {
|
|
that._select( item, event );
|
|
}
|
|
}
|
|
that.focusIndex = item.index;
|
|
|
|
that.button.attr( "aria-activedescendant",
|
|
that.menuItems.eq( item.index ).attr( "id" ) );
|
|
}
|
|
})
|
|
.menu( "instance" );
|
|
|
|
// Adjust menu styles to dropdown
|
|
this.menu
|
|
.addClass( "ui-corner-bottom" )
|
|
.removeClass( "ui-corner-all" );
|
|
|
|
// Don't close the menu on mouseleave
|
|
this.menuInstance._off( this.menu, "mouseleave" );
|
|
|
|
// Cancel the menu's collapseAll on document click
|
|
this.menuInstance._closeOnDocumentClick = function() {
|
|
return false;
|
|
};
|
|
|
|
// Selects often contain empty items, but never contain dividers
|
|
this.menuInstance._isDivider = function() {
|
|
return false;
|
|
};
|
|
},
|
|
|
|
refresh: function() {
|
|
this._refreshMenu();
|
|
this._setText( this.buttonText, this._getSelectedItem().text() );
|
|
if ( !this.options.width ) {
|
|
this._resizeButton();
|
|
}
|
|
},
|
|
|
|
_refreshMenu: function() {
|
|
this.menu.empty();
|
|
|
|
var item,
|
|
options = this.element.find( "option" );
|
|
|
|
if ( !options.length ) {
|
|
return;
|
|
}
|
|
|
|
this._parseOptions( options );
|
|
this._renderMenu( this.menu, this.items );
|
|
|
|
this.menuInstance.refresh();
|
|
this.menuItems = this.menu.find( "li" ).not( ".ui-selectmenu-optgroup" );
|
|
|
|
item = this._getSelectedItem();
|
|
|
|
// Update the menu to have the correct item focused
|
|
this.menuInstance.focus( null, item );
|
|
this._setAria( item.data( "ui-selectmenu-item" ) );
|
|
|
|
// Set disabled state
|
|
this._setOption( "disabled", this.element.prop( "disabled" ) );
|
|
},
|
|
|
|
open: function( event ) {
|
|
if ( this.options.disabled ) {
|
|
return;
|
|
}
|
|
|
|
// If this is the first time the menu is being opened, render the items
|
|
if ( !this.menuItems ) {
|
|
this._refreshMenu();
|
|
} else {
|
|
|
|
// Menu clears focus on close, reset focus to selected item
|
|
this.menu.find( ".ui-state-focus" ).removeClass( "ui-state-focus" );
|
|
this.menuInstance.focus( null, this._getSelectedItem() );
|
|
}
|
|
|
|
this.isOpen = true;
|
|
this._toggleAttr();
|
|
this._resizeMenu();
|
|
this._position();
|
|
|
|
this._on( this.document, this._documentClick );
|
|
|
|
this._trigger( "open", event );
|
|
},
|
|
|
|
_position: function() {
|
|
this.menuWrap.position( $.extend( { of: this.button }, this.options.position ) );
|
|
},
|
|
|
|
close: function( event ) {
|
|
if ( !this.isOpen ) {
|
|
return;
|
|
}
|
|
|
|
this.isOpen = false;
|
|
this._toggleAttr();
|
|
|
|
this.range = null;
|
|
this._off( this.document );
|
|
|
|
this._trigger( "close", event );
|
|
},
|
|
|
|
widget: function() {
|
|
return this.button;
|
|
},
|
|
|
|
menuWidget: function() {
|
|
return this.menu;
|
|
},
|
|
|
|
_renderMenu: function( ul, items ) {
|
|
var that = this,
|
|
currentOptgroup = "";
|
|
|
|
$.each( items, function( index, item ) {
|
|
if ( item.optgroup !== currentOptgroup ) {
|
|
$( "<li>", {
|
|
"class": "ui-selectmenu-optgroup ui-menu-divider" +
|
|
( item.element.parent( "optgroup" ).prop( "disabled" ) ?
|
|
" ui-state-disabled" :
|
|
"" ),
|
|
text: item.optgroup
|
|
})
|
|
.appendTo( ul );
|
|
|
|
currentOptgroup = item.optgroup;
|
|
}
|
|
|
|
that._renderItemData( ul, item );
|
|
});
|
|
},
|
|
|
|
_renderItemData: function( ul, item ) {
|
|
return this._renderItem( ul, item ).data( "ui-selectmenu-item", item );
|
|
},
|
|
|
|
_renderItem: function( ul, item ) {
|
|
var li = $( "<li>" );
|
|
|
|
if ( item.disabled ) {
|
|
li.addClass( "ui-state-disabled" );
|
|
}
|
|
this._setText( li, item.label );
|
|
|
|
return li.appendTo( ul );
|
|
},
|
|
|
|
_setText: function( element, value ) {
|
|
if ( value ) {
|
|
element.text( value );
|
|
} else {
|
|
element.html( " " );
|
|
}
|
|
},
|
|
|
|
_move: function( direction, event ) {
|
|
var item, next,
|
|
filter = ".ui-menu-item";
|
|
|
|
if ( this.isOpen ) {
|
|
item = this.menuItems.eq( this.focusIndex );
|
|
} else {
|
|
item = this.menuItems.eq( this.element[ 0 ].selectedIndex );
|
|
filter += ":not(.ui-state-disabled)";
|
|
}
|
|
|
|
if ( direction === "first" || direction === "last" ) {
|
|
next = item[ direction === "first" ? "prevAll" : "nextAll" ]( filter ).eq( -1 );
|
|
} else {
|
|
next = item[ direction + "All" ]( filter ).eq( 0 );
|
|
}
|
|
|
|
if ( next.length ) {
|
|
this.menuInstance.focus( event, next );
|
|
}
|
|
},
|
|
|
|
_getSelectedItem: function() {
|
|
return this.menuItems.eq( this.element[ 0 ].selectedIndex );
|
|
},
|
|
|
|
_toggle: function( event ) {
|
|
this[ this.isOpen ? "close" : "open" ]( event );
|
|
},
|
|
|
|
_setSelection: function() {
|
|
var selection;
|
|
|
|
if ( !this.range ) {
|
|
return;
|
|
}
|
|
|
|
if ( window.getSelection ) {
|
|
selection = window.getSelection();
|
|
selection.removeAllRanges();
|
|
selection.addRange( this.range );
|
|
|
|
// support: IE8
|
|
} else {
|
|
this.range.select();
|
|
}
|
|
|
|
// support: IE
|
|
// Setting the text selection kills the button focus in IE, but
|
|
// restoring the focus doesn't kill the selection.
|
|
this.button.focus();
|
|
},
|
|
|
|
_documentClick: {
|
|
mousedown: function( event ) {
|
|
if ( !this.isOpen ) {
|
|
return;
|
|
}
|
|
|
|
if ( !$( event.target ).closest( ".ui-selectmenu-menu, #" + this.ids.button ).length ) {
|
|
this.close( event );
|
|
}
|
|
}
|
|
},
|
|
|
|
_buttonEvents: {
|
|
|
|
// Prevent text selection from being reset when interacting with the selectmenu (#10144)
|
|
mousedown: function() {
|
|
var selection;
|
|
|
|
if ( window.getSelection ) {
|
|
selection = window.getSelection();
|
|
if ( selection.rangeCount ) {
|
|
this.range = selection.getRangeAt( 0 );
|
|
}
|
|
|
|
// support: IE8
|
|
} else {
|
|
this.range = document.selection.createRange();
|
|
}
|
|
},
|
|
|
|
click: function( event ) {
|
|
this._setSelection();
|
|
this._toggle( event );
|
|
},
|
|
|
|
keydown: function( event ) {
|
|
var preventDefault = true;
|
|
switch ( event.keyCode ) {
|
|
case $.ui.keyCode.TAB:
|
|
case $.ui.keyCode.ESCAPE:
|
|
this.close( event );
|
|
preventDefault = false;
|
|
break;
|
|
case $.ui.keyCode.ENTER:
|
|
if ( this.isOpen ) {
|
|
this._selectFocusedItem( event );
|
|
}
|
|
break;
|
|
case $.ui.keyCode.UP:
|
|
if ( event.altKey ) {
|
|
this._toggle( event );
|
|
} else {
|
|
this._move( "prev", event );
|
|
}
|
|
break;
|
|
case $.ui.keyCode.DOWN:
|
|
if ( event.altKey ) {
|
|
this._toggle( event );
|
|
} else {
|
|
this._move( "next", event );
|
|
}
|
|
break;
|
|
case $.ui.keyCode.SPACE:
|
|
if ( this.isOpen ) {
|
|
this._selectFocusedItem( event );
|
|
} else {
|
|
this._toggle( event );
|
|
}
|
|
break;
|
|
case $.ui.keyCode.LEFT:
|
|
this._move( "prev", event );
|
|
break;
|
|
case $.ui.keyCode.RIGHT:
|
|
this._move( "next", event );
|
|
break;
|
|
case $.ui.keyCode.HOME:
|
|
case $.ui.keyCode.PAGE_UP:
|
|
this._move( "first", event );
|
|
break;
|
|
case $.ui.keyCode.END:
|
|
case $.ui.keyCode.PAGE_DOWN:
|
|
this._move( "last", event );
|
|
break;
|
|
default:
|
|
this.menu.trigger( event );
|
|
preventDefault = false;
|
|
}
|
|
|
|
if ( preventDefault ) {
|
|
event.preventDefault();
|
|
}
|
|
}
|
|
},
|
|
|
|
_selectFocusedItem: function( event ) {
|
|
var item = this.menuItems.eq( this.focusIndex );
|
|
if ( !item.hasClass( "ui-state-disabled" ) ) {
|
|
this._select( item.data( "ui-selectmenu-item" ), event );
|
|
}
|
|
},
|
|
|
|
_select: function( item, event ) {
|
|
var oldIndex = this.element[ 0 ].selectedIndex;
|
|
|
|
// Change native select element
|
|
this.element[ 0 ].selectedIndex = item.index;
|
|
this._setText( this.buttonText, item.label );
|
|
this._setAria( item );
|
|
this._trigger( "select", event, { item: item } );
|
|
|
|
if ( item.index !== oldIndex ) {
|
|
this._trigger( "change", event, { item: item } );
|
|
}
|
|
|
|
this.close( event );
|
|
},
|
|
|
|
_setAria: function( item ) {
|
|
var id = this.menuItems.eq( item.index ).attr( "id" );
|
|
|
|
this.button.attr({
|
|
"aria-labelledby": id,
|
|
"aria-activedescendant": id
|
|
});
|
|
this.menu.attr( "aria-activedescendant", id );
|
|
},
|
|
|
|
_setOption: function( key, value ) {
|
|
if ( key === "icons" ) {
|
|
this.button.find( "span.ui-icon" )
|
|
.removeClass( this.options.icons.button )
|
|
.addClass( value.button );
|
|
}
|
|
|
|
this._super( key, value );
|
|
|
|
if ( key === "appendTo" ) {
|
|
this.menuWrap.appendTo( this._appendTo() );
|
|
}
|
|
|
|
if ( key === "disabled" ) {
|
|
this.menuInstance.option( "disabled", value );
|
|
this.button
|
|
.toggleClass( "ui-state-disabled", value )
|
|
.attr( "aria-disabled", value );
|
|
|
|
this.element.prop( "disabled", value );
|
|
if ( value ) {
|
|
this.button.attr( "tabindex", -1 );
|
|
this.close();
|
|
} else {
|
|
this.button.attr( "tabindex", 0 );
|
|
}
|
|
}
|
|
|
|
if ( key === "width" ) {
|
|
this._resizeButton();
|
|
}
|
|
},
|
|
|
|
_appendTo: function() {
|
|
var element = this.options.appendTo;
|
|
|
|
if ( element ) {
|
|
element = element.jquery || element.nodeType ?
|
|
$( element ) :
|
|
this.document.find( element ).eq( 0 );
|
|
}
|
|
|
|
if ( !element || !element[ 0 ] ) {
|
|
element = this.element.closest( ".ui-front" );
|
|
}
|
|
|
|
if ( !element.length ) {
|
|
element = this.document[ 0 ].body;
|
|
}
|
|
|
|
return element;
|
|
},
|
|
|
|
_toggleAttr: function() {
|
|
this.button
|
|
.toggleClass( "ui-corner-top", this.isOpen )
|
|
.toggleClass( "ui-corner-all", !this.isOpen )
|
|
.attr( "aria-expanded", this.isOpen );
|
|
this.menuWrap.toggleClass( "ui-selectmenu-open", this.isOpen );
|
|
this.menu.attr( "aria-hidden", !this.isOpen );
|
|
},
|
|
|
|
_resizeButton: function() {
|
|
var width = this.options.width;
|
|
|
|
if ( !width ) {
|
|
width = this.element.show().outerWidth();
|
|
this.element.hide();
|
|
}
|
|
|
|
this.button.outerWidth( width );
|
|
},
|
|
|
|
_resizeMenu: function() {
|
|
this.menu.outerWidth( Math.max(
|
|
this.button.outerWidth(),
|
|
|
|
// support: IE10
|
|
// IE10 wraps long text (possibly a rounding bug)
|
|
// so we add 1px to avoid the wrapping
|
|
this.menu.width( "" ).outerWidth() + 1
|
|
) );
|
|
},
|
|
|
|
_getCreateOptions: function() {
|
|
return { disabled: this.element.prop( "disabled" ) };
|
|
},
|
|
|
|
_parseOptions: function( options ) {
|
|
var data = [];
|
|
options.each(function( index, item ) {
|
|
var option = $( item ),
|
|
optgroup = option.parent( "optgroup" );
|
|
data.push({
|
|
element: option,
|
|
index: index,
|
|
value: option.val(),
|
|
label: option.text(),
|
|
optgroup: optgroup.attr( "label" ) || "",
|
|
disabled: optgroup.prop( "disabled" ) || option.prop( "disabled" )
|
|
});
|
|
});
|
|
this.items = data;
|
|
},
|
|
|
|
_destroy: function() {
|
|
this.menuWrap.remove();
|
|
this.button.remove();
|
|
this.element.show();
|
|
this.element.removeUniqueId();
|
|
this.label.attr( "for", this.ids.element );
|
|
}
|
|
});
|
|
|
|
|
|
/*!
|
|
* jQuery UI Slider 1.11.4
|
|
* http://jqueryui.com
|
|
*
|
|
* Copyright jQuery Foundation and other contributors
|
|
* Released under the MIT license.
|
|
* http://jquery.org/license
|
|
*
|
|
* http://api.jqueryui.com/slider/
|
|
*/
|
|
|
|
|
|
var slider = $.widget( "ui.slider", $.ui.mouse, {
|
|
version: "1.11.4",
|
|
widgetEventPrefix: "slide",
|
|
|
|
options: {
|
|
animate: false,
|
|
distance: 0,
|
|
max: 100,
|
|
min: 0,
|
|
orientation: "horizontal",
|
|
range: false,
|
|
step: 1,
|
|
value: 0,
|
|
values: null,
|
|
|
|
// callbacks
|
|
change: null,
|
|
slide: null,
|
|
start: null,
|
|
stop: null
|
|
},
|
|
|
|
// number of pages in a slider
|
|
// (how many times can you page up/down to go through the whole range)
|
|
numPages: 5,
|
|
|
|
_create: function() {
|
|
this._keySliding = false;
|
|
this._mouseSliding = false;
|
|
this._animateOff = true;
|
|
this._handleIndex = null;
|
|
this._detectOrientation();
|
|
this._mouseInit();
|
|
this._calculateNewMax();
|
|
|
|
this.element
|
|
.addClass( "ui-slider" +
|
|
" ui-slider-" + this.orientation +
|
|
" ui-widget" +
|
|
" ui-widget-content" +
|
|
" ui-corner-all");
|
|
|
|
this._refresh();
|
|
this._setOption( "disabled", this.options.disabled );
|
|
|
|
this._animateOff = false;
|
|
},
|
|
|
|
_refresh: function() {
|
|
this._createRange();
|
|
this._createHandles();
|
|
this._setupEvents();
|
|
this._refreshValue();
|
|
},
|
|
|
|
_createHandles: function() {
|
|
var i, handleCount,
|
|
options = this.options,
|
|
existingHandles = this.element.find( ".ui-slider-handle" ).addClass( "ui-state-default ui-corner-all" ),
|
|
handle = "<span class='ui-slider-handle ui-state-default ui-corner-all' tabindex='0'></span>",
|
|
handles = [];
|
|
|
|
handleCount = ( options.values && options.values.length ) || 1;
|
|
|
|
if ( existingHandles.length > handleCount ) {
|
|
existingHandles.slice( handleCount ).remove();
|
|
existingHandles = existingHandles.slice( 0, handleCount );
|
|
}
|
|
|
|
for ( i = existingHandles.length; i < handleCount; i++ ) {
|
|
handles.push( handle );
|
|
}
|
|
|
|
this.handles = existingHandles.add( $( handles.join( "" ) ).appendTo( this.element ) );
|
|
|
|
this.handle = this.handles.eq( 0 );
|
|
|
|
this.handles.each(function( i ) {
|
|
$( this ).data( "ui-slider-handle-index", i );
|
|
});
|
|
},
|
|
|
|
_createRange: function() {
|
|
var options = this.options,
|
|
classes = "";
|
|
|
|
if ( options.range ) {
|
|
if ( options.range === true ) {
|
|
if ( !options.values ) {
|
|
options.values = [ this._valueMin(), this._valueMin() ];
|
|
} else if ( options.values.length && options.values.length !== 2 ) {
|
|
options.values = [ options.values[0], options.values[0] ];
|
|
} else if ( $.isArray( options.values ) ) {
|
|
options.values = options.values.slice(0);
|
|
}
|
|
}
|
|
|
|
if ( !this.range || !this.range.length ) {
|
|
this.range = $( "<div></div>" )
|
|
.appendTo( this.element );
|
|
|
|
classes = "ui-slider-range" +
|
|
// note: this isn't the most fittingly semantic framework class for this element,
|
|
// but worked best visually with a variety of themes
|
|
" ui-widget-header ui-corner-all";
|
|
} else {
|
|
this.range.removeClass( "ui-slider-range-min ui-slider-range-max" )
|
|
// Handle range switching from true to min/max
|
|
.css({
|
|
"left": "",
|
|
"bottom": ""
|
|
});
|
|
}
|
|
|
|
this.range.addClass( classes +
|
|
( ( options.range === "min" || options.range === "max" ) ? " ui-slider-range-" + options.range : "" ) );
|
|
} else {
|
|
if ( this.range ) {
|
|
this.range.remove();
|
|
}
|
|
this.range = null;
|
|
}
|
|
},
|
|
|
|
_setupEvents: function() {
|
|
this._off( this.handles );
|
|
this._on( this.handles, this._handleEvents );
|
|
this._hoverable( this.handles );
|
|
this._focusable( this.handles );
|
|
},
|
|
|
|
_destroy: function() {
|
|
this.handles.remove();
|
|
if ( this.range ) {
|
|
this.range.remove();
|
|
}
|
|
|
|
this.element
|
|
.removeClass( "ui-slider" +
|
|
" ui-slider-horizontal" +
|
|
" ui-slider-vertical" +
|
|
" ui-widget" +
|
|
" ui-widget-content" +
|
|
" ui-corner-all" );
|
|
|
|
this._mouseDestroy();
|
|
},
|
|
|
|
_mouseCapture: function( event ) {
|
|
var position, normValue, distance, closestHandle, index, allowed, offset, mouseOverHandle,
|
|
that = this,
|
|
o = this.options;
|
|
|
|
if ( o.disabled ) {
|
|
return false;
|
|
}
|
|
|
|
this.elementSize = {
|
|
width: this.element.outerWidth(),
|
|
height: this.element.outerHeight()
|
|
};
|
|
this.elementOffset = this.element.offset();
|
|
|
|
position = { x: event.pageX, y: event.pageY };
|
|
normValue = this._normValueFromMouse( position );
|
|
distance = this._valueMax() - this._valueMin() + 1;
|
|
this.handles.each(function( i ) {
|
|
var thisDistance = Math.abs( normValue - that.values(i) );
|
|
if (( distance > thisDistance ) ||
|
|
( distance === thisDistance &&
|
|
(i === that._lastChangedValue || that.values(i) === o.min ))) {
|
|
distance = thisDistance;
|
|
closestHandle = $( this );
|
|
index = i;
|
|
}
|
|
});
|
|
|
|
allowed = this._start( event, index );
|
|
if ( allowed === false ) {
|
|
return false;
|
|
}
|
|
this._mouseSliding = true;
|
|
|
|
this._handleIndex = index;
|
|
|
|
closestHandle
|
|
.addClass( "ui-state-active" )
|
|
.focus();
|
|
|
|
offset = closestHandle.offset();
|
|
mouseOverHandle = !$( event.target ).parents().addBack().is( ".ui-slider-handle" );
|
|
this._clickOffset = mouseOverHandle ? { left: 0, top: 0 } : {
|
|
left: event.pageX - offset.left - ( closestHandle.width() / 2 ),
|
|
top: event.pageY - offset.top -
|
|
( closestHandle.height() / 2 ) -
|
|
( parseInt( closestHandle.css("borderTopWidth"), 10 ) || 0 ) -
|
|
( parseInt( closestHandle.css("borderBottomWidth"), 10 ) || 0) +
|
|
( parseInt( closestHandle.css("marginTop"), 10 ) || 0)
|
|
};
|
|
|
|
if ( !this.handles.hasClass( "ui-state-hover" ) ) {
|
|
this._slide( event, index, normValue );
|
|
}
|
|
this._animateOff = true;
|
|
return true;
|
|
},
|
|
|
|
_mouseStart: function() {
|
|
return true;
|
|
},
|
|
|
|
_mouseDrag: function( event ) {
|
|
var position = { x: event.pageX, y: event.pageY },
|
|
normValue = this._normValueFromMouse( position );
|
|
|
|
this._slide( event, this._handleIndex, normValue );
|
|
|
|
return false;
|
|
},
|
|
|
|
_mouseStop: function( event ) {
|
|
this.handles.removeClass( "ui-state-active" );
|
|
this._mouseSliding = false;
|
|
|
|
this._stop( event, this._handleIndex );
|
|
this._change( event, this._handleIndex );
|
|
|
|
this._handleIndex = null;
|
|
this._clickOffset = null;
|
|
this._animateOff = false;
|
|
|
|
return false;
|
|
},
|
|
|
|
_detectOrientation: function() {
|
|
this.orientation = ( this.options.orientation === "vertical" ) ? "vertical" : "horizontal";
|
|
},
|
|
|
|
_normValueFromMouse: function( position ) {
|
|
var pixelTotal,
|
|
pixelMouse,
|
|
percentMouse,
|
|
valueTotal,
|
|
valueMouse;
|
|
|
|
if ( this.orientation === "horizontal" ) {
|
|
pixelTotal = this.elementSize.width;
|
|
pixelMouse = position.x - this.elementOffset.left - ( this._clickOffset ? this._clickOffset.left : 0 );
|
|
} else {
|
|
pixelTotal = this.elementSize.height;
|
|
pixelMouse = position.y - this.elementOffset.top - ( this._clickOffset ? this._clickOffset.top : 0 );
|
|
}
|
|
|
|
percentMouse = ( pixelMouse / pixelTotal );
|
|
if ( percentMouse > 1 ) {
|
|
percentMouse = 1;
|
|
}
|
|
if ( percentMouse < 0 ) {
|
|
percentMouse = 0;
|
|
}
|
|
if ( this.orientation === "vertical" ) {
|
|
percentMouse = 1 - percentMouse;
|
|
}
|
|
|
|
valueTotal = this._valueMax() - this._valueMin();
|
|
valueMouse = this._valueMin() + percentMouse * valueTotal;
|
|
|
|
return this._trimAlignValue( valueMouse );
|
|
},
|
|
|
|
_start: function( event, index ) {
|
|
var uiHash = {
|
|
handle: this.handles[ index ],
|
|
value: this.value()
|
|
};
|
|
if ( this.options.values && this.options.values.length ) {
|
|
uiHash.value = this.values( index );
|
|
uiHash.values = this.values();
|
|
}
|
|
return this._trigger( "start", event, uiHash );
|
|
},
|
|
|
|
_slide: function( event, index, newVal ) {
|
|
var otherVal,
|
|
newValues,
|
|
allowed;
|
|
|
|
if ( this.options.values && this.options.values.length ) {
|
|
otherVal = this.values( index ? 0 : 1 );
|
|
|
|
if ( ( this.options.values.length === 2 && this.options.range === true ) &&
|
|
( ( index === 0 && newVal > otherVal) || ( index === 1 && newVal < otherVal ) )
|
|
) {
|
|
newVal = otherVal;
|
|
}
|
|
|
|
if ( newVal !== this.values( index ) ) {
|
|
newValues = this.values();
|
|
newValues[ index ] = newVal;
|
|
// A slide can be canceled by returning false from the slide callback
|
|
allowed = this._trigger( "slide", event, {
|
|
handle: this.handles[ index ],
|
|
value: newVal,
|
|
values: newValues
|
|
} );
|
|
otherVal = this.values( index ? 0 : 1 );
|
|
if ( allowed !== false ) {
|
|
this.values( index, newVal );
|
|
}
|
|
}
|
|
} else {
|
|
if ( newVal !== this.value() ) {
|
|
// A slide can be canceled by returning false from the slide callback
|
|
allowed = this._trigger( "slide", event, {
|
|
handle: this.handles[ index ],
|
|
value: newVal
|
|
} );
|
|
if ( allowed !== false ) {
|
|
this.value( newVal );
|
|
}
|
|
}
|
|
}
|
|
},
|
|
|
|
_stop: function( event, index ) {
|
|
var uiHash = {
|
|
handle: this.handles[ index ],
|
|
value: this.value()
|
|
};
|
|
if ( this.options.values && this.options.values.length ) {
|
|
uiHash.value = this.values( index );
|
|
uiHash.values = this.values();
|
|
}
|
|
|
|
this._trigger( "stop", event, uiHash );
|
|
},
|
|
|
|
_change: function( event, index ) {
|
|
if ( !this._keySliding && !this._mouseSliding ) {
|
|
var uiHash = {
|
|
handle: this.handles[ index ],
|
|
value: this.value()
|
|
};
|
|
if ( this.options.values && this.options.values.length ) {
|
|
uiHash.value = this.values( index );
|
|
uiHash.values = this.values();
|
|
}
|
|
|
|
//store the last changed value index for reference when handles overlap
|
|
this._lastChangedValue = index;
|
|
|
|
this._trigger( "change", event, uiHash );
|
|
}
|
|
},
|
|
|
|
value: function( newValue ) {
|
|
if ( arguments.length ) {
|
|
this.options.value = this._trimAlignValue( newValue );
|
|
this._refreshValue();
|
|
this._change( null, 0 );
|
|
return;
|
|
}
|
|
|
|
return this._value();
|
|
},
|
|
|
|
values: function( index, newValue ) {
|
|
var vals,
|
|
newValues,
|
|
i;
|
|
|
|
if ( arguments.length > 1 ) {
|
|
this.options.values[ index ] = this._trimAlignValue( newValue );
|
|
this._refreshValue();
|
|
this._change( null, index );
|
|
return;
|
|
}
|
|
|
|
if ( arguments.length ) {
|
|
if ( $.isArray( arguments[ 0 ] ) ) {
|
|
vals = this.options.values;
|
|
newValues = arguments[ 0 ];
|
|
for ( i = 0; i < vals.length; i += 1 ) {
|
|
vals[ i ] = this._trimAlignValue( newValues[ i ] );
|
|
this._change( null, i );
|
|
}
|
|
this._refreshValue();
|
|
} else {
|
|
if ( this.options.values && this.options.values.length ) {
|
|
return this._values( index );
|
|
} else {
|
|
return this.value();
|
|
}
|
|
}
|
|
} else {
|
|
return this._values();
|
|
}
|
|
},
|
|
|
|
_setOption: function( key, value ) {
|
|
var i,
|
|
valsLength = 0;
|
|
|
|
if ( key === "range" && this.options.range === true ) {
|
|
if ( value === "min" ) {
|
|
this.options.value = this._values( 0 );
|
|
this.options.values = null;
|
|
} else if ( value === "max" ) {
|
|
this.options.value = this._values( this.options.values.length - 1 );
|
|
this.options.values = null;
|
|
}
|
|
}
|
|
|
|
if ( $.isArray( this.options.values ) ) {
|
|
valsLength = this.options.values.length;
|
|
}
|
|
|
|
if ( key === "disabled" ) {
|
|
this.element.toggleClass( "ui-state-disabled", !!value );
|
|
}
|
|
|
|
this._super( key, value );
|
|
|
|
switch ( key ) {
|
|
case "orientation":
|
|
this._detectOrientation();
|
|
this.element
|
|
.removeClass( "ui-slider-horizontal ui-slider-vertical" )
|
|
.addClass( "ui-slider-" + this.orientation );
|
|
this._refreshValue();
|
|
|
|
// Reset positioning from previous orientation
|
|
this.handles.css( value === "horizontal" ? "bottom" : "left", "" );
|
|
break;
|
|
case "value":
|
|
this._animateOff = true;
|
|
this._refreshValue();
|
|
this._change( null, 0 );
|
|
this._animateOff = false;
|
|
break;
|
|
case "values":
|
|
this._animateOff = true;
|
|
this._refreshValue();
|
|
for ( i = 0; i < valsLength; i += 1 ) {
|
|
this._change( null, i );
|
|
}
|
|
this._animateOff = false;
|
|
break;
|
|
case "step":
|
|
case "min":
|
|
case "max":
|
|
this._animateOff = true;
|
|
this._calculateNewMax();
|
|
this._refreshValue();
|
|
this._animateOff = false;
|
|
break;
|
|
case "range":
|
|
this._animateOff = true;
|
|
this._refresh();
|
|
this._animateOff = false;
|
|
break;
|
|
}
|
|
},
|
|
|
|
//internal value getter
|
|
// _value() returns value trimmed by min and max, aligned by step
|
|
_value: function() {
|
|
var val = this.options.value;
|
|
val = this._trimAlignValue( val );
|
|
|
|
return val;
|
|
},
|
|
|
|
//internal values getter
|
|
// _values() returns array of values trimmed by min and max, aligned by step
|
|
// _values( index ) returns single value trimmed by min and max, aligned by step
|
|
_values: function( index ) {
|
|
var val,
|
|
vals,
|
|
i;
|
|
|
|
if ( arguments.length ) {
|
|
val = this.options.values[ index ];
|
|
val = this._trimAlignValue( val );
|
|
|
|
return val;
|
|
} else if ( this.options.values && this.options.values.length ) {
|
|
// .slice() creates a copy of the array
|
|
// this copy gets trimmed by min and max and then returned
|
|
vals = this.options.values.slice();
|
|
for ( i = 0; i < vals.length; i += 1) {
|
|
vals[ i ] = this._trimAlignValue( vals[ i ] );
|
|
}
|
|
|
|
return vals;
|
|
} else {
|
|
return [];
|
|
}
|
|
},
|
|
|
|
// returns the step-aligned value that val is closest to, between (inclusive) min and max
|
|
_trimAlignValue: function( val ) {
|
|
if ( val <= this._valueMin() ) {
|
|
return this._valueMin();
|
|
}
|
|
if ( val >= this._valueMax() ) {
|
|
return this._valueMax();
|
|
}
|
|
var step = ( this.options.step > 0 ) ? this.options.step : 1,
|
|
valModStep = (val - this._valueMin()) % step,
|
|
alignValue = val - valModStep;
|
|
|
|
if ( Math.abs(valModStep) * 2 >= step ) {
|
|
alignValue += ( valModStep > 0 ) ? step : ( -step );
|
|
}
|
|
|
|
// Since JavaScript has problems with large floats, round
|
|
// the final value to 5 digits after the decimal point (see #4124)
|
|
return parseFloat( alignValue.toFixed(5) );
|
|
},
|
|
|
|
_calculateNewMax: function() {
|
|
var max = this.options.max,
|
|
min = this._valueMin(),
|
|
step = this.options.step,
|
|
aboveMin = Math.floor( ( +( max - min ).toFixed( this._precision() ) ) / step ) * step;
|
|
max = aboveMin + min;
|
|
this.max = parseFloat( max.toFixed( this._precision() ) );
|
|
},
|
|
|
|
_precision: function() {
|
|
var precision = this._precisionOf( this.options.step );
|
|
if ( this.options.min !== null ) {
|
|
precision = Math.max( precision, this._precisionOf( this.options.min ) );
|
|
}
|
|
return precision;
|
|
},
|
|
|
|
_precisionOf: function( num ) {
|
|
var str = num.toString(),
|
|
decimal = str.indexOf( "." );
|
|
return decimal === -1 ? 0 : str.length - decimal - 1;
|
|
},
|
|
|
|
_valueMin: function() {
|
|
return this.options.min;
|
|
},
|
|
|
|
_valueMax: function() {
|
|
return this.max;
|
|
},
|
|
|
|
_refreshValue: function() {
|
|
var lastValPercent, valPercent, value, valueMin, valueMax,
|
|
oRange = this.options.range,
|
|
o = this.options,
|
|
that = this,
|
|
animate = ( !this._animateOff ) ? o.animate : false,
|
|
_set = {};
|
|
|
|
if ( this.options.values && this.options.values.length ) {
|
|
this.handles.each(function( i ) {
|
|
valPercent = ( that.values(i) - that._valueMin() ) / ( that._valueMax() - that._valueMin() ) * 100;
|
|
_set[ that.orientation === "horizontal" ? "left" : "bottom" ] = valPercent + "%";
|
|
$( this ).stop( 1, 1 )[ animate ? "animate" : "css" ]( _set, o.animate );
|
|
if ( that.options.range === true ) {
|
|
if ( that.orientation === "horizontal" ) {
|
|
if ( i === 0 ) {
|
|
that.range.stop( 1, 1 )[ animate ? "animate" : "css" ]( { left: valPercent + "%" }, o.animate );
|
|
}
|
|
if ( i === 1 ) {
|
|
that.range[ animate ? "animate" : "css" ]( { width: ( valPercent - lastValPercent ) + "%" }, { queue: false, duration: o.animate } );
|
|
}
|
|
} else {
|
|
if ( i === 0 ) {
|
|
that.range.stop( 1, 1 )[ animate ? "animate" : "css" ]( { bottom: ( valPercent ) + "%" }, o.animate );
|
|
}
|
|
if ( i === 1 ) {
|
|
that.range[ animate ? "animate" : "css" ]( { height: ( valPercent - lastValPercent ) + "%" }, { queue: false, duration: o.animate } );
|
|
}
|
|
}
|
|
}
|
|
lastValPercent = valPercent;
|
|
});
|
|
} else {
|
|
value = this.value();
|
|
valueMin = this._valueMin();
|
|
valueMax = this._valueMax();
|
|
valPercent = ( valueMax !== valueMin ) ?
|
|
( value - valueMin ) / ( valueMax - valueMin ) * 100 :
|
|
0;
|
|
_set[ this.orientation === "horizontal" ? "left" : "bottom" ] = valPercent + "%";
|
|
this.handle.stop( 1, 1 )[ animate ? "animate" : "css" ]( _set, o.animate );
|
|
|
|
if ( oRange === "min" && this.orientation === "horizontal" ) {
|
|
this.range.stop( 1, 1 )[ animate ? "animate" : "css" ]( { width: valPercent + "%" }, o.animate );
|
|
}
|
|
if ( oRange === "max" && this.orientation === "horizontal" ) {
|
|
this.range[ animate ? "animate" : "css" ]( { width: ( 100 - valPercent ) + "%" }, { queue: false, duration: o.animate } );
|
|
}
|
|
if ( oRange === "min" && this.orientation === "vertical" ) {
|
|
this.range.stop( 1, 1 )[ animate ? "animate" : "css" ]( { height: valPercent + "%" }, o.animate );
|
|
}
|
|
if ( oRange === "max" && this.orientation === "vertical" ) {
|
|
this.range[ animate ? "animate" : "css" ]( { height: ( 100 - valPercent ) + "%" }, { queue: false, duration: o.animate } );
|
|
}
|
|
}
|
|
},
|
|
|
|
_handleEvents: {
|
|
keydown: function( event ) {
|
|
var allowed, curVal, newVal, step,
|
|
index = $( event.target ).data( "ui-slider-handle-index" );
|
|
|
|
switch ( event.keyCode ) {
|
|
case $.ui.keyCode.HOME:
|
|
case $.ui.keyCode.END:
|
|
case $.ui.keyCode.PAGE_UP:
|
|
case $.ui.keyCode.PAGE_DOWN:
|
|
case $.ui.keyCode.UP:
|
|
case $.ui.keyCode.RIGHT:
|
|
case $.ui.keyCode.DOWN:
|
|
case $.ui.keyCode.LEFT:
|
|
event.preventDefault();
|
|
if ( !this._keySliding ) {
|
|
this._keySliding = true;
|
|
$( event.target ).addClass( "ui-state-active" );
|
|
allowed = this._start( event, index );
|
|
if ( allowed === false ) {
|
|
return;
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
|
|
step = this.options.step;
|
|
if ( this.options.values && this.options.values.length ) {
|
|
curVal = newVal = this.values( index );
|
|
} else {
|
|
curVal = newVal = this.value();
|
|
}
|
|
|
|
switch ( event.keyCode ) {
|
|
case $.ui.keyCode.HOME:
|
|
newVal = this._valueMin();
|
|
break;
|
|
case $.ui.keyCode.END:
|
|
newVal = this._valueMax();
|
|
break;
|
|
case $.ui.keyCode.PAGE_UP:
|
|
newVal = this._trimAlignValue(
|
|
curVal + ( ( this._valueMax() - this._valueMin() ) / this.numPages )
|
|
);
|
|
break;
|
|
case $.ui.keyCode.PAGE_DOWN:
|
|
newVal = this._trimAlignValue(
|
|
curVal - ( (this._valueMax() - this._valueMin()) / this.numPages ) );
|
|
break;
|
|
case $.ui.keyCode.UP:
|
|
case $.ui.keyCode.RIGHT:
|
|
if ( curVal === this._valueMax() ) {
|
|
return;
|
|
}
|
|
newVal = this._trimAlignValue( curVal + step );
|
|
break;
|
|
case $.ui.keyCode.DOWN:
|
|
case $.ui.keyCode.LEFT:
|
|
if ( curVal === this._valueMin() ) {
|
|
return;
|
|
}
|
|
newVal = this._trimAlignValue( curVal - step );
|
|
break;
|
|
}
|
|
|
|
this._slide( event, index, newVal );
|
|
},
|
|
keyup: function( event ) {
|
|
var index = $( event.target ).data( "ui-slider-handle-index" );
|
|
|
|
if ( this._keySliding ) {
|
|
this._keySliding = false;
|
|
this._stop( event, index );
|
|
this._change( event, index );
|
|
$( event.target ).removeClass( "ui-state-active" );
|
|
}
|
|
}
|
|
}
|
|
});
|
|
|
|
|
|
/*!
|
|
* jQuery UI Sortable 1.11.4
|
|
* http://jqueryui.com
|
|
*
|
|
* Copyright jQuery Foundation and other contributors
|
|
* Released under the MIT license.
|
|
* http://jquery.org/license
|
|
*
|
|
* http://api.jqueryui.com/sortable/
|
|
*/
|
|
|
|
|
|
var sortable = $.widget("ui.sortable", $.ui.mouse, {
|
|
version: "1.11.4",
|
|
widgetEventPrefix: "sort",
|
|
ready: false,
|
|
options: {
|
|
appendTo: "parent",
|
|
axis: false,
|
|
connectWith: false,
|
|
containment: false,
|
|
cursor: "auto",
|
|
cursorAt: false,
|
|
dropOnEmpty: true,
|
|
forcePlaceholderSize: false,
|
|
forceHelperSize: false,
|
|
grid: false,
|
|
handle: false,
|
|
helper: "original",
|
|
items: "> *",
|
|
opacity: false,
|
|
placeholder: false,
|
|
revert: false,
|
|
scroll: true,
|
|
scrollSensitivity: 20,
|
|
scrollSpeed: 20,
|
|
scope: "default",
|
|
tolerance: "intersect",
|
|
zIndex: 1000,
|
|
|
|
// callbacks
|
|
activate: null,
|
|
beforeStop: null,
|
|
change: null,
|
|
deactivate: null,
|
|
out: null,
|
|
over: null,
|
|
receive: null,
|
|
remove: null,
|
|
sort: null,
|
|
start: null,
|
|
stop: null,
|
|
update: null
|
|
},
|
|
|
|
_isOverAxis: function( x, reference, size ) {
|
|
return ( x >= reference ) && ( x < ( reference + size ) );
|
|
},
|
|
|
|
_isFloating: function( item ) {
|
|
return (/left|right/).test(item.css("float")) || (/inline|table-cell/).test(item.css("display"));
|
|
},
|
|
|
|
_create: function() {
|
|
this.containerCache = {};
|
|
this.element.addClass("ui-sortable");
|
|
|
|
//Get the items
|
|
this.refresh();
|
|
|
|
//Let's determine the parent's offset
|
|
this.offset = this.element.offset();
|
|
|
|
//Initialize mouse events for interaction
|
|
this._mouseInit();
|
|
|
|
this._setHandleClassName();
|
|
|
|
//We're ready to go
|
|
this.ready = true;
|
|
|
|
},
|
|
|
|
_setOption: function( key, value ) {
|
|
this._super( key, value );
|
|
|
|
if ( key === "handle" ) {
|
|
this._setHandleClassName();
|
|
}
|
|
},
|
|
|
|
_setHandleClassName: function() {
|
|
this.element.find( ".ui-sortable-handle" ).removeClass( "ui-sortable-handle" );
|
|
$.each( this.items, function() {
|
|
( this.instance.options.handle ?
|
|
this.item.find( this.instance.options.handle ) : this.item )
|
|
.addClass( "ui-sortable-handle" );
|
|
});
|
|
},
|
|
|
|
_destroy: function() {
|
|
this.element
|
|
.removeClass( "ui-sortable ui-sortable-disabled" )
|
|
.find( ".ui-sortable-handle" )
|
|
.removeClass( "ui-sortable-handle" );
|
|
this._mouseDestroy();
|
|
|
|
for ( var i = this.items.length - 1; i >= 0; i-- ) {
|
|
this.items[i].item.removeData(this.widgetName + "-item");
|
|
}
|
|
|
|
return this;
|
|
},
|
|
|
|
_mouseCapture: function(event, overrideHandle) {
|
|
var currentItem = null,
|
|
validHandle = false,
|
|
that = this;
|
|
|
|
if (this.reverting) {
|
|
return false;
|
|
}
|
|
|
|
if(this.options.disabled || this.options.type === "static") {
|
|
return false;
|
|
}
|
|
|
|
//We have to refresh the items data once first
|
|
this._refreshItems(event);
|
|
|
|
//Find out if the clicked node (or one of its parents) is a actual item in this.items
|
|
$(event.target).parents().each(function() {
|
|
if($.data(this, that.widgetName + "-item") === that) {
|
|
currentItem = $(this);
|
|
return false;
|
|
}
|
|
});
|
|
if($.data(event.target, that.widgetName + "-item") === that) {
|
|
currentItem = $(event.target);
|
|
}
|
|
|
|
if(!currentItem) {
|
|
return false;
|
|
}
|
|
if(this.options.handle && !overrideHandle) {
|
|
$(this.options.handle, currentItem).find("*").addBack().each(function() {
|
|
if(this === event.target) {
|
|
validHandle = true;
|
|
}
|
|
});
|
|
if(!validHandle) {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
this.currentItem = currentItem;
|
|
this._removeCurrentsFromItems();
|
|
return true;
|
|
|
|
},
|
|
|
|
_mouseStart: function(event, overrideHandle, noActivation) {
|
|
|
|
var i, body,
|
|
o = this.options;
|
|
|
|
this.currentContainer = this;
|
|
|
|
//We only need to call refreshPositions, because the refreshItems call has been moved to mouseCapture
|
|
this.refreshPositions();
|
|
|
|
//Create and append the visible helper
|
|
this.helper = this._createHelper(event);
|
|
|
|
//Cache the helper size
|
|
this._cacheHelperProportions();
|
|
|
|
/*
|
|
* - Position generation -
|
|
* This block generates everything position related - it's the core of draggables.
|
|
*/
|
|
|
|
//Cache the margins of the original element
|
|
this._cacheMargins();
|
|
|
|
//Get the next scrolling parent
|
|
this.scrollParent = this.helper.scrollParent();
|
|
|
|
//The element's absolute position on the page minus margins
|
|
this.offset = this.currentItem.offset();
|
|
this.offset = {
|
|
top: this.offset.top - this.margins.top,
|
|
left: this.offset.left - this.margins.left
|
|
};
|
|
|
|
$.extend(this.offset, {
|
|
click: { //Where the click happened, relative to the element
|
|
left: event.pageX - this.offset.left,
|
|
top: event.pageY - this.offset.top
|
|
},
|
|
parent: this._getParentOffset(),
|
|
relative: this._getRelativeOffset() //This is a relative to absolute position minus the actual position calculation - only used for relative positioned helper
|
|
});
|
|
|
|
// Only after we got the offset, we can change the helper's position to absolute
|
|
// TODO: Still need to figure out a way to make relative sorting possible
|
|
this.helper.css("position", "absolute");
|
|
this.cssPosition = this.helper.css("position");
|
|
|
|
//Generate the original position
|
|
this.originalPosition = this._generatePosition(event);
|
|
this.originalPageX = event.pageX;
|
|
this.originalPageY = event.pageY;
|
|
|
|
//Adjust the mouse offset relative to the helper if "cursorAt" is supplied
|
|
(o.cursorAt && this._adjustOffsetFromHelper(o.cursorAt));
|
|
|
|
//Cache the former DOM position
|
|
this.domPosition = { prev: this.currentItem.prev()[0], parent: this.currentItem.parent()[0] };
|
|
|
|
//If the helper is not the original, hide the original so it's not playing any role during the drag, won't cause anything bad this way
|
|
if(this.helper[0] !== this.currentItem[0]) {
|
|
this.currentItem.hide();
|
|
}
|
|
|
|
//Create the placeholder
|
|
this._createPlaceholder();
|
|
|
|
//Set a containment if given in the options
|
|
if(o.containment) {
|
|
this._setContainment();
|
|
}
|
|
|
|
if( o.cursor && o.cursor !== "auto" ) { // cursor option
|
|
body = this.document.find( "body" );
|
|
|
|
// support: IE
|
|
this.storedCursor = body.css( "cursor" );
|
|
body.css( "cursor", o.cursor );
|
|
|
|
this.storedStylesheet = $( "<style>*{ cursor: "+o.cursor+" !important; }</style>" ).appendTo( body );
|
|
}
|
|
|
|
if(o.opacity) { // opacity option
|
|
if (this.helper.css("opacity")) {
|
|
this._storedOpacity = this.helper.css("opacity");
|
|
}
|
|
this.helper.css("opacity", o.opacity);
|
|
}
|
|
|
|
if(o.zIndex) { // zIndex option
|
|
if (this.helper.css("zIndex")) {
|
|
this._storedZIndex = this.helper.css("zIndex");
|
|
}
|
|
this.helper.css("zIndex", o.zIndex);
|
|
}
|
|
|
|
//Prepare scrolling
|
|
if(this.scrollParent[0] !== this.document[0] && this.scrollParent[0].tagName !== "HTML") {
|
|
this.overflowOffset = this.scrollParent.offset();
|
|
}
|
|
|
|
//Call callbacks
|
|
this._trigger("start", event, this._uiHash());
|
|
|
|
//Recache the helper size
|
|
if(!this._preserveHelperProportions) {
|
|
this._cacheHelperProportions();
|
|
}
|
|
|
|
|
|
//Post "activate" events to possible containers
|
|
if( !noActivation ) {
|
|
for ( i = this.containers.length - 1; i >= 0; i-- ) {
|
|
this.containers[ i ]._trigger( "activate", event, this._uiHash( this ) );
|
|
}
|
|
}
|
|
|
|
//Prepare possible droppables
|
|
if($.ui.ddmanager) {
|
|
$.ui.ddmanager.current = this;
|
|
}
|
|
|
|
if ($.ui.ddmanager && !o.dropBehaviour) {
|
|
$.ui.ddmanager.prepareOffsets(this, event);
|
|
}
|
|
|
|
this.dragging = true;
|
|
|
|
this.helper.addClass("ui-sortable-helper");
|
|
this._mouseDrag(event); //Execute the drag once - this causes the helper not to be visible before getting its correct position
|
|
return true;
|
|
|
|
},
|
|
|
|
_mouseDrag: function(event) {
|
|
var i, item, itemElement, intersection,
|
|
o = this.options,
|
|
scrolled = false;
|
|
|
|
//Compute the helpers position
|
|
this.position = this._generatePosition(event);
|
|
this.positionAbs = this._convertPositionTo("absolute");
|
|
|
|
if (!this.lastPositionAbs) {
|
|
this.lastPositionAbs = this.positionAbs;
|
|
}
|
|
|
|
//Do scrolling
|
|
if(this.options.scroll) {
|
|
if(this.scrollParent[0] !== this.document[0] && this.scrollParent[0].tagName !== "HTML") {
|
|
|
|
if((this.overflowOffset.top + this.scrollParent[0].offsetHeight) - event.pageY < o.scrollSensitivity) {
|
|
this.scrollParent[0].scrollTop = scrolled = this.scrollParent[0].scrollTop + o.scrollSpeed;
|
|
} else if(event.pageY - this.overflowOffset.top < o.scrollSensitivity) {
|
|
this.scrollParent[0].scrollTop = scrolled = this.scrollParent[0].scrollTop - o.scrollSpeed;
|
|
}
|
|
|
|
if((this.overflowOffset.left + this.scrollParent[0].offsetWidth) - event.pageX < o.scrollSensitivity) {
|
|
this.scrollParent[0].scrollLeft = scrolled = this.scrollParent[0].scrollLeft + o.scrollSpeed;
|
|
} else if(event.pageX - this.overflowOffset.left < o.scrollSensitivity) {
|
|
this.scrollParent[0].scrollLeft = scrolled = this.scrollParent[0].scrollLeft - o.scrollSpeed;
|
|
}
|
|
|
|
} else {
|
|
|
|
if(event.pageY - this.document.scrollTop() < o.scrollSensitivity) {
|
|
scrolled = this.document.scrollTop(this.document.scrollTop() - o.scrollSpeed);
|
|
} else if(this.window.height() - (event.pageY - this.document.scrollTop()) < o.scrollSensitivity) {
|
|
scrolled = this.document.scrollTop(this.document.scrollTop() + o.scrollSpeed);
|
|
}
|
|
|
|
if(event.pageX - this.document.scrollLeft() < o.scrollSensitivity) {
|
|
scrolled = this.document.scrollLeft(this.document.scrollLeft() - o.scrollSpeed);
|
|
} else if(this.window.width() - (event.pageX - this.document.scrollLeft()) < o.scrollSensitivity) {
|
|
scrolled = this.document.scrollLeft(this.document.scrollLeft() + o.scrollSpeed);
|
|
}
|
|
|
|
}
|
|
|
|
if(scrolled !== false && $.ui.ddmanager && !o.dropBehaviour) {
|
|
$.ui.ddmanager.prepareOffsets(this, event);
|
|
}
|
|
}
|
|
|
|
//Regenerate the absolute position used for position checks
|
|
this.positionAbs = this._convertPositionTo("absolute");
|
|
|
|
//Set the helper position
|
|
if(!this.options.axis || this.options.axis !== "y") {
|
|
this.helper[0].style.left = this.position.left+"px";
|
|
}
|
|
if(!this.options.axis || this.options.axis !== "x") {
|
|
this.helper[0].style.top = this.position.top+"px";
|
|
}
|
|
|
|
//Rearrange
|
|
for (i = this.items.length - 1; i >= 0; i--) {
|
|
|
|
//Cache variables and intersection, continue if no intersection
|
|
item = this.items[i];
|
|
itemElement = item.item[0];
|
|
intersection = this._intersectsWithPointer(item);
|
|
if (!intersection) {
|
|
continue;
|
|
}
|
|
|
|
// Only put the placeholder inside the current Container, skip all
|
|
// items from other containers. This works because when moving
|
|
// an item from one container to another the
|
|
// currentContainer is switched before the placeholder is moved.
|
|
//
|
|
// Without this, moving items in "sub-sortables" can cause
|
|
// the placeholder to jitter between the outer and inner container.
|
|
if (item.instance !== this.currentContainer) {
|
|
continue;
|
|
}
|
|
|
|
// cannot intersect with itself
|
|
// no useless actions that have been done before
|
|
// no action if the item moved is the parent of the item checked
|
|
if (itemElement !== this.currentItem[0] &&
|
|
this.placeholder[intersection === 1 ? "next" : "prev"]()[0] !== itemElement &&
|
|
!$.contains(this.placeholder[0], itemElement) &&
|
|
(this.options.type === "semi-dynamic" ? !$.contains(this.element[0], itemElement) : true)
|
|
) {
|
|
|
|
this.direction = intersection === 1 ? "down" : "up";
|
|
|
|
if (this.options.tolerance === "pointer" || this._intersectsWithSides(item)) {
|
|
this._rearrange(event, item);
|
|
} else {
|
|
break;
|
|
}
|
|
|
|
this._trigger("change", event, this._uiHash());
|
|
break;
|
|
}
|
|
}
|
|
|
|
//Post events to containers
|
|
this._contactContainers(event);
|
|
|
|
//Interconnect with droppables
|
|
if($.ui.ddmanager) {
|
|
$.ui.ddmanager.drag(this, event);
|
|
}
|
|
|
|
//Call callbacks
|
|
this._trigger("sort", event, this._uiHash());
|
|
|
|
this.lastPositionAbs = this.positionAbs;
|
|
return false;
|
|
|
|
},
|
|
|
|
_mouseStop: function(event, noPropagation) {
|
|
|
|
if(!event) {
|
|
return;
|
|
}
|
|
|
|
//If we are using droppables, inform the manager about the drop
|
|
if ($.ui.ddmanager && !this.options.dropBehaviour) {
|
|
$.ui.ddmanager.drop(this, event);
|
|
}
|
|
|
|
if(this.options.revert) {
|
|
var that = this,
|
|
cur = this.placeholder.offset(),
|
|
axis = this.options.axis,
|
|
animation = {};
|
|
|
|
if ( !axis || axis === "x" ) {
|
|
animation.left = cur.left - this.offset.parent.left - this.margins.left + (this.offsetParent[0] === this.document[0].body ? 0 : this.offsetParent[0].scrollLeft);
|
|
}
|
|
if ( !axis || axis === "y" ) {
|
|
animation.top = cur.top - this.offset.parent.top - this.margins.top + (this.offsetParent[0] === this.document[0].body ? 0 : this.offsetParent[0].scrollTop);
|
|
}
|
|
this.reverting = true;
|
|
$(this.helper).animate( animation, parseInt(this.options.revert, 10) || 500, function() {
|
|
that._clear(event);
|
|
});
|
|
} else {
|
|
this._clear(event, noPropagation);
|
|
}
|
|
|
|
return false;
|
|
|
|
},
|
|
|
|
cancel: function() {
|
|
|
|
if(this.dragging) {
|
|
|
|
this._mouseUp({ target: null });
|
|
|
|
if(this.options.helper === "original") {
|
|
this.currentItem.css(this._storedCSS).removeClass("ui-sortable-helper");
|
|
} else {
|
|
this.currentItem.show();
|
|
}
|
|
|
|
//Post deactivating events to containers
|
|
for (var i = this.containers.length - 1; i >= 0; i--){
|
|
this.containers[i]._trigger("deactivate", null, this._uiHash(this));
|
|
if(this.containers[i].containerCache.over) {
|
|
this.containers[i]._trigger("out", null, this._uiHash(this));
|
|
this.containers[i].containerCache.over = 0;
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
if (this.placeholder) {
|
|
//$(this.placeholder[0]).remove(); would have been the jQuery way - unfortunately, it unbinds ALL events from the original node!
|
|
if(this.placeholder[0].parentNode) {
|
|
this.placeholder[0].parentNode.removeChild(this.placeholder[0]);
|
|
}
|
|
if(this.options.helper !== "original" && this.helper && this.helper[0].parentNode) {
|
|
this.helper.remove();
|
|
}
|
|
|
|
$.extend(this, {
|
|
helper: null,
|
|
dragging: false,
|
|
reverting: false,
|
|
_noFinalSort: null
|
|
});
|
|
|
|
if(this.domPosition.prev) {
|
|
$(this.domPosition.prev).after(this.currentItem);
|
|
} else {
|
|
$(this.domPosition.parent).prepend(this.currentItem);
|
|
}
|
|
}
|
|
|
|
return this;
|
|
|
|
},
|
|
|
|
serialize: function(o) {
|
|
|
|
var items = this._getItemsAsjQuery(o && o.connected),
|
|
str = [];
|
|
o = o || {};
|
|
|
|
$(items).each(function() {
|
|
var res = ($(o.item || this).attr(o.attribute || "id") || "").match(o.expression || (/(.+)[\-=_](.+)/));
|
|
if (res) {
|
|
str.push((o.key || res[1]+"[]")+"="+(o.key && o.expression ? res[1] : res[2]));
|
|
}
|
|
});
|
|
|
|
if(!str.length && o.key) {
|
|
str.push(o.key + "=");
|
|
}
|
|
|
|
return str.join("&");
|
|
|
|
},
|
|
|
|
toArray: function(o) {
|
|
|
|
var items = this._getItemsAsjQuery(o && o.connected),
|
|
ret = [];
|
|
|
|
o = o || {};
|
|
|
|
items.each(function() { ret.push($(o.item || this).attr(o.attribute || "id") || ""); });
|
|
return ret;
|
|
|
|
},
|
|
|
|
/* Be careful with the following core functions */
|
|
_intersectsWith: function(item) {
|
|
|
|
var x1 = this.positionAbs.left,
|
|
x2 = x1 + this.helperProportions.width,
|
|
y1 = this.positionAbs.top,
|
|
y2 = y1 + this.helperProportions.height,
|
|
l = item.left,
|
|
r = l + item.width,
|
|
t = item.top,
|
|
b = t + item.height,
|
|
dyClick = this.offset.click.top,
|
|
dxClick = this.offset.click.left,
|
|
isOverElementHeight = ( this.options.axis === "x" ) || ( ( y1 + dyClick ) > t && ( y1 + dyClick ) < b ),
|
|
isOverElementWidth = ( this.options.axis === "y" ) || ( ( x1 + dxClick ) > l && ( x1 + dxClick ) < r ),
|
|
isOverElement = isOverElementHeight && isOverElementWidth;
|
|
|
|
if ( this.options.tolerance === "pointer" ||
|
|
this.options.forcePointerForContainers ||
|
|
(this.options.tolerance !== "pointer" && this.helperProportions[this.floating ? "width" : "height"] > item[this.floating ? "width" : "height"])
|
|
) {
|
|
return isOverElement;
|
|
} else {
|
|
|
|
return (l < x1 + (this.helperProportions.width / 2) && // Right Half
|
|
x2 - (this.helperProportions.width / 2) < r && // Left Half
|
|
t < y1 + (this.helperProportions.height / 2) && // Bottom Half
|
|
y2 - (this.helperProportions.height / 2) < b ); // Top Half
|
|
|
|
}
|
|
},
|
|
|
|
_intersectsWithPointer: function(item) {
|
|
|
|
var isOverElementHeight = (this.options.axis === "x") || this._isOverAxis(this.positionAbs.top + this.offset.click.top, item.top, item.height),
|
|
isOverElementWidth = (this.options.axis === "y") || this._isOverAxis(this.positionAbs.left + this.offset.click.left, item.left, item.width),
|
|
isOverElement = isOverElementHeight && isOverElementWidth,
|
|
verticalDirection = this._getDragVerticalDirection(),
|
|
horizontalDirection = this._getDragHorizontalDirection();
|
|
|
|
if (!isOverElement) {
|
|
return false;
|
|
}
|
|
|
|
return this.floating ?
|
|
( ((horizontalDirection && horizontalDirection === "right") || verticalDirection === "down") ? 2 : 1 )
|
|
: ( verticalDirection && (verticalDirection === "down" ? 2 : 1) );
|
|
|
|
},
|
|
|
|
_intersectsWithSides: function(item) {
|
|
|
|
var isOverBottomHalf = this._isOverAxis(this.positionAbs.top + this.offset.click.top, item.top + (item.height/2), item.height),
|
|
isOverRightHalf = this._isOverAxis(this.positionAbs.left + this.offset.click.left, item.left + (item.width/2), item.width),
|
|
verticalDirection = this._getDragVerticalDirection(),
|
|
horizontalDirection = this._getDragHorizontalDirection();
|
|
|
|
if (this.floating && horizontalDirection) {
|
|
return ((horizontalDirection === "right" && isOverRightHalf) || (horizontalDirection === "left" && !isOverRightHalf));
|
|
} else {
|
|
return verticalDirection && ((verticalDirection === "down" && isOverBottomHalf) || (verticalDirection === "up" && !isOverBottomHalf));
|
|
}
|
|
|
|
},
|
|
|
|
_getDragVerticalDirection: function() {
|
|
var delta = this.positionAbs.top - this.lastPositionAbs.top;
|
|
return delta !== 0 && (delta > 0 ? "down" : "up");
|
|
},
|
|
|
|
_getDragHorizontalDirection: function() {
|
|
var delta = this.positionAbs.left - this.lastPositionAbs.left;
|
|
return delta !== 0 && (delta > 0 ? "right" : "left");
|
|
},
|
|
|
|
refresh: function(event) {
|
|
this._refreshItems(event);
|
|
this._setHandleClassName();
|
|
this.refreshPositions();
|
|
return this;
|
|
},
|
|
|
|
_connectWith: function() {
|
|
var options = this.options;
|
|
return options.connectWith.constructor === String ? [options.connectWith] : options.connectWith;
|
|
},
|
|
|
|
_getItemsAsjQuery: function(connected) {
|
|
|
|
var i, j, cur, inst,
|
|
items = [],
|
|
queries = [],
|
|
connectWith = this._connectWith();
|
|
|
|
if(connectWith && connected) {
|
|
for (i = connectWith.length - 1; i >= 0; i--){
|
|
cur = $(connectWith[i], this.document[0]);
|
|
for ( j = cur.length - 1; j >= 0; j--){
|
|
inst = $.data(cur[j], this.widgetFullName);
|
|
if(inst && inst !== this && !inst.options.disabled) {
|
|
queries.push([$.isFunction(inst.options.items) ? inst.options.items.call(inst.element) : $(inst.options.items, inst.element).not(".ui-sortable-helper").not(".ui-sortable-placeholder"), inst]);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
queries.push([$.isFunction(this.options.items) ? this.options.items.call(this.element, null, { options: this.options, item: this.currentItem }) : $(this.options.items, this.element).not(".ui-sortable-helper").not(".ui-sortable-placeholder"), this]);
|
|
|
|
function addItems() {
|
|
items.push( this );
|
|
}
|
|
for (i = queries.length - 1; i >= 0; i--){
|
|
queries[i][0].each( addItems );
|
|
}
|
|
|
|
return $(items);
|
|
|
|
},
|
|
|
|
_removeCurrentsFromItems: function() {
|
|
|
|
var list = this.currentItem.find(":data(" + this.widgetName + "-item)");
|
|
|
|
this.items = $.grep(this.items, function (item) {
|
|
for (var j=0; j < list.length; j++) {
|
|
if(list[j] === item.item[0]) {
|
|
return false;
|
|
}
|
|
}
|
|
return true;
|
|
});
|
|
|
|
},
|
|
|
|
_refreshItems: function(event) {
|
|
|
|
this.items = [];
|
|
this.containers = [this];
|
|
|
|
var i, j, cur, inst, targetData, _queries, item, queriesLength,
|
|
items = this.items,
|
|
queries = [[$.isFunction(this.options.items) ? this.options.items.call(this.element[0], event, { item: this.currentItem }) : $(this.options.items, this.element), this]],
|
|
connectWith = this._connectWith();
|
|
|
|
if(connectWith && this.ready) { //Shouldn't be run the first time through due to massive slow-down
|
|
for (i = connectWith.length - 1; i >= 0; i--){
|
|
cur = $(connectWith[i], this.document[0]);
|
|
for (j = cur.length - 1; j >= 0; j--){
|
|
inst = $.data(cur[j], this.widgetFullName);
|
|
if(inst && inst !== this && !inst.options.disabled) {
|
|
queries.push([$.isFunction(inst.options.items) ? inst.options.items.call(inst.element[0], event, { item: this.currentItem }) : $(inst.options.items, inst.element), inst]);
|
|
this.containers.push(inst);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
for (i = queries.length - 1; i >= 0; i--) {
|
|
targetData = queries[i][1];
|
|
_queries = queries[i][0];
|
|
|
|
for (j=0, queriesLength = _queries.length; j < queriesLength; j++) {
|
|
item = $(_queries[j]);
|
|
|
|
item.data(this.widgetName + "-item", targetData); // Data for target checking (mouse manager)
|
|
|
|
items.push({
|
|
item: item,
|
|
instance: targetData,
|
|
width: 0, height: 0,
|
|
left: 0, top: 0
|
|
});
|
|
}
|
|
}
|
|
|
|
},
|
|
|
|
refreshPositions: function(fast) {
|
|
|
|
// Determine whether items are being displayed horizontally
|
|
this.floating = this.items.length ?
|
|
this.options.axis === "x" || this._isFloating( this.items[ 0 ].item ) :
|
|
false;
|
|
|
|
//This has to be redone because due to the item being moved out/into the offsetParent, the offsetParent's position will change
|
|
if(this.offsetParent && this.helper) {
|
|
this.offset.parent = this._getParentOffset();
|
|
}
|
|
|
|
var i, item, t, p;
|
|
|
|
for (i = this.items.length - 1; i >= 0; i--){
|
|
item = this.items[i];
|
|
|
|
//We ignore calculating positions of all connected containers when we're not over them
|
|
if(item.instance !== this.currentContainer && this.currentContainer && item.item[0] !== this.currentItem[0]) {
|
|
continue;
|
|
}
|
|
|
|
t = this.options.toleranceElement ? $(this.options.toleranceElement, item.item) : item.item;
|
|
|
|
if (!fast) {
|
|
item.width = t.outerWidth();
|
|
item.height = t.outerHeight();
|
|
}
|
|
|
|
p = t.offset();
|
|
item.left = p.left;
|
|
item.top = p.top;
|
|
}
|
|
|
|
if(this.options.custom && this.options.custom.refreshContainers) {
|
|
this.options.custom.refreshContainers.call(this);
|
|
} else {
|
|
for (i = this.containers.length - 1; i >= 0; i--){
|
|
p = this.containers[i].element.offset();
|
|
this.containers[i].containerCache.left = p.left;
|
|
this.containers[i].containerCache.top = p.top;
|
|
this.containers[i].containerCache.width = this.containers[i].element.outerWidth();
|
|
this.containers[i].containerCache.height = this.containers[i].element.outerHeight();
|
|
}
|
|
}
|
|
|
|
return this;
|
|
},
|
|
|
|
_createPlaceholder: function(that) {
|
|
that = that || this;
|
|
var className,
|
|
o = that.options;
|
|
|
|
if(!o.placeholder || o.placeholder.constructor === String) {
|
|
className = o.placeholder;
|
|
o.placeholder = {
|
|
element: function() {
|
|
|
|
var nodeName = that.currentItem[0].nodeName.toLowerCase(),
|
|
element = $( "<" + nodeName + ">", that.document[0] )
|
|
.addClass(className || that.currentItem[0].className+" ui-sortable-placeholder")
|
|
.removeClass("ui-sortable-helper");
|
|
|
|
if ( nodeName === "tbody" ) {
|
|
that._createTrPlaceholder(
|
|
that.currentItem.find( "tr" ).eq( 0 ),
|
|
$( "<tr>", that.document[ 0 ] ).appendTo( element )
|
|
);
|
|
} else if ( nodeName === "tr" ) {
|
|
that._createTrPlaceholder( that.currentItem, element );
|
|
} else if ( nodeName === "img" ) {
|
|
element.attr( "src", that.currentItem.attr( "src" ) );
|
|
}
|
|
|
|
if ( !className ) {
|
|
element.css( "visibility", "hidden" );
|
|
}
|
|
|
|
return element;
|
|
},
|
|
update: function(container, p) {
|
|
|
|
// 1. If a className is set as 'placeholder option, we don't force sizes - the class is responsible for that
|
|
// 2. The option 'forcePlaceholderSize can be enabled to force it even if a class name is specified
|
|
if(className && !o.forcePlaceholderSize) {
|
|
return;
|
|
}
|
|
|
|
//If the element doesn't have a actual height by itself (without styles coming from a stylesheet), it receives the inline height from the dragged item
|
|
if(!p.height()) { p.height(that.currentItem.innerHeight() - parseInt(that.currentItem.css("paddingTop")||0, 10) - parseInt(that.currentItem.css("paddingBottom")||0, 10)); }
|
|
if(!p.width()) { p.width(that.currentItem.innerWidth() - parseInt(that.currentItem.css("paddingLeft")||0, 10) - parseInt(that.currentItem.css("paddingRight")||0, 10)); }
|
|
}
|
|
};
|
|
}
|
|
|
|
//Create the placeholder
|
|
that.placeholder = $(o.placeholder.element.call(that.element, that.currentItem));
|
|
|
|
//Append it after the actual current item
|
|
that.currentItem.after(that.placeholder);
|
|
|
|
//Update the size of the placeholder (TODO: Logic to fuzzy, see line 316/317)
|
|
o.placeholder.update(that, that.placeholder);
|
|
|
|
},
|
|
|
|
_createTrPlaceholder: function( sourceTr, targetTr ) {
|
|
var that = this;
|
|
|
|
sourceTr.children().each(function() {
|
|
$( "<td> </td>", that.document[ 0 ] )
|
|
.attr( "colspan", $( this ).attr( "colspan" ) || 1 )
|
|
.appendTo( targetTr );
|
|
});
|
|
},
|
|
|
|
_contactContainers: function(event) {
|
|
var i, j, dist, itemWithLeastDistance, posProperty, sizeProperty, cur, nearBottom, floating, axis,
|
|
innermostContainer = null,
|
|
innermostIndex = null;
|
|
|
|
// get innermost container that intersects with item
|
|
for (i = this.containers.length - 1; i >= 0; i--) {
|
|
|
|
// never consider a container that's located within the item itself
|
|
if($.contains(this.currentItem[0], this.containers[i].element[0])) {
|
|
continue;
|
|
}
|
|
|
|
if(this._intersectsWith(this.containers[i].containerCache)) {
|
|
|
|
// if we've already found a container and it's more "inner" than this, then continue
|
|
if(innermostContainer && $.contains(this.containers[i].element[0], innermostContainer.element[0])) {
|
|
continue;
|
|
}
|
|
|
|
innermostContainer = this.containers[i];
|
|
innermostIndex = i;
|
|
|
|
} else {
|
|
// container doesn't intersect. trigger "out" event if necessary
|
|
if(this.containers[i].containerCache.over) {
|
|
this.containers[i]._trigger("out", event, this._uiHash(this));
|
|
this.containers[i].containerCache.over = 0;
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
// if no intersecting containers found, return
|
|
if(!innermostContainer) {
|
|
return;
|
|
}
|
|
|
|
// move the item into the container if it's not there already
|
|
if(this.containers.length === 1) {
|
|
if (!this.containers[innermostIndex].containerCache.over) {
|
|
this.containers[innermostIndex]._trigger("over", event, this._uiHash(this));
|
|
this.containers[innermostIndex].containerCache.over = 1;
|
|
}
|
|
} else {
|
|
|
|
//When entering a new container, we will find the item with the least distance and append our item near it
|
|
dist = 10000;
|
|
itemWithLeastDistance = null;
|
|
floating = innermostContainer.floating || this._isFloating(this.currentItem);
|
|
posProperty = floating ? "left" : "top";
|
|
sizeProperty = floating ? "width" : "height";
|
|
axis = floating ? "clientX" : "clientY";
|
|
|
|
for (j = this.items.length - 1; j >= 0; j--) {
|
|
if(!$.contains(this.containers[innermostIndex].element[0], this.items[j].item[0])) {
|
|
continue;
|
|
}
|
|
if(this.items[j].item[0] === this.currentItem[0]) {
|
|
continue;
|
|
}
|
|
|
|
cur = this.items[j].item.offset()[posProperty];
|
|
nearBottom = false;
|
|
if ( event[ axis ] - cur > this.items[ j ][ sizeProperty ] / 2 ) {
|
|
nearBottom = true;
|
|
}
|
|
|
|
if ( Math.abs( event[ axis ] - cur ) < dist ) {
|
|
dist = Math.abs( event[ axis ] - cur );
|
|
itemWithLeastDistance = this.items[ j ];
|
|
this.direction = nearBottom ? "up": "down";
|
|
}
|
|
}
|
|
|
|
//Check if dropOnEmpty is enabled
|
|
if(!itemWithLeastDistance && !this.options.dropOnEmpty) {
|
|
return;
|
|
}
|
|
|
|
if(this.currentContainer === this.containers[innermostIndex]) {
|
|
if ( !this.currentContainer.containerCache.over ) {
|
|
this.containers[ innermostIndex ]._trigger( "over", event, this._uiHash() );
|
|
this.currentContainer.containerCache.over = 1;
|
|
}
|
|
return;
|
|
}
|
|
|
|
itemWithLeastDistance ? this._rearrange(event, itemWithLeastDistance, null, true) : this._rearrange(event, null, this.containers[innermostIndex].element, true);
|
|
this._trigger("change", event, this._uiHash());
|
|
this.containers[innermostIndex]._trigger("change", event, this._uiHash(this));
|
|
this.currentContainer = this.containers[innermostIndex];
|
|
|
|
//Update the placeholder
|
|
this.options.placeholder.update(this.currentContainer, this.placeholder);
|
|
|
|
this.containers[innermostIndex]._trigger("over", event, this._uiHash(this));
|
|
this.containers[innermostIndex].containerCache.over = 1;
|
|
}
|
|
|
|
|
|
},
|
|
|
|
_createHelper: function(event) {
|
|
|
|
var o = this.options,
|
|
helper = $.isFunction(o.helper) ? $(o.helper.apply(this.element[0], [event, this.currentItem])) : (o.helper === "clone" ? this.currentItem.clone() : this.currentItem);
|
|
|
|
//Add the helper to the DOM if that didn't happen already
|
|
if(!helper.parents("body").length) {
|
|
$(o.appendTo !== "parent" ? o.appendTo : this.currentItem[0].parentNode)[0].appendChild(helper[0]);
|
|
}
|
|
|
|
if(helper[0] === this.currentItem[0]) {
|
|
this._storedCSS = { width: this.currentItem[0].style.width, height: this.currentItem[0].style.height, position: this.currentItem.css("position"), top: this.currentItem.css("top"), left: this.currentItem.css("left") };
|
|
}
|
|
|
|
if(!helper[0].style.width || o.forceHelperSize) {
|
|
helper.width(this.currentItem.width());
|
|
}
|
|
if(!helper[0].style.height || o.forceHelperSize) {
|
|
helper.height(this.currentItem.height());
|
|
}
|
|
|
|
return helper;
|
|
|
|
},
|
|
|
|
_adjustOffsetFromHelper: function(obj) {
|
|
if (typeof obj === "string") {
|
|
obj = obj.split(" ");
|
|
}
|
|
if ($.isArray(obj)) {
|
|
obj = {left: +obj[0], top: +obj[1] || 0};
|
|
}
|
|
if ("left" in obj) {
|
|
this.offset.click.left = obj.left + this.margins.left;
|
|
}
|
|
if ("right" in obj) {
|
|
this.offset.click.left = this.helperProportions.width - obj.right + this.margins.left;
|
|
}
|
|
if ("top" in obj) {
|
|
this.offset.click.top = obj.top + this.margins.top;
|
|
}
|
|
if ("bottom" in obj) {
|
|
this.offset.click.top = this.helperProportions.height - obj.bottom + this.margins.top;
|
|
}
|
|
},
|
|
|
|
_getParentOffset: function() {
|
|
|
|
|
|
//Get the offsetParent and cache its position
|
|
this.offsetParent = this.helper.offsetParent();
|
|
var po = this.offsetParent.offset();
|
|
|
|
// This is a special case where we need to modify a offset calculated on start, since the following happened:
|
|
// 1. The position of the helper is absolute, so it's position is calculated based on the next positioned parent
|
|
// 2. The actual offset parent is a child of the scroll parent, and the scroll parent isn't the document, which means that
|
|
// the scroll is included in the initial calculation of the offset of the parent, and never recalculated upon drag
|
|
if(this.cssPosition === "absolute" && this.scrollParent[0] !== this.document[0] && $.contains(this.scrollParent[0], this.offsetParent[0])) {
|
|
po.left += this.scrollParent.scrollLeft();
|
|
po.top += this.scrollParent.scrollTop();
|
|
}
|
|
|
|
// This needs to be actually done for all browsers, since pageX/pageY includes this information
|
|
// with an ugly IE fix
|
|
if( this.offsetParent[0] === this.document[0].body || (this.offsetParent[0].tagName && this.offsetParent[0].tagName.toLowerCase() === "html" && $.ui.ie)) {
|
|
po = { top: 0, left: 0 };
|
|
}
|
|
|
|
return {
|
|
top: po.top + (parseInt(this.offsetParent.css("borderTopWidth"),10) || 0),
|
|
left: po.left + (parseInt(this.offsetParent.css("borderLeftWidth"),10) || 0)
|
|
};
|
|
|
|
},
|
|
|
|
_getRelativeOffset: function() {
|
|
|
|
if(this.cssPosition === "relative") {
|
|
var p = this.currentItem.position();
|
|
return {
|
|
top: p.top - (parseInt(this.helper.css("top"),10) || 0) + this.scrollParent.scrollTop(),
|
|
left: p.left - (parseInt(this.helper.css("left"),10) || 0) + this.scrollParent.scrollLeft()
|
|
};
|
|
} else {
|
|
return { top: 0, left: 0 };
|
|
}
|
|
|
|
},
|
|
|
|
_cacheMargins: function() {
|
|
this.margins = {
|
|
left: (parseInt(this.currentItem.css("marginLeft"),10) || 0),
|
|
top: (parseInt(this.currentItem.css("marginTop"),10) || 0)
|
|
};
|
|
},
|
|
|
|
_cacheHelperProportions: function() {
|
|
this.helperProportions = {
|
|
width: this.helper.outerWidth(),
|
|
height: this.helper.outerHeight()
|
|
};
|
|
},
|
|
|
|
_setContainment: function() {
|
|
|
|
var ce, co, over,
|
|
o = this.options;
|
|
if(o.containment === "parent") {
|
|
o.containment = this.helper[0].parentNode;
|
|
}
|
|
if(o.containment === "document" || o.containment === "window") {
|
|
this.containment = [
|
|
0 - this.offset.relative.left - this.offset.parent.left,
|
|
0 - this.offset.relative.top - this.offset.parent.top,
|
|
o.containment === "document" ? this.document.width() : this.window.width() - this.helperProportions.width - this.margins.left,
|
|
(o.containment === "document" ? this.document.width() : this.window.height() || this.document[0].body.parentNode.scrollHeight) - this.helperProportions.height - this.margins.top
|
|
];
|
|
}
|
|
|
|
if(!(/^(document|window|parent)$/).test(o.containment)) {
|
|
ce = $(o.containment)[0];
|
|
co = $(o.containment).offset();
|
|
over = ($(ce).css("overflow") !== "hidden");
|
|
|
|
this.containment = [
|
|
co.left + (parseInt($(ce).css("borderLeftWidth"),10) || 0) + (parseInt($(ce).css("paddingLeft"),10) || 0) - this.margins.left,
|
|
co.top + (parseInt($(ce).css("borderTopWidth"),10) || 0) + (parseInt($(ce).css("paddingTop"),10) || 0) - this.margins.top,
|
|
co.left+(over ? Math.max(ce.scrollWidth,ce.offsetWidth) : ce.offsetWidth) - (parseInt($(ce).css("borderLeftWidth"),10) || 0) - (parseInt($(ce).css("paddingRight"),10) || 0) - this.helperProportions.width - this.margins.left,
|
|
co.top+(over ? Math.max(ce.scrollHeight,ce.offsetHeight) : ce.offsetHeight) - (parseInt($(ce).css("borderTopWidth"),10) || 0) - (parseInt($(ce).css("paddingBottom"),10) || 0) - this.helperProportions.height - this.margins.top
|
|
];
|
|
}
|
|
|
|
},
|
|
|
|
_convertPositionTo: function(d, pos) {
|
|
|
|
if(!pos) {
|
|
pos = this.position;
|
|
}
|
|
var mod = d === "absolute" ? 1 : -1,
|
|
scroll = this.cssPosition === "absolute" && !(this.scrollParent[0] !== this.document[0] && $.contains(this.scrollParent[0], this.offsetParent[0])) ? this.offsetParent : this.scrollParent,
|
|
scrollIsRootNode = (/(html|body)/i).test(scroll[0].tagName);
|
|
|
|
return {
|
|
top: (
|
|
pos.top + // The absolute mouse position
|
|
this.offset.relative.top * mod + // Only for relative positioned nodes: Relative offset from element to offset parent
|
|
this.offset.parent.top * mod - // The offsetParent's offset without borders (offset + border)
|
|
( ( this.cssPosition === "fixed" ? -this.scrollParent.scrollTop() : ( scrollIsRootNode ? 0 : scroll.scrollTop() ) ) * mod)
|
|
),
|
|
left: (
|
|
pos.left + // The absolute mouse position
|
|
this.offset.relative.left * mod + // Only for relative positioned nodes: Relative offset from element to offset parent
|
|
this.offset.parent.left * mod - // The offsetParent's offset without borders (offset + border)
|
|
( ( this.cssPosition === "fixed" ? -this.scrollParent.scrollLeft() : scrollIsRootNode ? 0 : scroll.scrollLeft() ) * mod)
|
|
)
|
|
};
|
|
|
|
},
|
|
|
|
_generatePosition: function(event) {
|
|
|
|
var top, left,
|
|
o = this.options,
|
|
pageX = event.pageX,
|
|
pageY = event.pageY,
|
|
scroll = this.cssPosition === "absolute" && !(this.scrollParent[0] !== this.document[0] && $.contains(this.scrollParent[0], this.offsetParent[0])) ? this.offsetParent : this.scrollParent, scrollIsRootNode = (/(html|body)/i).test(scroll[0].tagName);
|
|
|
|
// This is another very weird special case that only happens for relative elements:
|
|
// 1. If the css position is relative
|
|
// 2. and the scroll parent is the document or similar to the offset parent
|
|
// we have to refresh the relative offset during the scroll so there are no jumps
|
|
if(this.cssPosition === "relative" && !(this.scrollParent[0] !== this.document[0] && this.scrollParent[0] !== this.offsetParent[0])) {
|
|
this.offset.relative = this._getRelativeOffset();
|
|
}
|
|
|
|
/*
|
|
* - Position constraining -
|
|
* Constrain the position to a mix of grid, containment.
|
|
*/
|
|
|
|
if(this.originalPosition) { //If we are not dragging yet, we won't check for options
|
|
|
|
if(this.containment) {
|
|
if(event.pageX - this.offset.click.left < this.containment[0]) {
|
|
pageX = this.containment[0] + this.offset.click.left;
|
|
}
|
|
if(event.pageY - this.offset.click.top < this.containment[1]) {
|
|
pageY = this.containment[1] + this.offset.click.top;
|
|
}
|
|
if(event.pageX - this.offset.click.left > this.containment[2]) {
|
|
pageX = this.containment[2] + this.offset.click.left;
|
|
}
|
|
if(event.pageY - this.offset.click.top > this.containment[3]) {
|
|
pageY = this.containment[3] + this.offset.click.top;
|
|
}
|
|
}
|
|
|
|
if(o.grid) {
|
|
top = this.originalPageY + Math.round((pageY - this.originalPageY) / o.grid[1]) * o.grid[1];
|
|
pageY = this.containment ? ( (top - this.offset.click.top >= this.containment[1] && top - this.offset.click.top <= this.containment[3]) ? top : ((top - this.offset.click.top >= this.containment[1]) ? top - o.grid[1] : top + o.grid[1])) : top;
|
|
|
|
left = this.originalPageX + Math.round((pageX - this.originalPageX) / o.grid[0]) * o.grid[0];
|
|
pageX = this.containment ? ( (left - this.offset.click.left >= this.containment[0] && left - this.offset.click.left <= this.containment[2]) ? left : ((left - this.offset.click.left >= this.containment[0]) ? left - o.grid[0] : left + o.grid[0])) : left;
|
|
}
|
|
|
|
}
|
|
|
|
return {
|
|
top: (
|
|
pageY - // The absolute mouse position
|
|
this.offset.click.top - // Click offset (relative to the element)
|
|
this.offset.relative.top - // Only for relative positioned nodes: Relative offset from element to offset parent
|
|
this.offset.parent.top + // The offsetParent's offset without borders (offset + border)
|
|
( ( this.cssPosition === "fixed" ? -this.scrollParent.scrollTop() : ( scrollIsRootNode ? 0 : scroll.scrollTop() ) ))
|
|
),
|
|
left: (
|
|
pageX - // The absolute mouse position
|
|
this.offset.click.left - // Click offset (relative to the element)
|
|
this.offset.relative.left - // Only for relative positioned nodes: Relative offset from element to offset parent
|
|
this.offset.parent.left + // The offsetParent's offset without borders (offset + border)
|
|
( ( this.cssPosition === "fixed" ? -this.scrollParent.scrollLeft() : scrollIsRootNode ? 0 : scroll.scrollLeft() ))
|
|
)
|
|
};
|
|
|
|
},
|
|
|
|
_rearrange: function(event, i, a, hardRefresh) {
|
|
|
|
a ? a[0].appendChild(this.placeholder[0]) : i.item[0].parentNode.insertBefore(this.placeholder[0], (this.direction === "down" ? i.item[0] : i.item[0].nextSibling));
|
|
|
|
//Various things done here to improve the performance:
|
|
// 1. we create a setTimeout, that calls refreshPositions
|
|
// 2. on the instance, we have a counter variable, that get's higher after every append
|
|
// 3. on the local scope, we copy the counter variable, and check in the timeout, if it's still the same
|
|
// 4. this lets only the last addition to the timeout stack through
|
|
this.counter = this.counter ? ++this.counter : 1;
|
|
var counter = this.counter;
|
|
|
|
this._delay(function() {
|
|
if(counter === this.counter) {
|
|
this.refreshPositions(!hardRefresh); //Precompute after each DOM insertion, NOT on mousemove
|
|
}
|
|
});
|
|
|
|
},
|
|
|
|
_clear: function(event, noPropagation) {
|
|
|
|
this.reverting = false;
|
|
// We delay all events that have to be triggered to after the point where the placeholder has been removed and
|
|
// everything else normalized again
|
|
var i,
|
|
delayedTriggers = [];
|
|
|
|
// We first have to update the dom position of the actual currentItem
|
|
// Note: don't do it if the current item is already removed (by a user), or it gets reappended (see #4088)
|
|
if(!this._noFinalSort && this.currentItem.parent().length) {
|
|
this.placeholder.before(this.currentItem);
|
|
}
|
|
this._noFinalSort = null;
|
|
|
|
if(this.helper[0] === this.currentItem[0]) {
|
|
for(i in this._storedCSS) {
|
|
if(this._storedCSS[i] === "auto" || this._storedCSS[i] === "static") {
|
|
this._storedCSS[i] = "";
|
|
}
|
|
}
|
|
this.currentItem.css(this._storedCSS).removeClass("ui-sortable-helper");
|
|
} else {
|
|
this.currentItem.show();
|
|
}
|
|
|
|
if(this.fromOutside && !noPropagation) {
|
|
delayedTriggers.push(function(event) { this._trigger("receive", event, this._uiHash(this.fromOutside)); });
|
|
}
|
|
if((this.fromOutside || this.domPosition.prev !== this.currentItem.prev().not(".ui-sortable-helper")[0] || this.domPosition.parent !== this.currentItem.parent()[0]) && !noPropagation) {
|
|
delayedTriggers.push(function(event) { this._trigger("update", event, this._uiHash()); }); //Trigger update callback if the DOM position has changed
|
|
}
|
|
|
|
// Check if the items Container has Changed and trigger appropriate
|
|
// events.
|
|
if (this !== this.currentContainer) {
|
|
if(!noPropagation) {
|
|
delayedTriggers.push(function(event) { this._trigger("remove", event, this._uiHash()); });
|
|
delayedTriggers.push((function(c) { return function(event) { c._trigger("receive", event, this._uiHash(this)); }; }).call(this, this.currentContainer));
|
|
delayedTriggers.push((function(c) { return function(event) { c._trigger("update", event, this._uiHash(this)); }; }).call(this, this.currentContainer));
|
|
}
|
|
}
|
|
|
|
|
|
//Post events to containers
|
|
function delayEvent( type, instance, container ) {
|
|
return function( event ) {
|
|
container._trigger( type, event, instance._uiHash( instance ) );
|
|
};
|
|
}
|
|
for (i = this.containers.length - 1; i >= 0; i--){
|
|
if (!noPropagation) {
|
|
delayedTriggers.push( delayEvent( "deactivate", this, this.containers[ i ] ) );
|
|
}
|
|
if(this.containers[i].containerCache.over) {
|
|
delayedTriggers.push( delayEvent( "out", this, this.containers[ i ] ) );
|
|
this.containers[i].containerCache.over = 0;
|
|
}
|
|
}
|
|
|
|
//Do what was originally in plugins
|
|
if ( this.storedCursor ) {
|
|
this.document.find( "body" ).css( "cursor", this.storedCursor );
|
|
this.storedStylesheet.remove();
|
|
}
|
|
if(this._storedOpacity) {
|
|
this.helper.css("opacity", this._storedOpacity);
|
|
}
|
|
if(this._storedZIndex) {
|
|
this.helper.css("zIndex", this._storedZIndex === "auto" ? "" : this._storedZIndex);
|
|
}
|
|
|
|
this.dragging = false;
|
|
|
|
if(!noPropagation) {
|
|
this._trigger("beforeStop", event, this._uiHash());
|
|
}
|
|
|
|
//$(this.placeholder[0]).remove(); would have been the jQuery way - unfortunately, it unbinds ALL events from the original node!
|
|
this.placeholder[0].parentNode.removeChild(this.placeholder[0]);
|
|
|
|
if ( !this.cancelHelperRemoval ) {
|
|
if ( this.helper[ 0 ] !== this.currentItem[ 0 ] ) {
|
|
this.helper.remove();
|
|
}
|
|
this.helper = null;
|
|
}
|
|
|
|
if(!noPropagation) {
|
|
for (i=0; i < delayedTriggers.length; i++) {
|
|
delayedTriggers[i].call(this, event);
|
|
} //Trigger all delayed events
|
|
this._trigger("stop", event, this._uiHash());
|
|
}
|
|
|
|
this.fromOutside = false;
|
|
return !this.cancelHelperRemoval;
|
|
|
|
},
|
|
|
|
_trigger: function() {
|
|
if ($.Widget.prototype._trigger.apply(this, arguments) === false) {
|
|
this.cancel();
|
|
}
|
|
},
|
|
|
|
_uiHash: function(_inst) {
|
|
var inst = _inst || this;
|
|
return {
|
|
helper: inst.helper,
|
|
placeholder: inst.placeholder || $([]),
|
|
position: inst.position,
|
|
originalPosition: inst.originalPosition,
|
|
offset: inst.positionAbs,
|
|
item: inst.currentItem,
|
|
sender: _inst ? _inst.element : null
|
|
};
|
|
}
|
|
|
|
});
|
|
|
|
|
|
/*!
|
|
* jQuery UI Spinner 1.11.4
|
|
* http://jqueryui.com
|
|
*
|
|
* Copyright jQuery Foundation and other contributors
|
|
* Released under the MIT license.
|
|
* http://jquery.org/license
|
|
*
|
|
* http://api.jqueryui.com/spinner/
|
|
*/
|
|
|
|
|
|
function spinner_modifier( fn ) {
|
|
return function() {
|
|
var previous = this.element.val();
|
|
fn.apply( this, arguments );
|
|
this._refresh();
|
|
if ( previous !== this.element.val() ) {
|
|
this._trigger( "change" );
|
|
}
|
|
};
|
|
}
|
|
|
|
var spinner = $.widget( "ui.spinner", {
|
|
version: "1.11.4",
|
|
defaultElement: "<input>",
|
|
widgetEventPrefix: "spin",
|
|
options: {
|
|
culture: null,
|
|
icons: {
|
|
down: "ui-icon-triangle-1-s",
|
|
up: "ui-icon-triangle-1-n"
|
|
},
|
|
incremental: true,
|
|
max: null,
|
|
min: null,
|
|
numberFormat: null,
|
|
page: 10,
|
|
step: 1,
|
|
|
|
change: null,
|
|
spin: null,
|
|
start: null,
|
|
stop: null
|
|
},
|
|
|
|
_create: function() {
|
|
// handle string values that need to be parsed
|
|
this._setOption( "max", this.options.max );
|
|
this._setOption( "min", this.options.min );
|
|
this._setOption( "step", this.options.step );
|
|
|
|
// Only format if there is a value, prevents the field from being marked
|
|
// as invalid in Firefox, see #9573.
|
|
if ( this.value() !== "" ) {
|
|
// Format the value, but don't constrain.
|
|
this._value( this.element.val(), true );
|
|
}
|
|
|
|
this._draw();
|
|
this._on( this._events );
|
|
this._refresh();
|
|
|
|
// turning off autocomplete prevents the browser from remembering the
|
|
// value when navigating through history, so we re-enable autocomplete
|
|
// if the page is unloaded before the widget is destroyed. #7790
|
|
this._on( this.window, {
|
|
beforeunload: function() {
|
|
this.element.removeAttr( "autocomplete" );
|
|
}
|
|
});
|
|
},
|
|
|
|
_getCreateOptions: function() {
|
|
var options = {},
|
|
element = this.element;
|
|
|
|
$.each( [ "min", "max", "step" ], function( i, option ) {
|
|
var value = element.attr( option );
|
|
if ( value !== undefined && value.length ) {
|
|
options[ option ] = value;
|
|
}
|
|
});
|
|
|
|
return options;
|
|
},
|
|
|
|
_events: {
|
|
keydown: function( event ) {
|
|
if ( this._start( event ) && this._keydown( event ) ) {
|
|
event.preventDefault();
|
|
}
|
|
},
|
|
keyup: "_stop",
|
|
focus: function() {
|
|
this.previous = this.element.val();
|
|
},
|
|
blur: function( event ) {
|
|
if ( this.cancelBlur ) {
|
|
delete this.cancelBlur;
|
|
return;
|
|
}
|
|
|
|
this._stop();
|
|
this._refresh();
|
|
if ( this.previous !== this.element.val() ) {
|
|
this._trigger( "change", event );
|
|
}
|
|
},
|
|
mousewheel: function( event, delta ) {
|
|
if ( !delta ) {
|
|
return;
|
|
}
|
|
if ( !this.spinning && !this._start( event ) ) {
|
|
return false;
|
|
}
|
|
|
|
this._spin( (delta > 0 ? 1 : -1) * this.options.step, event );
|
|
clearTimeout( this.mousewheelTimer );
|
|
this.mousewheelTimer = this._delay(function() {
|
|
if ( this.spinning ) {
|
|
this._stop( event );
|
|
}
|
|
}, 100 );
|
|
event.preventDefault();
|
|
},
|
|
"mousedown .ui-spinner-button": function( event ) {
|
|
var previous;
|
|
|
|
// We never want the buttons to have focus; whenever the user is
|
|
// interacting with the spinner, the focus should be on the input.
|
|
// If the input is focused then this.previous is properly set from
|
|
// when the input first received focus. If the input is not focused
|
|
// then we need to set this.previous based on the value before spinning.
|
|
previous = this.element[0] === this.document[0].activeElement ?
|
|
this.previous : this.element.val();
|
|
function checkFocus() {
|
|
var isActive = this.element[0] === this.document[0].activeElement;
|
|
if ( !isActive ) {
|
|
this.element.focus();
|
|
this.previous = previous;
|
|
// support: IE
|
|
// IE sets focus asynchronously, so we need to check if focus
|
|
// moved off of the input because the user clicked on the button.
|
|
this._delay(function() {
|
|
this.previous = previous;
|
|
});
|
|
}
|
|
}
|
|
|
|
// ensure focus is on (or stays on) the text field
|
|
event.preventDefault();
|
|
checkFocus.call( this );
|
|
|
|
// support: IE
|
|
// IE doesn't prevent moving focus even with event.preventDefault()
|
|
// so we set a flag to know when we should ignore the blur event
|
|
// and check (again) if focus moved off of the input.
|
|
this.cancelBlur = true;
|
|
this._delay(function() {
|
|
delete this.cancelBlur;
|
|
checkFocus.call( this );
|
|
});
|
|
|
|
if ( this._start( event ) === false ) {
|
|
return;
|
|
}
|
|
|
|
this._repeat( null, $( event.currentTarget ).hasClass( "ui-spinner-up" ) ? 1 : -1, event );
|
|
},
|
|
"mouseup .ui-spinner-button": "_stop",
|
|
"mouseenter .ui-spinner-button": function( event ) {
|
|
// button will add ui-state-active if mouse was down while mouseleave and kept down
|
|
if ( !$( event.currentTarget ).hasClass( "ui-state-active" ) ) {
|
|
return;
|
|
}
|
|
|
|
if ( this._start( event ) === false ) {
|
|
return false;
|
|
}
|
|
this._repeat( null, $( event.currentTarget ).hasClass( "ui-spinner-up" ) ? 1 : -1, event );
|
|
},
|
|
// TODO: do we really want to consider this a stop?
|
|
// shouldn't we just stop the repeater and wait until mouseup before
|
|
// we trigger the stop event?
|
|
"mouseleave .ui-spinner-button": "_stop"
|
|
},
|
|
|
|
_draw: function() {
|
|
var uiSpinner = this.uiSpinner = this.element
|
|
.addClass( "ui-spinner-input" )
|
|
.attr( "autocomplete", "off" )
|
|
.wrap( this._uiSpinnerHtml() )
|
|
.parent()
|
|
// add buttons
|
|
.append( this._buttonHtml() );
|
|
|
|
this.element.attr( "role", "spinbutton" );
|
|
|
|
// button bindings
|
|
this.buttons = uiSpinner.find( ".ui-spinner-button" )
|
|
.attr( "tabIndex", -1 )
|
|
.button()
|
|
.removeClass( "ui-corner-all" );
|
|
|
|
// IE 6 doesn't understand height: 50% for the buttons
|
|
// unless the wrapper has an explicit height
|
|
if ( this.buttons.height() > Math.ceil( uiSpinner.height() * 0.5 ) &&
|
|
uiSpinner.height() > 0 ) {
|
|
uiSpinner.height( uiSpinner.height() );
|
|
}
|
|
|
|
// disable spinner if element was already disabled
|
|
if ( this.options.disabled ) {
|
|
this.disable();
|
|
}
|
|
},
|
|
|
|
_keydown: function( event ) {
|
|
var options = this.options,
|
|
keyCode = $.ui.keyCode;
|
|
|
|
switch ( event.keyCode ) {
|
|
case keyCode.UP:
|
|
this._repeat( null, 1, event );
|
|
return true;
|
|
case keyCode.DOWN:
|
|
this._repeat( null, -1, event );
|
|
return true;
|
|
case keyCode.PAGE_UP:
|
|
this._repeat( null, options.page, event );
|
|
return true;
|
|
case keyCode.PAGE_DOWN:
|
|
this._repeat( null, -options.page, event );
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
},
|
|
|
|
_uiSpinnerHtml: function() {
|
|
return "<span class='ui-spinner ui-widget ui-widget-content ui-corner-all'></span>";
|
|
},
|
|
|
|
_buttonHtml: function() {
|
|
return "" +
|
|
"<a class='ui-spinner-button ui-spinner-up ui-corner-tr'>" +
|
|
"<span class='ui-icon " + this.options.icons.up + "'>▲</span>" +
|
|
"</a>" +
|
|
"<a class='ui-spinner-button ui-spinner-down ui-corner-br'>" +
|
|
"<span class='ui-icon " + this.options.icons.down + "'>▼</span>" +
|
|
"</a>";
|
|
},
|
|
|
|
_start: function( event ) {
|
|
if ( !this.spinning && this._trigger( "start", event ) === false ) {
|
|
return false;
|
|
}
|
|
|
|
if ( !this.counter ) {
|
|
this.counter = 1;
|
|
}
|
|
this.spinning = true;
|
|
return true;
|
|
},
|
|
|
|
_repeat: function( i, steps, event ) {
|
|
i = i || 500;
|
|
|
|
clearTimeout( this.timer );
|
|
this.timer = this._delay(function() {
|
|
this._repeat( 40, steps, event );
|
|
}, i );
|
|
|
|
this._spin( steps * this.options.step, event );
|
|
},
|
|
|
|
_spin: function( step, event ) {
|
|
var value = this.value() || 0;
|
|
|
|
if ( !this.counter ) {
|
|
this.counter = 1;
|
|
}
|
|
|
|
value = this._adjustValue( value + step * this._increment( this.counter ) );
|
|
|
|
if ( !this.spinning || this._trigger( "spin", event, { value: value } ) !== false) {
|
|
this._value( value );
|
|
this.counter++;
|
|
}
|
|
},
|
|
|
|
_increment: function( i ) {
|
|
var incremental = this.options.incremental;
|
|
|
|
if ( incremental ) {
|
|
return $.isFunction( incremental ) ?
|
|
incremental( i ) :
|
|
Math.floor( i * i * i / 50000 - i * i / 500 + 17 * i / 200 + 1 );
|
|
}
|
|
|
|
return 1;
|
|
},
|
|
|
|
_precision: function() {
|
|
var precision = this._precisionOf( this.options.step );
|
|
if ( this.options.min !== null ) {
|
|
precision = Math.max( precision, this._precisionOf( this.options.min ) );
|
|
}
|
|
return precision;
|
|
},
|
|
|
|
_precisionOf: function( num ) {
|
|
var str = num.toString(),
|
|
decimal = str.indexOf( "." );
|
|
return decimal === -1 ? 0 : str.length - decimal - 1;
|
|
},
|
|
|
|
_adjustValue: function( value ) {
|
|
var base, aboveMin,
|
|
options = this.options;
|
|
|
|
// make sure we're at a valid step
|
|
// - find out where we are relative to the base (min or 0)
|
|
base = options.min !== null ? options.min : 0;
|
|
aboveMin = value - base;
|
|
// - round to the nearest step
|
|
aboveMin = Math.round(aboveMin / options.step) * options.step;
|
|
// - rounding is based on 0, so adjust back to our base
|
|
value = base + aboveMin;
|
|
|
|
// fix precision from bad JS floating point math
|
|
value = parseFloat( value.toFixed( this._precision() ) );
|
|
|
|
// clamp the value
|
|
if ( options.max !== null && value > options.max) {
|
|
return options.max;
|
|
}
|
|
if ( options.min !== null && value < options.min ) {
|
|
return options.min;
|
|
}
|
|
|
|
return value;
|
|
},
|
|
|
|
_stop: function( event ) {
|
|
if ( !this.spinning ) {
|
|
return;
|
|
}
|
|
|
|
clearTimeout( this.timer );
|
|
clearTimeout( this.mousewheelTimer );
|
|
this.counter = 0;
|
|
this.spinning = false;
|
|
this._trigger( "stop", event );
|
|
},
|
|
|
|
_setOption: function( key, value ) {
|
|
if ( key === "culture" || key === "numberFormat" ) {
|
|
var prevValue = this._parse( this.element.val() );
|
|
this.options[ key ] = value;
|
|
this.element.val( this._format( prevValue ) );
|
|
return;
|
|
}
|
|
|
|
if ( key === "max" || key === "min" || key === "step" ) {
|
|
if ( typeof value === "string" ) {
|
|
value = this._parse( value );
|
|
}
|
|
}
|
|
if ( key === "icons" ) {
|
|
this.buttons.first().find( ".ui-icon" )
|
|
.removeClass( this.options.icons.up )
|
|
.addClass( value.up );
|
|
this.buttons.last().find( ".ui-icon" )
|
|
.removeClass( this.options.icons.down )
|
|
.addClass( value.down );
|
|
}
|
|
|
|
this._super( key, value );
|
|
|
|
if ( key === "disabled" ) {
|
|
this.widget().toggleClass( "ui-state-disabled", !!value );
|
|
this.element.prop( "disabled", !!value );
|
|
this.buttons.button( value ? "disable" : "enable" );
|
|
}
|
|
},
|
|
|
|
_setOptions: spinner_modifier(function( options ) {
|
|
this._super( options );
|
|
}),
|
|
|
|
_parse: function( val ) {
|
|
if ( typeof val === "string" && val !== "" ) {
|
|
val = window.Globalize && this.options.numberFormat ?
|
|
Globalize.parseFloat( val, 10, this.options.culture ) : +val;
|
|
}
|
|
return val === "" || isNaN( val ) ? null : val;
|
|
},
|
|
|
|
_format: function( value ) {
|
|
if ( value === "" ) {
|
|
return "";
|
|
}
|
|
return window.Globalize && this.options.numberFormat ?
|
|
Globalize.format( value, this.options.numberFormat, this.options.culture ) :
|
|
value;
|
|
},
|
|
|
|
_refresh: function() {
|
|
this.element.attr({
|
|
"aria-valuemin": this.options.min,
|
|
"aria-valuemax": this.options.max,
|
|
// TODO: what should we do with values that can't be parsed?
|
|
"aria-valuenow": this._parse( this.element.val() )
|
|
});
|
|
},
|
|
|
|
isValid: function() {
|
|
var value = this.value();
|
|
|
|
// null is invalid
|
|
if ( value === null ) {
|
|
return false;
|
|
}
|
|
|
|
// if value gets adjusted, it's invalid
|
|
return value === this._adjustValue( value );
|
|
},
|
|
|
|
// update the value without triggering change
|
|
_value: function( value, allowAny ) {
|
|
var parsed;
|
|
if ( value !== "" ) {
|
|
parsed = this._parse( value );
|
|
if ( parsed !== null ) {
|
|
if ( !allowAny ) {
|
|
parsed = this._adjustValue( parsed );
|
|
}
|
|
value = this._format( parsed );
|
|
}
|
|
}
|
|
this.element.val( value );
|
|
this._refresh();
|
|
},
|
|
|
|
_destroy: function() {
|
|
this.element
|
|
.removeClass( "ui-spinner-input" )
|
|
.prop( "disabled", false )
|
|
.removeAttr( "autocomplete" )
|
|
.removeAttr( "role" )
|
|
.removeAttr( "aria-valuemin" )
|
|
.removeAttr( "aria-valuemax" )
|
|
.removeAttr( "aria-valuenow" );
|
|
this.uiSpinner.replaceWith( this.element );
|
|
},
|
|
|
|
stepUp: spinner_modifier(function( steps ) {
|
|
this._stepUp( steps );
|
|
}),
|
|
_stepUp: function( steps ) {
|
|
if ( this._start() ) {
|
|
this._spin( (steps || 1) * this.options.step );
|
|
this._stop();
|
|
}
|
|
},
|
|
|
|
stepDown: spinner_modifier(function( steps ) {
|
|
this._stepDown( steps );
|
|
}),
|
|
_stepDown: function( steps ) {
|
|
if ( this._start() ) {
|
|
this._spin( (steps || 1) * -this.options.step );
|
|
this._stop();
|
|
}
|
|
},
|
|
|
|
pageUp: spinner_modifier(function( pages ) {
|
|
this._stepUp( (pages || 1) * this.options.page );
|
|
}),
|
|
|
|
pageDown: spinner_modifier(function( pages ) {
|
|
this._stepDown( (pages || 1) * this.options.page );
|
|
}),
|
|
|
|
value: function( newVal ) {
|
|
if ( !arguments.length ) {
|
|
return this._parse( this.element.val() );
|
|
}
|
|
spinner_modifier( this._value ).call( this, newVal );
|
|
},
|
|
|
|
widget: function() {
|
|
return this.uiSpinner;
|
|
}
|
|
});
|
|
|
|
|
|
/*!
|
|
* jQuery UI Tabs 1.11.4
|
|
* http://jqueryui.com
|
|
*
|
|
* Copyright jQuery Foundation and other contributors
|
|
* Released under the MIT license.
|
|
* http://jquery.org/license
|
|
*
|
|
* http://api.jqueryui.com/tabs/
|
|
*/
|
|
|
|
|
|
var tabs = $.widget( "ui.tabs", {
|
|
version: "1.11.4",
|
|
delay: 300,
|
|
options: {
|
|
active: null,
|
|
collapsible: false,
|
|
event: "click",
|
|
heightStyle: "content",
|
|
hide: null,
|
|
show: null,
|
|
|
|
// callbacks
|
|
activate: null,
|
|
beforeActivate: null,
|
|
beforeLoad: null,
|
|
load: null
|
|
},
|
|
|
|
_isLocal: (function() {
|
|
var rhash = /#.*$/;
|
|
|
|
return function( anchor ) {
|
|
var anchorUrl, locationUrl;
|
|
|
|
// support: IE7
|
|
// IE7 doesn't normalize the href property when set via script (#9317)
|
|
anchor = anchor.cloneNode( false );
|
|
|
|
anchorUrl = anchor.href.replace( rhash, "" );
|
|
locationUrl = location.href.replace( rhash, "" );
|
|
|
|
// decoding may throw an error if the URL isn't UTF-8 (#9518)
|
|
try {
|
|
anchorUrl = decodeURIComponent( anchorUrl );
|
|
} catch ( error ) {}
|
|
try {
|
|
locationUrl = decodeURIComponent( locationUrl );
|
|
} catch ( error ) {}
|
|
|
|
return anchor.hash.length > 1 && anchorUrl === locationUrl;
|
|
};
|
|
})(),
|
|
|
|
_create: function() {
|
|
var that = this,
|
|
options = this.options;
|
|
|
|
this.running = false;
|
|
|
|
this.element
|
|
.addClass( "ui-tabs ui-widget ui-widget-content ui-corner-all" )
|
|
.toggleClass( "ui-tabs-collapsible", options.collapsible );
|
|
|
|
this._processTabs();
|
|
options.active = this._initialActive();
|
|
|
|
// Take disabling tabs via class attribute from HTML
|
|
// into account and update option properly.
|
|
if ( $.isArray( options.disabled ) ) {
|
|
options.disabled = $.unique( options.disabled.concat(
|
|
$.map( this.tabs.filter( ".ui-state-disabled" ), function( li ) {
|
|
return that.tabs.index( li );
|
|
})
|
|
) ).sort();
|
|
}
|
|
|
|
// check for length avoids error when initializing empty list
|
|
if ( this.options.active !== false && this.anchors.length ) {
|
|
this.active = this._findActive( options.active );
|
|
} else {
|
|
this.active = $();
|
|
}
|
|
|
|
this._refresh();
|
|
|
|
if ( this.active.length ) {
|
|
this.load( options.active );
|
|
}
|
|
},
|
|
|
|
_initialActive: function() {
|
|
var active = this.options.active,
|
|
collapsible = this.options.collapsible,
|
|
locationHash = location.hash.substring( 1 );
|
|
|
|
if ( active === null ) {
|
|
// check the fragment identifier in the URL
|
|
if ( locationHash ) {
|
|
this.tabs.each(function( i, tab ) {
|
|
if ( $( tab ).attr( "aria-controls" ) === locationHash ) {
|
|
active = i;
|
|
return false;
|
|
}
|
|
});
|
|
}
|
|
|
|
// check for a tab marked active via a class
|
|
if ( active === null ) {
|
|
active = this.tabs.index( this.tabs.filter( ".ui-tabs-active" ) );
|
|
}
|
|
|
|
// no active tab, set to false
|
|
if ( active === null || active === -1 ) {
|
|
active = this.tabs.length ? 0 : false;
|
|
}
|
|
}
|
|
|
|
// handle numbers: negative, out of range
|
|
if ( active !== false ) {
|
|
active = this.tabs.index( this.tabs.eq( active ) );
|
|
if ( active === -1 ) {
|
|
active = collapsible ? false : 0;
|
|
}
|
|
}
|
|
|
|
// don't allow collapsible: false and active: false
|
|
if ( !collapsible && active === false && this.anchors.length ) {
|
|
active = 0;
|
|
}
|
|
|
|
return active;
|
|
},
|
|
|
|
_getCreateEventData: function() {
|
|
return {
|
|
tab: this.active,
|
|
panel: !this.active.length ? $() : this._getPanelForTab( this.active )
|
|
};
|
|
},
|
|
|
|
_tabKeydown: function( event ) {
|
|
var focusedTab = $( this.document[0].activeElement ).closest( "li" ),
|
|
selectedIndex = this.tabs.index( focusedTab ),
|
|
goingForward = true;
|
|
|
|
if ( this._handlePageNav( event ) ) {
|
|
return;
|
|
}
|
|
|
|
switch ( event.keyCode ) {
|
|
case $.ui.keyCode.RIGHT:
|
|
case $.ui.keyCode.DOWN:
|
|
selectedIndex++;
|
|
break;
|
|
case $.ui.keyCode.UP:
|
|
case $.ui.keyCode.LEFT:
|
|
goingForward = false;
|
|
selectedIndex--;
|
|
break;
|
|
case $.ui.keyCode.END:
|
|
selectedIndex = this.anchors.length - 1;
|
|
break;
|
|
case $.ui.keyCode.HOME:
|
|
selectedIndex = 0;
|
|
break;
|
|
case $.ui.keyCode.SPACE:
|
|
// Activate only, no collapsing
|
|
event.preventDefault();
|
|
clearTimeout( this.activating );
|
|
this._activate( selectedIndex );
|
|
return;
|
|
case $.ui.keyCode.ENTER:
|
|
// Toggle (cancel delayed activation, allow collapsing)
|
|
event.preventDefault();
|
|
clearTimeout( this.activating );
|
|
// Determine if we should collapse or activate
|
|
this._activate( selectedIndex === this.options.active ? false : selectedIndex );
|
|
return;
|
|
default:
|
|
return;
|
|
}
|
|
|
|
// Focus the appropriate tab, based on which key was pressed
|
|
event.preventDefault();
|
|
clearTimeout( this.activating );
|
|
selectedIndex = this._focusNextTab( selectedIndex, goingForward );
|
|
|
|
// Navigating with control/command key will prevent automatic activation
|
|
if ( !event.ctrlKey && !event.metaKey ) {
|
|
|
|
// Update aria-selected immediately so that AT think the tab is already selected.
|
|
// Otherwise AT may confuse the user by stating that they need to activate the tab,
|
|
// but the tab will already be activated by the time the announcement finishes.
|
|
focusedTab.attr( "aria-selected", "false" );
|
|
this.tabs.eq( selectedIndex ).attr( "aria-selected", "true" );
|
|
|
|
this.activating = this._delay(function() {
|
|
this.option( "active", selectedIndex );
|
|
}, this.delay );
|
|
}
|
|
},
|
|
|
|
_panelKeydown: function( event ) {
|
|
if ( this._handlePageNav( event ) ) {
|
|
return;
|
|
}
|
|
|
|
// Ctrl+up moves focus to the current tab
|
|
if ( event.ctrlKey && event.keyCode === $.ui.keyCode.UP ) {
|
|
event.preventDefault();
|
|
this.active.focus();
|
|
}
|
|
},
|
|
|
|
// Alt+page up/down moves focus to the previous/next tab (and activates)
|
|
_handlePageNav: function( event ) {
|
|
if ( event.altKey && event.keyCode === $.ui.keyCode.PAGE_UP ) {
|
|
this._activate( this._focusNextTab( this.options.active - 1, false ) );
|
|
return true;
|
|
}
|
|
if ( event.altKey && event.keyCode === $.ui.keyCode.PAGE_DOWN ) {
|
|
this._activate( this._focusNextTab( this.options.active + 1, true ) );
|
|
return true;
|
|
}
|
|
},
|
|
|
|
_findNextTab: function( index, goingForward ) {
|
|
var lastTabIndex = this.tabs.length - 1;
|
|
|
|
function constrain() {
|
|
if ( index > lastTabIndex ) {
|
|
index = 0;
|
|
}
|
|
if ( index < 0 ) {
|
|
index = lastTabIndex;
|
|
}
|
|
return index;
|
|
}
|
|
|
|
while ( $.inArray( constrain(), this.options.disabled ) !== -1 ) {
|
|
index = goingForward ? index + 1 : index - 1;
|
|
}
|
|
|
|
return index;
|
|
},
|
|
|
|
_focusNextTab: function( index, goingForward ) {
|
|
index = this._findNextTab( index, goingForward );
|
|
this.tabs.eq( index ).focus();
|
|
return index;
|
|
},
|
|
|
|
_setOption: function( key, value ) {
|
|
if ( key === "active" ) {
|
|
// _activate() will handle invalid values and update this.options
|
|
this._activate( value );
|
|
return;
|
|
}
|
|
|
|
if ( key === "disabled" ) {
|
|
// don't use the widget factory's disabled handling
|
|
this._setupDisabled( value );
|
|
return;
|
|
}
|
|
|
|
this._super( key, value);
|
|
|
|
if ( key === "collapsible" ) {
|
|
this.element.toggleClass( "ui-tabs-collapsible", value );
|
|
// Setting collapsible: false while collapsed; open first panel
|
|
if ( !value && this.options.active === false ) {
|
|
this._activate( 0 );
|
|
}
|
|
}
|
|
|
|
if ( key === "event" ) {
|
|
this._setupEvents( value );
|
|
}
|
|
|
|
if ( key === "heightStyle" ) {
|
|
this._setupHeightStyle( value );
|
|
}
|
|
},
|
|
|
|
_sanitizeSelector: function( hash ) {
|
|
return hash ? hash.replace( /[!"$%&'()*+,.\/:;<=>?@\[\]\^`{|}~]/g, "\\$&" ) : "";
|
|
},
|
|
|
|
refresh: function() {
|
|
var options = this.options,
|
|
lis = this.tablist.children( ":has(a[href])" );
|
|
|
|
// get disabled tabs from class attribute from HTML
|
|
// this will get converted to a boolean if needed in _refresh()
|
|
options.disabled = $.map( lis.filter( ".ui-state-disabled" ), function( tab ) {
|
|
return lis.index( tab );
|
|
});
|
|
|
|
this._processTabs();
|
|
|
|
// was collapsed or no tabs
|
|
if ( options.active === false || !this.anchors.length ) {
|
|
options.active = false;
|
|
this.active = $();
|
|
// was active, but active tab is gone
|
|
} else if ( this.active.length && !$.contains( this.tablist[ 0 ], this.active[ 0 ] ) ) {
|
|
// all remaining tabs are disabled
|
|
if ( this.tabs.length === options.disabled.length ) {
|
|
options.active = false;
|
|
this.active = $();
|
|
// activate previous tab
|
|
} else {
|
|
this._activate( this._findNextTab( Math.max( 0, options.active - 1 ), false ) );
|
|
}
|
|
// was active, active tab still exists
|
|
} else {
|
|
// make sure active index is correct
|
|
options.active = this.tabs.index( this.active );
|
|
}
|
|
|
|
this._refresh();
|
|
},
|
|
|
|
_refresh: function() {
|
|
this._setupDisabled( this.options.disabled );
|
|
this._setupEvents( this.options.event );
|
|
this._setupHeightStyle( this.options.heightStyle );
|
|
|
|
this.tabs.not( this.active ).attr({
|
|
"aria-selected": "false",
|
|
"aria-expanded": "false",
|
|
tabIndex: -1
|
|
});
|
|
this.panels.not( this._getPanelForTab( this.active ) )
|
|
.hide()
|
|
.attr({
|
|
"aria-hidden": "true"
|
|
});
|
|
|
|
// Make sure one tab is in the tab order
|
|
if ( !this.active.length ) {
|
|
this.tabs.eq( 0 ).attr( "tabIndex", 0 );
|
|
} else {
|
|
this.active
|
|
.addClass( "ui-tabs-active ui-state-active" )
|
|
.attr({
|
|
"aria-selected": "true",
|
|
"aria-expanded": "true",
|
|
tabIndex: 0
|
|
});
|
|
this._getPanelForTab( this.active )
|
|
.show()
|
|
.attr({
|
|
"aria-hidden": "false"
|
|
});
|
|
}
|
|
},
|
|
|
|
_processTabs: function() {
|
|
var that = this,
|
|
prevTabs = this.tabs,
|
|
prevAnchors = this.anchors,
|
|
prevPanels = this.panels;
|
|
|
|
this.tablist = this._getList()
|
|
.addClass( "ui-tabs-nav ui-helper-reset ui-helper-clearfix ui-widget-header ui-corner-all" )
|
|
.attr( "role", "tablist" )
|
|
|
|
// Prevent users from focusing disabled tabs via click
|
|
.delegate( "> li", "mousedown" + this.eventNamespace, function( event ) {
|
|
if ( $( this ).is( ".ui-state-disabled" ) ) {
|
|
event.preventDefault();
|
|
}
|
|
})
|
|
|
|
// support: IE <9
|
|
// Preventing the default action in mousedown doesn't prevent IE
|
|
// from focusing the element, so if the anchor gets focused, blur.
|
|
// We don't have to worry about focusing the previously focused
|
|
// element since clicking on a non-focusable element should focus
|
|
// the body anyway.
|
|
.delegate( ".ui-tabs-anchor", "focus" + this.eventNamespace, function() {
|
|
if ( $( this ).closest( "li" ).is( ".ui-state-disabled" ) ) {
|
|
this.blur();
|
|
}
|
|
});
|
|
|
|
this.tabs = this.tablist.find( "> li:has(a[href])" )
|
|
.addClass( "ui-state-default ui-corner-top" )
|
|
.attr({
|
|
role: "tab",
|
|
tabIndex: -1
|
|
});
|
|
|
|
this.anchors = this.tabs.map(function() {
|
|
return $( "a", this )[ 0 ];
|
|
})
|
|
.addClass( "ui-tabs-anchor" )
|
|
.attr({
|
|
role: "presentation",
|
|
tabIndex: -1
|
|
});
|
|
|
|
this.panels = $();
|
|
|
|
this.anchors.each(function( i, anchor ) {
|
|
var selector, panel, panelId,
|
|
anchorId = $( anchor ).uniqueId().attr( "id" ),
|
|
tab = $( anchor ).closest( "li" ),
|
|
originalAriaControls = tab.attr( "aria-controls" );
|
|
|
|
// inline tab
|
|
if ( that._isLocal( anchor ) ) {
|
|
selector = anchor.hash;
|
|
panelId = selector.substring( 1 );
|
|
panel = that.element.find( that._sanitizeSelector( selector ) );
|
|
// remote tab
|
|
} else {
|
|
// If the tab doesn't already have aria-controls,
|
|
// generate an id by using a throw-away element
|
|
panelId = tab.attr( "aria-controls" ) || $( {} ).uniqueId()[ 0 ].id;
|
|
selector = "#" + panelId;
|
|
panel = that.element.find( selector );
|
|
if ( !panel.length ) {
|
|
panel = that._createPanel( panelId );
|
|
panel.insertAfter( that.panels[ i - 1 ] || that.tablist );
|
|
}
|
|
panel.attr( "aria-live", "polite" );
|
|
}
|
|
|
|
if ( panel.length) {
|
|
that.panels = that.panels.add( panel );
|
|
}
|
|
if ( originalAriaControls ) {
|
|
tab.data( "ui-tabs-aria-controls", originalAriaControls );
|
|
}
|
|
tab.attr({
|
|
"aria-controls": panelId,
|
|
"aria-labelledby": anchorId
|
|
});
|
|
panel.attr( "aria-labelledby", anchorId );
|
|
});
|
|
|
|
this.panels
|
|
.addClass( "ui-tabs-panel ui-widget-content ui-corner-bottom" )
|
|
.attr( "role", "tabpanel" );
|
|
|
|
// Avoid memory leaks (#10056)
|
|
if ( prevTabs ) {
|
|
this._off( prevTabs.not( this.tabs ) );
|
|
this._off( prevAnchors.not( this.anchors ) );
|
|
this._off( prevPanels.not( this.panels ) );
|
|
}
|
|
},
|
|
|
|
// allow overriding how to find the list for rare usage scenarios (#7715)
|
|
_getList: function() {
|
|
return this.tablist || this.element.find( "ol,ul" ).eq( 0 );
|
|
},
|
|
|
|
_createPanel: function( id ) {
|
|
return $( "<div>" )
|
|
.attr( "id", id )
|
|
.addClass( "ui-tabs-panel ui-widget-content ui-corner-bottom" )
|
|
.data( "ui-tabs-destroy", true );
|
|
},
|
|
|
|
_setupDisabled: function( disabled ) {
|
|
if ( $.isArray( disabled ) ) {
|
|
if ( !disabled.length ) {
|
|
disabled = false;
|
|
} else if ( disabled.length === this.anchors.length ) {
|
|
disabled = true;
|
|
}
|
|
}
|
|
|
|
// disable tabs
|
|
for ( var i = 0, li; ( li = this.tabs[ i ] ); i++ ) {
|
|
if ( disabled === true || $.inArray( i, disabled ) !== -1 ) {
|
|
$( li )
|
|
.addClass( "ui-state-disabled" )
|
|
.attr( "aria-disabled", "true" );
|
|
} else {
|
|
$( li )
|
|
.removeClass( "ui-state-disabled" )
|
|
.removeAttr( "aria-disabled" );
|
|
}
|
|
}
|
|
|
|
this.options.disabled = disabled;
|
|
},
|
|
|
|
_setupEvents: function( event ) {
|
|
var events = {};
|
|
if ( event ) {
|
|
$.each( event.split(" "), function( index, eventName ) {
|
|
events[ eventName ] = "_eventHandler";
|
|
});
|
|
}
|
|
|
|
this._off( this.anchors.add( this.tabs ).add( this.panels ) );
|
|
// Always prevent the default action, even when disabled
|
|
this._on( true, this.anchors, {
|
|
click: function( event ) {
|
|
event.preventDefault();
|
|
}
|
|
});
|
|
this._on( this.anchors, events );
|
|
this._on( this.tabs, { keydown: "_tabKeydown" } );
|
|
this._on( this.panels, { keydown: "_panelKeydown" } );
|
|
|
|
this._focusable( this.tabs );
|
|
this._hoverable( this.tabs );
|
|
},
|
|
|
|
_setupHeightStyle: function( heightStyle ) {
|
|
var maxHeight,
|
|
parent = this.element.parent();
|
|
|
|
if ( heightStyle === "fill" ) {
|
|
maxHeight = parent.height();
|
|
maxHeight -= this.element.outerHeight() - this.element.height();
|
|
|
|
this.element.siblings( ":visible" ).each(function() {
|
|
var elem = $( this ),
|
|
position = elem.css( "position" );
|
|
|
|
if ( position === "absolute" || position === "fixed" ) {
|
|
return;
|
|
}
|
|
maxHeight -= elem.outerHeight( true );
|
|
});
|
|
|
|
this.element.children().not( this.panels ).each(function() {
|
|
maxHeight -= $( this ).outerHeight( true );
|
|
});
|
|
|
|
this.panels.each(function() {
|
|
$( this ).height( Math.max( 0, maxHeight -
|
|
$( this ).innerHeight() + $( this ).height() ) );
|
|
})
|
|
.css( "overflow", "auto" );
|
|
} else if ( heightStyle === "auto" ) {
|
|
maxHeight = 0;
|
|
this.panels.each(function() {
|
|
maxHeight = Math.max( maxHeight, $( this ).height( "" ).height() );
|
|
}).height( maxHeight );
|
|
}
|
|
},
|
|
|
|
_eventHandler: function( event ) {
|
|
var options = this.options,
|
|
active = this.active,
|
|
anchor = $( event.currentTarget ),
|
|
tab = anchor.closest( "li" ),
|
|
clickedIsActive = tab[ 0 ] === active[ 0 ],
|
|
collapsing = clickedIsActive && options.collapsible,
|
|
toShow = collapsing ? $() : this._getPanelForTab( tab ),
|
|
toHide = !active.length ? $() : this._getPanelForTab( active ),
|
|
eventData = {
|
|
oldTab: active,
|
|
oldPanel: toHide,
|
|
newTab: collapsing ? $() : tab,
|
|
newPanel: toShow
|
|
};
|
|
|
|
event.preventDefault();
|
|
|
|
if ( tab.hasClass( "ui-state-disabled" ) ||
|
|
// tab is already loading
|
|
tab.hasClass( "ui-tabs-loading" ) ||
|
|
// can't switch durning an animation
|
|
this.running ||
|
|
// click on active header, but not collapsible
|
|
( clickedIsActive && !options.collapsible ) ||
|
|
// allow canceling activation
|
|
( this._trigger( "beforeActivate", event, eventData ) === false ) ) {
|
|
return;
|
|
}
|
|
|
|
options.active = collapsing ? false : this.tabs.index( tab );
|
|
|
|
this.active = clickedIsActive ? $() : tab;
|
|
if ( this.xhr ) {
|
|
this.xhr.abort();
|
|
}
|
|
|
|
if ( !toHide.length && !toShow.length ) {
|
|
$.error( "jQuery UI Tabs: Mismatching fragment identifier." );
|
|
}
|
|
|
|
if ( toShow.length ) {
|
|
this.load( this.tabs.index( tab ), event );
|
|
}
|
|
this._toggle( event, eventData );
|
|
},
|
|
|
|
// handles show/hide for selecting tabs
|
|
_toggle: function( event, eventData ) {
|
|
var that = this,
|
|
toShow = eventData.newPanel,
|
|
toHide = eventData.oldPanel;
|
|
|
|
this.running = true;
|
|
|
|
function complete() {
|
|
that.running = false;
|
|
that._trigger( "activate", event, eventData );
|
|
}
|
|
|
|
function show() {
|
|
eventData.newTab.closest( "li" ).addClass( "ui-tabs-active ui-state-active" );
|
|
|
|
if ( toShow.length && that.options.show ) {
|
|
that._show( toShow, that.options.show, complete );
|
|
} else {
|
|
toShow.show();
|
|
complete();
|
|
}
|
|
}
|
|
|
|
// start out by hiding, then showing, then completing
|
|
if ( toHide.length && this.options.hide ) {
|
|
this._hide( toHide, this.options.hide, function() {
|
|
eventData.oldTab.closest( "li" ).removeClass( "ui-tabs-active ui-state-active" );
|
|
show();
|
|
});
|
|
} else {
|
|
eventData.oldTab.closest( "li" ).removeClass( "ui-tabs-active ui-state-active" );
|
|
toHide.hide();
|
|
show();
|
|
}
|
|
|
|
toHide.attr( "aria-hidden", "true" );
|
|
eventData.oldTab.attr({
|
|
"aria-selected": "false",
|
|
"aria-expanded": "false"
|
|
});
|
|
// If we're switching tabs, remove the old tab from the tab order.
|
|
// If we're opening from collapsed state, remove the previous tab from the tab order.
|
|
// If we're collapsing, then keep the collapsing tab in the tab order.
|
|
if ( toShow.length && toHide.length ) {
|
|
eventData.oldTab.attr( "tabIndex", -1 );
|
|
} else if ( toShow.length ) {
|
|
this.tabs.filter(function() {
|
|
return $( this ).attr( "tabIndex" ) === 0;
|
|
})
|
|
.attr( "tabIndex", -1 );
|
|
}
|
|
|
|
toShow.attr( "aria-hidden", "false" );
|
|
eventData.newTab.attr({
|
|
"aria-selected": "true",
|
|
"aria-expanded": "true",
|
|
tabIndex: 0
|
|
});
|
|
},
|
|
|
|
_activate: function( index ) {
|
|
var anchor,
|
|
active = this._findActive( index );
|
|
|
|
// trying to activate the already active panel
|
|
if ( active[ 0 ] === this.active[ 0 ] ) {
|
|
return;
|
|
}
|
|
|
|
// trying to collapse, simulate a click on the current active header
|
|
if ( !active.length ) {
|
|
active = this.active;
|
|
}
|
|
|
|
anchor = active.find( ".ui-tabs-anchor" )[ 0 ];
|
|
this._eventHandler({
|
|
target: anchor,
|
|
currentTarget: anchor,
|
|
preventDefault: $.noop
|
|
});
|
|
},
|
|
|
|
_findActive: function( index ) {
|
|
return index === false ? $() : this.tabs.eq( index );
|
|
},
|
|
|
|
_getIndex: function( index ) {
|
|
// meta-function to give users option to provide a href string instead of a numerical index.
|
|
if ( typeof index === "string" ) {
|
|
index = this.anchors.index( this.anchors.filter( "[href$='" + index + "']" ) );
|
|
}
|
|
|
|
return index;
|
|
},
|
|
|
|
_destroy: function() {
|
|
if ( this.xhr ) {
|
|
this.xhr.abort();
|
|
}
|
|
|
|
this.element.removeClass( "ui-tabs ui-widget ui-widget-content ui-corner-all ui-tabs-collapsible" );
|
|
|
|
this.tablist
|
|
.removeClass( "ui-tabs-nav ui-helper-reset ui-helper-clearfix ui-widget-header ui-corner-all" )
|
|
.removeAttr( "role" );
|
|
|
|
this.anchors
|
|
.removeClass( "ui-tabs-anchor" )
|
|
.removeAttr( "role" )
|
|
.removeAttr( "tabIndex" )
|
|
.removeUniqueId();
|
|
|
|
this.tablist.unbind( this.eventNamespace );
|
|
|
|
this.tabs.add( this.panels ).each(function() {
|
|
if ( $.data( this, "ui-tabs-destroy" ) ) {
|
|
$( this ).remove();
|
|
} else {
|
|
$( this )
|
|
.removeClass( "ui-state-default ui-state-active ui-state-disabled " +
|
|
"ui-corner-top ui-corner-bottom ui-widget-content ui-tabs-active ui-tabs-panel" )
|
|
.removeAttr( "tabIndex" )
|
|
.removeAttr( "aria-live" )
|
|
.removeAttr( "aria-busy" )
|
|
.removeAttr( "aria-selected" )
|
|
.removeAttr( "aria-labelledby" )
|
|
.removeAttr( "aria-hidden" )
|
|
.removeAttr( "aria-expanded" )
|
|
.removeAttr( "role" );
|
|
}
|
|
});
|
|
|
|
this.tabs.each(function() {
|
|
var li = $( this ),
|
|
prev = li.data( "ui-tabs-aria-controls" );
|
|
if ( prev ) {
|
|
li
|
|
.attr( "aria-controls", prev )
|
|
.removeData( "ui-tabs-aria-controls" );
|
|
} else {
|
|
li.removeAttr( "aria-controls" );
|
|
}
|
|
});
|
|
|
|
this.panels.show();
|
|
|
|
if ( this.options.heightStyle !== "content" ) {
|
|
this.panels.css( "height", "" );
|
|
}
|
|
},
|
|
|
|
enable: function( index ) {
|
|
var disabled = this.options.disabled;
|
|
if ( disabled === false ) {
|
|
return;
|
|
}
|
|
|
|
if ( index === undefined ) {
|
|
disabled = false;
|
|
} else {
|
|
index = this._getIndex( index );
|
|
if ( $.isArray( disabled ) ) {
|
|
disabled = $.map( disabled, function( num ) {
|
|
return num !== index ? num : null;
|
|
});
|
|
} else {
|
|
disabled = $.map( this.tabs, function( li, num ) {
|
|
return num !== index ? num : null;
|
|
});
|
|
}
|
|
}
|
|
this._setupDisabled( disabled );
|
|
},
|
|
|
|
disable: function( index ) {
|
|
var disabled = this.options.disabled;
|
|
if ( disabled === true ) {
|
|
return;
|
|
}
|
|
|
|
if ( index === undefined ) {
|
|
disabled = true;
|
|
} else {
|
|
index = this._getIndex( index );
|
|
if ( $.inArray( index, disabled ) !== -1 ) {
|
|
return;
|
|
}
|
|
if ( $.isArray( disabled ) ) {
|
|
disabled = $.merge( [ index ], disabled ).sort();
|
|
} else {
|
|
disabled = [ index ];
|
|
}
|
|
}
|
|
this._setupDisabled( disabled );
|
|
},
|
|
|
|
load: function( index, event ) {
|
|
index = this._getIndex( index );
|
|
var that = this,
|
|
tab = this.tabs.eq( index ),
|
|
anchor = tab.find( ".ui-tabs-anchor" ),
|
|
panel = this._getPanelForTab( tab ),
|
|
eventData = {
|
|
tab: tab,
|
|
panel: panel
|
|
},
|
|
complete = function( jqXHR, status ) {
|
|
if ( status === "abort" ) {
|
|
that.panels.stop( false, true );
|
|
}
|
|
|
|
tab.removeClass( "ui-tabs-loading" );
|
|
panel.removeAttr( "aria-busy" );
|
|
|
|
if ( jqXHR === that.xhr ) {
|
|
delete that.xhr;
|
|
}
|
|
};
|
|
|
|
// not remote
|
|
if ( this._isLocal( anchor[ 0 ] ) ) {
|
|
return;
|
|
}
|
|
|
|
this.xhr = $.ajax( this._ajaxSettings( anchor, event, eventData ) );
|
|
|
|
// support: jQuery <1.8
|
|
// jQuery <1.8 returns false if the request is canceled in beforeSend,
|
|
// but as of 1.8, $.ajax() always returns a jqXHR object.
|
|
if ( this.xhr && this.xhr.statusText !== "canceled" ) {
|
|
tab.addClass( "ui-tabs-loading" );
|
|
panel.attr( "aria-busy", "true" );
|
|
|
|
this.xhr
|
|
.done(function( response, status, jqXHR ) {
|
|
// support: jQuery <1.8
|
|
// http://bugs.jquery.com/ticket/11778
|
|
setTimeout(function() {
|
|
panel.html( response );
|
|
that._trigger( "load", event, eventData );
|
|
|
|
complete( jqXHR, status );
|
|
}, 1 );
|
|
})
|
|
.fail(function( jqXHR, status ) {
|
|
// support: jQuery <1.8
|
|
// http://bugs.jquery.com/ticket/11778
|
|
setTimeout(function() {
|
|
complete( jqXHR, status );
|
|
}, 1 );
|
|
});
|
|
}
|
|
},
|
|
|
|
_ajaxSettings: function( anchor, event, eventData ) {
|
|
var that = this;
|
|
return {
|
|
url: anchor.attr( "href" ),
|
|
beforeSend: function( jqXHR, settings ) {
|
|
return that._trigger( "beforeLoad", event,
|
|
$.extend( { jqXHR: jqXHR, ajaxSettings: settings }, eventData ) );
|
|
}
|
|
};
|
|
},
|
|
|
|
_getPanelForTab: function( tab ) {
|
|
var id = $( tab ).attr( "aria-controls" );
|
|
return this.element.find( this._sanitizeSelector( "#" + id ) );
|
|
}
|
|
});
|
|
|
|
|
|
/*!
|
|
* jQuery UI Tooltip 1.11.4
|
|
* http://jqueryui.com
|
|
*
|
|
* Copyright jQuery Foundation and other contributors
|
|
* Released under the MIT license.
|
|
* http://jquery.org/license
|
|
*
|
|
* http://api.jqueryui.com/tooltip/
|
|
*/
|
|
|
|
|
|
var tooltip = $.widget( "ui.tooltip", {
|
|
version: "1.11.4",
|
|
options: {
|
|
content: function() {
|
|
// support: IE<9, Opera in jQuery <1.7
|
|
// .text() can't accept undefined, so coerce to a string
|
|
var title = $( this ).attr( "title" ) || "";
|
|
// Escape title, since we're going from an attribute to raw HTML
|
|
return $( "<a>" ).text( title ).html();
|
|
},
|
|
hide: true,
|
|
// Disabled elements have inconsistent behavior across browsers (#8661)
|
|
items: "[title]:not([disabled])",
|
|
position: {
|
|
my: "left top+15",
|
|
at: "left bottom",
|
|
collision: "flipfit flip"
|
|
},
|
|
show: true,
|
|
tooltipClass: null,
|
|
track: false,
|
|
|
|
// callbacks
|
|
close: null,
|
|
open: null
|
|
},
|
|
|
|
_addDescribedBy: function( elem, id ) {
|
|
var describedby = (elem.attr( "aria-describedby" ) || "").split( /\s+/ );
|
|
describedby.push( id );
|
|
elem
|
|
.data( "ui-tooltip-id", id )
|
|
.attr( "aria-describedby", $.trim( describedby.join( " " ) ) );
|
|
},
|
|
|
|
_removeDescribedBy: function( elem ) {
|
|
var id = elem.data( "ui-tooltip-id" ),
|
|
describedby = (elem.attr( "aria-describedby" ) || "").split( /\s+/ ),
|
|
index = $.inArray( id, describedby );
|
|
|
|
if ( index !== -1 ) {
|
|
describedby.splice( index, 1 );
|
|
}
|
|
|
|
elem.removeData( "ui-tooltip-id" );
|
|
describedby = $.trim( describedby.join( " " ) );
|
|
if ( describedby ) {
|
|
elem.attr( "aria-describedby", describedby );
|
|
} else {
|
|
elem.removeAttr( "aria-describedby" );
|
|
}
|
|
},
|
|
|
|
_create: function() {
|
|
this._on({
|
|
mouseover: "open",
|
|
focusin: "open"
|
|
});
|
|
|
|
// IDs of generated tooltips, needed for destroy
|
|
this.tooltips = {};
|
|
|
|
// IDs of parent tooltips where we removed the title attribute
|
|
this.parents = {};
|
|
|
|
if ( this.options.disabled ) {
|
|
this._disable();
|
|
}
|
|
|
|
// Append the aria-live region so tooltips announce correctly
|
|
this.liveRegion = $( "<div>" )
|
|
.attr({
|
|
role: "log",
|
|
"aria-live": "assertive",
|
|
"aria-relevant": "additions"
|
|
})
|
|
.addClass( "ui-helper-hidden-accessible" )
|
|
.appendTo( this.document[ 0 ].body );
|
|
},
|
|
|
|
_setOption: function( key, value ) {
|
|
var that = this;
|
|
|
|
if ( key === "disabled" ) {
|
|
this[ value ? "_disable" : "_enable" ]();
|
|
this.options[ key ] = value;
|
|
// disable element style changes
|
|
return;
|
|
}
|
|
|
|
this._super( key, value );
|
|
|
|
if ( key === "content" ) {
|
|
$.each( this.tooltips, function( id, tooltipData ) {
|
|
that._updateContent( tooltipData.element );
|
|
});
|
|
}
|
|
},
|
|
|
|
_disable: function() {
|
|
var that = this;
|
|
|
|
// close open tooltips
|
|
$.each( this.tooltips, function( id, tooltipData ) {
|
|
var event = $.Event( "blur" );
|
|
event.target = event.currentTarget = tooltipData.element[ 0 ];
|
|
that.close( event, true );
|
|
});
|
|
|
|
// remove title attributes to prevent native tooltips
|
|
this.element.find( this.options.items ).addBack().each(function() {
|
|
var element = $( this );
|
|
if ( element.is( "[title]" ) ) {
|
|
element
|
|
.data( "ui-tooltip-title", element.attr( "title" ) )
|
|
.removeAttr( "title" );
|
|
}
|
|
});
|
|
},
|
|
|
|
_enable: function() {
|
|
// restore title attributes
|
|
this.element.find( this.options.items ).addBack().each(function() {
|
|
var element = $( this );
|
|
if ( element.data( "ui-tooltip-title" ) ) {
|
|
element.attr( "title", element.data( "ui-tooltip-title" ) );
|
|
}
|
|
});
|
|
},
|
|
|
|
open: function( event ) {
|
|
var that = this,
|
|
target = $( event ? event.target : this.element )
|
|
// we need closest here due to mouseover bubbling,
|
|
// but always pointing at the same event target
|
|
.closest( this.options.items );
|
|
|
|
// No element to show a tooltip for or the tooltip is already open
|
|
if ( !target.length || target.data( "ui-tooltip-id" ) ) {
|
|
return;
|
|
}
|
|
|
|
if ( target.attr( "title" ) ) {
|
|
target.data( "ui-tooltip-title", target.attr( "title" ) );
|
|
}
|
|
|
|
target.data( "ui-tooltip-open", true );
|
|
|
|
// kill parent tooltips, custom or native, for hover
|
|
if ( event && event.type === "mouseover" ) {
|
|
target.parents().each(function() {
|
|
var parent = $( this ),
|
|
blurEvent;
|
|
if ( parent.data( "ui-tooltip-open" ) ) {
|
|
blurEvent = $.Event( "blur" );
|
|
blurEvent.target = blurEvent.currentTarget = this;
|
|
that.close( blurEvent, true );
|
|
}
|
|
if ( parent.attr( "title" ) ) {
|
|
parent.uniqueId();
|
|
that.parents[ this.id ] = {
|
|
element: this,
|
|
title: parent.attr( "title" )
|
|
};
|
|
parent.attr( "title", "" );
|
|
}
|
|
});
|
|
}
|
|
|
|
this._registerCloseHandlers( event, target );
|
|
this._updateContent( target, event );
|
|
},
|
|
|
|
_updateContent: function( target, event ) {
|
|
var content,
|
|
contentOption = this.options.content,
|
|
that = this,
|
|
eventType = event ? event.type : null;
|
|
|
|
if ( typeof contentOption === "string" ) {
|
|
return this._open( event, target, contentOption );
|
|
}
|
|
|
|
content = contentOption.call( target[0], function( response ) {
|
|
|
|
// IE may instantly serve a cached response for ajax requests
|
|
// delay this call to _open so the other call to _open runs first
|
|
that._delay(function() {
|
|
|
|
// Ignore async response if tooltip was closed already
|
|
if ( !target.data( "ui-tooltip-open" ) ) {
|
|
return;
|
|
}
|
|
|
|
// jQuery creates a special event for focusin when it doesn't
|
|
// exist natively. To improve performance, the native event
|
|
// object is reused and the type is changed. Therefore, we can't
|
|
// rely on the type being correct after the event finished
|
|
// bubbling, so we set it back to the previous value. (#8740)
|
|
if ( event ) {
|
|
event.type = eventType;
|
|
}
|
|
this._open( event, target, response );
|
|
});
|
|
});
|
|
if ( content ) {
|
|
this._open( event, target, content );
|
|
}
|
|
},
|
|
|
|
_open: function( event, target, content ) {
|
|
var tooltipData, tooltip, delayedShow, a11yContent,
|
|
positionOption = $.extend( {}, this.options.position );
|
|
|
|
if ( !content ) {
|
|
return;
|
|
}
|
|
|
|
// Content can be updated multiple times. If the tooltip already
|
|
// exists, then just update the content and bail.
|
|
tooltipData = this._find( target );
|
|
if ( tooltipData ) {
|
|
tooltipData.tooltip.find( ".ui-tooltip-content" ).html( content );
|
|
return;
|
|
}
|
|
|
|
// if we have a title, clear it to prevent the native tooltip
|
|
// we have to check first to avoid defining a title if none exists
|
|
// (we don't want to cause an element to start matching [title])
|
|
//
|
|
// We use removeAttr only for key events, to allow IE to export the correct
|
|
// accessible attributes. For mouse events, set to empty string to avoid
|
|
// native tooltip showing up (happens only when removing inside mouseover).
|
|
if ( target.is( "[title]" ) ) {
|
|
if ( event && event.type === "mouseover" ) {
|
|
target.attr( "title", "" );
|
|
} else {
|
|
target.removeAttr( "title" );
|
|
}
|
|
}
|
|
|
|
tooltipData = this._tooltip( target );
|
|
tooltip = tooltipData.tooltip;
|
|
this._addDescribedBy( target, tooltip.attr( "id" ) );
|
|
tooltip.find( ".ui-tooltip-content" ).html( content );
|
|
|
|
// Support: Voiceover on OS X, JAWS on IE <= 9
|
|
// JAWS announces deletions even when aria-relevant="additions"
|
|
// Voiceover will sometimes re-read the entire log region's contents from the beginning
|
|
this.liveRegion.children().hide();
|
|
if ( content.clone ) {
|
|
a11yContent = content.clone();
|
|
a11yContent.removeAttr( "id" ).find( "[id]" ).removeAttr( "id" );
|
|
} else {
|
|
a11yContent = content;
|
|
}
|
|
$( "<div>" ).html( a11yContent ).appendTo( this.liveRegion );
|
|
|
|
function position( event ) {
|
|
positionOption.of = event;
|
|
if ( tooltip.is( ":hidden" ) ) {
|
|
return;
|
|
}
|
|
tooltip.position( positionOption );
|
|
}
|
|
if ( this.options.track && event && /^mouse/.test( event.type ) ) {
|
|
this._on( this.document, {
|
|
mousemove: position
|
|
});
|
|
// trigger once to override element-relative positioning
|
|
position( event );
|
|
} else {
|
|
tooltip.position( $.extend({
|
|
of: target
|
|
}, this.options.position ) );
|
|
}
|
|
|
|
tooltip.hide();
|
|
|
|
this._show( tooltip, this.options.show );
|
|
// Handle tracking tooltips that are shown with a delay (#8644). As soon
|
|
// as the tooltip is visible, position the tooltip using the most recent
|
|
// event.
|
|
if ( this.options.show && this.options.show.delay ) {
|
|
delayedShow = this.delayedShow = setInterval(function() {
|
|
if ( tooltip.is( ":visible" ) ) {
|
|
position( positionOption.of );
|
|
clearInterval( delayedShow );
|
|
}
|
|
}, $.fx.interval );
|
|
}
|
|
|
|
this._trigger( "open", event, { tooltip: tooltip } );
|
|
},
|
|
|
|
_registerCloseHandlers: function( event, target ) {
|
|
var events = {
|
|
keyup: function( event ) {
|
|
if ( event.keyCode === $.ui.keyCode.ESCAPE ) {
|
|
var fakeEvent = $.Event(event);
|
|
fakeEvent.currentTarget = target[0];
|
|
this.close( fakeEvent, true );
|
|
}
|
|
}
|
|
};
|
|
|
|
// Only bind remove handler for delegated targets. Non-delegated
|
|
// tooltips will handle this in destroy.
|
|
if ( target[ 0 ] !== this.element[ 0 ] ) {
|
|
events.remove = function() {
|
|
this._removeTooltip( this._find( target ).tooltip );
|
|
};
|
|
}
|
|
|
|
if ( !event || event.type === "mouseover" ) {
|
|
events.mouseleave = "close";
|
|
}
|
|
if ( !event || event.type === "focusin" ) {
|
|
events.focusout = "close";
|
|
}
|
|
this._on( true, target, events );
|
|
},
|
|
|
|
close: function( event ) {
|
|
var tooltip,
|
|
that = this,
|
|
target = $( event ? event.currentTarget : this.element ),
|
|
tooltipData = this._find( target );
|
|
|
|
// The tooltip may already be closed
|
|
if ( !tooltipData ) {
|
|
|
|
// We set ui-tooltip-open immediately upon open (in open()), but only set the
|
|
// additional data once there's actually content to show (in _open()). So even if the
|
|
// tooltip doesn't have full data, we always remove ui-tooltip-open in case we're in
|
|
// the period between open() and _open().
|
|
target.removeData( "ui-tooltip-open" );
|
|
return;
|
|
}
|
|
|
|
tooltip = tooltipData.tooltip;
|
|
|
|
// disabling closes the tooltip, so we need to track when we're closing
|
|
// to avoid an infinite loop in case the tooltip becomes disabled on close
|
|
if ( tooltipData.closing ) {
|
|
return;
|
|
}
|
|
|
|
// Clear the interval for delayed tracking tooltips
|
|
clearInterval( this.delayedShow );
|
|
|
|
// only set title if we had one before (see comment in _open())
|
|
// If the title attribute has changed since open(), don't restore
|
|
if ( target.data( "ui-tooltip-title" ) && !target.attr( "title" ) ) {
|
|
target.attr( "title", target.data( "ui-tooltip-title" ) );
|
|
}
|
|
|
|
this._removeDescribedBy( target );
|
|
|
|
tooltipData.hiding = true;
|
|
tooltip.stop( true );
|
|
this._hide( tooltip, this.options.hide, function() {
|
|
that._removeTooltip( $( this ) );
|
|
});
|
|
|
|
target.removeData( "ui-tooltip-open" );
|
|
this._off( target, "mouseleave focusout keyup" );
|
|
|
|
// Remove 'remove' binding only on delegated targets
|
|
if ( target[ 0 ] !== this.element[ 0 ] ) {
|
|
this._off( target, "remove" );
|
|
}
|
|
this._off( this.document, "mousemove" );
|
|
|
|
if ( event && event.type === "mouseleave" ) {
|
|
$.each( this.parents, function( id, parent ) {
|
|
$( parent.element ).attr( "title", parent.title );
|
|
delete that.parents[ id ];
|
|
});
|
|
}
|
|
|
|
tooltipData.closing = true;
|
|
this._trigger( "close", event, { tooltip: tooltip } );
|
|
if ( !tooltipData.hiding ) {
|
|
tooltipData.closing = false;
|
|
}
|
|
},
|
|
|
|
_tooltip: function( element ) {
|
|
var tooltip = $( "<div>" )
|
|
.attr( "role", "tooltip" )
|
|
.addClass( "ui-tooltip ui-widget ui-corner-all ui-widget-content " +
|
|
( this.options.tooltipClass || "" ) ),
|
|
id = tooltip.uniqueId().attr( "id" );
|
|
|
|
$( "<div>" )
|
|
.addClass( "ui-tooltip-content" )
|
|
.appendTo( tooltip );
|
|
|
|
tooltip.appendTo( this.document[0].body );
|
|
|
|
return this.tooltips[ id ] = {
|
|
element: element,
|
|
tooltip: tooltip
|
|
};
|
|
},
|
|
|
|
_find: function( target ) {
|
|
var id = target.data( "ui-tooltip-id" );
|
|
return id ? this.tooltips[ id ] : null;
|
|
},
|
|
|
|
_removeTooltip: function( tooltip ) {
|
|
tooltip.remove();
|
|
delete this.tooltips[ tooltip.attr( "id" ) ];
|
|
},
|
|
|
|
_destroy: function() {
|
|
var that = this;
|
|
|
|
// close open tooltips
|
|
$.each( this.tooltips, function( id, tooltipData ) {
|
|
// Delegate to close method to handle common cleanup
|
|
var event = $.Event( "blur" ),
|
|
element = tooltipData.element;
|
|
event.target = event.currentTarget = element[ 0 ];
|
|
that.close( event, true );
|
|
|
|
// Remove immediately; destroying an open tooltip doesn't use the
|
|
// hide animation
|
|
$( "#" + id ).remove();
|
|
|
|
// Restore the title
|
|
if ( element.data( "ui-tooltip-title" ) ) {
|
|
// If the title attribute has changed since open(), don't restore
|
|
if ( !element.attr( "title" ) ) {
|
|
element.attr( "title", element.data( "ui-tooltip-title" ) );
|
|
}
|
|
element.removeData( "ui-tooltip-title" );
|
|
}
|
|
});
|
|
this.liveRegion.remove();
|
|
}
|
|
});
|
|
|
|
|
|
|
|
}));
|
|
/*!
|
|
* jQuery Validation Plugin v1.13.1
|
|
*
|
|
* http://jqueryvalidation.org/
|
|
*
|
|
* Copyright (c) 2014 Jörn Zaefferer
|
|
* Released under the MIT license
|
|
*/
|
|
(function( factory ) {
|
|
if ( typeof define === "function" && define.amd ) {
|
|
define( ["jquery"], factory );
|
|
} else {
|
|
factory( jQuery );
|
|
}
|
|
}(function( $ ) {
|
|
|
|
$.extend($.fn, {
|
|
// http://jqueryvalidation.org/validate/
|
|
validate: function( options ) {
|
|
|
|
// if nothing is selected, return nothing; can't chain anyway
|
|
if ( !this.length ) {
|
|
if ( options && options.debug && window.console ) {
|
|
console.warn( "Nothing selected, can't validate, returning nothing." );
|
|
}
|
|
return;
|
|
}
|
|
|
|
// check if a validator for this form was already created
|
|
var validator = $.data( this[ 0 ], "validator" );
|
|
if ( validator ) {
|
|
return validator;
|
|
}
|
|
|
|
// Add novalidate tag if HTML5.
|
|
this.attr( "novalidate", "novalidate" );
|
|
|
|
validator = new $.validator( options, this[ 0 ] );
|
|
$.data( this[ 0 ], "validator", validator );
|
|
|
|
if ( validator.settings.onsubmit ) {
|
|
|
|
this.validateDelegate( ":submit", "click", function( event ) {
|
|
if ( validator.settings.submitHandler ) {
|
|
validator.submitButton = event.target;
|
|
}
|
|
// allow suppressing validation by adding a cancel class to the submit button
|
|
if ( $( event.target ).hasClass( "cancel" ) ) {
|
|
validator.cancelSubmit = true;
|
|
}
|
|
|
|
// allow suppressing validation by adding the html5 formnovalidate attribute to the submit button
|
|
if ( $( event.target ).attr( "formnovalidate" ) !== undefined ) {
|
|
validator.cancelSubmit = true;
|
|
}
|
|
});
|
|
|
|
// validate the form on submit
|
|
this.submit( function( event ) {
|
|
if ( validator.settings.debug ) {
|
|
// prevent form submit to be able to see console output
|
|
event.preventDefault();
|
|
}
|
|
function handle() {
|
|
var hidden, result;
|
|
if ( validator.settings.submitHandler ) {
|
|
if ( validator.submitButton ) {
|
|
// insert a hidden input as a replacement for the missing submit button
|
|
hidden = $( "<input type='hidden'/>" )
|
|
.attr( "name", validator.submitButton.name )
|
|
.val( $( validator.submitButton ).val() )
|
|
.appendTo( validator.currentForm );
|
|
}
|
|
result = validator.settings.submitHandler.call( validator, validator.currentForm, event );
|
|
if ( validator.submitButton ) {
|
|
// and clean up afterwards; thanks to no-block-scope, hidden can be referenced
|
|
hidden.remove();
|
|
}
|
|
if ( result !== undefined ) {
|
|
return result;
|
|
}
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
// prevent submit for invalid forms or custom submit handlers
|
|
if ( validator.cancelSubmit ) {
|
|
validator.cancelSubmit = false;
|
|
return handle();
|
|
}
|
|
if ( validator.form() ) {
|
|
if ( validator.pendingRequest ) {
|
|
validator.formSubmitted = true;
|
|
return false;
|
|
}
|
|
return handle();
|
|
} else {
|
|
validator.focusInvalid();
|
|
return false;
|
|
}
|
|
});
|
|
}
|
|
|
|
return validator;
|
|
},
|
|
// http://jqueryvalidation.org/valid/
|
|
valid: function() {
|
|
var valid, validator;
|
|
|
|
if ( $( this[ 0 ] ).is( "form" ) ) {
|
|
valid = this.validate().form();
|
|
} else {
|
|
valid = true;
|
|
validator = $( this[ 0 ].form ).validate();
|
|
this.each( function() {
|
|
valid = validator.element( this ) && valid;
|
|
});
|
|
}
|
|
return valid;
|
|
},
|
|
// attributes: space separated list of attributes to retrieve and remove
|
|
removeAttrs: function( attributes ) {
|
|
var result = {},
|
|
$element = this;
|
|
$.each( attributes.split( /\s/ ), function( index, value ) {
|
|
result[ value ] = $element.attr( value );
|
|
$element.removeAttr( value );
|
|
});
|
|
return result;
|
|
},
|
|
// http://jqueryvalidation.org/rules/
|
|
rules: function( command, argument ) {
|
|
var element = this[ 0 ],
|
|
settings, staticRules, existingRules, data, param, filtered;
|
|
|
|
if ( command ) {
|
|
settings = $.data( element.form, "validator" ).settings;
|
|
staticRules = settings.rules;
|
|
existingRules = $.validator.staticRules( element );
|
|
switch ( command ) {
|
|
case "add":
|
|
$.extend( existingRules, $.validator.normalizeRule( argument ) );
|
|
// remove messages from rules, but allow them to be set separately
|
|
delete existingRules.messages;
|
|
staticRules[ element.name ] = existingRules;
|
|
if ( argument.messages ) {
|
|
settings.messages[ element.name ] = $.extend( settings.messages[ element.name ], argument.messages );
|
|
}
|
|
break;
|
|
case "remove":
|
|
if ( !argument ) {
|
|
delete staticRules[ element.name ];
|
|
return existingRules;
|
|
}
|
|
filtered = {};
|
|
$.each( argument.split( /\s/ ), function( index, method ) {
|
|
filtered[ method ] = existingRules[ method ];
|
|
delete existingRules[ method ];
|
|
if ( method === "required" ) {
|
|
$( element ).removeAttr( "aria-required" );
|
|
}
|
|
});
|
|
return filtered;
|
|
}
|
|
}
|
|
|
|
data = $.validator.normalizeRules(
|
|
$.extend(
|
|
{},
|
|
$.validator.classRules( element ),
|
|
$.validator.attributeRules( element ),
|
|
$.validator.dataRules( element ),
|
|
$.validator.staticRules( element )
|
|
), element );
|
|
|
|
// make sure required is at front
|
|
if ( data.required ) {
|
|
param = data.required;
|
|
delete data.required;
|
|
data = $.extend( { required: param }, data );
|
|
$( element ).attr( "aria-required", "true" );
|
|
}
|
|
|
|
// make sure remote is at back
|
|
if ( data.remote ) {
|
|
param = data.remote;
|
|
delete data.remote;
|
|
data = $.extend( data, { remote: param });
|
|
}
|
|
|
|
return data;
|
|
}
|
|
});
|
|
|
|
// Custom selectors
|
|
$.extend( $.expr[ ":" ], {
|
|
// http://jqueryvalidation.org/blank-selector/
|
|
blank: function( a ) {
|
|
return !$.trim( "" + $( a ).val() );
|
|
},
|
|
// http://jqueryvalidation.org/filled-selector/
|
|
filled: function( a ) {
|
|
return !!$.trim( "" + $( a ).val() );
|
|
},
|
|
// http://jqueryvalidation.org/unchecked-selector/
|
|
unchecked: function( a ) {
|
|
return !$( a ).prop( "checked" );
|
|
}
|
|
});
|
|
|
|
// constructor for validator
|
|
$.validator = function( options, form ) {
|
|
this.settings = $.extend( true, {}, $.validator.defaults, options );
|
|
this.currentForm = form;
|
|
this.init();
|
|
};
|
|
|
|
// http://jqueryvalidation.org/jQuery.validator.format/
|
|
$.validator.format = function( source, params ) {
|
|
if ( arguments.length === 1 ) {
|
|
return function() {
|
|
var args = $.makeArray( arguments );
|
|
args.unshift( source );
|
|
return $.validator.format.apply( this, args );
|
|
};
|
|
}
|
|
if ( arguments.length > 2 && params.constructor !== Array ) {
|
|
params = $.makeArray( arguments ).slice( 1 );
|
|
}
|
|
if ( params.constructor !== Array ) {
|
|
params = [ params ];
|
|
}
|
|
$.each( params, function( i, n ) {
|
|
source = source.replace( new RegExp( "\\{" + i + "\\}", "g" ), function() {
|
|
return n;
|
|
});
|
|
});
|
|
return source;
|
|
};
|
|
|
|
$.extend( $.validator, {
|
|
|
|
defaults: {
|
|
messages: {},
|
|
groups: {},
|
|
rules: {},
|
|
errorClass: "error",
|
|
validClass: "valid",
|
|
errorElement: "label",
|
|
focusCleanup: false,
|
|
focusInvalid: true,
|
|
errorContainer: $( [] ),
|
|
errorLabelContainer: $( [] ),
|
|
onsubmit: true,
|
|
ignore: ":hidden",
|
|
ignoreTitle: false,
|
|
onfocusin: function( element ) {
|
|
this.lastActive = element;
|
|
|
|
// Hide error label and remove error class on focus if enabled
|
|
if ( this.settings.focusCleanup ) {
|
|
if ( this.settings.unhighlight ) {
|
|
this.settings.unhighlight.call( this, element, this.settings.errorClass, this.settings.validClass );
|
|
}
|
|
this.hideThese( this.errorsFor( element ) );
|
|
}
|
|
},
|
|
onfocusout: function( element ) {
|
|
if ( !this.checkable( element ) && ( element.name in this.submitted || !this.optional( element ) ) ) {
|
|
this.element( element );
|
|
}
|
|
},
|
|
onkeyup: function( element, event ) {
|
|
if ( event.which === 9 && this.elementValue( element ) === "" ) {
|
|
return;
|
|
} else if ( element.name in this.submitted || element === this.lastElement ) {
|
|
this.element( element );
|
|
}
|
|
},
|
|
onclick: function( element ) {
|
|
// click on selects, radiobuttons and checkboxes
|
|
if ( element.name in this.submitted ) {
|
|
this.element( element );
|
|
|
|
// or option elements, check parent select in that case
|
|
} else if ( element.parentNode.name in this.submitted ) {
|
|
this.element( element.parentNode );
|
|
}
|
|
},
|
|
highlight: function( element, errorClass, validClass ) {
|
|
if ( element.type === "radio" ) {
|
|
this.findByName( element.name ).addClass( errorClass ).removeClass( validClass );
|
|
} else {
|
|
$( element ).addClass( errorClass ).removeClass( validClass );
|
|
}
|
|
},
|
|
unhighlight: function( element, errorClass, validClass ) {
|
|
if ( element.type === "radio" ) {
|
|
this.findByName( element.name ).removeClass( errorClass ).addClass( validClass );
|
|
} else {
|
|
$( element ).removeClass( errorClass ).addClass( validClass );
|
|
}
|
|
}
|
|
},
|
|
|
|
// http://jqueryvalidation.org/jQuery.validator.setDefaults/
|
|
setDefaults: function( settings ) {
|
|
$.extend( $.validator.defaults, settings );
|
|
},
|
|
|
|
messages: {
|
|
required: "This field is required.",
|
|
remote: "Please fix this field.",
|
|
email: "Please enter a valid email address.",
|
|
url: "Please enter a valid URL.",
|
|
date: "Please enter a valid date.",
|
|
dateISO: "Please enter a valid date ( ISO ).",
|
|
number: "Please enter a valid number.",
|
|
digits: "Please enter only digits.",
|
|
creditcard: "Please enter a valid credit card number.",
|
|
equalTo: "Please enter the same value again.",
|
|
maxlength: $.validator.format( "Please enter no more than {0} characters." ),
|
|
minlength: $.validator.format( "Please enter at least {0} characters." ),
|
|
rangelength: $.validator.format( "Please enter a value between {0} and {1} characters long." ),
|
|
range: $.validator.format( "Please enter a value between {0} and {1}." ),
|
|
max: $.validator.format( "Please enter a value less than or equal to {0}." ),
|
|
min: $.validator.format( "Please enter a value greater than or equal to {0}." )
|
|
},
|
|
|
|
autoCreateRanges: false,
|
|
|
|
prototype: {
|
|
|
|
init: function() {
|
|
this.labelContainer = $( this.settings.errorLabelContainer );
|
|
this.errorContext = this.labelContainer.length && this.labelContainer || $( this.currentForm );
|
|
this.containers = $( this.settings.errorContainer ).add( this.settings.errorLabelContainer );
|
|
this.submitted = {};
|
|
this.valueCache = {};
|
|
this.pendingRequest = 0;
|
|
this.pending = {};
|
|
this.invalid = {};
|
|
this.reset();
|
|
|
|
var groups = ( this.groups = {} ),
|
|
rules;
|
|
$.each( this.settings.groups, function( key, value ) {
|
|
if ( typeof value === "string" ) {
|
|
value = value.split( /\s/ );
|
|
}
|
|
$.each( value, function( index, name ) {
|
|
groups[ name ] = key;
|
|
});
|
|
});
|
|
rules = this.settings.rules;
|
|
$.each( rules, function( key, value ) {
|
|
rules[ key ] = $.validator.normalizeRule( value );
|
|
});
|
|
|
|
function delegate( event ) {
|
|
var validator = $.data( this[ 0 ].form, "validator" ),
|
|
eventType = "on" + event.type.replace( /^validate/, "" ),
|
|
settings = validator.settings;
|
|
if ( settings[ eventType ] && !this.is( settings.ignore ) ) {
|
|
settings[ eventType ].call( validator, this[ 0 ], event );
|
|
}
|
|
}
|
|
$( this.currentForm )
|
|
.validateDelegate( ":text, [type='password'], [type='file'], select, textarea, " +
|
|
"[type='number'], [type='search'] ,[type='tel'], [type='url'], " +
|
|
"[type='email'], [type='datetime'], [type='date'], [type='month'], " +
|
|
"[type='week'], [type='time'], [type='datetime-local'], " +
|
|
"[type='range'], [type='color'], [type='radio'], [type='checkbox']",
|
|
"focusin focusout keyup", delegate)
|
|
// Support: Chrome, oldIE
|
|
// "select" is provided as event.target when clicking a option
|
|
.validateDelegate("select, option, [type='radio'], [type='checkbox']", "click", delegate);
|
|
|
|
if ( this.settings.invalidHandler ) {
|
|
$( this.currentForm ).bind( "invalid-form.validate", this.settings.invalidHandler );
|
|
}
|
|
|
|
// Add aria-required to any Static/Data/Class required fields before first validation
|
|
// Screen readers require this attribute to be present before the initial submission http://www.w3.org/TR/WCAG-TECHS/ARIA2.html
|
|
$( this.currentForm ).find( "[required], [data-rule-required], .required" ).attr( "aria-required", "true" );
|
|
},
|
|
|
|
// http://jqueryvalidation.org/Validator.form/
|
|
form: function() {
|
|
this.checkForm();
|
|
$.extend( this.submitted, this.errorMap );
|
|
this.invalid = $.extend({}, this.errorMap );
|
|
if ( !this.valid() ) {
|
|
$( this.currentForm ).triggerHandler( "invalid-form", [ this ]);
|
|
}
|
|
this.showErrors();
|
|
return this.valid();
|
|
},
|
|
|
|
checkForm: function() {
|
|
this.prepareForm();
|
|
for ( var i = 0, elements = ( this.currentElements = this.elements() ); elements[ i ]; i++ ) {
|
|
this.check( elements[ i ] );
|
|
}
|
|
return this.valid();
|
|
},
|
|
|
|
// http://jqueryvalidation.org/Validator.element/
|
|
element: function( element ) {
|
|
var cleanElement = this.clean( element ),
|
|
checkElement = this.validationTargetFor( cleanElement ),
|
|
result = true;
|
|
|
|
this.lastElement = checkElement;
|
|
|
|
if ( checkElement === undefined ) {
|
|
delete this.invalid[ cleanElement.name ];
|
|
} else {
|
|
this.prepareElement( checkElement );
|
|
this.currentElements = $( checkElement );
|
|
|
|
result = this.check( checkElement ) !== false;
|
|
if ( result ) {
|
|
delete this.invalid[ checkElement.name ];
|
|
} else {
|
|
this.invalid[ checkElement.name ] = true;
|
|
}
|
|
}
|
|
// Add aria-invalid status for screen readers
|
|
$( element ).attr( "aria-invalid", !result );
|
|
|
|
if ( !this.numberOfInvalids() ) {
|
|
// Hide error containers on last error
|
|
this.toHide = this.toHide.add( this.containers );
|
|
}
|
|
this.showErrors();
|
|
return result;
|
|
},
|
|
|
|
// http://jqueryvalidation.org/Validator.showErrors/
|
|
showErrors: function( errors ) {
|
|
if ( errors ) {
|
|
// add items to error list and map
|
|
$.extend( this.errorMap, errors );
|
|
this.errorList = [];
|
|
for ( var name in errors ) {
|
|
this.errorList.push({
|
|
message: errors[ name ],
|
|
element: this.findByName( name )[ 0 ]
|
|
});
|
|
}
|
|
// remove items from success list
|
|
this.successList = $.grep( this.successList, function( element ) {
|
|
return !( element.name in errors );
|
|
});
|
|
}
|
|
if ( this.settings.showErrors ) {
|
|
this.settings.showErrors.call( this, this.errorMap, this.errorList );
|
|
} else {
|
|
this.defaultShowErrors();
|
|
}
|
|
},
|
|
|
|
// http://jqueryvalidation.org/Validator.resetForm/
|
|
resetForm: function() {
|
|
if ( $.fn.resetForm ) {
|
|
$( this.currentForm ).resetForm();
|
|
}
|
|
this.submitted = {};
|
|
this.lastElement = null;
|
|
this.prepareForm();
|
|
this.hideErrors();
|
|
this.elements()
|
|
.removeClass( this.settings.errorClass )
|
|
.removeData( "previousValue" )
|
|
.removeAttr( "aria-invalid" );
|
|
},
|
|
|
|
numberOfInvalids: function() {
|
|
return this.objectLength( this.invalid );
|
|
},
|
|
|
|
objectLength: function( obj ) {
|
|
/* jshint unused: false */
|
|
var count = 0,
|
|
i;
|
|
for ( i in obj ) {
|
|
count++;
|
|
}
|
|
return count;
|
|
},
|
|
|
|
hideErrors: function() {
|
|
this.hideThese( this.toHide );
|
|
},
|
|
|
|
hideThese: function( errors ) {
|
|
errors.not( this.containers ).text( "" );
|
|
this.addWrapper( errors ).hide();
|
|
},
|
|
|
|
valid: function() {
|
|
return this.size() === 0;
|
|
},
|
|
|
|
size: function() {
|
|
return this.errorList.length;
|
|
},
|
|
|
|
focusInvalid: function() {
|
|
if ( this.settings.focusInvalid ) {
|
|
try {
|
|
$( this.findLastActive() || this.errorList.length && this.errorList[ 0 ].element || [])
|
|
.filter( ":visible" )
|
|
.focus()
|
|
// manually trigger focusin event; without it, focusin handler isn't called, findLastActive won't have anything to find
|
|
.trigger( "focusin" );
|
|
} catch ( e ) {
|
|
// ignore IE throwing errors when focusing hidden elements
|
|
}
|
|
}
|
|
},
|
|
|
|
findLastActive: function() {
|
|
var lastActive = this.lastActive;
|
|
return lastActive && $.grep( this.errorList, function( n ) {
|
|
return n.element.name === lastActive.name;
|
|
}).length === 1 && lastActive;
|
|
},
|
|
|
|
elements: function() {
|
|
var validator = this,
|
|
rulesCache = {};
|
|
|
|
// select all valid inputs inside the form (no submit or reset buttons)
|
|
return $( this.currentForm )
|
|
.find( "input, select, textarea" )
|
|
.not( ":submit, :reset, :image, [disabled], [readonly]" )
|
|
.not( this.settings.ignore )
|
|
.filter( function() {
|
|
if ( !this.name && validator.settings.debug && window.console ) {
|
|
console.error( "%o has no name assigned", this );
|
|
}
|
|
|
|
// select only the first element for each name, and only those with rules specified
|
|
if ( this.name in rulesCache || !validator.objectLength( $( this ).rules() ) ) {
|
|
return false;
|
|
}
|
|
|
|
rulesCache[ this.name ] = true;
|
|
return true;
|
|
});
|
|
},
|
|
|
|
clean: function( selector ) {
|
|
return $( selector )[ 0 ];
|
|
},
|
|
|
|
errors: function() {
|
|
var errorClass = this.settings.errorClass.split( " " ).join( "." );
|
|
return $( this.settings.errorElement + "." + errorClass, this.errorContext );
|
|
},
|
|
|
|
reset: function() {
|
|
this.successList = [];
|
|
this.errorList = [];
|
|
this.errorMap = {};
|
|
this.toShow = $( [] );
|
|
this.toHide = $( [] );
|
|
this.currentElements = $( [] );
|
|
},
|
|
|
|
prepareForm: function() {
|
|
this.reset();
|
|
this.toHide = this.errors().add( this.containers );
|
|
},
|
|
|
|
prepareElement: function( element ) {
|
|
this.reset();
|
|
this.toHide = this.errorsFor( element );
|
|
},
|
|
|
|
elementValue: function( element ) {
|
|
var val,
|
|
$element = $( element ),
|
|
type = element.type;
|
|
|
|
if ( type === "radio" || type === "checkbox" ) {
|
|
return $( "input[name='" + element.name + "']:checked" ).val();
|
|
} else if ( type === "number" && typeof element.validity !== "undefined" ) {
|
|
return element.validity.badInput ? false : $element.val();
|
|
}
|
|
|
|
val = $element.val();
|
|
if ( typeof val === "string" ) {
|
|
return val.replace(/\r/g, "" );
|
|
}
|
|
return val;
|
|
},
|
|
|
|
check: function( element ) {
|
|
element = this.validationTargetFor( this.clean( element ) );
|
|
|
|
var rules = $( element ).rules(),
|
|
rulesCount = $.map( rules, function( n, i ) {
|
|
return i;
|
|
}).length,
|
|
dependencyMismatch = false,
|
|
val = this.elementValue( element ),
|
|
result, method, rule;
|
|
|
|
for ( method in rules ) {
|
|
rule = { method: method, parameters: rules[ method ] };
|
|
try {
|
|
|
|
result = $.validator.methods[ method ].call( this, val, element, rule.parameters );
|
|
|
|
// if a method indicates that the field is optional and therefore valid,
|
|
// don't mark it as valid when there are no other rules
|
|
if ( result === "dependency-mismatch" && rulesCount === 1 ) {
|
|
dependencyMismatch = true;
|
|
continue;
|
|
}
|
|
dependencyMismatch = false;
|
|
|
|
if ( result === "pending" ) {
|
|
this.toHide = this.toHide.not( this.errorsFor( element ) );
|
|
return;
|
|
}
|
|
|
|
if ( !result ) {
|
|
this.formatAndAdd( element, rule );
|
|
return false;
|
|
}
|
|
} catch ( e ) {
|
|
if ( this.settings.debug && window.console ) {
|
|
console.log( "Exception occurred when checking element " + element.id + ", check the '" + rule.method + "' method.", e );
|
|
}
|
|
throw e;
|
|
}
|
|
}
|
|
if ( dependencyMismatch ) {
|
|
return;
|
|
}
|
|
if ( this.objectLength( rules ) ) {
|
|
this.successList.push( element );
|
|
}
|
|
return true;
|
|
},
|
|
|
|
// return the custom message for the given element and validation method
|
|
// specified in the element's HTML5 data attribute
|
|
// return the generic message if present and no method specific message is present
|
|
customDataMessage: function( element, method ) {
|
|
return $( element ).data( "msg" + method.charAt( 0 ).toUpperCase() +
|
|
method.substring( 1 ).toLowerCase() ) || $( element ).data( "msg" );
|
|
},
|
|
|
|
// return the custom message for the given element name and validation method
|
|
customMessage: function( name, method ) {
|
|
var m = this.settings.messages[ name ];
|
|
return m && ( m.constructor === String ? m : m[ method ]);
|
|
},
|
|
|
|
// return the first defined argument, allowing empty strings
|
|
findDefined: function() {
|
|
for ( var i = 0; i < arguments.length; i++) {
|
|
if ( arguments[ i ] !== undefined ) {
|
|
return arguments[ i ];
|
|
}
|
|
}
|
|
return undefined;
|
|
},
|
|
|
|
defaultMessage: function( element, method ) {
|
|
return this.findDefined(
|
|
this.customMessage( element.name, method ),
|
|
this.customDataMessage( element, method ),
|
|
// title is never undefined, so handle empty string as undefined
|
|
!this.settings.ignoreTitle && element.title || undefined,
|
|
$.validator.messages[ method ],
|
|
"<strong>Warning: No message defined for " + element.name + "</strong>"
|
|
);
|
|
},
|
|
|
|
formatAndAdd: function( element, rule ) {
|
|
var message = this.defaultMessage( element, rule.method ),
|
|
theregex = /\$?\{(\d+)\}/g;
|
|
if ( typeof message === "function" ) {
|
|
message = message.call( this, rule.parameters, element );
|
|
} else if ( theregex.test( message ) ) {
|
|
message = $.validator.format( message.replace( theregex, "{$1}" ), rule.parameters );
|
|
}
|
|
this.errorList.push({
|
|
message: message,
|
|
element: element,
|
|
method: rule.method
|
|
});
|
|
|
|
this.errorMap[ element.name ] = message;
|
|
this.submitted[ element.name ] = message;
|
|
},
|
|
|
|
addWrapper: function( toToggle ) {
|
|
if ( this.settings.wrapper ) {
|
|
toToggle = toToggle.add( toToggle.parent( this.settings.wrapper ) );
|
|
}
|
|
return toToggle;
|
|
},
|
|
|
|
defaultShowErrors: function() {
|
|
var i, elements, error;
|
|
for ( i = 0; this.errorList[ i ]; i++ ) {
|
|
error = this.errorList[ i ];
|
|
if ( this.settings.highlight ) {
|
|
this.settings.highlight.call( this, error.element, this.settings.errorClass, this.settings.validClass );
|
|
}
|
|
this.showLabel( error.element, error.message );
|
|
}
|
|
if ( this.errorList.length ) {
|
|
this.toShow = this.toShow.add( this.containers );
|
|
}
|
|
if ( this.settings.success ) {
|
|
for ( i = 0; this.successList[ i ]; i++ ) {
|
|
this.showLabel( this.successList[ i ] );
|
|
}
|
|
}
|
|
if ( this.settings.unhighlight ) {
|
|
for ( i = 0, elements = this.validElements(); elements[ i ]; i++ ) {
|
|
this.settings.unhighlight.call( this, elements[ i ], this.settings.errorClass, this.settings.validClass );
|
|
}
|
|
}
|
|
this.toHide = this.toHide.not( this.toShow );
|
|
this.hideErrors();
|
|
this.addWrapper( this.toShow ).show();
|
|
},
|
|
|
|
validElements: function() {
|
|
return this.currentElements.not( this.invalidElements() );
|
|
},
|
|
|
|
invalidElements: function() {
|
|
return $( this.errorList ).map(function() {
|
|
return this.element;
|
|
});
|
|
},
|
|
|
|
showLabel: function( element, message ) {
|
|
var place, group, errorID,
|
|
error = this.errorsFor( element ),
|
|
elementID = this.idOrName( element ),
|
|
describedBy = $( element ).attr( "aria-describedby" );
|
|
if ( error.length ) {
|
|
// refresh error/success class
|
|
error.removeClass( this.settings.validClass ).addClass( this.settings.errorClass );
|
|
// replace message on existing label
|
|
error.html( message );
|
|
} else {
|
|
// create error element
|
|
error = $( "<" + this.settings.errorElement + ">" )
|
|
.attr( "id", elementID + "-error" )
|
|
.addClass( this.settings.errorClass )
|
|
.html( message || "" );
|
|
|
|
// Maintain reference to the element to be placed into the DOM
|
|
place = error;
|
|
if ( this.settings.wrapper ) {
|
|
// make sure the element is visible, even in IE
|
|
// actually showing the wrapped element is handled elsewhere
|
|
place = error.hide().show().wrap( "<" + this.settings.wrapper + "/>" ).parent();
|
|
}
|
|
if ( this.labelContainer.length ) {
|
|
this.labelContainer.append( place );
|
|
} else if ( this.settings.errorPlacement ) {
|
|
this.settings.errorPlacement( place, $( element ) );
|
|
} else {
|
|
place.insertAfter( element );
|
|
}
|
|
|
|
// Link error back to the element
|
|
if ( error.is( "label" ) ) {
|
|
// If the error is a label, then associate using 'for'
|
|
error.attr( "for", elementID );
|
|
} else if ( error.parents( "label[for='" + elementID + "']" ).length === 0 ) {
|
|
// If the element is not a child of an associated label, then it's necessary
|
|
// to explicitly apply aria-describedby
|
|
|
|
errorID = error.attr( "id" ).replace( /(:|\.|\[|\])/g, "\\$1");
|
|
// Respect existing non-error aria-describedby
|
|
if ( !describedBy ) {
|
|
describedBy = errorID;
|
|
} else if ( !describedBy.match( new RegExp( "\\b" + errorID + "\\b" ) ) ) {
|
|
// Add to end of list if not already present
|
|
describedBy += " " + errorID;
|
|
}
|
|
$( element ).attr( "aria-describedby", describedBy );
|
|
|
|
// If this element is grouped, then assign to all elements in the same group
|
|
group = this.groups[ element.name ];
|
|
if ( group ) {
|
|
$.each( this.groups, function( name, testgroup ) {
|
|
if ( testgroup === group ) {
|
|
$( "[name='" + name + "']", this.currentForm )
|
|
.attr( "aria-describedby", error.attr( "id" ) );
|
|
}
|
|
});
|
|
}
|
|
}
|
|
}
|
|
if ( !message && this.settings.success ) {
|
|
error.text( "" );
|
|
if ( typeof this.settings.success === "string" ) {
|
|
error.addClass( this.settings.success );
|
|
} else {
|
|
this.settings.success( error, element );
|
|
}
|
|
}
|
|
this.toShow = this.toShow.add( error );
|
|
},
|
|
|
|
errorsFor: function( element ) {
|
|
var name = this.idOrName( element ),
|
|
describer = $( element ).attr( "aria-describedby" ),
|
|
selector = "label[for='" + name + "'], label[for='" + name + "'] *";
|
|
|
|
// aria-describedby should directly reference the error element
|
|
if ( describer ) {
|
|
selector = selector + ", #" + describer.replace( /\s+/g, ", #" );
|
|
}
|
|
return this
|
|
.errors()
|
|
.filter( selector );
|
|
},
|
|
|
|
idOrName: function( element ) {
|
|
return this.groups[ element.name ] || ( this.checkable( element ) ? element.name : element.id || element.name );
|
|
},
|
|
|
|
validationTargetFor: function( element ) {
|
|
|
|
// If radio/checkbox, validate first element in group instead
|
|
if ( this.checkable( element ) ) {
|
|
element = this.findByName( element.name );
|
|
}
|
|
|
|
// Always apply ignore filter
|
|
return $( element ).not( this.settings.ignore )[ 0 ];
|
|
},
|
|
|
|
checkable: function( element ) {
|
|
return ( /radio|checkbox/i ).test( element.type );
|
|
},
|
|
|
|
findByName: function( name ) {
|
|
return $( this.currentForm ).find( "[name='" + name + "']" );
|
|
},
|
|
|
|
getLength: function( value, element ) {
|
|
switch ( element.nodeName.toLowerCase() ) {
|
|
case "select":
|
|
return $( "option:selected", element ).length;
|
|
case "input":
|
|
if ( this.checkable( element ) ) {
|
|
return this.findByName( element.name ).filter( ":checked" ).length;
|
|
}
|
|
}
|
|
return value.length;
|
|
},
|
|
|
|
depend: function( param, element ) {
|
|
return this.dependTypes[typeof param] ? this.dependTypes[typeof param]( param, element ) : true;
|
|
},
|
|
|
|
dependTypes: {
|
|
"boolean": function( param ) {
|
|
return param;
|
|
},
|
|
"string": function( param, element ) {
|
|
return !!$( param, element.form ).length;
|
|
},
|
|
"function": function( param, element ) {
|
|
return param( element );
|
|
}
|
|
},
|
|
|
|
optional: function( element ) {
|
|
var val = this.elementValue( element );
|
|
return !$.validator.methods.required.call( this, val, element ) && "dependency-mismatch";
|
|
},
|
|
|
|
startRequest: function( element ) {
|
|
if ( !this.pending[ element.name ] ) {
|
|
this.pendingRequest++;
|
|
this.pending[ element.name ] = true;
|
|
}
|
|
},
|
|
|
|
stopRequest: function( element, valid ) {
|
|
this.pendingRequest--;
|
|
// sometimes synchronization fails, make sure pendingRequest is never < 0
|
|
if ( this.pendingRequest < 0 ) {
|
|
this.pendingRequest = 0;
|
|
}
|
|
delete this.pending[ element.name ];
|
|
if ( valid && this.pendingRequest === 0 && this.formSubmitted && this.form() ) {
|
|
$( this.currentForm ).submit();
|
|
this.formSubmitted = false;
|
|
} else if (!valid && this.pendingRequest === 0 && this.formSubmitted ) {
|
|
$( this.currentForm ).triggerHandler( "invalid-form", [ this ]);
|
|
this.formSubmitted = false;
|
|
}
|
|
},
|
|
|
|
previousValue: function( element ) {
|
|
return $.data( element, "previousValue" ) || $.data( element, "previousValue", {
|
|
old: null,
|
|
valid: true,
|
|
message: this.defaultMessage( element, "remote" )
|
|
});
|
|
}
|
|
|
|
},
|
|
|
|
classRuleSettings: {
|
|
required: { required: true },
|
|
email: { email: true },
|
|
url: { url: true },
|
|
date: { date: true },
|
|
dateISO: { dateISO: true },
|
|
number: { number: true },
|
|
digits: { digits: true },
|
|
creditcard: { creditcard: true }
|
|
},
|
|
|
|
addClassRules: function( className, rules ) {
|
|
if ( className.constructor === String ) {
|
|
this.classRuleSettings[ className ] = rules;
|
|
} else {
|
|
$.extend( this.classRuleSettings, className );
|
|
}
|
|
},
|
|
|
|
classRules: function( element ) {
|
|
var rules = {},
|
|
classes = $( element ).attr( "class" );
|
|
|
|
if ( classes ) {
|
|
$.each( classes.split( " " ), function() {
|
|
if ( this in $.validator.classRuleSettings ) {
|
|
$.extend( rules, $.validator.classRuleSettings[ this ]);
|
|
}
|
|
});
|
|
}
|
|
return rules;
|
|
},
|
|
|
|
attributeRules: function( element ) {
|
|
var rules = {},
|
|
$element = $( element ),
|
|
type = element.getAttribute( "type" ),
|
|
method, value;
|
|
|
|
for ( method in $.validator.methods ) {
|
|
|
|
// support for <input required> in both html5 and older browsers
|
|
if ( method === "required" ) {
|
|
value = element.getAttribute( method );
|
|
// Some browsers return an empty string for the required attribute
|
|
// and non-HTML5 browsers might have required="" markup
|
|
if ( value === "" ) {
|
|
value = true;
|
|
}
|
|
// force non-HTML5 browsers to return bool
|
|
value = !!value;
|
|
} else {
|
|
value = $element.attr( method );
|
|
}
|
|
|
|
// convert the value to a number for number inputs, and for text for backwards compability
|
|
// allows type="date" and others to be compared as strings
|
|
if ( /min|max/.test( method ) && ( type === null || /number|range|text/.test( type ) ) ) {
|
|
value = Number( value );
|
|
}
|
|
|
|
if ( value || value === 0 ) {
|
|
rules[ method ] = value;
|
|
} else if ( type === method && type !== "range" ) {
|
|
// exception: the jquery validate 'range' method
|
|
// does not test for the html5 'range' type
|
|
rules[ method ] = true;
|
|
}
|
|
}
|
|
|
|
// maxlength may be returned as -1, 2147483647 ( IE ) and 524288 ( safari ) for text inputs
|
|
if ( rules.maxlength && /-1|2147483647|524288/.test( rules.maxlength ) ) {
|
|
delete rules.maxlength;
|
|
}
|
|
|
|
return rules;
|
|
},
|
|
|
|
dataRules: function( element ) {
|
|
var method, value,
|
|
rules = {}, $element = $( element );
|
|
for ( method in $.validator.methods ) {
|
|
value = $element.data( "rule" + method.charAt( 0 ).toUpperCase() + method.substring( 1 ).toLowerCase() );
|
|
if ( value !== undefined ) {
|
|
rules[ method ] = value;
|
|
}
|
|
}
|
|
return rules;
|
|
},
|
|
|
|
staticRules: function( element ) {
|
|
var rules = {},
|
|
validator = $.data( element.form, "validator" );
|
|
|
|
if ( validator.settings.rules ) {
|
|
rules = $.validator.normalizeRule( validator.settings.rules[ element.name ] ) || {};
|
|
}
|
|
return rules;
|
|
},
|
|
|
|
normalizeRules: function( rules, element ) {
|
|
// handle dependency check
|
|
$.each( rules, function( prop, val ) {
|
|
// ignore rule when param is explicitly false, eg. required:false
|
|
if ( val === false ) {
|
|
delete rules[ prop ];
|
|
return;
|
|
}
|
|
if ( val.param || val.depends ) {
|
|
var keepRule = true;
|
|
switch ( typeof val.depends ) {
|
|
case "string":
|
|
keepRule = !!$( val.depends, element.form ).length;
|
|
break;
|
|
case "function":
|
|
keepRule = val.depends.call( element, element );
|
|
break;
|
|
}
|
|
if ( keepRule ) {
|
|
rules[ prop ] = val.param !== undefined ? val.param : true;
|
|
} else {
|
|
delete rules[ prop ];
|
|
}
|
|
}
|
|
});
|
|
|
|
// evaluate parameters
|
|
$.each( rules, function( rule, parameter ) {
|
|
rules[ rule ] = $.isFunction( parameter ) ? parameter( element ) : parameter;
|
|
});
|
|
|
|
// clean number parameters
|
|
$.each([ "minlength", "maxlength" ], function() {
|
|
if ( rules[ this ] ) {
|
|
rules[ this ] = Number( rules[ this ] );
|
|
}
|
|
});
|
|
$.each([ "rangelength", "range" ], function() {
|
|
var parts;
|
|
if ( rules[ this ] ) {
|
|
if ( $.isArray( rules[ this ] ) ) {
|
|
rules[ this ] = [ Number( rules[ this ][ 0 ]), Number( rules[ this ][ 1 ] ) ];
|
|
} else if ( typeof rules[ this ] === "string" ) {
|
|
parts = rules[ this ].replace(/[\[\]]/g, "" ).split( /[\s,]+/ );
|
|
rules[ this ] = [ Number( parts[ 0 ]), Number( parts[ 1 ] ) ];
|
|
}
|
|
}
|
|
});
|
|
|
|
if ( $.validator.autoCreateRanges ) {
|
|
// auto-create ranges
|
|
if ( rules.min != null && rules.max != null ) {
|
|
rules.range = [ rules.min, rules.max ];
|
|
delete rules.min;
|
|
delete rules.max;
|
|
}
|
|
if ( rules.minlength != null && rules.maxlength != null ) {
|
|
rules.rangelength = [ rules.minlength, rules.maxlength ];
|
|
delete rules.minlength;
|
|
delete rules.maxlength;
|
|
}
|
|
}
|
|
|
|
return rules;
|
|
},
|
|
|
|
// Converts a simple string to a {string: true} rule, e.g., "required" to {required:true}
|
|
normalizeRule: function( data ) {
|
|
if ( typeof data === "string" ) {
|
|
var transformed = {};
|
|
$.each( data.split( /\s/ ), function() {
|
|
transformed[ this ] = true;
|
|
});
|
|
data = transformed;
|
|
}
|
|
return data;
|
|
},
|
|
|
|
// http://jqueryvalidation.org/jQuery.validator.addMethod/
|
|
addMethod: function( name, method, message ) {
|
|
$.validator.methods[ name ] = method;
|
|
$.validator.messages[ name ] = message !== undefined ? message : $.validator.messages[ name ];
|
|
if ( method.length < 3 ) {
|
|
$.validator.addClassRules( name, $.validator.normalizeRule( name ) );
|
|
}
|
|
},
|
|
|
|
methods: {
|
|
|
|
// http://jqueryvalidation.org/required-method/
|
|
required: function( value, element, param ) {
|
|
// check if dependency is met
|
|
if ( !this.depend( param, element ) ) {
|
|
return "dependency-mismatch";
|
|
}
|
|
if ( element.nodeName.toLowerCase() === "select" ) {
|
|
// could be an array for select-multiple or a string, both are fine this way
|
|
var val = $( element ).val();
|
|
return val && val.length > 0;
|
|
}
|
|
if ( this.checkable( element ) ) {
|
|
return this.getLength( value, element ) > 0;
|
|
}
|
|
return $.trim( value ).length > 0;
|
|
},
|
|
|
|
// http://jqueryvalidation.org/email-method/
|
|
email: function( value, element ) {
|
|
// From http://www.whatwg.org/specs/web-apps/current-work/multipage/states-of-the-type-attribute.html#e-mail-state-%28type=email%29
|
|
// Retrieved 2014-01-14
|
|
// If you have a problem with this implementation, report a bug against the above spec
|
|
// Or use custom methods to implement your own email validation
|
|
return this.optional( element ) || /^[a-zA-Z0-9.!#$%&'*+\/=?^_`{|}~-]+@[a-zA-Z0-9](?:[a-zA-Z0-9-]{0,61}[a-zA-Z0-9])?(?:\.[a-zA-Z0-9](?:[a-zA-Z0-9-]{0,61}[a-zA-Z0-9])?)*$/.test( value );
|
|
},
|
|
|
|
// http://jqueryvalidation.org/url-method/
|
|
url: function( value, element ) {
|
|
// contributed by Scott Gonzalez: http://projects.scottsplayground.com/iri/
|
|
return this.optional( element ) || /^(https?|s?ftp):\/\/(((([a-z]|\d|-|\.|_|~|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])|(%[\da-f]{2})|[!\$&'\(\)\*\+,;=]|:)*@)?(((\d|[1-9]\d|1\d\d|2[0-4]\d|25[0-5])\.(\d|[1-9]\d|1\d\d|2[0-4]\d|25[0-5])\.(\d|[1-9]\d|1\d\d|2[0-4]\d|25[0-5])\.(\d|[1-9]\d|1\d\d|2[0-4]\d|25[0-5]))|((([a-z]|\d|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])|(([a-z]|\d|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])([a-z]|\d|-|\.|_|~|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])*([a-z]|\d|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])))\.)+(([a-z]|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])|(([a-z]|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])([a-z]|\d|-|\.|_|~|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])*([a-z]|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])))\.?)(:\d*)?)(\/((([a-z]|\d|-|\.|_|~|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])|(%[\da-f]{2})|[!\$&'\(\)\*\+,;=]|:|@)+(\/(([a-z]|\d|-|\.|_|~|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])|(%[\da-f]{2})|[!\$&'\(\)\*\+,;=]|:|@)*)*)?)?(\?((([a-z]|\d|-|\.|_|~|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])|(%[\da-f]{2})|[!\$&'\(\)\*\+,;=]|:|@)|[\uE000-\uF8FF]|\/|\?)*)?(#((([a-z]|\d|-|\.|_|~|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])|(%[\da-f]{2})|[!\$&'\(\)\*\+,;=]|:|@)|\/|\?)*)?$/i.test( value );
|
|
},
|
|
|
|
// http://jqueryvalidation.org/date-method/
|
|
date: function( value, element ) {
|
|
return this.optional( element ) || !/Invalid|NaN/.test( new Date( value ).toString() );
|
|
},
|
|
|
|
// http://jqueryvalidation.org/dateISO-method/
|
|
dateISO: function( value, element ) {
|
|
return this.optional( element ) || /^\d{4}[\/\-](0?[1-9]|1[012])[\/\-](0?[1-9]|[12][0-9]|3[01])$/.test( value );
|
|
},
|
|
|
|
// http://jqueryvalidation.org/number-method/
|
|
number: function( value, element ) {
|
|
return this.optional( element ) || /^-?(?:\d+|\d{1,3}(?:,\d{3})+)?(?:\.\d+)?$/.test( value );
|
|
},
|
|
|
|
// http://jqueryvalidation.org/digits-method/
|
|
digits: function( value, element ) {
|
|
return this.optional( element ) || /^\d+$/.test( value );
|
|
},
|
|
|
|
// http://jqueryvalidation.org/creditcard-method/
|
|
// based on http://en.wikipedia.org/wiki/Luhn/
|
|
creditcard: function( value, element ) {
|
|
if ( this.optional( element ) ) {
|
|
return "dependency-mismatch";
|
|
}
|
|
// accept only spaces, digits and dashes
|
|
if ( /[^0-9 \-]+/.test( value ) ) {
|
|
return false;
|
|
}
|
|
var nCheck = 0,
|
|
nDigit = 0,
|
|
bEven = false,
|
|
n, cDigit;
|
|
|
|
value = value.replace( /\D/g, "" );
|
|
|
|
// Basing min and max length on
|
|
// http://developer.ean.com/general_info/Valid_Credit_Card_Types
|
|
if ( value.length < 13 || value.length > 19 ) {
|
|
return false;
|
|
}
|
|
|
|
for ( n = value.length - 1; n >= 0; n--) {
|
|
cDigit = value.charAt( n );
|
|
nDigit = parseInt( cDigit, 10 );
|
|
if ( bEven ) {
|
|
if ( ( nDigit *= 2 ) > 9 ) {
|
|
nDigit -= 9;
|
|
}
|
|
}
|
|
nCheck += nDigit;
|
|
bEven = !bEven;
|
|
}
|
|
|
|
return ( nCheck % 10 ) === 0;
|
|
},
|
|
|
|
// http://jqueryvalidation.org/minlength-method/
|
|
minlength: function( value, element, param ) {
|
|
var length = $.isArray( value ) ? value.length : this.getLength( value, element );
|
|
return this.optional( element ) || length >= param;
|
|
},
|
|
|
|
// http://jqueryvalidation.org/maxlength-method/
|
|
maxlength: function( value, element, param ) {
|
|
var length = $.isArray( value ) ? value.length : this.getLength( value, element );
|
|
return this.optional( element ) || length <= param;
|
|
},
|
|
|
|
// http://jqueryvalidation.org/rangelength-method/
|
|
rangelength: function( value, element, param ) {
|
|
var length = $.isArray( value ) ? value.length : this.getLength( value, element );
|
|
return this.optional( element ) || ( length >= param[ 0 ] && length <= param[ 1 ] );
|
|
},
|
|
|
|
// http://jqueryvalidation.org/min-method/
|
|
min: function( value, element, param ) {
|
|
return this.optional( element ) || value >= param;
|
|
},
|
|
|
|
// http://jqueryvalidation.org/max-method/
|
|
max: function( value, element, param ) {
|
|
return this.optional( element ) || value <= param;
|
|
},
|
|
|
|
// http://jqueryvalidation.org/range-method/
|
|
range: function( value, element, param ) {
|
|
return this.optional( element ) || ( value >= param[ 0 ] && value <= param[ 1 ] );
|
|
},
|
|
|
|
// http://jqueryvalidation.org/equalTo-method/
|
|
equalTo: function( value, element, param ) {
|
|
// bind to the blur event of the target in order to revalidate whenever the target field is updated
|
|
// TODO find a way to bind the event just once, avoiding the unbind-rebind overhead
|
|
var target = $( param );
|
|
if ( this.settings.onfocusout ) {
|
|
target.unbind( ".validate-equalTo" ).bind( "blur.validate-equalTo", function() {
|
|
$( element ).valid();
|
|
});
|
|
}
|
|
return value === target.val();
|
|
},
|
|
|
|
// http://jqueryvalidation.org/remote-method/
|
|
remote: function( value, element, param ) {
|
|
if ( this.optional( element ) ) {
|
|
return "dependency-mismatch";
|
|
}
|
|
|
|
var previous = this.previousValue( element ),
|
|
validator, data;
|
|
|
|
if (!this.settings.messages[ element.name ] ) {
|
|
this.settings.messages[ element.name ] = {};
|
|
}
|
|
previous.originalMessage = this.settings.messages[ element.name ].remote;
|
|
this.settings.messages[ element.name ].remote = previous.message;
|
|
|
|
param = typeof param === "string" && { url: param } || param;
|
|
|
|
if ( previous.old === value ) {
|
|
return previous.valid;
|
|
}
|
|
|
|
previous.old = value;
|
|
validator = this;
|
|
this.startRequest( element );
|
|
data = {};
|
|
data[ element.name ] = value;
|
|
$.ajax( $.extend( true, {
|
|
url: param,
|
|
mode: "abort",
|
|
port: "validate" + element.name,
|
|
dataType: "json",
|
|
data: data,
|
|
context: validator.currentForm,
|
|
success: function( response ) {
|
|
var valid = response === true || response === "true",
|
|
errors, message, submitted;
|
|
|
|
validator.settings.messages[ element.name ].remote = previous.originalMessage;
|
|
if ( valid ) {
|
|
submitted = validator.formSubmitted;
|
|
validator.prepareElement( element );
|
|
validator.formSubmitted = submitted;
|
|
validator.successList.push( element );
|
|
delete validator.invalid[ element.name ];
|
|
validator.showErrors();
|
|
} else {
|
|
errors = {};
|
|
message = response || validator.defaultMessage( element, "remote" );
|
|
errors[ element.name ] = previous.message = $.isFunction( message ) ? message( value ) : message;
|
|
validator.invalid[ element.name ] = true;
|
|
validator.showErrors( errors );
|
|
}
|
|
previous.valid = valid;
|
|
validator.stopRequest( element, valid );
|
|
}
|
|
}, param ) );
|
|
return "pending";
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
$.format = function deprecated() {
|
|
throw "$.format has been deprecated. Please use $.validator.format instead.";
|
|
};
|
|
|
|
// ajax mode: abort
|
|
// usage: $.ajax({ mode: "abort"[, port: "uniqueport"]});
|
|
// if mode:"abort" is used, the previous request on that port (port can be undefined) is aborted via XMLHttpRequest.abort()
|
|
|
|
var pendingRequests = {},
|
|
ajax;
|
|
// Use a prefilter if available (1.5+)
|
|
if ( $.ajaxPrefilter ) {
|
|
$.ajaxPrefilter(function( settings, _, xhr ) {
|
|
var port = settings.port;
|
|
if ( settings.mode === "abort" ) {
|
|
if ( pendingRequests[port] ) {
|
|
pendingRequests[port].abort();
|
|
}
|
|
pendingRequests[port] = xhr;
|
|
}
|
|
});
|
|
} else {
|
|
// Proxy ajax
|
|
ajax = $.ajax;
|
|
$.ajax = function( settings ) {
|
|
var mode = ( "mode" in settings ? settings : $.ajaxSettings ).mode,
|
|
port = ( "port" in settings ? settings : $.ajaxSettings ).port;
|
|
if ( mode === "abort" ) {
|
|
if ( pendingRequests[port] ) {
|
|
pendingRequests[port].abort();
|
|
}
|
|
pendingRequests[port] = ajax.apply(this, arguments);
|
|
return pendingRequests[port];
|
|
}
|
|
return ajax.apply(this, arguments);
|
|
};
|
|
}
|
|
|
|
// provides delegate(type: String, delegate: Selector, handler: Callback) plugin for easier event delegation
|
|
// handler is only called when $(event.target).is(delegate), in the scope of the jquery-object for event.target
|
|
|
|
$.extend($.fn, {
|
|
validateDelegate: function( delegate, type, handler ) {
|
|
return this.bind(type, function( event ) {
|
|
var target = $(event.target);
|
|
if ( target.is(delegate) ) {
|
|
return handler.apply(target, arguments);
|
|
}
|
|
});
|
|
}
|
|
});
|
|
|
|
}));
|
|
(function(a){if(typeof define==="function"&&define.amd){define(["jquery"],a)}else{if(typeof exports==="object"){a(require("jquery"))}else{a(jQuery)}}}(function(f,c){if(!("indexOf" in Array.prototype)){Array.prototype.indexOf=function(k,j){if(j===c){j=0}if(j<0){j+=this.length}if(j<0){j=0}for(var l=this.length;j<l;j++){if(j in this&&this[j]===k){return j}}return -1}}function e(l){var k=f(l);var j=k.add(k.parents());var m=false;j.each(function(){if(f(this).css("position")==="fixed"){m=true;return false}});return m}function h(){return new Date(Date.UTC.apply(Date,arguments))}function d(){var j=new Date();return h(j.getUTCFullYear(),j.getUTCMonth(),j.getUTCDate(),j.getUTCHours(),j.getUTCMinutes(),j.getUTCSeconds(),0)}var i=function(l,k){var n=this;this.element=f(l);this.container=k.container||"body";this.language=k.language||this.element.data("date-language")||"en";this.language=this.language in a?this.language:this.language.split("-")[0];this.language=this.language in a?this.language:"en";this.isRTL=a[this.language].rtl||false;this.formatType=k.formatType||this.element.data("format-type")||"standard";this.format=g.parseFormat(k.format||this.element.data("date-format")||a[this.language].format||g.getDefaultFormat(this.formatType,"input"),this.formatType);this.isInline=false;this.isVisible=false;this.isInput=this.element.is("input");this.fontAwesome=k.fontAwesome||this.element.data("font-awesome")||false;this.bootcssVer=k.bootcssVer||(this.isInput?(this.element.is(".form-control")?3:2):(this.bootcssVer=this.element.is(".input-group")?3:2));this.component=this.element.is(".date")?(this.bootcssVer==3?this.element.find(".input-group-addon .glyphicon-th, .input-group-addon .glyphicon-time, .input-group-addon .glyphicon-remove, .input-group-addon .glyphicon-calendar, .input-group-addon .fa-calendar, .input-group-addon .fa-clock-o").parent():this.element.find(".add-on .icon-th, .add-on .icon-time, .add-on .icon-calendar, .add-on .fa-calendar, .add-on .fa-clock-o").parent()):false;this.componentReset=this.element.is(".date")?(this.bootcssVer==3?this.element.find(".input-group-addon .glyphicon-remove, .input-group-addon .fa-times").parent():this.element.find(".add-on .icon-remove, .add-on .fa-times").parent()):false;this.hasInput=this.component&&this.element.find("input").length;if(this.component&&this.component.length===0){this.component=false}this.linkField=k.linkField||this.element.data("link-field")||false;this.linkFormat=g.parseFormat(k.linkFormat||this.element.data("link-format")||g.getDefaultFormat(this.formatType,"link"),this.formatType);this.minuteStep=k.minuteStep||this.element.data("minute-step")||5;this.pickerPosition=k.pickerPosition||this.element.data("picker-position")||"bottom-right";this.showMeridian=k.showMeridian||this.element.data("show-meridian")||false;this.initialDate=k.initialDate||new Date();this.zIndex=k.zIndex||this.element.data("z-index")||c;this.title=typeof k.title==="undefined"?false:k.title;this.icons={leftArrow:this.fontAwesome?"fa-arrow-left":(this.bootcssVer===3?"glyphicon-arrow-left":"icon-arrow-left"),rightArrow:this.fontAwesome?"fa-arrow-right":(this.bootcssVer===3?"glyphicon-arrow-right":"icon-arrow-right")};this.icontype=this.fontAwesome?"fa":"glyphicon";this._attachEvents();this.clickedOutside=function(o){if(f(o.target).closest(".datetimepicker").length===0){n.hide()}};this.formatViewType="datetime";if("formatViewType" in k){this.formatViewType=k.formatViewType}else{if("formatViewType" in this.element.data()){this.formatViewType=this.element.data("formatViewType")}}this.minView=0;if("minView" in k){this.minView=k.minView}else{if("minView" in this.element.data()){this.minView=this.element.data("min-view")}}this.minView=g.convertViewMode(this.minView);this.maxView=g.modes.length-1;if("maxView" in k){this.maxView=k.maxView}else{if("maxView" in this.element.data()){this.maxView=this.element.data("max-view")}}this.maxView=g.convertViewMode(this.maxView);this.wheelViewModeNavigation=false;if("wheelViewModeNavigation" in k){this.wheelViewModeNavigation=k.wheelViewModeNavigation}else{if("wheelViewModeNavigation" in this.element.data()){this.wheelViewModeNavigation=this.element.data("view-mode-wheel-navigation")}}this.wheelViewModeNavigationInverseDirection=false;if("wheelViewModeNavigationInverseDirection" in k){this.wheelViewModeNavigationInverseDirection=k.wheelViewModeNavigationInverseDirection}else{if("wheelViewModeNavigationInverseDirection" in this.element.data()){this.wheelViewModeNavigationInverseDirection=this.element.data("view-mode-wheel-navigation-inverse-dir")}}this.wheelViewModeNavigationDelay=100;if("wheelViewModeNavigationDelay" in k){this.wheelViewModeNavigationDelay=k.wheelViewModeNavigationDelay}else{if("wheelViewModeNavigationDelay" in this.element.data()){this.wheelViewModeNavigationDelay=this.element.data("view-mode-wheel-navigation-delay")}}this.startViewMode=2;if("startView" in k){this.startViewMode=k.startView}else{if("startView" in this.element.data()){this.startViewMode=this.element.data("start-view")}}this.startViewMode=g.convertViewMode(this.startViewMode);this.viewMode=this.startViewMode;this.viewSelect=this.minView;if("viewSelect" in k){this.viewSelect=k.viewSelect}else{if("viewSelect" in this.element.data()){this.viewSelect=this.element.data("view-select")}}this.viewSelect=g.convertViewMode(this.viewSelect);this.forceParse=true;if("forceParse" in k){this.forceParse=k.forceParse}else{if("dateForceParse" in this.element.data()){this.forceParse=this.element.data("date-force-parse")}}var m=this.bootcssVer===3?g.templateV3:g.template;while(m.indexOf("{iconType}")!==-1){m=m.replace("{iconType}",this.icontype)}while(m.indexOf("{leftArrow}")!==-1){m=m.replace("{leftArrow}",this.icons.leftArrow)}while(m.indexOf("{rightArrow}")!==-1){m=m.replace("{rightArrow}",this.icons.rightArrow)}this.picker=f(m).appendTo(this.isInline?this.element:this.container).on({click:f.proxy(this.click,this),mousedown:f.proxy(this.mousedown,this)});if(this.wheelViewModeNavigation){if(f.fn.mousewheel){this.picker.on({mousewheel:f.proxy(this.mousewheel,this)})}else{console.log("Mouse Wheel event is not supported. Please include the jQuery Mouse Wheel plugin before enabling this option")}}if(this.isInline){this.picker.addClass("datetimepicker-inline")}else{this.picker.addClass("datetimepicker-dropdown-"+this.pickerPosition+" dropdown-menu")}if(this.isRTL){this.picker.addClass("datetimepicker-rtl");var j=this.bootcssVer===3?".prev span, .next span":".prev i, .next i";this.picker.find(j).toggleClass(this.icons.leftArrow+" "+this.icons.rightArrow)}f(document).on("mousedown",this.clickedOutside);this.autoclose=false;if("autoclose" in k){this.autoclose=k.autoclose}else{if("dateAutoclose" in this.element.data()){this.autoclose=this.element.data("date-autoclose")}}this.keyboardNavigation=true;if("keyboardNavigation" in k){this.keyboardNavigation=k.keyboardNavigation}else{if("dateKeyboardNavigation" in this.element.data()){this.keyboardNavigation=this.element.data("date-keyboard-navigation")}}this.todayBtn=(k.todayBtn||this.element.data("date-today-btn")||false);this.clearBtn=(k.clearBtn||this.element.data("date-clear-btn")||false);this.todayHighlight=(k.todayHighlight||this.element.data("date-today-highlight")||false);this.weekStart=((k.weekStart||this.element.data("date-weekstart")||a[this.language].weekStart||0)%7);this.weekEnd=((this.weekStart+6)%7);this.startDate=-Infinity;this.endDate=Infinity;this.datesDisabled=[];this.daysOfWeekDisabled=[];this.setStartDate(k.startDate||this.element.data("date-startdate"));this.setEndDate(k.endDate||this.element.data("date-enddate"));this.setDatesDisabled(k.datesDisabled||this.element.data("date-dates-disabled"));this.setDaysOfWeekDisabled(k.daysOfWeekDisabled||this.element.data("date-days-of-week-disabled"));this.setMinutesDisabled(k.minutesDisabled||this.element.data("date-minute-disabled"));this.setHoursDisabled(k.hoursDisabled||this.element.data("date-hour-disabled"));this.fillDow();this.fillMonths();this.update();this.showMode();if(this.isInline){this.show()}};i.prototype={constructor:i,_events:[],_attachEvents:function(){this._detachEvents();if(this.isInput){this._events=[[this.element,{focus:f.proxy(this.show,this),keyup:f.proxy(this.update,this),keydown:f.proxy(this.keydown,this)}]]}else{if(this.component&&this.hasInput){this._events=[[this.element.find("input"),{focus:f.proxy(this.show,this),keyup:f.proxy(this.update,this),keydown:f.proxy(this.keydown,this)}],[this.component,{click:f.proxy(this.show,this)}]];if(this.componentReset){this._events.push([this.componentReset,{click:f.proxy(this.reset,this)}])}}else{if(this.element.is("div")){this.isInline=true}else{this._events=[[this.element,{click:f.proxy(this.show,this)}]]}}}for(var j=0,k,l;j<this._events.length;j++){k=this._events[j][0];l=this._events[j][1];k.on(l)}},_detachEvents:function(){for(var j=0,k,l;j<this._events.length;j++){k=this._events[j][0];l=this._events[j][1];k.off(l)}this._events=[]},show:function(j){this.picker.show();this.height=this.component?this.component.outerHeight():this.element.outerHeight();if(this.forceParse){this.update()}this.place();f(window).on("resize",f.proxy(this.place,this));if(j){j.stopPropagation();j.preventDefault()}this.isVisible=true;this.element.trigger({type:"show",date:this.date})},hide:function(j){if(!this.isVisible){return}if(this.isInline){return}this.picker.hide();f(window).off("resize",this.place);this.viewMode=this.startViewMode;this.showMode();if(!this.isInput){f(document).off("mousedown",this.hide)}if(this.forceParse&&(this.isInput&&this.element.val()||this.hasInput&&this.element.find("input").val())){this.setValue()}this.isVisible=false;this.element.trigger({type:"hide",date:this.date})},remove:function(){this._detachEvents();f(document).off("mousedown",this.clickedOutside);this.picker.remove();delete this.picker;delete this.element.data().datetimepicker},getDate:function(){var j=this.getUTCDate();return new Date(j.getTime()+(j.getTimezoneOffset()*60000))},getUTCDate:function(){return this.date},getInitialDate:function(){return this.initialDate},setInitialDate:function(j){this.initialDate=j},setDate:function(j){this.setUTCDate(new Date(j.getTime()-(j.getTimezoneOffset()*60000)))},setUTCDate:function(j){if(j>=this.startDate&&j<=this.endDate){this.date=j;this.setValue();this.viewDate=this.date;this.fill()}else{this.element.trigger({type:"outOfRange",date:j,startDate:this.startDate,endDate:this.endDate})}},setFormat:function(k){this.format=g.parseFormat(k,this.formatType);var j;if(this.isInput){j=this.element}else{if(this.component){j=this.element.find("input")}}if(j&&j.val()){this.setValue()}},setValue:function(){var j=this.getFormattedDate();if(!this.isInput){if(this.component){this.element.find("input").val(j)}this.element.data("date",j)}else{this.element.val(j)}if(this.linkField){f("#"+this.linkField).val(this.getFormattedDate(this.linkFormat))}},getFormattedDate:function(j){if(j==c){j=this.format}return g.formatDate(this.date,j,this.language,this.formatType)},setStartDate:function(j){this.startDate=j||-Infinity;if(this.startDate!==-Infinity){this.startDate=g.parseDate(this.startDate,this.format,this.language,this.formatType)}this.update();this.updateNavArrows()},setEndDate:function(j){this.endDate=j||Infinity;if(this.endDate!==Infinity){this.endDate=g.parseDate(this.endDate,this.format,this.language,this.formatType)}this.update();this.updateNavArrows()},setDatesDisabled:function(j){this.datesDisabled=j||[];if(!f.isArray(this.datesDisabled)){this.datesDisabled=this.datesDisabled.split(/,\s*/)}this.datesDisabled=f.map(this.datesDisabled,function(k){return g.parseDate(k,this.format,this.language,this.formatType).toDateString()});this.update();this.updateNavArrows()},setTitle:function(j,k){return this.picker.find(j).find("th:eq(1)").text(this.title===false?k:this.title)},setDaysOfWeekDisabled:function(j){this.daysOfWeekDisabled=j||[];if(!f.isArray(this.daysOfWeekDisabled)){this.daysOfWeekDisabled=this.daysOfWeekDisabled.split(/,\s*/)}this.daysOfWeekDisabled=f.map(this.daysOfWeekDisabled,function(k){return parseInt(k,10)});this.update();this.updateNavArrows()},setMinutesDisabled:function(j){this.minutesDisabled=j||[];if(!f.isArray(this.minutesDisabled)){this.minutesDisabled=this.minutesDisabled.split(/,\s*/)}this.minutesDisabled=f.map(this.minutesDisabled,function(k){return parseInt(k,10)});this.update();this.updateNavArrows()},setHoursDisabled:function(j){this.hoursDisabled=j||[];if(!f.isArray(this.hoursDisabled)){this.hoursDisabled=this.hoursDisabled.split(/,\s*/)}this.hoursDisabled=f.map(this.hoursDisabled,function(k){return parseInt(k,10)});this.update();this.updateNavArrows()},place:function(){if(this.isInline){return}if(!this.zIndex){var k=0;f("div").each(function(){var p=parseInt(f(this).css("zIndex"),10);if(p>k){k=p}});this.zIndex=k+10}var o,n,m,l;if(this.container instanceof f){l=this.container.offset()}else{l=f(this.container).offset()}if(this.component){o=this.component.offset();m=o.left;if(this.pickerPosition=="bottom-left"||this.pickerPosition=="top-left"){m+=this.component.outerWidth()-this.picker.outerWidth()}}else{o=this.element.offset();m=o.left;if(this.pickerPosition=="bottom-left"||this.pickerPosition=="top-left"){m+=this.element.outerWidth()-this.picker.outerWidth()}}var j=document.body.clientWidth||window.innerWidth;if(m+220>j){m=j-220}if(this.pickerPosition=="top-left"||this.pickerPosition=="top-right"){n=o.top-this.picker.outerHeight()}else{n=o.top+this.height}n=n-l.top;m=m-l.left;this.picker.css({top:n,left:m,zIndex:this.zIndex})},update:function(){var j,k=false;if(arguments&&arguments.length&&(typeof arguments[0]==="string"||arguments[0] instanceof Date)){j=arguments[0];k=true}else{j=(this.isInput?this.element.val():this.element.find("input").val())||this.element.data("date")||this.initialDate;if(typeof j=="string"||j instanceof String){j=j.replace(/^\s+|\s+$/g,"")}}if(!j){j=new Date();k=false}this.date=g.parseDate(j,this.format,this.language,this.formatType);if(k){this.setValue()}if(this.date<this.startDate){this.viewDate=new Date(this.startDate)}else{if(this.date>this.endDate){this.viewDate=new Date(this.endDate)}else{this.viewDate=new Date(this.date)}}this.fill()},fillDow:function(){var j=this.weekStart,k="<tr>";while(j<this.weekStart+7){k+='<th class="dow">'+a[this.language].daysMin[(j++)%7]+"</th>"}k+="</tr>";this.picker.find(".datetimepicker-days thead").append(k)},fillMonths:function(){var k="",j=0;while(j<12){k+='<span class="month">'+a[this.language].monthsShort[j++]+"</span>"}this.picker.find(".datetimepicker-months td").html(k)},fill:function(){if(this.date==null||this.viewDate==null){return}var H=new Date(this.viewDate),u=H.getUTCFullYear(),I=H.getUTCMonth(),n=H.getUTCDate(),D=H.getUTCHours(),y=H.getUTCMinutes(),z=this.startDate!==-Infinity?this.startDate.getUTCFullYear():-Infinity,E=this.startDate!==-Infinity?this.startDate.getUTCMonth()+1:-Infinity,q=this.endDate!==Infinity?this.endDate.getUTCFullYear():Infinity,A=this.endDate!==Infinity?this.endDate.getUTCMonth()+1:Infinity,r=(new h(this.date.getUTCFullYear(),this.date.getUTCMonth(),this.date.getUTCDate())).valueOf(),G=new Date();this.setTitle(".datetimepicker-days",a[this.language].months[I]+" "+u);if(this.formatViewType=="time"){var k=this.getFormattedDate();this.setTitle(".datetimepicker-hours",k);this.setTitle(".datetimepicker-minutes",k)}else{this.setTitle(".datetimepicker-hours",n+" "+a[this.language].months[I]+" "+u);this.setTitle(".datetimepicker-minutes",n+" "+a[this.language].months[I]+" "+u)}this.picker.find("tfoot th.today").text(a[this.language].today||a.en.today).toggle(this.todayBtn!==false);this.picker.find("tfoot th.clear").text(a[this.language].clear||a.en.clear).toggle(this.clearBtn!==false);this.updateNavArrows();this.fillMonths();var K=h(u,I-1,28,0,0,0,0),C=g.getDaysInMonth(K.getUTCFullYear(),K.getUTCMonth());K.setUTCDate(C);K.setUTCDate(C-(K.getUTCDay()-this.weekStart+7)%7);var j=new Date(K);j.setUTCDate(j.getUTCDate()+42);j=j.valueOf();var s=[];var v;while(K.valueOf()<j){if(K.getUTCDay()==this.weekStart){s.push("<tr>")}v="";if(K.getUTCFullYear()<u||(K.getUTCFullYear()==u&&K.getUTCMonth()<I)){v+=" old"}else{if(K.getUTCFullYear()>u||(K.getUTCFullYear()==u&&K.getUTCMonth()>I)){v+=" new"}}if(this.todayHighlight&&K.getUTCFullYear()==G.getFullYear()&&K.getUTCMonth()==G.getMonth()&&K.getUTCDate()==G.getDate()){v+=" today"}if(K.valueOf()==r){v+=" active"}if((K.valueOf()+86400000)<=this.startDate||K.valueOf()>this.endDate||f.inArray(K.getUTCDay(),this.daysOfWeekDisabled)!==-1||f.inArray(K.toDateString(),this.datesDisabled)!==-1){v+=" disabled"}s.push('<td class="day'+v+'">'+K.getUTCDate()+"</td>");if(K.getUTCDay()==this.weekEnd){s.push("</tr>")}K.setUTCDate(K.getUTCDate()+1)}this.picker.find(".datetimepicker-days tbody").empty().append(s.join(""));s=[];var w="",F="",t="";var l=this.hoursDisabled||[];for(var B=0;B<24;B++){if(l.indexOf(B)!==-1){continue}var x=h(u,I,n,B);v="";if((x.valueOf()+3600000)<=this.startDate||x.valueOf()>this.endDate){v+=" disabled"}else{if(D==B){v+=" active"}}if(this.showMeridian&&a[this.language].meridiem.length==2){F=(B<12?a[this.language].meridiem[0]:a[this.language].meridiem[1]);if(F!=t){if(t!=""){s.push("</fieldset>")}s.push('<fieldset class="hour"><legend>'+F.toUpperCase()+"</legend>")}t=F;w=(B%12?B%12:12);s.push('<span class="hour'+v+" hour_"+(B<12?"am":"pm")+'">'+w+"</span>");if(B==23){s.push("</fieldset>")}}else{w=B+":00";s.push('<span class="hour'+v+'">'+w+"</span>")}}this.picker.find(".datetimepicker-hours td").html(s.join(""));s=[];w="",F="",t="";var m=this.minutesDisabled||[];for(var B=0;B<60;B+=this.minuteStep){if(m.indexOf(B)!==-1){continue}var x=h(u,I,n,D,B,0);v="";if(x.valueOf()<this.startDate||x.valueOf()>this.endDate){v+=" disabled"}else{if(Math.floor(y/this.minuteStep)==Math.floor(B/this.minuteStep)){v+=" active"}}if(this.showMeridian&&a[this.language].meridiem.length==2){F=(D<12?a[this.language].meridiem[0]:a[this.language].meridiem[1]);if(F!=t){if(t!=""){s.push("</fieldset>")}s.push('<fieldset class="minute"><legend>'+F.toUpperCase()+"</legend>")}t=F;w=(D%12?D%12:12);s.push('<span class="minute'+v+'">'+w+":"+(B<10?"0"+B:B)+"</span>");if(B==59){s.push("</fieldset>")}}else{w=B+":00";s.push('<span class="minute'+v+'">'+D+":"+(B<10?"0"+B:B)+"</span>")}}this.picker.find(".datetimepicker-minutes td").html(s.join(""));var L=this.date.getUTCFullYear();var p=this.setTitle(".datetimepicker-months",u).end().find("span").removeClass("active");if(L==u){var o=p.length-12;p.eq(this.date.getUTCMonth()+o).addClass("active")}if(u<z||u>q){p.addClass("disabled")}if(u==z){p.slice(0,E+1).addClass("disabled")}if(u==q){p.slice(A).addClass("disabled")}s="";u=parseInt(u/10,10)*10;var J=this.setTitle(".datetimepicker-years",u+"-"+(u+9)).end().find("td");u-=1;for(var B=-1;B<11;B++){s+='<span class="year'+(B==-1||B==10?" old":"")+(L==u?" active":"")+(u<z||u>q?" disabled":"")+'">'+u+"</span>";u+=1}J.html(s);this.place()},updateNavArrows:function(){var n=new Date(this.viewDate),l=n.getUTCFullYear(),m=n.getUTCMonth(),k=n.getUTCDate(),j=n.getUTCHours();switch(this.viewMode){case 0:if(this.startDate!==-Infinity&&l<=this.startDate.getUTCFullYear()&&m<=this.startDate.getUTCMonth()&&k<=this.startDate.getUTCDate()&&j<=this.startDate.getUTCHours()){this.picker.find(".prev").css({visibility:"hidden"})}else{this.picker.find(".prev").css({visibility:"visible"})}if(this.endDate!==Infinity&&l>=this.endDate.getUTCFullYear()&&m>=this.endDate.getUTCMonth()&&k>=this.endDate.getUTCDate()&&j>=this.endDate.getUTCHours()){this.picker.find(".next").css({visibility:"hidden"})}else{this.picker.find(".next").css({visibility:"visible"})}break;case 1:if(this.startDate!==-Infinity&&l<=this.startDate.getUTCFullYear()&&m<=this.startDate.getUTCMonth()&&k<=this.startDate.getUTCDate()){this.picker.find(".prev").css({visibility:"hidden"})}else{this.picker.find(".prev").css({visibility:"visible"})}if(this.endDate!==Infinity&&l>=this.endDate.getUTCFullYear()&&m>=this.endDate.getUTCMonth()&&k>=this.endDate.getUTCDate()){this.picker.find(".next").css({visibility:"hidden"})}else{this.picker.find(".next").css({visibility:"visible"})}break;case 2:if(this.startDate!==-Infinity&&l<=this.startDate.getUTCFullYear()&&m<=this.startDate.getUTCMonth()){this.picker.find(".prev").css({visibility:"hidden"})}else{this.picker.find(".prev").css({visibility:"visible"})}if(this.endDate!==Infinity&&l>=this.endDate.getUTCFullYear()&&m>=this.endDate.getUTCMonth()){this.picker.find(".next").css({visibility:"hidden"})}else{this.picker.find(".next").css({visibility:"visible"})}break;case 3:case 4:if(this.startDate!==-Infinity&&l<=this.startDate.getUTCFullYear()){this.picker.find(".prev").css({visibility:"hidden"})}else{this.picker.find(".prev").css({visibility:"visible"})}if(this.endDate!==Infinity&&l>=this.endDate.getUTCFullYear()){this.picker.find(".next").css({visibility:"hidden"})}else{this.picker.find(".next").css({visibility:"visible"})}break}},mousewheel:function(k){k.preventDefault();k.stopPropagation();if(this.wheelPause){return}this.wheelPause=true;var j=k.originalEvent;var m=j.wheelDelta;var l=m>0?1:(m===0)?0:-1;if(this.wheelViewModeNavigationInverseDirection){l=-l}this.showMode(l);setTimeout(f.proxy(function(){this.wheelPause=false},this),this.wheelViewModeNavigationDelay)},click:function(n){n.stopPropagation();n.preventDefault();var o=f(n.target).closest("span, td, th, legend");if(o.is("."+this.icontype)){o=f(o).parent().closest("span, td, th, legend")}if(o.length==1){if(o.is(".disabled")){this.element.trigger({type:"outOfRange",date:this.viewDate,startDate:this.startDate,endDate:this.endDate});return}switch(o[0].nodeName.toLowerCase()){case"th":switch(o[0].className){case"switch":this.showMode(1);break;case"prev":case"next":var j=g.modes[this.viewMode].navStep*(o[0].className=="prev"?-1:1);switch(this.viewMode){case 0:this.viewDate=this.moveHour(this.viewDate,j);break;case 1:this.viewDate=this.moveDate(this.viewDate,j);break;case 2:this.viewDate=this.moveMonth(this.viewDate,j);break;case 3:case 4:this.viewDate=this.moveYear(this.viewDate,j);break}this.fill();this.element.trigger({type:o[0].className+":"+this.convertViewModeText(this.viewMode),date:this.viewDate,startDate:this.startDate,endDate:this.endDate});break;case"clear":this.reset();if(this.autoclose){this.hide()}break;case"today":var k=new Date();k=h(k.getFullYear(),k.getMonth(),k.getDate(),k.getHours(),k.getMinutes(),k.getSeconds(),0);if(k<this.startDate){k=this.startDate}else{if(k>this.endDate){k=this.endDate}}this.viewMode=this.startViewMode;this.showMode(0);this._setDate(k);this.fill();if(this.autoclose){this.hide()}break}break;case"span":if(!o.is(".disabled")){var q=this.viewDate.getUTCFullYear(),p=this.viewDate.getUTCMonth(),r=this.viewDate.getUTCDate(),s=this.viewDate.getUTCHours(),l=this.viewDate.getUTCMinutes(),t=this.viewDate.getUTCSeconds();if(o.is(".month")){this.viewDate.setUTCDate(1);p=o.parent().find("span").index(o);r=this.viewDate.getUTCDate();this.viewDate.setUTCMonth(p);this.element.trigger({type:"changeMonth",date:this.viewDate});if(this.viewSelect>=3){this._setDate(h(q,p,r,s,l,t,0))}}else{if(o.is(".year")){this.viewDate.setUTCDate(1);q=parseInt(o.text(),10)||0;this.viewDate.setUTCFullYear(q);this.element.trigger({type:"changeYear",date:this.viewDate});if(this.viewSelect>=4){this._setDate(h(q,p,r,s,l,t,0))}}else{if(o.is(".hour")){s=parseInt(o.text(),10)||0;if(o.hasClass("hour_am")||o.hasClass("hour_pm")){if(s==12&&o.hasClass("hour_am")){s=0}else{if(s!=12&&o.hasClass("hour_pm")){s+=12}}}this.viewDate.setUTCHours(s);this.element.trigger({type:"changeHour",date:this.viewDate});if(this.viewSelect>=1){this._setDate(h(q,p,r,s,l,t,0))}}else{if(o.is(".minute")){l=parseInt(o.text().substr(o.text().indexOf(":")+1),10)||0;this.viewDate.setUTCMinutes(l);this.element.trigger({type:"changeMinute",date:this.viewDate});if(this.viewSelect>=0){this._setDate(h(q,p,r,s,l,t,0))}}}}}if(this.viewMode!=0){var m=this.viewMode;this.showMode(-1);this.fill();if(m==this.viewMode&&this.autoclose){this.hide()}}else{this.fill();if(this.autoclose){this.hide()}}}break;case"td":if(o.is(".day")&&!o.is(".disabled")){var r=parseInt(o.text(),10)||1;var q=this.viewDate.getUTCFullYear(),p=this.viewDate.getUTCMonth(),s=this.viewDate.getUTCHours(),l=this.viewDate.getUTCMinutes(),t=this.viewDate.getUTCSeconds();if(o.is(".old")){if(p===0){p=11;q-=1}else{p-=1}}else{if(o.is(".new")){if(p==11){p=0;q+=1}else{p+=1}}}this.viewDate.setUTCFullYear(q);this.viewDate.setUTCMonth(p,r);this.element.trigger({type:"changeDay",date:this.viewDate});if(this.viewSelect>=2){this._setDate(h(q,p,r,s,l,t,0))}}var m=this.viewMode;this.showMode(-1);this.fill();if(m==this.viewMode&&this.autoclose){this.hide()}break}}},_setDate:function(j,l){if(!l||l=="date"){this.date=j}if(!l||l=="view"){this.viewDate=j}this.fill();this.setValue();var k;if(this.isInput){k=this.element}else{if(this.component){k=this.element.find("input")}}if(k){k.change();if(this.autoclose&&(!l||l=="date")){}}this.element.trigger({type:"changeDate",date:this.getDate()});if(j==null){this.date=this.viewDate}},moveMinute:function(k,j){if(!j){return k}var l=new Date(k.valueOf());l.setUTCMinutes(l.getUTCMinutes()+(j*this.minuteStep));return l},moveHour:function(k,j){if(!j){return k}var l=new Date(k.valueOf());l.setUTCHours(l.getUTCHours()+j);return l},moveDate:function(k,j){if(!j){return k}var l=new Date(k.valueOf());l.setUTCDate(l.getUTCDate()+j);return l},moveMonth:function(j,k){if(!k){return j}var n=new Date(j.valueOf()),r=n.getUTCDate(),o=n.getUTCMonth(),m=Math.abs(k),q,p;k=k>0?1:-1;if(m==1){p=k==-1?function(){return n.getUTCMonth()==o}:function(){return n.getUTCMonth()!=q};q=o+k;n.setUTCMonth(q);if(q<0||q>11){q=(q+12)%12}}else{for(var l=0;l<m;l++){n=this.moveMonth(n,k)}q=n.getUTCMonth();n.setUTCDate(r);p=function(){return q!=n.getUTCMonth()}}while(p()){n.setUTCDate(--r);n.setUTCMonth(q)}return n},moveYear:function(k,j){return this.moveMonth(k,j*12)},dateWithinRange:function(j){return j>=this.startDate&&j<=this.endDate},keydown:function(n){if(this.picker.is(":not(:visible)")){if(n.keyCode==27){this.show()}return}var p=false,k,q,o,r,j;switch(n.keyCode){case 27:this.hide();n.preventDefault();break;case 37:case 39:if(!this.keyboardNavigation){break}k=n.keyCode==37?-1:1;viewMode=this.viewMode;if(n.ctrlKey){viewMode+=2}else{if(n.shiftKey){viewMode+=1}}if(viewMode==4){r=this.moveYear(this.date,k);j=this.moveYear(this.viewDate,k)}else{if(viewMode==3){r=this.moveMonth(this.date,k);j=this.moveMonth(this.viewDate,k)}else{if(viewMode==2){r=this.moveDate(this.date,k);j=this.moveDate(this.viewDate,k)}else{if(viewMode==1){r=this.moveHour(this.date,k);j=this.moveHour(this.viewDate,k)}else{if(viewMode==0){r=this.moveMinute(this.date,k);j=this.moveMinute(this.viewDate,k)}}}}}if(this.dateWithinRange(r)){this.date=r;this.viewDate=j;this.setValue();this.update();n.preventDefault();p=true}break;case 38:case 40:if(!this.keyboardNavigation){break}k=n.keyCode==38?-1:1;viewMode=this.viewMode;if(n.ctrlKey){viewMode+=2}else{if(n.shiftKey){viewMode+=1}}if(viewMode==4){r=this.moveYear(this.date,k);j=this.moveYear(this.viewDate,k)}else{if(viewMode==3){r=this.moveMonth(this.date,k);j=this.moveMonth(this.viewDate,k)}else{if(viewMode==2){r=this.moveDate(this.date,k*7);j=this.moveDate(this.viewDate,k*7)}else{if(viewMode==1){if(this.showMeridian){r=this.moveHour(this.date,k*6);j=this.moveHour(this.viewDate,k*6)}else{r=this.moveHour(this.date,k*4);j=this.moveHour(this.viewDate,k*4)}}else{if(viewMode==0){r=this.moveMinute(this.date,k*4);j=this.moveMinute(this.viewDate,k*4)}}}}}if(this.dateWithinRange(r)){this.date=r;this.viewDate=j;this.setValue();this.update();n.preventDefault();p=true}break;case 13:if(this.viewMode!=0){var m=this.viewMode;this.showMode(-1);this.fill();if(m==this.viewMode&&this.autoclose){this.hide()}}else{this.fill();if(this.autoclose){this.hide()}}n.preventDefault();break;case 9:this.hide();break}if(p){var l;if(this.isInput){l=this.element}else{if(this.component){l=this.element.find("input")}}if(l){l.change()}this.element.trigger({type:"changeDate",date:this.getDate()})}},showMode:function(j){if(j){var k=Math.max(0,Math.min(g.modes.length-1,this.viewMode+j));if(k>=this.minView&&k<=this.maxView){this.element.trigger({type:"changeMode",date:this.viewDate,oldViewMode:this.viewMode,newViewMode:k});this.viewMode=k}}this.picker.find(">div").hide().filter(".datetimepicker-"+g.modes[this.viewMode].clsName).css("display","block");this.updateNavArrows()},reset:function(j){this._setDate(null,"date")},convertViewModeText:function(j){switch(j){case 4:return"decade";case 3:return"year";case 2:return"month";case 1:return"day";case 0:return"hour"}}};var b=f.fn.datetimepicker;f.fn.datetimepicker=function(l){var j=Array.apply(null,arguments);j.shift();var k;this.each(function(){var o=f(this),n=o.data("datetimepicker"),m=typeof l=="object"&&l;if(!n){o.data("datetimepicker",(n=new i(this,f.extend({},f.fn.datetimepicker.defaults,m))))}if(typeof l=="string"&&typeof n[l]=="function"){k=n[l].apply(n,j);if(k!==c){return false}}});if(k!==c){return k}else{return this}};f.fn.datetimepicker.defaults={};f.fn.datetimepicker.Constructor=i;var a=f.fn.datetimepicker.dates={en:{days:["Sunday","Monday","Tuesday","Wednesday","Thursday","Friday","Saturday","Sunday"],daysShort:["Sun","Mon","Tue","Wed","Thu","Fri","Sat","Sun"],daysMin:["Su","Mo","Tu","We","Th","Fr","Sa","Su"],months:["January","February","March","April","May","June","July","August","September","October","November","December"],monthsShort:["Jan","Feb","Mar","Apr","May","Jun","Jul","Aug","Sep","Oct","Nov","Dec"],meridiem:["am","pm"],suffix:["st","nd","rd","th"],today:"Today",clear:"Clear"}};var g={modes:[{clsName:"minutes",navFnc:"Hours",navStep:1},{clsName:"hours",navFnc:"Date",navStep:1},{clsName:"days",navFnc:"Month",navStep:1},{clsName:"months",navFnc:"FullYear",navStep:1},{clsName:"years",navFnc:"FullYear",navStep:10}],isLeapYear:function(j){return(((j%4===0)&&(j%100!==0))||(j%400===0))},getDaysInMonth:function(j,k){return[31,(g.isLeapYear(j)?29:28),31,30,31,30,31,31,30,31,30,31][k]},getDefaultFormat:function(j,k){if(j=="standard"){if(k=="input"){return"yyyy-mm-dd hh:ii"}else{return"yyyy-mm-dd hh:ii:ss"}}else{if(j=="php"){if(k=="input"){return"Y-m-d H:i"}else{return"Y-m-d H:i:s"}}else{throw new Error("Invalid format type.")}}},validParts:function(j){if(j=="standard"){return/t|hh?|HH?|p|P|ii?|ss?|dd?|DD?|mm?|MM?|yy(?:yy)?/g}else{if(j=="php"){return/[dDjlNwzFmMnStyYaABgGhHis]/g}else{throw new Error("Invalid format type.")}}},nonpunctuation:/[^ -\/:-@\[-`{-~\t\n\rTZ]+/g,parseFormat:function(m,k){var j=m.replace(this.validParts(k),"\0").split("\0"),l=m.match(this.validParts(k));if(!j||!j.length||!l||l.length==0){throw new Error("Invalid date format.")}return{separators:j,parts:l}},parseDate:function(n,w,q,u){if(n instanceof Date){var y=new Date(n.valueOf()-n.getTimezoneOffset()*60000);y.setMilliseconds(0);return y}if(/^\d{4}\-\d{1,2}\-\d{1,2}$/.test(n)){w=this.parseFormat("yyyy-mm-dd",u)}if(/^\d{4}\-\d{1,2}\-\d{1,2}[T ]\d{1,2}\:\d{1,2}$/.test(n)){w=this.parseFormat("yyyy-mm-dd hh:ii",u)}if(/^\d{4}\-\d{1,2}\-\d{1,2}[T ]\d{1,2}\:\d{1,2}\:\d{1,2}[Z]{0,1}$/.test(n)){w=this.parseFormat("yyyy-mm-dd hh:ii:ss",u)}if(/^[-+]\d+[dmwy]([\s,]+[-+]\d+[dmwy])*$/.test(n)){var z=/([-+]\d+)([dmwy])/,o=n.match(/([-+]\d+)([dmwy])/g),j,m;n=new Date();for(var p=0;p<o.length;p++){j=z.exec(o[p]);m=parseInt(j[1]);switch(j[2]){case"d":n.setUTCDate(n.getUTCDate()+m);break;case"m":n=i.prototype.moveMonth.call(i.prototype,n,m);break;case"w":n.setUTCDate(n.getUTCDate()+m*7);break;case"y":n=i.prototype.moveYear.call(i.prototype,n,m);break}}return h(n.getUTCFullYear(),n.getUTCMonth(),n.getUTCDate(),n.getUTCHours(),n.getUTCMinutes(),n.getUTCSeconds(),0)}var o=n&&n.toString().match(this.nonpunctuation)||[],n=new Date(0,0,0,0,0,0,0),t={},v=["hh","h","ii","i","ss","s","yyyy","yy","M","MM","m","mm","D","DD","d","dd","H","HH","p","P"],x={hh:function(B,s){return B.setUTCHours(s)},h:function(B,s){return B.setUTCHours(s)},HH:function(B,s){return B.setUTCHours(s==12?0:s)},H:function(B,s){return B.setUTCHours(s==12?0:s)},ii:function(B,s){return B.setUTCMinutes(s)},i:function(B,s){return B.setUTCMinutes(s)},ss:function(B,s){return B.setUTCSeconds(s)},s:function(B,s){return B.setUTCSeconds(s)},yyyy:function(B,s){return B.setUTCFullYear(s)},yy:function(B,s){return B.setUTCFullYear(2000+s)},m:function(B,s){s-=1;while(s<0){s+=12}s%=12;B.setUTCMonth(s);while(B.getUTCMonth()!=s){if(isNaN(B.getUTCMonth())){return B}else{B.setUTCDate(B.getUTCDate()-1)}}return B},d:function(B,s){return B.setUTCDate(s)},p:function(B,s){return B.setUTCHours(s==1?B.getUTCHours()+12:B.getUTCHours())}},l,r,j;x.M=x.MM=x.mm=x.m;x.dd=x.d;x.P=x.p;n=h(n.getFullYear(),n.getMonth(),n.getDate(),n.getHours(),n.getMinutes(),n.getSeconds());if(o.length==w.parts.length){for(var p=0,k=w.parts.length;p<k;p++){l=parseInt(o[p],10);j=w.parts[p];if(isNaN(l)){switch(j){case"MM":r=f(a[q].months).filter(function(){var s=this.slice(0,o[p].length),B=o[p].slice(0,s.length);return s==B});l=f.inArray(r[0],a[q].months)+1;break;case"M":r=f(a[q].monthsShort).filter(function(){var s=this.slice(0,o[p].length),B=o[p].slice(0,s.length);return s.toLowerCase()==B.toLowerCase()});l=f.inArray(r[0],a[q].monthsShort)+1;break;case"p":case"P":l=f.inArray(o[p].toLowerCase(),a[q].meridiem);break}}t[j]=l}for(var p=0,A;p<v.length;p++){A=v[p];if(A in t&&!isNaN(t[A])){x[A](n,t[A])}}}return n},formatDate:function(j,o,q,m){if(j==null){return""}var p;if(m=="standard"){p={t:j.getTime(),yy:j.getUTCFullYear().toString().substring(2),yyyy:j.getUTCFullYear(),m:j.getUTCMonth()+1,M:a[q].monthsShort[j.getUTCMonth()],MM:a[q].months[j.getUTCMonth()],d:j.getUTCDate(),D:a[q].daysShort[j.getUTCDay()],DD:a[q].days[j.getUTCDay()],p:(a[q].meridiem.length==2?a[q].meridiem[j.getUTCHours()<12?0:1]:""),h:j.getUTCHours(),i:j.getUTCMinutes(),s:j.getUTCSeconds()};if(a[q].meridiem.length==2){p.H=(p.h%12==0?12:p.h%12)}else{p.H=p.h}p.HH=(p.H<10?"0":"")+p.H;p.P=p.p.toUpperCase();p.hh=(p.h<10?"0":"")+p.h;p.ii=(p.i<10?"0":"")+p.i;p.ss=(p.s<10?"0":"")+p.s;p.dd=(p.d<10?"0":"")+p.d;p.mm=(p.m<10?"0":"")+p.m}else{if(m=="php"){p={y:j.getUTCFullYear().toString().substring(2),Y:j.getUTCFullYear(),F:a[q].months[j.getUTCMonth()],M:a[q].monthsShort[j.getUTCMonth()],n:j.getUTCMonth()+1,t:g.getDaysInMonth(j.getUTCFullYear(),j.getUTCMonth()),j:j.getUTCDate(),l:a[q].days[j.getUTCDay()],D:a[q].daysShort[j.getUTCDay()],w:j.getUTCDay(),N:(j.getUTCDay()==0?7:j.getUTCDay()),S:(j.getUTCDate()%10<=a[q].suffix.length?a[q].suffix[j.getUTCDate()%10-1]:""),a:(a[q].meridiem.length==2?a[q].meridiem[j.getUTCHours()<12?0:1]:""),g:(j.getUTCHours()%12==0?12:j.getUTCHours()%12),G:j.getUTCHours(),i:j.getUTCMinutes(),s:j.getUTCSeconds()};p.m=(p.n<10?"0":"")+p.n;p.d=(p.j<10?"0":"")+p.j;p.A=p.a.toString().toUpperCase();p.h=(p.g<10?"0":"")+p.g;p.H=(p.G<10?"0":"")+p.G;p.i=(p.i<10?"0":"")+p.i;p.s=(p.s<10?"0":"")+p.s}else{throw new Error("Invalid format type.")}}var j=[],n=f.extend([],o.separators);for(var l=0,k=o.parts.length;l<k;l++){if(n.length){j.push(n.shift())}j.push(p[o.parts[l]])}if(n.length){j.push(n.shift())}return j.join("")},convertViewMode:function(j){switch(j){case 4:case"decade":j=4;break;case 3:case"year":j=3;break;case 2:case"month":j=2;break;case 1:case"day":j=1;break;case 0:case"hour":j=0;break}return j},headTemplate:'<thead><tr><th class="prev"><i class="{iconType} {leftArrow}"/></th><th colspan="5" class="switch"></th><th class="next"><i class="{iconType} {rightArrow}"/></th></tr></thead>',headTemplateV3:'<thead><tr><th class="prev"><span class="{iconType} {leftArrow}"></span> </th><th colspan="5" class="switch"></th><th class="next"><span class="{iconType} {rightArrow}"></span> </th></tr></thead>',contTemplate:'<tbody><tr><td colspan="7"></td></tr></tbody>',footTemplate:'<tfoot><tr><th colspan="7" class="today"></th></tr><tr><th colspan="7" class="clear"></th></tr></tfoot>'};g.template='<div class="datetimepicker"><div class="datetimepicker-minutes"><table class=" table-condensed">'+g.headTemplate+g.contTemplate+g.footTemplate+'</table></div><div class="datetimepicker-hours"><table class=" table-condensed">'+g.headTemplate+g.contTemplate+g.footTemplate+'</table></div><div class="datetimepicker-days"><table class=" table-condensed">'+g.headTemplate+"<tbody></tbody>"+g.footTemplate+'</table></div><div class="datetimepicker-months"><table class="table-condensed">'+g.headTemplate+g.contTemplate+g.footTemplate+'</table></div><div class="datetimepicker-years"><table class="table-condensed">'+g.headTemplate+g.contTemplate+g.footTemplate+"</table></div></div>";g.templateV3='<div class="datetimepicker"><div class="datetimepicker-minutes"><table class=" table-condensed">'+g.headTemplateV3+g.contTemplate+g.footTemplate+'</table></div><div class="datetimepicker-hours"><table class=" table-condensed">'+g.headTemplateV3+g.contTemplate+g.footTemplate+'</table></div><div class="datetimepicker-days"><table class=" table-condensed">'+g.headTemplateV3+"<tbody></tbody>"+g.footTemplate+'</table></div><div class="datetimepicker-months"><table class="table-condensed">'+g.headTemplateV3+g.contTemplate+g.footTemplate+'</table></div><div class="datetimepicker-years"><table class="table-condensed">'+g.headTemplateV3+g.contTemplate+g.footTemplate+"</table></div></div>";f.fn.datetimepicker.DPGlobal=g;f.fn.datetimepicker.noConflict=function(){f.fn.datetimepicker=b;return this};f(document).on("focus.datetimepicker.data-api click.datetimepicker.data-api",'[data-provide="datetimepicker"]',function(k){var j=f(this);if(j.data("datetimepicker")){return}k.preventDefault();j.datetimepicker("show")});f(function(){f('[data-provide="datetimepicker-inline"]').datetimepicker()})}));
|
|
/*! SWFObject v2.3.20130521 <http://github.com/swfobject/swfobject>
|
|
is released under the MIT License <http://www.opensource.org/licenses/mit-license.php>
|
|
*/
|
|
var swfobject=function(){var D="undefined",r="object",T="Shockwave Flash",Z="ShockwaveFlash.ShockwaveFlash",q="application/x-shockwave-flash",S="SWFObjectExprInst",x="onreadystatechange",Q=window,h=document,t=navigator,V=false,X=[],o=[],P=[],K=[],I,p,E,B,L=false,a=false,m,G,j=true,l=false,O=function(){var ad=typeof h.getElementById!=D&&typeof h.getElementsByTagName!=D&&typeof h.createElement!=D,ak=t.userAgent.toLowerCase(),ab=t.platform.toLowerCase(),ah=ab?/win/.test(ab):/win/.test(ak),af=ab?/mac/.test(ab):/mac/.test(ak),ai=/webkit/.test(ak)?parseFloat(ak.replace(/^.*webkit\/(\d+(\.\d+)?).*$/,"$1")):false,aa=t.appName==="Microsoft Internet Explorer",aj=[0,0,0],ae=null;if(typeof t.plugins!=D&&typeof t.plugins[T]==r){ae=t.plugins[T].description;if(ae&&(typeof t.mimeTypes!=D&&t.mimeTypes[q]&&t.mimeTypes[q].enabledPlugin)){V=true;aa=false;ae=ae.replace(/^.*\s+(\S+\s+\S+$)/,"$1");aj[0]=n(ae.replace(/^(.*)\..*$/,"$1"));aj[1]=n(ae.replace(/^.*\.(.*)\s.*$/,"$1"));aj[2]=/[a-zA-Z]/.test(ae)?n(ae.replace(/^.*[a-zA-Z]+(.*)$/,"$1")):0}}else{if(typeof Q.ActiveXObject!=D){try{var ag=new ActiveXObject(Z);if(ag){ae=ag.GetVariable("$version");if(ae){aa=true;ae=ae.split(" ")[1].split(",");aj=[n(ae[0]),n(ae[1]),n(ae[2])]}}}catch(ac){}}}return{w3:ad,pv:aj,wk:ai,ie:aa,win:ah,mac:af}}(),i=function(){if(!O.w3){return}if((typeof h.readyState!=D&&(h.readyState==="complete"||h.readyState==="interactive"))||(typeof h.readyState==D&&(h.getElementsByTagName("body")[0]||h.body))){f()}if(!L){if(typeof h.addEventListener!=D){h.addEventListener("DOMContentLoaded",f,false)}if(O.ie){h.attachEvent(x,function aa(){if(h.readyState=="complete"){h.detachEvent(x,aa);f()}});if(Q==top){(function ac(){if(L){return}try{h.documentElement.doScroll("left")}catch(ad){setTimeout(ac,0);return}f()}())}}if(O.wk){(function ab(){if(L){return}if(!/loaded|complete/.test(h.readyState)){setTimeout(ab,0);return}f()}())}}}();function f(){if(L||!document.getElementsByTagName("body")[0]){return}try{var ac,ad=C("span");ad.style.display="none";ac=h.getElementsByTagName("body")[0].appendChild(ad);ac.parentNode.removeChild(ac);ac=null;ad=null}catch(ae){return}L=true;var aa=X.length;for(var ab=0;ab<aa;ab++){X[ab]()}}function M(aa){if(L){aa()}else{X[X.length]=aa}}function s(ab){if(typeof Q.addEventListener!=D){Q.addEventListener("load",ab,false)}else{if(typeof h.addEventListener!=D){h.addEventListener("load",ab,false)}else{if(typeof Q.attachEvent!=D){g(Q,"onload",ab)}else{if(typeof Q.onload=="function"){var aa=Q.onload;Q.onload=function(){aa();ab()}}else{Q.onload=ab}}}}}function Y(){var aa=h.getElementsByTagName("body")[0];var ae=C(r);ae.setAttribute("style","visibility: hidden;");ae.setAttribute("type",q);var ad=aa.appendChild(ae);if(ad){var ac=0;(function ab(){if(typeof ad.GetVariable!=D){try{var ag=ad.GetVariable("$version");if(ag){ag=ag.split(" ")[1].split(",");O.pv=[n(ag[0]),n(ag[1]),n(ag[2])]}}catch(af){O.pv=[8,0,0]}}else{if(ac<10){ac++;setTimeout(ab,10);return}}aa.removeChild(ae);ad=null;H()}())}else{H()}}function H(){var aj=o.length;if(aj>0){for(var ai=0;ai<aj;ai++){var ab=o[ai].id;var ae=o[ai].callbackFn;var ad={success:false,id:ab};if(O.pv[0]>0){var ah=c(ab);if(ah){if(F(o[ai].swfVersion)&&!(O.wk&&O.wk<312)){w(ab,true);if(ae){ad.success=true;ad.ref=z(ab);ad.id=ab;ae(ad)}}else{if(o[ai].expressInstall&&A()){var al={};al.data=o[ai].expressInstall;al.width=ah.getAttribute("width")||"0";al.height=ah.getAttribute("height")||"0";if(ah.getAttribute("class")){al.styleclass=ah.getAttribute("class")}if(ah.getAttribute("align")){al.align=ah.getAttribute("align")}var ak={};var aa=ah.getElementsByTagName("param");var af=aa.length;for(var ag=0;ag<af;ag++){if(aa[ag].getAttribute("name").toLowerCase()!="movie"){ak[aa[ag].getAttribute("name")]=aa[ag].getAttribute("value")}}R(al,ak,ab,ae)}else{b(ah);if(ae){ae(ad)}}}}}else{w(ab,true);if(ae){var ac=z(ab);if(ac&&typeof ac.SetVariable!=D){ad.success=true;ad.ref=ac;ad.id=ac.id}ae(ad)}}}}}X[0]=function(){if(V){Y()}else{H()}};function z(ac){var aa=null,ab=c(ac);if(ab&&ab.nodeName.toUpperCase()==="OBJECT"){if(typeof ab.SetVariable!==D){aa=ab}else{aa=ab.getElementsByTagName(r)[0]||ab}}return aa}function A(){return !a&&F("6.0.65")&&(O.win||O.mac)&&!(O.wk&&O.wk<312)}function R(ad,ae,aa,ac){var ah=c(aa);aa=W(aa);a=true;E=ac||null;B={success:false,id:aa};if(ah){if(ah.nodeName.toUpperCase()=="OBJECT"){I=J(ah);p=null}else{I=ah;p=aa}ad.id=S;if(typeof ad.width==D||(!/%$/.test(ad.width)&&n(ad.width)<310)){ad.width="310"}if(typeof ad.height==D||(!/%$/.test(ad.height)&&n(ad.height)<137)){ad.height="137"}var ag=O.ie?"ActiveX":"PlugIn",af="MMredirectURL="+encodeURIComponent(Q.location.toString().replace(/&/g,"%26"))+"&MMplayerType="+ag+"&MMdoctitle="+encodeURIComponent(h.title.slice(0,47)+" - Flash Player Installation");if(typeof ae.flashvars!=D){ae.flashvars+="&"+af}else{ae.flashvars=af}if(O.ie&&ah.readyState!=4){var ab=C("div");
|
|
aa+="SWFObjectNew";ab.setAttribute("id",aa);ah.parentNode.insertBefore(ab,ah);ah.style.display="none";y(ah)}u(ad,ae,aa)}}function b(ab){if(O.ie&&ab.readyState!=4){ab.style.display="none";var aa=C("div");ab.parentNode.insertBefore(aa,ab);aa.parentNode.replaceChild(J(ab),aa);y(ab)}else{ab.parentNode.replaceChild(J(ab),ab)}}function J(af){var ae=C("div");if(O.win&&O.ie){ae.innerHTML=af.innerHTML}else{var ab=af.getElementsByTagName(r)[0];if(ab){var ag=ab.childNodes;if(ag){var aa=ag.length;for(var ad=0;ad<aa;ad++){if(!(ag[ad].nodeType==1&&ag[ad].nodeName=="PARAM")&&!(ag[ad].nodeType==8)){ae.appendChild(ag[ad].cloneNode(true))}}}}}return ae}function k(aa,ab){var ac=C("div");ac.innerHTML="<object classid='clsid:D27CDB6E-AE6D-11cf-96B8-444553540000'><param name='movie' value='"+aa+"'>"+ab+"</object>";return ac.firstChild}function u(ai,ag,ab){var aa,ad=c(ab);ab=W(ab);if(O.wk&&O.wk<312){return aa}if(ad){var ac=(O.ie)?C("div"):C(r),af,ah,ae;if(typeof ai.id==D){ai.id=ab}for(ae in ag){if(ag.hasOwnProperty(ae)&&ae.toLowerCase()!=="movie"){e(ac,ae,ag[ae])}}if(O.ie){ac=k(ai.data,ac.innerHTML)}for(af in ai){if(ai.hasOwnProperty(af)){ah=af.toLowerCase();if(ah==="styleclass"){ac.setAttribute("class",ai[af])}else{if(ah!=="classid"&&ah!=="data"){ac.setAttribute(af,ai[af])}}}}if(O.ie){P[P.length]=ai.id}else{ac.setAttribute("type",q);ac.setAttribute("data",ai.data)}ad.parentNode.replaceChild(ac,ad);aa=ac}return aa}function e(ac,aa,ab){var ad=C("param");ad.setAttribute("name",aa);ad.setAttribute("value",ab);ac.appendChild(ad)}function y(ac){var ab=c(ac);if(ab&&ab.nodeName.toUpperCase()=="OBJECT"){if(O.ie){ab.style.display="none";(function aa(){if(ab.readyState==4){for(var ad in ab){if(typeof ab[ad]=="function"){ab[ad]=null}}ab.parentNode.removeChild(ab)}else{setTimeout(aa,10)}}())}else{ab.parentNode.removeChild(ab)}}}function U(aa){return(aa&&aa.nodeType&&aa.nodeType===1)}function W(aa){return(U(aa))?aa.id:aa}function c(ac){if(U(ac)){return ac}var aa=null;try{aa=h.getElementById(ac)}catch(ab){}return aa}function C(aa){return h.createElement(aa)}function n(aa){return parseInt(aa,10)}function g(ac,aa,ab){ac.attachEvent(aa,ab);K[K.length]=[ac,aa,ab]}function F(ac){ac+="";var ab=O.pv,aa=ac.split(".");aa[0]=n(aa[0]);aa[1]=n(aa[1])||0;aa[2]=n(aa[2])||0;return(ab[0]>aa[0]||(ab[0]==aa[0]&&ab[1]>aa[1])||(ab[0]==aa[0]&&ab[1]==aa[1]&&ab[2]>=aa[2]))?true:false}function v(af,ab,ag,ae){var ad=h.getElementsByTagName("head")[0];if(!ad){return}var aa=(typeof ag=="string")?ag:"screen";if(ae){m=null;G=null}if(!m||G!=aa){var ac=C("style");ac.setAttribute("type","text/css");ac.setAttribute("media",aa);m=ad.appendChild(ac);if(O.ie&&typeof h.styleSheets!=D&&h.styleSheets.length>0){m=h.styleSheets[h.styleSheets.length-1]}G=aa}if(m){if(typeof m.addRule!=D){m.addRule(af,ab)}else{if(typeof h.createTextNode!=D){m.appendChild(h.createTextNode(af+" {"+ab+"}"))}}}}function w(ad,aa){if(!j){return}var ab=aa?"visible":"hidden",ac=c(ad);if(L&&ac){ac.style.visibility=ab}else{if(typeof ad==="string"){v("#"+ad,"visibility:"+ab)}}}function N(ab){var ac=/[\\\"<>\.;]/;var aa=ac.exec(ab)!=null;return aa&&typeof encodeURIComponent!=D?encodeURIComponent(ab):ab}var d=function(){if(O.ie){window.attachEvent("onunload",function(){var af=K.length;for(var ae=0;ae<af;ae++){K[ae][0].detachEvent(K[ae][1],K[ae][2])}var ac=P.length;for(var ad=0;ad<ac;ad++){y(P[ad])}for(var ab in O){O[ab]=null}O=null;for(var aa in swfobject){swfobject[aa]=null}swfobject=null})}}();return{registerObject:function(ae,aa,ad,ac){if(O.w3&&ae&&aa){var ab={};ab.id=ae;ab.swfVersion=aa;ab.expressInstall=ad;ab.callbackFn=ac;o[o.length]=ab;w(ae,false)}else{if(ac){ac({success:false,id:ae})}}},getObjectById:function(aa){if(O.w3){return z(aa)}},embedSWF:function(af,al,ai,ak,ab,ae,ad,ah,aj,ag){var ac=W(al),aa={success:false,id:ac};if(O.w3&&!(O.wk&&O.wk<312)&&af&&al&&ai&&ak&&ab){w(ac,false);M(function(){ai+="";ak+="";var an={};if(aj&&typeof aj===r){for(var aq in aj){an[aq]=aj[aq]}}an.data=af;an.width=ai;an.height=ak;var ar={};if(ah&&typeof ah===r){for(var ao in ah){ar[ao]=ah[ao]}}if(ad&&typeof ad===r){for(var am in ad){if(ad.hasOwnProperty(am)){var ap=(l)?encodeURIComponent(am):am,at=(l)?encodeURIComponent(ad[am]):ad[am];if(typeof ar.flashvars!=D){ar.flashvars+="&"+ap+"="+at}else{ar.flashvars=ap+"="+at}}}}if(F(ab)){var au=u(an,ar,al);if(an.id==ac){w(ac,true)}aa.success=true;aa.ref=au;aa.id=au.id}else{if(ae&&A()){an.data=ae;R(an,ar,al,ag);return}else{w(ac,true)}}if(ag){ag(aa)}})}else{if(ag){ag(aa)}}},switchOffAutoHideShow:function(){j=false},enableUriEncoding:function(aa){l=(typeof aa===D)?true:aa},ua:O,getFlashPlayerVersion:function(){return{major:O.pv[0],minor:O.pv[1],release:O.pv[2]}},hasFlashPlayerVersion:F,createSWF:function(ac,ab,aa){if(O.w3){return u(ac,ab,aa)}else{return undefined}},showExpressInstall:function(ac,ad,aa,ab){if(O.w3&&A()){R(ac,ad,aa,ab)}},removeSWF:function(aa){if(O.w3){y(aa)}},createCSS:function(ad,ac,ab,aa){if(O.w3){v(ad,ac,ab,aa)}},addDomLoadEvent:M,addLoadEvent:s,getQueryParamValue:function(ad){var ac=h.location.search||h.location.hash;
|
|
if(ac){if(/\?/.test(ac)){ac=ac.split("?")[1]}if(ad==null){return N(ac)}var ab=ac.split("&");for(var aa=0;aa<ab.length;aa++){if(ab[aa].substring(0,ab[aa].indexOf("="))==ad){return N(ab[aa].substring((ab[aa].indexOf("=")+1)))}}}return""},expressInstallCallback:function(){if(a){var aa=c(S);if(aa&&I){aa.parentNode.replaceChild(I,aa);if(p){w(p,true);if(O.ie){I.style.display="block"}}if(E){E(B)}}a=false}},version:"2.3"}}();
|
|
|
|
/*! tablesorter (FORK) - updated 04-11-2016 (v2.25.8)*/
|
|
/* Includes widgets ( storage,uitheme,columns,filter,stickyHeaders,resizable,saveSort ) */
|
|
(function(factory) {
|
|
if (typeof define === 'function' && define.amd) {
|
|
define(['jquery'], factory);
|
|
} else if (typeof module === 'object' && typeof module.exports === 'object') {
|
|
module.exports = factory(require('jquery'));
|
|
} else {
|
|
factory(jQuery);
|
|
}
|
|
}(function($) {
|
|
|
|
/*! TableSorter (FORK) v2.25.8 *//*
|
|
* Client-side table sorting with ease!
|
|
* @requires jQuery v1.2.6+
|
|
*
|
|
* Copyright (c) 2007 Christian Bach
|
|
* fork maintained by Rob Garrison
|
|
*
|
|
* Examples and docs at: http://tablesorter.com
|
|
* Dual licensed under the MIT and GPL licenses:
|
|
* http://www.opensource.org/licenses/mit-license.php
|
|
* http://www.gnu.org/licenses/gpl.html
|
|
*
|
|
* @type jQuery
|
|
* @name tablesorter (FORK)
|
|
* @cat Plugins/Tablesorter
|
|
* @author Christian Bach - christian.bach@polyester.se
|
|
* @contributor Rob Garrison - https://github.com/Mottie/tablesorter
|
|
*/
|
|
/*jshint browser:true, jquery:true, unused:false, expr: true */
|
|
;( function( $ ) {
|
|
'use strict';
|
|
var ts = $.tablesorter = {
|
|
|
|
version : '2.25.8',
|
|
|
|
parsers : [],
|
|
widgets : [],
|
|
defaults : {
|
|
|
|
// *** appearance
|
|
theme : 'default', // adds tablesorter-{theme} to the table for styling
|
|
widthFixed : false, // adds colgroup to fix widths of columns
|
|
showProcessing : false, // show an indeterminate timer icon in the header when the table is sorted or filtered.
|
|
|
|
headerTemplate : '{content}',// header layout template (HTML ok); {content} = innerHTML, {icon} = <i/> // class from cssIcon
|
|
onRenderTemplate : null, // function( index, template ){ return template; }, // template is a string
|
|
onRenderHeader : null, // function( index ){}, // nothing to return
|
|
|
|
// *** functionality
|
|
cancelSelection : true, // prevent text selection in the header
|
|
tabIndex : true, // add tabindex to header for keyboard accessibility
|
|
dateFormat : 'mmddyyyy', // other options: 'ddmmyyy' or 'yyyymmdd'
|
|
sortMultiSortKey : 'shiftKey', // key used to select additional columns
|
|
sortResetKey : 'ctrlKey', // key used to remove sorting on a column
|
|
usNumberFormat : true, // false for German '1.234.567,89' or French '1 234 567,89'
|
|
delayInit : false, // if false, the parsed table contents will not update until the first sort
|
|
serverSideSorting: false, // if true, server-side sorting should be performed because client-side sorting will be disabled, but the ui and events will still be used.
|
|
resort : true, // default setting to trigger a resort after an 'update', 'addRows', 'updateCell', etc has completed
|
|
|
|
// *** sort options
|
|
headers : {}, // set sorter, string, empty, locked order, sortInitialOrder, filter, etc.
|
|
ignoreCase : true, // ignore case while sorting
|
|
sortForce : null, // column(s) first sorted; always applied
|
|
sortList : [], // Initial sort order; applied initially; updated when manually sorted
|
|
sortAppend : null, // column(s) sorted last; always applied
|
|
sortStable : false, // when sorting two rows with exactly the same content, the original sort order is maintained
|
|
|
|
sortInitialOrder : 'asc', // sort direction on first click
|
|
sortLocaleCompare: false, // replace equivalent character (accented characters)
|
|
sortReset : false, // third click on the header will reset column to default - unsorted
|
|
sortRestart : false, // restart sort to 'sortInitialOrder' when clicking on previously unsorted columns
|
|
|
|
emptyTo : 'bottom', // sort empty cell to bottom, top, none, zero, emptyMax, emptyMin
|
|
stringTo : 'max', // sort strings in numerical column as max, min, top, bottom, zero
|
|
duplicateSpan : true, // colspan cells in the tbody will have duplicated content in the cache for each spanned column
|
|
textExtraction : 'basic', // text extraction method/function - function( node, table, cellIndex ){}
|
|
textAttribute : 'data-text',// data-attribute that contains alternate cell text (used in default textExtraction function)
|
|
textSorter : null, // choose overall or specific column sorter function( a, b, direction, table, columnIndex ) [alt: ts.sortText]
|
|
numberSorter : null, // choose overall numeric sorter function( a, b, direction, maxColumnValue )
|
|
|
|
// *** widget options
|
|
widgets: [], // method to add widgets, e.g. widgets: ['zebra']
|
|
widgetOptions : {
|
|
zebra : [ 'even', 'odd' ] // zebra widget alternating row class names
|
|
},
|
|
initWidgets : true, // apply widgets on tablesorter initialization
|
|
widgetClass : 'widget-{name}', // table class name template to match to include a widget
|
|
|
|
// *** callbacks
|
|
initialized : null, // function( table ){},
|
|
|
|
// *** extra css class names
|
|
tableClass : '',
|
|
cssAsc : '',
|
|
cssDesc : '',
|
|
cssNone : '',
|
|
cssHeader : '',
|
|
cssHeaderRow : '',
|
|
cssProcessing : '', // processing icon applied to header during sort/filter
|
|
|
|
cssChildRow : 'tablesorter-childRow', // class name indiciating that a row is to be attached to the its parent
|
|
cssInfoBlock : 'tablesorter-infoOnly', // don't sort tbody with this class name (only one class name allowed here!)
|
|
cssNoSort : 'tablesorter-noSort', // class name added to element inside header; clicking on it won't cause a sort
|
|
cssIgnoreRow : 'tablesorter-ignoreRow', // header row to ignore; cells within this row will not be added to c.$headers
|
|
|
|
cssIcon : 'tablesorter-icon', // if this class does not exist, the {icon} will not be added from the headerTemplate
|
|
cssIconNone : '', // class name added to the icon when there is no column sort
|
|
cssIconAsc : '', // class name added to the icon when the column has an ascending sort
|
|
cssIconDesc : '', // class name added to the icon when the column has a descending sort
|
|
|
|
// *** events
|
|
pointerClick : 'click',
|
|
pointerDown : 'mousedown',
|
|
pointerUp : 'mouseup',
|
|
|
|
// *** selectors
|
|
selectorHeaders : '> thead th, > thead td',
|
|
selectorSort : 'th, td', // jQuery selector of content within selectorHeaders that is clickable to trigger a sort
|
|
selectorRemove : '.remove-me',
|
|
|
|
// *** advanced
|
|
debug : false,
|
|
|
|
// *** Internal variables
|
|
headerList: [],
|
|
empties: {},
|
|
strings: {},
|
|
parsers: []
|
|
|
|
// removed: widgetZebra: { css: ['even', 'odd'] }
|
|
|
|
},
|
|
|
|
// internal css classes - these will ALWAYS be added to
|
|
// the table and MUST only contain one class name - fixes #381
|
|
css : {
|
|
table : 'tablesorter',
|
|
cssHasChild: 'tablesorter-hasChildRow',
|
|
childRow : 'tablesorter-childRow',
|
|
colgroup : 'tablesorter-colgroup',
|
|
header : 'tablesorter-header',
|
|
headerRow : 'tablesorter-headerRow',
|
|
headerIn : 'tablesorter-header-inner',
|
|
icon : 'tablesorter-icon',
|
|
processing : 'tablesorter-processing',
|
|
sortAsc : 'tablesorter-headerAsc',
|
|
sortDesc : 'tablesorter-headerDesc',
|
|
sortNone : 'tablesorter-headerUnSorted'
|
|
},
|
|
|
|
// labels applied to sortable headers for accessibility (aria) support
|
|
language : {
|
|
sortAsc : 'Ascending sort applied, ',
|
|
sortDesc : 'Descending sort applied, ',
|
|
sortNone : 'No sort applied, ',
|
|
sortDisabled : 'sorting is disabled',
|
|
nextAsc : 'activate to apply an ascending sort',
|
|
nextDesc : 'activate to apply a descending sort',
|
|
nextNone : 'activate to remove the sort'
|
|
},
|
|
|
|
regex : {
|
|
templateContent : /\{content\}/g,
|
|
templateIcon : /\{icon\}/g,
|
|
templateName : /\{name\}/i,
|
|
spaces : /\s+/g,
|
|
nonWord : /\W/g,
|
|
formElements : /(input|select|button|textarea)/i,
|
|
|
|
// *** sort functions ***
|
|
// regex used in natural sort
|
|
// chunk/tokenize numbers & letters
|
|
chunk : /(^([+\-]?(?:\d*)(?:\.\d*)?(?:[eE][+\-]?\d+)?)?$|^0x[0-9a-f]+$|\d+)/gi,
|
|
// replace chunks @ ends
|
|
chunks : /(^\\0|\\0$)/,
|
|
hex : /^0x[0-9a-f]+$/i,
|
|
|
|
// *** formatFloat ***
|
|
comma : /,/g,
|
|
digitNonUS : /[\s|\.]/g,
|
|
digitNegativeTest : /^\s*\([.\d]+\)/,
|
|
digitNegativeReplace : /^\s*\(([.\d]+)\)/,
|
|
|
|
// *** isDigit ***
|
|
digitTest : /^[\-+(]?\d+[)]?$/,
|
|
digitReplace : /[,.'"\s]/g
|
|
|
|
},
|
|
|
|
// digit sort, text location
|
|
string : {
|
|
max : 1,
|
|
min : -1,
|
|
emptymin : 1,
|
|
emptymax : -1,
|
|
zero : 0,
|
|
none : 0,
|
|
'null' : 0,
|
|
top : true,
|
|
bottom : false
|
|
},
|
|
|
|
keyCodes : {
|
|
enter : 13
|
|
},
|
|
|
|
// placeholder date parser data (globalize)
|
|
dates : {},
|
|
|
|
// These methods can be applied on table.config instance
|
|
instanceMethods : {},
|
|
|
|
/*
|
|
▄█████ ██████ ██████ ██ ██ █████▄
|
|
▀█▄ ██▄▄ ██ ██ ██ ██▄▄██
|
|
▀█▄ ██▀▀ ██ ██ ██ ██▀▀▀
|
|
█████▀ ██████ ██ ▀████▀ ██
|
|
*/
|
|
|
|
setup : function( table, c ) {
|
|
// if no thead or tbody, or tablesorter is already present, quit
|
|
if ( !table || !table.tHead || table.tBodies.length === 0 || table.hasInitialized === true ) {
|
|
if ( c.debug ) {
|
|
if ( table.hasInitialized ) {
|
|
console.warn( 'Stopping initialization. Tablesorter has already been initialized' );
|
|
} else {
|
|
console.error( 'Stopping initialization! No table, thead or tbody', table );
|
|
}
|
|
}
|
|
return;
|
|
}
|
|
|
|
var tmp = '',
|
|
$table = $( table ),
|
|
meta = $.metadata;
|
|
// initialization flag
|
|
table.hasInitialized = false;
|
|
// table is being processed flag
|
|
table.isProcessing = true;
|
|
// make sure to store the config object
|
|
table.config = c;
|
|
// save the settings where they read
|
|
$.data( table, 'tablesorter', c );
|
|
if ( c.debug ) {
|
|
console[ console.group ? 'group' : 'log' ]( 'Initializing tablesorter' );
|
|
$.data( table, 'startoveralltimer', new Date() );
|
|
}
|
|
|
|
// removing this in version 3 (only supports jQuery 1.7+)
|
|
c.supportsDataObject = ( function( version ) {
|
|
version[ 0 ] = parseInt( version[ 0 ], 10 );
|
|
return ( version[ 0 ] > 1 ) || ( version[ 0 ] === 1 && parseInt( version[ 1 ], 10 ) >= 4 );
|
|
})( $.fn.jquery.split( '.' ) );
|
|
// ensure case insensitivity
|
|
c.emptyTo = c.emptyTo.toLowerCase();
|
|
c.stringTo = c.stringTo.toLowerCase();
|
|
c.last = { sortList : [], clickedIndex : -1 };
|
|
// add table theme class only if there isn't already one there
|
|
if ( !/tablesorter\-/.test( $table.attr( 'class' ) ) ) {
|
|
tmp = ( c.theme !== '' ? ' tablesorter-' + c.theme : '' );
|
|
}
|
|
c.table = table;
|
|
c.$table = $table
|
|
.addClass( ts.css.table + ' ' + c.tableClass + tmp )
|
|
.attr( 'role', 'grid' );
|
|
c.$headers = $table.find( c.selectorHeaders );
|
|
|
|
// give the table a unique id, which will be used in namespace binding
|
|
if ( !c.namespace ) {
|
|
c.namespace = '.tablesorter' + Math.random().toString( 16 ).slice( 2 );
|
|
} else {
|
|
// make sure namespace starts with a period & doesn't have weird characters
|
|
c.namespace = '.' + c.namespace.replace( ts.regex.nonWord, '' );
|
|
}
|
|
|
|
c.$table.children().children( 'tr' ).attr( 'role', 'row' );
|
|
c.$tbodies = $table.children( 'tbody:not(.' + c.cssInfoBlock + ')' ).attr({
|
|
'aria-live' : 'polite',
|
|
'aria-relevant' : 'all'
|
|
});
|
|
if ( c.$table.children( 'caption' ).length ) {
|
|
tmp = c.$table.children( 'caption' )[ 0 ];
|
|
if ( !tmp.id ) { tmp.id = c.namespace.slice( 1 ) + 'caption'; }
|
|
c.$table.attr( 'aria-labelledby', tmp.id );
|
|
}
|
|
c.widgetInit = {}; // keep a list of initialized widgets
|
|
// change textExtraction via data-attribute
|
|
c.textExtraction = c.$table.attr( 'data-text-extraction' ) || c.textExtraction || 'basic';
|
|
// build headers
|
|
ts.buildHeaders( c );
|
|
// fixate columns if the users supplies the fixedWidth option
|
|
// do this after theme has been applied
|
|
ts.fixColumnWidth( table );
|
|
// add widgets from class name
|
|
ts.addWidgetFromClass( table );
|
|
// add widget options before parsing (e.g. grouping widget has parser settings)
|
|
ts.applyWidgetOptions( table );
|
|
// try to auto detect column type, and store in tables config
|
|
ts.setupParsers( c );
|
|
// start total row count at zero
|
|
c.totalRows = 0;
|
|
// build the cache for the tbody cells
|
|
// delayInit will delay building the cache until the user starts a sort
|
|
if ( !c.delayInit ) { ts.buildCache( c ); }
|
|
// bind all header events and methods
|
|
ts.bindEvents( table, c.$headers, true );
|
|
ts.bindMethods( c );
|
|
// get sort list from jQuery data or metadata
|
|
// in jQuery < 1.4, an error occurs when calling $table.data()
|
|
if ( c.supportsDataObject && typeof $table.data().sortlist !== 'undefined' ) {
|
|
c.sortList = $table.data().sortlist;
|
|
} else if ( meta && ( $table.metadata() && $table.metadata().sortlist ) ) {
|
|
c.sortList = $table.metadata().sortlist;
|
|
}
|
|
// apply widget init code
|
|
ts.applyWidget( table, true );
|
|
// if user has supplied a sort list to constructor
|
|
if ( c.sortList.length > 0 ) {
|
|
ts.sortOn( c, c.sortList, {}, !c.initWidgets );
|
|
} else {
|
|
ts.setHeadersCss( c );
|
|
if ( c.initWidgets ) {
|
|
// apply widget format
|
|
ts.applyWidget( table, false );
|
|
}
|
|
}
|
|
|
|
// show processesing icon
|
|
if ( c.showProcessing ) {
|
|
$table
|
|
.unbind( 'sortBegin' + c.namespace + ' sortEnd' + c.namespace )
|
|
.bind( 'sortBegin' + c.namespace + ' sortEnd' + c.namespace, function( e ) {
|
|
clearTimeout( c.timerProcessing );
|
|
ts.isProcessing( table );
|
|
if ( e.type === 'sortBegin' ) {
|
|
c.timerProcessing = setTimeout( function() {
|
|
ts.isProcessing( table, true );
|
|
}, 500 );
|
|
}
|
|
});
|
|
}
|
|
|
|
// initialized
|
|
table.hasInitialized = true;
|
|
table.isProcessing = false;
|
|
if ( c.debug ) {
|
|
console.log( 'Overall initialization time: ' + ts.benchmark( $.data( table, 'startoveralltimer' ) ) );
|
|
if ( c.debug && console.groupEnd ) { console.groupEnd(); }
|
|
}
|
|
$table.triggerHandler( 'tablesorter-initialized', table );
|
|
if ( typeof c.initialized === 'function' ) {
|
|
c.initialized( table );
|
|
}
|
|
},
|
|
|
|
bindMethods : function( c ) {
|
|
var $table = c.$table,
|
|
namespace = c.namespace,
|
|
events = ( 'sortReset update updateRows updateAll updateHeaders addRows updateCell updateComplete ' +
|
|
'sorton appendCache updateCache applyWidgetId applyWidgets refreshWidgets destroy mouseup ' +
|
|
'mouseleave ' ).split( ' ' )
|
|
.join( namespace + ' ' );
|
|
// apply easy methods that trigger bound events
|
|
$table
|
|
.unbind( events.replace( ts.regex.spaces, ' ' ) )
|
|
.bind( 'sortReset' + namespace, function( e, callback ) {
|
|
e.stopPropagation();
|
|
// using this.config to ensure functions are getting a non-cached version of the config
|
|
ts.sortReset( this.config, callback );
|
|
})
|
|
.bind( 'updateAll' + namespace, function( e, resort, callback ) {
|
|
e.stopPropagation();
|
|
ts.updateAll( this.config, resort, callback );
|
|
})
|
|
.bind( 'update' + namespace + ' updateRows' + namespace, function( e, resort, callback ) {
|
|
e.stopPropagation();
|
|
ts.update( this.config, resort, callback );
|
|
})
|
|
.bind( 'updateHeaders' + namespace, function( e, callback ) {
|
|
e.stopPropagation();
|
|
ts.updateHeaders( this.config, callback );
|
|
})
|
|
.bind( 'updateCell' + namespace, function( e, cell, resort, callback ) {
|
|
e.stopPropagation();
|
|
ts.updateCell( this.config, cell, resort, callback );
|
|
})
|
|
.bind( 'addRows' + namespace, function( e, $row, resort, callback ) {
|
|
e.stopPropagation();
|
|
ts.addRows( this.config, $row, resort, callback );
|
|
})
|
|
.bind( 'updateComplete' + namespace, function() {
|
|
this.isUpdating = false;
|
|
})
|
|
.bind( 'sorton' + namespace, function( e, list, callback, init ) {
|
|
e.stopPropagation();
|
|
ts.sortOn( this.config, list, callback, init );
|
|
})
|
|
.bind( 'appendCache' + namespace, function( e, callback, init ) {
|
|
e.stopPropagation();
|
|
ts.appendCache( this.config, init );
|
|
if ( $.isFunction( callback ) ) {
|
|
callback( this );
|
|
}
|
|
})
|
|
// $tbodies variable is used by the tbody sorting widget
|
|
.bind( 'updateCache' + namespace, function( e, callback, $tbodies ) {
|
|
e.stopPropagation();
|
|
ts.updateCache( this.config, callback, $tbodies );
|
|
})
|
|
.bind( 'applyWidgetId' + namespace, function( e, id ) {
|
|
e.stopPropagation();
|
|
ts.applyWidgetId( this, id );
|
|
})
|
|
.bind( 'applyWidgets' + namespace, function( e, init ) {
|
|
e.stopPropagation();
|
|
// apply widgets
|
|
ts.applyWidget( this, init );
|
|
})
|
|
.bind( 'refreshWidgets' + namespace, function( e, all, dontapply ) {
|
|
e.stopPropagation();
|
|
ts.refreshWidgets( this, all, dontapply );
|
|
})
|
|
.bind( 'removeWidget' + namespace, function( e, name, refreshing ) {
|
|
e.stopPropagation();
|
|
ts.removeWidget( this, name, refreshing );
|
|
})
|
|
.bind( 'destroy' + namespace, function( e, removeClasses, callback ) {
|
|
e.stopPropagation();
|
|
ts.destroy( this, removeClasses, callback );
|
|
})
|
|
.bind( 'resetToLoadState' + namespace, function( e ) {
|
|
e.stopPropagation();
|
|
// remove all widgets
|
|
ts.removeWidget( this, true, false );
|
|
// restore original settings; this clears out current settings, but does not clear
|
|
// values saved to storage.
|
|
c = $.extend( true, ts.defaults, c.originalSettings );
|
|
this.hasInitialized = false;
|
|
// setup the entire table again
|
|
ts.setup( this, c );
|
|
});
|
|
},
|
|
|
|
bindEvents : function( table, $headers, core ) {
|
|
table = $( table )[ 0 ];
|
|
var tmp,
|
|
c = table.config,
|
|
namespace = c.namespace,
|
|
downTarget = null;
|
|
if ( core !== true ) {
|
|
$headers.addClass( namespace.slice( 1 ) + '_extra_headers' );
|
|
tmp = $.fn.closest ? $headers.closest( 'table' )[ 0 ] : $headers.parents( 'table' )[ 0 ];
|
|
if ( tmp && tmp.nodeName === 'TABLE' && tmp !== table ) {
|
|
$( tmp ).addClass( namespace.slice( 1 ) + '_extra_table' );
|
|
}
|
|
}
|
|
tmp = ( c.pointerDown + ' ' + c.pointerUp + ' ' + c.pointerClick + ' sort keyup ' )
|
|
.replace( ts.regex.spaces, ' ' )
|
|
.split( ' ' )
|
|
.join( namespace + ' ' );
|
|
// apply event handling to headers and/or additional headers (stickyheaders, scroller, etc)
|
|
$headers
|
|
// http://stackoverflow.com/questions/5312849/jquery-find-self;
|
|
.find( c.selectorSort )
|
|
.add( $headers.filter( c.selectorSort ) )
|
|
.unbind( tmp )
|
|
.bind( tmp, function( e, external ) {
|
|
var $cell, cell, temp,
|
|
$target = $( e.target ),
|
|
// wrap event type in spaces, so the match doesn't trigger on inner words
|
|
type = ' ' + e.type + ' ';
|
|
// only recognize left clicks
|
|
if ( ( ( e.which || e.button ) !== 1 && !type.match( ' ' + c.pointerClick + ' | sort | keyup ' ) ) ||
|
|
// allow pressing enter
|
|
( type === ' keyup ' && e.which !== ts.keyCodes.enter ) ||
|
|
// allow triggering a click event (e.which is undefined) & ignore physical clicks
|
|
( type.match( ' ' + c.pointerClick + ' ' ) && typeof e.which !== 'undefined' ) ) {
|
|
return;
|
|
}
|
|
// ignore mouseup if mousedown wasn't on the same target
|
|
if ( type.match( ' ' + c.pointerUp + ' ' ) && downTarget !== e.target && external !== true ) {
|
|
return;
|
|
}
|
|
// set target on mousedown
|
|
if ( type.match( ' ' + c.pointerDown + ' ' ) ) {
|
|
downTarget = e.target;
|
|
// preventDefault needed or jQuery v1.3.2 and older throws an
|
|
// "Uncaught TypeError: handler.apply is not a function" error
|
|
temp = $target.jquery.split( '.' );
|
|
if ( temp[ 0 ] === '1' && temp[ 1 ] < 4 ) { e.preventDefault(); }
|
|
return;
|
|
}
|
|
downTarget = null;
|
|
// prevent sort being triggered on form elements
|
|
if ( ts.regex.formElements.test( e.target.nodeName ) ||
|
|
// nosort class name, or elements within a nosort container
|
|
$target.hasClass( c.cssNoSort ) || $target.parents( '.' + c.cssNoSort ).length > 0 ||
|
|
// elements within a button
|
|
$target.parents( 'button' ).length > 0 ) {
|
|
return !c.cancelSelection;
|
|
}
|
|
if ( c.delayInit && ts.isEmptyObject( c.cache ) ) {
|
|
ts.buildCache( c );
|
|
}
|
|
// jQuery v1.2.6 doesn't have closest()
|
|
$cell = $.fn.closest ? $( this ).closest( 'th, td' ) :
|
|
/TH|TD/.test( this.nodeName ) ? $( this ) : $( this ).parents( 'th, td' );
|
|
// reference original table headers and find the same cell
|
|
// don't use $headers or IE8 throws an error - see #987
|
|
temp = $headers.index( $cell );
|
|
c.last.clickedIndex = ( temp < 0 ) ? $cell.attr( 'data-column' ) : temp;
|
|
// use column index if $headers is undefined
|
|
cell = c.$headers[ c.last.clickedIndex ];
|
|
if ( cell && !cell.sortDisabled ) {
|
|
ts.initSort( c, cell, e );
|
|
}
|
|
});
|
|
if ( c.cancelSelection ) {
|
|
// cancel selection
|
|
$headers
|
|
.attr( 'unselectable', 'on' )
|
|
.bind( 'selectstart', false )
|
|
.css({
|
|
'user-select' : 'none',
|
|
'MozUserSelect' : 'none' // not needed for jQuery 1.8+
|
|
});
|
|
}
|
|
},
|
|
|
|
buildHeaders : function( c ) {
|
|
var $temp, icon, timer, indx;
|
|
c.headerList = [];
|
|
c.headerContent = [];
|
|
c.sortVars = [];
|
|
if ( c.debug ) {
|
|
timer = new Date();
|
|
}
|
|
// children tr in tfoot - see issue #196 & #547
|
|
// don't pass table.config to computeColumnIndex here - widgets (math) pass it to "quickly" index tbody cells
|
|
c.columns = ts.computeColumnIndex( c.$table.children( 'thead, tfoot' ).children( 'tr' ) );
|
|
// add icon if cssIcon option exists
|
|
icon = c.cssIcon ?
|
|
'<i class="' + ( c.cssIcon === ts.css.icon ? ts.css.icon : c.cssIcon + ' ' + ts.css.icon ) + '"></i>' :
|
|
'';
|
|
// redefine c.$headers here in case of an updateAll that replaces or adds an entire header cell - see #683
|
|
c.$headers = $( $.map( c.$table.find( c.selectorHeaders ), function( elem, index ) {
|
|
var configHeaders, header, column, template, tmp,
|
|
$elem = $( elem );
|
|
// ignore cell (don't add it to c.$headers) if row has ignoreRow class
|
|
if ( $elem.parent().hasClass( c.cssIgnoreRow ) ) { return; }
|
|
// make sure to get header cell & not column indexed cell
|
|
configHeaders = ts.getColumnData( c.table, c.headers, index, true );
|
|
// save original header content
|
|
c.headerContent[ index ] = $elem.html();
|
|
// if headerTemplate is empty, don't reformat the header cell
|
|
if ( c.headerTemplate !== '' && !$elem.find( '.' + ts.css.headerIn ).length ) {
|
|
// set up header template
|
|
template = c.headerTemplate
|
|
.replace( ts.regex.templateContent, $elem.html() )
|
|
.replace( ts.regex.templateIcon, $elem.find( '.' + ts.css.icon ).length ? '' : icon );
|
|
if ( c.onRenderTemplate ) {
|
|
header = c.onRenderTemplate.apply( $elem, [ index, template ] );
|
|
// only change t if something is returned
|
|
if ( header && typeof header === 'string' ) {
|
|
template = header;
|
|
}
|
|
}
|
|
$elem.html( '<div class="' + ts.css.headerIn + '">' + template + '</div>' ); // faster than wrapInner
|
|
}
|
|
if ( c.onRenderHeader ) {
|
|
c.onRenderHeader.apply( $elem, [ index, c, c.$table ] );
|
|
}
|
|
column = parseInt( $elem.attr( 'data-column' ), 10 );
|
|
elem.column = column;
|
|
tmp = ts.getData( $elem, configHeaders, 'sortInitialOrder' ) || c.sortInitialOrder;
|
|
// this may get updated numerous times if there are multiple rows
|
|
c.sortVars[ column ] = {
|
|
count : -1, // set to -1 because clicking on the header automatically adds one
|
|
order: ts.getOrder( tmp ) ?
|
|
[ 1, 0, 2 ] : // desc, asc, unsorted
|
|
[ 0, 1, 2 ], // asc, desc, unsorted
|
|
lockedOrder : false
|
|
};
|
|
tmp = ts.getData( $elem, configHeaders, 'lockedOrder' ) || false;
|
|
if ( typeof tmp !== 'undefined' && tmp !== false ) {
|
|
c.sortVars[ column ].lockedOrder = true;
|
|
c.sortVars[ column ].order = ts.getOrder( tmp ) ? [ 1, 1, 1 ] : [ 0, 0, 0 ];
|
|
}
|
|
// add cell to headerList
|
|
c.headerList[ index ] = elem;
|
|
// add to parent in case there are multiple rows
|
|
$elem
|
|
.addClass( ts.css.header + ' ' + c.cssHeader )
|
|
.parent()
|
|
.addClass( ts.css.headerRow + ' ' + c.cssHeaderRow )
|
|
.attr( 'role', 'row' );
|
|
// allow keyboard cursor to focus on element
|
|
if ( c.tabIndex ) {
|
|
$elem.attr( 'tabindex', 0 );
|
|
}
|
|
return elem;
|
|
}) );
|
|
// cache headers per column
|
|
c.$headerIndexed = [];
|
|
for ( indx = 0; indx < c.columns; indx++ ) {
|
|
// colspan in header making a column undefined
|
|
if ( ts.isEmptyObject( c.sortVars[ indx ] ) ) {
|
|
c.sortVars[ indx ] = {};
|
|
}
|
|
$temp = c.$headers.filter( '[data-column="' + indx + '"]' );
|
|
// target sortable column cells, unless there are none, then use non-sortable cells
|
|
// .last() added in jQuery 1.4; use .filter(':last') to maintain compatibility with jQuery v1.2.6
|
|
c.$headerIndexed[ indx ] = $temp.length ?
|
|
$temp.not( '.sorter-false' ).length ?
|
|
$temp.not( '.sorter-false' ).filter( ':last' ) :
|
|
$temp.filter( ':last' ) :
|
|
$();
|
|
}
|
|
c.$table.find( c.selectorHeaders ).attr({
|
|
scope: 'col',
|
|
role : 'columnheader'
|
|
});
|
|
// enable/disable sorting
|
|
ts.updateHeader( c );
|
|
if ( c.debug ) {
|
|
console.log( 'Built headers:' + ts.benchmark( timer ) );
|
|
console.log( c.$headers );
|
|
}
|
|
},
|
|
|
|
// Use it to add a set of methods to table.config which will be available for all tables.
|
|
// This should be done before table initialization
|
|
addInstanceMethods : function( methods ) {
|
|
$.extend( ts.instanceMethods, methods );
|
|
},
|
|
|
|
/*
|
|
█████▄ ▄████▄ █████▄ ▄█████ ██████ █████▄ ▄█████
|
|
██▄▄██ ██▄▄██ ██▄▄██ ▀█▄ ██▄▄ ██▄▄██ ▀█▄
|
|
██▀▀▀ ██▀▀██ ██▀██ ▀█▄ ██▀▀ ██▀██ ▀█▄
|
|
██ ██ ██ ██ ██ █████▀ ██████ ██ ██ █████▀
|
|
*/
|
|
setupParsers : function( c, $tbodies ) {
|
|
var rows, list, span, max, colIndex, indx, header, configHeaders,
|
|
noParser, parser, extractor, time, tbody, len,
|
|
table = c.table,
|
|
tbodyIndex = 0,
|
|
debug = {};
|
|
// update table bodies in case we start with an empty table
|
|
c.$tbodies = c.$table.children( 'tbody:not(.' + c.cssInfoBlock + ')' );
|
|
tbody = typeof $tbodies === 'undefined' ? c.$tbodies : $tbodies;
|
|
len = tbody.length;
|
|
if ( len === 0 ) {
|
|
return c.debug ? console.warn( 'Warning: *Empty table!* Not building a parser cache' ) : '';
|
|
} else if ( c.debug ) {
|
|
time = new Date();
|
|
console[ console.group ? 'group' : 'log' ]( 'Detecting parsers for each column' );
|
|
}
|
|
list = {
|
|
extractors: [],
|
|
parsers: []
|
|
};
|
|
while ( tbodyIndex < len ) {
|
|
rows = tbody[ tbodyIndex ].rows;
|
|
if ( rows.length ) {
|
|
colIndex = 0;
|
|
max = c.columns;
|
|
for ( indx = 0; indx < max; indx++ ) {
|
|
header = c.$headerIndexed[ colIndex ];
|
|
if ( header && header.length ) {
|
|
// get column indexed table cell
|
|
configHeaders = ts.getColumnData( table, c.headers, colIndex );
|
|
// get column parser/extractor
|
|
extractor = ts.getParserById( ts.getData( header, configHeaders, 'extractor' ) );
|
|
parser = ts.getParserById( ts.getData( header, configHeaders, 'sorter' ) );
|
|
noParser = ts.getData( header, configHeaders, 'parser' ) === 'false';
|
|
// empty cells behaviour - keeping emptyToBottom for backwards compatibility
|
|
c.empties[colIndex] = (
|
|
ts.getData( header, configHeaders, 'empty' ) ||
|
|
c.emptyTo || ( c.emptyToBottom ? 'bottom' : 'top' ) ).toLowerCase();
|
|
// text strings behaviour in numerical sorts
|
|
c.strings[colIndex] = (
|
|
ts.getData( header, configHeaders, 'string' ) ||
|
|
c.stringTo ||
|
|
'max' ).toLowerCase();
|
|
if ( noParser ) {
|
|
parser = ts.getParserById( 'no-parser' );
|
|
}
|
|
if ( !extractor ) {
|
|
// For now, maybe detect someday
|
|
extractor = false;
|
|
}
|
|
if ( !parser ) {
|
|
parser = ts.detectParserForColumn( c, rows, -1, colIndex );
|
|
}
|
|
if ( c.debug ) {
|
|
debug[ '(' + colIndex + ') ' + header.text() ] = {
|
|
parser : parser.id,
|
|
extractor : extractor ? extractor.id : 'none',
|
|
string : c.strings[ colIndex ],
|
|
empty : c.empties[ colIndex ]
|
|
};
|
|
}
|
|
list.parsers[ colIndex ] = parser;
|
|
list.extractors[ colIndex ] = extractor;
|
|
span = header[ 0 ].colSpan - 1;
|
|
if ( span > 0 ) {
|
|
colIndex += span;
|
|
max += span;
|
|
while ( span + 1 > 0 ) {
|
|
// set colspan columns to use the same parsers & extractors
|
|
list.parsers[ colIndex - span ] = parser;
|
|
list.extractors[ colIndex - span ] = extractor;
|
|
span--;
|
|
}
|
|
}
|
|
}
|
|
colIndex++;
|
|
}
|
|
}
|
|
tbodyIndex += ( list.parsers.length ) ? len : 1;
|
|
}
|
|
if ( c.debug ) {
|
|
if ( !ts.isEmptyObject( debug ) ) {
|
|
console[ console.table ? 'table' : 'log' ]( debug );
|
|
} else {
|
|
console.warn( ' No parsers detected!' );
|
|
}
|
|
console.log( 'Completed detecting parsers' + ts.benchmark( time ) );
|
|
if ( console.groupEnd ) { console.groupEnd(); }
|
|
}
|
|
c.parsers = list.parsers;
|
|
c.extractors = list.extractors;
|
|
},
|
|
|
|
addParser : function( parser ) {
|
|
var indx,
|
|
len = ts.parsers.length,
|
|
add = true;
|
|
for ( indx = 0; indx < len; indx++ ) {
|
|
if ( ts.parsers[ indx ].id.toLowerCase() === parser.id.toLowerCase() ) {
|
|
add = false;
|
|
}
|
|
}
|
|
if ( add ) {
|
|
ts.parsers[ ts.parsers.length ] = parser;
|
|
}
|
|
},
|
|
|
|
getParserById : function( name ) {
|
|
/*jshint eqeqeq:false */
|
|
if ( name == 'false' ) { return false; }
|
|
var indx,
|
|
len = ts.parsers.length;
|
|
for ( indx = 0; indx < len; indx++ ) {
|
|
if ( ts.parsers[ indx ].id.toLowerCase() === ( name.toString() ).toLowerCase() ) {
|
|
return ts.parsers[ indx ];
|
|
}
|
|
}
|
|
return false;
|
|
},
|
|
|
|
detectParserForColumn : function( c, rows, rowIndex, cellIndex ) {
|
|
var cur, $node, row,
|
|
indx = ts.parsers.length,
|
|
node = false,
|
|
nodeValue = '',
|
|
keepLooking = true;
|
|
while ( nodeValue === '' && keepLooking ) {
|
|
rowIndex++;
|
|
row = rows[ rowIndex ];
|
|
// stop looking after 50 empty rows
|
|
if ( row && rowIndex < 50 ) {
|
|
if ( row.className.indexOf( ts.cssIgnoreRow ) < 0 ) {
|
|
node = rows[ rowIndex ].cells[ cellIndex ];
|
|
nodeValue = ts.getElementText( c, node, cellIndex );
|
|
$node = $( node );
|
|
if ( c.debug ) {
|
|
console.log( 'Checking if value was empty on row ' + rowIndex + ', column: ' +
|
|
cellIndex + ': "' + nodeValue + '"' );
|
|
}
|
|
}
|
|
} else {
|
|
keepLooking = false;
|
|
}
|
|
}
|
|
while ( --indx >= 0 ) {
|
|
cur = ts.parsers[ indx ];
|
|
// ignore the default text parser because it will always be true
|
|
if ( cur && cur.id !== 'text' && cur.is && cur.is( nodeValue, c.table, node, $node ) ) {
|
|
return cur;
|
|
}
|
|
}
|
|
// nothing found, return the generic parser (text)
|
|
return ts.getParserById( 'text' );
|
|
},
|
|
|
|
getElementText : function( c, node, cellIndex ) {
|
|
if ( !node ) { return ''; }
|
|
var tmp,
|
|
extract = c.textExtraction || '',
|
|
// node could be a jquery object
|
|
// http://jsperf.com/jquery-vs-instanceof-jquery/2
|
|
$node = node.jquery ? node : $( node );
|
|
if ( typeof extract === 'string' ) {
|
|
// check data-attribute first when set to 'basic'; don't use node.innerText - it's really slow!
|
|
// http://www.kellegous.com/j/2013/02/27/innertext-vs-textcontent/
|
|
if ( extract === 'basic' && typeof ( tmp = $node.attr( c.textAttribute ) ) !== 'undefined' ) {
|
|
return $.trim( tmp );
|
|
}
|
|
return $.trim( node.textContent || $node.text() );
|
|
} else {
|
|
if ( typeof extract === 'function' ) {
|
|
return $.trim( extract( $node[ 0 ], c.table, cellIndex ) );
|
|
} else if ( typeof ( tmp = ts.getColumnData( c.table, extract, cellIndex ) ) === 'function' ) {
|
|
return $.trim( tmp( $node[ 0 ], c.table, cellIndex ) );
|
|
}
|
|
}
|
|
// fallback
|
|
return $.trim( $node[ 0 ].textContent || $node.text() );
|
|
},
|
|
|
|
// centralized function to extract/parse cell contents
|
|
getParsedText : function( c, cell, colIndex, txt ) {
|
|
if ( typeof txt === 'undefined' ) {
|
|
txt = ts.getElementText( c, cell, colIndex );
|
|
}
|
|
// if no parser, make sure to return the txt
|
|
var val = '' + txt,
|
|
parser = c.parsers[ colIndex ],
|
|
extractor = c.extractors[ colIndex ];
|
|
if ( parser ) {
|
|
// do extract before parsing, if there is one
|
|
if ( extractor && typeof extractor.format === 'function' ) {
|
|
txt = extractor.format( txt, c.table, cell, colIndex );
|
|
}
|
|
// allow parsing if the string is empty, previously parsing would change it to zero,
|
|
// in case the parser needs to extract data from the table cell attributes
|
|
val = parser.id === 'no-parser' ? '' :
|
|
// make sure txt is a string (extractor may have converted it)
|
|
parser.format( '' + txt, c.table, cell, colIndex );
|
|
if ( c.ignoreCase && typeof val === 'string' ) {
|
|
val = val.toLowerCase();
|
|
}
|
|
}
|
|
return val;
|
|
},
|
|
|
|
/*
|
|
▄████▄ ▄████▄ ▄████▄ ██ ██ ██████
|
|
██ ▀▀ ██▄▄██ ██ ▀▀ ██▄▄██ ██▄▄
|
|
██ ▄▄ ██▀▀██ ██ ▄▄ ██▀▀██ ██▀▀
|
|
▀████▀ ██ ██ ▀████▀ ██ ██ ██████
|
|
*/
|
|
buildCache : function( c, callback, $tbodies ) {
|
|
var cache, val, txt, rowIndex, colIndex, tbodyIndex, $tbody, $row,
|
|
cols, $cells, cell, cacheTime, totalRows, rowData, prevRowData,
|
|
colMax, span, cacheIndex, hasParser, max, len, index,
|
|
table = c.table,
|
|
parsers = c.parsers;
|
|
// update tbody variable
|
|
c.$tbodies = c.$table.children( 'tbody:not(.' + c.cssInfoBlock + ')' );
|
|
$tbody = typeof $tbodies === 'undefined' ? c.$tbodies : $tbodies,
|
|
c.cache = {};
|
|
c.totalRows = 0;
|
|
// if no parsers found, return - it's an empty table.
|
|
if ( !parsers ) {
|
|
return c.debug ? console.warn( 'Warning: *Empty table!* Not building a cache' ) : '';
|
|
}
|
|
if ( c.debug ) {
|
|
cacheTime = new Date();
|
|
}
|
|
// processing icon
|
|
if ( c.showProcessing ) {
|
|
ts.isProcessing( table, true );
|
|
}
|
|
for ( tbodyIndex = 0; tbodyIndex < $tbody.length; tbodyIndex++ ) {
|
|
colMax = []; // column max value per tbody
|
|
cache = c.cache[ tbodyIndex ] = {
|
|
normalized: [] // array of normalized row data; last entry contains 'rowData' above
|
|
// colMax: # // added at the end
|
|
};
|
|
|
|
totalRows = ( $tbody[ tbodyIndex ] && $tbody[ tbodyIndex ].rows.length ) || 0;
|
|
for ( rowIndex = 0; rowIndex < totalRows; ++rowIndex ) {
|
|
rowData = {
|
|
// order: original row order #
|
|
// $row : jQuery Object[]
|
|
child: [], // child row text (filter widget)
|
|
raw: [] // original row text
|
|
};
|
|
/** Add the table data to main data array */
|
|
$row = $( $tbody[ tbodyIndex ].rows[ rowIndex ] );
|
|
cols = [];
|
|
// if this is a child row, add it to the last row's children and continue to the next row
|
|
// ignore child row class, if it is the first row
|
|
if ( $row.hasClass( c.cssChildRow ) && rowIndex !== 0 ) {
|
|
len = cache.normalized.length - 1;
|
|
prevRowData = cache.normalized[ len ][ c.columns ];
|
|
prevRowData.$row = prevRowData.$row.add( $row );
|
|
// add 'hasChild' class name to parent row
|
|
if ( !$row.prev().hasClass( c.cssChildRow ) ) {
|
|
$row.prev().addClass( ts.css.cssHasChild );
|
|
}
|
|
// save child row content (un-parsed!)
|
|
$cells = $row.children( 'th, td' );
|
|
len = prevRowData.child.length;
|
|
prevRowData.child[ len ] = [];
|
|
// child row content does not account for colspans/rowspans; so indexing may be off
|
|
cacheIndex = 0;
|
|
max = c.columns;
|
|
for ( colIndex = 0; colIndex < max; colIndex++ ) {
|
|
cell = $cells[ colIndex ];
|
|
if ( cell ) {
|
|
prevRowData.child[ len ][ colIndex ] = ts.getParsedText( c, cell, colIndex );
|
|
span = $cells[ colIndex ].colSpan - 1;
|
|
if ( span > 0 ) {
|
|
cacheIndex += span;
|
|
max += span;
|
|
}
|
|
}
|
|
cacheIndex++;
|
|
}
|
|
// go to the next for loop
|
|
continue;
|
|
}
|
|
rowData.$row = $row;
|
|
rowData.order = rowIndex; // add original row position to rowCache
|
|
cacheIndex = 0;
|
|
max = c.columns;
|
|
for ( colIndex = 0; colIndex < max; ++colIndex ) {
|
|
cell = $row[ 0 ].cells[ colIndex ];
|
|
if ( cell && cacheIndex < c.columns ) {
|
|
hasParser = typeof parsers[ cacheIndex ] !== 'undefined';
|
|
if ( !hasParser && c.debug ) {
|
|
console.warn( 'No parser found for row: ' + rowIndex + ', column: ' + colIndex +
|
|
'; cell containing: "' + $(cell).text() + '"; does it have a header?' );
|
|
}
|
|
val = ts.getElementText( c, cell, cacheIndex );
|
|
rowData.raw[ cacheIndex ] = val; // save original row text
|
|
// save raw column text even if there is no parser set
|
|
txt = ts.getParsedText( c, cell, cacheIndex, val );
|
|
cols[ cacheIndex ] = txt;
|
|
if ( hasParser && ( parsers[ cacheIndex ].type || '' ).toLowerCase() === 'numeric' ) {
|
|
// determine column max value (ignore sign)
|
|
colMax[ cacheIndex ] = Math.max( Math.abs( txt ) || 0, colMax[ cacheIndex ] || 0 );
|
|
}
|
|
// allow colSpan in tbody
|
|
span = cell.colSpan - 1;
|
|
if ( span > 0 ) {
|
|
index = 0;
|
|
while ( index <= span ) {
|
|
// duplicate text (or not) to spanned columns
|
|
// instead of setting duplicate span to empty string, use textExtraction to try to get a value
|
|
// see http://stackoverflow.com/q/36449711/145346
|
|
txt = c.duplicateSpan || index === 0 ?
|
|
val :
|
|
typeof c.textExtraction !== 'string' ?
|
|
ts.getElementText( c, cell, cacheIndex + index ) || '' :
|
|
'';
|
|
rowData.raw[ cacheIndex + index ] = txt;
|
|
cols[ cacheIndex + index ] = txt;
|
|
index++;
|
|
}
|
|
cacheIndex += span;
|
|
max += span;
|
|
}
|
|
}
|
|
cacheIndex++;
|
|
}
|
|
// ensure rowData is always in the same location (after the last column)
|
|
cols[ c.columns ] = rowData;
|
|
cache.normalized[ cache.normalized.length ] = cols;
|
|
}
|
|
cache.colMax = colMax;
|
|
// total up rows, not including child rows
|
|
c.totalRows += cache.normalized.length;
|
|
|
|
}
|
|
if ( c.showProcessing ) {
|
|
ts.isProcessing( table ); // remove processing icon
|
|
}
|
|
if ( c.debug ) {
|
|
len = Math.min( 5, c.cache[ 0 ].normalized.length );
|
|
console[ console.group ? 'group' : 'log' ]( 'Building cache for ' + c.totalRows +
|
|
' rows (showing ' + len + ' rows in log)' + ts.benchmark( cacheTime ) );
|
|
val = {};
|
|
for ( colIndex = 0; colIndex < c.columns; colIndex++ ) {
|
|
for ( cacheIndex = 0; cacheIndex < len; cacheIndex++ ) {
|
|
if ( !val[ 'row: ' + cacheIndex ] ) {
|
|
val[ 'row: ' + cacheIndex ] = {};
|
|
}
|
|
val[ 'row: ' + cacheIndex ][ c.$headerIndexed[ colIndex ].text() ] =
|
|
c.cache[ 0 ].normalized[ cacheIndex ][ colIndex ];
|
|
}
|
|
}
|
|
console[ console.table ? 'table' : 'log' ]( val );
|
|
if ( console.groupEnd ) { console.groupEnd(); }
|
|
}
|
|
if ( $.isFunction( callback ) ) {
|
|
callback( table );
|
|
}
|
|
},
|
|
|
|
getColumnText : function( table, column, callback, rowFilter ) {
|
|
table = $( table )[0];
|
|
var tbodyIndex, rowIndex, cache, row, tbodyLen, rowLen, raw, parsed, $cell, result,
|
|
hasCallback = typeof callback === 'function',
|
|
allColumns = column === 'all',
|
|
data = { raw : [], parsed: [], $cell: [] },
|
|
c = table.config;
|
|
if ( ts.isEmptyObject( c ) ) {
|
|
if ( c.debug ) {
|
|
console.warn( 'No cache found - aborting getColumnText function!' );
|
|
}
|
|
} else {
|
|
tbodyLen = c.$tbodies.length;
|
|
for ( tbodyIndex = 0; tbodyIndex < tbodyLen; tbodyIndex++ ) {
|
|
cache = c.cache[ tbodyIndex ].normalized;
|
|
rowLen = cache.length;
|
|
for ( rowIndex = 0; rowIndex < rowLen; rowIndex++ ) {
|
|
row = cache[ rowIndex ];
|
|
if ( rowFilter && !row[ c.columns ].$row.is( rowFilter ) ) {
|
|
continue;
|
|
}
|
|
result = true;
|
|
parsed = ( allColumns ) ? row.slice( 0, c.columns ) : row[ column ];
|
|
row = row[ c.columns ];
|
|
raw = ( allColumns ) ? row.raw : row.raw[ column ];
|
|
$cell = ( allColumns ) ? row.$row.children() : row.$row.children().eq( column );
|
|
if ( hasCallback ) {
|
|
result = callback({
|
|
tbodyIndex : tbodyIndex,
|
|
rowIndex : rowIndex,
|
|
parsed : parsed,
|
|
raw : raw,
|
|
$row : row.$row,
|
|
$cell : $cell
|
|
});
|
|
}
|
|
if ( result !== false ) {
|
|
data.parsed[ data.parsed.length ] = parsed;
|
|
data.raw[ data.raw.length ] = raw;
|
|
data.$cell[ data.$cell.length ] = $cell;
|
|
}
|
|
}
|
|
}
|
|
// return everything
|
|
return data;
|
|
}
|
|
},
|
|
|
|
/*
|
|
██ ██ █████▄ █████▄ ▄████▄ ██████ ██████
|
|
██ ██ ██▄▄██ ██ ██ ██▄▄██ ██ ██▄▄
|
|
██ ██ ██▀▀▀ ██ ██ ██▀▀██ ██ ██▀▀
|
|
▀████▀ ██ █████▀ ██ ██ ██ ██████
|
|
*/
|
|
setHeadersCss : function( c ) {
|
|
var $sorted, indx, column,
|
|
list = c.sortList,
|
|
len = list.length,
|
|
none = ts.css.sortNone + ' ' + c.cssNone,
|
|
css = [ ts.css.sortAsc + ' ' + c.cssAsc, ts.css.sortDesc + ' ' + c.cssDesc ],
|
|
cssIcon = [ c.cssIconAsc, c.cssIconDesc, c.cssIconNone ],
|
|
aria = [ 'ascending', 'descending' ],
|
|
// find the footer
|
|
$headers = c.$table
|
|
.find( 'tfoot tr' )
|
|
.children( 'td, th' )
|
|
.add( $( c.namespace + '_extra_headers' ) )
|
|
.removeClass( css.join( ' ' ) );
|
|
// remove all header information
|
|
c.$headers
|
|
.removeClass( css.join( ' ' ) )
|
|
.addClass( none )
|
|
.attr( 'aria-sort', 'none' )
|
|
.find( '.' + ts.css.icon )
|
|
.removeClass( cssIcon.join( ' ' ) )
|
|
.addClass( cssIcon[ 2 ] );
|
|
for ( indx = 0; indx < len; indx++ ) {
|
|
// direction = 2 means reset!
|
|
if ( list[ indx ][ 1 ] !== 2 ) {
|
|
// multicolumn sorting updating - see #1005
|
|
// .not(function(){}) needs jQuery 1.4
|
|
// filter(function(i, el){}) <- el is undefined in jQuery v1.2.6
|
|
$sorted = c.$headers.filter( function( i ) {
|
|
// only include headers that are in the sortList (this includes colspans)
|
|
var include = true,
|
|
$el = c.$headers.eq( i ),
|
|
col = parseInt( $el.attr( 'data-column' ), 10 ),
|
|
end = col + c.$headers[ i ].colSpan;
|
|
for ( ; col < end; col++ ) {
|
|
include = include ? include || ts.isValueInArray( col, c.sortList ) > -1 : false;
|
|
}
|
|
return include;
|
|
});
|
|
|
|
// choose the :last in case there are nested columns
|
|
$sorted = $sorted
|
|
.not( '.sorter-false' )
|
|
.filter( '[data-column="' + list[ indx ][ 0 ] + '"]' + ( len === 1 ? ':last' : '' ) );
|
|
if ( $sorted.length ) {
|
|
for ( column = 0; column < $sorted.length; column++ ) {
|
|
if ( !$sorted[ column ].sortDisabled ) {
|
|
$sorted
|
|
.eq( column )
|
|
.removeClass( none )
|
|
.addClass( css[ list[ indx ][ 1 ] ] )
|
|
.attr( 'aria-sort', aria[ list[ indx ][ 1 ] ] )
|
|
.find( '.' + ts.css.icon )
|
|
.removeClass( cssIcon[ 2 ] )
|
|
.addClass( cssIcon[ list[ indx ][ 1 ] ] );
|
|
}
|
|
}
|
|
// add sorted class to footer & extra headers, if they exist
|
|
if ( $headers.length ) {
|
|
$headers
|
|
.filter( '[data-column="' + list[ indx ][ 0 ] + '"]' )
|
|
.removeClass( none )
|
|
.addClass( css[ list[ indx ][ 1 ] ] );
|
|
}
|
|
}
|
|
}
|
|
}
|
|
// add verbose aria labels
|
|
len = c.$headers.length;
|
|
for ( indx = 0; indx < len; indx++ ) {
|
|
ts.setColumnAriaLabel( c, c.$headers.eq( indx ) );
|
|
}
|
|
},
|
|
|
|
// nextSort (optional), lets you disable next sort text
|
|
setColumnAriaLabel : function( c, $header, nextSort ) {
|
|
if ( $header.length ) {
|
|
var column = parseInt( $header.attr( 'data-column' ), 10 ),
|
|
tmp = $header.hasClass( ts.css.sortAsc ) ?
|
|
'sortAsc' :
|
|
$header.hasClass( ts.css.sortDesc ) ? 'sortDesc' : 'sortNone',
|
|
txt = $.trim( $header.text() ) + ': ' + ts.language[ tmp ];
|
|
if ( $header.hasClass( 'sorter-false' ) || nextSort === false ) {
|
|
txt += ts.language.sortDisabled;
|
|
} else {
|
|
nextSort = c.sortVars[ column ].order[ ( c.sortVars[ column ].count + 1 ) % ( c.sortReset ? 3 : 2 ) ];
|
|
// if nextSort
|
|
txt += ts.language[ nextSort === 0 ? 'nextAsc' : nextSort === 1 ? 'nextDesc' : 'nextNone' ];
|
|
}
|
|
$header.attr( 'aria-label', txt );
|
|
}
|
|
},
|
|
|
|
updateHeader : function( c ) {
|
|
var index, isDisabled, $header, col,
|
|
table = c.table,
|
|
len = c.$headers.length;
|
|
for ( index = 0; index < len; index++ ) {
|
|
$header = c.$headers.eq( index );
|
|
col = ts.getColumnData( table, c.headers, index, true );
|
|
// add 'sorter-false' class if 'parser-false' is set
|
|
isDisabled = ts.getData( $header, col, 'sorter' ) === 'false' || ts.getData( $header, col, 'parser' ) === 'false';
|
|
ts.setColumnSort( c, $header, isDisabled );
|
|
}
|
|
},
|
|
|
|
setColumnSort : function( c, $header, isDisabled ) {
|
|
var id = c.table.id;
|
|
$header[ 0 ].sortDisabled = isDisabled;
|
|
$header[ isDisabled ? 'addClass' : 'removeClass' ]( 'sorter-false' )
|
|
.attr( 'aria-disabled', '' + isDisabled );
|
|
// disable tab index on disabled cells
|
|
if ( c.tabIndex ) {
|
|
if ( isDisabled ) {
|
|
$header.removeAttr( 'tabindex' );
|
|
} else {
|
|
$header.attr( 'tabindex', '0' );
|
|
}
|
|
}
|
|
// aria-controls - requires table ID
|
|
if ( id ) {
|
|
if ( isDisabled ) {
|
|
$header.removeAttr( 'aria-controls' );
|
|
} else {
|
|
$header.attr( 'aria-controls', id );
|
|
}
|
|
}
|
|
},
|
|
|
|
updateHeaderSortCount : function( c, list ) {
|
|
var col, dir, group, indx, primary, temp, val, order,
|
|
sortList = list || c.sortList,
|
|
len = sortList.length;
|
|
c.sortList = [];
|
|
for ( indx = 0; indx < len; indx++ ) {
|
|
val = sortList[ indx ];
|
|
// ensure all sortList values are numeric - fixes #127
|
|
col = parseInt( val[ 0 ], 10 );
|
|
// prevents error if sorton array is wrong
|
|
if ( col < c.columns ) {
|
|
|
|
// set order if not already defined - due to colspan header without associated header cell
|
|
// adding this check prevents a javascript error
|
|
if ( !c.sortVars[ col ].order ) {
|
|
order = c.sortVars[ col ].order = ts.getOrder( c.sortInitialOrder ) ? [ 1, 0, 2 ] : [ 0, 1, 2 ];
|
|
c.sortVars[ col ].count = 0;
|
|
}
|
|
|
|
order = c.sortVars[ col ].order;
|
|
dir = ( '' + val[ 1 ] ).match( /^(1|d|s|o|n)/ );
|
|
dir = dir ? dir[ 0 ] : '';
|
|
// 0/(a)sc (default), 1/(d)esc, (s)ame, (o)pposite, (n)ext
|
|
switch ( dir ) {
|
|
case '1' : case 'd' : // descending
|
|
dir = 1;
|
|
break;
|
|
case 's' : // same direction (as primary column)
|
|
// if primary sort is set to 's', make it ascending
|
|
dir = primary || 0;
|
|
break;
|
|
case 'o' :
|
|
temp = order[ ( primary || 0 ) % ( c.sortReset ? 3 : 2 ) ];
|
|
// opposite of primary column; but resets if primary resets
|
|
dir = temp === 0 ? 1 : temp === 1 ? 0 : 2;
|
|
break;
|
|
case 'n' :
|
|
dir = order[ ( ++c.sortVars[ col ].count ) % ( c.sortReset ? 3 : 2 ) ];
|
|
break;
|
|
default : // ascending
|
|
dir = 0;
|
|
break;
|
|
}
|
|
primary = indx === 0 ? dir : primary;
|
|
group = [ col, parseInt( dir, 10 ) || 0 ];
|
|
c.sortList[ c.sortList.length ] = group;
|
|
dir = $.inArray( group[ 1 ], order ); // fixes issue #167
|
|
c.sortVars[ col ].count = dir >= 0 ? dir : group[ 1 ] % ( c.sortReset ? 3 : 2 );
|
|
}
|
|
}
|
|
},
|
|
|
|
updateAll : function( c, resort, callback ) {
|
|
var table = c.table;
|
|
table.isUpdating = true;
|
|
ts.refreshWidgets( table, true, true );
|
|
ts.buildHeaders( c );
|
|
ts.bindEvents( table, c.$headers, true );
|
|
ts.bindMethods( c );
|
|
ts.commonUpdate( c, resort, callback );
|
|
},
|
|
|
|
update : function( c, resort, callback ) {
|
|
var table = c.table;
|
|
table.isUpdating = true;
|
|
// update sorting (if enabled/disabled)
|
|
ts.updateHeader( c );
|
|
ts.commonUpdate( c, resort, callback );
|
|
},
|
|
|
|
// simple header update - see #989
|
|
updateHeaders : function( c, callback ) {
|
|
c.table.isUpdating = true;
|
|
ts.buildHeaders( c );
|
|
ts.bindEvents( c.table, c.$headers, true );
|
|
ts.resortComplete( c, callback );
|
|
},
|
|
|
|
updateCell : function( c, cell, resort, callback ) {
|
|
if ( ts.isEmptyObject( c.cache ) ) {
|
|
// empty table, do an update instead - fixes #1099
|
|
ts.updateHeader( c );
|
|
ts.commonUpdate( c, resort, callback );
|
|
return;
|
|
}
|
|
c.table.isUpdating = true;
|
|
c.$table.find( c.selectorRemove ).remove();
|
|
// get position from the dom
|
|
var tmp, indx, row, icell, cache, len,
|
|
$tbodies = c.$tbodies,
|
|
$cell = $( cell ),
|
|
// update cache - format: function( s, table, cell, cellIndex )
|
|
// no closest in jQuery v1.2.6
|
|
tbodyIndex = $tbodies
|
|
.index( $.fn.closest ? $cell.closest( 'tbody' ) : $cell.parents( 'tbody' ).filter( ':first' ) ),
|
|
tbcache = c.cache[ tbodyIndex ],
|
|
$row = $.fn.closest ? $cell.closest( 'tr' ) : $cell.parents( 'tr' ).filter( ':first' );
|
|
cell = $cell[ 0 ]; // in case cell is a jQuery object
|
|
// tbody may not exist if update is initialized while tbody is removed for processing
|
|
if ( $tbodies.length && tbodyIndex >= 0 ) {
|
|
row = $tbodies.eq( tbodyIndex ).find( 'tr' ).index( $row );
|
|
cache = tbcache.normalized[ row ];
|
|
len = $row[ 0 ].cells.length;
|
|
if ( len !== c.columns ) {
|
|
// colspan in here somewhere!
|
|
icell = 0;
|
|
tmp = false;
|
|
for ( indx = 0; indx < len; indx++ ) {
|
|
if ( !tmp && $row[ 0 ].cells[ indx ] !== cell ) {
|
|
icell += $row[ 0 ].cells[ indx ].colSpan;
|
|
} else {
|
|
tmp = true;
|
|
}
|
|
}
|
|
} else {
|
|
icell = $cell.index();
|
|
}
|
|
tmp = ts.getElementText( c, cell, icell ); // raw
|
|
cache[ c.columns ].raw[ icell ] = tmp;
|
|
tmp = ts.getParsedText( c, cell, icell, tmp );
|
|
cache[ icell ] = tmp; // parsed
|
|
cache[ c.columns ].$row = $row;
|
|
if ( ( c.parsers[ icell ].type || '' ).toLowerCase() === 'numeric' ) {
|
|
// update column max value (ignore sign)
|
|
tbcache.colMax[ icell ] = Math.max( Math.abs( tmp ) || 0, tbcache.colMax[ icell ] || 0 );
|
|
}
|
|
tmp = resort !== 'undefined' ? resort : c.resort;
|
|
if ( tmp !== false ) {
|
|
// widgets will be reapplied
|
|
ts.checkResort( c, tmp, callback );
|
|
} else {
|
|
// don't reapply widgets is resort is false, just in case it causes
|
|
// problems with element focus
|
|
ts.resortComplete( c, callback );
|
|
}
|
|
} else {
|
|
if ( c.debug ) {
|
|
console.error( 'updateCell aborted, tbody missing or not within the indicated table' );
|
|
}
|
|
c.table.isUpdating = false;
|
|
}
|
|
},
|
|
|
|
addRows : function( c, $row, resort, callback ) {
|
|
var txt, val, tbodyIndex, rowIndex, rows, cellIndex, len, order,
|
|
cacheIndex, rowData, cells, cell, span,
|
|
// allow passing a row string if only one non-info tbody exists in the table
|
|
valid = typeof $row === 'string' && c.$tbodies.length === 1 && /<tr/.test( $row || '' ),
|
|
table = c.table;
|
|
if ( valid ) {
|
|
$row = $( $row );
|
|
c.$tbodies.append( $row );
|
|
} else if ( !$row ||
|
|
// row is a jQuery object?
|
|
!( $row instanceof jQuery ) ||
|
|
// row contained in the table?
|
|
( $.fn.closest ? $row.closest( 'table' )[ 0 ] : $row.parents( 'table' )[ 0 ] ) !== c.table ) {
|
|
if ( c.debug ) {
|
|
console.error( 'addRows method requires (1) a jQuery selector reference to rows that have already ' +
|
|
'been added to the table, or (2) row HTML string to be added to a table with only one tbody' );
|
|
}
|
|
return false;
|
|
}
|
|
table.isUpdating = true;
|
|
if ( ts.isEmptyObject( c.cache ) ) {
|
|
// empty table, do an update instead - fixes #450
|
|
ts.updateHeader( c );
|
|
ts.commonUpdate( c, resort, callback );
|
|
} else {
|
|
rows = $row.filter( 'tr' ).attr( 'role', 'row' ).length;
|
|
tbodyIndex = c.$tbodies.index( $row.parents( 'tbody' ).filter( ':first' ) );
|
|
// fixes adding rows to an empty table - see issue #179
|
|
if ( !( c.parsers && c.parsers.length ) ) {
|
|
ts.setupParsers( c );
|
|
}
|
|
// add each row
|
|
for ( rowIndex = 0; rowIndex < rows; rowIndex++ ) {
|
|
cacheIndex = 0;
|
|
len = $row[ rowIndex ].cells.length;
|
|
order = c.cache[ tbodyIndex ].normalized.length;
|
|
cells = [];
|
|
rowData = {
|
|
child : [],
|
|
raw : [],
|
|
$row : $row.eq( rowIndex ),
|
|
order : order
|
|
};
|
|
// add each cell
|
|
for ( cellIndex = 0; cellIndex < len; cellIndex++ ) {
|
|
cell = $row[ rowIndex ].cells[ cellIndex ];
|
|
txt = ts.getElementText( c, cell, cacheIndex );
|
|
rowData.raw[ cacheIndex ] = txt;
|
|
val = ts.getParsedText( c, cell, cacheIndex, txt );
|
|
cells[ cacheIndex ] = val;
|
|
if ( ( c.parsers[ cacheIndex ].type || '' ).toLowerCase() === 'numeric' ) {
|
|
// update column max value (ignore sign)
|
|
c.cache[ tbodyIndex ].colMax[ cacheIndex ] =
|
|
Math.max( Math.abs( val ) || 0, c.cache[ tbodyIndex ].colMax[ cacheIndex ] || 0 );
|
|
}
|
|
span = cell.colSpan - 1;
|
|
if ( span > 0 ) {
|
|
cacheIndex += span;
|
|
}
|
|
cacheIndex++;
|
|
}
|
|
// add the row data to the end
|
|
cells[ c.columns ] = rowData;
|
|
// update cache
|
|
c.cache[ tbodyIndex ].normalized[ order ] = cells;
|
|
}
|
|
// resort using current settings
|
|
ts.checkResort( c, resort, callback );
|
|
}
|
|
},
|
|
|
|
updateCache : function( c, callback, $tbodies ) {
|
|
// rebuild parsers
|
|
if ( !( c.parsers && c.parsers.length ) ) {
|
|
ts.setupParsers( c, $tbodies );
|
|
}
|
|
// rebuild the cache map
|
|
ts.buildCache( c, callback, $tbodies );
|
|
},
|
|
|
|
// init flag (true) used by pager plugin to prevent widget application
|
|
// renamed from appendToTable
|
|
appendCache : function( c, init ) {
|
|
var parsed, totalRows, $tbody, $curTbody, rowIndex, tbodyIndex, appendTime,
|
|
table = c.table,
|
|
wo = c.widgetOptions,
|
|
$tbodies = c.$tbodies,
|
|
rows = [],
|
|
cache = c.cache;
|
|
// empty table - fixes #206/#346
|
|
if ( ts.isEmptyObject( cache ) ) {
|
|
// run pager appender in case the table was just emptied
|
|
return c.appender ? c.appender( table, rows ) :
|
|
table.isUpdating ? c.$table.triggerHandler( 'updateComplete', table ) : ''; // Fixes #532
|
|
}
|
|
if ( c.debug ) {
|
|
appendTime = new Date();
|
|
}
|
|
for ( tbodyIndex = 0; tbodyIndex < $tbodies.length; tbodyIndex++ ) {
|
|
$tbody = $tbodies.eq( tbodyIndex );
|
|
if ( $tbody.length ) {
|
|
// detach tbody for manipulation
|
|
$curTbody = ts.processTbody( table, $tbody, true );
|
|
parsed = cache[ tbodyIndex ].normalized;
|
|
totalRows = parsed.length;
|
|
for ( rowIndex = 0; rowIndex < totalRows; rowIndex++ ) {
|
|
rows[rows.length] = parsed[ rowIndex ][ c.columns ].$row;
|
|
// removeRows used by the pager plugin; don't render if using ajax - fixes #411
|
|
if ( !c.appender || ( c.pager && ( !c.pager.removeRows || !wo.pager_removeRows ) && !c.pager.ajax ) ) {
|
|
$curTbody.append( parsed[ rowIndex ][ c.columns ].$row );
|
|
}
|
|
}
|
|
// restore tbody
|
|
ts.processTbody( table, $curTbody, false );
|
|
}
|
|
}
|
|
if ( c.appender ) {
|
|
c.appender( table, rows );
|
|
}
|
|
if ( c.debug ) {
|
|
console.log( 'Rebuilt table' + ts.benchmark( appendTime ) );
|
|
}
|
|
// apply table widgets; but not before ajax completes
|
|
if ( !init && !c.appender ) {
|
|
ts.applyWidget( table );
|
|
}
|
|
if ( table.isUpdating ) {
|
|
c.$table.triggerHandler( 'updateComplete', table );
|
|
}
|
|
},
|
|
|
|
commonUpdate : function( c, resort, callback ) {
|
|
// remove rows/elements before update
|
|
c.$table.find( c.selectorRemove ).remove();
|
|
// rebuild parsers
|
|
ts.setupParsers( c );
|
|
// rebuild the cache map
|
|
ts.buildCache( c );
|
|
ts.checkResort( c, resort, callback );
|
|
},
|
|
|
|
/*
|
|
▄█████ ▄████▄ █████▄ ██████ ██ █████▄ ▄████▄
|
|
▀█▄ ██ ██ ██▄▄██ ██ ██ ██ ██ ██ ▄▄▄
|
|
▀█▄ ██ ██ ██▀██ ██ ██ ██ ██ ██ ▀██
|
|
█████▀ ▀████▀ ██ ██ ██ ██ ██ ██ ▀████▀
|
|
*/
|
|
initSort : function( c, cell, event ) {
|
|
if ( c.table.isUpdating ) {
|
|
// let any updates complete before initializing a sort
|
|
return setTimeout( function(){
|
|
ts.initSort( c, cell, event );
|
|
}, 50 );
|
|
}
|
|
|
|
var arry, indx, headerIndx, dir, temp, tmp, $header,
|
|
notMultiSort = !event[ c.sortMultiSortKey ],
|
|
table = c.table,
|
|
len = c.$headers.length,
|
|
// get current column index
|
|
col = parseInt( $( cell ).attr( 'data-column' ), 10 ),
|
|
order = c.sortVars[ col ].order;
|
|
|
|
// Only call sortStart if sorting is enabled
|
|
c.$table.triggerHandler( 'sortStart', table );
|
|
// get current column sort order
|
|
c.sortVars[ col ].count =
|
|
event[ c.sortResetKey ] ? 2 : ( c.sortVars[ col ].count + 1 ) % ( c.sortReset ? 3 : 2 );
|
|
// reset all sorts on non-current column - issue #30
|
|
if ( c.sortRestart ) {
|
|
for ( headerIndx = 0; headerIndx < len; headerIndx++ ) {
|
|
$header = c.$headers.eq( headerIndx );
|
|
tmp = parseInt( $header.attr( 'data-column' ), 10 );
|
|
// only reset counts on columns that weren't just clicked on and if not included in a multisort
|
|
if ( col !== tmp && ( notMultiSort || $header.hasClass( ts.css.sortNone ) ) ) {
|
|
c.sortVars[ tmp ].count = -1;
|
|
}
|
|
}
|
|
}
|
|
// user only wants to sort on one column
|
|
if ( notMultiSort ) {
|
|
// flush the sort list
|
|
c.sortList = [];
|
|
c.last.sortList = [];
|
|
if ( c.sortForce !== null ) {
|
|
arry = c.sortForce;
|
|
for ( indx = 0; indx < arry.length; indx++ ) {
|
|
if ( arry[ indx ][ 0 ] !== col ) {
|
|
c.sortList[ c.sortList.length ] = arry[ indx ];
|
|
}
|
|
}
|
|
}
|
|
// add column to sort list
|
|
dir = order[ c.sortVars[ col ].count ];
|
|
if ( dir < 2 ) {
|
|
c.sortList[ c.sortList.length ] = [ col, dir ];
|
|
// add other columns if header spans across multiple
|
|
if ( cell.colSpan > 1 ) {
|
|
for ( indx = 1; indx < cell.colSpan; indx++ ) {
|
|
c.sortList[ c.sortList.length ] = [ col + indx, dir ];
|
|
// update count on columns in colSpan
|
|
c.sortVars[ col + indx ].count = $.inArray( dir, order );
|
|
}
|
|
}
|
|
}
|
|
// multi column sorting
|
|
} else {
|
|
// get rid of the sortAppend before adding more - fixes issue #115 & #523
|
|
c.sortList = $.extend( [], c.last.sortList );
|
|
|
|
// the user has clicked on an already sorted column
|
|
if ( ts.isValueInArray( col, c.sortList ) >= 0 ) {
|
|
// reverse the sorting direction
|
|
for ( indx = 0; indx < c.sortList.length; indx++ ) {
|
|
tmp = c.sortList[ indx ];
|
|
if ( tmp[ 0 ] === col ) {
|
|
// order.count seems to be incorrect when compared to cell.count
|
|
tmp[ 1 ] = order[ c.sortVars[ col ].count ];
|
|
if ( tmp[1] === 2 ) {
|
|
c.sortList.splice( indx, 1 );
|
|
c.sortVars[ col ].count = -1;
|
|
}
|
|
}
|
|
}
|
|
} else {
|
|
// add column to sort list array
|
|
dir = order[ c.sortVars[ col ].count ];
|
|
if ( dir < 2 ) {
|
|
c.sortList[ c.sortList.length ] = [ col, dir ];
|
|
// add other columns if header spans across multiple
|
|
if ( cell.colSpan > 1 ) {
|
|
for ( indx = 1; indx < cell.colSpan; indx++ ) {
|
|
c.sortList[ c.sortList.length ] = [ col + indx, dir ];
|
|
// update count on columns in colSpan
|
|
c.sortVars[ col + indx ].count = $.inArray( dir, order );
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
// save sort before applying sortAppend
|
|
c.last.sortList = $.extend( [], c.sortList );
|
|
if ( c.sortList.length && c.sortAppend ) {
|
|
arry = $.isArray( c.sortAppend ) ? c.sortAppend : c.sortAppend[ c.sortList[ 0 ][ 0 ] ];
|
|
if ( !ts.isEmptyObject( arry ) ) {
|
|
for ( indx = 0; indx < arry.length; indx++ ) {
|
|
if ( arry[ indx ][ 0 ] !== col && ts.isValueInArray( arry[ indx ][ 0 ], c.sortList ) < 0 ) {
|
|
dir = arry[ indx ][ 1 ];
|
|
temp = ( '' + dir ).match( /^(a|d|s|o|n)/ );
|
|
if ( temp ) {
|
|
tmp = c.sortList[ 0 ][ 1 ];
|
|
switch ( temp[ 0 ] ) {
|
|
case 'd' :
|
|
dir = 1;
|
|
break;
|
|
case 's' :
|
|
dir = tmp;
|
|
break;
|
|
case 'o' :
|
|
dir = tmp === 0 ? 1 : 0;
|
|
break;
|
|
case 'n' :
|
|
dir = ( tmp + 1 ) % ( c.sortReset ? 3 : 2 );
|
|
break;
|
|
default:
|
|
dir = 0;
|
|
break;
|
|
}
|
|
}
|
|
c.sortList[ c.sortList.length ] = [ arry[ indx ][ 0 ], dir ];
|
|
}
|
|
}
|
|
}
|
|
}
|
|
// sortBegin event triggered immediately before the sort
|
|
c.$table.triggerHandler( 'sortBegin', table );
|
|
// setTimeout needed so the processing icon shows up
|
|
setTimeout( function() {
|
|
// set css for headers
|
|
ts.setHeadersCss( c );
|
|
ts.multisort( c );
|
|
ts.appendCache( c );
|
|
c.$table.triggerHandler( 'sortBeforeEnd', table );
|
|
c.$table.triggerHandler( 'sortEnd', table );
|
|
}, 1 );
|
|
},
|
|
|
|
// sort multiple columns
|
|
multisort : function( c ) { /*jshint loopfunc:true */
|
|
var tbodyIndex, sortTime, colMax, rows,
|
|
table = c.table,
|
|
dir = 0,
|
|
textSorter = c.textSorter || '',
|
|
sortList = c.sortList,
|
|
sortLen = sortList.length,
|
|
len = c.$tbodies.length;
|
|
if ( c.serverSideSorting || ts.isEmptyObject( c.cache ) ) {
|
|
// empty table - fixes #206/#346
|
|
return;
|
|
}
|
|
if ( c.debug ) { sortTime = new Date(); }
|
|
for ( tbodyIndex = 0; tbodyIndex < len; tbodyIndex++ ) {
|
|
colMax = c.cache[ tbodyIndex ].colMax;
|
|
rows = c.cache[ tbodyIndex ].normalized;
|
|
|
|
rows.sort( function( a, b ) {
|
|
var sortIndex, num, col, order, sort, x, y;
|
|
// rows is undefined here in IE, so don't use it!
|
|
for ( sortIndex = 0; sortIndex < sortLen; sortIndex++ ) {
|
|
col = sortList[ sortIndex ][ 0 ];
|
|
order = sortList[ sortIndex ][ 1 ];
|
|
// sort direction, true = asc, false = desc
|
|
dir = order === 0;
|
|
|
|
if ( c.sortStable && a[ col ] === b[ col ] && sortLen === 1 ) {
|
|
return a[ c.columns ].order - b[ c.columns ].order;
|
|
}
|
|
|
|
// fallback to natural sort since it is more robust
|
|
num = /n/i.test( ts.getSortType( c.parsers, col ) );
|
|
if ( num && c.strings[ col ] ) {
|
|
// sort strings in numerical columns
|
|
if ( typeof ( ts.string[ c.strings[ col ] ] ) === 'boolean' ) {
|
|
num = ( dir ? 1 : -1 ) * ( ts.string[ c.strings[ col ] ] ? -1 : 1 );
|
|
} else {
|
|
num = ( c.strings[ col ] ) ? ts.string[ c.strings[ col ] ] || 0 : 0;
|
|
}
|
|
// fall back to built-in numeric sort
|
|
// var sort = $.tablesorter['sort' + s]( a[col], b[col], dir, colMax[col], table );
|
|
sort = c.numberSorter ? c.numberSorter( a[ col ], b[ col ], dir, colMax[ col ], table ) :
|
|
ts[ 'sortNumeric' + ( dir ? 'Asc' : 'Desc' ) ]( a[ col ], b[ col ], num, colMax[ col ], col, c );
|
|
} else {
|
|
// set a & b depending on sort direction
|
|
x = dir ? a : b;
|
|
y = dir ? b : a;
|
|
// text sort function
|
|
if ( typeof textSorter === 'function' ) {
|
|
// custom OVERALL text sorter
|
|
sort = textSorter( x[ col ], y[ col ], dir, col, table );
|
|
} else if ( typeof textSorter === 'object' && textSorter.hasOwnProperty( col ) ) {
|
|
// custom text sorter for a SPECIFIC COLUMN
|
|
sort = textSorter[ col ]( x[ col ], y[ col ], dir, col, table );
|
|
} else {
|
|
// fall back to natural sort
|
|
sort = ts[ 'sortNatural' + ( dir ? 'Asc' : 'Desc' ) ]( a[ col ], b[ col ], col, c );
|
|
}
|
|
}
|
|
if ( sort ) { return sort; }
|
|
}
|
|
return a[ c.columns ].order - b[ c.columns ].order;
|
|
});
|
|
}
|
|
if ( c.debug ) {
|
|
console.log( 'Applying sort ' + sortList.toString() + ts.benchmark( sortTime ) );
|
|
}
|
|
},
|
|
|
|
resortComplete : function( c, callback ) {
|
|
if ( c.table.isUpdating ) {
|
|
c.$table.triggerHandler( 'updateComplete', c.table );
|
|
}
|
|
if ( $.isFunction( callback ) ) {
|
|
callback( c.table );
|
|
}
|
|
},
|
|
|
|
checkResort : function( c, resort, callback ) {
|
|
var sortList = $.isArray( resort ) ? resort : c.sortList,
|
|
// if no resort parameter is passed, fallback to config.resort (true by default)
|
|
resrt = typeof resort === 'undefined' ? c.resort : resort;
|
|
// don't try to resort if the table is still processing
|
|
// this will catch spamming of the updateCell method
|
|
if ( resrt !== false && !c.serverSideSorting && !c.table.isProcessing ) {
|
|
if ( sortList.length ) {
|
|
ts.sortOn( c, sortList, function() {
|
|
ts.resortComplete( c, callback );
|
|
}, true );
|
|
} else {
|
|
ts.sortReset( c, function() {
|
|
ts.resortComplete( c, callback );
|
|
ts.applyWidget( c.table, false );
|
|
} );
|
|
}
|
|
} else {
|
|
ts.resortComplete( c, callback );
|
|
ts.applyWidget( c.table, false );
|
|
}
|
|
},
|
|
|
|
sortOn : function( c, list, callback, init ) {
|
|
var table = c.table;
|
|
c.$table.triggerHandler( 'sortStart', table );
|
|
// update header count index
|
|
ts.updateHeaderSortCount( c, list );
|
|
// set css for headers
|
|
ts.setHeadersCss( c );
|
|
// fixes #346
|
|
if ( c.delayInit && ts.isEmptyObject( c.cache ) ) {
|
|
ts.buildCache( c );
|
|
}
|
|
c.$table.triggerHandler( 'sortBegin', table );
|
|
// sort the table and append it to the dom
|
|
ts.multisort( c );
|
|
ts.appendCache( c, init );
|
|
c.$table.triggerHandler( 'sortBeforeEnd', table );
|
|
c.$table.triggerHandler( 'sortEnd', table );
|
|
ts.applyWidget( table );
|
|
if ( $.isFunction( callback ) ) {
|
|
callback( table );
|
|
}
|
|
},
|
|
|
|
sortReset : function( c, callback ) {
|
|
c.sortList = [];
|
|
ts.setHeadersCss( c );
|
|
ts.multisort( c );
|
|
ts.appendCache( c );
|
|
if ( $.isFunction( callback ) ) {
|
|
callback( c.table );
|
|
}
|
|
},
|
|
|
|
getSortType : function( parsers, column ) {
|
|
return ( parsers && parsers[ column ] ) ? parsers[ column ].type || '' : '';
|
|
},
|
|
|
|
getOrder : function( val ) {
|
|
// look for 'd' in 'desc' order; return true
|
|
return ( /^d/i.test( val ) || val === 1 );
|
|
},
|
|
|
|
// Natural sort - https://github.com/overset/javascript-natural-sort (date sorting removed)
|
|
// this function will only accept strings, or you'll see 'TypeError: undefined is not a function'
|
|
// I could add a = a.toString(); b = b.toString(); but it'll slow down the sort overall
|
|
sortNatural : function( a, b ) {
|
|
if ( a === b ) { return 0; }
|
|
var aNum, bNum, aFloat, bFloat, indx, max,
|
|
regex = ts.regex;
|
|
// first try and sort Hex codes
|
|
if ( regex.hex.test( b ) ) {
|
|
aNum = parseInt( a.match( regex.hex ), 16 );
|
|
bNum = parseInt( b.match( regex.hex ), 16 );
|
|
if ( aNum < bNum ) { return -1; }
|
|
if ( aNum > bNum ) { return 1; }
|
|
}
|
|
// chunk/tokenize
|
|
aNum = a.replace( regex.chunk, '\\0$1\\0' ).replace( regex.chunks, '' ).split( '\\0' );
|
|
bNum = b.replace( regex.chunk, '\\0$1\\0' ).replace( regex.chunks, '' ).split( '\\0' );
|
|
max = Math.max( aNum.length, bNum.length );
|
|
// natural sorting through split numeric strings and default strings
|
|
for ( indx = 0; indx < max; indx++ ) {
|
|
// find floats not starting with '0', string or 0 if not defined
|
|
aFloat = isNaN( aNum[ indx ] ) ? aNum[ indx ] || 0 : parseFloat( aNum[ indx ] ) || 0;
|
|
bFloat = isNaN( bNum[ indx ] ) ? bNum[ indx ] || 0 : parseFloat( bNum[ indx ] ) || 0;
|
|
// handle numeric vs string comparison - number < string - (Kyle Adams)
|
|
if ( isNaN( aFloat ) !== isNaN( bFloat ) ) { return isNaN( aFloat ) ? 1 : -1; }
|
|
// rely on string comparison if different types - i.e. '02' < 2 != '02' < '2'
|
|
if ( typeof aFloat !== typeof bFloat ) {
|
|
aFloat += '';
|
|
bFloat += '';
|
|
}
|
|
if ( aFloat < bFloat ) { return -1; }
|
|
if ( aFloat > bFloat ) { return 1; }
|
|
}
|
|
return 0;
|
|
},
|
|
|
|
sortNaturalAsc : function( a, b, col, c ) {
|
|
if ( a === b ) { return 0; }
|
|
var empty = ts.string[ ( c.empties[ col ] || c.emptyTo ) ];
|
|
if ( a === '' && empty !== 0 ) { return typeof empty === 'boolean' ? ( empty ? -1 : 1 ) : -empty || -1; }
|
|
if ( b === '' && empty !== 0 ) { return typeof empty === 'boolean' ? ( empty ? 1 : -1 ) : empty || 1; }
|
|
return ts.sortNatural( a, b );
|
|
},
|
|
|
|
sortNaturalDesc : function( a, b, col, c ) {
|
|
if ( a === b ) { return 0; }
|
|
var empty = ts.string[ ( c.empties[ col ] || c.emptyTo ) ];
|
|
if ( a === '' && empty !== 0 ) { return typeof empty === 'boolean' ? ( empty ? -1 : 1 ) : empty || 1; }
|
|
if ( b === '' && empty !== 0 ) { return typeof empty === 'boolean' ? ( empty ? 1 : -1 ) : -empty || -1; }
|
|
return ts.sortNatural( b, a );
|
|
},
|
|
|
|
// basic alphabetical sort
|
|
sortText : function( a, b ) {
|
|
return a > b ? 1 : ( a < b ? -1 : 0 );
|
|
},
|
|
|
|
// return text string value by adding up ascii value
|
|
// so the text is somewhat sorted when using a digital sort
|
|
// this is NOT an alphanumeric sort
|
|
getTextValue : function( val, num, max ) {
|
|
if ( max ) {
|
|
// make sure the text value is greater than the max numerical value (max)
|
|
var indx,
|
|
len = val ? val.length : 0,
|
|
n = max + num;
|
|
for ( indx = 0; indx < len; indx++ ) {
|
|
n += val.charCodeAt( indx );
|
|
}
|
|
return num * n;
|
|
}
|
|
return 0;
|
|
},
|
|
|
|
sortNumericAsc : function( a, b, num, max, col, c ) {
|
|
if ( a === b ) { return 0; }
|
|
var empty = ts.string[ ( c.empties[ col ] || c.emptyTo ) ];
|
|
if ( a === '' && empty !== 0 ) { return typeof empty === 'boolean' ? ( empty ? -1 : 1 ) : -empty || -1; }
|
|
if ( b === '' && empty !== 0 ) { return typeof empty === 'boolean' ? ( empty ? 1 : -1 ) : empty || 1; }
|
|
if ( isNaN( a ) ) { a = ts.getTextValue( a, num, max ); }
|
|
if ( isNaN( b ) ) { b = ts.getTextValue( b, num, max ); }
|
|
return a - b;
|
|
},
|
|
|
|
sortNumericDesc : function( a, b, num, max, col, c ) {
|
|
if ( a === b ) { return 0; }
|
|
var empty = ts.string[ ( c.empties[ col ] || c.emptyTo ) ];
|
|
if ( a === '' && empty !== 0 ) { return typeof empty === 'boolean' ? ( empty ? -1 : 1 ) : empty || 1; }
|
|
if ( b === '' && empty !== 0 ) { return typeof empty === 'boolean' ? ( empty ? 1 : -1 ) : -empty || -1; }
|
|
if ( isNaN( a ) ) { a = ts.getTextValue( a, num, max ); }
|
|
if ( isNaN( b ) ) { b = ts.getTextValue( b, num, max ); }
|
|
return b - a;
|
|
},
|
|
|
|
sortNumeric : function( a, b ) {
|
|
return a - b;
|
|
},
|
|
|
|
/*
|
|
██ ██ ██ ██ █████▄ ▄████▄ ██████ ██████ ▄█████
|
|
██ ██ ██ ██ ██ ██ ██ ▄▄▄ ██▄▄ ██ ▀█▄
|
|
██ ██ ██ ██ ██ ██ ██ ▀██ ██▀▀ ██ ▀█▄
|
|
███████▀ ██ █████▀ ▀████▀ ██████ ██ █████▀
|
|
*/
|
|
addWidget : function( widget ) {
|
|
if ( widget.id && !ts.isEmptyObject( ts.getWidgetById( widget.id ) ) ) {
|
|
console.warn( '"' + widget.id + '" widget was loaded more than once!' );
|
|
}
|
|
ts.widgets[ ts.widgets.length ] = widget;
|
|
},
|
|
|
|
hasWidget : function( $table, name ) {
|
|
$table = $( $table );
|
|
return $table.length && $table[ 0 ].config && $table[ 0 ].config.widgetInit[ name ] || false;
|
|
},
|
|
|
|
getWidgetById : function( name ) {
|
|
var indx, widget,
|
|
len = ts.widgets.length;
|
|
for ( indx = 0; indx < len; indx++ ) {
|
|
widget = ts.widgets[ indx ];
|
|
if ( widget && widget.id && widget.id.toLowerCase() === name.toLowerCase() ) {
|
|
return widget;
|
|
}
|
|
}
|
|
},
|
|
|
|
applyWidgetOptions : function( table ) {
|
|
var indx, widget,
|
|
c = table.config,
|
|
len = c.widgets.length;
|
|
if ( len ) {
|
|
for ( indx = 0; indx < len; indx++ ) {
|
|
widget = ts.getWidgetById( c.widgets[ indx ] );
|
|
if ( widget && widget.options ) {
|
|
c.widgetOptions = $.extend( true, {}, widget.options, c.widgetOptions );
|
|
}
|
|
}
|
|
}
|
|
},
|
|
|
|
addWidgetFromClass : function( table ) {
|
|
var len, indx,
|
|
c = table.config,
|
|
// look for widgets to apply from table class
|
|
// don't match from 'ui-widget-content'; use \S instead of \w to include widgets
|
|
// with dashes in the name, e.g. "widget-test-2" extracts out "test-2"
|
|
regex = '^' + c.widgetClass.replace( ts.regex.templateName, '(\\S+)+' ) + '$',
|
|
widgetClass = new RegExp( regex, 'g' ),
|
|
// split up table class (widget id's can include dashes) - stop using match
|
|
// otherwise only one widget gets extracted, see #1109
|
|
widgets = ( table.className || '' ).split( ts.regex.spaces );
|
|
if ( widgets.length ) {
|
|
len = widgets.length;
|
|
for ( indx = 0; indx < len; indx++ ) {
|
|
if ( widgets[ indx ].match( widgetClass ) ) {
|
|
c.widgets[ c.widgets.length ] = widgets[ indx ].replace( widgetClass, '$1' );
|
|
}
|
|
}
|
|
}
|
|
},
|
|
|
|
applyWidgetId : function( table, id, init ) {
|
|
table = $(table)[0];
|
|
var applied, time, name,
|
|
c = table.config,
|
|
wo = c.widgetOptions,
|
|
widget = ts.getWidgetById( id );
|
|
if ( widget ) {
|
|
name = widget.id;
|
|
applied = false;
|
|
// add widget name to option list so it gets reapplied after sorting, filtering, etc
|
|
if ( $.inArray( name, c.widgets ) < 0 ) {
|
|
c.widgets[ c.widgets.length ] = name;
|
|
}
|
|
if ( c.debug ) { time = new Date(); }
|
|
|
|
if ( init || !( c.widgetInit[ name ] ) ) {
|
|
// set init flag first to prevent calling init more than once (e.g. pager)
|
|
c.widgetInit[ name ] = true;
|
|
if ( table.hasInitialized ) {
|
|
// don't reapply widget options on tablesorter init
|
|
ts.applyWidgetOptions( table );
|
|
}
|
|
if ( typeof widget.init === 'function' ) {
|
|
applied = true;
|
|
if ( c.debug ) {
|
|
console[ console.group ? 'group' : 'log' ]( 'Initializing ' + name + ' widget' );
|
|
}
|
|
widget.init( table, widget, c, wo );
|
|
}
|
|
}
|
|
if ( !init && typeof widget.format === 'function' ) {
|
|
applied = true;
|
|
if ( c.debug ) {
|
|
console[ console.group ? 'group' : 'log' ]( 'Updating ' + name + ' widget' );
|
|
}
|
|
widget.format( table, c, wo, false );
|
|
}
|
|
if ( c.debug ) {
|
|
if ( applied ) {
|
|
console.log( 'Completed ' + ( init ? 'initializing ' : 'applying ' ) + name + ' widget' + ts.benchmark( time ) );
|
|
if ( console.groupEnd ) { console.groupEnd(); }
|
|
}
|
|
}
|
|
}
|
|
},
|
|
|
|
applyWidget : function( table, init, callback ) {
|
|
table = $( table )[ 0 ]; // in case this is called externally
|
|
var indx, len, names, widget, time,
|
|
c = table.config,
|
|
widgets = [];
|
|
// prevent numerous consecutive widget applications
|
|
if ( init !== false && table.hasInitialized && ( table.isApplyingWidgets || table.isUpdating ) ) {
|
|
return;
|
|
}
|
|
if ( c.debug ) { time = new Date(); }
|
|
ts.addWidgetFromClass( table );
|
|
// prevent "tablesorter-ready" from firing multiple times in a row
|
|
clearTimeout( c.timerReady );
|
|
if ( c.widgets.length ) {
|
|
table.isApplyingWidgets = true;
|
|
// ensure unique widget ids
|
|
c.widgets = $.grep( c.widgets, function( val, index ) {
|
|
return $.inArray( val, c.widgets ) === index;
|
|
});
|
|
names = c.widgets || [];
|
|
len = names.length;
|
|
// build widget array & add priority as needed
|
|
for ( indx = 0; indx < len; indx++ ) {
|
|
widget = ts.getWidgetById( names[ indx ] );
|
|
if ( widget && widget.id ) {
|
|
// set priority to 10 if not defined
|
|
if ( !widget.priority ) { widget.priority = 10; }
|
|
widgets[ indx ] = widget;
|
|
} else if ( c.debug ) {
|
|
console.warn( '"' + names[ indx ] + '" widget code does not exist!' );
|
|
}
|
|
}
|
|
// sort widgets by priority
|
|
widgets.sort( function( a, b ) {
|
|
return a.priority < b.priority ? -1 : a.priority === b.priority ? 0 : 1;
|
|
});
|
|
// add/update selected widgets
|
|
len = widgets.length;
|
|
if ( c.debug ) {
|
|
console[ console.group ? 'group' : 'log' ]( 'Start ' + ( init ? 'initializing' : 'applying' ) + ' widgets' );
|
|
}
|
|
for ( indx = 0; indx < len; indx++ ) {
|
|
widget = widgets[ indx ];
|
|
if ( widget && widget.id ) {
|
|
ts.applyWidgetId( table, widget.id, init );
|
|
}
|
|
}
|
|
if ( c.debug && console.groupEnd ) { console.groupEnd(); }
|
|
// callback executed on init only
|
|
if ( !init && typeof callback === 'function' ) {
|
|
callback( table );
|
|
}
|
|
}
|
|
c.timerReady = setTimeout( function() {
|
|
table.isApplyingWidgets = false;
|
|
$.data( table, 'lastWidgetApplication', new Date() );
|
|
c.$table.triggerHandler( 'tablesorter-ready' );
|
|
}, 10 );
|
|
if ( c.debug ) {
|
|
widget = c.widgets.length;
|
|
console.log( 'Completed ' +
|
|
( init === true ? 'initializing ' : 'applying ' ) + widget +
|
|
' widget' + ( widget !== 1 ? 's' : '' ) + ts.benchmark( time ) );
|
|
}
|
|
},
|
|
|
|
removeWidget : function( table, name, refreshing ) {
|
|
table = $( table )[ 0 ];
|
|
var index, widget, indx, len,
|
|
c = table.config;
|
|
// if name === true, add all widgets from $.tablesorter.widgets
|
|
if ( name === true ) {
|
|
name = [];
|
|
len = ts.widgets.length;
|
|
for ( indx = 0; indx < len; indx++ ) {
|
|
widget = ts.widgets[ indx ];
|
|
if ( widget && widget.id ) {
|
|
name[ name.length ] = widget.id;
|
|
}
|
|
}
|
|
} else {
|
|
// name can be either an array of widgets names,
|
|
// or a space/comma separated list of widget names
|
|
name = ( $.isArray( name ) ? name.join( ',' ) : name || '' ).toLowerCase().split( /[\s,]+/ );
|
|
}
|
|
len = name.length;
|
|
for ( index = 0; index < len; index++ ) {
|
|
widget = ts.getWidgetById( name[ index ] );
|
|
indx = $.inArray( name[ index ], c.widgets );
|
|
// don't remove the widget from config.widget if refreshing
|
|
if ( indx >= 0 && refreshing !== true ) {
|
|
c.widgets.splice( indx, 1 );
|
|
}
|
|
if ( widget && widget.remove ) {
|
|
if ( c.debug ) {
|
|
console.log( ( refreshing ? 'Refreshing' : 'Removing' ) + ' "' + name[ index ] + '" widget' );
|
|
}
|
|
widget.remove( table, c, c.widgetOptions, refreshing );
|
|
c.widgetInit[ name[ index ] ] = false;
|
|
}
|
|
}
|
|
},
|
|
|
|
refreshWidgets : function( table, doAll, dontapply ) {
|
|
table = $( table )[ 0 ]; // see issue #243
|
|
var indx, widget,
|
|
c = table.config,
|
|
curWidgets = c.widgets,
|
|
widgets = ts.widgets,
|
|
len = widgets.length,
|
|
list = [],
|
|
callback = function( table ) {
|
|
$( table ).triggerHandler( 'refreshComplete' );
|
|
};
|
|
// remove widgets not defined in config.widgets, unless doAll is true
|
|
for ( indx = 0; indx < len; indx++ ) {
|
|
widget = widgets[ indx ];
|
|
if ( widget && widget.id && ( doAll || $.inArray( widget.id, curWidgets ) < 0 ) ) {
|
|
list[ list.length ] = widget.id;
|
|
}
|
|
}
|
|
ts.removeWidget( table, list.join( ',' ), true );
|
|
if ( dontapply !== true ) {
|
|
// call widget init if
|
|
ts.applyWidget( table, doAll || false, callback );
|
|
if ( doAll ) {
|
|
// apply widget format
|
|
ts.applyWidget( table, false, callback );
|
|
}
|
|
} else {
|
|
callback( table );
|
|
}
|
|
},
|
|
|
|
/*
|
|
██ ██ ██████ ██ ██ ██ ██████ ██ ██████ ▄█████
|
|
██ ██ ██ ██ ██ ██ ██ ██ ██▄▄ ▀█▄
|
|
██ ██ ██ ██ ██ ██ ██ ██ ██▀▀ ▀█▄
|
|
▀████▀ ██ ██ ██████ ██ ██ ██ ██████ █████▀
|
|
*/
|
|
benchmark : function( diff ) {
|
|
return ( ' ( ' + ( new Date().getTime() - diff.getTime() ) + 'ms )' );
|
|
},
|
|
// deprecated ts.log
|
|
log : function() {
|
|
console.log( arguments );
|
|
},
|
|
|
|
// $.isEmptyObject from jQuery v1.4
|
|
isEmptyObject : function( obj ) {
|
|
/*jshint forin: false */
|
|
for ( var name in obj ) {
|
|
return false;
|
|
}
|
|
return true;
|
|
},
|
|
|
|
isValueInArray : function( column, arry ) {
|
|
var indx,
|
|
len = arry && arry.length || 0;
|
|
for ( indx = 0; indx < len; indx++ ) {
|
|
if ( arry[ indx ][ 0 ] === column ) {
|
|
return indx;
|
|
}
|
|
}
|
|
return -1;
|
|
},
|
|
|
|
formatFloat : function( str, table ) {
|
|
if ( typeof str !== 'string' || str === '' ) { return str; }
|
|
// allow using formatFloat without a table; defaults to US number format
|
|
var num,
|
|
usFormat = table && table.config ? table.config.usNumberFormat !== false :
|
|
typeof table !== 'undefined' ? table : true;
|
|
if ( usFormat ) {
|
|
// US Format - 1,234,567.89 -> 1234567.89
|
|
str = str.replace( ts.regex.comma, '' );
|
|
} else {
|
|
// German Format = 1.234.567,89 -> 1234567.89
|
|
// French Format = 1 234 567,89 -> 1234567.89
|
|
str = str.replace( ts.regex.digitNonUS, '' ).replace( ts.regex.comma, '.' );
|
|
}
|
|
if ( ts.regex.digitNegativeTest.test( str ) ) {
|
|
// make (#) into a negative number -> (10) = -10
|
|
str = str.replace( ts.regex.digitNegativeReplace, '-$1' );
|
|
}
|
|
num = parseFloat( str );
|
|
// return the text instead of zero
|
|
return isNaN( num ) ? $.trim( str ) : num;
|
|
},
|
|
|
|
isDigit : function( str ) {
|
|
// replace all unwanted chars and match
|
|
return isNaN( str ) ?
|
|
ts.regex.digitTest.test( str.toString().replace( ts.regex.digitReplace, '' ) ) :
|
|
str !== '';
|
|
},
|
|
|
|
// computeTableHeaderCellIndexes from:
|
|
// http://www.javascripttoolbox.com/lib/table/examples.php
|
|
// http://www.javascripttoolbox.com/temp/table_cellindex.html
|
|
computeColumnIndex : function( $rows, c ) {
|
|
var i, j, k, l, cell, cells, rowIndex, rowSpan, colSpan, firstAvailCol,
|
|
// total columns has been calculated, use it to set the matrixrow
|
|
columns = c && c.columns || 0,
|
|
matrix = [],
|
|
matrixrow = new Array( columns );
|
|
for ( i = 0; i < $rows.length; i++ ) {
|
|
cells = $rows[ i ].cells;
|
|
for ( j = 0; j < cells.length; j++ ) {
|
|
cell = cells[ j ];
|
|
rowIndex = cell.parentNode.rowIndex;
|
|
rowSpan = cell.rowSpan || 1;
|
|
colSpan = cell.colSpan || 1;
|
|
if ( typeof matrix[ rowIndex ] === 'undefined' ) {
|
|
matrix[ rowIndex ] = [];
|
|
}
|
|
// Find first available column in the first row
|
|
for ( k = 0; k < matrix[ rowIndex ].length + 1; k++ ) {
|
|
if ( typeof matrix[ rowIndex ][ k ] === 'undefined' ) {
|
|
firstAvailCol = k;
|
|
break;
|
|
}
|
|
}
|
|
// jscs:disable disallowEmptyBlocks
|
|
if ( columns && cell.cellIndex === firstAvailCol ) {
|
|
// don't to anything
|
|
} else if ( cell.setAttribute ) {
|
|
// jscs:enable disallowEmptyBlocks
|
|
// add data-column (setAttribute = IE8+)
|
|
cell.setAttribute( 'data-column', firstAvailCol );
|
|
} else {
|
|
// remove once we drop support for IE7 - 1/12/2016
|
|
$( cell ).attr( 'data-column', firstAvailCol );
|
|
}
|
|
for ( k = rowIndex; k < rowIndex + rowSpan; k++ ) {
|
|
if ( typeof matrix[ k ] === 'undefined' ) {
|
|
matrix[ k ] = [];
|
|
}
|
|
matrixrow = matrix[ k ];
|
|
for ( l = firstAvailCol; l < firstAvailCol + colSpan; l++ ) {
|
|
matrixrow[ l ] = 'x';
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return matrixrow.length;
|
|
},
|
|
|
|
// automatically add a colgroup with col elements set to a percentage width
|
|
fixColumnWidth : function( table ) {
|
|
table = $( table )[ 0 ];
|
|
var overallWidth, percent, $tbodies, len, index,
|
|
c = table.config,
|
|
$colgroup = c.$table.children( 'colgroup' );
|
|
// remove plugin-added colgroup, in case we need to refresh the widths
|
|
if ( $colgroup.length && $colgroup.hasClass( ts.css.colgroup ) ) {
|
|
$colgroup.remove();
|
|
}
|
|
if ( c.widthFixed && c.$table.children( 'colgroup' ).length === 0 ) {
|
|
$colgroup = $( '<colgroup class="' + ts.css.colgroup + '">' );
|
|
overallWidth = c.$table.width();
|
|
// only add col for visible columns - fixes #371
|
|
$tbodies = c.$tbodies.find( 'tr:first' ).children( ':visible' );
|
|
len = $tbodies.length;
|
|
for ( index = 0; index < len; index++ ) {
|
|
percent = parseInt( ( $tbodies.eq( index ).width() / overallWidth ) * 1000, 10 ) / 10 + '%';
|
|
$colgroup.append( $( '<col>' ).css( 'width', percent ) );
|
|
}
|
|
c.$table.prepend( $colgroup );
|
|
}
|
|
},
|
|
|
|
// get sorter, string, empty, etc options for each column from
|
|
// jQuery data, metadata, header option or header class name ('sorter-false')
|
|
// priority = jQuery data > meta > headers option > header class name
|
|
getData : function( header, configHeader, key ) {
|
|
var meta, cl4ss,
|
|
val = '',
|
|
$header = $( header );
|
|
if ( !$header.length ) { return ''; }
|
|
meta = $.metadata ? $header.metadata() : false;
|
|
cl4ss = ' ' + ( $header.attr( 'class' ) || '' );
|
|
if ( typeof $header.data( key ) !== 'undefined' ||
|
|
typeof $header.data( key.toLowerCase() ) !== 'undefined' ) {
|
|
// 'data-lockedOrder' is assigned to 'lockedorder'; but 'data-locked-order' is assigned to 'lockedOrder'
|
|
// 'data-sort-initial-order' is assigned to 'sortInitialOrder'
|
|
val += $header.data( key ) || $header.data( key.toLowerCase() );
|
|
} else if ( meta && typeof meta[ key ] !== 'undefined' ) {
|
|
val += meta[ key ];
|
|
} else if ( configHeader && typeof configHeader[ key ] !== 'undefined' ) {
|
|
val += configHeader[ key ];
|
|
} else if ( cl4ss !== ' ' && cl4ss.match( ' ' + key + '-' ) ) {
|
|
// include sorter class name 'sorter-text', etc; now works with 'sorter-my-custom-parser'
|
|
val = cl4ss.match( new RegExp( '\\s' + key + '-([\\w-]+)' ) )[ 1 ] || '';
|
|
}
|
|
return $.trim( val );
|
|
},
|
|
|
|
getColumnData : function( table, obj, indx, getCell, $headers ) {
|
|
if ( typeof obj === 'undefined' || obj === null ) { return; }
|
|
table = $( table )[ 0 ];
|
|
var $header, key,
|
|
c = table.config,
|
|
$cells = ( $headers || c.$headers ),
|
|
// c.$headerIndexed is not defined initially
|
|
$cell = c.$headerIndexed && c.$headerIndexed[ indx ] ||
|
|
$cells.filter( '[data-column="' + indx + '"]:last' );
|
|
if ( obj[ indx ] ) {
|
|
return getCell ? obj[ indx ] : obj[ $cells.index( $cell ) ];
|
|
}
|
|
for ( key in obj ) {
|
|
if ( typeof key === 'string' ) {
|
|
$header = $cell
|
|
// header cell with class/id
|
|
.filter( key )
|
|
// find elements within the header cell with cell/id
|
|
.add( $cell.find( key ) );
|
|
if ( $header.length ) {
|
|
return obj[ key ];
|
|
}
|
|
}
|
|
}
|
|
return;
|
|
},
|
|
|
|
// *** Process table ***
|
|
// add processing indicator
|
|
isProcessing : function( $table, toggle, $headers ) {
|
|
$table = $( $table );
|
|
var c = $table[ 0 ].config,
|
|
// default to all headers
|
|
$header = $headers || $table.find( '.' + ts.css.header );
|
|
if ( toggle ) {
|
|
// don't use sortList if custom $headers used
|
|
if ( typeof $headers !== 'undefined' && c.sortList.length > 0 ) {
|
|
// get headers from the sortList
|
|
$header = $header.filter( function() {
|
|
// get data-column from attr to keep compatibility with jQuery 1.2.6
|
|
return this.sortDisabled ?
|
|
false :
|
|
ts.isValueInArray( parseFloat( $( this ).attr( 'data-column' ) ), c.sortList ) >= 0;
|
|
});
|
|
}
|
|
$table.add( $header ).addClass( ts.css.processing + ' ' + c.cssProcessing );
|
|
} else {
|
|
$table.add( $header ).removeClass( ts.css.processing + ' ' + c.cssProcessing );
|
|
}
|
|
},
|
|
|
|
// detach tbody but save the position
|
|
// don't use tbody because there are portions that look for a tbody index (updateCell)
|
|
processTbody : function( table, $tb, getIt ) {
|
|
table = $( table )[ 0 ];
|
|
if ( getIt ) {
|
|
table.isProcessing = true;
|
|
$tb.before( '<colgroup class="tablesorter-savemyplace"/>' );
|
|
return $.fn.detach ? $tb.detach() : $tb.remove();
|
|
}
|
|
var holdr = $( table ).find( 'colgroup.tablesorter-savemyplace' );
|
|
$tb.insertAfter( holdr );
|
|
holdr.remove();
|
|
table.isProcessing = false;
|
|
},
|
|
|
|
clearTableBody : function( table ) {
|
|
$( table )[ 0 ].config.$tbodies.children().detach();
|
|
},
|
|
|
|
// used when replacing accented characters during sorting
|
|
characterEquivalents : {
|
|
'a' : '\u00e1\u00e0\u00e2\u00e3\u00e4\u0105\u00e5', // áàâãäąå
|
|
'A' : '\u00c1\u00c0\u00c2\u00c3\u00c4\u0104\u00c5', // ÁÀÂÃÄĄÅ
|
|
'c' : '\u00e7\u0107\u010d', // çćč
|
|
'C' : '\u00c7\u0106\u010c', // ÇĆČ
|
|
'e' : '\u00e9\u00e8\u00ea\u00eb\u011b\u0119', // éèêëěę
|
|
'E' : '\u00c9\u00c8\u00ca\u00cb\u011a\u0118', // ÉÈÊËĚĘ
|
|
'i' : '\u00ed\u00ec\u0130\u00ee\u00ef\u0131', // íìİîïı
|
|
'I' : '\u00cd\u00cc\u0130\u00ce\u00cf', // ÍÌİÎÏ
|
|
'o' : '\u00f3\u00f2\u00f4\u00f5\u00f6\u014d', // óòôõöō
|
|
'O' : '\u00d3\u00d2\u00d4\u00d5\u00d6\u014c', // ÓÒÔÕÖŌ
|
|
'ss': '\u00df', // ß (s sharp)
|
|
'SS': '\u1e9e', // ẞ (Capital sharp s)
|
|
'u' : '\u00fa\u00f9\u00fb\u00fc\u016f', // úùûüů
|
|
'U' : '\u00da\u00d9\u00db\u00dc\u016e' // ÚÙÛÜŮ
|
|
},
|
|
|
|
replaceAccents : function( str ) {
|
|
var chr,
|
|
acc = '[',
|
|
eq = ts.characterEquivalents;
|
|
if ( !ts.characterRegex ) {
|
|
ts.characterRegexArray = {};
|
|
for ( chr in eq ) {
|
|
if ( typeof chr === 'string' ) {
|
|
acc += eq[ chr ];
|
|
ts.characterRegexArray[ chr ] = new RegExp( '[' + eq[ chr ] + ']', 'g' );
|
|
}
|
|
}
|
|
ts.characterRegex = new RegExp( acc + ']' );
|
|
}
|
|
if ( ts.characterRegex.test( str ) ) {
|
|
for ( chr in eq ) {
|
|
if ( typeof chr === 'string' ) {
|
|
str = str.replace( ts.characterRegexArray[ chr ], chr );
|
|
}
|
|
}
|
|
}
|
|
return str;
|
|
},
|
|
|
|
// restore headers
|
|
restoreHeaders : function( table ) {
|
|
var index, $cell,
|
|
c = $( table )[ 0 ].config,
|
|
$headers = c.$table.find( c.selectorHeaders ),
|
|
len = $headers.length;
|
|
// don't use c.$headers here in case header cells were swapped
|
|
for ( index = 0; index < len; index++ ) {
|
|
$cell = $headers.eq( index );
|
|
// only restore header cells if it is wrapped
|
|
// because this is also used by the updateAll method
|
|
if ( $cell.find( '.' + ts.css.headerIn ).length ) {
|
|
$cell.html( c.headerContent[ index ] );
|
|
}
|
|
}
|
|
},
|
|
|
|
destroy : function( table, removeClasses, callback ) {
|
|
table = $( table )[ 0 ];
|
|
if ( !table.hasInitialized ) { return; }
|
|
// remove all widgets
|
|
ts.removeWidget( table, true, false );
|
|
var events,
|
|
$t = $( table ),
|
|
c = table.config,
|
|
debug = c.debug,
|
|
$h = $t.find( 'thead:first' ),
|
|
$r = $h.find( 'tr.' + ts.css.headerRow ).removeClass( ts.css.headerRow + ' ' + c.cssHeaderRow ),
|
|
$f = $t.find( 'tfoot:first > tr' ).children( 'th, td' );
|
|
if ( removeClasses === false && $.inArray( 'uitheme', c.widgets ) >= 0 ) {
|
|
// reapply uitheme classes, in case we want to maintain appearance
|
|
$t.triggerHandler( 'applyWidgetId', [ 'uitheme' ] );
|
|
$t.triggerHandler( 'applyWidgetId', [ 'zebra' ] );
|
|
}
|
|
// remove widget added rows, just in case
|
|
$h.find( 'tr' ).not( $r ).remove();
|
|
// disable tablesorter - not using .unbind( namespace ) because namespacing was
|
|
// added in jQuery v1.4.3 - see http://api.jquery.com/event.namespace/
|
|
events = 'sortReset update updateRows updateAll updateHeaders updateCell addRows updateComplete sorton ' +
|
|
'appendCache updateCache applyWidgetId applyWidgets refreshWidgets removeWidget destroy mouseup mouseleave ' +
|
|
'keypress sortBegin sortEnd resetToLoadState '.split( ' ' )
|
|
.join( c.namespace + ' ' );
|
|
$t
|
|
.removeData( 'tablesorter' )
|
|
.unbind( events.replace( ts.regex.spaces, ' ' ) );
|
|
c.$headers
|
|
.add( $f )
|
|
.removeClass( [ ts.css.header, c.cssHeader, c.cssAsc, c.cssDesc, ts.css.sortAsc, ts.css.sortDesc, ts.css.sortNone ].join( ' ' ) )
|
|
.removeAttr( 'data-column' )
|
|
.removeAttr( 'aria-label' )
|
|
.attr( 'aria-disabled', 'true' );
|
|
$r
|
|
.find( c.selectorSort )
|
|
.unbind( ( 'mousedown mouseup keypress '.split( ' ' ).join( c.namespace + ' ' ) ).replace( ts.regex.spaces, ' ' ) );
|
|
ts.restoreHeaders( table );
|
|
$t.toggleClass( ts.css.table + ' ' + c.tableClass + ' tablesorter-' + c.theme, removeClasses === false );
|
|
// clear flag in case the plugin is initialized again
|
|
table.hasInitialized = false;
|
|
delete table.config.cache;
|
|
if ( typeof callback === 'function' ) {
|
|
callback( table );
|
|
}
|
|
if ( debug ) {
|
|
console.log( 'tablesorter has been removed' );
|
|
}
|
|
}
|
|
|
|
};
|
|
|
|
$.fn.tablesorter = function( settings ) {
|
|
return this.each( function() {
|
|
var table = this,
|
|
// merge & extend config options
|
|
c = $.extend( true, {}, ts.defaults, settings, ts.instanceMethods );
|
|
// save initial settings
|
|
c.originalSettings = settings;
|
|
// create a table from data (build table widget)
|
|
if ( !table.hasInitialized && ts.buildTable && this.nodeName !== 'TABLE' ) {
|
|
// return the table (in case the original target is the table's container)
|
|
ts.buildTable( table, c );
|
|
} else {
|
|
ts.setup( table, c );
|
|
}
|
|
});
|
|
};
|
|
|
|
// set up debug logs
|
|
if ( !( window.console && window.console.log ) ) {
|
|
// access $.tablesorter.logs for browsers that don't have a console...
|
|
ts.logs = [];
|
|
/*jshint -W020 */
|
|
console = {};
|
|
console.log = console.warn = console.error = console.table = function() {
|
|
var arg = arguments.length > 1 ? arguments : arguments[0];
|
|
ts.logs[ ts.logs.length ] = { date: Date.now(), log: arg };
|
|
};
|
|
}
|
|
|
|
// add default parsers
|
|
ts.addParser({
|
|
id : 'no-parser',
|
|
is : function() {
|
|
return false;
|
|
},
|
|
format : function() {
|
|
return '';
|
|
},
|
|
type : 'text'
|
|
});
|
|
|
|
ts.addParser({
|
|
id : 'text',
|
|
is : function() {
|
|
return true;
|
|
},
|
|
format : function( str, table ) {
|
|
var c = table.config;
|
|
if ( str ) {
|
|
str = $.trim( c.ignoreCase ? str.toLocaleLowerCase() : str );
|
|
str = c.sortLocaleCompare ? ts.replaceAccents( str ) : str;
|
|
}
|
|
return str;
|
|
},
|
|
type : 'text'
|
|
});
|
|
|
|
ts.regex.nondigit = /[^\w,. \-()]/g;
|
|
ts.addParser({
|
|
id : 'digit',
|
|
is : function( str ) {
|
|
return ts.isDigit( str );
|
|
},
|
|
format : function( str, table ) {
|
|
var num = ts.formatFloat( ( str || '' ).replace( ts.regex.nondigit, '' ), table );
|
|
return str && typeof num === 'number' ? num :
|
|
str ? $.trim( str && table.config.ignoreCase ? str.toLocaleLowerCase() : str ) : str;
|
|
},
|
|
type : 'numeric'
|
|
});
|
|
|
|
ts.regex.currencyReplace = /[+\-,. ]/g;
|
|
ts.regex.currencyTest = /^\(?\d+[\u00a3$\u20ac\u00a4\u00a5\u00a2?.]|[\u00a3$\u20ac\u00a4\u00a5\u00a2?.]\d+\)?$/;
|
|
ts.addParser({
|
|
id : 'currency',
|
|
is : function( str ) {
|
|
str = ( str || '' ).replace( ts.regex.currencyReplace, '' );
|
|
// test for £$€¤¥¢
|
|
return ts.regex.currencyTest.test( str );
|
|
},
|
|
format : function( str, table ) {
|
|
var num = ts.formatFloat( ( str || '' ).replace( ts.regex.nondigit, '' ), table );
|
|
return str && typeof num === 'number' ? num :
|
|
str ? $.trim( str && table.config.ignoreCase ? str.toLocaleLowerCase() : str ) : str;
|
|
},
|
|
type : 'numeric'
|
|
});
|
|
|
|
// too many protocols to add them all https://en.wikipedia.org/wiki/URI_scheme
|
|
// now, this regex can be updated before initialization
|
|
ts.regex.urlProtocolTest = /^(https?|ftp|file):\/\//;
|
|
ts.regex.urlProtocolReplace = /(https?|ftp|file):\/\//;
|
|
ts.addParser({
|
|
id : 'url',
|
|
is : function( str ) {
|
|
return ts.regex.urlProtocolTest.test( str );
|
|
},
|
|
format : function( str ) {
|
|
return str ? $.trim( str.replace( ts.regex.urlProtocolReplace, '' ) ) : str;
|
|
},
|
|
parsed : true, // filter widget flag
|
|
type : 'text'
|
|
});
|
|
|
|
ts.regex.dash = /-/g;
|
|
ts.regex.isoDate = /^\d{4}[\/\-]\d{1,2}[\/\-]\d{1,2}/;
|
|
ts.addParser({
|
|
id : 'isoDate',
|
|
is : function( str ) {
|
|
return ts.regex.isoDate.test( str );
|
|
},
|
|
format : function( str, table ) {
|
|
var date = str ? new Date( str.replace( ts.regex.dash, '/' ) ) : str;
|
|
return date instanceof Date && isFinite( date ) ? date.getTime() : str;
|
|
},
|
|
type : 'numeric'
|
|
});
|
|
|
|
ts.regex.percent = /%/g;
|
|
ts.regex.percentTest = /(\d\s*?%|%\s*?\d)/;
|
|
ts.addParser({
|
|
id : 'percent',
|
|
is : function( str ) {
|
|
return ts.regex.percentTest.test( str ) && str.length < 15;
|
|
},
|
|
format : function( str, table ) {
|
|
return str ? ts.formatFloat( str.replace( ts.regex.percent, '' ), table ) : str;
|
|
},
|
|
type : 'numeric'
|
|
});
|
|
|
|
// added image parser to core v2.17.9
|
|
ts.addParser({
|
|
id : 'image',
|
|
is : function( str, table, node, $node ) {
|
|
return $node.find( 'img' ).length > 0;
|
|
},
|
|
format : function( str, table, cell ) {
|
|
return $( cell ).find( 'img' ).attr( table.config.imgAttr || 'alt' ) || str;
|
|
},
|
|
parsed : true, // filter widget flag
|
|
type : 'text'
|
|
});
|
|
|
|
ts.regex.dateReplace = /(\S)([AP]M)$/i; // used by usLongDate & time parser
|
|
ts.regex.usLongDateTest1 = /^[A-Z]{3,10}\.?\s+\d{1,2},?\s+(\d{4})(\s+\d{1,2}:\d{2}(:\d{2})?(\s+[AP]M)?)?$/i;
|
|
ts.regex.usLongDateTest2 = /^\d{1,2}\s+[A-Z]{3,10}\s+\d{4}/i;
|
|
ts.addParser({
|
|
id : 'usLongDate',
|
|
is : function( str ) {
|
|
// two digit years are not allowed cross-browser
|
|
// Jan 01, 2013 12:34:56 PM or 01 Jan 2013
|
|
return ts.regex.usLongDateTest1.test( str ) || ts.regex.usLongDateTest2.test( str );
|
|
},
|
|
format : function( str, table ) {
|
|
var date = str ? new Date( str.replace( ts.regex.dateReplace, '$1 $2' ) ) : str;
|
|
return date instanceof Date && isFinite( date ) ? date.getTime() : str;
|
|
},
|
|
type : 'numeric'
|
|
});
|
|
|
|
// testing for ##-##-#### or ####-##-##, so it's not perfect; time can be included
|
|
ts.regex.shortDateTest = /(^\d{1,2}[\/\s]\d{1,2}[\/\s]\d{4})|(^\d{4}[\/\s]\d{1,2}[\/\s]\d{1,2})/;
|
|
// escaped "-" because JSHint in Firefox was showing it as an error
|
|
ts.regex.shortDateReplace = /[\-.,]/g;
|
|
// XXY covers MDY & DMY formats
|
|
ts.regex.shortDateXXY = /(\d{1,2})[\/\s](\d{1,2})[\/\s](\d{4})/;
|
|
ts.regex.shortDateYMD = /(\d{4})[\/\s](\d{1,2})[\/\s](\d{1,2})/;
|
|
ts.convertFormat = function( dateString, format ) {
|
|
dateString = ( dateString || '' )
|
|
.replace( ts.regex.spaces, ' ' )
|
|
.replace( ts.regex.shortDateReplace, '/' );
|
|
if ( format === 'mmddyyyy' ) {
|
|
dateString = dateString.replace( ts.regex.shortDateXXY, '$3/$1/$2' );
|
|
} else if ( format === 'ddmmyyyy' ) {
|
|
dateString = dateString.replace( ts.regex.shortDateXXY, '$3/$2/$1' );
|
|
} else if ( format === 'yyyymmdd' ) {
|
|
dateString = dateString.replace( ts.regex.shortDateYMD, '$1/$2/$3' );
|
|
}
|
|
var date = new Date( dateString );
|
|
return date instanceof Date && isFinite( date ) ? date.getTime() : '';
|
|
};
|
|
|
|
ts.addParser({
|
|
id : 'shortDate', // 'mmddyyyy', 'ddmmyyyy' or 'yyyymmdd'
|
|
is : function( str ) {
|
|
str = ( str || '' ).replace( ts.regex.spaces, ' ' ).replace( ts.regex.shortDateReplace, '/' );
|
|
return ts.regex.shortDateTest.test( str );
|
|
},
|
|
format : function( str, table, cell, cellIndex ) {
|
|
if ( str ) {
|
|
var c = table.config,
|
|
$header = c.$headerIndexed[ cellIndex ],
|
|
format = $header.length && $header.data( 'dateFormat' ) ||
|
|
ts.getData( $header, ts.getColumnData( table, c.headers, cellIndex ), 'dateFormat' ) ||
|
|
c.dateFormat;
|
|
// save format because getData can be slow...
|
|
if ( $header.length ) {
|
|
$header.data( 'dateFormat', format );
|
|
}
|
|
return ts.convertFormat( str, format ) || str;
|
|
}
|
|
return str;
|
|
},
|
|
type : 'numeric'
|
|
});
|
|
|
|
// match 24 hour time & 12 hours time + am/pm - see http://regexr.com/3c3tk
|
|
ts.regex.timeTest = /^([1-9]|1[0-2]):([0-5]\d)(\s[AP]M)$|^((?:[01]\d|[2][0-4]):[0-5]\d)$/i;
|
|
ts.regex.timeMatch = /([1-9]|1[0-2]):([0-5]\d)(\s[AP]M)|((?:[01]\d|[2][0-4]):[0-5]\d)/i;
|
|
ts.addParser({
|
|
id : 'time',
|
|
is : function( str ) {
|
|
return ts.regex.timeTest.test( str );
|
|
},
|
|
format : function( str, table ) {
|
|
// isolate time... ignore month, day and year
|
|
var temp,
|
|
timePart = ( str || '' ).match( ts.regex.timeMatch ),
|
|
orig = new Date( str ),
|
|
// no time component? default to 00:00 by leaving it out, but only if str is defined
|
|
time = str && ( timePart !== null ? timePart[ 0 ] : '00:00 AM' ),
|
|
date = time ? new Date( '2000/01/01 ' + time.replace( ts.regex.dateReplace, '$1 $2' ) ) : time;
|
|
if ( date instanceof Date && isFinite( date ) ) {
|
|
temp = orig instanceof Date && isFinite( orig ) ? orig.getTime() : 0;
|
|
// if original string was a valid date, add it to the decimal so the column sorts in some kind of order
|
|
// luckily new Date() ignores the decimals
|
|
return temp ? parseFloat( date.getTime() + '.' + orig.getTime() ) : date.getTime();
|
|
}
|
|
return str;
|
|
},
|
|
type : 'numeric'
|
|
});
|
|
|
|
ts.addParser({
|
|
id : 'metadata',
|
|
is : function() {
|
|
return false;
|
|
},
|
|
format : function( str, table, cell ) {
|
|
var c = table.config,
|
|
p = ( !c.parserMetadataName ) ? 'sortValue' : c.parserMetadataName;
|
|
return $( cell ).metadata()[ p ];
|
|
},
|
|
type : 'numeric'
|
|
});
|
|
|
|
/*
|
|
██████ ██████ █████▄ █████▄ ▄████▄
|
|
▄█▀ ██▄▄ ██▄▄██ ██▄▄██ ██▄▄██
|
|
▄█▀ ██▀▀ ██▀▀██ ██▀▀█ ██▀▀██
|
|
██████ ██████ █████▀ ██ ██ ██ ██
|
|
*/
|
|
// add default widgets
|
|
ts.addWidget({
|
|
id : 'zebra',
|
|
priority : 90,
|
|
format : function( table, c, wo ) {
|
|
var $visibleRows, $row, count, isEven, tbodyIndex, rowIndex, len,
|
|
child = new RegExp( c.cssChildRow, 'i' ),
|
|
$tbodies = c.$tbodies.add( $( c.namespace + '_extra_table' ).children( 'tbody:not(.' + c.cssInfoBlock + ')' ) );
|
|
for ( tbodyIndex = 0; tbodyIndex < $tbodies.length; tbodyIndex++ ) {
|
|
// loop through the visible rows
|
|
count = 0;
|
|
$visibleRows = $tbodies.eq( tbodyIndex ).children( 'tr:visible' ).not( c.selectorRemove );
|
|
len = $visibleRows.length;
|
|
for ( rowIndex = 0; rowIndex < len; rowIndex++ ) {
|
|
$row = $visibleRows.eq( rowIndex );
|
|
// style child rows the same way the parent row was styled
|
|
if ( !child.test( $row[ 0 ].className ) ) { count++; }
|
|
isEven = ( count % 2 === 0 );
|
|
$row
|
|
.removeClass( wo.zebra[ isEven ? 1 : 0 ] )
|
|
.addClass( wo.zebra[ isEven ? 0 : 1 ] );
|
|
}
|
|
}
|
|
},
|
|
remove : function( table, c, wo, refreshing ) {
|
|
if ( refreshing ) { return; }
|
|
var tbodyIndex, $tbody,
|
|
$tbodies = c.$tbodies,
|
|
toRemove = ( wo.zebra || [ 'even', 'odd' ] ).join( ' ' );
|
|
for ( tbodyIndex = 0; tbodyIndex < $tbodies.length; tbodyIndex++ ){
|
|
$tbody = ts.processTbody( table, $tbodies.eq( tbodyIndex ), true ); // remove tbody
|
|
$tbody.children().removeClass( toRemove );
|
|
ts.processTbody( table, $tbody, false ); // restore tbody
|
|
}
|
|
}
|
|
});
|
|
|
|
})( jQuery );
|
|
|
|
/*! Widget: storage - updated 3/1/2016 (v2.25.5) */
|
|
/*global JSON:false */
|
|
;(function ($, window, document) {
|
|
'use strict';
|
|
|
|
var ts = $.tablesorter || {};
|
|
// *** Store data in local storage, with a cookie fallback ***
|
|
/* IE7 needs JSON library for JSON.stringify - (http://caniuse.com/#search=json)
|
|
if you need it, then include https://github.com/douglascrockford/JSON-js
|
|
|
|
$.parseJSON is not available is jQuery versions older than 1.4.1, using older
|
|
versions will only allow storing information for one page at a time
|
|
|
|
// *** Save data (JSON format only) ***
|
|
// val must be valid JSON... use http://jsonlint.com/ to ensure it is valid
|
|
var val = { "mywidget" : "data1" }; // valid JSON uses double quotes
|
|
// $.tablesorter.storage(table, key, val);
|
|
$.tablesorter.storage(table, 'tablesorter-mywidget', val);
|
|
|
|
// *** Get data: $.tablesorter.storage(table, key); ***
|
|
v = $.tablesorter.storage(table, 'tablesorter-mywidget');
|
|
// val may be empty, so also check for your data
|
|
val = (v && v.hasOwnProperty('mywidget')) ? v.mywidget : '';
|
|
alert(val); // 'data1' if saved, or '' if not
|
|
*/
|
|
ts.storage = function(table, key, value, options) {
|
|
table = $(table)[0];
|
|
var cookieIndex, cookies, date,
|
|
hasStorage = false,
|
|
values = {},
|
|
c = table.config,
|
|
wo = c && c.widgetOptions,
|
|
storageType = ( options && options.useSessionStorage ) || ( wo && wo.storage_useSessionStorage ) ?
|
|
'sessionStorage' : 'localStorage',
|
|
$table = $(table),
|
|
// id from (1) options ID, (2) table 'data-table-group' attribute, (3) widgetOptions.storage_tableId,
|
|
// (4) table ID, then (5) table index
|
|
id = options && options.id ||
|
|
$table.attr( options && options.group || wo && wo.storage_group || 'data-table-group') ||
|
|
wo && wo.storage_tableId || table.id || $('.tablesorter').index( $table ),
|
|
// url from (1) options url, (2) table 'data-table-page' attribute, (3) widgetOptions.storage_fixedUrl,
|
|
// (4) table.config.fixedUrl (deprecated), then (5) window location path
|
|
url = options && options.url ||
|
|
$table.attr(options && options.page || wo && wo.storage_page || 'data-table-page') ||
|
|
wo && wo.storage_fixedUrl || c && c.fixedUrl || window.location.pathname;
|
|
// https://gist.github.com/paulirish/5558557
|
|
if (storageType in window) {
|
|
try {
|
|
window[storageType].setItem('_tmptest', 'temp');
|
|
hasStorage = true;
|
|
window[storageType].removeItem('_tmptest');
|
|
} catch (error) {
|
|
if (c && c.debug) {
|
|
console.warn( storageType + ' is not supported in this browser' );
|
|
}
|
|
}
|
|
}
|
|
// *** get value ***
|
|
if ($.parseJSON) {
|
|
if (hasStorage) {
|
|
values = $.parseJSON( window[storageType][key] || 'null' ) || {};
|
|
} else {
|
|
// old browser, using cookies
|
|
cookies = document.cookie.split(/[;\s|=]/);
|
|
// add one to get from the key to the value
|
|
cookieIndex = $.inArray(key, cookies) + 1;
|
|
values = (cookieIndex !== 0) ? $.parseJSON(cookies[cookieIndex] || 'null') || {} : {};
|
|
}
|
|
}
|
|
// allow value to be an empty string too
|
|
if (typeof value !== 'undefined' && window.JSON && JSON.hasOwnProperty('stringify')) {
|
|
// add unique identifiers = url pathname > table ID/index on page > data
|
|
if (!values[url]) {
|
|
values[url] = {};
|
|
}
|
|
values[url][id] = value;
|
|
// *** set value ***
|
|
if (hasStorage) {
|
|
window[storageType][key] = JSON.stringify(values);
|
|
} else {
|
|
date = new Date();
|
|
date.setTime(date.getTime() + (31536e+6)); // 365 days
|
|
document.cookie = key + '=' + (JSON.stringify(values)).replace(/\"/g, '\"') + '; expires=' + date.toGMTString() + '; path=/';
|
|
}
|
|
} else {
|
|
return values && values[url] ? values[url][id] : '';
|
|
}
|
|
};
|
|
|
|
})(jQuery, window, document);
|
|
|
|
/*! Widget: uitheme - updated 3/26/2015 (v2.21.3) */
|
|
;(function ($) {
|
|
'use strict';
|
|
var ts = $.tablesorter || {};
|
|
|
|
ts.themes = {
|
|
'bootstrap' : {
|
|
table : 'table table-bordered table-striped',
|
|
caption : 'caption',
|
|
// header class names
|
|
header : 'bootstrap-header', // give the header a gradient background (theme.bootstrap_2.css)
|
|
sortNone : '',
|
|
sortAsc : '',
|
|
sortDesc : '',
|
|
active : '', // applied when column is sorted
|
|
hover : '', // custom css required - a defined bootstrap style may not override other classes
|
|
// icon class names
|
|
icons : '', // add 'icon-white' to make them white; this icon class is added to the <i> in the header
|
|
iconSortNone : 'bootstrap-icon-unsorted', // class name added to icon when column is not sorted
|
|
iconSortAsc : 'icon-chevron-up glyphicon glyphicon-chevron-up', // class name added to icon when column has ascending sort
|
|
iconSortDesc : 'icon-chevron-down glyphicon glyphicon-chevron-down', // class name added to icon when column has descending sort
|
|
filterRow : '', // filter row class
|
|
footerRow : '',
|
|
footerCells : '',
|
|
even : '', // even row zebra striping
|
|
odd : '' // odd row zebra striping
|
|
},
|
|
'jui' : {
|
|
table : 'ui-widget ui-widget-content ui-corner-all', // table classes
|
|
caption : 'ui-widget-content',
|
|
// header class names
|
|
header : 'ui-widget-header ui-corner-all ui-state-default', // header classes
|
|
sortNone : '',
|
|
sortAsc : '',
|
|
sortDesc : '',
|
|
active : 'ui-state-active', // applied when column is sorted
|
|
hover : 'ui-state-hover', // hover class
|
|
// icon class names
|
|
icons : 'ui-icon', // icon class added to the <i> in the header
|
|
iconSortNone : 'ui-icon-carat-2-n-s', // class name added to icon when column is not sorted
|
|
iconSortAsc : 'ui-icon-carat-1-n', // class name added to icon when column has ascending sort
|
|
iconSortDesc : 'ui-icon-carat-1-s', // class name added to icon when column has descending sort
|
|
filterRow : '',
|
|
footerRow : '',
|
|
footerCells : '',
|
|
even : 'ui-widget-content', // even row zebra striping
|
|
odd : 'ui-state-default' // odd row zebra striping
|
|
}
|
|
};
|
|
|
|
$.extend(ts.css, {
|
|
wrapper : 'tablesorter-wrapper' // ui theme & resizable
|
|
});
|
|
|
|
ts.addWidget({
|
|
id: 'uitheme',
|
|
priority: 10,
|
|
format: function(table, c, wo) {
|
|
var i, hdr, icon, time, $header, $icon, $tfoot, $h, oldtheme, oldremove, oldIconRmv, hasOldTheme,
|
|
themesAll = ts.themes,
|
|
$table = c.$table.add( $( c.namespace + '_extra_table' ) ),
|
|
$headers = c.$headers.add( $( c.namespace + '_extra_headers' ) ),
|
|
theme = c.theme || 'jui',
|
|
themes = themesAll[theme] || {},
|
|
remove = $.trim( [ themes.sortNone, themes.sortDesc, themes.sortAsc, themes.active ].join( ' ' ) ),
|
|
iconRmv = $.trim( [ themes.iconSortNone, themes.iconSortDesc, themes.iconSortAsc ].join( ' ' ) );
|
|
if (c.debug) { time = new Date(); }
|
|
// initialization code - run once
|
|
if (!$table.hasClass('tablesorter-' + theme) || c.theme !== c.appliedTheme || !wo.uitheme_applied) {
|
|
wo.uitheme_applied = true;
|
|
oldtheme = themesAll[c.appliedTheme] || {};
|
|
hasOldTheme = !$.isEmptyObject(oldtheme);
|
|
oldremove = hasOldTheme ? [ oldtheme.sortNone, oldtheme.sortDesc, oldtheme.sortAsc, oldtheme.active ].join( ' ' ) : '';
|
|
oldIconRmv = hasOldTheme ? [ oldtheme.iconSortNone, oldtheme.iconSortDesc, oldtheme.iconSortAsc ].join( ' ' ) : '';
|
|
if (hasOldTheme) {
|
|
wo.zebra[0] = $.trim( ' ' + wo.zebra[0].replace(' ' + oldtheme.even, '') );
|
|
wo.zebra[1] = $.trim( ' ' + wo.zebra[1].replace(' ' + oldtheme.odd, '') );
|
|
c.$tbodies.children().removeClass( [ oldtheme.even, oldtheme.odd ].join(' ') );
|
|
}
|
|
// update zebra stripes
|
|
if (themes.even) { wo.zebra[0] += ' ' + themes.even; }
|
|
if (themes.odd) { wo.zebra[1] += ' ' + themes.odd; }
|
|
// add caption style
|
|
$table.children('caption')
|
|
.removeClass(oldtheme.caption || '')
|
|
.addClass(themes.caption);
|
|
// add table/footer class names
|
|
$tfoot = $table
|
|
// remove other selected themes
|
|
.removeClass( (c.appliedTheme ? 'tablesorter-' + (c.appliedTheme || '') : '') + ' ' + (oldtheme.table || '') )
|
|
.addClass('tablesorter-' + theme + ' ' + (themes.table || '')) // add theme widget class name
|
|
.children('tfoot');
|
|
c.appliedTheme = c.theme;
|
|
|
|
if ($tfoot.length) {
|
|
$tfoot
|
|
// if oldtheme.footerRow or oldtheme.footerCells are undefined, all class names are removed
|
|
.children('tr').removeClass(oldtheme.footerRow || '').addClass(themes.footerRow)
|
|
.children('th, td').removeClass(oldtheme.footerCells || '').addClass(themes.footerCells);
|
|
}
|
|
// update header classes
|
|
$headers
|
|
.removeClass( (hasOldTheme ? [ oldtheme.header, oldtheme.hover, oldremove ].join(' ') : '') || '' )
|
|
.addClass(themes.header)
|
|
.not('.sorter-false')
|
|
.unbind('mouseenter.tsuitheme mouseleave.tsuitheme')
|
|
.bind('mouseenter.tsuitheme mouseleave.tsuitheme', function(event) {
|
|
// toggleClass with switch added in jQuery 1.3
|
|
$(this)[ event.type === 'mouseenter' ? 'addClass' : 'removeClass' ](themes.hover || '');
|
|
});
|
|
|
|
$headers.each(function(){
|
|
var $this = $(this);
|
|
if (!$this.find('.' + ts.css.wrapper).length) {
|
|
// Firefox needs this inner div to position the icon & resizer correctly
|
|
$this.wrapInner('<div class="' + ts.css.wrapper + '" style="position:relative;height:100%;width:100%"></div>');
|
|
}
|
|
});
|
|
if (c.cssIcon) {
|
|
// if c.cssIcon is '', then no <i> is added to the header
|
|
$headers
|
|
.find('.' + ts.css.icon)
|
|
.removeClass(hasOldTheme ? [ oldtheme.icons, oldIconRmv ].join(' ') : '')
|
|
.addClass(themes.icons || '');
|
|
}
|
|
if ($table.hasClass('hasFilters')) {
|
|
$table.children('thead').children('.' + ts.css.filterRow)
|
|
.removeClass(hasOldTheme ? oldtheme.filterRow || '' : '')
|
|
.addClass(themes.filterRow || '');
|
|
}
|
|
}
|
|
for (i = 0; i < c.columns; i++) {
|
|
$header = c.$headers
|
|
.add($(c.namespace + '_extra_headers'))
|
|
.not('.sorter-false')
|
|
.filter('[data-column="' + i + '"]');
|
|
$icon = (ts.css.icon) ? $header.find('.' + ts.css.icon) : $();
|
|
$h = $headers.not('.sorter-false').filter('[data-column="' + i + '"]:last');
|
|
if ($h.length) {
|
|
$header.removeClass(remove);
|
|
$icon.removeClass(iconRmv);
|
|
if ($h[0].sortDisabled) {
|
|
// no sort arrows for disabled columns!
|
|
$icon.removeClass(themes.icons || '');
|
|
} else {
|
|
hdr = themes.sortNone;
|
|
icon = themes.iconSortNone;
|
|
if ($h.hasClass(ts.css.sortAsc)) {
|
|
hdr = [ themes.sortAsc, themes.active ].join(' ');
|
|
icon = themes.iconSortAsc;
|
|
} else if ($h.hasClass(ts.css.sortDesc)) {
|
|
hdr = [ themes.sortDesc, themes.active ].join(' ');
|
|
icon = themes.iconSortDesc;
|
|
}
|
|
$header.addClass(hdr);
|
|
$icon.addClass(icon || '');
|
|
}
|
|
}
|
|
}
|
|
if (c.debug) {
|
|
console.log('Applying ' + theme + ' theme' + ts.benchmark(time));
|
|
}
|
|
},
|
|
remove: function(table, c, wo, refreshing) {
|
|
if (!wo.uitheme_applied) { return; }
|
|
var $table = c.$table,
|
|
theme = c.appliedTheme || 'jui',
|
|
themes = ts.themes[ theme ] || ts.themes.jui,
|
|
$headers = $table.children('thead').children(),
|
|
remove = themes.sortNone + ' ' + themes.sortDesc + ' ' + themes.sortAsc,
|
|
iconRmv = themes.iconSortNone + ' ' + themes.iconSortDesc + ' ' + themes.iconSortAsc;
|
|
$table.removeClass('tablesorter-' + theme + ' ' + themes.table);
|
|
wo.uitheme_applied = false;
|
|
if (refreshing) { return; }
|
|
$table.find(ts.css.header).removeClass(themes.header);
|
|
$headers
|
|
.unbind('mouseenter.tsuitheme mouseleave.tsuitheme') // remove hover
|
|
.removeClass(themes.hover + ' ' + remove + ' ' + themes.active)
|
|
.filter('.' + ts.css.filterRow)
|
|
.removeClass(themes.filterRow);
|
|
$headers.find('.' + ts.css.icon).removeClass(themes.icons + ' ' + iconRmv);
|
|
}
|
|
});
|
|
|
|
})(jQuery);
|
|
|
|
/*! Widget: columns */
|
|
;(function ($) {
|
|
'use strict';
|
|
var ts = $.tablesorter || {};
|
|
|
|
ts.addWidget({
|
|
id: 'columns',
|
|
priority: 30,
|
|
options : {
|
|
columns : [ 'primary', 'secondary', 'tertiary' ]
|
|
},
|
|
format: function(table, c, wo) {
|
|
var $tbody, tbodyIndex, $rows, rows, $row, $cells, remove, indx,
|
|
$table = c.$table,
|
|
$tbodies = c.$tbodies,
|
|
sortList = c.sortList,
|
|
len = sortList.length,
|
|
// removed c.widgetColumns support
|
|
css = wo && wo.columns || [ 'primary', 'secondary', 'tertiary' ],
|
|
last = css.length - 1;
|
|
remove = css.join(' ');
|
|
// check if there is a sort (on initialization there may not be one)
|
|
for (tbodyIndex = 0; tbodyIndex < $tbodies.length; tbodyIndex++ ) {
|
|
$tbody = ts.processTbody(table, $tbodies.eq(tbodyIndex), true); // detach tbody
|
|
$rows = $tbody.children('tr');
|
|
// loop through the visible rows
|
|
$rows.each(function() {
|
|
$row = $(this);
|
|
if (this.style.display !== 'none') {
|
|
// remove all columns class names
|
|
$cells = $row.children().removeClass(remove);
|
|
// add appropriate column class names
|
|
if (sortList && sortList[0]) {
|
|
// primary sort column class
|
|
$cells.eq(sortList[0][0]).addClass(css[0]);
|
|
if (len > 1) {
|
|
for (indx = 1; indx < len; indx++) {
|
|
// secondary, tertiary, etc sort column classes
|
|
$cells.eq(sortList[indx][0]).addClass( css[indx] || css[last] );
|
|
}
|
|
}
|
|
}
|
|
}
|
|
});
|
|
ts.processTbody(table, $tbody, false);
|
|
}
|
|
// add classes to thead and tfoot
|
|
rows = wo.columns_thead !== false ? [ 'thead tr' ] : [];
|
|
if (wo.columns_tfoot !== false) {
|
|
rows.push('tfoot tr');
|
|
}
|
|
if (rows.length) {
|
|
$rows = $table.find( rows.join(',') ).children().removeClass(remove);
|
|
if (len) {
|
|
for (indx = 0; indx < len; indx++) {
|
|
// add primary. secondary, tertiary, etc sort column classes
|
|
$rows.filter('[data-column="' + sortList[indx][0] + '"]').addClass(css[indx] || css[last]);
|
|
}
|
|
}
|
|
}
|
|
},
|
|
remove: function(table, c, wo) {
|
|
var tbodyIndex, $tbody,
|
|
$tbodies = c.$tbodies,
|
|
remove = (wo.columns || [ 'primary', 'secondary', 'tertiary' ]).join(' ');
|
|
c.$headers.removeClass(remove);
|
|
c.$table.children('tfoot').children('tr').children('th, td').removeClass(remove);
|
|
for (tbodyIndex = 0; tbodyIndex < $tbodies.length; tbodyIndex++ ) {
|
|
$tbody = ts.processTbody(table, $tbodies.eq(tbodyIndex), true); // remove tbody
|
|
$tbody.children('tr').each(function() {
|
|
$(this).children().removeClass(remove);
|
|
});
|
|
ts.processTbody(table, $tbody, false); // restore tbody
|
|
}
|
|
}
|
|
});
|
|
|
|
})(jQuery);
|
|
|
|
/*! Widget: filter - updated 4/1/2016 (v2.25.7) *//*
|
|
* Requires tablesorter v2.8+ and jQuery 1.7+
|
|
* by Rob Garrison
|
|
*/
|
|
;( function ( $ ) {
|
|
'use strict';
|
|
var tsf, tsfRegex,
|
|
ts = $.tablesorter || {},
|
|
tscss = ts.css,
|
|
tskeyCodes = ts.keyCodes;
|
|
|
|
$.extend( tscss, {
|
|
filterRow : 'tablesorter-filter-row',
|
|
filter : 'tablesorter-filter',
|
|
filterDisabled : 'disabled',
|
|
filterRowHide : 'hideme'
|
|
});
|
|
|
|
$.extend( tskeyCodes, {
|
|
backSpace : 8,
|
|
escape : 27,
|
|
space : 32,
|
|
left : 37,
|
|
down : 40
|
|
});
|
|
|
|
ts.addWidget({
|
|
id: 'filter',
|
|
priority: 50,
|
|
options : {
|
|
filter_childRows : false, // if true, filter includes child row content in the search
|
|
filter_childByColumn : false, // ( filter_childRows must be true ) if true = search child rows by column; false = search all child row text grouped
|
|
filter_childWithSibs : true, // if true, include matching child row siblings
|
|
filter_columnFilters : true, // if true, a filter will be added to the top of each table column
|
|
filter_columnAnyMatch: true, // if true, allows using '#:{query}' in AnyMatch searches ( column:query )
|
|
filter_cellFilter : '', // css class name added to the filter cell ( string or array )
|
|
filter_cssFilter : '', // css class name added to the filter row & each input in the row ( tablesorter-filter is ALWAYS added )
|
|
filter_defaultFilter : {}, // add a default column filter type '~{query}' to make fuzzy searches default; '{q1} AND {q2}' to make all searches use a logical AND.
|
|
filter_excludeFilter : {}, // filters to exclude, per column
|
|
filter_external : '', // jQuery selector string ( or jQuery object ) of external filters
|
|
filter_filteredRow : 'filtered', // class added to filtered rows; define in css with "display:none" to hide the filtered-out rows
|
|
filter_formatter : null, // add custom filter elements to the filter row
|
|
filter_functions : null, // add custom filter functions using this option
|
|
filter_hideEmpty : true, // hide filter row when table is empty
|
|
filter_hideFilters : false, // collapse filter row when mouse leaves the area
|
|
filter_ignoreCase : true, // if true, make all searches case-insensitive
|
|
filter_liveSearch : true, // if true, search column content while the user types ( with a delay )
|
|
filter_matchType : { 'input': 'exact', 'select': 'exact' }, // global query settings ('exact' or 'match'); overridden by "filter-match" or "filter-exact" class
|
|
filter_onlyAvail : 'filter-onlyAvail', // a header with a select dropdown & this class name will only show available ( visible ) options within the drop down
|
|
filter_placeholder : { search : '', select : '' }, // default placeholder text ( overridden by any header 'data-placeholder' setting )
|
|
filter_reset : null, // jQuery selector string of an element used to reset the filters
|
|
filter_resetOnEsc : true, // Reset filter input when the user presses escape - normalized across browsers
|
|
filter_saveFilters : false, // Use the $.tablesorter.storage utility to save the most recent filters
|
|
filter_searchDelay : 300, // typing delay in milliseconds before starting a search
|
|
filter_searchFiltered: true, // allow searching through already filtered rows in special circumstances; will speed up searching in large tables if true
|
|
filter_selectSource : null, // include a function to return an array of values to be added to the column filter select
|
|
filter_startsWith : false, // if true, filter start from the beginning of the cell contents
|
|
filter_useParsedData : false, // filter all data using parsed content
|
|
filter_serversideFiltering : false, // if true, must perform server-side filtering b/c client-side filtering is disabled, but the ui and events will still be used.
|
|
filter_defaultAttrib : 'data-value', // data attribute in the header cell that contains the default filter value
|
|
filter_selectSourceSeparator : '|' // filter_selectSource array text left of the separator is added to the option value, right into the option text
|
|
},
|
|
format: function( table, c, wo ) {
|
|
if ( !c.$table.hasClass( 'hasFilters' ) ) {
|
|
tsf.init( table, c, wo );
|
|
}
|
|
},
|
|
remove: function( table, c, wo, refreshing ) {
|
|
var tbodyIndex, $tbody,
|
|
$table = c.$table,
|
|
$tbodies = c.$tbodies,
|
|
events = 'addRows updateCell update updateRows updateComplete appendCache filterReset filterEnd search '
|
|
.split( ' ' ).join( c.namespace + 'filter ' );
|
|
$table
|
|
.removeClass( 'hasFilters' )
|
|
// add filter namespace to all BUT search
|
|
.unbind( events.replace( ts.regex.spaces, ' ' ) )
|
|
// remove the filter row even if refreshing, because the column might have been moved
|
|
.find( '.' + tscss.filterRow ).remove();
|
|
if ( refreshing ) { return; }
|
|
for ( tbodyIndex = 0; tbodyIndex < $tbodies.length; tbodyIndex++ ) {
|
|
$tbody = ts.processTbody( table, $tbodies.eq( tbodyIndex ), true ); // remove tbody
|
|
$tbody.children().removeClass( wo.filter_filteredRow ).show();
|
|
ts.processTbody( table, $tbody, false ); // restore tbody
|
|
}
|
|
if ( wo.filter_reset ) {
|
|
$( document ).undelegate( wo.filter_reset, 'click' + c.namespace + 'filter' );
|
|
}
|
|
}
|
|
});
|
|
|
|
tsf = ts.filter = {
|
|
|
|
// regex used in filter 'check' functions - not for general use and not documented
|
|
regex: {
|
|
regex : /^\/((?:\\\/|[^\/])+)\/([mig]{0,3})?$/, // regex to test for regex
|
|
child : /tablesorter-childRow/, // child row class name; this gets updated in the script
|
|
filtered : /filtered/, // filtered (hidden) row class name; updated in the script
|
|
type : /undefined|number/, // check type
|
|
exact : /(^[\"\'=]+)|([\"\'=]+$)/g, // exact match (allow '==')
|
|
operators : /[<>=]/g, // replace operators
|
|
query : '(q|query)', // replace filter queries
|
|
wild01 : /\?/g, // wild card match 0 or 1
|
|
wild0More : /\*/g, // wild care match 0 or more
|
|
quote : /\"/g,
|
|
isNeg1 : /(>=?\s*-\d)/,
|
|
isNeg2 : /(<=?\s*\d)/
|
|
},
|
|
// function( c, data ) { }
|
|
// c = table.config
|
|
// data.$row = jQuery object of the row currently being processed
|
|
// data.$cells = jQuery object of all cells within the current row
|
|
// data.filters = array of filters for all columns ( some may be undefined )
|
|
// data.filter = filter for the current column
|
|
// data.iFilter = same as data.filter, except lowercase ( if wo.filter_ignoreCase is true )
|
|
// data.exact = table cell text ( or parsed data if column parser enabled; may be a number & not a string )
|
|
// data.iExact = same as data.exact, except lowercase ( if wo.filter_ignoreCase is true; may be a number & not a string )
|
|
// data.cache = table cell text from cache, so it has been parsed ( & in all lower case if c.ignoreCase is true )
|
|
// data.cacheArray = An array of parsed content from each table cell in the row being processed
|
|
// data.index = column index; table = table element ( DOM )
|
|
// data.parsed = array ( by column ) of boolean values ( from filter_useParsedData or 'filter-parsed' class )
|
|
types: {
|
|
or : function( c, data, vars ) {
|
|
// look for "|", but not if it is inside of a regular expression
|
|
if ( ( tsfRegex.orTest.test( data.iFilter ) || tsfRegex.orSplit.test( data.filter ) ) &&
|
|
// this test for regex has potential to slow down the overall search
|
|
!tsfRegex.regex.test( data.filter ) ) {
|
|
var indx, filterMatched, query, regex,
|
|
// duplicate data but split filter
|
|
data2 = $.extend( {}, data ),
|
|
filter = data.filter.split( tsfRegex.orSplit ),
|
|
iFilter = data.iFilter.split( tsfRegex.orSplit ),
|
|
len = filter.length;
|
|
for ( indx = 0; indx < len; indx++ ) {
|
|
data2.nestedFilters = true;
|
|
data2.filter = '' + ( tsf.parseFilter( c, filter[ indx ], data ) || '' );
|
|
data2.iFilter = '' + ( tsf.parseFilter( c, iFilter[ indx ], data ) || '' );
|
|
query = '(' + ( tsf.parseFilter( c, data2.filter, data ) || '' ) + ')';
|
|
try {
|
|
// use try/catch, because query may not be a valid regex if "|" is contained within a partial regex search,
|
|
// e.g "/(Alex|Aar" -> Uncaught SyntaxError: Invalid regular expression: /(/(Alex)/: Unterminated group
|
|
regex = new RegExp( data.isMatch ? query : '^' + query + '$', c.widgetOptions.filter_ignoreCase ? 'i' : '' );
|
|
// filterMatched = data2.filter === '' && indx > 0 ? true
|
|
// look for an exact match with the 'or' unless the 'filter-match' class is found
|
|
filterMatched = regex.test( data2.exact ) || tsf.processTypes( c, data2, vars );
|
|
if ( filterMatched ) {
|
|
return filterMatched;
|
|
}
|
|
} catch ( error ) {
|
|
return null;
|
|
}
|
|
}
|
|
// may be null from processing types
|
|
return filterMatched || false;
|
|
}
|
|
return null;
|
|
},
|
|
// Look for an AND or && operator ( logical and )
|
|
and : function( c, data, vars ) {
|
|
if ( tsfRegex.andTest.test( data.filter ) ) {
|
|
var indx, filterMatched, result, query, regex,
|
|
// duplicate data but split filter
|
|
data2 = $.extend( {}, data ),
|
|
filter = data.filter.split( tsfRegex.andSplit ),
|
|
iFilter = data.iFilter.split( tsfRegex.andSplit ),
|
|
len = filter.length;
|
|
for ( indx = 0; indx < len; indx++ ) {
|
|
data2.nestedFilters = true;
|
|
data2.filter = '' + ( tsf.parseFilter( c, filter[ indx ], data ) || '' );
|
|
data2.iFilter = '' + ( tsf.parseFilter( c, iFilter[ indx ], data ) || '' );
|
|
query = ( '(' + ( tsf.parseFilter( c, data2.filter, data ) || '' ) + ')' )
|
|
// replace wild cards since /(a*)/i will match anything
|
|
.replace( tsfRegex.wild01, '\\S{1}' ).replace( tsfRegex.wild0More, '\\S*' );
|
|
try {
|
|
// use try/catch just in case RegExp is invalid
|
|
regex = new RegExp( data.isMatch ? query : '^' + query + '$', c.widgetOptions.filter_ignoreCase ? 'i' : '' );
|
|
// look for an exact match with the 'and' unless the 'filter-match' class is found
|
|
result = ( regex.test( data2.exact ) || tsf.processTypes( c, data2, vars ) );
|
|
if ( indx === 0 ) {
|
|
filterMatched = result;
|
|
} else {
|
|
filterMatched = filterMatched && result;
|
|
}
|
|
} catch ( error ) {
|
|
return null;
|
|
}
|
|
}
|
|
// may be null from processing types
|
|
return filterMatched || false;
|
|
}
|
|
return null;
|
|
},
|
|
// Look for regex
|
|
regex: function( c, data ) {
|
|
if ( tsfRegex.regex.test( data.filter ) ) {
|
|
var matches,
|
|
// cache regex per column for optimal speed
|
|
regex = data.filter_regexCache[ data.index ] || tsfRegex.regex.exec( data.filter ),
|
|
isRegex = regex instanceof RegExp;
|
|
try {
|
|
if ( !isRegex ) {
|
|
// force case insensitive search if ignoreCase option set?
|
|
// if ( c.ignoreCase && !regex[2] ) { regex[2] = 'i'; }
|
|
data.filter_regexCache[ data.index ] = regex = new RegExp( regex[1], regex[2] );
|
|
}
|
|
matches = regex.test( data.exact );
|
|
} catch ( error ) {
|
|
matches = false;
|
|
}
|
|
return matches;
|
|
}
|
|
return null;
|
|
},
|
|
// Look for operators >, >=, < or <=
|
|
operators: function( c, data ) {
|
|
// ignore empty strings... because '' < 10 is true
|
|
if ( tsfRegex.operTest.test( data.iFilter ) && data.iExact !== '' ) {
|
|
var cachedValue, result, txt,
|
|
table = c.table,
|
|
parsed = data.parsed[ data.index ],
|
|
query = ts.formatFloat( data.iFilter.replace( tsfRegex.operators, '' ), table ),
|
|
parser = c.parsers[ data.index ] || {},
|
|
savedSearch = query;
|
|
// parse filter value in case we're comparing numbers ( dates )
|
|
if ( parsed || parser.type === 'numeric' ) {
|
|
txt = $.trim( '' + data.iFilter.replace( tsfRegex.operators, '' ) );
|
|
result = tsf.parseFilter( c, txt, data, true );
|
|
query = ( typeof result === 'number' && result !== '' && !isNaN( result ) ) ? result : query;
|
|
}
|
|
// iExact may be numeric - see issue #149;
|
|
// check if cached is defined, because sometimes j goes out of range? ( numeric columns )
|
|
if ( ( parsed || parser.type === 'numeric' ) && !isNaN( query ) &&
|
|
typeof data.cache !== 'undefined' ) {
|
|
cachedValue = data.cache;
|
|
} else {
|
|
txt = isNaN( data.iExact ) ? data.iExact.replace( ts.regex.nondigit, '' ) : data.iExact;
|
|
cachedValue = ts.formatFloat( txt, table );
|
|
}
|
|
if ( tsfRegex.gtTest.test( data.iFilter ) ) {
|
|
result = tsfRegex.gteTest.test( data.iFilter ) ? cachedValue >= query : cachedValue > query;
|
|
} else if ( tsfRegex.ltTest.test( data.iFilter ) ) {
|
|
result = tsfRegex.lteTest.test( data.iFilter ) ? cachedValue <= query : cachedValue < query;
|
|
}
|
|
// keep showing all rows if nothing follows the operator
|
|
if ( !result && savedSearch === '' ) {
|
|
result = true;
|
|
}
|
|
return result;
|
|
}
|
|
return null;
|
|
},
|
|
// Look for a not match
|
|
notMatch: function( c, data ) {
|
|
if ( tsfRegex.notTest.test( data.iFilter ) ) {
|
|
var indx,
|
|
txt = data.iFilter.replace( '!', '' ),
|
|
filter = tsf.parseFilter( c, txt, data ) || '';
|
|
if ( tsfRegex.exact.test( filter ) ) {
|
|
// look for exact not matches - see #628
|
|
filter = filter.replace( tsfRegex.exact, '' );
|
|
return filter === '' ? true : $.trim( filter ) !== data.iExact;
|
|
} else {
|
|
indx = data.iExact.search( $.trim( filter ) );
|
|
return filter === '' ? true : !( c.widgetOptions.filter_startsWith ? indx === 0 : indx >= 0 );
|
|
}
|
|
}
|
|
return null;
|
|
},
|
|
// Look for quotes or equals to get an exact match; ignore type since iExact could be numeric
|
|
exact: function( c, data ) {
|
|
/*jshint eqeqeq:false */
|
|
if ( tsfRegex.exact.test( data.iFilter ) ) {
|
|
var txt = data.iFilter.replace( tsfRegex.exact, '' ),
|
|
filter = tsf.parseFilter( c, txt, data ) || '';
|
|
return data.anyMatch ? $.inArray( filter, data.rowArray ) >= 0 : filter == data.iExact;
|
|
}
|
|
return null;
|
|
},
|
|
// Look for a range ( using ' to ' or ' - ' ) - see issue #166; thanks matzhu!
|
|
range : function( c, data ) {
|
|
if ( tsfRegex.toTest.test( data.iFilter ) ) {
|
|
var result, tmp, range1, range2,
|
|
table = c.table,
|
|
index = data.index,
|
|
parsed = data.parsed[index],
|
|
// make sure the dash is for a range and not indicating a negative number
|
|
query = data.iFilter.split( tsfRegex.toSplit );
|
|
|
|
tmp = query[0].replace( ts.regex.nondigit, '' ) || '';
|
|
range1 = ts.formatFloat( tsf.parseFilter( c, tmp, data ), table );
|
|
tmp = query[1].replace( ts.regex.nondigit, '' ) || '';
|
|
range2 = ts.formatFloat( tsf.parseFilter( c, tmp, data ), table );
|
|
// parse filter value in case we're comparing numbers ( dates )
|
|
if ( parsed || c.parsers[ index ].type === 'numeric' ) {
|
|
result = c.parsers[ index ].format( '' + query[0], table, c.$headers.eq( index ), index );
|
|
range1 = ( result !== '' && !isNaN( result ) ) ? result : range1;
|
|
result = c.parsers[ index ].format( '' + query[1], table, c.$headers.eq( index ), index );
|
|
range2 = ( result !== '' && !isNaN( result ) ) ? result : range2;
|
|
}
|
|
if ( ( parsed || c.parsers[ index ].type === 'numeric' ) && !isNaN( range1 ) && !isNaN( range2 ) ) {
|
|
result = data.cache;
|
|
} else {
|
|
tmp = isNaN( data.iExact ) ? data.iExact.replace( ts.regex.nondigit, '' ) : data.iExact;
|
|
result = ts.formatFloat( tmp, table );
|
|
}
|
|
if ( range1 > range2 ) {
|
|
tmp = range1; range1 = range2; range2 = tmp; // swap
|
|
}
|
|
return ( result >= range1 && result <= range2 ) || ( range1 === '' || range2 === '' );
|
|
}
|
|
return null;
|
|
},
|
|
// Look for wild card: ? = single, * = multiple, or | = logical OR
|
|
wild : function( c, data ) {
|
|
if ( tsfRegex.wildOrTest.test( data.iFilter ) ) {
|
|
var query = '' + ( tsf.parseFilter( c, data.iFilter, data ) || '' );
|
|
// look for an exact match with the 'or' unless the 'filter-match' class is found
|
|
if ( !tsfRegex.wildTest.test( query ) && data.nestedFilters ) {
|
|
query = data.isMatch ? query : '^(' + query + ')$';
|
|
}
|
|
// parsing the filter may not work properly when using wildcards =/
|
|
try {
|
|
return new RegExp(
|
|
query.replace( tsfRegex.wild01, '\\S{1}' ).replace( tsfRegex.wild0More, '\\S*' ),
|
|
c.widgetOptions.filter_ignoreCase ? 'i' : ''
|
|
)
|
|
.test( data.exact );
|
|
} catch ( error ) {
|
|
return null;
|
|
}
|
|
}
|
|
return null;
|
|
},
|
|
// fuzzy text search; modified from https://github.com/mattyork/fuzzy ( MIT license )
|
|
fuzzy: function( c, data ) {
|
|
if ( tsfRegex.fuzzyTest.test( data.iFilter ) ) {
|
|
var indx,
|
|
patternIndx = 0,
|
|
len = data.iExact.length,
|
|
txt = data.iFilter.slice( 1 ),
|
|
pattern = tsf.parseFilter( c, txt, data ) || '';
|
|
for ( indx = 0; indx < len; indx++ ) {
|
|
if ( data.iExact[ indx ] === pattern[ patternIndx ] ) {
|
|
patternIndx += 1;
|
|
}
|
|
}
|
|
return patternIndx === pattern.length;
|
|
}
|
|
return null;
|
|
}
|
|
},
|
|
init: function( table, c, wo ) {
|
|
// filter language options
|
|
ts.language = $.extend( true, {}, {
|
|
to : 'to',
|
|
or : 'or',
|
|
and : 'and'
|
|
}, ts.language );
|
|
|
|
var options, string, txt, $header, column, filters, val, fxn, noSelect;
|
|
c.$table.addClass( 'hasFilters' );
|
|
c.lastSearch = [];
|
|
|
|
// define timers so using clearTimeout won't cause an undefined error
|
|
wo.filter_searchTimer = null;
|
|
wo.filter_initTimer = null;
|
|
wo.filter_formatterCount = 0;
|
|
wo.filter_formatterInit = [];
|
|
wo.filter_anyColumnSelector = '[data-column="all"],[data-column="any"]';
|
|
wo.filter_multipleColumnSelector = '[data-column*="-"],[data-column*=","]';
|
|
|
|
val = '\\{' + tsfRegex.query + '\\}';
|
|
$.extend( tsfRegex, {
|
|
child : new RegExp( c.cssChildRow ),
|
|
filtered : new RegExp( wo.filter_filteredRow ),
|
|
alreadyFiltered : new RegExp( '(\\s+(' + ts.language.or + '|-|' + ts.language.to + ')\\s+)', 'i' ),
|
|
toTest : new RegExp( '\\s+(-|' + ts.language.to + ')\\s+', 'i' ),
|
|
toSplit : new RegExp( '(?:\\s+(?:-|' + ts.language.to + ')\\s+)', 'gi' ),
|
|
andTest : new RegExp( '\\s+(' + ts.language.and + '|&&)\\s+', 'i' ),
|
|
andSplit : new RegExp( '(?:\\s+(?:' + ts.language.and + '|&&)\\s+)', 'gi' ),
|
|
orTest : new RegExp( '(\\||\\s+' + ts.language.or + '\\s+)', 'i' ),
|
|
orSplit : new RegExp( '(?:\\s+(?:' + ts.language.or + ')\\s+|\\|)', 'gi' ),
|
|
iQuery : new RegExp( val, 'i' ),
|
|
igQuery : new RegExp( val, 'ig' ),
|
|
operTest : /^[<>]=?/,
|
|
gtTest : />/,
|
|
gteTest : />=/,
|
|
ltTest : /</,
|
|
lteTest : /<=/,
|
|
notTest : /^\!/,
|
|
wildOrTest : /[\?\*\|]/,
|
|
wildTest : /\?\*/,
|
|
fuzzyTest : /^~/,
|
|
exactTest : /[=\"\|!]/
|
|
});
|
|
|
|
// don't build filter row if columnFilters is false or all columns are set to 'filter-false'
|
|
// see issue #156
|
|
val = c.$headers.filter( '.filter-false, .parser-false' ).length;
|
|
if ( wo.filter_columnFilters !== false && val !== c.$headers.length ) {
|
|
// build filter row
|
|
tsf.buildRow( table, c, wo );
|
|
}
|
|
|
|
txt = 'addRows updateCell update updateRows updateComplete appendCache filterReset ' +
|
|
'filterResetSaved filterEnd search '.split( ' ' ).join( c.namespace + 'filter ' );
|
|
c.$table.bind( txt, function( event, filter ) {
|
|
val = wo.filter_hideEmpty &&
|
|
$.isEmptyObject( c.cache ) &&
|
|
!( c.delayInit && event.type === 'appendCache' );
|
|
// hide filter row using the 'filtered' class name
|
|
c.$table.find( '.' + tscss.filterRow ).toggleClass( wo.filter_filteredRow, val ); // fixes #450
|
|
if ( !/(search|filter)/.test( event.type ) ) {
|
|
event.stopPropagation();
|
|
tsf.buildDefault( table, true );
|
|
}
|
|
if ( event.type === 'filterReset' ) {
|
|
c.$table.find( '.' + tscss.filter ).add( wo.filter_$externalFilters ).val( '' );
|
|
tsf.searching( table, [] );
|
|
} else if ( event.type === 'filterResetSaved' ) {
|
|
ts.storage( table, 'tablesorter-filters', '' );
|
|
} else if ( event.type === 'filterEnd' ) {
|
|
tsf.buildDefault( table, true );
|
|
} else {
|
|
// send false argument to force a new search; otherwise if the filter hasn't changed,
|
|
// it will return
|
|
filter = event.type === 'search' ? filter :
|
|
event.type === 'updateComplete' ? c.$table.data( 'lastSearch' ) : '';
|
|
if ( /(update|add)/.test( event.type ) && event.type !== 'updateComplete' ) {
|
|
// force a new search since content has changed
|
|
c.lastCombinedFilter = null;
|
|
c.lastSearch = [];
|
|
}
|
|
// pass true ( skipFirst ) to prevent the tablesorter.setFilters function from skipping the first
|
|
// input ensures all inputs are updated when a search is triggered on the table
|
|
// $( 'table' ).trigger( 'search', [...] );
|
|
tsf.searching( table, filter, true );
|
|
}
|
|
return false;
|
|
});
|
|
|
|
// reset button/link
|
|
if ( wo.filter_reset ) {
|
|
if ( wo.filter_reset instanceof $ ) {
|
|
// reset contains a jQuery object, bind to it
|
|
wo.filter_reset.click( function() {
|
|
c.$table.triggerHandler( 'filterReset' );
|
|
});
|
|
} else if ( $( wo.filter_reset ).length ) {
|
|
// reset is a jQuery selector, use event delegation
|
|
$( document )
|
|
.undelegate( wo.filter_reset, 'click' + c.namespace + 'filter' )
|
|
.delegate( wo.filter_reset, 'click' + c.namespace + 'filter', function() {
|
|
// trigger a reset event, so other functions ( filter_formatter ) know when to reset
|
|
c.$table.triggerHandler( 'filterReset' );
|
|
});
|
|
}
|
|
}
|
|
if ( wo.filter_functions ) {
|
|
for ( column = 0; column < c.columns; column++ ) {
|
|
fxn = ts.getColumnData( table, wo.filter_functions, column );
|
|
if ( fxn ) {
|
|
// remove 'filter-select' from header otherwise the options added here are replaced with
|
|
// all options
|
|
$header = c.$headerIndexed[ column ].removeClass( 'filter-select' );
|
|
// don't build select if 'filter-false' or 'parser-false' set
|
|
noSelect = !( $header.hasClass( 'filter-false' ) || $header.hasClass( 'parser-false' ) );
|
|
options = '';
|
|
if ( fxn === true && noSelect ) {
|
|
tsf.buildSelect( table, column );
|
|
} else if ( typeof fxn === 'object' && noSelect ) {
|
|
// add custom drop down list
|
|
for ( string in fxn ) {
|
|
if ( typeof string === 'string' ) {
|
|
options += options === '' ?
|
|
'<option value="">' +
|
|
( $header.data( 'placeholder' ) ||
|
|
$header.attr( 'data-placeholder' ) ||
|
|
wo.filter_placeholder.select ||
|
|
''
|
|
) +
|
|
'</option>' : '';
|
|
val = string;
|
|
txt = string;
|
|
if ( string.indexOf( wo.filter_selectSourceSeparator ) >= 0 ) {
|
|
val = string.split( wo.filter_selectSourceSeparator );
|
|
txt = val[1];
|
|
val = val[0];
|
|
}
|
|
options += '<option ' +
|
|
( txt === val ? '' : 'data-function-name="' + string + '" ' ) +
|
|
'value="' + val + '">' + txt + '</option>';
|
|
}
|
|
}
|
|
c.$table
|
|
.find( 'thead' )
|
|
.find( 'select.' + tscss.filter + '[data-column="' + column + '"]' )
|
|
.append( options );
|
|
txt = wo.filter_selectSource;
|
|
fxn = typeof txt === 'function' ? true : ts.getColumnData( table, txt, column );
|
|
if ( fxn ) {
|
|
// updating so the extra options are appended
|
|
tsf.buildSelect( c.table, column, '', true, $header.hasClass( wo.filter_onlyAvail ) );
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
// not really updating, but if the column has both the 'filter-select' class &
|
|
// filter_functions set to true, it would append the same options twice.
|
|
tsf.buildDefault( table, true );
|
|
|
|
tsf.bindSearch( table, c.$table.find( '.' + tscss.filter ), true );
|
|
if ( wo.filter_external ) {
|
|
tsf.bindSearch( table, wo.filter_external );
|
|
}
|
|
|
|
if ( wo.filter_hideFilters ) {
|
|
tsf.hideFilters( c );
|
|
}
|
|
|
|
// show processing icon
|
|
if ( c.showProcessing ) {
|
|
txt = 'filterStart filterEnd '.split( ' ' ).join( c.namespace + 'filter ' );
|
|
c.$table
|
|
.unbind( txt.replace( ts.regex.spaces, ' ' ) )
|
|
.bind( txt, function( event, columns ) {
|
|
// only add processing to certain columns to all columns
|
|
$header = ( columns ) ?
|
|
c.$table
|
|
.find( '.' + tscss.header )
|
|
.filter( '[data-column]' )
|
|
.filter( function() {
|
|
return columns[ $( this ).data( 'column' ) ] !== '';
|
|
}) : '';
|
|
ts.isProcessing( table, event.type === 'filterStart', columns ? $header : '' );
|
|
});
|
|
}
|
|
|
|
// set filtered rows count ( intially unfiltered )
|
|
c.filteredRows = c.totalRows;
|
|
|
|
// add default values
|
|
txt = 'tablesorter-initialized pagerBeforeInitialized '.split( ' ' ).join( c.namespace + 'filter ' );
|
|
c.$table
|
|
.unbind( txt.replace( ts.regex.spaces, ' ' ) )
|
|
.bind( txt, function() {
|
|
// redefine 'wo' as it does not update properly inside this callback
|
|
var wo = this.config.widgetOptions;
|
|
filters = tsf.setDefaults( table, c, wo ) || [];
|
|
if ( filters.length ) {
|
|
// prevent delayInit from triggering a cache build if filters are empty
|
|
if ( !( c.delayInit && filters.join( '' ) === '' ) ) {
|
|
ts.setFilters( table, filters, true );
|
|
}
|
|
}
|
|
c.$table.triggerHandler( 'filterFomatterUpdate' );
|
|
// trigger init after setTimeout to prevent multiple filterStart/End/Init triggers
|
|
setTimeout( function() {
|
|
if ( !wo.filter_initialized ) {
|
|
tsf.filterInitComplete( c );
|
|
}
|
|
}, 100 );
|
|
});
|
|
// if filter widget is added after pager has initialized; then set filter init flag
|
|
if ( c.pager && c.pager.initialized && !wo.filter_initialized ) {
|
|
c.$table.triggerHandler( 'filterFomatterUpdate' );
|
|
setTimeout( function() {
|
|
tsf.filterInitComplete( c );
|
|
}, 100 );
|
|
}
|
|
},
|
|
// $cell parameter, but not the config, is passed to the filter_formatters,
|
|
// so we have to work with it instead
|
|
formatterUpdated: function( $cell, column ) {
|
|
// prevent error if $cell is undefined - see #1056
|
|
var wo = $cell && $cell.closest( 'table' )[0].config.widgetOptions;
|
|
if ( wo && !wo.filter_initialized ) {
|
|
// add updates by column since this function
|
|
// may be called numerous times before initialization
|
|
wo.filter_formatterInit[ column ] = 1;
|
|
}
|
|
},
|
|
filterInitComplete: function( c ) {
|
|
var indx, len,
|
|
wo = c.widgetOptions,
|
|
count = 0,
|
|
completed = function() {
|
|
wo.filter_initialized = true;
|
|
c.$table.triggerHandler( 'filterInit', c );
|
|
tsf.findRows( c.table, c.$table.data( 'lastSearch' ) || [] );
|
|
};
|
|
if ( $.isEmptyObject( wo.filter_formatter ) ) {
|
|
completed();
|
|
} else {
|
|
len = wo.filter_formatterInit.length;
|
|
for ( indx = 0; indx < len; indx++ ) {
|
|
if ( wo.filter_formatterInit[ indx ] === 1 ) {
|
|
count++;
|
|
}
|
|
}
|
|
clearTimeout( wo.filter_initTimer );
|
|
if ( !wo.filter_initialized && count === wo.filter_formatterCount ) {
|
|
// filter widget initialized
|
|
completed();
|
|
} else if ( !wo.filter_initialized ) {
|
|
// fall back in case a filter_formatter doesn't call
|
|
// $.tablesorter.filter.formatterUpdated( $cell, column ), and the count is off
|
|
wo.filter_initTimer = setTimeout( function() {
|
|
completed();
|
|
}, 500 );
|
|
}
|
|
}
|
|
},
|
|
// encode or decode filters for storage; see #1026
|
|
processFilters: function( filters, encode ) {
|
|
var indx,
|
|
mode = encode ? encodeURIComponent : decodeURIComponent,
|
|
len = filters.length;
|
|
for ( indx = 0; indx < len; indx++ ) {
|
|
if ( filters[ indx ] ) {
|
|
filters[ indx ] = mode( filters[ indx ] );
|
|
}
|
|
}
|
|
return filters;
|
|
},
|
|
setDefaults: function( table, c, wo ) {
|
|
var isArray, saved, indx, col, $filters,
|
|
// get current ( default ) filters
|
|
filters = ts.getFilters( table ) || [];
|
|
if ( wo.filter_saveFilters && ts.storage ) {
|
|
saved = ts.storage( table, 'tablesorter-filters' ) || [];
|
|
isArray = $.isArray( saved );
|
|
// make sure we're not just getting an empty array
|
|
if ( !( isArray && saved.join( '' ) === '' || !isArray ) ) {
|
|
filters = tsf.processFilters( saved );
|
|
}
|
|
}
|
|
// if no filters saved, then check default settings
|
|
if ( filters.join( '' ) === '' ) {
|
|
// allow adding default setting to external filters
|
|
$filters = c.$headers.add( wo.filter_$externalFilters )
|
|
.filter( '[' + wo.filter_defaultAttrib + ']' );
|
|
for ( indx = 0; indx <= c.columns; indx++ ) {
|
|
// include data-column='all' external filters
|
|
col = indx === c.columns ? 'all' : indx;
|
|
filters[ indx ] = $filters
|
|
.filter( '[data-column="' + col + '"]' )
|
|
.attr( wo.filter_defaultAttrib ) || filters[indx] || '';
|
|
}
|
|
}
|
|
c.$table.data( 'lastSearch', filters );
|
|
return filters;
|
|
},
|
|
parseFilter: function( c, filter, data, parsed ) {
|
|
return parsed || data.parsed[ data.index ] ?
|
|
c.parsers[ data.index ].format( filter, c.table, [], data.index ) :
|
|
filter;
|
|
},
|
|
buildRow: function( table, c, wo ) {
|
|
var $filter, col, column, $header, makeSelect, disabled, name, ffxn, tmp,
|
|
// c.columns defined in computeThIndexes()
|
|
cellFilter = wo.filter_cellFilter,
|
|
columns = c.columns,
|
|
arry = $.isArray( cellFilter ),
|
|
buildFilter = '<tr role="row" class="' + tscss.filterRow + ' ' + c.cssIgnoreRow + '">';
|
|
for ( column = 0; column < columns; column++ ) {
|
|
if ( c.$headerIndexed[ column ].length ) {
|
|
// account for entire column set with colspan. See #1047
|
|
tmp = c.$headerIndexed[ column ] && c.$headerIndexed[ column ][0].colSpan || 0;
|
|
if ( tmp > 1 ) {
|
|
buildFilter += '<td data-column="' + column + '-' + ( column + tmp - 1 ) + '" colspan="' + tmp + '"';
|
|
} else {
|
|
buildFilter += '<td data-column="' + column + '"';
|
|
}
|
|
if ( arry ) {
|
|
buildFilter += ( cellFilter[ column ] ? ' class="' + cellFilter[ column ] + '"' : '' );
|
|
} else {
|
|
buildFilter += ( cellFilter !== '' ? ' class="' + cellFilter + '"' : '' );
|
|
}
|
|
buildFilter += '></td>';
|
|
}
|
|
}
|
|
c.$filters = $( buildFilter += '</tr>' )
|
|
.appendTo( c.$table.children( 'thead' ).eq( 0 ) )
|
|
.children( 'td' );
|
|
// build each filter input
|
|
for ( column = 0; column < columns; column++ ) {
|
|
disabled = false;
|
|
// assuming last cell of a column is the main column
|
|
$header = c.$headerIndexed[ column ];
|
|
if ( $header && $header.length ) {
|
|
// $filter = c.$filters.filter( '[data-column="' + column + '"]' );
|
|
$filter = tsf.getColumnElm( c, c.$filters, column );
|
|
ffxn = ts.getColumnData( table, wo.filter_functions, column );
|
|
makeSelect = ( wo.filter_functions && ffxn && typeof ffxn !== 'function' ) ||
|
|
$header.hasClass( 'filter-select' );
|
|
// get data from jQuery data, metadata, headers option or header class name
|
|
col = ts.getColumnData( table, c.headers, column );
|
|
disabled = ts.getData( $header[0], col, 'filter' ) === 'false' ||
|
|
ts.getData( $header[0], col, 'parser' ) === 'false';
|
|
|
|
if ( makeSelect ) {
|
|
buildFilter = $( '<select>' ).appendTo( $filter );
|
|
} else {
|
|
ffxn = ts.getColumnData( table, wo.filter_formatter, column );
|
|
if ( ffxn ) {
|
|
wo.filter_formatterCount++;
|
|
buildFilter = ffxn( $filter, column );
|
|
// no element returned, so lets go find it
|
|
if ( buildFilter && buildFilter.length === 0 ) {
|
|
buildFilter = $filter.children( 'input' );
|
|
}
|
|
// element not in DOM, so lets attach it
|
|
if ( buildFilter && ( buildFilter.parent().length === 0 ||
|
|
( buildFilter.parent().length && buildFilter.parent()[0] !== $filter[0] ) ) ) {
|
|
$filter.append( buildFilter );
|
|
}
|
|
} else {
|
|
buildFilter = $( '<input type="search">' ).appendTo( $filter );
|
|
}
|
|
if ( buildFilter ) {
|
|
tmp = $header.data( 'placeholder' ) ||
|
|
$header.attr( 'data-placeholder' ) ||
|
|
wo.filter_placeholder.search || '';
|
|
buildFilter.attr( 'placeholder', tmp );
|
|
}
|
|
}
|
|
if ( buildFilter ) {
|
|
// add filter class name
|
|
name = ( $.isArray( wo.filter_cssFilter ) ?
|
|
( typeof wo.filter_cssFilter[column] !== 'undefined' ? wo.filter_cssFilter[column] || '' : '' ) :
|
|
wo.filter_cssFilter ) || '';
|
|
// copy data-column from table cell (it will include colspan)
|
|
buildFilter.addClass( tscss.filter + ' ' + name ).attr( 'data-column', $filter.attr( 'data-column' ) );
|
|
if ( disabled ) {
|
|
buildFilter.attr( 'placeholder', '' ).addClass( tscss.filterDisabled )[0].disabled = true;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
bindSearch: function( table, $el, internal ) {
|
|
table = $( table )[0];
|
|
$el = $( $el ); // allow passing a selector string
|
|
if ( !$el.length ) { return; }
|
|
var tmp,
|
|
c = table.config,
|
|
wo = c.widgetOptions,
|
|
namespace = c.namespace + 'filter',
|
|
$ext = wo.filter_$externalFilters;
|
|
if ( internal !== true ) {
|
|
// save anyMatch element
|
|
tmp = wo.filter_anyColumnSelector + ',' + wo.filter_multipleColumnSelector;
|
|
wo.filter_$anyMatch = $el.filter( tmp );
|
|
if ( $ext && $ext.length ) {
|
|
wo.filter_$externalFilters = wo.filter_$externalFilters.add( $el );
|
|
} else {
|
|
wo.filter_$externalFilters = $el;
|
|
}
|
|
// update values ( external filters added after table initialization )
|
|
ts.setFilters( table, c.$table.data( 'lastSearch' ) || [], internal === false );
|
|
}
|
|
// unbind events
|
|
tmp = ( 'keypress keyup keydown search change input '.split( ' ' ).join( namespace + ' ' ) );
|
|
$el
|
|
// use data attribute instead of jQuery data since the head is cloned without including
|
|
// the data/binding
|
|
.attr( 'data-lastSearchTime', new Date().getTime() )
|
|
.unbind( tmp.replace( ts.regex.spaces, ' ' ) )
|
|
.bind( 'keydown' + namespace, function( event ) {
|
|
if ( event.which === tskeyCodes.escape && !wo.filter_resetOnEsc ) {
|
|
// prevent keypress event
|
|
return false;
|
|
}
|
|
})
|
|
.bind( 'keyup' + namespace, function( event ) {
|
|
var column = parseInt( $( this ).attr( 'data-column' ), 10 );
|
|
$( this ).attr( 'data-lastSearchTime', new Date().getTime() );
|
|
// emulate what webkit does.... escape clears the filter
|
|
if ( event.which === tskeyCodes.escape ) {
|
|
// make sure to restore the last value on escape
|
|
this.value = wo.filter_resetOnEsc ? '' : c.lastSearch[column];
|
|
// live search
|
|
} else if ( wo.filter_liveSearch === false ) {
|
|
return;
|
|
// don't return if the search value is empty ( all rows need to be revealed )
|
|
} else if ( this.value !== '' && (
|
|
// liveSearch can contain a min value length; ignore arrow and meta keys, but allow backspace
|
|
( typeof wo.filter_liveSearch === 'number' && this.value.length < wo.filter_liveSearch ) ||
|
|
// let return & backspace continue on, but ignore arrows & non-valid characters
|
|
( event.which !== tskeyCodes.enter && event.which !== tskeyCodes.backSpace &&
|
|
( event.which < tskeyCodes.space || ( event.which >= tskeyCodes.left && event.which <= tskeyCodes.down ) ) ) ) ) {
|
|
return;
|
|
}
|
|
// change event = no delay; last true flag tells getFilters to skip newest timed input
|
|
tsf.searching( table, true, true );
|
|
})
|
|
// include change for select - fixes #473
|
|
.bind( 'search change keypress input '.split( ' ' ).join( namespace + ' ' ), function( event ) {
|
|
// don't get cached data, in case data-column changes dynamically
|
|
var column = parseInt( $( this ).attr( 'data-column' ), 10 );
|
|
// don't allow 'change' event to process if the input value is the same - fixes #685
|
|
if ( wo.filter_initialized && ( event.which === tskeyCodes.enter || event.type === 'search' ||
|
|
( event.type === 'change' ) && this.value !== c.lastSearch[column] ) ||
|
|
// only "input" event fires in MS Edge when clicking the "x" to clear the search
|
|
( event.type === 'input' && this.value === '' ) ) {
|
|
event.preventDefault();
|
|
// init search with no delay
|
|
$( this ).attr( 'data-lastSearchTime', new Date().getTime() );
|
|
tsf.searching( table, event.type !== 'keypress', true );
|
|
}
|
|
});
|
|
},
|
|
searching: function( table, filter, skipFirst ) {
|
|
var wo = table.config.widgetOptions;
|
|
clearTimeout( wo.filter_searchTimer );
|
|
if ( typeof filter === 'undefined' || filter === true ) {
|
|
// delay filtering
|
|
wo.filter_searchTimer = setTimeout( function() {
|
|
tsf.checkFilters( table, filter, skipFirst );
|
|
}, wo.filter_liveSearch ? wo.filter_searchDelay : 10 );
|
|
} else {
|
|
// skip delay
|
|
tsf.checkFilters( table, filter, skipFirst );
|
|
}
|
|
},
|
|
checkFilters: function( table, filter, skipFirst ) {
|
|
var c = table.config,
|
|
wo = c.widgetOptions,
|
|
filterArray = $.isArray( filter ),
|
|
filters = ( filterArray ) ? filter : ts.getFilters( table, true ),
|
|
combinedFilters = ( filters || [] ).join( '' ); // combined filter values
|
|
// prevent errors if delay init is set
|
|
if ( $.isEmptyObject( c.cache ) ) {
|
|
// update cache if delayInit set & pager has initialized ( after user initiates a search )
|
|
if ( c.delayInit && ( !c.pager || c.pager && c.pager.initialized ) ) {
|
|
ts.updateCache( c, function() {
|
|
tsf.checkFilters( table, false, skipFirst );
|
|
});
|
|
}
|
|
return;
|
|
}
|
|
// add filter array back into inputs
|
|
if ( filterArray ) {
|
|
ts.setFilters( table, filters, false, skipFirst !== true );
|
|
if ( !wo.filter_initialized ) { c.lastCombinedFilter = ''; }
|
|
}
|
|
if ( wo.filter_hideFilters ) {
|
|
// show/hide filter row as needed
|
|
c.$table
|
|
.find( '.' + tscss.filterRow )
|
|
.triggerHandler( combinedFilters === '' ? 'mouseleave' : 'mouseenter' );
|
|
}
|
|
// return if the last search is the same; but filter === false when updating the search
|
|
// see example-widget-filter.html filter toggle buttons
|
|
if ( c.lastCombinedFilter === combinedFilters && filter !== false ) {
|
|
return;
|
|
} else if ( filter === false ) {
|
|
// force filter refresh
|
|
c.lastCombinedFilter = null;
|
|
c.lastSearch = [];
|
|
}
|
|
// define filter inside it is false
|
|
filters = filters || [];
|
|
// convert filters to strings - see #1070
|
|
filters = Array.prototype.map ?
|
|
filters.map( String ) :
|
|
// for IE8 & older browsers - maybe not the best method
|
|
filters.join( '\ufffd' ).split( '\ufffd' );
|
|
|
|
if ( wo.filter_initialized ) {
|
|
c.$table.triggerHandler( 'filterStart', [ filters ] );
|
|
}
|
|
if ( c.showProcessing ) {
|
|
// give it time for the processing icon to kick in
|
|
setTimeout( function() {
|
|
tsf.findRows( table, filters, combinedFilters );
|
|
return false;
|
|
}, 30 );
|
|
} else {
|
|
tsf.findRows( table, filters, combinedFilters );
|
|
return false;
|
|
}
|
|
},
|
|
hideFilters: function( c, $table ) {
|
|
var timer,
|
|
$row = ( $table || c.$table ).find( '.' + tscss.filterRow ).addClass( tscss.filterRowHide );
|
|
$row
|
|
.bind( 'mouseenter mouseleave', function( e ) {
|
|
// save event object - http://bugs.jquery.com/ticket/12140
|
|
var event = e,
|
|
$filterRow = $( this );
|
|
clearTimeout( timer );
|
|
timer = setTimeout( function() {
|
|
if ( /enter|over/.test( event.type ) ) {
|
|
$filterRow.removeClass( tscss.filterRowHide );
|
|
} else {
|
|
// don't hide if input has focus
|
|
// $( ':focus' ) needs jQuery 1.6+
|
|
if ( $( document.activeElement ).closest( 'tr' )[0] !== $filterRow[0] ) {
|
|
// don't hide row if any filter has a value
|
|
if ( c.lastCombinedFilter === '' ) {
|
|
$filterRow.addClass( tscss.filterRowHide );
|
|
}
|
|
}
|
|
}
|
|
}, 200 );
|
|
})
|
|
.find( 'input, select' ).bind( 'focus blur', function( e ) {
|
|
var event = e,
|
|
$row = $( this ).closest( 'tr' );
|
|
clearTimeout( timer );
|
|
timer = setTimeout( function() {
|
|
clearTimeout( timer );
|
|
// don't hide row if any filter has a value
|
|
if ( ts.getFilters( c.$table ).join( '' ) === '' ) {
|
|
$row.toggleClass( tscss.filterRowHide, event.type !== 'focus' );
|
|
}
|
|
}, 200 );
|
|
});
|
|
},
|
|
defaultFilter: function( filter, mask ) {
|
|
if ( filter === '' ) { return filter; }
|
|
var regex = tsfRegex.iQuery,
|
|
maskLen = mask.match( tsfRegex.igQuery ).length,
|
|
query = maskLen > 1 ? $.trim( filter ).split( /\s/ ) : [ $.trim( filter ) ],
|
|
len = query.length - 1,
|
|
indx = 0,
|
|
val = mask;
|
|
if ( len < 1 && maskLen > 1 ) {
|
|
// only one 'word' in query but mask has >1 slots
|
|
query[1] = query[0];
|
|
}
|
|
// replace all {query} with query words...
|
|
// if query = 'Bob', then convert mask from '!{query}' to '!Bob'
|
|
// if query = 'Bob Joe Frank', then convert mask '{q} OR {q}' to 'Bob OR Joe OR Frank'
|
|
while ( regex.test( val ) ) {
|
|
val = val.replace( regex, query[indx++] || '' );
|
|
if ( regex.test( val ) && indx < len && ( query[indx] || '' ) !== '' ) {
|
|
val = mask.replace( regex, val );
|
|
}
|
|
}
|
|
return val;
|
|
},
|
|
getLatestSearch: function( $input ) {
|
|
if ( $input ) {
|
|
return $input.sort( function( a, b ) {
|
|
return $( b ).attr( 'data-lastSearchTime' ) - $( a ).attr( 'data-lastSearchTime' );
|
|
});
|
|
}
|
|
return $input || $();
|
|
},
|
|
findRange: function( c, val, ignoreRanges ) {
|
|
// look for multiple columns '1-3,4-6,8' in data-column
|
|
var temp, ranges, range, start, end, singles, i, indx, len,
|
|
columns = [];
|
|
if ( /^[0-9]+$/.test( val ) ) {
|
|
// always return an array
|
|
return [ parseInt( val, 10 ) ];
|
|
}
|
|
// process column range
|
|
if ( !ignoreRanges && /-/.test( val ) ) {
|
|
ranges = val.match( /(\d+)\s*-\s*(\d+)/g );
|
|
len = ranges ? ranges.length : 0;
|
|
for ( indx = 0; indx < len; indx++ ) {
|
|
range = ranges[indx].split( /\s*-\s*/ );
|
|
start = parseInt( range[0], 10 ) || 0;
|
|
end = parseInt( range[1], 10 ) || ( c.columns - 1 );
|
|
if ( start > end ) {
|
|
temp = start; start = end; end = temp; // swap
|
|
}
|
|
if ( end >= c.columns ) {
|
|
end = c.columns - 1;
|
|
}
|
|
for ( ; start <= end; start++ ) {
|
|
columns[ columns.length ] = start;
|
|
}
|
|
// remove processed range from val
|
|
val = val.replace( ranges[ indx ], '' );
|
|
}
|
|
}
|
|
// process single columns
|
|
if ( !ignoreRanges && /,/.test( val ) ) {
|
|
singles = val.split( /\s*,\s*/ );
|
|
len = singles.length;
|
|
for ( i = 0; i < len; i++ ) {
|
|
if ( singles[ i ] !== '' ) {
|
|
indx = parseInt( singles[ i ], 10 );
|
|
if ( indx < c.columns ) {
|
|
columns[ columns.length ] = indx;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
// return all columns
|
|
if ( !columns.length ) {
|
|
for ( indx = 0; indx < c.columns; indx++ ) {
|
|
columns[ columns.length ] = indx;
|
|
}
|
|
}
|
|
return columns;
|
|
},
|
|
getColumnElm: function( c, $elements, column ) {
|
|
// data-column may contain multiple columns '1-3,5-6,8'
|
|
// replaces: c.$filters.filter( '[data-column="' + column + '"]' );
|
|
return $elements.filter( function() {
|
|
var cols = tsf.findRange( c, $( this ).attr( 'data-column' ) );
|
|
return $.inArray( column, cols ) > -1;
|
|
});
|
|
},
|
|
multipleColumns: function( c, $input ) {
|
|
// look for multiple columns '1-3,4-6,8' in data-column
|
|
var wo = c.widgetOptions,
|
|
// only target 'all' column inputs on initialization
|
|
// & don't target 'all' column inputs if they don't exist
|
|
targets = wo.filter_initialized || !$input.filter( wo.filter_anyColumnSelector ).length,
|
|
val = $.trim( tsf.getLatestSearch( $input ).attr( 'data-column' ) || '' );
|
|
return tsf.findRange( c, val, !targets );
|
|
},
|
|
processTypes: function( c, data, vars ) {
|
|
var ffxn,
|
|
filterMatched = null,
|
|
matches = null;
|
|
for ( ffxn in tsf.types ) {
|
|
if ( $.inArray( ffxn, vars.excludeMatch ) < 0 && matches === null ) {
|
|
matches = tsf.types[ffxn]( c, data, vars );
|
|
if ( matches !== null ) {
|
|
filterMatched = matches;
|
|
}
|
|
}
|
|
}
|
|
return filterMatched;
|
|
},
|
|
matchType: function( c, columnIndex ) {
|
|
var isMatch,
|
|
wo = c.widgetOptions,
|
|
$el = c.$headerIndexed[ columnIndex ];
|
|
// filter-exact > filter-match > filter_matchType for type
|
|
if ( $el.hasClass( 'filter-exact' ) ) {
|
|
isMatch = false;
|
|
} else if ( $el.hasClass( 'filter-match' ) ) {
|
|
isMatch = true;
|
|
} else {
|
|
// filter-select is not applied when filter_functions are used, so look for a select
|
|
if ( wo.filter_columnFilters ) {
|
|
$el = c.$filters
|
|
.find( '.' + tscss.filter )
|
|
.add( wo.filter_$externalFilters )
|
|
.filter( '[data-column="' + columnIndex + '"]' );
|
|
} else if ( wo.filter_$externalFilters ) {
|
|
$el = wo.filter_$externalFilters.filter( '[data-column="' + columnIndex + '"]' );
|
|
}
|
|
isMatch = $el.length ?
|
|
c.widgetOptions.filter_matchType[ ( $el[ 0 ].nodeName || '' ).toLowerCase() ] === 'match' :
|
|
// default to exact, if no inputs found
|
|
false;
|
|
}
|
|
return isMatch;
|
|
},
|
|
processRow: function( c, data, vars ) {
|
|
var result, filterMatched,
|
|
fxn, ffxn, txt,
|
|
wo = c.widgetOptions,
|
|
showRow = true,
|
|
|
|
// if wo.filter_$anyMatch data-column attribute is changed dynamically
|
|
// we don't want to do an "anyMatch" search on one column using data
|
|
// for the entire row - see #998
|
|
columnIndex = wo.filter_$anyMatch && wo.filter_$anyMatch.length ?
|
|
// look for multiple columns '1-3,4-6,8'
|
|
tsf.multipleColumns( c, wo.filter_$anyMatch ) :
|
|
[];
|
|
|
|
data.$cells = data.$row.children();
|
|
|
|
if ( data.anyMatchFlag && columnIndex.length > 1 ) {
|
|
data.anyMatch = true;
|
|
data.isMatch = true;
|
|
data.rowArray = data.$cells.map( function( i ) {
|
|
if ( $.inArray( i, columnIndex ) > -1 ) {
|
|
if ( data.parsed[ i ] ) {
|
|
txt = data.cacheArray[ i ];
|
|
} else {
|
|
txt = data.rawArray[ i ];
|
|
txt = $.trim( wo.filter_ignoreCase ? txt.toLowerCase() : txt );
|
|
if ( c.sortLocaleCompare ) {
|
|
txt = ts.replaceAccents( txt );
|
|
}
|
|
}
|
|
return txt;
|
|
}
|
|
}).get();
|
|
data.filter = data.anyMatchFilter;
|
|
data.iFilter = data.iAnyMatchFilter;
|
|
data.exact = data.rowArray.join( ' ' );
|
|
data.iExact = wo.filter_ignoreCase ? data.exact.toLowerCase() : data.exact;
|
|
data.cache = data.cacheArray.slice( 0, -1 ).join( ' ' );
|
|
|
|
vars.excludeMatch = vars.noAnyMatch;
|
|
filterMatched = tsf.processTypes( c, data, vars );
|
|
if ( filterMatched !== null ) {
|
|
showRow = filterMatched;
|
|
} else {
|
|
if ( wo.filter_startsWith ) {
|
|
showRow = false;
|
|
// data.rowArray may not contain all columns
|
|
columnIndex = Math.min( c.columns, data.rowArray.length );
|
|
while ( !showRow && columnIndex > 0 ) {
|
|
columnIndex--;
|
|
showRow = showRow || data.rowArray[ columnIndex ].indexOf( data.iFilter ) === 0;
|
|
}
|
|
} else {
|
|
showRow = ( data.iExact + data.childRowText ).indexOf( data.iFilter ) >= 0;
|
|
}
|
|
}
|
|
data.anyMatch = false;
|
|
// no other filters to process
|
|
if ( data.filters.join( '' ) === data.filter ) {
|
|
return showRow;
|
|
}
|
|
}
|
|
|
|
for ( columnIndex = 0; columnIndex < c.columns; columnIndex++ ) {
|
|
data.filter = data.filters[ columnIndex ];
|
|
data.index = columnIndex;
|
|
|
|
// filter types to exclude, per column
|
|
vars.excludeMatch = vars.excludeFilter[ columnIndex ];
|
|
|
|
// ignore if filter is empty or disabled
|
|
if ( data.filter ) {
|
|
data.cache = data.cacheArray[ columnIndex ];
|
|
result = data.parsed[ columnIndex ] ? data.cache : data.rawArray[ columnIndex ] || '';
|
|
data.exact = c.sortLocaleCompare ? ts.replaceAccents( result ) : result; // issue #405
|
|
data.iExact = !tsfRegex.type.test( typeof data.exact ) && wo.filter_ignoreCase ?
|
|
data.exact.toLowerCase() : data.exact;
|
|
data.isMatch = tsf.matchType( c, columnIndex );
|
|
|
|
result = showRow; // if showRow is true, show that row
|
|
|
|
// in case select filter option has a different value vs text 'a - z|A through Z'
|
|
ffxn = wo.filter_columnFilters ?
|
|
c.$filters.add( wo.filter_$externalFilters )
|
|
.filter( '[data-column="' + columnIndex + '"]' )
|
|
.find( 'select option:selected' )
|
|
.attr( 'data-function-name' ) || '' : '';
|
|
// replace accents - see #357
|
|
if ( c.sortLocaleCompare ) {
|
|
data.filter = ts.replaceAccents( data.filter );
|
|
}
|
|
|
|
// replace column specific default filters - see #1088
|
|
if ( wo.filter_defaultFilter && tsfRegex.iQuery.test( vars.defaultColFilter[ columnIndex ] ) ) {
|
|
data.filter = tsf.defaultFilter( data.filter, vars.defaultColFilter[ columnIndex ] );
|
|
}
|
|
|
|
// data.iFilter = case insensitive ( if wo.filter_ignoreCase is true ),
|
|
// data.filter = case sensitive
|
|
data.iFilter = wo.filter_ignoreCase ? ( data.filter || '' ).toLowerCase() : data.filter;
|
|
fxn = vars.functions[ columnIndex ];
|
|
filterMatched = null;
|
|
if ( fxn ) {
|
|
if ( fxn === true ) {
|
|
// default selector uses exact match unless 'filter-match' class is found
|
|
filterMatched = data.isMatch ?
|
|
// data.iExact may be a number
|
|
( '' + data.iExact ).search( data.iFilter ) >= 0 :
|
|
data.filter === data.exact;
|
|
} else if ( typeof fxn === 'function' ) {
|
|
// filter callback( exact cell content, parser normalized content,
|
|
// filter input value, column index, jQuery row object )
|
|
filterMatched = fxn( data.exact, data.cache, data.filter, columnIndex, data.$row, c, data );
|
|
} else if ( typeof fxn[ ffxn || data.filter ] === 'function' ) {
|
|
// selector option function
|
|
txt = ffxn || data.filter;
|
|
filterMatched =
|
|
fxn[ txt ]( data.exact, data.cache, data.filter, columnIndex, data.$row, c, data );
|
|
}
|
|
}
|
|
if ( filterMatched === null ) {
|
|
// cycle through the different filters
|
|
// filters return a boolean or null if nothing matches
|
|
filterMatched = tsf.processTypes( c, data, vars );
|
|
if ( filterMatched !== null ) {
|
|
result = filterMatched;
|
|
// Look for match, and add child row data for matching
|
|
} else {
|
|
txt = ( data.iExact + data.childRowText ).indexOf( tsf.parseFilter( c, data.iFilter, data ) );
|
|
result = ( ( !wo.filter_startsWith && txt >= 0 ) || ( wo.filter_startsWith && txt === 0 ) );
|
|
}
|
|
} else {
|
|
result = filterMatched;
|
|
}
|
|
showRow = ( result ) ? showRow : false;
|
|
}
|
|
}
|
|
return showRow;
|
|
},
|
|
findRows: function( table, filters, combinedFilters ) {
|
|
if ( table.config.lastCombinedFilter === combinedFilters ||
|
|
!table.config.widgetOptions.filter_initialized ) {
|
|
return;
|
|
}
|
|
var len, norm_rows, rowData, $rows, $row, rowIndex, tbodyIndex, $tbody, columnIndex,
|
|
isChild, childRow, lastSearch, showRow, showParent, time, val, indx,
|
|
notFiltered, searchFiltered, query, injected, res, id, txt,
|
|
storedFilters = $.extend( [], filters ),
|
|
c = table.config,
|
|
wo = c.widgetOptions,
|
|
// data object passed to filters; anyMatch is a flag for the filters
|
|
data = {
|
|
anyMatch: false,
|
|
filters: filters,
|
|
// regex filter type cache
|
|
filter_regexCache : []
|
|
},
|
|
vars = {
|
|
// anyMatch really screws up with these types of filters
|
|
noAnyMatch: [ 'range', 'notMatch', 'operators' ],
|
|
// cache filter variables that use ts.getColumnData in the main loop
|
|
functions : [],
|
|
excludeFilter : [],
|
|
defaultColFilter : [],
|
|
defaultAnyFilter : ts.getColumnData( table, wo.filter_defaultFilter, c.columns, true ) || ''
|
|
};
|
|
|
|
// parse columns after formatter, in case the class is added at that point
|
|
data.parsed = [];
|
|
for ( columnIndex = 0; columnIndex < c.columns; columnIndex++ ) {
|
|
data.parsed[ columnIndex ] = wo.filter_useParsedData ||
|
|
// parser has a "parsed" parameter
|
|
( c.parsers && c.parsers[ columnIndex ] && c.parsers[ columnIndex ].parsed ||
|
|
// getData may not return 'parsed' if other 'filter-' class names exist
|
|
// ( e.g. <th class="filter-select filter-parsed"> )
|
|
ts.getData && ts.getData( c.$headerIndexed[ columnIndex ],
|
|
ts.getColumnData( table, c.headers, columnIndex ), 'filter' ) === 'parsed' ||
|
|
c.$headerIndexed[ columnIndex ].hasClass( 'filter-parsed' ) );
|
|
|
|
vars.functions[ columnIndex ] =
|
|
ts.getColumnData( table, wo.filter_functions, columnIndex ) ||
|
|
c.$headerIndexed[ columnIndex ].hasClass( 'filter-select' );
|
|
vars.defaultColFilter[ columnIndex ] =
|
|
ts.getColumnData( table, wo.filter_defaultFilter, columnIndex ) || '';
|
|
vars.excludeFilter[ columnIndex ] =
|
|
( ts.getColumnData( table, wo.filter_excludeFilter, columnIndex, true ) || '' ).split( /\s+/ );
|
|
}
|
|
|
|
if ( c.debug ) {
|
|
console.log( 'Filter: Starting filter widget search', filters );
|
|
time = new Date();
|
|
}
|
|
// filtered rows count
|
|
c.filteredRows = 0;
|
|
c.totalRows = 0;
|
|
// combindedFilters are undefined on init
|
|
combinedFilters = ( storedFilters || [] ).join( '' );
|
|
|
|
for ( tbodyIndex = 0; tbodyIndex < c.$tbodies.length; tbodyIndex++ ) {
|
|
$tbody = ts.processTbody( table, c.$tbodies.eq( tbodyIndex ), true );
|
|
// skip child rows & widget added ( removable ) rows - fixes #448 thanks to @hempel!
|
|
// $rows = $tbody.children( 'tr' ).not( c.selectorRemove );
|
|
columnIndex = c.columns;
|
|
// convert stored rows into a jQuery object
|
|
norm_rows = c.cache[ tbodyIndex ].normalized;
|
|
$rows = $( $.map( norm_rows, function( el ) {
|
|
return el[ columnIndex ].$row.get();
|
|
}) );
|
|
|
|
if ( combinedFilters === '' || wo.filter_serversideFiltering ) {
|
|
$rows
|
|
.removeClass( wo.filter_filteredRow )
|
|
.not( '.' + c.cssChildRow )
|
|
.css( 'display', '' );
|
|
} else {
|
|
// filter out child rows
|
|
$rows = $rows.not( '.' + c.cssChildRow );
|
|
len = $rows.length;
|
|
|
|
if ( ( wo.filter_$anyMatch && wo.filter_$anyMatch.length ) ||
|
|
typeof filters[c.columns] !== 'undefined' ) {
|
|
data.anyMatchFlag = true;
|
|
data.anyMatchFilter = '' + (
|
|
filters[ c.columns ] ||
|
|
wo.filter_$anyMatch && tsf.getLatestSearch( wo.filter_$anyMatch ).val() ||
|
|
''
|
|
);
|
|
if ( wo.filter_columnAnyMatch ) {
|
|
// specific columns search
|
|
query = data.anyMatchFilter.split( tsfRegex.andSplit );
|
|
injected = false;
|
|
for ( indx = 0; indx < query.length; indx++ ) {
|
|
res = query[ indx ].split( ':' );
|
|
if ( res.length > 1 ) {
|
|
// make the column a one-based index ( non-developers start counting from one :P )
|
|
id = parseInt( res[0], 10 ) - 1;
|
|
if ( id >= 0 && id < c.columns ) { // if id is an integer
|
|
filters[ id ] = res[1];
|
|
query.splice( indx, 1 );
|
|
indx--;
|
|
injected = true;
|
|
}
|
|
}
|
|
}
|
|
if ( injected ) {
|
|
data.anyMatchFilter = query.join( ' && ' );
|
|
}
|
|
}
|
|
}
|
|
|
|
// optimize searching only through already filtered rows - see #313
|
|
searchFiltered = wo.filter_searchFiltered;
|
|
lastSearch = c.lastSearch || c.$table.data( 'lastSearch' ) || [];
|
|
if ( searchFiltered ) {
|
|
// cycle through all filters; include last ( columnIndex + 1 = match any column ). Fixes #669
|
|
for ( indx = 0; indx < columnIndex + 1; indx++ ) {
|
|
val = filters[indx] || '';
|
|
// break out of loop if we've already determined not to search filtered rows
|
|
if ( !searchFiltered ) { indx = columnIndex; }
|
|
// search already filtered rows if...
|
|
searchFiltered = searchFiltered && lastSearch.length &&
|
|
// there are no changes from beginning of filter
|
|
val.indexOf( lastSearch[indx] || '' ) === 0 &&
|
|
// if there is NOT a logical 'or', or range ( 'to' or '-' ) in the string
|
|
!tsfRegex.alreadyFiltered.test( val ) &&
|
|
// if we are not doing exact matches, using '|' ( logical or ) or not '!'
|
|
!tsfRegex.exactTest.test( val ) &&
|
|
// don't search only filtered if the value is negative
|
|
// ( '> -10' => '> -100' will ignore hidden rows )
|
|
!( tsfRegex.isNeg1.test( val ) || tsfRegex.isNeg2.test( val ) ) &&
|
|
// if filtering using a select without a 'filter-match' class ( exact match ) - fixes #593
|
|
!( val !== '' && c.$filters && c.$filters.filter( '[data-column="' + indx + '"]' ).find( 'select' ).length &&
|
|
!tsf.matchType( c, indx ) );
|
|
}
|
|
}
|
|
notFiltered = $rows.not( '.' + wo.filter_filteredRow ).length;
|
|
// can't search when all rows are hidden - this happens when looking for exact matches
|
|
if ( searchFiltered && notFiltered === 0 ) { searchFiltered = false; }
|
|
if ( c.debug ) {
|
|
console.log( 'Filter: Searching through ' +
|
|
( searchFiltered && notFiltered < len ? notFiltered : 'all' ) + ' rows' );
|
|
}
|
|
if ( data.anyMatchFlag ) {
|
|
if ( c.sortLocaleCompare ) {
|
|
// replace accents
|
|
data.anyMatchFilter = ts.replaceAccents( data.anyMatchFilter );
|
|
}
|
|
if ( wo.filter_defaultFilter && tsfRegex.iQuery.test( vars.defaultAnyFilter ) ) {
|
|
data.anyMatchFilter = tsf.defaultFilter( data.anyMatchFilter, vars.defaultAnyFilter );
|
|
// clear search filtered flag because default filters are not saved to the last search
|
|
searchFiltered = false;
|
|
}
|
|
// make iAnyMatchFilter lowercase unless both filter widget & core ignoreCase options are true
|
|
// when c.ignoreCase is true, the cache contains all lower case data
|
|
data.iAnyMatchFilter = !( wo.filter_ignoreCase && c.ignoreCase ) ?
|
|
data.anyMatchFilter :
|
|
data.anyMatchFilter.toLowerCase();
|
|
}
|
|
|
|
// loop through the rows
|
|
for ( rowIndex = 0; rowIndex < len; rowIndex++ ) {
|
|
|
|
txt = $rows[ rowIndex ].className;
|
|
// the first row can never be a child row
|
|
isChild = rowIndex && tsfRegex.child.test( txt );
|
|
// skip child rows & already filtered rows
|
|
if ( isChild || ( searchFiltered && tsfRegex.filtered.test( txt ) ) ) {
|
|
continue;
|
|
}
|
|
|
|
data.$row = $rows.eq( rowIndex );
|
|
data.cacheArray = norm_rows[ rowIndex ];
|
|
rowData = data.cacheArray[ c.columns ];
|
|
data.rawArray = rowData.raw;
|
|
data.childRowText = '';
|
|
|
|
if ( !wo.filter_childByColumn ) {
|
|
txt = '';
|
|
// child row cached text
|
|
childRow = rowData.child;
|
|
// so, if 'table.config.widgetOptions.filter_childRows' is true and there is
|
|
// a match anywhere in the child row, then it will make the row visible
|
|
// checked here so the option can be changed dynamically
|
|
for ( indx = 0; indx < childRow.length; indx++ ) {
|
|
txt += ' ' + childRow[indx].join( ' ' ) || '';
|
|
}
|
|
data.childRowText = wo.filter_childRows ?
|
|
( wo.filter_ignoreCase ? txt.toLowerCase() : txt ) :
|
|
'';
|
|
}
|
|
|
|
showRow = false;
|
|
showParent = tsf.processRow( c, data, vars );
|
|
$row = rowData.$row;
|
|
|
|
// don't pass reference to val
|
|
val = showParent ? true : false;
|
|
childRow = rowData.$row.filter( ':gt(0)' );
|
|
if ( wo.filter_childRows && childRow.length ) {
|
|
if ( wo.filter_childByColumn ) {
|
|
if ( !wo.filter_childWithSibs ) {
|
|
// hide all child rows
|
|
childRow.addClass( wo.filter_filteredRow );
|
|
// if only showing resulting child row, only include parent
|
|
$row = $row.eq( 0 );
|
|
}
|
|
// cycle through each child row
|
|
for ( indx = 0; indx < childRow.length; indx++ ) {
|
|
data.$row = childRow.eq( indx );
|
|
data.cacheArray = rowData.child[ indx ];
|
|
data.rawArray = data.cacheArray;
|
|
val = tsf.processRow( c, data, vars );
|
|
// use OR comparison on child rows
|
|
showRow = showRow || val;
|
|
if ( !wo.filter_childWithSibs && val ) {
|
|
childRow.eq( indx ).removeClass( wo.filter_filteredRow );
|
|
}
|
|
}
|
|
}
|
|
// keep parent row match even if no child matches... see #1020
|
|
showRow = showRow || showParent;
|
|
} else {
|
|
showRow = val;
|
|
}
|
|
$row
|
|
.toggleClass( wo.filter_filteredRow, !showRow )[0]
|
|
.display = showRow ? '' : 'none';
|
|
}
|
|
}
|
|
c.filteredRows += $rows.not( '.' + wo.filter_filteredRow ).length;
|
|
c.totalRows += $rows.length;
|
|
ts.processTbody( table, $tbody, false );
|
|
}
|
|
c.lastCombinedFilter = combinedFilters; // save last search
|
|
// don't save 'filters' directly since it may have altered ( AnyMatch column searches )
|
|
c.lastSearch = storedFilters;
|
|
c.$table.data( 'lastSearch', storedFilters );
|
|
if ( wo.filter_saveFilters && ts.storage ) {
|
|
ts.storage( table, 'tablesorter-filters', tsf.processFilters( storedFilters, true ) );
|
|
}
|
|
if ( c.debug ) {
|
|
console.log( 'Completed filter widget search' + ts.benchmark(time) );
|
|
}
|
|
if ( wo.filter_initialized ) {
|
|
c.$table.triggerHandler( 'filterBeforeEnd', c );
|
|
c.$table.triggerHandler( 'filterEnd', c );
|
|
}
|
|
setTimeout( function() {
|
|
ts.applyWidget( c.table ); // make sure zebra widget is applied
|
|
}, 0 );
|
|
},
|
|
getOptionSource: function( table, column, onlyAvail ) {
|
|
table = $( table )[0];
|
|
var c = table.config,
|
|
wo = c.widgetOptions,
|
|
arry = false,
|
|
source = wo.filter_selectSource,
|
|
last = c.$table.data( 'lastSearch' ) || [],
|
|
fxn = typeof source === 'function' ? true : ts.getColumnData( table, source, column );
|
|
|
|
if ( onlyAvail && last[column] !== '' ) {
|
|
onlyAvail = false;
|
|
}
|
|
|
|
// filter select source option
|
|
if ( fxn === true ) {
|
|
// OVERALL source
|
|
arry = source( table, column, onlyAvail );
|
|
} else if ( fxn instanceof $ || ( $.type( fxn ) === 'string' && fxn.indexOf( '</option>' ) >= 0 ) ) {
|
|
// selectSource is a jQuery object or string of options
|
|
return fxn;
|
|
} else if ( $.isArray( fxn ) ) {
|
|
arry = fxn;
|
|
} else if ( $.type( source ) === 'object' && fxn ) {
|
|
// custom select source function for a SPECIFIC COLUMN
|
|
arry = fxn( table, column, onlyAvail );
|
|
}
|
|
if ( arry === false ) {
|
|
// fall back to original method
|
|
arry = tsf.getOptions( table, column, onlyAvail );
|
|
}
|
|
|
|
return tsf.processOptions( table, column, arry );
|
|
|
|
},
|
|
processOptions: function( table, column, arry ) {
|
|
if ( !$.isArray( arry ) ) {
|
|
return false;
|
|
}
|
|
table = $( table )[0];
|
|
var cts, txt, indx, len, parsedTxt, str,
|
|
c = table.config,
|
|
validColumn = typeof column !== 'undefined' && column !== null && column >= 0 && column < c.columns,
|
|
parsed = [];
|
|
// get unique elements and sort the list
|
|
// if $.tablesorter.sortText exists ( not in the original tablesorter ),
|
|
// then natural sort the list otherwise use a basic sort
|
|
arry = $.grep( arry, function( value, indx ) {
|
|
if ( value.text ) {
|
|
return true;
|
|
}
|
|
return $.inArray( value, arry ) === indx;
|
|
});
|
|
if ( validColumn && c.$headerIndexed[ column ].hasClass( 'filter-select-nosort' ) ) {
|
|
// unsorted select options
|
|
return arry;
|
|
} else {
|
|
len = arry.length;
|
|
// parse select option values
|
|
for ( indx = 0; indx < len; indx++ ) {
|
|
txt = arry[ indx ];
|
|
// check for object
|
|
str = txt.text ? txt.text : txt;
|
|
// sortNatural breaks if you don't pass it strings
|
|
parsedTxt = ( validColumn && c.parsers && c.parsers.length &&
|
|
c.parsers[ column ].format( str, table, [], column ) || str ).toString();
|
|
parsedTxt = c.widgetOptions.filter_ignoreCase ? parsedTxt.toLowerCase() : parsedTxt;
|
|
// parse array data using set column parser; this DOES NOT pass the original
|
|
// table cell to the parser format function
|
|
if ( txt.text ) {
|
|
txt.parsed = parsedTxt;
|
|
parsed[ parsed.length ] = txt;
|
|
} else {
|
|
parsed[ parsed.length ] = {
|
|
text : txt,
|
|
// check parser length - fixes #934
|
|
parsed : parsedTxt
|
|
};
|
|
}
|
|
}
|
|
// sort parsed select options
|
|
cts = c.textSorter || '';
|
|
parsed.sort( function( a, b ) {
|
|
var x = a.parsed,
|
|
y = b.parsed;
|
|
if ( validColumn && typeof cts === 'function' ) {
|
|
// custom OVERALL text sorter
|
|
return cts( x, y, true, column, table );
|
|
} else if ( validColumn && typeof cts === 'object' && cts.hasOwnProperty( column ) ) {
|
|
// custom text sorter for a SPECIFIC COLUMN
|
|
return cts[column]( x, y, true, column, table );
|
|
} else if ( ts.sortNatural ) {
|
|
// fall back to natural sort
|
|
return ts.sortNatural( x, y );
|
|
}
|
|
// using an older version! do a basic sort
|
|
return true;
|
|
});
|
|
// rebuild arry from sorted parsed data
|
|
arry = [];
|
|
len = parsed.length;
|
|
for ( indx = 0; indx < len; indx++ ) {
|
|
arry[ arry.length ] = parsed[indx];
|
|
}
|
|
return arry;
|
|
}
|
|
},
|
|
getOptions: function( table, column, onlyAvail ) {
|
|
table = $( table )[0];
|
|
var rowIndex, tbodyIndex, len, row, cache, indx, child, childLen,
|
|
c = table.config,
|
|
wo = c.widgetOptions,
|
|
arry = [];
|
|
for ( tbodyIndex = 0; tbodyIndex < c.$tbodies.length; tbodyIndex++ ) {
|
|
cache = c.cache[tbodyIndex];
|
|
len = c.cache[tbodyIndex].normalized.length;
|
|
// loop through the rows
|
|
for ( rowIndex = 0; rowIndex < len; rowIndex++ ) {
|
|
// get cached row from cache.row ( old ) or row data object
|
|
// ( new; last item in normalized array )
|
|
row = cache.row ?
|
|
cache.row[ rowIndex ] :
|
|
cache.normalized[ rowIndex ][ c.columns ].$row[0];
|
|
// check if has class filtered
|
|
if ( onlyAvail && row.className.match( wo.filter_filteredRow ) ) {
|
|
continue;
|
|
}
|
|
// get non-normalized cell content
|
|
if ( wo.filter_useParsedData ||
|
|
c.parsers[column].parsed ||
|
|
c.$headerIndexed[column].hasClass( 'filter-parsed' ) ) {
|
|
arry[ arry.length ] = '' + cache.normalized[ rowIndex ][ column ];
|
|
// child row parsed data
|
|
if ( wo.filter_childRows && wo.filter_childByColumn ) {
|
|
childLen = cache.normalized[ rowIndex ][ c.columns ].$row.length - 1;
|
|
for ( indx = 0; indx < childLen; indx++ ) {
|
|
arry[ arry.length ] = '' + cache.normalized[ rowIndex ][ c.columns ].child[ indx ][ column ];
|
|
}
|
|
}
|
|
} else {
|
|
// get raw cached data instead of content directly from the cells
|
|
arry[ arry.length ] = cache.normalized[ rowIndex ][ c.columns ].raw[ column ];
|
|
// child row unparsed data
|
|
if ( wo.filter_childRows && wo.filter_childByColumn ) {
|
|
childLen = cache.normalized[ rowIndex ][ c.columns ].$row.length;
|
|
for ( indx = 1; indx < childLen; indx++ ) {
|
|
child = cache.normalized[ rowIndex ][ c.columns ].$row.eq( indx ).children().eq( column );
|
|
arry[ arry.length ] = '' + ts.getElementText( c, child, column );
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return arry;
|
|
},
|
|
buildSelect: function( table, column, arry, updating, onlyAvail ) {
|
|
table = $( table )[0];
|
|
column = parseInt( column, 10 );
|
|
if ( !table.config.cache || $.isEmptyObject( table.config.cache ) ) {
|
|
return;
|
|
}
|
|
|
|
var indx, val, txt, t, $filters, $filter, option,
|
|
c = table.config,
|
|
wo = c.widgetOptions,
|
|
node = c.$headerIndexed[ column ],
|
|
// t.data( 'placeholder' ) won't work in jQuery older than 1.4.3
|
|
options = '<option value="">' +
|
|
( node.data( 'placeholder' ) ||
|
|
node.attr( 'data-placeholder' ) ||
|
|
wo.filter_placeholder.select || ''
|
|
) + '</option>',
|
|
// Get curent filter value
|
|
currentValue = c.$table
|
|
.find( 'thead' )
|
|
.find( 'select.' + tscss.filter + '[data-column="' + column + '"]' )
|
|
.val();
|
|
|
|
// nothing included in arry ( external source ), so get the options from
|
|
// filter_selectSource or column data
|
|
if ( typeof arry === 'undefined' || arry === '' ) {
|
|
arry = tsf.getOptionSource( table, column, onlyAvail );
|
|
}
|
|
|
|
if ( $.isArray( arry ) ) {
|
|
// build option list
|
|
for ( indx = 0; indx < arry.length; indx++ ) {
|
|
option = arry[ indx ];
|
|
if ( option.text ) {
|
|
// OBJECT!! add data-function-name in case the value is set in filter_functions
|
|
option['data-function-name'] = typeof option.value === 'undefined' ? option.text : option.value;
|
|
|
|
// support jQuery < v1.8, otherwise the below code could be shortened to
|
|
// options += $( '<option>', option )[ 0 ].outerHTML;
|
|
options += '<option';
|
|
for ( val in option ) {
|
|
if ( option.hasOwnProperty( val ) && val !== 'text' ) {
|
|
options += ' ' + val + '="' + option[ val ] + '"';
|
|
}
|
|
}
|
|
if ( !option.value ) {
|
|
options += ' value="' + option.text + '"';
|
|
}
|
|
options += '>' + option.text + '</option>';
|
|
// above code is needed in jQuery < v1.8
|
|
|
|
// make sure we don't turn an object into a string (objects without a "text" property)
|
|
} else if ( '' + option !== '[object Object]' ) {
|
|
txt = option = ( '' + option ).replace( tsfRegex.quote, '"' );
|
|
val = txt;
|
|
// allow including a symbol in the selectSource array
|
|
// 'a-z|A through Z' so that 'a-z' becomes the option value
|
|
// and 'A through Z' becomes the option text
|
|
if ( txt.indexOf( wo.filter_selectSourceSeparator ) >= 0 ) {
|
|
t = txt.split( wo.filter_selectSourceSeparator );
|
|
val = t[0];
|
|
txt = t[1];
|
|
}
|
|
// replace quotes - fixes #242 & ignore empty strings
|
|
// see http://stackoverflow.com/q/14990971/145346
|
|
options += option !== '' ?
|
|
'<option ' +
|
|
( val === txt ? '' : 'data-function-name="' + option + '" ' ) +
|
|
'value="' + val + '">' + txt +
|
|
'</option>' : '';
|
|
}
|
|
}
|
|
// clear arry so it doesn't get appended twice
|
|
arry = [];
|
|
}
|
|
|
|
// update all selects in the same column ( clone thead in sticky headers &
|
|
// any external selects ) - fixes 473
|
|
$filters = ( c.$filters ? c.$filters : c.$table.children( 'thead' ) )
|
|
.find( '.' + tscss.filter );
|
|
if ( wo.filter_$externalFilters ) {
|
|
$filters = $filters && $filters.length ?
|
|
$filters.add( wo.filter_$externalFilters ) :
|
|
wo.filter_$externalFilters;
|
|
}
|
|
$filter = $filters.filter( 'select[data-column="' + column + '"]' );
|
|
|
|
// make sure there is a select there!
|
|
if ( $filter.length ) {
|
|
$filter[ updating ? 'html' : 'append' ]( options );
|
|
if ( !$.isArray( arry ) ) {
|
|
// append options if arry is provided externally as a string or jQuery object
|
|
// options ( default value ) was already added
|
|
$filter.append( arry ).val( currentValue );
|
|
}
|
|
$filter.val( currentValue );
|
|
}
|
|
},
|
|
buildDefault: function( table, updating ) {
|
|
var columnIndex, $header, noSelect,
|
|
c = table.config,
|
|
wo = c.widgetOptions,
|
|
columns = c.columns;
|
|
// build default select dropdown
|
|
for ( columnIndex = 0; columnIndex < columns; columnIndex++ ) {
|
|
$header = c.$headerIndexed[columnIndex];
|
|
noSelect = !( $header.hasClass( 'filter-false' ) || $header.hasClass( 'parser-false' ) );
|
|
// look for the filter-select class; build/update it if found
|
|
if ( ( $header.hasClass( 'filter-select' ) ||
|
|
ts.getColumnData( table, wo.filter_functions, columnIndex ) === true ) && noSelect ) {
|
|
tsf.buildSelect( table, columnIndex, '', updating, $header.hasClass( wo.filter_onlyAvail ) );
|
|
}
|
|
}
|
|
}
|
|
};
|
|
|
|
// filter regex variable
|
|
tsfRegex = tsf.regex;
|
|
|
|
ts.getFilters = function( table, getRaw, setFilters, skipFirst ) {
|
|
var i, $filters, $column, cols,
|
|
filters = false,
|
|
c = table ? $( table )[0].config : '',
|
|
wo = c ? c.widgetOptions : '';
|
|
if ( ( getRaw !== true && wo && !wo.filter_columnFilters ) ||
|
|
// setFilters called, but last search is exactly the same as the current
|
|
// fixes issue #733 & #903 where calling update causes the input values to reset
|
|
( $.isArray(setFilters) && setFilters.join('') === c.lastCombinedFilter ) ) {
|
|
return $( table ).data( 'lastSearch' );
|
|
}
|
|
if ( c ) {
|
|
if ( c.$filters ) {
|
|
$filters = c.$filters.find( '.' + tscss.filter );
|
|
}
|
|
if ( wo.filter_$externalFilters ) {
|
|
$filters = $filters && $filters.length ?
|
|
$filters.add( wo.filter_$externalFilters ) :
|
|
wo.filter_$externalFilters;
|
|
}
|
|
if ( $filters && $filters.length ) {
|
|
filters = setFilters || [];
|
|
for ( i = 0; i < c.columns + 1; i++ ) {
|
|
cols = ( i === c.columns ?
|
|
// 'all' columns can now include a range or set of columms ( data-column='0-2,4,6-7' )
|
|
wo.filter_anyColumnSelector + ',' + wo.filter_multipleColumnSelector :
|
|
'[data-column="' + i + '"]' );
|
|
$column = $filters.filter( cols );
|
|
if ( $column.length ) {
|
|
// move the latest search to the first slot in the array
|
|
$column = tsf.getLatestSearch( $column );
|
|
if ( $.isArray( setFilters ) ) {
|
|
// skip first ( latest input ) to maintain cursor position while typing
|
|
if ( skipFirst && $column.length > 1 ) {
|
|
$column = $column.slice( 1 );
|
|
}
|
|
if ( i === c.columns ) {
|
|
// prevent data-column='all' from filling data-column='0,1' ( etc )
|
|
cols = $column.filter( wo.filter_anyColumnSelector );
|
|
$column = cols.length ? cols : $column;
|
|
}
|
|
$column
|
|
.val( setFilters[ i ] )
|
|
// must include a namespace here; but not c.namespace + 'filter'?
|
|
.trigger( 'change' + c.namespace );
|
|
} else {
|
|
filters[i] = $column.val() || '';
|
|
// don't change the first... it will move the cursor
|
|
if ( i === c.columns ) {
|
|
// don't update range columns from 'all' setting
|
|
$column
|
|
.slice( 1 )
|
|
.filter( '[data-column*="' + $column.attr( 'data-column' ) + '"]' )
|
|
.val( filters[ i ] );
|
|
} else {
|
|
$column
|
|
.slice( 1 )
|
|
.val( filters[ i ] );
|
|
}
|
|
}
|
|
// save any match input dynamically
|
|
if ( i === c.columns && $column.length ) {
|
|
wo.filter_$anyMatch = $column;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
if ( filters.length === 0 ) {
|
|
filters = false;
|
|
}
|
|
return filters;
|
|
};
|
|
|
|
ts.setFilters = function( table, filter, apply, skipFirst ) {
|
|
var c = table ? $( table )[0].config : '',
|
|
valid = ts.getFilters( table, true, filter, skipFirst );
|
|
// default apply to "true"
|
|
if ( typeof apply === 'undefined' ) {
|
|
apply = true;
|
|
}
|
|
if ( c && apply ) {
|
|
// ensure new set filters are applied, even if the search is the same
|
|
c.lastCombinedFilter = null;
|
|
c.lastSearch = [];
|
|
tsf.searching( c.table, filter, skipFirst );
|
|
c.$table.triggerHandler( 'filterFomatterUpdate' );
|
|
}
|
|
return !!valid;
|
|
};
|
|
|
|
})( jQuery );
|
|
|
|
/*! Widget: stickyHeaders - updated 4/1/2016 (v2.25.7) *//*
|
|
* Requires tablesorter v2.8+ and jQuery 1.4.3+
|
|
* by Rob Garrison
|
|
*/
|
|
;(function ($, window) {
|
|
'use strict';
|
|
var ts = $.tablesorter || {};
|
|
|
|
$.extend(ts.css, {
|
|
sticky : 'tablesorter-stickyHeader', // stickyHeader
|
|
stickyVis : 'tablesorter-sticky-visible',
|
|
stickyHide: 'tablesorter-sticky-hidden',
|
|
stickyWrap: 'tablesorter-sticky-wrapper'
|
|
});
|
|
|
|
// Add a resize event to table headers
|
|
ts.addHeaderResizeEvent = function(table, disable, settings) {
|
|
table = $(table)[0]; // make sure we're using a dom element
|
|
if ( !table.config ) { return; }
|
|
var defaults = {
|
|
timer : 250
|
|
},
|
|
options = $.extend({}, defaults, settings),
|
|
c = table.config,
|
|
wo = c.widgetOptions,
|
|
checkSizes = function( triggerEvent ) {
|
|
var index, headers, $header, sizes, width, height,
|
|
len = c.$headers.length;
|
|
wo.resize_flag = true;
|
|
headers = [];
|
|
for ( index = 0; index < len; index++ ) {
|
|
$header = c.$headers.eq( index );
|
|
sizes = $header.data( 'savedSizes' ) || [ 0, 0 ]; // fixes #394
|
|
width = $header[0].offsetWidth;
|
|
height = $header[0].offsetHeight;
|
|
if ( width !== sizes[0] || height !== sizes[1] ) {
|
|
$header.data( 'savedSizes', [ width, height ] );
|
|
headers.push( $header[0] );
|
|
}
|
|
}
|
|
if ( headers.length && triggerEvent !== false ) {
|
|
c.$table.triggerHandler( 'resize', [ headers ] );
|
|
}
|
|
wo.resize_flag = false;
|
|
};
|
|
clearInterval(wo.resize_timer);
|
|
if (disable) {
|
|
wo.resize_flag = false;
|
|
return false;
|
|
}
|
|
checkSizes( false );
|
|
wo.resize_timer = setInterval(function() {
|
|
if (wo.resize_flag) { return; }
|
|
checkSizes();
|
|
}, options.timer);
|
|
};
|
|
|
|
// Sticky headers based on this awesome article:
|
|
// http://css-tricks.com/13465-persistent-headers/
|
|
// and https://github.com/jmosbech/StickyTableHeaders by Jonas Mosbech
|
|
// **************************
|
|
ts.addWidget({
|
|
id: 'stickyHeaders',
|
|
priority: 60, // sticky widget must be initialized after the filter widget!
|
|
options: {
|
|
stickyHeaders : '', // extra class name added to the sticky header row
|
|
stickyHeaders_attachTo : null, // jQuery selector or object to attach sticky header to
|
|
stickyHeaders_xScroll : null, // jQuery selector or object to monitor horizontal scroll position (defaults: xScroll > attachTo > window)
|
|
stickyHeaders_yScroll : null, // jQuery selector or object to monitor vertical scroll position (defaults: yScroll > attachTo > window)
|
|
stickyHeaders_offset : 0, // number or jquery selector targeting the position:fixed element
|
|
stickyHeaders_filteredToTop: true, // scroll table top into view after filtering
|
|
stickyHeaders_cloneId : '-sticky', // added to table ID, if it exists
|
|
stickyHeaders_addResizeEvent : true, // trigger 'resize' event on headers
|
|
stickyHeaders_includeCaption : true, // if false and a caption exist, it won't be included in the sticky header
|
|
stickyHeaders_zIndex : 2 // The zIndex of the stickyHeaders, allows the user to adjust this to their needs
|
|
},
|
|
format: function(table, c, wo) {
|
|
// filter widget doesn't initialize on an empty table. Fixes #449
|
|
if ( c.$table.hasClass('hasStickyHeaders') || ($.inArray('filter', c.widgets) >= 0 && !c.$table.hasClass('hasFilters')) ) {
|
|
return;
|
|
}
|
|
var index, len, $t,
|
|
$table = c.$table,
|
|
// add position: relative to attach element, hopefully it won't cause trouble.
|
|
$attach = $(wo.stickyHeaders_attachTo),
|
|
namespace = c.namespace + 'stickyheaders ',
|
|
// element to watch for the scroll event
|
|
$yScroll = $(wo.stickyHeaders_yScroll || wo.stickyHeaders_attachTo || window),
|
|
$xScroll = $(wo.stickyHeaders_xScroll || wo.stickyHeaders_attachTo || window),
|
|
$thead = $table.children('thead:first'),
|
|
$header = $thead.children('tr').not('.sticky-false').children(),
|
|
$tfoot = $table.children('tfoot'),
|
|
$stickyOffset = isNaN(wo.stickyHeaders_offset) ? $(wo.stickyHeaders_offset) : '',
|
|
stickyOffset = $stickyOffset.length ? $stickyOffset.height() || 0 : parseInt(wo.stickyHeaders_offset, 10) || 0,
|
|
// is this table nested? If so, find parent sticky header wrapper (div, not table)
|
|
$nestedSticky = $table.parent().closest('.' + ts.css.table).hasClass('hasStickyHeaders') ?
|
|
$table.parent().closest('table.tablesorter')[0].config.widgetOptions.$sticky.parent() : [],
|
|
nestedStickyTop = $nestedSticky.length ? $nestedSticky.height() : 0,
|
|
// clone table, then wrap to make sticky header
|
|
$stickyTable = wo.$sticky = $table.clone()
|
|
.addClass('containsStickyHeaders ' + ts.css.sticky + ' ' + wo.stickyHeaders + ' ' + c.namespace.slice(1) + '_extra_table' )
|
|
.wrap('<div class="' + ts.css.stickyWrap + '">'),
|
|
$stickyWrap = $stickyTable.parent()
|
|
.addClass(ts.css.stickyHide)
|
|
.css({
|
|
position : $attach.length ? 'absolute' : 'fixed',
|
|
padding : parseInt( $stickyTable.parent().parent().css('padding-left'), 10 ),
|
|
top : stickyOffset + nestedStickyTop,
|
|
left : 0,
|
|
visibility : 'hidden',
|
|
zIndex : wo.stickyHeaders_zIndex || 2
|
|
}),
|
|
$stickyThead = $stickyTable.children('thead:first'),
|
|
$stickyCells,
|
|
laststate = '',
|
|
spacing = 0,
|
|
setWidth = function($orig, $clone){
|
|
var index, width, border, $cell, $this,
|
|
$cells = $orig.filter(':visible'),
|
|
len = $cells.length;
|
|
for ( index = 0; index < len; index++ ) {
|
|
$cell = $clone.filter(':visible').eq(index);
|
|
$this = $cells.eq(index);
|
|
// code from https://github.com/jmosbech/StickyTableHeaders
|
|
if ($this.css('box-sizing') === 'border-box') {
|
|
width = $this.outerWidth();
|
|
} else {
|
|
if ($cell.css('border-collapse') === 'collapse') {
|
|
if (window.getComputedStyle) {
|
|
width = parseFloat( window.getComputedStyle($this[0], null).width );
|
|
} else {
|
|
// ie8 only
|
|
border = parseFloat( $this.css('border-width') );
|
|
width = $this.outerWidth() - parseFloat( $this.css('padding-left') ) - parseFloat( $this.css('padding-right') ) - border;
|
|
}
|
|
} else {
|
|
width = $this.width();
|
|
}
|
|
}
|
|
$cell.css({
|
|
'width': width,
|
|
'min-width': width,
|
|
'max-width': width
|
|
});
|
|
}
|
|
},
|
|
resizeHeader = function() {
|
|
stickyOffset = $stickyOffset.length ? $stickyOffset.height() || 0 : parseInt(wo.stickyHeaders_offset, 10) || 0;
|
|
spacing = 0;
|
|
$stickyWrap.css({
|
|
left : $attach.length ? parseInt($attach.css('padding-left'), 10) || 0 :
|
|
$table.offset().left - parseInt($table.css('margin-left'), 10) - $xScroll.scrollLeft() - spacing,
|
|
width: $table.outerWidth()
|
|
});
|
|
setWidth( $table, $stickyTable );
|
|
setWidth( $header, $stickyCells );
|
|
},
|
|
scrollSticky = function( resizing ) {
|
|
if (!$table.is(':visible')) { return; } // fixes #278
|
|
// Detect nested tables - fixes #724
|
|
nestedStickyTop = $nestedSticky.length ? $nestedSticky.offset().top - $yScroll.scrollTop() + $nestedSticky.height() : 0;
|
|
var offset = $table.offset(),
|
|
yWindow = $.isWindow( $yScroll[0] ), // $.isWindow needs jQuery 1.4.3
|
|
xWindow = $.isWindow( $xScroll[0] ),
|
|
// scrollTop = ( $attach.length ? $attach.offset().top : $yScroll.scrollTop() ) + stickyOffset + nestedStickyTop,
|
|
scrollTop = ( $attach.length ? ( yWindow ? $yScroll.scrollTop() : $yScroll.offset().top ) : $yScroll.scrollTop() ) + stickyOffset + nestedStickyTop,
|
|
tableHeight = $table.height() - ($stickyWrap.height() + ($tfoot.height() || 0)),
|
|
isVisible = ( scrollTop > offset.top ) && ( scrollTop < offset.top + tableHeight ) ? 'visible' : 'hidden',
|
|
cssSettings = { visibility : isVisible };
|
|
|
|
if ($attach.length) {
|
|
cssSettings.top = yWindow ? scrollTop - $attach.offset().top : $attach.scrollTop();
|
|
}
|
|
if (xWindow) {
|
|
// adjust when scrolling horizontally - fixes issue #143
|
|
cssSettings.left = $table.offset().left - parseInt($table.css('margin-left'), 10) - $xScroll.scrollLeft() - spacing;
|
|
}
|
|
if ($nestedSticky.length) {
|
|
cssSettings.top = ( cssSettings.top || 0 ) + stickyOffset + nestedStickyTop;
|
|
}
|
|
$stickyWrap
|
|
.removeClass( ts.css.stickyVis + ' ' + ts.css.stickyHide )
|
|
.addClass( isVisible === 'visible' ? ts.css.stickyVis : ts.css.stickyHide )
|
|
.css(cssSettings);
|
|
if (isVisible !== laststate || resizing) {
|
|
// make sure the column widths match
|
|
resizeHeader();
|
|
laststate = isVisible;
|
|
}
|
|
};
|
|
// only add a position relative if a position isn't already defined
|
|
if ($attach.length && !$attach.css('position')) {
|
|
$attach.css('position', 'relative');
|
|
}
|
|
// fix clone ID, if it exists - fixes #271
|
|
if ($stickyTable.attr('id')) { $stickyTable[0].id += wo.stickyHeaders_cloneId; }
|
|
// clear out cloned table, except for sticky header
|
|
// include caption & filter row (fixes #126 & #249) - don't remove cells to get correct cell indexing
|
|
$stickyTable.find('thead:gt(0), tr.sticky-false').hide();
|
|
$stickyTable.find('tbody, tfoot').remove();
|
|
$stickyTable.find('caption').toggle(wo.stickyHeaders_includeCaption);
|
|
// issue #172 - find td/th in sticky header
|
|
$stickyCells = $stickyThead.children().children();
|
|
$stickyTable.css({ height:0, width:0, margin: 0 });
|
|
// remove resizable block
|
|
$stickyCells.find('.' + ts.css.resizer).remove();
|
|
// update sticky header class names to match real header after sorting
|
|
$table
|
|
.addClass('hasStickyHeaders')
|
|
.bind('pagerComplete' + namespace, function() {
|
|
resizeHeader();
|
|
});
|
|
|
|
ts.bindEvents(table, $stickyThead.children().children('.' + ts.css.header));
|
|
|
|
// add stickyheaders AFTER the table. If the table is selected by ID, the original one (first) will be returned.
|
|
$table.after( $stickyWrap );
|
|
|
|
// onRenderHeader is defined, we need to do something about it (fixes #641)
|
|
if (c.onRenderHeader) {
|
|
$t = $stickyThead.children('tr').children();
|
|
len = $t.length;
|
|
for ( index = 0; index < len; index++ ) {
|
|
// send second parameter
|
|
c.onRenderHeader.apply( $t.eq( index ), [ index, c, $stickyTable ] );
|
|
}
|
|
}
|
|
|
|
// make it sticky!
|
|
$xScroll.add($yScroll)
|
|
.unbind( ('scroll resize '.split(' ').join( namespace )).replace(/\s+/g, ' ') )
|
|
.bind('scroll resize '.split(' ').join( namespace ), function( event ) {
|
|
scrollSticky( event.type === 'resize' );
|
|
});
|
|
c.$table
|
|
.unbind('stickyHeadersUpdate' + namespace)
|
|
.bind('stickyHeadersUpdate' + namespace, function(){
|
|
scrollSticky( true );
|
|
});
|
|
|
|
if (wo.stickyHeaders_addResizeEvent) {
|
|
ts.addHeaderResizeEvent(table);
|
|
}
|
|
|
|
// look for filter widget
|
|
if ($table.hasClass('hasFilters') && wo.filter_columnFilters) {
|
|
// scroll table into view after filtering, if sticky header is active - #482
|
|
$table.bind('filterEnd' + namespace, function() {
|
|
// $(':focus') needs jQuery 1.6+
|
|
var $td = $(document.activeElement).closest('td'),
|
|
column = $td.parent().children().index($td);
|
|
// only scroll if sticky header is active
|
|
if ($stickyWrap.hasClass(ts.css.stickyVis) && wo.stickyHeaders_filteredToTop) {
|
|
// scroll to original table (not sticky clone)
|
|
window.scrollTo(0, $table.position().top);
|
|
// give same input/select focus; check if c.$filters exists; fixes #594
|
|
if (column >= 0 && c.$filters) {
|
|
c.$filters.eq(column).find('a, select, input').filter(':visible').focus();
|
|
}
|
|
}
|
|
});
|
|
ts.filter.bindSearch( $table, $stickyCells.find('.' + ts.css.filter) );
|
|
// support hideFilters
|
|
if (wo.filter_hideFilters) {
|
|
ts.filter.hideFilters(c, $stickyTable);
|
|
}
|
|
}
|
|
|
|
// resize table (Firefox)
|
|
if (wo.stickyHeaders_addResizeEvent) {
|
|
$table.bind('resize' + c.namespace + 'stickyheaders', function() {
|
|
resizeHeader();
|
|
});
|
|
}
|
|
|
|
$table.triggerHandler('stickyHeadersInit');
|
|
|
|
},
|
|
remove: function(table, c, wo) {
|
|
var namespace = c.namespace + 'stickyheaders ';
|
|
c.$table
|
|
.removeClass('hasStickyHeaders')
|
|
.unbind( ('pagerComplete resize filterEnd stickyHeadersUpdate '.split(' ').join(namespace)).replace(/\s+/g, ' ') )
|
|
.next('.' + ts.css.stickyWrap).remove();
|
|
if (wo.$sticky && wo.$sticky.length) { wo.$sticky.remove(); } // remove cloned table
|
|
$(window)
|
|
.add(wo.stickyHeaders_xScroll)
|
|
.add(wo.stickyHeaders_yScroll)
|
|
.add(wo.stickyHeaders_attachTo)
|
|
.unbind( ('scroll resize '.split(' ').join(namespace)).replace(/\s+/g, ' ') );
|
|
ts.addHeaderResizeEvent(table, true);
|
|
}
|
|
});
|
|
|
|
})(jQuery, window);
|
|
|
|
/*! Widget: resizable - updated 11/4/2015 (v2.24.3) */
|
|
/*jshint browser:true, jquery:true, unused:false */
|
|
;(function ($, window) {
|
|
'use strict';
|
|
var ts = $.tablesorter || {};
|
|
|
|
$.extend(ts.css, {
|
|
resizableContainer : 'tablesorter-resizable-container',
|
|
resizableHandle : 'tablesorter-resizable-handle',
|
|
resizableNoSelect : 'tablesorter-disableSelection',
|
|
resizableStorage : 'tablesorter-resizable'
|
|
});
|
|
|
|
// Add extra scroller css
|
|
$(function(){
|
|
var s = '<style>' +
|
|
'body.' + ts.css.resizableNoSelect + ' { -ms-user-select: none; -moz-user-select: -moz-none;' +
|
|
'-khtml-user-select: none; -webkit-user-select: none; user-select: none; }' +
|
|
'.' + ts.css.resizableContainer + ' { position: relative; height: 1px; }' +
|
|
// make handle z-index > than stickyHeader z-index, so the handle stays above sticky header
|
|
'.' + ts.css.resizableHandle + ' { position: absolute; display: inline-block; width: 8px;' +
|
|
'top: 1px; cursor: ew-resize; z-index: 3; user-select: none; -moz-user-select: none; }' +
|
|
'</style>';
|
|
$(s).appendTo('body');
|
|
});
|
|
|
|
ts.resizable = {
|
|
init : function( c, wo ) {
|
|
if ( c.$table.hasClass( 'hasResizable' ) ) { return; }
|
|
c.$table.addClass( 'hasResizable' );
|
|
|
|
var noResize, $header, column, storedSizes, tmp,
|
|
$table = c.$table,
|
|
$parent = $table.parent(),
|
|
marginTop = parseInt( $table.css( 'margin-top' ), 10 ),
|
|
|
|
// internal variables
|
|
vars = wo.resizable_vars = {
|
|
useStorage : ts.storage && wo.resizable !== false,
|
|
$wrap : $parent,
|
|
mouseXPosition : 0,
|
|
$target : null,
|
|
$next : null,
|
|
overflow : $parent.css('overflow') === 'auto' ||
|
|
$parent.css('overflow') === 'scroll' ||
|
|
$parent.css('overflow-x') === 'auto' ||
|
|
$parent.css('overflow-x') === 'scroll',
|
|
storedSizes : []
|
|
};
|
|
|
|
// set default widths
|
|
ts.resizableReset( c.table, true );
|
|
|
|
// now get measurements!
|
|
vars.tableWidth = $table.width();
|
|
// attempt to autodetect
|
|
vars.fullWidth = Math.abs( $parent.width() - vars.tableWidth ) < 20;
|
|
|
|
/*
|
|
// Hacky method to determine if table width is set to 'auto'
|
|
// http://stackoverflow.com/a/20892048/145346
|
|
if ( !vars.fullWidth ) {
|
|
tmp = $table.width();
|
|
$header = $table.wrap('<span>').parent(); // temp variable
|
|
storedSizes = parseInt( $table.css( 'margin-left' ), 10 ) || 0;
|
|
$table.css( 'margin-left', storedSizes + 50 );
|
|
vars.tableWidth = $header.width() > tmp ? 'auto' : tmp;
|
|
$table.css( 'margin-left', storedSizes ? storedSizes : '' );
|
|
$header = null;
|
|
$table.unwrap('<span>');
|
|
}
|
|
*/
|
|
|
|
if ( vars.useStorage && vars.overflow ) {
|
|
// save table width
|
|
ts.storage( c.table, 'tablesorter-table-original-css-width', vars.tableWidth );
|
|
tmp = ts.storage( c.table, 'tablesorter-table-resized-width' ) || 'auto';
|
|
ts.resizable.setWidth( $table, tmp, true );
|
|
}
|
|
wo.resizable_vars.storedSizes = storedSizes = ( vars.useStorage ?
|
|
ts.storage( c.table, ts.css.resizableStorage ) :
|
|
[] ) || [];
|
|
ts.resizable.setWidths( c, wo, storedSizes );
|
|
ts.resizable.updateStoredSizes( c, wo );
|
|
|
|
wo.$resizable_container = $( '<div class="' + ts.css.resizableContainer + '">' )
|
|
.css({ top : marginTop })
|
|
.insertBefore( $table );
|
|
// add container
|
|
for ( column = 0; column < c.columns; column++ ) {
|
|
$header = c.$headerIndexed[ column ];
|
|
tmp = ts.getColumnData( c.table, c.headers, column );
|
|
noResize = ts.getData( $header, tmp, 'resizable' ) === 'false';
|
|
if ( !noResize ) {
|
|
$( '<div class="' + ts.css.resizableHandle + '">' )
|
|
.appendTo( wo.$resizable_container )
|
|
.attr({
|
|
'data-column' : column,
|
|
'unselectable' : 'on'
|
|
})
|
|
.data( 'header', $header )
|
|
.bind( 'selectstart', false );
|
|
}
|
|
}
|
|
ts.resizable.setHandlePosition( c, wo );
|
|
ts.resizable.bindings( c, wo );
|
|
},
|
|
|
|
updateStoredSizes : function( c, wo ) {
|
|
var column, $header,
|
|
len = c.columns,
|
|
vars = wo.resizable_vars;
|
|
vars.storedSizes = [];
|
|
for ( column = 0; column < len; column++ ) {
|
|
$header = c.$headerIndexed[ column ];
|
|
vars.storedSizes[ column ] = $header.is(':visible') ? $header.width() : 0;
|
|
}
|
|
},
|
|
|
|
setWidth : function( $el, width, overflow ) {
|
|
// overflow tables need min & max width set as well
|
|
$el.css({
|
|
'width' : width,
|
|
'min-width' : overflow ? width : '',
|
|
'max-width' : overflow ? width : ''
|
|
});
|
|
},
|
|
|
|
setWidths : function( c, wo, storedSizes ) {
|
|
var column, $temp,
|
|
vars = wo.resizable_vars,
|
|
$extra = $( c.namespace + '_extra_headers' ),
|
|
$col = c.$table.children( 'colgroup' ).children( 'col' );
|
|
storedSizes = storedSizes || vars.storedSizes || [];
|
|
// process only if table ID or url match
|
|
if ( storedSizes.length ) {
|
|
for ( column = 0; column < c.columns; column++ ) {
|
|
// set saved resizable widths
|
|
ts.resizable.setWidth( c.$headerIndexed[ column ], storedSizes[ column ], vars.overflow );
|
|
if ( $extra.length ) {
|
|
// stickyHeaders needs to modify min & max width as well
|
|
$temp = $extra.eq( column ).add( $col.eq( column ) );
|
|
ts.resizable.setWidth( $temp, storedSizes[ column ], vars.overflow );
|
|
}
|
|
}
|
|
$temp = $( c.namespace + '_extra_table' );
|
|
if ( $temp.length && !ts.hasWidget( c.table, 'scroller' ) ) {
|
|
ts.resizable.setWidth( $temp, c.$table.outerWidth(), vars.overflow );
|
|
}
|
|
}
|
|
},
|
|
|
|
setHandlePosition : function( c, wo ) {
|
|
var startPosition,
|
|
hasScroller = ts.hasWidget( c.table, 'scroller' ),
|
|
tableHeight = c.$table.height(),
|
|
$handles = wo.$resizable_container.children(),
|
|
handleCenter = Math.floor( $handles.width() / 2 );
|
|
|
|
if ( hasScroller ) {
|
|
tableHeight = 0;
|
|
c.$table.closest( '.' + ts.css.scrollerWrap ).children().each(function(){
|
|
var $this = $(this);
|
|
// center table has a max-height set
|
|
tableHeight += $this.filter('[style*="height"]').length ? $this.height() : $this.children('table').height();
|
|
});
|
|
}
|
|
// subtract out table left position from resizable handles. Fixes #864
|
|
startPosition = c.$table.position().left;
|
|
$handles.each( function() {
|
|
var $this = $(this),
|
|
column = parseInt( $this.attr( 'data-column' ), 10 ),
|
|
columns = c.columns - 1,
|
|
$header = $this.data( 'header' );
|
|
if ( !$header ) { return; } // see #859
|
|
if ( !$header.is(':visible') ) {
|
|
$this.hide();
|
|
} else if ( column < columns || column === columns && wo.resizable_addLastColumn ) {
|
|
$this.css({
|
|
display: 'inline-block',
|
|
height : tableHeight,
|
|
left : $header.position().left - startPosition + $header.outerWidth() - handleCenter
|
|
});
|
|
}
|
|
});
|
|
},
|
|
|
|
// prevent text selection while dragging resize bar
|
|
toggleTextSelection : function( c, wo, toggle ) {
|
|
var namespace = c.namespace + 'tsresize';
|
|
wo.resizable_vars.disabled = toggle;
|
|
$( 'body' ).toggleClass( ts.css.resizableNoSelect, toggle );
|
|
if ( toggle ) {
|
|
$( 'body' )
|
|
.attr( 'unselectable', 'on' )
|
|
.bind( 'selectstart' + namespace, false );
|
|
} else {
|
|
$( 'body' )
|
|
.removeAttr( 'unselectable' )
|
|
.unbind( 'selectstart' + namespace );
|
|
}
|
|
},
|
|
|
|
bindings : function( c, wo ) {
|
|
var namespace = c.namespace + 'tsresize';
|
|
wo.$resizable_container.children().bind( 'mousedown', function( event ) {
|
|
// save header cell and mouse position
|
|
var column,
|
|
vars = wo.resizable_vars,
|
|
$extras = $( c.namespace + '_extra_headers' ),
|
|
$header = $( event.target ).data( 'header' );
|
|
|
|
column = parseInt( $header.attr( 'data-column' ), 10 );
|
|
vars.$target = $header = $header.add( $extras.filter('[data-column="' + column + '"]') );
|
|
vars.target = column;
|
|
|
|
// if table is not as wide as it's parent, then resize the table
|
|
vars.$next = event.shiftKey || wo.resizable_targetLast ?
|
|
$header.parent().children().not( '.resizable-false' ).filter( ':last' ) :
|
|
$header.nextAll( ':not(.resizable-false)' ).eq( 0 );
|
|
|
|
column = parseInt( vars.$next.attr( 'data-column' ), 10 );
|
|
vars.$next = vars.$next.add( $extras.filter('[data-column="' + column + '"]') );
|
|
vars.next = column;
|
|
|
|
vars.mouseXPosition = event.pageX;
|
|
ts.resizable.updateStoredSizes( c, wo );
|
|
ts.resizable.toggleTextSelection(c, wo, true );
|
|
});
|
|
|
|
$( document )
|
|
.bind( 'mousemove' + namespace, function( event ) {
|
|
var vars = wo.resizable_vars;
|
|
// ignore mousemove if no mousedown
|
|
if ( !vars.disabled || vars.mouseXPosition === 0 || !vars.$target ) { return; }
|
|
if ( wo.resizable_throttle ) {
|
|
clearTimeout( vars.timer );
|
|
vars.timer = setTimeout( function() {
|
|
ts.resizable.mouseMove( c, wo, event );
|
|
}, isNaN( wo.resizable_throttle ) ? 5 : wo.resizable_throttle );
|
|
} else {
|
|
ts.resizable.mouseMove( c, wo, event );
|
|
}
|
|
})
|
|
.bind( 'mouseup' + namespace, function() {
|
|
if (!wo.resizable_vars.disabled) { return; }
|
|
ts.resizable.toggleTextSelection( c, wo, false );
|
|
ts.resizable.stopResize( c, wo );
|
|
ts.resizable.setHandlePosition( c, wo );
|
|
});
|
|
|
|
// resizeEnd event triggered by scroller widget
|
|
$( window ).bind( 'resize' + namespace + ' resizeEnd' + namespace, function() {
|
|
ts.resizable.setHandlePosition( c, wo );
|
|
});
|
|
|
|
// right click to reset columns to default widths
|
|
c.$table
|
|
.bind( 'columnUpdate' + namespace, function() {
|
|
ts.resizable.setHandlePosition( c, wo );
|
|
})
|
|
.find( 'thead:first' )
|
|
.add( $( c.namespace + '_extra_table' ).find( 'thead:first' ) )
|
|
.bind( 'contextmenu' + namespace, function() {
|
|
// $.isEmptyObject() needs jQuery 1.4+; allow right click if already reset
|
|
var allowClick = wo.resizable_vars.storedSizes.length === 0;
|
|
ts.resizableReset( c.table );
|
|
ts.resizable.setHandlePosition( c, wo );
|
|
wo.resizable_vars.storedSizes = [];
|
|
return allowClick;
|
|
});
|
|
|
|
},
|
|
|
|
mouseMove : function( c, wo, event ) {
|
|
if ( wo.resizable_vars.mouseXPosition === 0 || !wo.resizable_vars.$target ) { return; }
|
|
// resize columns
|
|
var column,
|
|
total = 0,
|
|
vars = wo.resizable_vars,
|
|
$next = vars.$next,
|
|
tar = vars.storedSizes[ vars.target ],
|
|
leftEdge = event.pageX - vars.mouseXPosition;
|
|
if ( vars.overflow ) {
|
|
if ( tar + leftEdge > 0 ) {
|
|
vars.storedSizes[ vars.target ] += leftEdge;
|
|
ts.resizable.setWidth( vars.$target, vars.storedSizes[ vars.target ], true );
|
|
// update the entire table width
|
|
for ( column = 0; column < c.columns; column++ ) {
|
|
total += vars.storedSizes[ column ];
|
|
}
|
|
ts.resizable.setWidth( c.$table.add( $( c.namespace + '_extra_table' ) ), total );
|
|
}
|
|
if ( !$next.length ) {
|
|
// if expanding right-most column, scroll the wrapper
|
|
vars.$wrap[0].scrollLeft = c.$table.width();
|
|
}
|
|
} else if ( vars.fullWidth ) {
|
|
vars.storedSizes[ vars.target ] += leftEdge;
|
|
vars.storedSizes[ vars.next ] -= leftEdge;
|
|
ts.resizable.setWidths( c, wo );
|
|
} else {
|
|
vars.storedSizes[ vars.target ] += leftEdge;
|
|
ts.resizable.setWidths( c, wo );
|
|
}
|
|
vars.mouseXPosition = event.pageX;
|
|
// dynamically update sticky header widths
|
|
c.$table.triggerHandler('stickyHeadersUpdate');
|
|
},
|
|
|
|
stopResize : function( c, wo ) {
|
|
var vars = wo.resizable_vars;
|
|
ts.resizable.updateStoredSizes( c, wo );
|
|
if ( vars.useStorage ) {
|
|
// save all column widths
|
|
ts.storage( c.table, ts.css.resizableStorage, vars.storedSizes );
|
|
ts.storage( c.table, 'tablesorter-table-resized-width', c.$table.width() );
|
|
}
|
|
vars.mouseXPosition = 0;
|
|
vars.$target = vars.$next = null;
|
|
// will update stickyHeaders, just in case, see #912
|
|
c.$table.triggerHandler('stickyHeadersUpdate');
|
|
}
|
|
};
|
|
|
|
// this widget saves the column widths if
|
|
// $.tablesorter.storage function is included
|
|
// **************************
|
|
ts.addWidget({
|
|
id: 'resizable',
|
|
priority: 40,
|
|
options: {
|
|
resizable : true, // save column widths to storage
|
|
resizable_addLastColumn : false,
|
|
resizable_widths : [],
|
|
resizable_throttle : false, // set to true (5ms) or any number 0-10 range
|
|
resizable_targetLast : false,
|
|
resizable_fullWidth : null
|
|
},
|
|
init: function(table, thisWidget, c, wo) {
|
|
ts.resizable.init( c, wo );
|
|
},
|
|
remove: function( table, c, wo, refreshing ) {
|
|
if (wo.$resizable_container) {
|
|
var namespace = c.namespace + 'tsresize';
|
|
c.$table.add( $( c.namespace + '_extra_table' ) )
|
|
.removeClass('hasResizable')
|
|
.children( 'thead' )
|
|
.unbind( 'contextmenu' + namespace );
|
|
|
|
wo.$resizable_container.remove();
|
|
ts.resizable.toggleTextSelection( c, wo, false );
|
|
ts.resizableReset( table, refreshing );
|
|
$( document ).unbind( 'mousemove' + namespace + ' mouseup' + namespace );
|
|
}
|
|
}
|
|
});
|
|
|
|
ts.resizableReset = function( table, refreshing ) {
|
|
$( table ).each(function(){
|
|
var index, $t,
|
|
c = this.config,
|
|
wo = c && c.widgetOptions,
|
|
vars = wo.resizable_vars;
|
|
if ( table && c && c.$headerIndexed.length ) {
|
|
// restore the initial table width
|
|
if ( vars.overflow && vars.tableWidth ) {
|
|
ts.resizable.setWidth( c.$table, vars.tableWidth, true );
|
|
if ( vars.useStorage ) {
|
|
ts.storage( table, 'tablesorter-table-resized-width', 'auto' );
|
|
}
|
|
}
|
|
for ( index = 0; index < c.columns; index++ ) {
|
|
$t = c.$headerIndexed[ index ];
|
|
if ( wo.resizable_widths && wo.resizable_widths[ index ] ) {
|
|
ts.resizable.setWidth( $t, wo.resizable_widths[ index ], vars.overflow );
|
|
} else if ( !$t.hasClass( 'resizable-false' ) ) {
|
|
// don't clear the width of any column that is not resizable
|
|
ts.resizable.setWidth( $t, '', vars.overflow );
|
|
}
|
|
}
|
|
|
|
// reset stickyHeader widths
|
|
c.$table.triggerHandler( 'stickyHeadersUpdate' );
|
|
if ( ts.storage && !refreshing ) {
|
|
ts.storage( this, ts.css.resizableStorage, {} );
|
|
}
|
|
}
|
|
});
|
|
};
|
|
|
|
})( jQuery, window );
|
|
|
|
/*! Widget: saveSort - updated 10/31/2015 (v2.24.0) *//*
|
|
* Requires tablesorter v2.16+
|
|
* by Rob Garrison
|
|
*/
|
|
;(function ($) {
|
|
'use strict';
|
|
var ts = $.tablesorter || {};
|
|
|
|
// this widget saves the last sort only if the
|
|
// saveSort widget option is true AND the
|
|
// $.tablesorter.storage function is included
|
|
// **************************
|
|
ts.addWidget({
|
|
id: 'saveSort',
|
|
priority: 20,
|
|
options: {
|
|
saveSort : true
|
|
},
|
|
init: function(table, thisWidget, c, wo) {
|
|
// run widget format before all other widgets are applied to the table
|
|
thisWidget.format(table, c, wo, true);
|
|
},
|
|
format: function(table, c, wo, init) {
|
|
var stored, time,
|
|
$table = c.$table,
|
|
saveSort = wo.saveSort !== false, // make saveSort active/inactive; default to true
|
|
sortList = { 'sortList' : c.sortList };
|
|
if (c.debug) {
|
|
time = new Date();
|
|
}
|
|
if ($table.hasClass('hasSaveSort')) {
|
|
if (saveSort && table.hasInitialized && ts.storage) {
|
|
ts.storage( table, 'tablesorter-savesort', sortList );
|
|
if (c.debug) {
|
|
console.log('saveSort widget: Saving last sort: ' + c.sortList + ts.benchmark(time));
|
|
}
|
|
}
|
|
} else {
|
|
// set table sort on initial run of the widget
|
|
$table.addClass('hasSaveSort');
|
|
sortList = '';
|
|
// get data
|
|
if (ts.storage) {
|
|
stored = ts.storage( table, 'tablesorter-savesort' );
|
|
sortList = (stored && stored.hasOwnProperty('sortList') && $.isArray(stored.sortList)) ? stored.sortList : '';
|
|
if (c.debug) {
|
|
console.log('saveSort: Last sort loaded: "' + sortList + '"' + ts.benchmark(time));
|
|
}
|
|
$table.bind('saveSortReset', function(event) {
|
|
event.stopPropagation();
|
|
ts.storage( table, 'tablesorter-savesort', '' );
|
|
});
|
|
}
|
|
// init is true when widget init is run, this will run this widget before all other widgets have initialized
|
|
// this method allows using this widget in the original tablesorter plugin; but then it will run all widgets twice.
|
|
if (init && sortList && sortList.length > 0) {
|
|
c.sortList = sortList;
|
|
} else if (table.hasInitialized && sortList && sortList.length > 0) {
|
|
// update sort change
|
|
ts.sortOn( c, sortList );
|
|
}
|
|
}
|
|
},
|
|
remove: function(table, c) {
|
|
c.$table.removeClass('hasSaveSort');
|
|
// clear storage
|
|
if (ts.storage) { ts.storage( table, 'tablesorter-savesort', '' ); }
|
|
}
|
|
});
|
|
|
|
})(jQuery);
|
|
|
|
return $.tablesorter;
|
|
}));
|
|
;/*
|
|
*
|
|
* StaticRow widget for jQuery TableSorter 2.0
|
|
* Version 1.0
|
|
*
|
|
* Copyright (c) 2011 Nils Luxton
|
|
* Licensed under the MIT license:
|
|
* http://www.opensource.org/licenses/mit-license.php
|
|
*
|
|
*/
|
|
|
|
$.tablesorter.addWidget({
|
|
|
|
// Give the new Widget an ID to be used in the tablesorter() call, as follows:
|
|
// $('#myElement').tablesorter({ widgets: ['zebra','staticRow'] });
|
|
id: 'staticRow',
|
|
|
|
// "Format" is run on all widgets once when the tablesorter has finished initialising,
|
|
// and then again every time a sort has finished.
|
|
format: function(table) {
|
|
|
|
// Use a property of the function to determine
|
|
// whether this is the first run of "Format"
|
|
// (i.e. is this the table's default starting position,
|
|
// or has it been sorted?)
|
|
if (typeof $(table).data('hasSorted') == 'undefined')
|
|
{
|
|
$(table).data('hasSorted', true); // This will force us into the "else" block the next time "Format" is run
|
|
|
|
// "Index" the static rows, saving their current (starting)
|
|
// position in the table inside a data() param on the
|
|
// <tr> element itself for later use.
|
|
$('tbody .static', table).each(function() {
|
|
$(this).data('tableindex', $(this).index());
|
|
});
|
|
}
|
|
else
|
|
{
|
|
// Loop the static rows, moving them to their
|
|
// original "indexed" position, and keep doing
|
|
// this until no more re-shuffling needs doing
|
|
var hasShuffled = true;
|
|
|
|
while (hasShuffled)
|
|
{
|
|
hasShuffled = false;
|
|
$('tbody .static', table).each(function() {
|
|
var targetIndex = $(this).data('tableindex');
|
|
if (targetIndex != $(this).index())
|
|
{
|
|
hasShuffled = true;
|
|
var thisRow = $(this).detach();
|
|
var numRows = $('tbody tr', table).length;
|
|
|
|
// Are we trying to be the last row?
|
|
if (targetIndex >= numRows)
|
|
{
|
|
thisRow.appendTo($('tbody', table));
|
|
}
|
|
// Are we trying to be the first row?
|
|
else if (targetIndex == 0)
|
|
{
|
|
thisRow.prependTo($('tbody', table));
|
|
}
|
|
// No, we want to be somewhere in the middle!
|
|
else
|
|
{
|
|
thisRow.insertBefore($('tbody tr:eq(' + targetIndex + ')', table));
|
|
}
|
|
}
|
|
});
|
|
}
|
|
}
|
|
|
|
$('tbody .static-last', table).each(function() {
|
|
var row = $(this).detach();
|
|
row.appendTo($('tbody', table));
|
|
});
|
|
|
|
}
|
|
});
|
|
;function set_feedback(text, classname, keep_displayed)
|
|
{
|
|
if(text)
|
|
{
|
|
$('#feedback_bar').removeClass().addClass(classname).html(text).css('opacity','1');
|
|
|
|
if(!keep_displayed)
|
|
{
|
|
$('#feedback_bar').fadeTo(5000, 1).fadeTo("fast",0);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
$('#feedback_bar').css('opacity','0');
|
|
}
|
|
}
|
|
;/*
|
|
* imgPreview jQuery plugin
|
|
* Copyright (c) 2009 James Padolsey
|
|
* j@qd9.co.uk | http://james.padolsey.com
|
|
* Dual licensed under MIT and GPL.
|
|
* Updated: 09/02/09
|
|
* @author James Padolsey
|
|
* @version 0.22
|
|
*/
|
|
(function($){
|
|
|
|
$.expr[':'].linkingToImage = function(elem, index, match){
|
|
// This will return true if the specified attribute contains a valid link to an image:
|
|
return !! ($(elem).attr(match[3]) && $(elem).attr(match[3]).match(/\.(gif|jpe?g|png|bmp)$/i));
|
|
};
|
|
|
|
$.fn.imgPreview = function(userDefinedSettings){
|
|
|
|
var s = $.extend({
|
|
|
|
/* DEFAULTS */
|
|
|
|
// CSS to be applied to image:
|
|
imgCSS: {},
|
|
// Distance between cursor and preview:
|
|
distanceFromCursor: {top:10, left:10},
|
|
// Boolean, whether or not to preload images:
|
|
preloadImages: true,
|
|
// Callback: run when link is hovered: container is shown:
|
|
onShow: function(){},
|
|
// Callback: container is hidden:
|
|
onHide: function(){},
|
|
// Callback: Run when image within container has loaded:
|
|
onLoad: function(){},
|
|
// ID to give to container (for CSS styling):
|
|
containerID: 'imgPreviewContainer',
|
|
// Class to be given to container while image is loading:
|
|
containerLoadingClass: 'loading',
|
|
// Prefix (if using thumbnails), e.g. 'thumb_'
|
|
thumbPrefix: '',
|
|
// Where to retrieve the image from:
|
|
srcAttr: 'href'
|
|
|
|
}, userDefinedSettings),
|
|
|
|
$container = $('<div/>').attr('id', s.containerID)
|
|
.append('<img/>').hide()
|
|
.css('position','absolute')
|
|
.appendTo('body'),
|
|
|
|
$img = $('img', $container).css(s.imgCSS),
|
|
|
|
// Get all valid elements (linking to images / ATTR with image link):
|
|
$collection = this.filter(':linkingToImage(' + s.srcAttr + ')');
|
|
|
|
// Re-usable means to add prefix (from setting):
|
|
function addPrefix(src) {
|
|
return src && src.replace(/(\/?)([^\/]+)$/,'$1' + s.thumbPrefix + '$2');
|
|
}
|
|
|
|
if (s.preloadImages) {
|
|
(function(i){
|
|
var tempIMG = new Image(),
|
|
callee = arguments.callee;
|
|
var src = $($collection[i]).attr(s.srcAttr)
|
|
if (src)
|
|
{
|
|
tempIMG.src = addPrefix(src);
|
|
tempIMG.onload = function(){
|
|
$collection[i + 1] && callee(i + 1);
|
|
};
|
|
}
|
|
})(0);
|
|
}
|
|
|
|
$collection
|
|
.mousemove(function(e){
|
|
|
|
$container.css({
|
|
top: e.pageY + s.distanceFromCursor.top + 'px',
|
|
left: e.pageX + s.distanceFromCursor.left + 'px'
|
|
});
|
|
|
|
})
|
|
.hover(function(){
|
|
|
|
var link = this;
|
|
$container
|
|
.addClass(s.containerLoadingClass)
|
|
.show();
|
|
$img
|
|
.load(function(){
|
|
$container.removeClass(s.containerLoadingClass);
|
|
$img.show();
|
|
s.onLoad.call($img[0], link);
|
|
})
|
|
.attr( 'src' , addPrefix($(link).attr(s.srcAttr)) );
|
|
s.onShow.call($container[0], link);
|
|
|
|
}, function(){
|
|
|
|
$container.hide();
|
|
$img.unbind('load').attr('src','').hide();
|
|
s.onHide.call($container[0], this);
|
|
|
|
});
|
|
|
|
// Return full selection, not $collection!
|
|
return this;
|
|
|
|
};
|
|
|
|
})(jQuery);;(function(dialog_support, $) {
|
|
|
|
var btn_id, dialog_ref;
|
|
|
|
var hide = function() {
|
|
dialog_ref.close();
|
|
};
|
|
|
|
var clicked_id = function() {
|
|
return btn_id;
|
|
};
|
|
|
|
var submit = function(button_id) {
|
|
return function(dlog_ref) {
|
|
btn_id = button_id;
|
|
dialog_ref = dlog_ref;
|
|
if (button_id == 'submit') {
|
|
$('form', dlog_ref.$modalBody).first().submit();
|
|
}
|
|
}
|
|
};
|
|
|
|
var button_class = {
|
|
'submit' : 'btn-primary',
|
|
'delete' : 'btn-danger'
|
|
};
|
|
|
|
var init = function(selector) {
|
|
|
|
var buttons = function(event) {
|
|
var buttons = [];
|
|
var dialog_class = 'modal-dlg';
|
|
$.each($(this).attr('class').split(/\s+/), function(classIndex, className) {
|
|
var width_class = className.split("modal-dlg-");
|
|
if (width_class && width_class.length > 1) {
|
|
dialog_class = className;
|
|
}
|
|
var btn_class = className.split("modal-btn-");
|
|
if (btn_class && btn_class.length > 1) {
|
|
var btn_name = btn_class[1];
|
|
var is_submit = btn_name == 'submit';
|
|
buttons.push({
|
|
id: btn_name,
|
|
label: btn_name.charAt(0).toUpperCase() + btn_name.slice(1),
|
|
cssClass: button_class[btn_name],
|
|
hotkey: is_submit ? 13 : undefined, // Enter.
|
|
action: submit(btn_name)
|
|
});
|
|
}
|
|
});
|
|
|
|
!buttons.length && buttons.push({
|
|
id: 'close',
|
|
label: 'Close',
|
|
cssClass: 'btn-primary',
|
|
action: function(dialog_ref) {
|
|
dialog_ref.close();
|
|
}
|
|
});
|
|
return { buttons: buttons, cssClass: dialog_class};
|
|
};
|
|
|
|
$(selector).each(function(index, $element) {
|
|
|
|
return $(selector).off('click').on('click', function(event) {
|
|
var $link = $(event.target);
|
|
$link = !$link.is("a, button") ? $link.parents("a") : $link ;
|
|
BootstrapDialog.show($.extend({
|
|
title: $link.attr('title'),
|
|
message: (function() {
|
|
var node = $('<div></div>');
|
|
$.get($link.attr('href') || $link.data('href'), function(data) {
|
|
node.html(data);
|
|
});
|
|
return node;
|
|
})
|
|
}, buttons.call(this, event)));
|
|
|
|
return false;
|
|
});
|
|
});
|
|
};
|
|
|
|
dialog_support.error = {
|
|
errorClass: "has-error",
|
|
errorLabelContainer: "#error_message_box",
|
|
wrapper: "li",
|
|
highlight: function (e) {
|
|
$(e).closest('.form-group').addClass('has-error');
|
|
},
|
|
unhighlight: function (e) {
|
|
$(e).closest('.form-group').removeClass('has-error');
|
|
}
|
|
};
|
|
|
|
$.extend(dialog_support, {
|
|
init: init,
|
|
submit: submit,
|
|
hide: hide,
|
|
clicked_id: clicked_id
|
|
});
|
|
|
|
})(window.dialog_support = window.dialog_support || {}, jQuery);
|
|
|
|
(function(table_support, $) {
|
|
|
|
var enable_actions = function(callback) {
|
|
return function() {
|
|
var selection_empty = selected_rows().length == 0;
|
|
$("#toolbar button:not(.dropdown-toggle)").attr('disabled', selection_empty);
|
|
typeof callback == 'function' && callback();
|
|
}
|
|
};
|
|
|
|
var table = function() {
|
|
return $("#table").data('bootstrap.table');
|
|
}
|
|
|
|
var selected_ids = function () {
|
|
return $.map(table().getSelections(), function (element) {
|
|
return element.id;
|
|
});
|
|
};
|
|
|
|
var selected_rows = function () {
|
|
return $("#table input:checkbox:checked").parents("tr");
|
|
};
|
|
|
|
var row_selector = function(id) {
|
|
return "tr[data-uniqueid='" + id + "']";
|
|
};
|
|
|
|
var rows_selector = function(ids) {
|
|
var selectors = [];
|
|
ids = ids instanceof Array ? ids : ("" + ids).split(",");
|
|
$.each(ids, function(index, element) {
|
|
selectors.push(row_selector(element));
|
|
});
|
|
return selectors;;
|
|
};
|
|
|
|
var highlight_row = function (id, color) {
|
|
var original = $(row_selector(id)).css('backgroundColor');
|
|
$(row_selector(id)).find("td").animate({backgroundColor: color || '#e1ffdd'}, "slow", "linear")
|
|
.animate({backgroundColor: color || '#e1ffdd'}, 5000)
|
|
.animate({backgroundColor: original}, "slow", "linear");
|
|
};
|
|
|
|
var do_delete = function () {
|
|
if (confirm(options.confirmDeleteMessage)) {
|
|
$.post(options.resource + '/delete', {'ids[]': selected_ids()}, function (response) {
|
|
//delete was successful, remove checkbox rows
|
|
if (response.success) {
|
|
$(selected_rows()).each(function (index, element) {
|
|
$(this).find("td").animate({backgroundColor: "green"}, 1200, "linear")
|
|
.end().animate({opacity: 0}, 1200, "linear", function () {
|
|
table().remove({
|
|
field: 'id',
|
|
values: selected_ids()
|
|
});
|
|
enable_actions();
|
|
});
|
|
});
|
|
set_feedback(response.message, 'alert alert-dismissible alert-success', false);
|
|
} else {
|
|
set_feedback(response.message, 'alert alert-dismissible alert-danger', true);
|
|
}
|
|
}, "json");
|
|
} else {
|
|
return false;
|
|
}
|
|
};
|
|
|
|
var load_success = function(callback) {
|
|
return function(response) {
|
|
typeof options.load_callback == 'function' && options.load_callback();
|
|
options.load_callback = undefined;
|
|
dialog_support.init("a.modal-dlg, button.modal-dlg");
|
|
typeof callback == 'function' && callback.call(this, response);
|
|
}
|
|
};
|
|
|
|
var options;
|
|
|
|
var init = function (_options) {
|
|
options = _options;
|
|
enable_actions = enable_actions(options.enableActions);
|
|
$('#table').bootstrapTable($.extend(options, {
|
|
columns: options.headers,
|
|
url: options.resource + '/search',
|
|
sidePagination: 'server',
|
|
pageSize: options.pageSize,
|
|
striped: true,
|
|
pagination: true,
|
|
search: options.resource || false,
|
|
showColumns: true,
|
|
clickToSelect: true,
|
|
toolbar: '#toolbar',
|
|
uniqueId: 'id',
|
|
onCheck: enable_actions,
|
|
onUncheck: enable_actions,
|
|
onCheckAll: enable_actions,
|
|
onUncheckAll: enable_actions,
|
|
onLoadSuccess: load_success(options.onLoadSuccess),
|
|
queryParamsType: 'limit',
|
|
iconSize: 'sm',
|
|
silentSort: true,
|
|
paginationVAlign: 'bottom'
|
|
}));
|
|
enable_actions();
|
|
init_delete();
|
|
};
|
|
|
|
var init_delete = function (confirmMessage) {
|
|
$("#delete").click(function (event) {
|
|
do_delete();
|
|
});
|
|
};
|
|
|
|
var refresh = function() {
|
|
table().refresh();
|
|
}
|
|
|
|
var submit_handler = function(url) {
|
|
return function (resource, response) {
|
|
var id = response.id;
|
|
|
|
if (!response.success) {
|
|
set_feedback(response.message, 'alert alert-dismissible alert-danger', true);
|
|
} else {
|
|
var message = response.message;
|
|
var selector = rows_selector(response.id);
|
|
if ($(selector.join(",")).length > 0) {
|
|
$.each(selector, function (index, element) {
|
|
var id = $(element).data('uniqueid');
|
|
$.get({
|
|
url: url + id || resource + '/get_row/' + id,
|
|
success: function (response) {
|
|
table().updateByUniqueId({id: id, row: response});
|
|
// TODO make selector more specific?
|
|
dialog_support.init("a.modal-dlg");
|
|
enable_actions();
|
|
highlight_row(id);
|
|
},
|
|
dataType: 'json'
|
|
});
|
|
});
|
|
} else {
|
|
// call hightlight function once after refresh
|
|
options.load_callback = function () {
|
|
enable_actions();
|
|
highlight_row(id);
|
|
};
|
|
refresh();
|
|
}
|
|
set_feedback(message, 'alert alert-dismissible alert-success', false);
|
|
}
|
|
};
|
|
};
|
|
|
|
var handle_submit = submit_handler();
|
|
|
|
$.extend(table_support, {
|
|
submit_handler: function(url) {
|
|
handle_submit = submit_handler(url);
|
|
},
|
|
handle_submit: handle_submit,
|
|
init: init,
|
|
do_delete: do_delete,
|
|
refresh : refresh,
|
|
selected_ids : selected_ids,
|
|
});
|
|
|
|
})(window.table_support = window.table_support || {}, jQuery);;(function($) {
|
|
|
|
function http_s(url)
|
|
{
|
|
return document.location.protocol + '//' + url;
|
|
}
|
|
|
|
if (window.sessionStorage && !sessionStorage['country'])
|
|
{
|
|
/*$.ajax({
|
|
type: "GET",
|
|
url: http_s('ipinfo.io/json'),
|
|
success: function(response) {
|
|
sessionStorage['country'] = response.country;
|
|
}, dataType: 'jsonp'
|
|
})*/;
|
|
}
|
|
|
|
var url = http_s('nominatim.openstreetmap.org/search');
|
|
|
|
var handle_auto_completion = function(fields) {
|
|
return function(event, ui) {
|
|
var results = ui.item.results;
|
|
if (results != null && results.length > 0) {
|
|
// handle auto completion
|
|
for(var i in fields) {
|
|
$("#" + fields[i]).val(results[i]);
|
|
}
|
|
return false;
|
|
}
|
|
return true;
|
|
};
|
|
};
|
|
|
|
var create_parser = function(field_name, parse_format)
|
|
{
|
|
var parse_field = function(format, address)
|
|
{
|
|
var fields = [];
|
|
$.each(format.split("|"), function(key, value)
|
|
{
|
|
if (address[value] && fields.length < 2 && $.inArray(address[value], fields) === -1)
|
|
{
|
|
fields.push(address[value]);
|
|
}
|
|
});
|
|
return fields[0] + (fields[1] ? ' (' + fields[1] + ')' : '');
|
|
};
|
|
|
|
return function(data)
|
|
{
|
|
var parsed = [];
|
|
$.each(data, function(index, value)
|
|
{
|
|
var row = [];
|
|
var address = value.address;
|
|
$.each(parse_format, function(key, format)
|
|
{
|
|
row.push(parse_field(format, address));
|
|
});
|
|
parsed[index] = {
|
|
label: row.join(", "),
|
|
results: row,
|
|
value: address[field_name]
|
|
};
|
|
});
|
|
return parsed;
|
|
};
|
|
};
|
|
|
|
var init = function(options) {
|
|
|
|
var default_params = function(id, key, language)
|
|
{
|
|
return function() {
|
|
var result = {
|
|
format: 'json',
|
|
limit: 5,
|
|
addressdetails: 1,
|
|
countrycodes: window['sessionStorage'] ? sessionStorage['country'] : 'be',
|
|
'accept-language' : language || navigator.language
|
|
};
|
|
result[key || id] = $("#"+id).val();
|
|
return result;
|
|
}
|
|
|
|
};
|
|
|
|
$.each(options.fields, function(key, value)
|
|
{
|
|
var handle_field_completion = handle_auto_completion(value.dependencies);
|
|
|
|
$("#" + key).autocomplete({
|
|
source: function (request, response) {
|
|
var params = default_params(key, value.response && value.response.field, options.language);
|
|
var request_params = {q: request.term};
|
|
$.each(options.extra_params, function(key, param) {
|
|
request_params[key] = typeof param == "function" ? param() : param;
|
|
});
|
|
|
|
$.ajax({
|
|
type: "GET",
|
|
url: url,
|
|
dataType: "json",
|
|
data: $.extend(request_params, params()),
|
|
success: function(data) {
|
|
response($.map(data, function(item) {
|
|
return (create_parser(key, (value.response && value.response.format) || value.dependencies))(data)
|
|
}))
|
|
}
|
|
});
|
|
},
|
|
minChars:3,
|
|
delay:500,
|
|
appendTo: '.modal-content',
|
|
select: handle_field_completion
|
|
});
|
|
|
|
});
|
|
};
|
|
|
|
var nominatim = {
|
|
|
|
init : init
|
|
|
|
};
|
|
|
|
window['nominatim'] = nominatim;
|
|
|
|
})(jQuery);;/*
|
|
* Copyright (c) 2013 Kevin van Zonneveld (http://kvz.io)
|
|
* and Contributors (http://phpjs.org/authors)
|
|
*
|
|
* Permission is hereby granted, free of charge, to any person obtaining a copy of
|
|
* this software and associated documentation files (the "Software"), to deal in
|
|
* the Software without restriction, including without limitation the rights to
|
|
* use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies
|
|
* of the Software, and to permit persons to whom the Software is furnished to do
|
|
* so, subject to the following conditions:
|
|
*
|
|
* The above copyright notice and this permission notice shall be included in all
|
|
* copies or substantial portions of the Software.
|
|
*
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
* SOFTWARE.
|
|
*
|
|
*/
|
|
|
|
function phpjsDate(format, timestamp) {
|
|
// discuss at: http://phpjs.org/functions/date/
|
|
// original by: Carlos R. L. Rodrigues (http://www.jsfromhell.com)
|
|
// original by: gettimeofday
|
|
// parts by: Peter-Paul Koch (http://www.quirksmode.org/js/beat.html)
|
|
// improved by: Kevin van Zonneveld (http://kevin.vanzonneveld.net)
|
|
// improved by: MeEtc (http://yass.meetcweb.com)
|
|
// improved by: Brad Touesnard
|
|
// improved by: Tim Wiel
|
|
// improved by: Bryan Elliott
|
|
// improved by: David Randall
|
|
// improved by: Theriault
|
|
// improved by: Theriault
|
|
// improved by: Brett Zamir (http://brett-zamir.me)
|
|
// improved by: Theriault
|
|
// improved by: Thomas Beaucourt (http://www.webapp.fr)
|
|
// improved by: JT
|
|
// improved by: Theriault
|
|
// improved by: Rafal Kukawski (http://blog.kukawski.pl)
|
|
// improved by: Theriault
|
|
// input by: Brett Zamir (http://brett-zamir.me)
|
|
// input by: majak
|
|
// input by: Alex
|
|
// input by: Martin
|
|
// input by: Alex Wilson
|
|
// input by: Haravikk
|
|
// bugfixed by: Kevin van Zonneveld (http://kevin.vanzonneveld.net)
|
|
// bugfixed by: majak
|
|
// bugfixed by: Kevin van Zonneveld (http://kevin.vanzonneveld.net)
|
|
// bugfixed by: Brett Zamir (http://brett-zamir.me)
|
|
// bugfixed by: omid (http://phpjs.org/functions/380:380#comment_137122)
|
|
// bugfixed by: Chris (http://www.devotis.nl/)
|
|
// note: Uses global: php_js to store the default timezone
|
|
// note: Although the function potentially allows timezone info (see notes), it currently does not set
|
|
// note: per a timezone specified by date_default_timezone_set(). Implementers might use
|
|
// note: this.php_js.currentTimezoneOffset and this.php_js.currentTimezoneDST set by that function
|
|
// note: in order to adjust the dates in this function (or our other date functions!) accordingly
|
|
// example 1: date('H:m:s \\m \\i\\s \\m\\o\\n\\t\\h', 1062402400);
|
|
// returns 1: '09:09:40 m is month'
|
|
// example 2: date('F j, Y, g:i a', 1062462400);
|
|
// returns 2: 'September 2, 2003, 2:26 am'
|
|
// example 3: date('Y W o', 1062462400);
|
|
// returns 3: '2003 36 2003'
|
|
// example 4: x = date('Y m d', (new Date()).getTime()/1000);
|
|
// example 4: (x+'').length == 10 // 2009 01 09
|
|
// returns 4: true
|
|
// example 5: date('W', 1104534000);
|
|
// returns 5: '53'
|
|
// example 6: date('B t', 1104534000);
|
|
// returns 6: '999 31'
|
|
// example 7: date('W U', 1293750000.82); // 2010-12-31
|
|
// returns 7: '52 1293750000'
|
|
// example 8: date('W', 1293836400); // 2011-01-01
|
|
// returns 8: '52'
|
|
// example 9: date('W Y-m-d', 1293974054); // 2011-01-02
|
|
// returns 9: '52 2011-01-02'
|
|
|
|
var that = this;
|
|
var jsdate, f;
|
|
// Keep this here (works, but for code commented-out below for file size reasons)
|
|
// var tal= [];
|
|
var txt_words = [
|
|
'Sun', 'Mon', 'Tues', 'Wednes', 'Thurs', 'Fri', 'Satur',
|
|
'January', 'February', 'March', 'April', 'May', 'June',
|
|
'July', 'August', 'September', 'October', 'November', 'December'
|
|
];
|
|
// trailing backslash -> (dropped)
|
|
// a backslash followed by any character (including backslash) -> the character
|
|
// empty string -> empty string
|
|
var formatChr = /\\?(.?)/gi;
|
|
var formatChrCb = function(t, s) {
|
|
return f[t] ? f[t]() : s;
|
|
};
|
|
var _pad = function(n, c) {
|
|
n = String(n);
|
|
while (n.length < c) {
|
|
n = '0' + n;
|
|
}
|
|
return n;
|
|
};
|
|
f = {
|
|
// Day
|
|
d : function() {
|
|
// Day of month w/leading 0; 01..31
|
|
return _pad(f.j(), 2);
|
|
},
|
|
D : function() {
|
|
// Shorthand day name; Mon...Sun
|
|
return f.l()
|
|
.slice(0, 3);
|
|
},
|
|
j : function() {
|
|
// Day of month; 1..31
|
|
return jsdate.getDate();
|
|
},
|
|
l : function() {
|
|
// Full day name; Monday...Sunday
|
|
return txt_words[f.w()] + 'day';
|
|
},
|
|
N : function() {
|
|
// ISO-8601 day of week; 1[Mon]..7[Sun]
|
|
return f.w() || 7;
|
|
},
|
|
S : function() {
|
|
// Ordinal suffix for day of month; st, nd, rd, th
|
|
var j = f.j();
|
|
var i = j % 10;
|
|
if (i <= 3 && parseInt((j % 100) / 10, 10) == 1) {
|
|
i = 0;
|
|
}
|
|
return ['st', 'nd', 'rd'][i - 1] || 'th';
|
|
},
|
|
w : function() {
|
|
// Day of week; 0[Sun]..6[Sat]
|
|
return jsdate.getDay();
|
|
},
|
|
z : function() {
|
|
// Day of year; 0..365
|
|
var a = new Date(f.Y(), f.n() - 1, f.j());
|
|
var b = new Date(f.Y(), 0, 1);
|
|
return Math.round((a - b) / 864e5);
|
|
},
|
|
|
|
// Week
|
|
W : function() {
|
|
// ISO-8601 week number
|
|
var a = new Date(f.Y(), f.n() - 1, f.j() - f.N() + 3);
|
|
var b = new Date(a.getFullYear(), 0, 4);
|
|
return _pad(1 + Math.round((a - b) / 864e5 / 7), 2);
|
|
},
|
|
|
|
// Month
|
|
F : function() {
|
|
// Full month name; January...December
|
|
return txt_words[6 + f.n()];
|
|
},
|
|
m : function() {
|
|
// Month w/leading 0; 01...12
|
|
return _pad(f.n(), 2);
|
|
},
|
|
M : function() {
|
|
// Shorthand month name; Jan...Dec
|
|
return f.F()
|
|
.slice(0, 3);
|
|
},
|
|
n : function() {
|
|
// Month; 1...12
|
|
return jsdate.getMonth() + 1;
|
|
},
|
|
t : function() {
|
|
// Days in month; 28...31
|
|
return (new Date(f.Y(), f.n(), 0))
|
|
.getDate();
|
|
},
|
|
|
|
// Year
|
|
L : function() {
|
|
// Is leap year?; 0 or 1
|
|
var j = f.Y();
|
|
return j % 4 === 0 & j % 100 !== 0 | j % 400 === 0;
|
|
},
|
|
o : function() {
|
|
// ISO-8601 year
|
|
var n = f.n();
|
|
var W = f.W();
|
|
var Y = f.Y();
|
|
return Y + (n === 12 && W < 9 ? 1 : n === 1 && W > 9 ? -1 : 0);
|
|
},
|
|
Y : function() {
|
|
// Full year; e.g. 1980...2010
|
|
return jsdate.getFullYear();
|
|
},
|
|
y : function() {
|
|
// Last two digits of year; 00...99
|
|
return f.Y()
|
|
.toString()
|
|
.slice(-2);
|
|
},
|
|
|
|
// Time
|
|
a : function() {
|
|
// am or pm
|
|
return jsdate.getHours() > 11 ? 'pm' : 'am';
|
|
},
|
|
A : function() {
|
|
// AM or PM
|
|
return f.a()
|
|
.toUpperCase();
|
|
},
|
|
B : function() {
|
|
// Swatch Internet time; 000..999
|
|
var H = jsdate.getUTCHours() * 36e2;
|
|
// Hours
|
|
var i = jsdate.getUTCMinutes() * 60;
|
|
// Minutes
|
|
// Seconds
|
|
var s = jsdate.getUTCSeconds();
|
|
return _pad(Math.floor((H + i + s + 36e2) / 86.4) % 1e3, 3);
|
|
},
|
|
g : function() {
|
|
// 12-Hours; 1..12
|
|
return f.G() % 12 || 12;
|
|
},
|
|
G : function() {
|
|
// 24-Hours; 0..23
|
|
return jsdate.getHours();
|
|
},
|
|
h : function() {
|
|
// 12-Hours w/leading 0; 01..12
|
|
return _pad(f.g(), 2);
|
|
},
|
|
H : function() {
|
|
// 24-Hours w/leading 0; 00..23
|
|
return _pad(f.G(), 2);
|
|
},
|
|
i : function() {
|
|
// Minutes w/leading 0; 00..59
|
|
return _pad(jsdate.getMinutes(), 2);
|
|
},
|
|
s : function() {
|
|
// Seconds w/leading 0; 00..59
|
|
return _pad(jsdate.getSeconds(), 2);
|
|
},
|
|
u : function() {
|
|
// Microseconds; 000000-999000
|
|
return _pad(jsdate.getMilliseconds() * 1000, 6);
|
|
},
|
|
|
|
// Timezone
|
|
e : function() {
|
|
// Timezone identifier; e.g. Atlantic/Azores, ...
|
|
// The following works, but requires inclusion of the very large
|
|
// timezone_abbreviations_list() function.
|
|
/* return that.date_default_timezone_get();
|
|
*/
|
|
throw 'Not supported (see source code of date() for timezone on how to add support)';
|
|
},
|
|
I : function() {
|
|
// DST observed?; 0 or 1
|
|
// Compares Jan 1 minus Jan 1 UTC to Jul 1 minus Jul 1 UTC.
|
|
// If they are not equal, then DST is observed.
|
|
var a = new Date(f.Y(), 0);
|
|
// Jan 1
|
|
var c = Date.UTC(f.Y(), 0);
|
|
// Jan 1 UTC
|
|
var b = new Date(f.Y(), 6);
|
|
// Jul 1
|
|
// Jul 1 UTC
|
|
var d = Date.UTC(f.Y(), 6);
|
|
return ((a - c) !== (b - d)) ? 1 : 0;
|
|
},
|
|
O : function() {
|
|
// Difference to GMT in hour format; e.g. +0200
|
|
var tzo = jsdate.getTimezoneOffset();
|
|
var a = Math.abs(tzo);
|
|
return (tzo > 0 ? '-' : '+') + _pad(Math.floor(a / 60) * 100 + a % 60, 4);
|
|
},
|
|
P : function() {
|
|
// Difference to GMT w/colon; e.g. +02:00
|
|
var O = f.O();
|
|
return (O.substr(0, 3) + ':' + O.substr(3, 2));
|
|
},
|
|
T : function() {
|
|
// Timezone abbreviation; e.g. EST, MDT, ...
|
|
// The following works, but requires inclusion of the very
|
|
// large timezone_abbreviations_list() function.
|
|
/* var abbr, i, os, _default;
|
|
if (!tal.length) {
|
|
tal = that.timezone_abbreviations_list();
|
|
}
|
|
if (that.php_js && that.php_js.default_timezone) {
|
|
_default = that.php_js.default_timezone;
|
|
for (abbr in tal) {
|
|
for (i = 0; i < tal[abbr].length; i++) {
|
|
if (tal[abbr][i].timezone_id === _default) {
|
|
return abbr.toUpperCase();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
for (abbr in tal) {
|
|
for (i = 0; i < tal[abbr].length; i++) {
|
|
os = -jsdate.getTimezoneOffset() * 60;
|
|
if (tal[abbr][i].offset === os) {
|
|
return abbr.toUpperCase();
|
|
}
|
|
}
|
|
}
|
|
*/
|
|
return 'UTC';
|
|
},
|
|
Z : function() {
|
|
// Timezone offset in seconds (-43200...50400)
|
|
return -jsdate.getTimezoneOffset() * 60;
|
|
},
|
|
|
|
// Full Date/Time
|
|
c : function() {
|
|
// ISO-8601 date.
|
|
return 'Y-m-d\\TH:i:sP'.replace(formatChr, formatChrCb);
|
|
},
|
|
r : function() {
|
|
// RFC 2822
|
|
return 'D, d M Y H:i:s O'.replace(formatChr, formatChrCb);
|
|
},
|
|
U : function() {
|
|
// Seconds since UNIX epoch
|
|
return jsdate / 1000 | 0;
|
|
}
|
|
};
|
|
this.date = function(format, timestamp) {
|
|
that = this;
|
|
jsdate = (timestamp === undefined ? new Date() : // Not provided
|
|
(timestamp instanceof Date) ? new Date(timestamp) : // JS Date()
|
|
new Date(timestamp * 1000) // UNIX timestamp (auto-convert to int)
|
|
);
|
|
return format.replace(formatChr, formatChrCb);
|
|
};
|
|
return this.date(format, timestamp);
|
|
} |