|
|
/*!
|
|
|
* jQuery JavaScript Library v2.1.1-pre
|
|
|
* http://jquery.com/
|
|
|
*
|
|
|
* Includes Sizzle.js
|
|
|
* http://sizzlejs.com/
|
|
|
*
|
|
|
* Copyright 2005, 2014 jQuery Foundation, Inc. and other contributors
|
|
|
* Released under the MIT license
|
|
|
* http://jquery.org/license
|
|
|
*
|
|
|
* Date: 2014-10-30T07:56Z
|
|
|
*/
|
|
|
|
|
|
(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 inherently posses a window with a document
|
|
|
// (such as Node.js), expose a jQuery-making 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 ) {
|
|
|
|
|
|
// Can't do this because several apps including ASP.NET trace
|
|
|
// the stack via arguments.caller.callee and Firefox dies if
|
|
|
// you try to trace through "use strict" call chains. (#13335)
|
|
|
// Support: Firefox 18+
|
|
|
//
|
|
|
|
|
|
var arr = [];
|
|
|
|
|
|
var slice = arr.slice;
|
|
|
|
|
|
var concat = arr.concat;
|
|
|
|
|
|
var push = arr.push;
|
|
|
|
|
|
var indexOf = arr.indexOf;
|
|
|
|
|
|
var class2type = {};
|
|
|
|
|
|
var toString = class2type.toString;
|
|
|
|
|
|
var hasOwn = class2type.hasOwnProperty;
|
|
|
|
|
|
var support = {};
|
|
|
|
|
|
|
|
|
|
|
|
var
|
|
|
// Use the correct document accordingly with window argument (sandbox)
|
|
|
document = window.document,
|
|
|
|
|
|
version = "2.1.1-pre",
|
|
|
|
|
|
// 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
|
|
|
// 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.
|
|
|
// (You can seed the arguments with an array of args, but this is
|
|
|
// only used internally.)
|
|
|
each: function( callback, args ) {
|
|
|
return jQuery.each( this, callback, args );
|
|
|
},
|
|
|
|
|
|
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(null);
|
|
|
},
|
|
|
|
|
|
// For internal use only.
|
|
|
// Behaves like an Array's method, not like a jQuery method.
|
|
|
push: push,
|
|
|
sort: arr.sort,
|
|
|
splice: arr.splice
|
|
|
};
|
|
|
|
|
|
jQuery.extend = jQuery.fn.extend = function() {
|
|
|
var options, name, src, copy, copyIsArray, 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,
|
|
|
|
|
|
isWindow: function( obj ) {
|
|
|
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
|
|
|
return !jQuery.isArray( obj ) && obj - parseFloat( obj ) >= 0;
|
|
|
},
|
|
|
|
|
|
isPlainObject: function( obj ) {
|
|
|
// Not plain objects:
|
|
|
// - Any object or value whose internal [[Class]] property is not "[object Object]"
|
|
|
// - DOM nodes
|
|
|
// - window
|
|
|
if ( jQuery.type( obj ) !== "object" || obj.nodeType || jQuery.isWindow( obj ) ) {
|
|
|
return false;
|
|
|
}
|
|
|
|
|
|
if ( obj.constructor &&
|
|
|
!hasOwn.call( obj.constructor.prototype, "isPrototypeOf" ) ) {
|
|
|
return false;
|
|
|
}
|
|
|
|
|
|
// If the function hasn't returned already, we're confident that
|
|
|
// |obj| is a plain object, created by {} or constructed with new Object
|
|
|
return true;
|
|
|
},
|
|
|
|
|
|
isEmptyObject: function( obj ) {
|
|
|
var name;
|
|
|
for ( name in obj ) {
|
|
|
return false;
|
|
|
}
|
|
|
return true;
|
|
|
},
|
|
|
|
|
|
type: function( obj ) {
|
|
|
if ( obj == null ) {
|
|
|
return obj + "";
|
|
|
}
|
|
|
// Support: Android < 4.0, iOS < 6 (functionish RegExp)
|
|
|
return typeof obj === "object" || typeof obj === "function" ?
|
|
|
class2type[ toString.call(obj) ] || "object" :
|
|
|
typeof obj;
|
|
|
},
|
|
|
|
|
|
// Evaluates a script in a global context
|
|
|
globalEval: function( code ) {
|
|
|
var script,
|
|
|
indirect = eval;
|
|
|
|
|
|
code = jQuery.trim( code );
|
|
|
|
|
|
if ( code ) {
|
|
|
// If the code includes a valid, prologue position
|
|
|
// strict mode pragma, execute code by injecting a
|
|
|
// script tag into the document.
|
|
|
if ( code.indexOf("use strict") === 1 ) {
|
|
|
script = document.createElement("script");
|
|
|
script.text = code;
|
|
|
document.head.appendChild( script ).parentNode.removeChild( script );
|
|
|
} else {
|
|
|
// Otherwise, avoid the DOM node creation, insertion
|
|
|
// and removal by using an indirect global eval
|
|
|
indirect( code );
|
|
|
}
|
|
|
}
|
|
|
},
|
|
|
|
|
|
// 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();
|
|
|
},
|
|
|
|
|
|
// args is for internal usage only
|
|
|
each: function( obj, callback, args ) {
|
|
|
var value,
|
|
|
i = 0,
|
|
|
length = obj.length,
|
|
|
isArray = isArraylike( obj );
|
|
|
|
|
|
if ( args ) {
|
|
|
if ( isArray ) {
|
|
|
for ( ; i < length; i++ ) {
|
|
|
value = callback.apply( obj[ i ], args );
|
|
|
|
|
|
if ( value === false ) {
|
|
|
break;
|
|
|
}
|
|
|
}
|
|
|
} else {
|
|
|
for ( i in obj ) {
|
|
|
value = callback.apply( obj[ i ], args );
|
|
|
|
|
|
if ( value === false ) {
|
|
|
break;
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
|
|
|
// A special, fast, case for the most common use of each
|
|
|
} else {
|
|
|
if ( isArray ) {
|
|
|
for ( ; i < length; i++ ) {
|
|
|
value = callback.call( obj[ i ], i, obj[ i ] );
|
|
|
|
|
|
if ( value === false ) {
|
|
|
break;
|
|
|
}
|
|
|
}
|
|
|
} else {
|
|
|
for ( i in obj ) {
|
|
|
value = callback.call( obj[ i ], i, obj[ i ] );
|
|
|
|
|
|
if ( value === false ) {
|
|
|
break;
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
|
|
|
return obj;
|
|
|
},
|
|
|
|
|
|
// Support: Android<4.1
|
|
|
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 ) {
|
|
|
return arr == null ? -1 : indexOf.call( arr, elem, i );
|
|
|
},
|
|
|
|
|
|
merge: function( first, second ) {
|
|
|
var len = +second.length,
|
|
|
j = 0,
|
|
|
i = first.length;
|
|
|
|
|
|
for ( ; j < len; j++ ) {
|
|
|
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 value,
|
|
|
i = 0,
|
|
|
length = elems.length,
|
|
|
isArray = isArraylike( elems ),
|
|
|
ret = [];
|
|
|
|
|
|
// Go through the array, translating each of the items to their new values
|
|
|
if ( isArray ) {
|
|
|
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 tmp, args, proxy;
|
|
|
|
|
|
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: Date.now,
|
|
|
|
|
|
// jQuery.support is not used in Core but other projects attach their
|
|
|
// properties to it so it needs to exist.
|
|
|
support: support
|
|
|
});
|
|
|
|
|
|
// Populate the class2type map
|
|
|
jQuery.each("Boolean Number String Function Array Date RegExp Object Error".split(" "), function(i, name) {
|
|
|
class2type[ "[object " + name + "]" ] = name.toLowerCase();
|
|
|
});
|
|
|
|
|
|
function isArraylike( obj ) {
|
|
|
var length = obj.length,
|
|
|
type = jQuery.type( obj );
|
|
|
|
|
|
if ( type === "function" || jQuery.isWindow( obj ) ) {
|
|
|
return false;
|
|
|
}
|
|
|
|
|
|
if ( obj.nodeType === 1 && length ) {
|
|
|
return true;
|
|
|
}
|
|
|
|
|
|
return type === "array" || length === 0 ||
|
|
|
typeof length === "number" && length > 0 && ( length - 1 ) in obj;
|
|
|
}
|
|
|
var Sizzle =
|
|
|
/*!
|
|
|
* Sizzle CSS Selector Engine v1.10.19
|
|
|
* http://sizzlejs.com/
|
|
|
*
|
|
|
* Copyright 2013 jQuery Foundation, Inc. and other contributors
|
|
|
* Released under the MIT license
|
|
|
* http://jquery.org/license
|
|
|
*
|
|
|
* Date: 2014-04-18
|
|
|
*/
|
|
|
(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" + -(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
|
|
|
strundefined = typeof undefined,
|
|
|
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 if we can't use a native one
|
|
|
indexOf = arr.indexOf || function( elem ) {
|
|
|
var i = 0,
|
|
|
len = this.length;
|
|
|
for ( ; i < len; i++ ) {
|
|
|
if ( this[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
|
|
|
|
|
|
// Whitespace characters http://www.w3.org/TR/css3-selectors/#whitespace
|
|
|
whitespace = "[\\x20\\t\\r\\n\\f]",
|
|
|
// http://www.w3.org/TR/css3-syntax/#characters
|
|
|
characterEncoding = "(?:\\\\.|[\\w-]|[^\\x00-\\xa0])+",
|
|
|
|
|
|
// Loosely modeled on CSS identifier characters
|
|
|
// An unquoted value should be a CSS identifier http://www.w3.org/TR/css3-selectors/#attribute-selectors
|
|
|
// Proper syntax: http://www.w3.org/TR/CSS21/syndata.html#value-def-identifier
|
|
|
identifier = characterEncoding.replace( "w", "w#" ),
|
|
|
|
|
|
// Attribute selectors: http://www.w3.org/TR/selectors/#attribute-selectors
|
|
|
attributes = "\\[" + whitespace + "*(" + characterEncoding + ")(?:" + whitespace +
|
|
|
// Operator (capture 2)
|
|
|
"*([*^$|!~]?=)" + whitespace +
|
|
|
// "Attribute values must be CSS identifiers [capture 5] or strings [capture 3 or capture 4]"
|
|
|
"*(?:'((?:\\\\.|[^\\\\'])*)'|\"((?:\\\\.|[^\\\\\"])*)\"|(" + identifier + "))|)" + whitespace +
|
|
|
"*\\]",
|
|
|
|
|
|
pseudos = ":(" + characterEncoding + ")(?:\\((" +
|
|
|
// 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
|
|
|
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( "^#(" + characterEncoding + ")" ),
|
|
|
"CLASS": new RegExp( "^\\.(" + characterEncoding + ")" ),
|
|
|
"TAG": new RegExp( "^(" + characterEncoding.replace( "w", "w*" ) + ")" ),
|
|
|
"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 );
|
|
|
};
|
|
|
|
|
|
// 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 match, elem, m, nodeType,
|
|
|
// QSA vars
|
|
|
i, groups, old, nid, newContext, newSelector;
|
|
|
|
|
|
if ( ( context ? context.ownerDocument || context : preferredDoc ) !== document ) {
|
|
|
setDocument( context );
|
|
|
}
|
|
|
|
|
|
context = context || document;
|
|
|
results = results || [];
|
|
|
|
|
|
if ( !selector || typeof selector !== "string" ) {
|
|
|
return results;
|
|
|
}
|
|
|
|
|
|
if ( (nodeType = context.nodeType) !== 1 && nodeType !== 9 ) {
|
|
|
return [];
|
|
|
}
|
|
|
|
|
|
if ( documentIsHTML && !seed ) {
|
|
|
|
|
|
// Shortcuts
|
|
|
if ( (match = rquickExpr.exec( selector )) ) {
|
|
|
// Speed-up: Sizzle("#ID")
|
|
|
if ( (m = match[1]) ) {
|
|
|
if ( nodeType === 9 ) {
|
|
|
elem = context.getElementById( m );
|
|
|
// Check parentNode to catch when Blackberry 4.6 returns
|
|
|
// nodes that are no longer in the document (jQuery #6963)
|
|
|
if ( elem && elem.parentNode ) {
|
|
|
// Handle the case where IE, Opera, and Webkit return items
|
|
|
// by name instead of ID
|
|
|
if ( elem.id === m ) {
|
|
|
results.push( elem );
|
|
|
return results;
|
|
|
}
|
|
|
} else {
|
|
|
return results;
|
|
|
}
|
|
|
} else {
|
|
|
// Context is not a document
|
|
|
if ( context.ownerDocument && (elem = context.ownerDocument.getElementById( m )) &&
|
|
|
contains( context, elem ) && elem.id === m ) {
|
|
|
results.push( elem );
|
|
|
return results;
|
|
|
}
|
|
|
}
|
|
|
|
|
|
// Speed-up: Sizzle("TAG")
|
|
|
} else if ( match[2] ) {
|
|
|
push.apply( results, context.getElementsByTagName( selector ) );
|
|
|
return results;
|
|
|
|
|
|
// Speed-up: Sizzle(".CLASS")
|
|
|
} else if ( (m = match[3]) && support.getElementsByClassName && context.getElementsByClassName ) {
|
|
|
push.apply( results, context.getElementsByClassName( m ) );
|
|
|
return results;
|
|
|
}
|
|
|
}
|
|
|
|
|
|
// QSA path
|
|
|
if ( support.qsa && (!rbuggyQSA || !rbuggyQSA.test( selector )) ) {
|
|
|
nid = old = expando;
|
|
|
newContext = context;
|
|
|
newSelector = nodeType === 9 && selector;
|
|
|
|
|
|
// qSA works strangely on Element-rooted queries
|
|
|
// We can work around this by specifying an extra ID on the root
|
|
|
// and working up from there (Thanks to Andrew Dupont for the technique)
|
|
|
// IE 8 doesn't work on object elements
|
|
|
if ( nodeType === 1 && context.nodeName.toLowerCase() !== "object" ) {
|
|
|
groups = tokenize( selector );
|
|
|
|
|
|
if ( (old = context.getAttribute("id")) ) {
|
|
|
nid = old.replace( rescape, "\\$&" );
|
|
|
} else {
|
|
|
context.setAttribute( "id", nid );
|
|
|
}
|
|
|
nid = "[id='" + nid + "'] ";
|
|
|
|
|
|
i = groups.length;
|
|
|
while ( i-- ) {
|
|
|
groups[i] = nid + toSelector( groups[i] );
|
|
|
}
|
|
|
newContext = rsibling.test( selector ) && testContext( context.parentNode ) || context;
|
|
|
newSelector = groups.join(",");
|
|
|
}
|
|
|
|
|
|
if ( newSelector ) {
|
|
|
try {
|
|
|
push.apply( results,
|
|
|
newContext.querySelectorAll( newSelector )
|
|
|
);
|
|
|
return results;
|
|
|
} catch(qsaError) {
|
|
|
} finally {
|
|
|
if ( !old ) {
|
|
|
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 = attrs.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 !== strundefined && 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,
|
|
|
doc = node ? node.ownerDocument || node : preferredDoc,
|
|
|
parent = doc.defaultView;
|
|
|
|
|
|
// If no document and documentElement is available, return
|
|
|
if ( doc === document || doc.nodeType !== 9 || !doc.documentElement ) {
|
|
|
return document;
|
|
|
}
|
|
|
|
|
|
// Set our document
|
|
|
document = doc;
|
|
|
docElem = doc.documentElement;
|
|
|
|
|
|
// Support tests
|
|
|
documentIsHTML = !isXML( doc );
|
|
|
|
|
|
// Support: IE>8
|
|
|
// If iframe document is assigned to "document" variable and if iframe has been reloaded,
|
|
|
// IE will throw "permission denied" error when accessing "document" variable, see jQuery #13936
|
|
|
// IE6-8 do not support the defaultView property so parent will be undefined
|
|
|
if ( parent && parent !== parent.top ) {
|
|
|
// IE11 does not have attachEvent, so all must suffer
|
|
|
if ( parent.addEventListener ) {
|
|
|
parent.addEventListener( "unload", function() {
|
|
|
setDocument();
|
|
|
}, false );
|
|
|
} else if ( parent.attachEvent ) {
|
|
|
parent.attachEvent( "onunload", function() {
|
|
|
setDocument();
|
|
|
});
|
|
|
}
|
|
|
}
|
|
|
|
|
|
/* 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( doc.createComment("") );
|
|
|
return !div.getElementsByTagName("*").length;
|
|
|
});
|
|
|
|
|
|
// Check if getElementsByClassName can be trusted
|
|
|
support.getElementsByClassName = rnative.test( doc.getElementsByClassName ) && assert(function( div ) {
|
|
|
div.innerHTML = "<div class='a'></div><div class='a i'></div>";
|
|
|
|
|
|
// Support: Safari<4
|
|
|
// Catch class over-caching
|
|
|
div.firstChild.className = "i";
|
|
|
// Support: Opera<10
|
|
|
// Catch gEBCN failure to find non-leading classes
|
|
|
return div.getElementsByClassName("i").length === 2;
|
|
|
});
|
|
|
|
|
|
// 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 !doc.getElementsByName || !doc.getElementsByName( expando ).length;
|
|
|
});
|
|
|
|
|
|
// ID find and filter
|
|
|
if ( support.getById ) {
|
|
|
Expr.find["ID"] = function( id, context ) {
|
|
|
if ( typeof context.getElementById !== strundefined && documentIsHTML ) {
|
|
|
var m = context.getElementById( id );
|
|
|
// Check parentNode to catch when Blackberry 4.6 returns
|
|
|
// nodes that are no longer in the document #6963
|
|
|
return m && m.parentNode ? [ 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 !== strundefined && elem.getAttributeNode("id");
|
|
|
return node && node.value === attrId;
|
|
|
};
|
|
|
};
|
|
|
}
|
|
|
|
|
|
// Tag
|
|
|
Expr.find["TAG"] = support.getElementsByTagName ?
|
|
|
function( tag, context ) {
|
|
|
if ( typeof context.getElementsByTagName !== strundefined ) {
|
|
|
return context.getElementsByTagName( tag );
|
|
|
}
|
|
|
} :
|
|
|
function( tag, context ) {
|
|
|
var elem,
|
|
|
tmp = [],
|
|
|
i = 0,
|
|
|
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 !== strundefined && 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( doc.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
|
|
|
div.innerHTML = "<select msallowclip=''><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("[msallowclip^='']").length ) {
|
|
|
rbuggyQSA.push( "[*^$]=" + whitespace + "*(?:''|\"\")" );
|
|
|
}
|
|
|
|
|
|
// Support: IE8
|
|
|
// Boolean attributes and "value" are not treated correctly
|
|
|
if ( !div.querySelectorAll("[selected]").length ) {
|
|
|
rbuggyQSA.push( "\\[" + whitespace + "*(?:value|" + booleans + ")" );
|
|
|
}
|
|
|
|
|
|
// 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");
|
|
|
}
|
|
|
});
|
|
|
|
|
|
assert(function( div ) {
|
|
|
// Support: Windows 8 Native Apps
|
|
|
// The type and name attributes are restricted during .innerHTML assignment
|
|
|
var input = doc.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 does not implement inclusive descendent
|
|
|
// 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 === doc || a.ownerDocument === preferredDoc && contains(preferredDoc, a) ) {
|
|
|
return -1;
|
|
|
}
|
|
|
if ( b === doc || b.ownerDocument === preferredDoc && contains(preferredDoc, b) ) {
|
|
|
return 1;
|
|
|
}
|
|
|
|
|
|
// Maintain original order
|
|
|
return sortInput ?
|
|
|
( indexOf.call( sortInput, a ) - indexOf.call( 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 === doc ? -1 :
|
|
|
b === doc ? 1 :
|
|
|
aup ? -1 :
|
|
|
bup ? 1 :
|
|
|
sortInput ?
|
|
|
( indexOf.call( sortInput, a ) - indexOf.call( 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 doc;
|
|
|
};
|
|
|
|
|
|
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 &&
|
|
|
( !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 !== strundefined && 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 + " " ).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, outerCache, node, diff, nodeIndex, start,
|
|
|
dir = simple !== forward ? "nextSibling" : "previousSibling",
|
|
|
parent = elem.parentNode,
|
|
|
name = ofType && elem.nodeName.toLowerCase(),
|
|
|
useCache = !xml && !ofType;
|
|
|
|
|
|
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
|
|
|
outerCache = parent[ expando ] || (parent[ expando ] = {});
|
|
|
cache = outerCache[ type ] || [];
|
|
|
nodeIndex = cache[0] === dirruns && cache[1];
|
|
|
diff = cache[0] === dirruns && 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 ) {
|
|
|
outerCache[ type ] = [ dirruns, nodeIndex, diff ];
|
|
|
break;
|
|
|
}
|
|
|
}
|
|
|
|
|
|
// Use previously-cached element index if available
|
|
|
} else if ( useCache && (cache = (elem[ expando ] || (elem[ expando ] = {}))[ type ]) && cache[0] === dirruns ) {
|
|
|
diff = cache[1];
|
|
|
|
|
|
// xml :nth-child(...) or :nth-last-child(...) or :nth(-last)?-of-type(...)
|
|
|
} else {
|
|
|
// 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 ) {
|
|
|
(node[ expando ] || (node[ expando ] = {}))[ 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.call( 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 );
|
|
|
return !results.pop();
|
|
|
};
|
|
|
}),
|
|
|
|
|
|
"has": markFunction(function( selector ) {
|
|
|
return function( elem ) {
|
|
|
return Sizzle( selector, elem ).length > 0;
|
|
|
};
|
|
|
}),
|
|
|
|
|
|
"contains": markFunction(function( text ) {
|
|
|
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, outerCache,
|
|
|
newCache = [ dirruns, doneName ];
|
|
|
|
|
|
// We can't set arbitrary data on XML nodes, so they don't benefit from dir 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 ] = {});
|
|
|
if ( (oldCache = outerCache[ 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
|
|
|
outerCache[ 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.call( 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.call( checkContext, elem ) > -1;
|
|
|
}, implicitRelative, true ),
|
|
|
matchers = [ function( elem, context, xml ) {
|
|
|
return ( !leadingRelative && ( xml || context !== outermostContext ) ) || (
|
|
|
(checkContext = context).nodeType ?
|
|
|
matchContext( elem, context, xml ) :
|
|
|
matchAnyContext( elem, context, xml ) );
|
|
|
} ];
|
|
|
|
|
|
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;
|
|
|
}
|
|
|
|
|
|
// Add elements passing elementMatchers directly to results
|
|
|
// Keep `i` a string if there are no elements so `matchedCount` will be "00" below
|
|
|
// 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;
|
|
|
while ( (matcher = elementMatchers[j++]) ) {
|
|
|
if ( matcher( elem, context, 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 );
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
|
|
|
// Apply set filters to unmatched elements
|
|
|
matchedCount += i;
|
|
|
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 no seed and only one group
|
|
|
if ( match.length === 1 ) {
|
|
|
|
|
|
// Take a shortcut and set the context if the root 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,
|
|
|
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
|
|
|
// 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.unique = Sizzle.uniqueSort;
|
|
|
jQuery.text = Sizzle.getText;
|
|
|
jQuery.isXMLDoc = Sizzle.isXML;
|
|
|
jQuery.contains = Sizzle.contains;
|
|
|
|
|
|
|
|
|
|
|
|
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 ( indexOf.call( qualifier, elem ) >= 0 ) !== 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,
|
|
|
len = this.length,
|
|
|
ret = [],
|
|
|
self = this;
|
|
|
|
|
|
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 ) {
|
|
|
var match, elem;
|
|
|
|
|
|
// HANDLE: $(""), $(null), $(undefined), $(false)
|
|
|
if ( !selector ) {
|
|
|
return this;
|
|
|
}
|
|
|
|
|
|
// Handle HTML strings
|
|
|
if ( typeof selector === "string" ) {
|
|
|
if ( selector[0] === "<" && selector[ 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 ) {
|
|
|
// 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 || rootjQuery ).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 rootjQuery.ready !== "undefined" ?
|
|
|
rootjQuery.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.extend({
|
|
|
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;
|
|
|
},
|
|
|
|
|
|
sibling: function( n, elem ) {
|
|
|
var matched = [];
|
|
|
|
|
|
for ( ; n; n = n.nextSibling ) {
|
|
|
if ( n.nodeType === 1 && n !== elem ) {
|
|
|
matched.push( n );
|
|
|
}
|
|
|
}
|
|
|
|
|
|
return matched;
|
|
|
}
|
|
|
});
|
|
|
|
|
|
jQuery.fn.extend({
|
|
|
has: function( target ) {
|
|
|
var targets = jQuery( target, this ),
|
|
|
l = targets.length;
|
|
|
|
|
|
return this.filter(function() {
|
|
|
var i = 0;
|
|
|
for ( ; i < l; 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.unique( 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 indexOf.call( jQuery( elem ), this[ 0 ] );
|
|
|
}
|
|
|
|
|
|
// Locate the position of the desired element
|
|
|
return indexOf.call( this,
|
|
|
|
|
|
// If it receives a jQuery object, the first element is used
|
|
|
elem.jquery ? elem[ 0 ] : elem
|
|
|
);
|
|
|
},
|
|
|
|
|
|
add: function( selector, context ) {
|
|
|
return this.pushStack(
|
|
|
jQuery.unique(
|
|
|
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 ) {
|
|
|
while ( (cur = cur[dir]) && 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 jQuery.dir( elem, "parentNode" );
|
|
|
},
|
|
|
parentsUntil: function( elem, i, until ) {
|
|
|
return jQuery.dir( elem, "parentNode", until );
|
|
|
},
|
|
|
next: function( elem ) {
|
|
|
return sibling( elem, "nextSibling" );
|
|
|
},
|
|
|
prev: function( elem ) {
|
|
|
return sibling( elem, "previousSibling" );
|
|
|
},
|
|
|
nextAll: function( elem ) {
|
|
|
return jQuery.dir( elem, "nextSibling" );
|
|
|
},
|
|
|
prevAll: function( elem ) {
|
|
|
return jQuery.dir( elem, "previousSibling" );
|
|
|
},
|
|
|
nextUntil: function( elem, i, until ) {
|
|
|
return jQuery.dir( elem, "nextSibling", until );
|
|
|
},
|
|
|
prevUntil: function( elem, i, until ) {
|
|
|
return jQuery.dir( elem, "previousSibling", until );
|
|
|
},
|
|
|
siblings: function( elem ) {
|
|
|
return jQuery.sibling( ( elem.parentNode || {} ).firstChild, elem );
|
|
|
},
|
|
|
children: function( elem ) {
|
|
|
return jQuery.sibling( elem.firstChild );
|
|
|
},
|
|
|
contents: function( elem ) {
|
|
|
return elem.contentDocument || jQuery.merge( [], elem.childNodes );
|
|
|
}
|
|
|
}, function( name, fn ) {
|
|
|
jQuery.fn[ name ] = function( until, selector ) {
|
|
|
var matched = jQuery.map( this, fn, until );
|
|
|
|
|
|
if ( name.slice( -5 ) !== "Until" ) {
|
|
|
selector = until;
|
|
|
}
|
|
|
|
|
|
if ( selector && typeof selector === "string" ) {
|
|
|
matched = jQuery.filter( selector, matched );
|
|
|
}
|
|
|
|
|
|
if ( this.length > 1 ) {
|
|
|
// Remove duplicates
|
|
|
if ( !guaranteedUnique[ name ] ) {
|
|
|
jQuery.unique( matched );
|
|
|
}
|
|
|
|
|
|
// Reverse order for parents* and prev-derivatives
|
|
|
if ( rparentsprev.test( name ) ) {
|
|
|
matched.reverse();
|
|
|
}
|
|
|
}
|
|
|
|
|
|
return this.pushStack( matched );
|
|
|
};
|
|
|
});
|
|
|
var rnotwhite = (/\S+/g);
|
|
|
|
|
|
|
|
|
|
|
|
// String to Object options format cache
|
|
|
var optionsCache = {};
|
|
|
|
|
|
// Convert String-formatted options into Object-formatted ones and store in cache
|
|
|
function createOptions( options ) {
|
|
|
var object = optionsCache[ options ] = {};
|
|
|
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" ?
|
|
|
( optionsCache[ options ] || createOptions( options ) ) :
|
|
|
jQuery.extend( {}, options );
|
|
|
|
|
|
var // Last fire value (for non-forgettable lists)
|
|
|
memory,
|
|
|
// Flag to know if list was already fired
|
|
|
fired,
|
|
|
// Flag to know if list is currently firing
|
|
|
firing,
|
|
|
// First callback to fire (used internally by add and fireWith)
|
|
|
firingStart,
|
|
|
// End of the loop when firing
|
|
|
firingLength,
|
|
|
// Index of currently firing callback (modified by remove if needed)
|
|
|
firingIndex,
|
|
|
// Actual callback list
|
|
|
list = [],
|
|
|
// Stack of fire calls for repeatable lists
|
|
|
stack = !options.once && [],
|
|
|
// Fire callbacks
|
|
|
fire = function( data ) {
|
|
|
memory = options.memory && data;
|
|
|
fired = true;
|
|
|
firingIndex = firingStart || 0;
|
|
|
firingStart = 0;
|
|
|
firingLength = list.length;
|
|
|
firing = true;
|
|
|
for ( ; list && firingIndex < firingLength; firingIndex++ ) {
|
|
|
if ( list[ firingIndex ].apply( data[ 0 ], data[ 1 ] ) === false && options.stopOnFalse ) {
|
|
|
memory = false; // To prevent further calls using add
|
|
|
break;
|
|
|
}
|
|
|
}
|
|
|
firing = false;
|
|
|
if ( list ) {
|
|
|
if ( stack ) {
|
|
|
if ( stack.length ) {
|
|
|
fire( stack.shift() );
|
|
|
}
|
|
|
} else if ( memory ) {
|
|
|
list = [];
|
|
|
} else {
|
|
|
self.disable();
|
|
|
}
|
|
|
}
|
|
|
},
|
|
|
// Actual Callbacks object
|
|
|
self = {
|
|
|
// Add a callback or a collection of callbacks to the list
|
|
|
add: function() {
|
|
|
if ( list ) {
|
|
|
// First, we save the current length
|
|
|
var start = list.length;
|
|
|
(function add( args ) {
|
|
|
jQuery.each( args, function( _, arg ) {
|
|
|
var type = jQuery.type( arg );
|
|
|
if ( type === "function" ) {
|
|
|
if ( !options.unique || !self.has( arg ) ) {
|
|
|
list.push( arg );
|
|
|
}
|
|
|
} else if ( arg && arg.length && type !== "string" ) {
|
|
|
// Inspect recursively
|
|
|
add( arg );
|
|
|
}
|
|
|
});
|
|
|
})( arguments );
|
|
|
// Do we need to add the callbacks to the
|
|
|
// current firing batch?
|
|
|
if ( firing ) {
|
|
|
firingLength = list.length;
|
|
|
// With memory, if we're not firing then
|
|
|
// we should call right away
|
|
|
} else if ( memory ) {
|
|
|
firingStart = start;
|
|
|
fire( memory );
|
|
|
}
|
|
|
}
|
|
|
return this;
|
|
|
},
|
|
|
// Remove a callback from the list
|
|
|
remove: function() {
|
|
|
if ( list ) {
|
|
|
jQuery.each( arguments, function( _, arg ) {
|
|
|
var index;
|
|
|
while ( ( index = jQuery.inArray( arg, list, index ) ) > -1 ) {
|
|
|
list.splice( index, 1 );
|
|
|
// Handle firing indexes
|
|
|
if ( firing ) {
|
|
|
if ( index <= firingLength ) {
|
|
|
firingLength--;
|
|
|
}
|
|
|
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 && list.length );
|
|
|
},
|
|
|
// Remove all callbacks from the list
|
|
|
empty: function() {
|
|
|
list = [];
|
|
|
firingLength = 0;
|
|
|
return this;
|
|
|
},
|
|
|
// Have the list do nothing anymore
|
|
|
disable: function() {
|
|
|
list = stack = memory = undefined;
|
|
|
return this;
|
|
|
},
|
|
|
// Is it disabled?
|
|
|
disabled: function() {
|
|
|
return !list;
|
|
|
},
|
|
|
// Lock the list in its current state
|
|
|
lock: function() {
|
|
|
stack = undefined;
|
|
|
if ( !memory ) {
|
|
|
self.disable();
|
|
|
}
|
|
|
return this;
|
|
|
},
|
|
|
// Is it locked?
|
|
|
locked: function() {
|
|
|
return !stack;
|
|
|
},
|
|
|
// Call all callbacks with the given context and arguments
|
|
|
fireWith: function( context, args ) {
|
|
|
if ( list && ( !fired || stack ) ) {
|
|
|
args = args || [];
|
|
|
args = [ context, args.slice ? args.slice() : args ];
|
|
|
if ( firing ) {
|
|
|
stack.push( args );
|
|
|
} else {
|
|
|
fire( args );
|
|
|
}
|
|
|
}
|
|
|
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()
|
|
|
.done( newDefer.resolve )
|
|
|
.fail( newDefer.reject )
|
|
|
.progress( newDefer.notify );
|
|
|
} 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()
|
|
|
.done( updateFunc( i, resolveContexts, resolveValues ) )
|
|
|
.fail( deferred.reject )
|
|
|
.progress( updateFunc( i, progressContexts, progressValues ) );
|
|
|
} 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" );
|
|
|
}
|
|
|
}
|
|
|
});
|
|
|
|
|
|
/**
|
|
|
* The ready event handler and self cleanup method
|
|
|
*/
|
|
|
function completed() {
|
|
|
document.removeEventListener( "DOMContentLoaded", completed, false );
|
|
|
window.removeEventListener( "load", completed, false );
|
|
|
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.
|
|
|
// we once tried to use readyState "interactive" here, but it caused issues like the one
|
|
|
// discovered by ChrisS here: http://bugs.jquery.com/ticket/12282#comment:15
|
|
|
if ( document.readyState === "complete" ) {
|
|
|
// Handle it asynchronously to allow scripts the opportunity to delay ready
|
|
|
setTimeout( jQuery.ready );
|
|
|
|
|
|
} else {
|
|
|
|
|
|
// Use the handy event callback
|
|
|
document.addEventListener( "DOMContentLoaded", completed, false );
|
|
|
|
|
|
// A fallback to window.onload, that will always work
|
|
|
window.addEventListener( "load", completed, false );
|
|
|
}
|
|
|
}
|
|
|
return readyList.promise( obj );
|
|
|
};
|
|
|
|
|
|
// Kick off the DOM ready check even if the user does not
|
|
|
jQuery.ready.promise();
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Multifunctional method to get and set values of a collection
|
|
|
// The value/s can optionally be executed if it's a function
|
|
|
var access = jQuery.access = function( elems, fn, key, value, chainable, emptyGet, raw ) {
|
|
|
var i = 0,
|
|
|
len = elems.length,
|
|
|
bulk = key == null;
|
|
|
|
|
|
// Sets many values
|
|
|
if ( jQuery.type( key ) === "object" ) {
|
|
|
chainable = true;
|
|
|
for ( i in key ) {
|
|
|
jQuery.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 < len; i++ ) {
|
|
|
fn( elems[i], key, raw ? value : value.call( elems[i], i, fn( elems[i], key ) ) );
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
|
|
|
return chainable ?
|
|
|
elems :
|
|
|
|
|
|
// Gets
|
|
|
bulk ?
|
|
|
fn.call( elems ) :
|
|
|
len ? fn( elems[0], key ) : emptyGet;
|
|
|
};
|
|
|
|
|
|
|
|
|
/**
|
|
|
* Determines whether an object can have data
|
|
|
*/
|
|
|
jQuery.acceptData = function( owner ) {
|
|
|
// Accepts only:
|
|
|
// - Node
|
|
|
// - Node.ELEMENT_NODE
|
|
|
// - Node.DOCUMENT_NODE
|
|
|
// - Object
|
|
|
// - Any
|
|
|
/* jshint -W018 */
|
|
|
return owner.nodeType === 1 || owner.nodeType === 9 || !( +owner.nodeType );
|
|
|
};
|
|
|
|
|
|
|
|
|
function Data() {
|
|
|
// Support: Android < 4,
|
|
|
// Old WebKit does not have Object.preventExtensions/freeze method,
|
|
|
// return new empty object instead with no [[set]] accessor
|
|
|
Object.defineProperty( this.cache = {}, 0, {
|
|
|
get: function() {
|
|
|
return {};
|
|
|
}
|
|
|
});
|
|
|
|
|
|
this.expando = jQuery.expando + Math.random();
|
|
|
}
|
|
|
|
|
|
Data.uid = 1;
|
|
|
Data.accepts = jQuery.acceptData;
|
|
|
|
|
|
Data.prototype = {
|
|
|
key: function( owner ) {
|
|
|
// We can accept data for non-element nodes in modern browsers,
|
|
|
// but we should not, see #8335.
|
|
|
// Always return the key for a frozen object.
|
|
|
if ( !Data.accepts( owner ) ) {
|
|
|
return 0;
|
|
|
}
|
|
|
|
|
|
var descriptor = {},
|
|
|
// Check if the owner object already has a cache key
|
|
|
unlock = owner[ this.expando ];
|
|
|
|
|
|
// If not, create one
|
|
|
if ( !unlock ) {
|
|
|
unlock = Data.uid++;
|
|
|
|
|
|
// Secure it in a non-enumerable, non-writable property
|
|
|
try {
|
|
|
descriptor[ this.expando ] = { value: unlock };
|
|
|
Object.defineProperties( owner, descriptor );
|
|
|
|
|
|
// Support: Android < 4
|
|
|
// Fallback to a less secure definition
|
|
|
} catch ( e ) {
|
|
|
descriptor[ this.expando ] = unlock;
|
|
|
jQuery.extend( owner, descriptor );
|
|
|
}
|
|
|
}
|
|
|
|
|
|
// Ensure the cache object
|
|
|
if ( !this.cache[ unlock ] ) {
|
|
|
this.cache[ unlock ] = {};
|
|
|
}
|
|
|
|
|
|
return unlock;
|
|
|
},
|
|
|
set: function( owner, data, value ) {
|
|
|
var prop,
|
|
|
// There may be an unlock assigned to this node,
|
|
|
// if there is no entry for this "owner", create one inline
|
|
|
// and set the unlock as though an owner entry had always existed
|
|
|
unlock = this.key( owner ),
|
|
|
cache = this.cache[ unlock ];
|
|
|
|
|
|
// Handle: [ owner, key, value ] args
|
|
|
if ( typeof data === "string" ) {
|
|
|
cache[ data ] = value;
|
|
|
|
|
|
// Handle: [ owner, { properties } ] args
|
|
|
} else {
|
|
|
// Fresh assignments by object are shallow copied
|
|
|
if ( jQuery.isEmptyObject( cache ) ) {
|
|
|
jQuery.extend( this.cache[ unlock ], data );
|
|
|
// Otherwise, copy the properties one-by-one to the cache object
|
|
|
} else {
|
|
|
for ( prop in data ) {
|
|
|
cache[ prop ] = data[ prop ];
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
return cache;
|
|
|
},
|
|
|
get: function( owner, key ) {
|
|
|
// Either a valid cache is found, or will be created.
|
|
|
// New caches will be created and the unlock returned,
|
|
|
// allowing direct access to the newly created
|
|
|
// empty data object. A valid owner object must be provided.
|
|
|
var cache = this.cache[ this.key( owner ) ];
|
|
|
|
|
|
return key === undefined ?
|
|
|
cache : cache[ key ];
|
|
|
},
|
|
|
access: function( owner, key, value ) {
|
|
|
var stored;
|
|
|
// In cases where either:
|
|
|
//
|
|
|
// 1. No key was specified
|
|
|
// 2. A string key was specified, but no value provided
|
|
|
//
|
|
|
// Take the "read" path and allow the get method to determine
|
|
|
// which value to return, respectively either:
|
|
|
//
|
|
|
// 1. The entire cache object
|
|
|
// 2. The data stored at the key
|
|
|
//
|
|
|
if ( key === undefined ||
|
|
|
((key && typeof key === "string") && value === undefined) ) {
|
|
|
|
|
|
stored = this.get( owner, key );
|
|
|
|
|
|
return stored !== undefined ?
|
|
|
stored : this.get( owner, jQuery.camelCase(key) );
|
|
|
}
|
|
|
|
|
|
// [*]When the key is not a string, or both a key and value
|
|
|
// are specified, set or extend (existing objects) with either:
|
|
|
//
|
|
|
// 1. An object of properties
|
|
|
// 2. A key and value
|
|
|
//
|
|
|
this.set( owner, key, value );
|
|
|
|
|
|
// Since the "set" path can have two possible entry points
|
|
|
// return the expected data based on which path was taken[*]
|
|
|
return value !== undefined ? value : key;
|
|
|
},
|
|
|
remove: function( owner, key ) {
|
|
|
var i, name, camel,
|
|
|
unlock = this.key( owner ),
|
|
|
cache = this.cache[ unlock ];
|
|
|
|
|
|
if ( key === undefined ) {
|
|
|
this.cache[ unlock ] = {};
|
|
|
|
|
|
} else {
|
|
|
// Support array or space separated string of keys
|
|
|
if ( jQuery.isArray( key ) ) {
|
|
|
// 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 = key.concat( key.map( jQuery.camelCase ) );
|
|
|
} else {
|
|
|
camel = jQuery.camelCase( key );
|
|
|
// Try the string as a key before any manipulation
|
|
|
if ( key in cache ) {
|
|
|
name = [ key, camel ];
|
|
|
} else {
|
|
|
// If a key with the spaces exists, use it.
|
|
|
// Otherwise, create an array by matching non-whitespace
|
|
|
name = camel;
|
|
|
name = name in cache ?
|
|
|
[ name ] : ( name.match( rnotwhite ) || [] );
|
|
|
}
|
|
|
}
|
|
|
|
|
|
i = name.length;
|
|
|
while ( i-- ) {
|
|
|
delete cache[ name[ i ] ];
|
|
|
}
|
|
|
}
|
|
|
},
|
|
|
hasData: function( owner ) {
|
|
|
return !jQuery.isEmptyObject(
|
|
|
this.cache[ owner[ this.expando ] ] || {}
|
|
|
);
|
|
|
},
|
|
|
discard: function( owner ) {
|
|
|
if ( owner[ this.expando ] ) {
|
|
|
delete this.cache[ owner[ this.expando ] ];
|
|
|
}
|
|
|
}
|
|
|
};
|
|
|
var data_priv = new Data();
|
|
|
|
|
|
var data_user = new Data();
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
Implementation Summary
|
|
|
|
|
|
1. Enforce API surface and semantic compatibility with 1.9.x branch
|
|
|
2. Improve the module's maintainability by reducing the storage
|
|
|
paths to a single mechanism.
|
|
|
3. Use the same single mechanism to support "private" and "user" data.
|
|
|
4. _Never_ expose "private" data to user code (TODO: Drop _data, _removeData)
|
|
|
5. Avoid exposing implementation details on user objects (eg. expando properties)
|
|
|
6. Provide a clear path for implementation upgrade to WeakMap in 2014
|
|
|
*/
|
|
|
var rbrace = /^(?:\{[\w\W]*\}|\[[\w\W]*\])$/,
|
|
|
rmultiDash = /([A-Z])/g;
|
|
|
|
|
|
function dataAttr( elem, key, data ) {
|
|
|
var name;
|
|
|
|
|
|
// If nothing was found internally, try to fetch any
|
|
|
// data from the HTML5 data-* attribute
|
|
|
if ( data === undefined && elem.nodeType === 1 ) {
|
|
|
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
|
|
|
data_user.set( elem, key, data );
|
|
|
} else {
|
|
|
data = undefined;
|
|
|
}
|
|
|
}
|
|
|
return data;
|
|
|
}
|
|
|
|
|
|
jQuery.extend({
|
|
|
hasData: function( elem ) {
|
|
|
return data_user.hasData( elem ) || data_priv.hasData( elem );
|
|
|
},
|
|
|
|
|
|
data: function( elem, name, data ) {
|
|
|
return data_user.access( elem, name, data );
|
|
|
},
|
|
|
|
|
|
removeData: function( elem, name ) {
|
|
|
data_user.remove( elem, name );
|
|
|
},
|
|
|
|
|
|
// TODO: Now that all calls to _data and _removeData have been replaced
|
|
|
// with direct calls to data_priv methods, these can be deprecated.
|
|
|
_data: function( elem, name, data ) {
|
|
|
return data_priv.access( elem, name, data );
|
|
|
},
|
|
|
|
|
|
_removeData: function( elem, name ) {
|
|
|
data_priv.remove( elem, name );
|
|
|
}
|
|
|
});
|
|
|
|
|
|
jQuery.fn.extend({
|
|
|
data: function( key, value ) {
|
|
|
var i, name, data,
|
|
|
elem = this[ 0 ],
|
|
|
attrs = elem && elem.attributes;
|
|
|
|
|
|
// Gets all values
|
|
|
if ( key === undefined ) {
|
|
|
if ( this.length ) {
|
|
|
data = data_user.get( elem );
|
|
|
|
|
|
if ( elem.nodeType === 1 && !data_priv.get( elem, "hasDataAttrs" ) ) {
|
|
|
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 ] );
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
data_priv.set( elem, "hasDataAttrs", true );
|
|
|
}
|
|
|
}
|
|
|
|
|
|
return data;
|
|
|
}
|
|
|
|
|
|
// Sets multiple values
|
|
|
if ( typeof key === "object" ) {
|
|
|
return this.each(function() {
|
|
|
data_user.set( this, key );
|
|
|
});
|
|
|
}
|
|
|
|
|
|
return access( this, function( value ) {
|
|
|
var data,
|
|
|
camelKey = jQuery.camelCase( key );
|
|
|
|
|
|
// The calling jQuery object (element matches) is not empty
|
|
|
// (and therefore has an element appears at this[ 0 ]) and the
|
|
|
// `value` parameter was not undefined. An empty jQuery object
|
|
|
// will result in `undefined` for elem = this[ 0 ] which will
|
|
|
// throw an exception if an attempt to read a data cache is made.
|
|
|
if ( elem && value === undefined ) {
|
|
|
// Attempt to get data from the cache
|
|
|
// with the key as-is
|
|
|
data = data_user.get( elem, key );
|
|
|
if ( data !== undefined ) {
|
|
|
return data;
|
|
|
}
|
|
|
|
|
|
// Attempt to get data from the cache
|
|
|
// with the key camelized
|
|
|
data = data_user.get( elem, camelKey );
|
|
|
if ( data !== undefined ) {
|
|
|
return data;
|
|
|
}
|
|
|
|
|
|
// Attempt to "discover" the data in
|
|
|
// HTML5 custom data-* attrs
|
|
|
data = dataAttr( elem, camelKey, undefined );
|
|
|
if ( data !== undefined ) {
|
|
|
return data;
|
|
|
}
|
|
|
|
|
|
// We tried really hard, but the data doesn't exist.
|
|
|
return;
|
|
|
}
|
|
|
|
|
|
// Set the data...
|
|
|
this.each(function() {
|
|
|
// First, attempt to store a copy or reference of any
|
|
|
// data that might've been store with a camelCased key.
|
|
|
var data = data_user.get( this, camelKey );
|
|
|
|
|
|
// For HTML5 data-* attribute interop, we have to
|
|
|
// store property names with dashes in a camelCase form.
|
|
|
// This might not apply to all properties...*
|
|
|
data_user.set( this, camelKey, value );
|
|
|
|
|
|
// *... In the case of properties that might _actually_
|
|
|
// have dashes, we need to also store a copy of that
|
|
|
// unchanged property.
|
|
|
if ( key.indexOf("-") !== -1 && data !== undefined ) {
|
|
|
data_user.set( this, key, value );
|
|
|
}
|
|
|
});
|
|
|
}, null, value, arguments.length > 1, null, true );
|
|
|
},
|
|
|
|
|
|
removeData: function( key ) {
|
|
|
return this.each(function() {
|
|
|
data_user.remove( this, key );
|
|
|
});
|
|
|
}
|
|
|
});
|
|
|
|
|
|
|
|
|
jQuery.extend({
|
|
|
queue: function( elem, type, data ) {
|
|
|
var queue;
|
|
|
|
|
|
if ( elem ) {
|
|
|
type = ( type || "fx" ) + "queue";
|
|
|
queue = data_priv.get( elem, type );
|
|
|
|
|
|
// Speed up dequeue by getting out quickly if this is just a lookup
|
|
|
if ( data ) {
|
|
|
if ( !queue || jQuery.isArray( data ) ) {
|
|
|
queue = data_priv.access( 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 data_priv.get( elem, key ) || data_priv.access( elem, key, {
|
|
|
empty: jQuery.Callbacks("once memory").add(function() {
|
|
|
data_priv.remove( elem, [ type + "queue", 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 = data_priv.get( elements[ i ], type + "queueHooks" );
|
|
|
if ( tmp && tmp.empty ) {
|
|
|
count++;
|
|
|
tmp.empty.add( resolve );
|
|
|
}
|
|
|
}
|
|
|
resolve();
|
|
|
return defer.promise( obj );
|
|
|
}
|
|
|
});
|
|
|
var pnum = (/[+-]?(?:\d*\.|)\d+(?:[eE][+-]?\d+|)/).source;
|
|
|
|
|
|
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 );
|
|
|
};
|
|
|
|
|
|
var rcheckableType = (/^(?:checkbox|radio)$/i);
|
|
|
|
|
|
|
|
|
|
|
|
(function() {
|
|
|
var fragment = document.createDocumentFragment(),
|
|
|
div = fragment.appendChild( document.createElement( "div" ) ),
|
|
|
input = document.createElement( "input" );
|
|
|
|
|
|
// #11217 - WebKit loses check when the name is after the checked attribute
|
|
|
// Support: Windows Web Apps (WWA)
|
|
|
// `name` and `type` need .setAttribute for WWA
|
|
|
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;
|
|
|
|
|
|
// Make sure textarea (and checkbox) defaultValue is properly cloned
|
|
|
// Support: IE9-IE11+
|
|
|
div.innerHTML = "<textarea>x</textarea>";
|
|
|
support.noCloneChecked = !!div.cloneNode( true ).lastChild.defaultValue;
|
|
|
})();
|
|
|
var strundefined = typeof undefined;
|
|
|
|
|
|
|
|
|
|
|
|
support.focusinBubbles = "onfocusin" in window;
|
|
|
|
|
|
|
|
|
var
|
|
|
rkeyEvent = /^key/,
|
|
|
rmouseEvent = /^(?:mouse|pointer|contextmenu)|click/,
|
|
|
rfocusMorph = /^(?:focusinfocus|focusoutblur)$/,
|
|
|
rtypenamespace = /^([^.]*)(?:\.(.+)|)$/;
|
|
|
|
|
|
function returnTrue() {
|
|
|
return true;
|
|
|
}
|
|
|
|
|
|
function returnFalse() {
|
|
|
return false;
|
|
|
}
|
|
|
|
|
|
function safeActiveElement() {
|
|
|
try {
|
|
|
return document.activeElement;
|
|
|
} catch ( err ) { }
|
|
|
}
|
|
|
|
|
|
/*
|
|
|
* 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 handleObjIn, eventHandle, tmp,
|
|
|
events, t, handleObj,
|
|
|
special, handlers, type, namespaces, origType,
|
|
|
elemData = data_priv.get( 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 !== strundefined && jQuery.event.triggered !== e.type ?
|
|
|
jQuery.event.dispatch.apply( elem, arguments ) : undefined;
|
|
|
};
|
|
|
}
|
|
|
|
|
|
// 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 if the special events handler returns false
|
|
|
if ( !special.setup || special.setup.call( elem, data, namespaces, eventHandle ) === false ) {
|
|
|
if ( elem.addEventListener ) {
|
|
|
elem.addEventListener( type, eventHandle, false );
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
|
|
|
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;
|
|
|
}
|
|
|
|
|
|
},
|
|
|
|
|
|
// Detach an event or set of events from an element
|
|
|
remove: function( elem, types, handler, selector, mappedTypes ) {
|
|
|
|
|
|
var j, origCount, tmp,
|
|
|
events, t, handleObj,
|
|
|
special, handlers, type, namespaces, origType,
|
|
|
elemData = data_priv.hasData( elem ) && data_priv.get( 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;
|
|
|
data_priv.remove( elem, "events" );
|
|
|
}
|
|
|
},
|
|
|
|
|
|
trigger: function( event, data, elem, onlyHandlers ) {
|
|
|
|
|
|
var i, cur, tmp, bubbleType, ontype, handle, special,
|
|
|
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(".") >= 0 ) {
|
|
|
// 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.namespace_re = 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 = ( data_priv.get( cur, "events" ) || {} )[ event.type ] && data_priv.get( cur, "handle" );
|
|
|
if ( handle ) {
|
|
|
handle.apply( cur, data );
|
|
|
}
|
|
|
|
|
|
// Native handler
|
|
|
handle = ontype && cur[ ontype ];
|
|
|
if ( handle && handle.apply && jQuery.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) &&
|
|
|
jQuery.acceptData( elem ) ) {
|
|
|
|
|
|
// Call a native DOM method on the target with the same name name as the event.
|
|
|
// Don't do default actions on window, that's where global variables be (#6170)
|
|
|
if ( ontype && jQuery.isFunction( 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;
|
|
|
elem[ type ]();
|
|
|
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 = ( data_priv.get( 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.namespace_re || event.namespace_re.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;
|
|
|
|
|
|
// Find delegate handlers
|
|
|
// Black-hole SVG <use> instance trees (#13180)
|
|
|
// Avoid non-left-click bubbling in Firefox (#3861)
|
|
|
if ( delegateCount && cur.nodeType && (!event.button || event.type !== "click") ) {
|
|
|
|
|
|
for ( ; cur !== this; cur = cur.parentNode || this ) {
|
|
|
|
|
|
// Don't process clicks on disabled elements (#6911, #8165, #11382, #11764)
|
|
|
if ( 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 ) >= 0 :
|
|
|
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;
|
|
|
},
|
|
|
|
|
|
// Includes some event props shared by KeyEvent and MouseEvent
|
|
|
props: "altKey bubbles cancelable ctrlKey currentTarget 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 offsetX offsetY pageX pageY screenX screenY toElement".split(" "),
|
|
|
filter: function( event, original ) {
|
|
|
var eventDoc, doc, body,
|
|
|
button = original.button;
|
|
|
|
|
|
// 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 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;
|
|
|
}
|
|
|
},
|
|
|
|
|
|
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: Cordova 2.5 (WebKit) (#13255)
|
|
|
// All events should have a target; Cordova deviceready doesn't
|
|
|
if ( !event.target ) {
|
|
|
event.target = document;
|
|
|
}
|
|
|
|
|
|
// Support: Safari 6.0+, Chrome < 28
|
|
|
// Target should not be a text node (#504, #13143)
|
|
|
if ( event.target.nodeType === 3 ) {
|
|
|
event.target = event.target.parentNode;
|
|
|
}
|
|
|
|
|
|
return fixHook.filter ? fixHook.filter( event, originalEvent ) : 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 ) {
|
|
|
this.focus();
|
|
|
return false;
|
|
|
}
|
|
|
},
|
|
|
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 ( this.type === "checkbox" && this.click && jQuery.nodeName( this, "input" ) ) {
|
|
|
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;
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
},
|
|
|
|
|
|
simulate: function( type, elem, event, bubble ) {
|
|
|
// Piggyback on a donor event to simulate a different one.
|
|
|
// Fake originalEvent to avoid donor's stopPropagation, but if the
|
|
|
// simulated event prevents default then we do the same on the donor.
|
|
|
var e = jQuery.extend(
|
|
|
new jQuery.Event(),
|
|
|
event,
|
|
|
{
|
|
|
type: type,
|
|
|
isSimulated: true,
|
|
|
originalEvent: {}
|
|
|
}
|
|
|
);
|
|
|
if ( bubble ) {
|
|
|
jQuery.event.trigger( e, null, elem );
|
|
|
} else {
|
|
|
jQuery.event.dispatch.call( elem, e );
|
|
|
}
|
|
|
if ( e.isDefaultPrevented() ) {
|
|
|
event.preventDefault();
|
|
|
}
|
|
|
}
|
|
|
};
|
|
|
|
|
|
jQuery.removeEvent = function( elem, type, handle ) {
|
|
|
if ( elem.removeEventListener ) {
|
|
|
elem.removeEventListener( type, handle, false );
|
|
|
}
|
|
|
};
|
|
|
|
|
|
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: 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 = {
|
|
|
isDefaultPrevented: returnFalse,
|
|
|
isPropagationStopped: returnFalse,
|
|
|
isImmediatePropagationStopped: returnFalse,
|
|
|
|
|
|
preventDefault: function() {
|
|
|
var e = this.originalEvent;
|
|
|
|
|
|
this.isDefaultPrevented = returnTrue;
|
|
|
|
|
|
if ( e && e.preventDefault ) {
|
|
|
e.preventDefault();
|
|
|
}
|
|
|
},
|
|
|
stopPropagation: function() {
|
|
|
var e = this.originalEvent;
|
|
|
|
|
|
this.isPropagationStopped = returnTrue;
|
|
|
|
|
|
if ( e && e.stopPropagation ) {
|
|
|
e.stopPropagation();
|
|
|
}
|
|
|
},
|
|
|
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
|
|
|
// Support: Chrome 15+
|
|
|
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 mousenter/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;
|
|
|
}
|
|
|
};
|
|
|
});
|
|
|
|
|
|
// Create "bubbling" focus and blur events
|
|
|
// Support: Firefox, Chrome, Safari
|
|
|
if ( !support.focusinBubbles ) {
|
|
|
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 ), true );
|
|
|
};
|
|
|
|
|
|
jQuery.event.special[ fix ] = {
|
|
|
setup: function() {
|
|
|
var doc = this.ownerDocument || this,
|
|
|
attaches = data_priv.access( doc, fix );
|
|
|
|
|
|
if ( !attaches ) {
|
|
|
doc.addEventListener( orig, handler, true );
|
|
|
}
|
|
|
data_priv.access( doc, fix, ( attaches || 0 ) + 1 );
|
|
|
},
|
|
|
teardown: function() {
|
|
|
var doc = this.ownerDocument || this,
|
|
|
attaches = data_priv.access( doc, fix ) - 1;
|
|
|
|
|
|
if ( !attaches ) {
|
|
|
doc.removeEventListener( orig, handler, true );
|
|
|
data_priv.remove( doc, fix );
|
|
|
|
|
|
} else {
|
|
|
data_priv.access( doc, fix, attaches );
|
|
|
}
|
|
|
}
|
|
|
};
|
|
|
});
|
|
|
}
|
|
|
|
|
|
jQuery.fn.extend({
|
|
|
|
|
|
on: function( types, selector, data, fn, /*INTERNAL*/ 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 ) {
|
|
|
this.on( type, selector, data, types[ type ], one );
|
|
|
}
|
|
|
return this;
|
|
|
}
|
|
|
|
|
|
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 this;
|
|
|
}
|
|
|
|
|
|
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 this.each( function() {
|
|
|
jQuery.event.add( this, types, fn, data, selector );
|
|
|
});
|
|
|
},
|
|
|
one: function( types, selector, data, fn ) {
|
|
|
return this.on( 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
|
|
|
rxhtmlTag = /<(?!area|br|col|embed|hr|img|input|link|meta|param)(([\w:]+)[^>]*)\/>/gi,
|
|
|
rtagName = /<([\w:]+)/,
|
|
|
rhtml = /<|&#?\w+;/,
|
|
|
rnoInnerhtml = /<(?:script|style|link)/i,
|
|
|
// checked="checked" or checked
|
|
|
rchecked = /checked\s*(?:[^=]|=\s*.checked.)/i,
|
|
|
rscriptType = /^$|\/(?:java|ecma)script/i,
|
|
|
rscriptTypeMasked = /^true\/(.*)/,
|
|
|
rcleanScript = /^\s*<!(?:\[CDATA\[|--)|(?:\]\]|--)>\s*$/g,
|
|
|
|
|
|
// We have to close these tags to support XHTML (#13200)
|
|
|
wrapMap = {
|
|
|
|
|
|
// Support: IE 9
|
|
|
option: [ 1, "<select multiple='multiple'>", "</select>" ],
|
|
|
|
|
|
thead: [ 1, "<table>", "</table>" ],
|
|
|
col: [ 2, "<table><colgroup>", "</colgroup></table>" ],
|
|
|
tr: [ 2, "<table><tbody>", "</tbody></table>" ],
|
|
|
td: [ 3, "<table><tbody><tr>", "</tr></tbody></table>" ],
|
|
|
|
|
|
_default: [ 0, "", "" ]
|
|
|
};
|
|
|
|
|
|
// Support: IE 9
|
|
|
wrapMap.optgroup = wrapMap.option;
|
|
|
|
|
|
wrapMap.tbody = wrapMap.tfoot = wrapMap.colgroup = wrapMap.caption = wrapMap.thead;
|
|
|
wrapMap.th = wrapMap.td;
|
|
|
|
|
|
// Support: 1.x compatibility
|
|
|
// 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 = (elem.getAttribute("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;
|
|
|
}
|
|
|
|
|
|
// Mark scripts as having already been evaluated
|
|
|
function setGlobalEval( elems, refElements ) {
|
|
|
var i = 0,
|
|
|
l = elems.length;
|
|
|
|
|
|
for ( ; i < l; i++ ) {
|
|
|
data_priv.set(
|
|
|
elems[ i ], "globalEval", !refElements || data_priv.get( refElements[ i ], "globalEval" )
|
|
|
);
|
|
|
}
|
|
|
}
|
|
|
|
|
|
function cloneCopyEvent( src, dest ) {
|
|
|
var i, l, type, pdataOld, pdataCur, udataOld, udataCur, events;
|
|
|
|
|
|
if ( dest.nodeType !== 1 ) {
|
|
|
return;
|
|
|
}
|
|
|
|
|
|
// 1. Copy private data: events, handlers, etc.
|
|
|
if ( data_priv.hasData( src ) ) {
|
|
|
pdataOld = data_priv.access( src );
|
|
|
pdataCur = data_priv.set( dest, pdataOld );
|
|
|
events = pdataOld.events;
|
|
|
|
|
|
if ( events ) {
|
|
|
delete pdataCur.handle;
|
|
|
pdataCur.events = {};
|
|
|
|
|
|
for ( type in events ) {
|
|
|
for ( i = 0, l = events[ type ].length; i < l; i++ ) {
|
|
|
jQuery.event.add( dest, type, events[ type ][ i ] );
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
|
|
|
// 2. Copy user data
|
|
|
if ( data_user.hasData( src ) ) {
|
|
|
udataOld = data_user.access( src );
|
|
|
udataCur = jQuery.extend( {}, udataOld );
|
|
|
|
|
|
data_user.set( dest, udataCur );
|
|
|
}
|
|
|
}
|
|
|
|
|
|
function getAll( context, tag ) {
|
|
|
var ret = context.getElementsByTagName ? context.getElementsByTagName( tag || "*" ) :
|
|
|
context.querySelectorAll ? context.querySelectorAll( tag || "*" ) :
|
|
|
[];
|
|
|
|
|
|
return tag === undefined || tag && jQuery.nodeName( context, tag ) ?
|
|
|
jQuery.merge( [ context ], ret ) :
|
|
|
ret;
|
|
|
}
|
|
|
|
|
|
// Support: IE >= 9
|
|
|
function fixInput( src, dest ) {
|
|
|
var nodeName = dest.nodeName.toLowerCase();
|
|
|
|
|
|
// Fails to persist the checked state of a cloned checkbox or radio button.
|
|
|
if ( nodeName === "input" && rcheckableType.test( src.type ) ) {
|
|
|
dest.checked = src.checked;
|
|
|
|
|
|
// Fails to return the selected option to the default selected state when cloning options
|
|
|
} else if ( nodeName === "input" || nodeName === "textarea" ) {
|
|
|
dest.defaultValue = src.defaultValue;
|
|
|
}
|
|
|
}
|
|
|
|
|
|
jQuery.extend({
|
|
|
clone: function( elem, dataAndEvents, deepDataAndEvents ) {
|
|
|
var i, l, srcElements, destElements,
|
|
|
clone = elem.cloneNode( true ),
|
|
|
inPage = jQuery.contains( elem.ownerDocument, elem );
|
|
|
|
|
|
// Support: IE >= 9
|
|
|
// Fix Cloning issues
|
|
|
if ( !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 );
|
|
|
|
|
|
for ( i = 0, l = srcElements.length; i < l; i++ ) {
|
|
|
fixInput( srcElements[ i ], 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, l = srcElements.length; i < l; i++ ) {
|
|
|
cloneCopyEvent( srcElements[ i ], destElements[ i ] );
|
|
|
}
|
|
|
} else {
|
|
|
cloneCopyEvent( elem, clone );
|
|
|
}
|
|
|
}
|
|
|
|
|
|
// Preserve script evaluation history
|
|
|
destElements = getAll( clone, "script" );
|
|
|
if ( destElements.length > 0 ) {
|
|
|
setGlobalEval( destElements, !inPage && getAll( elem, "script" ) );
|
|
|
}
|
|
|
|
|
|
// Return the cloned set
|
|
|
return clone;
|
|
|
},
|
|
|
|
|
|
buildFragment: function( elems, context, scripts, selection ) {
|
|
|
var elem, tmp, tag, wrap, contains, j,
|
|
|
fragment = context.createDocumentFragment(),
|
|
|
nodes = [],
|
|
|
i = 0,
|
|
|
l = elems.length;
|
|
|
|
|
|
for ( ; i < l; i++ ) {
|
|
|
elem = elems[ i ];
|
|
|
|
|
|
if ( elem || elem === 0 ) {
|
|
|
|
|
|
// Add nodes directly
|
|
|
if ( jQuery.type( elem ) === "object" ) {
|
|
|
// Support: QtWebKit
|
|
|
// jQuery.merge because push.apply(_, arraylike) throws
|
|
|
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 || fragment.appendChild( context.createElement("div") );
|
|
|
|
|
|
// Deserialize a standard representation
|
|
|
tag = ( rtagName.exec( elem ) || [ "", "" ] )[ 1 ].toLowerCase();
|
|
|
wrap = wrapMap[ tag ] || wrapMap._default;
|
|
|
tmp.innerHTML = wrap[ 1 ] + elem.replace( rxhtmlTag, "<$1></$2>" ) + wrap[ 2 ];
|
|
|
|
|
|
// Descend through wrappers to the right content
|
|
|
j = wrap[ 0 ];
|
|
|
while ( j-- ) {
|
|
|
tmp = tmp.lastChild;
|
|
|
}
|
|
|
|
|
|
// Support: QtWebKit
|
|
|
// jQuery.merge because push.apply(_, arraylike) throws
|
|
|
jQuery.merge( nodes, tmp.childNodes );
|
|
|
|
|
|
// Remember the top-level container
|
|
|
tmp = fragment.firstChild;
|
|
|
|
|
|
// Fixes #12346
|
|
|
// Support: Webkit, IE
|
|
|
tmp.textContent = "";
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
|
|
|
// Remove wrapper from fragment
|
|
|
fragment.textContent = "";
|
|
|
|
|
|
i = 0;
|
|
|
while ( (elem = nodes[ i++ ]) ) {
|
|
|
|
|
|
// #4087 - If origin and destination elements are the same, and this is
|
|
|
// that element, do not do anything
|
|
|
if ( selection && jQuery.inArray( elem, selection ) !== -1 ) {
|
|
|
continue;
|
|
|
}
|
|
|
|
|
|
contains = jQuery.contains( elem.ownerDocument, elem );
|
|
|
|
|
|
// Append to fragment
|
|
|
tmp = getAll( fragment.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 );
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
|
|
|
return fragment;
|
|
|
},
|
|
|
|
|
|
cleanData: function( elems ) {
|
|
|
var data, elem, type, key,
|
|
|
special = jQuery.event.special,
|
|
|
i = 0;
|
|
|
|
|
|
for ( ; (elem = elems[ i ]) !== undefined; i++ ) {
|
|
|
if ( jQuery.acceptData( elem ) ) {
|
|
|
key = elem[ data_priv.expando ];
|
|
|
|
|
|
if ( key && (data = data_priv.cache[ key ]) ) {
|
|
|
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 );
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
if ( data_priv.cache[ key ] ) {
|
|
|
// Discard any remaining `private` data
|
|
|
delete data_priv.cache[ key ];
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
// Discard any remaining `user` data
|
|
|
delete data_user.cache[ elem[ data_user.expando ] ];
|
|
|
}
|
|
|
}
|
|
|
});
|
|
|
|
|
|
jQuery.fn.extend({
|
|
|
text: function( value ) {
|
|
|
return access( this, function( value ) {
|
|
|
return value === undefined ?
|
|
|
jQuery.text( this ) :
|
|
|
this.empty().each(function() {
|
|
|
if ( this.nodeType === 1 || this.nodeType === 11 || this.nodeType === 9 ) {
|
|
|
this.textContent = value;
|
|
|
}
|
|
|
});
|
|
|
}, null, value, arguments.length );
|
|
|
},
|
|
|
|
|
|
append: function() {
|
|
|
return this.domManip( arguments, function( elem ) {
|
|
|
if ( this.nodeType === 1 || this.nodeType === 11 || this.nodeType === 9 ) {
|
|
|
var target = manipulationTarget( this, elem );
|
|
|
target.appendChild( elem );
|
|
|
}
|
|
|
});
|
|
|
},
|
|
|
|
|
|
prepend: function() {
|
|
|
return this.domManip( 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 this.domManip( arguments, function( elem ) {
|
|
|
if ( this.parentNode ) {
|
|
|
this.parentNode.insertBefore( elem, this );
|
|
|
}
|
|
|
});
|
|
|
},
|
|
|
|
|
|
after: function() {
|
|
|
return this.domManip( arguments, function( elem ) {
|
|
|
if ( this.parentNode ) {
|
|
|
this.parentNode.insertBefore( elem, this.nextSibling );
|
|
|
}
|
|
|
});
|
|
|
},
|
|
|
|
|
|
remove: function( selector, keepData /* Internal Use Only */ ) {
|
|
|
var elem,
|
|
|
elems = selector ? jQuery.filter( selector, this ) : this,
|
|
|
i = 0;
|
|
|
|
|
|
for ( ; (elem = elems[i]) != null; i++ ) {
|
|
|
if ( !keepData && elem.nodeType === 1 ) {
|
|
|
jQuery.cleanData( getAll( elem ) );
|
|
|
}
|
|
|
|
|
|
if ( elem.parentNode ) {
|
|
|
if ( keepData && jQuery.contains( elem.ownerDocument, elem ) ) {
|
|
|
setGlobalEval( getAll( elem, "script" ) );
|
|
|
}
|
|
|
elem.parentNode.removeChild( elem );
|
|
|
}
|
|
|
}
|
|
|
|
|
|
return this;
|
|
|
},
|
|
|
|
|
|
empty: function() {
|
|
|
var elem,
|
|
|
i = 0;
|
|
|
|
|
|
for ( ; (elem = this[i]) != null; i++ ) {
|
|
|
if ( elem.nodeType === 1 ) {
|
|
|
|
|
|
// Prevent memory leaks
|
|
|
jQuery.cleanData( getAll( elem, false ) );
|
|
|
|
|
|
// Remove any remaining nodes
|
|
|
elem.textContent = "";
|
|
|
}
|
|
|
}
|
|
|
|
|
|
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 && elem.nodeType === 1 ) {
|
|
|
return elem.innerHTML;
|
|
|
}
|
|
|
|
|
|
// See if we can take a shortcut and just use innerHTML
|
|
|
if ( typeof value === "string" && !rnoInnerhtml.test( value ) &&
|
|
|
!wrapMap[ ( rtagName.exec( value ) || [ "", "" ] )[ 1 ].toLowerCase() ] ) {
|
|
|
|
|
|
value = value.replace( rxhtmlTag, "<$1></$2>" );
|
|
|
|
|
|
try {
|
|
|
for ( ; i < l; i++ ) {
|
|
|
elem = this[ i ] || {};
|
|
|
|
|
|
// Remove element nodes and prevent memory leaks
|
|
|
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 arg = arguments[ 0 ];
|
|
|
|
|
|
// Make the changes, replacing each context element with the new content
|
|
|
this.domManip( arguments, function( elem ) {
|
|
|
arg = this.parentNode;
|
|
|
|
|
|
jQuery.cleanData( getAll( this ) );
|
|
|
|
|
|
if ( arg ) {
|
|
|
arg.replaceChild( elem, this );
|
|
|
}
|
|
|
});
|
|
|
|
|
|
// Force removal if there was no new content (e.g., from empty arguments)
|
|
|
return arg && (arg.length || arg.nodeType) ? this : this.remove();
|
|
|
},
|
|
|
|
|
|
detach: function( selector ) {
|
|
|
return this.remove( selector, true );
|
|
|
},
|
|
|
|
|
|
domManip: function( args, callback ) {
|
|
|
|
|
|
// Flatten any nested arrays
|
|
|
args = concat.apply( [], args );
|
|
|
|
|
|
var fragment, first, scripts, hasScripts, node, doc,
|
|
|
i = 0,
|
|
|
l = this.length,
|
|
|
set = this,
|
|
|
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 this.each(function( index ) {
|
|
|
var self = set.eq( index );
|
|
|
if ( isFunction ) {
|
|
|
args[ 0 ] = value.call( this, index, self.html() );
|
|
|
}
|
|
|
self.domManip( args, callback );
|
|
|
});
|
|
|
}
|
|
|
|
|
|
if ( l ) {
|
|
|
fragment = jQuery.buildFragment( args, this[ 0 ].ownerDocument, false, this );
|
|
|
first = fragment.firstChild;
|
|
|
|
|
|
if ( fragment.childNodes.length === 1 ) {
|
|
|
fragment = first;
|
|
|
}
|
|
|
|
|
|
if ( first ) {
|
|
|
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: QtWebKit
|
|
|
// jQuery.merge because push.apply(_, arraylike) throws
|
|
|
jQuery.merge( scripts, getAll( node, "script" ) );
|
|
|
}
|
|
|
}
|
|
|
|
|
|
callback.call( this[ 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 || "" ) &&
|
|
|
!data_priv.access( 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.textContent.replace( rcleanScript, "" ) );
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
|
|
|
return this;
|
|
|
}
|
|
|
});
|
|
|
|
|
|
jQuery.each({
|
|
|
appendTo: "append",
|
|
|
prependTo: "prepend",
|
|
|
insertBefore: "before",
|
|
|
insertAfter: "after",
|
|
|
replaceAll: "replaceWith"
|
|
|
}, function( name, original ) {
|
|
|
jQuery.fn[ name ] = function( selector ) {
|
|
|
var elems,
|
|
|
ret = [],
|
|
|
insert = jQuery( selector ),
|
|
|
last = insert.length - 1,
|
|
|
i = 0;
|
|
|
|
|
|
for ( ; i <= last; i++ ) {
|
|
|
elems = i === last ? this : this.clone( true );
|
|
|
jQuery( insert[ i ] )[ original ]( elems );
|
|
|
|
|
|
// Support: QtWebKit
|
|
|
// .get() because push.apply(_, arraylike) throws
|
|
|
push.apply( ret, elems.get() );
|
|
|
}
|
|
|
|
|
|
return this.pushStack( ret );
|
|
|
};
|
|
|
});
|
|
|
|
|
|
|
|
|
var iframe,
|
|
|
elemdisplay = {};
|
|
|
|
|
|
/**
|
|
|
* 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 style,
|
|
|
elem = jQuery( doc.createElement( name ) ).appendTo( doc.body ),
|
|
|
|
|
|
// getDefaultComputedStyle might be reliably used only on attached element
|
|
|
display = window.getDefaultComputedStyle && ( style = window.getDefaultComputedStyle( elem[ 0 ] ) ) ?
|
|
|
|
|
|
// Use of this method is a temporary fix (more like optmization) until something better comes along,
|
|
|
// since it was removed from specification and supported only in FF
|
|
|
style.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 ].contentDocument;
|
|
|
|
|
|
// 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 getStyles = function( elem ) {
|
|
|
return elem.ownerDocument.defaultView.getComputedStyle( elem, null );
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
function curCSS( elem, name, computed ) {
|
|
|
var width, minWidth, maxWidth, ret,
|
|
|
style = elem.style;
|
|
|
|
|
|
computed = computed || getStyles( elem );
|
|
|
|
|
|
// Support: IE9
|
|
|
// getPropertyValue is only needed for .css('filter') in IE9, see #12537
|
|
|
if ( computed ) {
|
|
|
ret = computed.getPropertyValue( name ) || computed[ name ];
|
|
|
}
|
|
|
|
|
|
if ( computed ) {
|
|
|
|
|
|
if ( ret === "" && !jQuery.contains( elem.ownerDocument, elem ) ) {
|
|
|
ret = jQuery.style( elem, name );
|
|
|
}
|
|
|
|
|
|
// Support: iOS < 6
|
|
|
// A tribute to the "awesome hack by Dean Edwards"
|
|
|
// iOS < 6 (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 ( 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;
|
|
|
}
|
|
|
}
|
|
|
|
|
|
return ret !== undefined ?
|
|
|
// Support: IE
|
|
|
// IE returns zIndex value as an integer.
|
|
|
ret + "" :
|
|
|
ret;
|
|
|
}
|
|
|
|
|
|
|
|
|
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.
|
|
|
// Since there are no other hooks for marginRight, remove the whole object.
|
|
|
delete this.get;
|
|
|
return;
|
|
|
}
|
|
|
|
|
|
// Hook needed; redefine it so that the support test is not executed again.
|
|
|
|
|
|
return (this.get = hookFn).apply( this, arguments );
|
|
|
}
|
|
|
};
|
|
|
}
|
|
|
|
|
|
|
|
|
(function() {
|
|
|
var pixelPositionVal, boxSizingReliableVal,
|
|
|
docElem = document.documentElement,
|
|
|
container = document.createElement( "div" ),
|
|
|
div = document.createElement( "div" );
|
|
|
|
|
|
if ( !div.style ) {
|
|
|
return;
|
|
|
}
|
|
|
|
|
|
div.style.backgroundClip = "content-box";
|
|
|
div.cloneNode( true ).style.backgroundClip = "";
|
|
|
support.clearCloneStyle = div.style.backgroundClip === "content-box";
|
|
|
|
|
|
container.style.cssText = "border:0;width:0;height:0;top:0;left:-9999px;margin-top:1px;" +
|
|
|
"position:absolute";
|
|
|
container.appendChild( div );
|
|
|
|
|
|
// Executing both pixelPosition & boxSizingReliable tests require only one layout
|
|
|
// so they're executed at the same time to save the second computation.
|
|
|
function computePixelPositionAndBoxSizingReliable() {
|
|
|
div.style.cssText =
|
|
|
// Support: Firefox<29, Android 2.3
|
|
|
// Vendor-prefix box-sizing
|
|
|
"-webkit-box-sizing:border-box;-moz-box-sizing:border-box;" +
|
|
|
"box-sizing:border-box;display:block;margin-top:1%;top:1%;" +
|
|
|
"border:1px;padding:1px;width:4px;position:absolute";
|
|
|
div.innerHTML = "";
|
|
|
docElem.appendChild( container );
|
|
|
|
|
|
var divStyle = window.getComputedStyle( div, null );
|
|
|
pixelPositionVal = divStyle.top !== "1%";
|
|
|
boxSizingReliableVal = divStyle.width === "4px";
|
|
|
|
|
|
docElem.removeChild( container );
|
|
|
}
|
|
|
|
|
|
// Support: node.js jsdom
|
|
|
// Don't assume that getComputedStyle is a property of the global object
|
|
|
if ( window.getComputedStyle ) {
|
|
|
jQuery.extend( support, {
|
|
|
pixelPosition: function() {
|
|
|
// This test is executed only once but we still do memoizing
|
|
|
// since we can use the boxSizingReliable pre-computing.
|
|
|
// No need to check if the test was already performed, though.
|
|
|
computePixelPositionAndBoxSizingReliable();
|
|
|
return pixelPositionVal;
|
|
|
},
|
|
|
boxSizingReliable: function() {
|
|
|
if ( boxSizingReliableVal == null ) {
|
|
|
computePixelPositionAndBoxSizingReliable();
|
|
|
}
|
|
|
return boxSizingReliableVal;
|
|
|
},
|
|
|
reliableMarginRight: function() {
|
|
|
// Support: Android 2.3
|
|
|
// Check if 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
|
|
|
// This support function is only executed once so no memoizing is needed.
|
|
|
var ret,
|
|
|
marginDiv = div.appendChild( document.createElement( "div" ) );
|
|
|
|
|
|
// Reset CSS: box-sizing; display; margin; border; padding
|
|
|
marginDiv.style.cssText = 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:0";
|
|
|
marginDiv.style.marginRight = marginDiv.style.width = "0";
|
|
|
div.style.width = "1px";
|
|
|
docElem.appendChild( container );
|
|
|
|
|
|
ret = !parseFloat( window.getComputedStyle( marginDiv, null ).marginRight );
|
|
|
|
|
|
docElem.removeChild( container );
|
|
|
|
|
|
return ret;
|
|
|
}
|
|
|
});
|
|
|
}
|
|
|
})();
|
|
|
|
|
|
|
|
|
// A method for quickly swapping in/out CSS properties to get correct calculations.
|
|
|
jQuery.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
|
|
|
// 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" ),
|
|
|
rrelNum = new RegExp( "^([+-])=(" + pnum + ")", "i" ),
|
|
|
|
|
|
cssShow = { position: "absolute", visibility: "hidden", display: "block" },
|
|
|
cssNormalTransform = {
|
|
|
letterSpacing: "0",
|
|
|
fontWeight: "400"
|
|
|
},
|
|
|
|
|
|
cssPrefixes = [ "Webkit", "O", "Moz", "ms" ];
|
|
|
|
|
|
// return a css property mapped to a potentially vendor prefixed property
|
|
|
function vendorPropName( style, name ) {
|
|
|
|
|
|
// shortcut for names that are not vendor prefixed
|
|
|
if ( name in style ) {
|
|
|
return name;
|
|
|
}
|
|
|
|
|
|
// check for vendor prefixed names
|
|
|
var capName = name[0].toUpperCase() + name.slice(1),
|
|
|
origName = name,
|
|
|
i = cssPrefixes.length;
|
|
|
|
|
|
while ( i-- ) {
|
|
|
name = cssPrefixes[ i ] + capName;
|
|
|
if ( name in style ) {
|
|
|
return name;
|
|
|
}
|
|
|
}
|
|
|
|
|
|
return origName;
|
|
|
}
|
|
|
|
|
|
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 = jQuery.css( elem, "boxSizing", false, styles ) === "border-box";
|
|
|
|
|
|
// 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";
|
|
|
}
|
|
|
|
|
|
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 ] = data_priv.get( 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 ] = data_priv.access( elem, "olddisplay", defaultDisplay(elem.nodeName) );
|
|
|
}
|
|
|
} else {
|
|
|
hidden = isHidden( elem );
|
|
|
|
|
|
if ( display !== "none" || !hidden ) {
|
|
|
data_priv.set( 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;
|
|
|
}
|
|
|
|
|
|
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: {
|
|
|
"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": "cssFloat"
|
|
|
},
|
|
|
|
|
|
// 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( style, 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 relative number strings (+= or -=) to relative numbers. #7345
|
|
|
if ( type === "string" && (ret = rrelNum.exec( value )) ) {
|
|
|
value = ( ret[1] + 1 ) * ret[2] + parseFloat( jQuery.css( elem, name ) );
|
|
|
// 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 'px' to the (except for certain CSS properties)
|
|
|
if ( type === "number" && !jQuery.cssNumber[ origName ] ) {
|
|
|
value += "px";
|
|
|
}
|
|
|
|
|
|
// Fixes #8908, it can be done more correctly by specifying 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 ) {
|
|
|
style[ name ] = value;
|
|
|
}
|
|
|
|
|
|
} 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 val, num, hooks,
|
|
|
origName = jQuery.camelCase( name );
|
|
|
|
|
|
// Make sure that we're working with the right name
|
|
|
name = jQuery.cssProps[ origName ] || ( jQuery.cssProps[ origName ] = vendorPropName( elem.style, 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 || jQuery.isNumeric( 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 ?
|
|
|
jQuery.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,
|
|
|
jQuery.css( elem, "boxSizing", false, styles ) === "border-box",
|
|
|
styles
|
|
|
) : 0
|
|
|
);
|
|
|
}
|
|
|
};
|
|
|
});
|
|
|
|
|
|
// Support: Android 2.3
|
|
|
jQuery.cssHooks.marginRight = addGetHookIf( support.reliableMarginRight,
|
|
|
function( elem, computed ) {
|
|
|
if ( computed ) {
|
|
|
// WebKit Bug 13343 - getComputedStyle returns wrong value for margin-right
|
|
|
// Work around by temporarily setting element display to inline-block
|
|
|
return jQuery.swap( elem, { "display": "inline-block" },
|
|
|
curCSS, [ elem, "marginRight" ] );
|
|
|
}
|
|
|
}
|
|
|
);
|
|
|
|
|
|
// 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 || "swing";
|
|
|
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;
|
|
|
|
|
|
if ( tween.elem[ tween.prop ] != null &&
|
|
|
(!tween.elem.style || 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.style && ( 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: IE9
|
|
|
// 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;
|
|
|
}
|
|
|
};
|
|
|
|
|
|
jQuery.fx = Tween.prototype.init;
|
|
|
|
|
|
// Back Compat <1.8 extension point
|
|
|
jQuery.fx.step = {};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
var
|
|
|
fxNow, timerId,
|
|
|
rfxtypes = /^(?:toggle|show|hide)$/,
|
|
|
rfxnum = new RegExp( "^(?:([+-])=|)(" + pnum + ")([a-z%]*)$", "i" ),
|
|
|
rrun = /queueHooks$/,
|
|
|
animationPrefilters = [ defaultPrefilter ],
|
|
|
tweeners = {
|
|
|
"*": [ function( prop, value ) {
|
|
|
var tween = this.createTween( prop, value ),
|
|
|
target = tween.cur(),
|
|
|
parts = rfxnum.exec( value ),
|
|
|
unit = parts && parts[ 3 ] || ( jQuery.cssNumber[ prop ] ? "" : "px" ),
|
|
|
|
|
|
// Starting value computation is required for potential unit mismatches
|
|
|
start = ( jQuery.cssNumber[ prop ] || unit !== "px" && +target ) &&
|
|
|
rfxnum.exec( jQuery.css( tween.elem, prop ) ),
|
|
|
scale = 1,
|
|
|
maxIterations = 20;
|
|
|
|
|
|
if ( start && start[ 3 ] !== unit ) {
|
|
|
// Trust units reported by jQuery.css
|
|
|
unit = unit || start[ 3 ];
|
|
|
|
|
|
// Make sure we update the tween properties later on
|
|
|
parts = parts || [];
|
|
|
|
|
|
// Iteratively approximate from a nonzero starting point
|
|
|
start = +target || 1;
|
|
|
|
|
|
do {
|
|
|
// If previous iteration zeroed out, double until we get *something*
|
|
|
// Use a string for doubling factor so we don't accidentally see scale as unchanged below
|
|
|
scale = scale || ".5";
|
|
|
|
|
|
// Adjust and apply
|
|
|
start = start / scale;
|
|
|
jQuery.style( tween.elem, prop, start + unit );
|
|
|
|
|
|
// Update scale, tolerating zero or NaN from tween.cur()
|
|
|
// And breaking the loop if scale is unchanged or perfect, or if we've just had enough
|
|
|
} while ( scale !== (scale = tween.cur() / target) && scale !== 1 && --maxIterations );
|
|
|
}
|
|
|
|
|
|
// Update tween properties
|
|
|
if ( parts ) {
|
|
|
start = tween.start = +start || +target || 0;
|
|
|
tween.unit = unit;
|
|
|
// If a +=/-= token was provided, we're doing a relative animation
|
|
|
tween.end = parts[ 1 ] ?
|
|
|
start + ( parts[ 1 ] + 1 ) * parts[ 2 ] :
|
|
|
+parts[ 2 ];
|
|
|
}
|
|
|
|
|
|
return tween;
|
|
|
} ]
|
|
|
};
|
|
|
|
|
|
// Animations created synchronously will run synchronously
|
|
|
function createFxNow() {
|
|
|
setTimeout(function() {
|
|
|
fxNow = undefined;
|
|
|
});
|
|
|
return ( fxNow = jQuery.now() );
|
|
|
}
|
|
|
|
|
|
// Generate parameters to create a standard animation
|
|
|
function genFx( type, includeWidth ) {
|
|
|
var which,
|
|
|
i = 0,
|
|
|
attrs = { height: type };
|
|
|
|
|
|
// 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 = ( tweeners[ prop ] || [] ).concat( 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 = data_priv.get( 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 IE9-10 do 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" ?
|
|
|
data_priv.get( elem, "olddisplay" ) || defaultDisplay( elem.nodeName ) : display;
|
|
|
|
|
|
if ( checkDisplay === "inline" && jQuery.css( elem, "float" ) === "none" ) {
|
|
|
style.display = "inline-block";
|
|
|
}
|
|
|
}
|
|
|
|
|
|
if ( opts.overflow ) {
|
|
|
style.overflow = "hidden";
|
|
|
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 = data_priv.access( 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;
|
|
|
|
|
|
data_priv.remove( 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 = animationPrefilters.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 ),
|
|
|
// 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: {} }, 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.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 = animationPrefilters[ index ].call( animation, elem, props, animation.opts );
|
|
|
if ( 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, {
|
|
|
|
|
|
tweener: function( props, callback ) {
|
|
|
if ( jQuery.isFunction( props ) ) {
|
|
|
callback = props;
|
|
|
props = [ "*" ];
|
|
|
} else {
|
|
|
props = props.split(" ");
|
|
|
}
|
|
|
|
|
|
var prop,
|
|
|
index = 0,
|
|
|
length = props.length;
|
|
|
|
|
|
for ( ; index < length ; index++ ) {
|
|
|
prop = props[ index ];
|
|
|
tweeners[ prop ] = tweeners[ prop ] || [];
|
|
|
tweeners[ prop ].unshift( callback );
|
|
|
}
|
|
|
},
|
|
|
|
|
|
prefilter: function( callback, prepend ) {
|
|
|
if ( prepend ) {
|
|
|
animationPrefilters.unshift( callback );
|
|
|
} else {
|
|
|
animationPrefilters.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 || data_priv.get( 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 = data_priv.get( 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 = data_priv.get( 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,
|
|
|
i = 0,
|
|
|
timers = jQuery.timers;
|
|
|
|
|
|
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 = setInterval( jQuery.fx.tick, jQuery.fx.interval );
|
|
|
}
|
|
|
};
|
|
|
|
|
|
jQuery.fx.stop = function() {
|
|
|
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://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 = setTimeout( next, time );
|
|
|
hooks.stop = function() {
|
|
|
clearTimeout( timeout );
|
|
|
};
|
|
|
});
|
|
|
};
|
|
|
|
|
|
|
|
|
(function() {
|
|
|
var input = document.createElement( "input" ),
|
|
|
select = document.createElement( "select" ),
|
|
|
opt = select.appendChild( document.createElement( "option" ) );
|
|
|
|
|
|
input.type = "checkbox";
|
|
|
|
|
|
// Support: iOS 5.1, Android 4.x, Android 2.3
|
|
|
// Check the default checkbox/radio value ("" on old WebKit; "on" elsewhere)
|
|
|
support.checkOn = input.value !== "";
|
|
|
|
|
|
// Must access the parent to make an option select properly
|
|
|
// Support: IE9, IE10
|
|
|
support.optSelected = opt.selected;
|
|
|
|
|
|
// 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;
|
|
|
|
|
|
// Check if an input maintains its value after becoming a radio
|
|
|
// Support: IE9, IE10
|
|
|
input = document.createElement( "input" );
|
|
|
input.value = "t";
|
|
|
input.type = "radio";
|
|
|
support.radioValue = input.value === "t";
|
|
|
})();
|
|
|
|
|
|
|
|
|
var nodeHook, boolHook,
|
|
|
attrHandle = jQuery.expr.attrHandle;
|
|
|
|
|
|
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 hooks, ret,
|
|
|
nType = elem.nodeType;
|
|
|
|
|
|
// don't get/set attributes on text, comment and attribute nodes
|
|
|
if ( !elem || nType === 3 || nType === 8 || nType === 2 ) {
|
|
|
return;
|
|
|
}
|
|
|
|
|
|
// Fallback to prop when attributes are not supported
|
|
|
if ( typeof elem.getAttribute === strundefined ) {
|
|
|
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 );
|
|
|
|
|
|
} else if ( hooks && "set" in hooks && (ret = hooks.set( elem, value, name )) !== undefined ) {
|
|
|
return ret;
|
|
|
|
|
|
} else {
|
|
|
elem.setAttribute( name, value + "" );
|
|
|
return value;
|
|
|
}
|
|
|
|
|
|
} else if ( hooks && "get" in hooks && (ret = hooks.get( elem, name )) !== null ) {
|
|
|
return ret;
|
|
|
|
|
|
} else {
|
|
|
ret = jQuery.find.attr( elem, name );
|
|
|
|
|
|
// Non-existent attributes return null, we normalize to undefined
|
|
|
return ret == null ?
|
|
|
undefined :
|
|
|
ret;
|
|
|
}
|
|
|
},
|
|
|
|
|
|
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
|
|
|
elem[ propName ] = false;
|
|
|
}
|
|
|
|
|
|
elem.removeAttribute( name );
|
|
|
}
|
|
|
}
|
|
|
},
|
|
|
|
|
|
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 IE6-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;
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
});
|
|
|
|
|
|
// 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 {
|
|
|
elem.setAttribute( name, name );
|
|
|
}
|
|
|
return name;
|
|
|
}
|
|
|
};
|
|
|
jQuery.each( jQuery.expr.match.bool.source.match( /\w+/g ), function( i, name ) {
|
|
|
var getter = attrHandle[ name ] || jQuery.find.attr;
|
|
|
|
|
|
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;
|
|
|
};
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
var rfocusable = /^(?:input|select|textarea|button)$/i;
|
|
|
|
|
|
jQuery.fn.extend({
|
|
|
prop: function( name, value ) {
|
|
|
return access( this, jQuery.prop, name, value, arguments.length > 1 );
|
|
|
},
|
|
|
|
|
|
removeProp: function( name ) {
|
|
|
return this.each(function() {
|
|
|
delete this[ jQuery.propFix[ name ] || name ];
|
|
|
});
|
|
|
}
|
|
|
});
|
|
|
|
|
|
jQuery.extend({
|
|
|
propFix: {
|
|
|
"for": "htmlFor",
|
|
|
"class": "className"
|
|
|
},
|
|
|
|
|
|
prop: function( elem, name, value ) {
|
|
|
var ret, hooks, notxml,
|
|
|
nType = elem.nodeType;
|
|
|
|
|
|
// don't get/set properties on text, comment and attribute nodes
|
|
|
if ( !elem || nType === 3 || nType === 8 || nType === 2 ) {
|
|
|
return;
|
|
|
}
|
|
|
|
|
|
notxml = nType !== 1 || !jQuery.isXMLDoc( elem );
|
|
|
|
|
|
if ( notxml ) {
|
|
|
// Fix name and attach hooks
|
|
|
name = jQuery.propFix[ name ] || name;
|
|
|
hooks = jQuery.propHooks[ name ];
|
|
|
}
|
|
|
|
|
|
if ( value !== undefined ) {
|
|
|
return hooks && "set" in hooks && (ret = hooks.set( elem, value, name )) !== undefined ?
|
|
|
ret :
|
|
|
( elem[ name ] = value );
|
|
|
|
|
|
} else {
|
|
|
return hooks && "get" in hooks && (ret = hooks.get( elem, name )) !== null ?
|
|
|
ret :
|
|
|
elem[ name ];
|
|
|
}
|
|
|
},
|
|
|
|
|
|
propHooks: {
|
|
|
tabIndex: {
|
|
|
get: function( elem ) {
|
|
|
return elem.hasAttribute( "tabindex" ) || rfocusable.test( elem.nodeName ) || elem.href ?
|
|
|
elem.tabIndex :
|
|
|
-1;
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
});
|
|
|
|
|
|
// Support: IE9+
|
|
|
// Selectedness for an option in an optgroup can be inaccurate
|
|
|
if ( !support.optSelected ) {
|
|
|
jQuery.propHooks.selected = {
|
|
|
get: function( elem ) {
|
|
|
var parent = elem.parentNode;
|
|
|
if ( parent && parent.parentNode ) {
|
|
|
parent.parentNode.selectedIndex;
|
|
|
}
|
|
|
return null;
|
|
|
}
|
|
|
};
|
|
|
}
|
|
|
|
|
|
jQuery.each([
|
|
|
"tabIndex",
|
|
|
"readOnly",
|
|
|
"maxLength",
|
|
|
"cellSpacing",
|
|
|
"cellPadding",
|
|
|
"rowSpan",
|
|
|
"colSpan",
|
|
|
"useMap",
|
|
|
"frameBorder",
|
|
|
"contentEditable"
|
|
|
], function() {
|
|
|
jQuery.propFix[ this.toLowerCase() ] = this;
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
var rclass = /[\t\r\n\f]/g;
|
|
|
|
|
|
jQuery.fn.extend({
|
|
|
addClass: function( value ) {
|
|
|
var classes, elem, cur, clazz, j, finalValue,
|
|
|
proceed = typeof value === "string" && value,
|
|
|
i = 0,
|
|
|
len = this.length;
|
|
|
|
|
|
if ( jQuery.isFunction( value ) ) {
|
|
|
return this.each(function( j ) {
|
|
|
jQuery( this ).addClass( value.call( this, j, this.className ) );
|
|
|
});
|
|
|
}
|
|
|
|
|
|
if ( proceed ) {
|
|
|
// The disjunction here is for better compressibility (see removeClass)
|
|
|
classes = ( value || "" ).match( rnotwhite ) || [];
|
|
|
|
|
|
for ( ; i < len; i++ ) {
|
|
|
elem = this[ i ];
|
|
|
cur = elem.nodeType === 1 && ( elem.className ?
|
|
|
( " " + elem.className + " " ).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 ( elem.className !== finalValue ) {
|
|
|
elem.className = finalValue;
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
|
|
|
return this;
|
|
|
},
|
|
|
|
|
|
removeClass: function( value ) {
|
|
|
var classes, elem, cur, clazz, j, finalValue,
|
|
|
proceed = arguments.length === 0 || typeof value === "string" && value,
|
|
|
i = 0,
|
|
|
len = this.length;
|
|
|
|
|
|
if ( jQuery.isFunction( value ) ) {
|
|
|
return this.each(function( j ) {
|
|
|
jQuery( this ).removeClass( value.call( this, j, this.className ) );
|
|
|
});
|
|
|
}
|
|
|
if ( proceed ) {
|
|
|
classes = ( value || "" ).match( rnotwhite ) || [];
|
|
|
|
|
|
for ( ; i < len; i++ ) {
|
|
|
elem = this[ i ];
|
|
|
// This expression is here for better compressibility (see addClass)
|
|
|
cur = elem.nodeType === 1 && ( elem.className ?
|
|
|
( " " + elem.className + " " ).replace( rclass, " " ) :
|
|
|
""
|
|
|
);
|
|
|
|
|
|
if ( cur ) {
|
|
|
j = 0;
|
|
|
while ( (clazz = classes[j++]) ) {
|
|
|
// Remove *all* instances
|
|
|
while ( cur.indexOf( " " + clazz + " " ) >= 0 ) {
|
|
|
cur = cur.replace( " " + clazz + " ", " " );
|
|
|
}
|
|
|
}
|
|
|
|
|
|
// only assign if different to avoid unneeded rendering.
|
|
|
finalValue = value ? jQuery.trim( cur ) : "";
|
|
|
if ( elem.className !== finalValue ) {
|
|
|
elem.className = 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, this.className, stateVal), stateVal );
|
|
|
});
|
|
|
}
|
|
|
|
|
|
return this.each(function() {
|
|
|
if ( type === "string" ) {
|
|
|
// toggle individual class names
|
|
|
var className,
|
|
|
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 ( type === strundefined || type === "boolean" ) {
|
|
|
if ( this.className ) {
|
|
|
// store className if set
|
|
|
data_priv.set( this, "__className__", this.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.
|
|
|
this.className = this.className || value === false ? "" : data_priv.get( this, "__className__" ) || "";
|
|
|
}
|
|
|
});
|
|
|
},
|
|
|
|
|
|
hasClass: function( selector ) {
|
|
|
var className = " " + selector + " ",
|
|
|
i = 0,
|
|
|
l = this.length;
|
|
|
for ( ; i < l; i++ ) {
|
|
|
if ( this[i].nodeType === 1 && (" " + this[i].className + " ").replace(rclass, " ").indexOf( className ) >= 0 ) {
|
|
|
return true;
|
|
|
}
|
|
|
}
|
|
|
|
|
|
return false;
|
|
|
}
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
var rreturn = /\r/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)
|
|
|
jQuery.trim( jQuery.text( elem ) );
|
|
|
}
|
|
|
},
|
|
|
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 ];
|
|
|
|
|
|
// IE6-9 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 ( (option.selected = jQuery.inArray( option.value, values ) >= 0) ) {
|
|
|
optionSet = true;
|
|
|
}
|
|
|
}
|
|
|
|
|
|
// force browsers to behave consistently when non-matching value is set
|
|
|
if ( !optionSet ) {
|
|
|
elem.selectedIndex = -1;
|
|
|
}
|
|
|
return values;
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
});
|
|
|
|
|
|
// 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 ) >= 0 );
|
|
|
}
|
|
|
}
|
|
|
};
|
|
|
if ( !support.checkOn ) {
|
|
|
jQuery.valHooks[ this ].get = function( elem ) {
|
|
|
// Support: Webkit
|
|
|
// "" is returned instead of "on" if a value isn't specified
|
|
|
return elem.getAttribute("value") === null ? "on" : elem.value;
|
|
|
};
|
|
|
}
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// 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 );
|
|
|
},
|
|
|
|
|
|
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 );
|
|
|
}
|
|
|
});
|
|
|
|
|
|
|
|
|
var nonce = jQuery.now();
|
|
|
|
|
|
var rquery = (/\?/);
|
|
|
|
|
|
|
|
|
|
|
|
// Support: Android 2.3
|
|
|
// Workaround failure to string-cast null input
|
|
|
jQuery.parseJSON = function( data ) {
|
|
|
return JSON.parse( data + "" );
|
|
|
};
|
|
|
|
|
|
|
|
|
// Cross-browser xml parsing
|
|
|
jQuery.parseXML = function( data ) {
|
|
|
var xml, tmp;
|
|
|
if ( !data || typeof data !== "string" ) {
|
|
|
return null;
|
|
|
}
|
|
|
|
|
|
// Support: IE9
|
|
|
try {
|
|
|
tmp = new DOMParser();
|
|
|
xml = tmp.parseFromString( data, "text/xml" );
|
|
|
} catch ( e ) {
|
|
|
xml = undefined;
|
|
|
}
|
|
|
|
|
|
if ( !xml || xml.getElementsByTagName( "parsererror" ).length ) {
|
|
|
jQuery.error( "Invalid XML: " + data );
|
|
|
}
|
|
|
return xml;
|
|
|
};
|
|
|
|
|
|
|
|
|
var
|
|
|
// Document location
|
|
|
ajaxLocParts,
|
|
|
ajaxLocation,
|
|
|
|
|
|
rhash = /#.*$/,
|
|
|
rts = /([?&])_=[^&]*/,
|
|
|
rheaders = /^(.*?):[ \t]*([^\r\n]*)$/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("*");
|
|
|
|
|
|
// #8138, IE may throw an exception when accessing
|
|
|
// a field from window.location if document.domain has been set
|
|
|
try {
|
|
|
ajaxLocation = location.href;
|
|
|
} catch( e ) {
|
|
|
// Use the href attribute of an A element
|
|
|
// since IE will modify it given document.location
|
|
|
ajaxLocation = document.createElement( "a" );
|
|
|
ajaxLocation.href = "";
|
|
|
ajaxLocation = ajaxLocation.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[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 key, deep,
|
|
|
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 ct, type, finalDataType, firstDataType,
|
|
|
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" ] ) {
|
|
|
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: /xml/,
|
|
|
html: /html/,
|
|
|
json: /json/
|
|
|
},
|
|
|
|
|
|
responseFields: {
|
|
|
xml: "responseXML",
|
|
|
text: "responseText",
|
|
|
json: "responseJSON",
|
|
|
native: "responseNative"
|
|
|
},
|
|
|
|
|
|
// 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,
|
|
|
|
|
|
// Don't convert a native response
|
|
|
"* native": true
|
|
|
},
|
|
|
|
|
|
// 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 transport,
|
|
|
// URL without anti-cache param
|
|
|
cacheURL,
|
|
|
// Response headers
|
|
|
responseHeadersString,
|
|
|
responseHeaders,
|
|
|
// timeout handle
|
|
|
timeoutTimer,
|
|
|
// Cross-domain detection vars
|
|
|
parts,
|
|
|
// To know if global events are to be dispatched
|
|
|
fireGlobals,
|
|
|
// Loop variable
|
|
|
i,
|
|
|
// 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 (prefilters might expect it)
|
|
|
// 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
|
|
|
fireGlobals = 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 ] );
|
|
|
}
|
|
|
// Timeout
|
|
|
if ( s.async && s.timeout > 0 ) {
|
|
|
timeoutTimer = 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 ) {
|
|
|
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;
|
|
|
}
|
|
|
|
|
|
return jQuery.ajax({
|
|
|
url: url,
|
|
|
type: method,
|
|
|
dataType: type,
|
|
|
data: data,
|
|
|
success: callback
|
|
|
});
|
|
|
};
|
|
|
});
|
|
|
|
|
|
// 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._evalUrl = function( url ) {
|
|
|
return jQuery.ajax({
|
|
|
url: url,
|
|
|
type: "GET",
|
|
|
dataType: "script",
|
|
|
async: false,
|
|
|
global: false,
|
|
|
"throws": true
|
|
|
});
|
|
|
};
|
|
|
|
|
|
|
|
|
jQuery.fn.extend({
|
|
|
wrapAll: function( html ) {
|
|
|
var wrap;
|
|
|
|
|
|
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
|
|
|
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.firstElementChild ) {
|
|
|
elem = elem.firstElementChild;
|
|
|
}
|
|
|
|
|
|
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();
|
|
|
}
|
|
|
});
|
|
|
|
|
|
|
|
|
jQuery.expr.filters.hidden = function( elem ) {
|
|
|
// Support: Opera <= 12.12
|
|
|
// Opera reports offsetWidths and offsetHeights less than zero on some elements
|
|
|
return elem.offsetWidth <= 0 && elem.offsetHeight <= 0;
|
|
|
};
|
|
|
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" ? 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();
|
|
|
}
|
|
|
});
|
|
|
|
|
|
|
|
|
jQuery.ajaxSettings.xhr = function() {
|
|
|
try {
|
|
|
return new XMLHttpRequest();
|
|
|
} catch( e ) {}
|
|
|
};
|
|
|
|
|
|
var xhrId = 0,
|
|
|
xhrCallbacks = {},
|
|
|
xhrSuccessStatus = {
|
|
|
// file protocol always yields status code 0, assume 200
|
|
|
0: 200,
|
|
|
// Support: IE9
|
|
|
// #1450: sometimes IE returns 1223 when it should be 204
|
|
|
1223: 204
|
|
|
},
|
|
|
xhrSupported = jQuery.ajaxSettings.xhr();
|
|
|
|
|
|
// Support: IE9
|
|
|
// Open requests must be manually aborted on unload (#5280)
|
|
|
if ( window.ActiveXObject ) {
|
|
|
jQuery( window ).on( "unload", function() {
|
|
|
for ( var key in xhrCallbacks ) {
|
|
|
xhrCallbacks[ key ]();
|
|
|
}
|
|
|
});
|
|
|
}
|
|
|
|
|
|
support.cors = !!xhrSupported && ( "withCredentials" in xhrSupported );
|
|
|
support.ajax = xhrSupported = !!xhrSupported;
|
|
|
|
|
|
jQuery.ajaxTransport(function( options ) {
|
|
|
var callback;
|
|
|
|
|
|
// Cross domain only allowed if supported through XMLHttpRequest
|
|
|
if ( support.cors || xhrSupported && !options.crossDomain ) {
|
|
|
return {
|
|
|
send: function( headers, complete ) {
|
|
|
var i,
|
|
|
xhr = options.xhr(),
|
|
|
id = ++xhrId,
|
|
|
responses = {};
|
|
|
|
|
|
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 ) {
|
|
|
xhr.setRequestHeader( i, headers[ i ] );
|
|
|
}
|
|
|
|
|
|
// Callback
|
|
|
callback = function( type ) {
|
|
|
return function() {
|
|
|
if ( callback ) {
|
|
|
delete xhrCallbacks[ id ];
|
|
|
callback = xhr.onload = xhr.onerror = null;
|
|
|
|
|
|
if ( type === "abort" ) {
|
|
|
xhr.abort();
|
|
|
} else if ( type === "error" ) {
|
|
|
complete(
|
|
|
// file: protocol always yields status 0; see #8605, #14207
|
|
|
xhr.status,
|
|
|
xhr.statusText
|
|
|
);
|
|
|
} else {
|
|
|
// Verify the responseType attribute to avoid InvalidStateError Exception (XHR2 Spec)
|
|
|
// Support: IE9
|
|
|
// Accessing binary-data responseText throws an exception
|
|
|
// (#11426)
|
|
|
if ( (!xhr.responseType || xhr.responseType === "text") &&
|
|
|
typeof xhr.responseText === "string" ) {
|
|
|
responses.text = xhr.responseText;
|
|
|
}
|
|
|
|
|
|
// The native response associated with the responseType
|
|
|
// Stored in the xhr.response attribute (XHR2 Spec)
|
|
|
if ( xhr.response ) {
|
|
|
responses.native = xhr.response;
|
|
|
}
|
|
|
|
|
|
complete(
|
|
|
xhrSuccessStatus[ xhr.status ] || xhr.status,
|
|
|
xhr.statusText,
|
|
|
responses,
|
|
|
xhr.getAllResponseHeaders()
|
|
|
);
|
|
|
}
|
|
|
}
|
|
|
};
|
|
|
};
|
|
|
|
|
|
// Listen to events
|
|
|
xhr.onload = callback();
|
|
|
xhr.onerror = callback("error");
|
|
|
|
|
|
// Create the abort callback
|
|
|
callback = xhrCallbacks[ id ] = callback("abort");
|
|
|
|
|
|
try {
|
|
|
// Do send the request (this may raise an exception)
|
|
|
xhr.send( options.hasContent && options.data || null );
|
|
|
} catch ( e ) {
|
|
|
// #14683: Only rethrow if this hasn't been notified as an error yet
|
|
|
if ( callback ) {
|
|
|
throw e;
|
|
|
}
|
|
|
}
|
|
|
},
|
|
|
|
|
|
abort: function() {
|
|
|
if ( callback ) {
|
|
|
callback();
|
|
|
}
|
|
|
}
|
|
|
};
|
|
|
}
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Install script dataType
|
|
|
jQuery.ajaxSetup({
|
|
|
accepts: {
|
|
|
script: "text/javascript, application/javascript, application/ecmascript, application/x-ecmascript"
|
|
|
},
|
|
|
contents: {
|
|
|
script: /(?:java|ecma)script/
|
|
|
},
|
|
|
converters: {
|
|
|
"text script": function( text ) {
|
|
|
jQuery.globalEval( text );
|
|
|
return text;
|
|
|
}
|
|
|
}
|
|
|
});
|
|
|
|
|
|
// Handle cache's special case and crossDomain
|
|
|
jQuery.ajaxPrefilter( "script", function( s ) {
|
|
|
if ( s.cache === undefined ) {
|
|
|
s.cache = false;
|
|
|
}
|
|
|
if ( s.crossDomain ) {
|
|
|
s.type = "GET";
|
|
|
}
|
|
|
});
|
|
|
|
|
|
// Bind script tag hack transport
|
|
|
jQuery.ajaxTransport( "script", function( s ) {
|
|
|
// This transport only deals with cross domain requests
|
|
|
if ( s.crossDomain ) {
|
|
|
var script, callback;
|
|
|
return {
|
|
|
send: function( _, complete ) {
|
|
|
script = jQuery("<script>").prop({
|
|
|
async: true,
|
|
|
charset: s.scriptCharset,
|
|
|
src: s.url
|
|
|
}).on(
|
|
|
"load error",
|
|
|
callback = function( evt ) {
|
|
|
script.remove();
|
|
|
callback = null;
|
|
|
if ( evt ) {
|
|
|
complete( evt.type === "error" ? 404 : 200, evt.type );
|
|
|
}
|
|
|
}
|
|
|
);
|
|
|
document.head.appendChild( script[ 0 ] );
|
|
|
},
|
|
|
abort: function() {
|
|
|
if ( callback ) {
|
|
|
callback();
|
|
|
}
|
|
|
}
|
|
|
};
|
|
|
}
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
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") && 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() {
|
|
|
// Restore preexisting value
|
|
|
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 = jQuery.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 >= 0 ) {
|
|
|
selector = jQuery.trim( url.slice( off ) );
|
|
|
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
|
|
|
type: type,
|
|
|
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 );
|
|
|
|
|
|
}).complete( callback && function( jqXHR, status ) {
|
|
|
self.each( callback, response || [ jqXHR.responseText, status, jqXHR ] );
|
|
|
});
|
|
|
}
|
|
|
|
|
|
return this;
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
jQuery.expr.filters.animated = function( elem ) {
|
|
|
return jQuery.grep(jQuery.timers, function( fn ) {
|
|
|
return elem === fn.elem;
|
|
|
}).length;
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
var docElem = window.document.documentElement;
|
|
|
|
|
|
/**
|
|
|
* Gets a window from an element
|
|
|
*/
|
|
|
function getWindow( elem ) {
|
|
|
return jQuery.isWindow( elem ) ? elem : elem.nodeType === 9 && elem.defaultView;
|
|
|
}
|
|
|
|
|
|
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" ) &&
|
|
|
( curCSSTop + curCSSLeft ).indexOf("auto") > -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 ) ) {
|
|
|
options = options.call( elem, i, 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,
|
|
|
elem = this[ 0 ],
|
|
|
box = { top: 0, left: 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 !== strundefined ) {
|
|
|
box = elem.getBoundingClientRect();
|
|
|
}
|
|
|
win = getWindow( doc );
|
|
|
return {
|
|
|
top: box.top + win.pageYOffset - docElem.clientTop,
|
|
|
left: box.left + win.pageXOffset - docElem.clientLeft
|
|
|
};
|
|
|
},
|
|
|
|
|
|
position: function() {
|
|
|
if ( !this[ 0 ] ) {
|
|
|
return;
|
|
|
}
|
|
|
|
|
|
var offsetParent, offset,
|
|
|
elem = this[ 0 ],
|
|
|
parentOffset = { top: 0, left: 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
|
|
|
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 || docElem;
|
|
|
|
|
|
while ( offsetParent && ( !jQuery.nodeName( offsetParent, "html" ) && jQuery.css( offsetParent, "position" ) === "static" ) ) {
|
|
|
offsetParent = offsetParent.offsetParent;
|
|
|
}
|
|
|
|
|
|
return offsetParent || docElem;
|
|
|
});
|
|
|
}
|
|
|
});
|
|
|
|
|
|
// Create scrollLeft and scrollTop methods
|
|
|
jQuery.each( { scrollLeft: "pageXOffset", scrollTop: "pageYOffset" }, function( method, prop ) {
|
|
|
var top = "pageYOffset" === prop;
|
|
|
|
|
|
jQuery.fn[ method ] = function( val ) {
|
|
|
return access( this, function( elem, method, val ) {
|
|
|
var win = getWindow( elem );
|
|
|
|
|
|
if ( val === undefined ) {
|
|
|
return win ? win[ prop ] : elem[ method ];
|
|
|
}
|
|
|
|
|
|
if ( win ) {
|
|
|
win.scrollTo(
|
|
|
!top ? val : window.pageXOffset,
|
|
|
top ? val : window.pageYOffset
|
|
|
);
|
|
|
|
|
|
} else {
|
|
|
elem[ method ] = val;
|
|
|
}
|
|
|
}, method, val, arguments.length, null );
|
|
|
};
|
|
|
});
|
|
|
|
|
|
// 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
|
|
|
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 );
|
|
|
};
|
|
|
});
|
|
|
});
|
|
|
|
|
|
|
|
|
// 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 ( typeof noGlobal === strundefined ) {
|
|
|
window.jQuery = window.$ = jQuery;
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
return jQuery;
|
|
|
|
|
|
}));
|
|
|
|
|
|
/*
|
|
|
Copyright 2013 Daniel Wirtz <dcode@dcode.io>
|
|
|
Copyright 2009 The Closure Library Authors. All Rights Reserved.
|
|
|
|
|
|
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.
|
|
|
*/
|
|
|
|
|
|
/**
|
|
|
* @license Long.js (c) 2013 Daniel Wirtz <dcode@dcode.io>
|
|
|
* Released under the Apache License, Version 2.0
|
|
|
* see: https://github.com/dcodeIO/Long.js for details
|
|
|
*/
|
|
|
(function(global) {
|
|
|
"use strict";
|
|
|
|
|
|
/**
|
|
|
* Constructs a 64 bit two's-complement integer, given its low and high 32 bit values as *signed* integers.
|
|
|
* See the from* functions below for more convenient ways of constructing Longs.
|
|
|
* @exports Long
|
|
|
* @class A Long class for representing a 64 bit two's-complement integer value.
|
|
|
* @param {number} low The low (signed) 32 bits of the long
|
|
|
* @param {number} high The high (signed) 32 bits of the long
|
|
|
* @param {boolean=} unsigned Whether unsigned or not, defaults to `false` for signed
|
|
|
* @constructor
|
|
|
*/
|
|
|
var Long = function(low, high, unsigned) {
|
|
|
|
|
|
/**
|
|
|
* The low 32 bits as a signed value.
|
|
|
* @type {number}
|
|
|
* @expose
|
|
|
*/
|
|
|
this.low = low|0;
|
|
|
|
|
|
/**
|
|
|
* The high 32 bits as a signed value.
|
|
|
* @type {number}
|
|
|
* @expose
|
|
|
*/
|
|
|
this.high = high|0;
|
|
|
|
|
|
/**
|
|
|
* Whether unsigned or not.
|
|
|
* @type {boolean}
|
|
|
* @expose
|
|
|
*/
|
|
|
this.unsigned = !!unsigned;
|
|
|
};
|
|
|
|
|
|
// The internal representation of a long is the two given signed, 32-bit values.
|
|
|
// We use 32-bit pieces because these are the size of integers on which
|
|
|
// Javascript performs bit-operations. For operations like addition and
|
|
|
// multiplication, we split each number into 16 bit pieces, which can easily be
|
|
|
// multiplied within Javascript's floating-point representation without overflow
|
|
|
// or change in sign.
|
|
|
//
|
|
|
// In the algorithms below, we frequently reduce the negative case to the
|
|
|
// positive case by negating the input(s) and then post-processing the result.
|
|
|
// Note that we must ALWAYS check specially whether those values are MIN_VALUE
|
|
|
// (-2^63) because -MIN_VALUE == MIN_VALUE (since 2^63 cannot be represented as
|
|
|
// a positive number, it overflows back into a negative). Not handling this
|
|
|
// case would often result in infinite recursion.
|
|
|
//
|
|
|
// Common constant values ZERO, ONE, NEG_ONE, etc. are defined below the from*
|
|
|
// methods on which they depend.
|
|
|
|
|
|
/**
|
|
|
* Tests if the specified object is a Long.
|
|
|
* @param {*} obj Object
|
|
|
* @returns {boolean}
|
|
|
* @expose
|
|
|
*/
|
|
|
Long.isLong = function(obj) {
|
|
|
return (obj && obj instanceof Long) === true;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* A cache of the Long representations of small integer values.
|
|
|
* @type {!Object}
|
|
|
* @inner
|
|
|
*/
|
|
|
var INT_CACHE = {};
|
|
|
|
|
|
/**
|
|
|
* A cache of the Long representations of small unsigned integer values.
|
|
|
* @type {!Object}
|
|
|
* @inner
|
|
|
*/
|
|
|
var UINT_CACHE = {};
|
|
|
|
|
|
/**
|
|
|
* Returns a Long representing the given 32 bit integer value.
|
|
|
* @param {number} value The 32 bit integer in question
|
|
|
* @param {boolean=} unsigned Whether unsigned or not, defaults to `false` for signed
|
|
|
* @returns {!Long} The corresponding Long value
|
|
|
* @expose
|
|
|
*/
|
|
|
Long.fromInt = function(value, unsigned) {
|
|
|
var obj, cachedObj;
|
|
|
if (!unsigned) {
|
|
|
value = value | 0;
|
|
|
if (-128 <= value && value < 128) {
|
|
|
cachedObj = INT_CACHE[value];
|
|
|
if (cachedObj)
|
|
|
return cachedObj;
|
|
|
}
|
|
|
obj = new Long(value, value < 0 ? -1 : 0, false);
|
|
|
if (-128 <= value && value < 128)
|
|
|
INT_CACHE[value] = obj;
|
|
|
return obj;
|
|
|
} else {
|
|
|
value = value >>> 0;
|
|
|
if (0 <= value && value < 256) {
|
|
|
cachedObj = UINT_CACHE[value];
|
|
|
if (cachedObj)
|
|
|
return cachedObj;
|
|
|
}
|
|
|
obj = new Long(value, (value | 0) < 0 ? -1 : 0, true);
|
|
|
if (0 <= value && value < 256)
|
|
|
UINT_CACHE[value] = obj;
|
|
|
return obj;
|
|
|
}
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Returns a Long representing the given value, provided that it is a finite number. Otherwise, zero is returned.
|
|
|
* @param {number} value The number in question
|
|
|
* @param {boolean=} unsigned Whether unsigned or not, defaults to `false` for signed
|
|
|
* @returns {!Long} The corresponding Long value
|
|
|
* @expose
|
|
|
*/
|
|
|
Long.fromNumber = function(value, unsigned) {
|
|
|
unsigned = !!unsigned;
|
|
|
if (isNaN(value) || !isFinite(value))
|
|
|
return Long.ZERO;
|
|
|
if (!unsigned && value <= -TWO_PWR_63_DBL)
|
|
|
return Long.MIN_VALUE;
|
|
|
if (!unsigned && value + 1 >= TWO_PWR_63_DBL)
|
|
|
return Long.MAX_VALUE;
|
|
|
if (unsigned && value >= TWO_PWR_64_DBL)
|
|
|
return Long.MAX_UNSIGNED_VALUE;
|
|
|
if (value < 0)
|
|
|
return Long.fromNumber(-value, unsigned).negate();
|
|
|
return new Long((value % TWO_PWR_32_DBL) | 0, (value / TWO_PWR_32_DBL) | 0, unsigned);
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Returns a Long representing the 64 bit integer that comes by concatenating the given low and high bits. Each is
|
|
|
* assumed to use 32 bits.
|
|
|
* @param {number} lowBits The low 32 bits
|
|
|
* @param {number} highBits The high 32 bits
|
|
|
* @param {boolean=} unsigned Whether unsigned or not, defaults to `false` for signed
|
|
|
* @returns {!Long} The corresponding Long value
|
|
|
* @expose
|
|
|
*/
|
|
|
Long.fromBits = function(lowBits, highBits, unsigned) {
|
|
|
return new Long(lowBits, highBits, unsigned);
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Returns a Long representation of the given string, written using the specified radix.
|
|
|
* @param {string} str The textual representation of the Long
|
|
|
* @param {(boolean|number)=} unsigned Whether unsigned or not, defaults to `false` for signed
|
|
|
* @param {number=} radix The radix in which the text is written (2-36), defaults to 10
|
|
|
* @returns {!Long} The corresponding Long value
|
|
|
* @expose
|
|
|
*/
|
|
|
Long.fromString = function(str, unsigned, radix) {
|
|
|
if (str.length === 0)
|
|
|
throw Error('number format error: empty string');
|
|
|
if (str === "NaN" || str === "Infinity" || str === "+Infinity" || str === "-Infinity")
|
|
|
return Long.ZERO;
|
|
|
if (typeof unsigned === 'number') // For goog.math.long compatibility
|
|
|
radix = unsigned,
|
|
|
unsigned = false;
|
|
|
radix = radix || 10;
|
|
|
if (radix < 2 || 36 < radix)
|
|
|
throw Error('radix out of range: ' + radix);
|
|
|
|
|
|
var p;
|
|
|
if ((p = str.indexOf('-')) > 0)
|
|
|
throw Error('number format error: interior "-" character: ' + str);
|
|
|
else if (p === 0)
|
|
|
return Long.fromString(str.substring(1), unsigned, radix).negate();
|
|
|
|
|
|
// Do several (8) digits each time through the loop, so as to
|
|
|
// minimize the calls to the very expensive emulated div.
|
|
|
var radixToPower = Long.fromNumber(Math.pow(radix, 8));
|
|
|
|
|
|
var result = Long.ZERO;
|
|
|
for (var i = 0; i < str.length; i += 8) {
|
|
|
var size = Math.min(8, str.length - i);
|
|
|
var value = parseInt(str.substring(i, i + size), radix);
|
|
|
if (size < 8) {
|
|
|
var power = Long.fromNumber(Math.pow(radix, size));
|
|
|
result = result.multiply(power).add(Long.fromNumber(value));
|
|
|
} else {
|
|
|
result = result.multiply(radixToPower);
|
|
|
result = result.add(Long.fromNumber(value));
|
|
|
}
|
|
|
}
|
|
|
result.unsigned = unsigned;
|
|
|
return result;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Converts the specified value to a Long.
|
|
|
* @param {!Long|number|string|!{low: number, high: number, unsigned: boolean}} val Value
|
|
|
* @returns {!Long}
|
|
|
* @expose
|
|
|
*/
|
|
|
Long.fromValue = function(val) {
|
|
|
if (typeof val === 'number')
|
|
|
return Long.fromNumber(val);
|
|
|
if (typeof val === 'string')
|
|
|
return Long.fromString(val);
|
|
|
if (Long.isLong(val))
|
|
|
return val;
|
|
|
// Throws for not an object (undefined, null):
|
|
|
return new Long(val.low, val.high, val.unsigned);
|
|
|
};
|
|
|
|
|
|
// NOTE: the compiler should inline these constant values below and then remove these variables, so there should be
|
|
|
// no runtime penalty for these.
|
|
|
|
|
|
/**
|
|
|
* @type {number}
|
|
|
* @const
|
|
|
* @inner
|
|
|
*/
|
|
|
var TWO_PWR_16_DBL = 1 << 16;
|
|
|
|
|
|
/**
|
|
|
* @type {number}
|
|
|
* @const
|
|
|
* @inner
|
|
|
*/
|
|
|
var TWO_PWR_24_DBL = 1 << 24;
|
|
|
|
|
|
/**
|
|
|
* @type {number}
|
|
|
* @const
|
|
|
* @inner
|
|
|
*/
|
|
|
var TWO_PWR_32_DBL = TWO_PWR_16_DBL * TWO_PWR_16_DBL;
|
|
|
|
|
|
/**
|
|
|
* @type {number}
|
|
|
* @const
|
|
|
* @inner
|
|
|
*/
|
|
|
var TWO_PWR_64_DBL = TWO_PWR_32_DBL * TWO_PWR_32_DBL;
|
|
|
|
|
|
/**
|
|
|
* @type {number}
|
|
|
* @const
|
|
|
* @inner
|
|
|
*/
|
|
|
var TWO_PWR_63_DBL = TWO_PWR_64_DBL / 2;
|
|
|
|
|
|
/**
|
|
|
* @type {!Long}
|
|
|
* @const
|
|
|
* @inner
|
|
|
*/
|
|
|
var TWO_PWR_24 = Long.fromInt(TWO_PWR_24_DBL);
|
|
|
|
|
|
/**
|
|
|
* Signed zero.
|
|
|
* @type {!Long}
|
|
|
* @expose
|
|
|
*/
|
|
|
Long.ZERO = Long.fromInt(0);
|
|
|
|
|
|
/**
|
|
|
* Unsigned zero.
|
|
|
* @type {!Long}
|
|
|
* @expose
|
|
|
*/
|
|
|
Long.UZERO = Long.fromInt(0, true);
|
|
|
|
|
|
/**
|
|
|
* Signed one.
|
|
|
* @type {!Long}
|
|
|
* @expose
|
|
|
*/
|
|
|
Long.ONE = Long.fromInt(1);
|
|
|
|
|
|
/**
|
|
|
* Unsigned one.
|
|
|
* @type {!Long}
|
|
|
* @expose
|
|
|
*/
|
|
|
Long.UONE = Long.fromInt(1, true);
|
|
|
|
|
|
/**
|
|
|
* Signed negative one.
|
|
|
* @type {!Long}
|
|
|
* @expose
|
|
|
*/
|
|
|
Long.NEG_ONE = Long.fromInt(-1);
|
|
|
|
|
|
/**
|
|
|
* Maximum signed value.
|
|
|
* @type {!Long}
|
|
|
* @expose
|
|
|
*/
|
|
|
Long.MAX_VALUE = Long.fromBits(0xFFFFFFFF|0, 0x7FFFFFFF|0, false);
|
|
|
|
|
|
/**
|
|
|
* Maximum unsigned value.
|
|
|
* @type {!Long}
|
|
|
* @expose
|
|
|
*/
|
|
|
Long.MAX_UNSIGNED_VALUE = Long.fromBits(0xFFFFFFFF|0, 0xFFFFFFFF|0, true);
|
|
|
|
|
|
/**
|
|
|
* Minimum signed value.
|
|
|
* @type {!Long}
|
|
|
* @expose
|
|
|
*/
|
|
|
Long.MIN_VALUE = Long.fromBits(0, 0x80000000|0, false);
|
|
|
|
|
|
/**
|
|
|
* Converts the Long to a 32 bit integer, assuming it is a 32 bit integer.
|
|
|
* @returns {number}
|
|
|
* @expose
|
|
|
*/
|
|
|
Long.prototype.toInt = function() {
|
|
|
return this.unsigned ? this.low >>> 0 : this.low;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Converts the Long to a the nearest floating-point representation of this value (double, 53 bit mantissa).
|
|
|
* @returns {number}
|
|
|
* @expose
|
|
|
*/
|
|
|
Long.prototype.toNumber = function() {
|
|
|
if (this.unsigned) {
|
|
|
return ((this.high >>> 0) * TWO_PWR_32_DBL) + (this.low >>> 0);
|
|
|
}
|
|
|
return this.high * TWO_PWR_32_DBL + (this.low >>> 0);
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Converts the Long to a string written in the specified radix.
|
|
|
* @param {number=} radix Radix (2-36), defaults to 10
|
|
|
* @returns {string}
|
|
|
* @override
|
|
|
* @throws {RangeError} If `radix` is out of range
|
|
|
* @expose
|
|
|
*/
|
|
|
Long.prototype.toString = function(radix) {
|
|
|
radix = radix || 10;
|
|
|
if (radix < 2 || 36 < radix)
|
|
|
throw RangeError('radix out of range: ' + radix);
|
|
|
if (this.isZero())
|
|
|
return '0';
|
|
|
var rem;
|
|
|
if (this.isNegative()) { // Unsigned Longs are never negative
|
|
|
if (this.equals(Long.MIN_VALUE)) {
|
|
|
// We need to change the Long value before it can be negated, so we remove
|
|
|
// the bottom-most digit in this base and then recurse to do the rest.
|
|
|
var radixLong = Long.fromNumber(radix);
|
|
|
var div = this.div(radixLong);
|
|
|
rem = div.multiply(radixLong).subtract(this);
|
|
|
return div.toString(radix) + rem.toInt().toString(radix);
|
|
|
} else
|
|
|
return '-' + this.negate().toString(radix);
|
|
|
}
|
|
|
|
|
|
// Do several (6) digits each time through the loop, so as to
|
|
|
// minimize the calls to the very expensive emulated div.
|
|
|
var radixToPower = Long.fromNumber(Math.pow(radix, 6), this.unsigned);
|
|
|
rem = this;
|
|
|
var result = '';
|
|
|
while (true) {
|
|
|
var remDiv = rem.div(radixToPower),
|
|
|
intval = rem.subtract(remDiv.multiply(radixToPower)).toInt() >>> 0,
|
|
|
digits = intval.toString(radix);
|
|
|
rem = remDiv;
|
|
|
if (rem.isZero())
|
|
|
return digits + result;
|
|
|
else {
|
|
|
while (digits.length < 6)
|
|
|
digits = '0' + digits;
|
|
|
result = '' + digits + result;
|
|
|
}
|
|
|
}
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Gets the high 32 bits as a signed integer.
|
|
|
* @returns {number} Signed high bits
|
|
|
* @expose
|
|
|
*/
|
|
|
Long.prototype.getHighBits = function() {
|
|
|
return this.high;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Gets the high 32 bits as an unsigned integer.
|
|
|
* @returns {number} Unsigned high bits
|
|
|
* @expose
|
|
|
*/
|
|
|
Long.prototype.getHighBitsUnsigned = function() {
|
|
|
return this.high >>> 0;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Gets the low 32 bits as a signed integer.
|
|
|
* @returns {number} Signed low bits
|
|
|
* @expose
|
|
|
*/
|
|
|
Long.prototype.getLowBits = function() {
|
|
|
return this.low;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Gets the low 32 bits as an unsigned integer.
|
|
|
* @returns {number} Unsigned low bits
|
|
|
* @expose
|
|
|
*/
|
|
|
Long.prototype.getLowBitsUnsigned = function() {
|
|
|
return this.low >>> 0;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Gets the number of bits needed to represent the absolute value of this Long.
|
|
|
* @returns {number}
|
|
|
* @expose
|
|
|
*/
|
|
|
Long.prototype.getNumBitsAbs = function() {
|
|
|
if (this.isNegative()) // Unsigned Longs are never negative
|
|
|
return this.equals(Long.MIN_VALUE) ? 64 : this.negate().getNumBitsAbs();
|
|
|
var val = this.high != 0 ? this.high : this.low;
|
|
|
for (var bit = 31; bit > 0; bit--)
|
|
|
if ((val & (1 << bit)) != 0)
|
|
|
break;
|
|
|
return this.high != 0 ? bit + 33 : bit + 1;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Tests if this Long's value equals zero.
|
|
|
* @returns {boolean}
|
|
|
* @expose
|
|
|
*/
|
|
|
Long.prototype.isZero = function() {
|
|
|
return this.high === 0 && this.low === 0;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Tests if this Long's value is negative.
|
|
|
* @returns {boolean}
|
|
|
* @expose
|
|
|
*/
|
|
|
Long.prototype.isNegative = function() {
|
|
|
return !this.unsigned && this.high < 0;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Tests if this Long's value is positive.
|
|
|
* @returns {boolean}
|
|
|
* @expose
|
|
|
*/
|
|
|
Long.prototype.isPositive = function() {
|
|
|
return this.unsigned || this.high >= 0;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Tests if this Long's value is odd.
|
|
|
* @returns {boolean}
|
|
|
* @expose
|
|
|
*/
|
|
|
Long.prototype.isOdd = function() {
|
|
|
return (this.low & 1) === 1;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Tests if this Long's value is even.
|
|
|
* @returns {boolean}
|
|
|
* @expose
|
|
|
*/
|
|
|
Long.prototype.isEven = function() {
|
|
|
return (this.low & 1) === 0;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Tests if this Long's value equals the specified's.
|
|
|
* @param {!Long|number|string} other Other value
|
|
|
* @returns {boolean}
|
|
|
* @expose
|
|
|
*/
|
|
|
Long.prototype.equals = function(other) {
|
|
|
if (!Long.isLong(other))
|
|
|
other = Long.fromValue(other);
|
|
|
if (this.unsigned !== other.unsigned && (this.high >>> 31) === 1 && (other.high >>> 31) === 1)
|
|
|
return false;
|
|
|
return this.high === other.high && this.low === other.low;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Tests if this Long's value differs from the specified's.
|
|
|
* @param {!Long|number|string} other Other value
|
|
|
* @returns {boolean}
|
|
|
* @expose
|
|
|
*/
|
|
|
Long.prototype.notEquals = function(other) {
|
|
|
if (!Long.isLong(other))
|
|
|
other = Long.fromValue(other);
|
|
|
return !this.equals(other);
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Tests if this Long's value is less than the specified's.
|
|
|
* @param {!Long|number|string} other Other value
|
|
|
* @returns {boolean}
|
|
|
* @expose
|
|
|
*/
|
|
|
Long.prototype.lessThan = function(other) {
|
|
|
if (!Long.isLong(other))
|
|
|
other = Long.fromValue(other);
|
|
|
return this.compare(other) < 0;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Tests if this Long's value is less than or equal the specified's.
|
|
|
* @param {!Long|number|string} other Other value
|
|
|
* @returns {boolean}
|
|
|
* @expose
|
|
|
*/
|
|
|
Long.prototype.lessThanOrEqual = function(other) {
|
|
|
if (!Long.isLong(other))
|
|
|
other = Long.fromValue(other);
|
|
|
return this.compare(other) <= 0;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Tests if this Long's value is greater than the specified's.
|
|
|
* @param {!Long|number|string} other Other value
|
|
|
* @returns {boolean}
|
|
|
* @expose
|
|
|
*/
|
|
|
Long.prototype.greaterThan = function(other) {
|
|
|
if (!Long.isLong(other))
|
|
|
other = Long.fromValue(other);
|
|
|
return this.compare(other) > 0;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Tests if this Long's value is greater than or equal the specified's.
|
|
|
* @param {!Long|number|string} other Other value
|
|
|
* @returns {boolean}
|
|
|
* @expose
|
|
|
*/
|
|
|
Long.prototype.greaterThanOrEqual = function(other) {
|
|
|
return this.compare(other) >= 0;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Compares this Long's value with the specified's.
|
|
|
* @param {!Long|number|string} other Other value
|
|
|
* @returns {number} 0 if they are the same, 1 if the this is greater and -1
|
|
|
* if the given one is greater
|
|
|
* @expose
|
|
|
*/
|
|
|
Long.prototype.compare = function(other) {
|
|
|
if (this.equals(other))
|
|
|
return 0;
|
|
|
var thisNeg = this.isNegative(),
|
|
|
otherNeg = other.isNegative();
|
|
|
if (thisNeg && !otherNeg)
|
|
|
return -1;
|
|
|
if (!thisNeg && otherNeg)
|
|
|
return 1;
|
|
|
// At this point the sign bits are the same
|
|
|
if (!this.unsigned)
|
|
|
return this.subtract(other).isNegative() ? -1 : 1;
|
|
|
// Both are positive if at least one is unsigned
|
|
|
return (other.high >>> 0) > (this.high >>> 0) || (other.high === this.high && (other.low >>> 0) > (this.low >>> 0)) ? -1 : 1;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Negates this Long's value.
|
|
|
* @returns {!Long} Negated Long
|
|
|
* @expose
|
|
|
*/
|
|
|
Long.prototype.negate = function() {
|
|
|
if (!this.unsigned && this.equals(Long.MIN_VALUE))
|
|
|
return Long.MIN_VALUE;
|
|
|
return this.not().add(Long.ONE);
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Returns the sum of this and the specified Long.
|
|
|
* @param {!Long|number|string} addend Addend
|
|
|
* @returns {!Long} Sum
|
|
|
* @expose
|
|
|
*/
|
|
|
Long.prototype.add = function(addend) {
|
|
|
if (!Long.isLong(addend))
|
|
|
addend = Long.fromValue(addend);
|
|
|
|
|
|
// Divide each number into 4 chunks of 16 bits, and then sum the chunks.
|
|
|
|
|
|
var a48 = this.high >>> 16;
|
|
|
var a32 = this.high & 0xFFFF;
|
|
|
var a16 = this.low >>> 16;
|
|
|
var a00 = this.low & 0xFFFF;
|
|
|
|
|
|
var b48 = addend.high >>> 16;
|
|
|
var b32 = addend.high & 0xFFFF;
|
|
|
var b16 = addend.low >>> 16;
|
|
|
var b00 = addend.low & 0xFFFF;
|
|
|
|
|
|
var c48 = 0, c32 = 0, c16 = 0, c00 = 0;
|
|
|
c00 += a00 + b00;
|
|
|
c16 += c00 >>> 16;
|
|
|
c00 &= 0xFFFF;
|
|
|
c16 += a16 + b16;
|
|
|
c32 += c16 >>> 16;
|
|
|
c16 &= 0xFFFF;
|
|
|
c32 += a32 + b32;
|
|
|
c48 += c32 >>> 16;
|
|
|
c32 &= 0xFFFF;
|
|
|
c48 += a48 + b48;
|
|
|
c48 &= 0xFFFF;
|
|
|
return Long.fromBits((c16 << 16) | c00, (c48 << 16) | c32, this.unsigned);
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Returns the difference of this and the specified Long.
|
|
|
* @param {!Long|number|string} subtrahend Subtrahend
|
|
|
* @returns {!Long} Difference
|
|
|
* @expose
|
|
|
*/
|
|
|
Long.prototype.subtract = function(subtrahend) {
|
|
|
if (!Long.isLong(subtrahend))
|
|
|
subtrahend = Long.fromValue(subtrahend);
|
|
|
return this.add(subtrahend.negate());
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Returns the product of this and the specified Long.
|
|
|
* @param {!Long|number|string} multiplier Multiplier
|
|
|
* @returns {!Long} Product
|
|
|
* @expose
|
|
|
*/
|
|
|
Long.prototype.multiply = function(multiplier) {
|
|
|
if (this.isZero())
|
|
|
return Long.ZERO;
|
|
|
if (!Long.isLong(multiplier))
|
|
|
multiplier = Long.fromValue(multiplier);
|
|
|
if (multiplier.isZero())
|
|
|
return Long.ZERO;
|
|
|
if (this.equals(Long.MIN_VALUE))
|
|
|
return multiplier.isOdd() ? Long.MIN_VALUE : Long.ZERO;
|
|
|
if (multiplier.equals(Long.MIN_VALUE))
|
|
|
return this.isOdd() ? Long.MIN_VALUE : Long.ZERO;
|
|
|
|
|
|
if (this.isNegative()) {
|
|
|
if (multiplier.isNegative())
|
|
|
return this.negate().multiply(multiplier.negate());
|
|
|
else
|
|
|
return this.negate().multiply(multiplier).negate();
|
|
|
} else if (multiplier.isNegative())
|
|
|
return this.multiply(multiplier.negate()).negate();
|
|
|
|
|
|
// If both longs are small, use float multiplication
|
|
|
if (this.lessThan(TWO_PWR_24) && multiplier.lessThan(TWO_PWR_24))
|
|
|
return Long.fromNumber(this.toNumber() * multiplier.toNumber(), this.unsigned);
|
|
|
|
|
|
// Divide each long into 4 chunks of 16 bits, and then add up 4x4 products.
|
|
|
// We can skip products that would overflow.
|
|
|
|
|
|
var a48 = this.high >>> 16;
|
|
|
var a32 = this.high & 0xFFFF;
|
|
|
var a16 = this.low >>> 16;
|
|
|
var a00 = this.low & 0xFFFF;
|
|
|
|
|
|
var b48 = multiplier.high >>> 16;
|
|
|
var b32 = multiplier.high & 0xFFFF;
|
|
|
var b16 = multiplier.low >>> 16;
|
|
|
var b00 = multiplier.low & 0xFFFF;
|
|
|
|
|
|
var c48 = 0, c32 = 0, c16 = 0, c00 = 0;
|
|
|
c00 += a00 * b00;
|
|
|
c16 += c00 >>> 16;
|
|
|
c00 &= 0xFFFF;
|
|
|
c16 += a16 * b00;
|
|
|
c32 += c16 >>> 16;
|
|
|
c16 &= 0xFFFF;
|
|
|
c16 += a00 * b16;
|
|
|
c32 += c16 >>> 16;
|
|
|
c16 &= 0xFFFF;
|
|
|
c32 += a32 * b00;
|
|
|
c48 += c32 >>> 16;
|
|
|
c32 &= 0xFFFF;
|
|
|
c32 += a16 * b16;
|
|
|
c48 += c32 >>> 16;
|
|
|
c32 &= 0xFFFF;
|
|
|
c32 += a00 * b32;
|
|
|
c48 += c32 >>> 16;
|
|
|
c32 &= 0xFFFF;
|
|
|
c48 += a48 * b00 + a32 * b16 + a16 * b32 + a00 * b48;
|
|
|
c48 &= 0xFFFF;
|
|
|
return Long.fromBits((c16 << 16) | c00, (c48 << 16) | c32, this.unsigned);
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Returns this Long divided by the specified.
|
|
|
* @param {!Long|number|string} divisor Divisor
|
|
|
* @returns {!Long} Quotient
|
|
|
* @expose
|
|
|
*/
|
|
|
Long.prototype.div = function(divisor) {
|
|
|
if (!Long.isLong(divisor))
|
|
|
divisor = Long.fromValue(divisor);
|
|
|
if (divisor.isZero())
|
|
|
throw(new Error('division by zero'));
|
|
|
if (this.isZero())
|
|
|
return this.unsigned ? Long.UZERO : Long.ZERO;
|
|
|
var approx, rem, res;
|
|
|
if (this.equals(Long.MIN_VALUE)) {
|
|
|
if (divisor.equals(Long.ONE) || divisor.equals(Long.NEG_ONE))
|
|
|
return Long.MIN_VALUE; // recall that -MIN_VALUE == MIN_VALUE
|
|
|
else if (divisor.equals(Long.MIN_VALUE))
|
|
|
return Long.ONE;
|
|
|
else {
|
|
|
// At this point, we have |other| >= 2, so |this/other| < |MIN_VALUE|.
|
|
|
var halfThis = this.shiftRight(1);
|
|
|
approx = halfThis.div(divisor).shiftLeft(1);
|
|
|
if (approx.equals(Long.ZERO)) {
|
|
|
return divisor.isNegative() ? Long.ONE : Long.NEG_ONE;
|
|
|
} else {
|
|
|
rem = this.subtract(divisor.multiply(approx));
|
|
|
res = approx.add(rem.div(divisor));
|
|
|
return res;
|
|
|
}
|
|
|
}
|
|
|
} else if (divisor.equals(Long.MIN_VALUE))
|
|
|
return this.unsigned ? Long.UZERO : Long.ZERO;
|
|
|
if (this.isNegative()) {
|
|
|
if (divisor.isNegative())
|
|
|
return this.negate().div(divisor.negate());
|
|
|
return this.negate().div(divisor).negate();
|
|
|
} else if (divisor.isNegative())
|
|
|
return this.div(divisor.negate()).negate();
|
|
|
|
|
|
// Repeat the following until the remainder is less than other: find a
|
|
|
// floating-point that approximates remainder / other *from below*, add this
|
|
|
// into the result, and subtract it from the remainder. It is critical that
|
|
|
// the approximate value is less than or equal to the real value so that the
|
|
|
// remainder never becomes negative.
|
|
|
res = Long.ZERO;
|
|
|
rem = this;
|
|
|
while (rem.greaterThanOrEqual(divisor)) {
|
|
|
// Approximate the result of division. This may be a little greater or
|
|
|
// smaller than the actual value.
|
|
|
approx = Math.max(1, Math.floor(rem.toNumber() / divisor.toNumber()));
|
|
|
|
|
|
// We will tweak the approximate result by changing it in the 48-th digit or
|
|
|
// the smallest non-fractional digit, whichever is larger.
|
|
|
var log2 = Math.ceil(Math.log(approx) / Math.LN2),
|
|
|
delta = (log2 <= 48) ? 1 : Math.pow(2, log2 - 48),
|
|
|
|
|
|
// Decrease the approximation until it is smaller than the remainder. Note
|
|
|
// that if it is too large, the product overflows and is negative.
|
|
|
approxRes = Long.fromNumber(approx),
|
|
|
approxRem = approxRes.multiply(divisor);
|
|
|
while (approxRem.isNegative() || approxRem.greaterThan(rem)) {
|
|
|
approx -= delta;
|
|
|
approxRes = Long.fromNumber(approx, this.unsigned);
|
|
|
approxRem = approxRes.multiply(divisor);
|
|
|
}
|
|
|
|
|
|
// We know the answer can't be zero... and actually, zero would cause
|
|
|
// infinite recursion since we would make no progress.
|
|
|
if (approxRes.isZero())
|
|
|
approxRes = Long.ONE;
|
|
|
|
|
|
res = res.add(approxRes);
|
|
|
rem = rem.subtract(approxRem);
|
|
|
}
|
|
|
return res;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Returns this Long modulo the specified.
|
|
|
* @param {!Long|number|string} divisor Divisor
|
|
|
* @returns {!Long} Remainder
|
|
|
* @expose
|
|
|
*/
|
|
|
Long.prototype.modulo = function(divisor) {
|
|
|
if (!Long.isLong(divisor))
|
|
|
divisor = Long.fromValue(divisor);
|
|
|
return this.subtract(this.div(divisor).multiply(divisor));
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Returns the bitwise NOT of this Long.
|
|
|
* @returns {!Long}
|
|
|
* @expose
|
|
|
*/
|
|
|
Long.prototype.not = function() {
|
|
|
return Long.fromBits(~this.low, ~this.high, this.unsigned);
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Returns the bitwise AND of this Long and the specified.
|
|
|
* @param {!Long|number|string} other Other Long
|
|
|
* @returns {!Long}
|
|
|
* @expose
|
|
|
*/
|
|
|
Long.prototype.and = function(other) {
|
|
|
if (!Long.isLong(other))
|
|
|
other = Long.fromValue(other);
|
|
|
return Long.fromBits(this.low & other.low, this.high & other.high, this.unsigned);
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Returns the bitwise OR of this Long and the specified.
|
|
|
* @param {!Long|number|string} other Other Long
|
|
|
* @returns {!Long}
|
|
|
* @expose
|
|
|
*/
|
|
|
Long.prototype.or = function(other) {
|
|
|
if (!Long.isLong(other))
|
|
|
other = Long.fromValue(other);
|
|
|
return Long.fromBits(this.low | other.low, this.high | other.high, this.unsigned);
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Returns the bitwise XOR of this Long and the given one.
|
|
|
* @param {!Long|number|string} other Other Long
|
|
|
* @returns {!Long}
|
|
|
* @expose
|
|
|
*/
|
|
|
Long.prototype.xor = function(other) {
|
|
|
if (!Long.isLong(other))
|
|
|
other = Long.fromValue(other);
|
|
|
return Long.fromBits(this.low ^ other.low, this.high ^ other.high, this.unsigned);
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Returns this Long with bits shifted to the left by the given amount.
|
|
|
* @param {number|!Long} numBits Number of bits
|
|
|
* @returns {!Long} Shifted Long
|
|
|
* @expose
|
|
|
*/
|
|
|
Long.prototype.shiftLeft = function(numBits) {
|
|
|
if (Long.isLong(numBits))
|
|
|
numBits = numBits.toInt();
|
|
|
if ((numBits &= 63) === 0)
|
|
|
return this;
|
|
|
else if (numBits < 32)
|
|
|
return Long.fromBits(this.low << numBits, (this.high << numBits) | (this.low >>> (32 - numBits)), this.unsigned);
|
|
|
else
|
|
|
return Long.fromBits(0, this.low << (numBits - 32), this.unsigned);
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Returns this Long with bits arithmetically shifted to the right by the given amount.
|
|
|
* @param {number|!Long} numBits Number of bits
|
|
|
* @returns {!Long} Shifted Long
|
|
|
* @expose
|
|
|
*/
|
|
|
Long.prototype.shiftRight = function(numBits) {
|
|
|
if (Long.isLong(numBits))
|
|
|
numBits = numBits.toInt();
|
|
|
if ((numBits &= 63) === 0)
|
|
|
return this;
|
|
|
else if (numBits < 32)
|
|
|
return Long.fromBits((this.low >>> numBits) | (this.high << (32 - numBits)), this.high >> numBits, this.unsigned);
|
|
|
else
|
|
|
return Long.fromBits(this.high >> (numBits - 32), this.high >= 0 ? 0 : -1, this.unsigned);
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Returns this Long with bits logically shifted to the right by the given amount.
|
|
|
* @param {number|!Long} numBits Number of bits
|
|
|
* @returns {!Long} Shifted Long
|
|
|
* @expose
|
|
|
*/
|
|
|
Long.prototype.shiftRightUnsigned = function(numBits) {
|
|
|
if (Long.isLong(numBits))
|
|
|
numBits = numBits.toInt();
|
|
|
numBits &= 63;
|
|
|
if (numBits === 0)
|
|
|
return this;
|
|
|
else {
|
|
|
var high = this.high;
|
|
|
if (numBits < 32) {
|
|
|
var low = this.low;
|
|
|
return Long.fromBits((low >>> numBits) | (high << (32 - numBits)), high >>> numBits, this.unsigned);
|
|
|
} else if (numBits === 32)
|
|
|
return Long.fromBits(high, 0, this.unsigned);
|
|
|
else
|
|
|
return Long.fromBits(high >>> (numBits - 32), 0, this.unsigned);
|
|
|
}
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Converts this Long to signed.
|
|
|
* @returns {!Long} Signed long
|
|
|
* @expose
|
|
|
*/
|
|
|
Long.prototype.toSigned = function() {
|
|
|
if (!this.unsigned)
|
|
|
return this;
|
|
|
return new Long(this.low, this.high, false);
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Converts this Long to unsigned.
|
|
|
* @returns {!Long} Unsigned long
|
|
|
* @expose
|
|
|
*/
|
|
|
Long.prototype.toUnsigned = function() {
|
|
|
if (this.unsigned)
|
|
|
return this;
|
|
|
return new Long(this.low, this.high, true);
|
|
|
};
|
|
|
|
|
|
/* CommonJS */ if (typeof require === 'function' && typeof module === 'object' && module && typeof exports === 'object' && exports)
|
|
|
module["exports"] = Long;
|
|
|
/* AMD */ else if (typeof define === 'function' && define["amd"])
|
|
|
define(function() { return Long; });
|
|
|
/* Global */ else
|
|
|
(global["dcodeIO"] = global["dcodeIO"] || {})["Long"] = Long;
|
|
|
|
|
|
})(this);
|
|
|
|
|
|
/*
|
|
|
Copyright 2013-2014 Daniel Wirtz <dcode@dcode.io>
|
|
|
|
|
|
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.
|
|
|
*/
|
|
|
|
|
|
/**
|
|
|
* @license ByteBuffer.js (c) 2013-2014 Daniel Wirtz <dcode@dcode.io>
|
|
|
* This version of ByteBuffer.js uses an ArrayBuffer (AB) as its backing buffer and is compatible with modern browsers.
|
|
|
* Released under the Apache License, Version 2.0
|
|
|
* see: https://github.com/dcodeIO/ByteBuffer.js for details
|
|
|
*/ //
|
|
|
(function(global) {
|
|
|
"use strict";
|
|
|
|
|
|
/**
|
|
|
* @param {function(new: Long, number, number, boolean=)=} Long
|
|
|
* @returns {function(new: ByteBuffer, number=, boolean=, boolean=)}}
|
|
|
* @inner
|
|
|
*/
|
|
|
function loadByteBuffer(Long) {
|
|
|
|
|
|
/**
|
|
|
* Constructs a new ByteBuffer.
|
|
|
* @class The swiss army knife for binary data in JavaScript.
|
|
|
* @exports ByteBuffer
|
|
|
* @constructor
|
|
|
* @param {number=} capacity Initial capacity. Defaults to {@link ByteBuffer.DEFAULT_CAPACITY}.
|
|
|
* @param {boolean=} littleEndian Whether to use little or big endian byte order. Defaults to
|
|
|
* {@link ByteBuffer.DEFAULT_ENDIAN}.
|
|
|
* @param {boolean=} noAssert Whether to skip assertions of offsets and values. Defaults to
|
|
|
* {@link ByteBuffer.DEFAULT_NOASSERT}.
|
|
|
* @expose
|
|
|
*/
|
|
|
var ByteBuffer = function(capacity, littleEndian, noAssert) {
|
|
|
if (typeof capacity === 'undefined') capacity = ByteBuffer.DEFAULT_CAPACITY;
|
|
|
if (typeof littleEndian === 'undefined') littleEndian = ByteBuffer.DEFAULT_ENDIAN;
|
|
|
if (typeof noAssert === 'undefined') noAssert = ByteBuffer.DEFAULT_NOASSERT;
|
|
|
if (!noAssert) {
|
|
|
capacity = capacity | 0;
|
|
|
if (capacity < 0)
|
|
|
throw RangeError("Illegal capacity");
|
|
|
littleEndian = !!littleEndian;
|
|
|
noAssert = !!noAssert;
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* Backing buffer.
|
|
|
* @type {!ArrayBuffer}
|
|
|
* @expose
|
|
|
*/
|
|
|
this.buffer = capacity === 0 ? EMPTY_BUFFER : new ArrayBuffer(capacity);
|
|
|
|
|
|
/**
|
|
|
* Data view to manipulate the backing buffer. Becomes `null` if the backing buffer has a capacity of `0`.
|
|
|
* @type {?DataView}
|
|
|
* @expose
|
|
|
*/
|
|
|
this.view = capacity === 0 ? null : new DataView(this.buffer);
|
|
|
|
|
|
/**
|
|
|
* Absolute read/write offset.
|
|
|
* @type {number}
|
|
|
* @expose
|
|
|
* @see ByteBuffer#flip
|
|
|
* @see ByteBuffer#clear
|
|
|
*/
|
|
|
this.offset = 0;
|
|
|
|
|
|
/**
|
|
|
* Marked offset.
|
|
|
* @type {number}
|
|
|
* @expose
|
|
|
* @see ByteBuffer#mark
|
|
|
* @see ByteBuffer#reset
|
|
|
*/
|
|
|
this.markedOffset = -1;
|
|
|
|
|
|
/**
|
|
|
* Absolute limit of the contained data. Set to the backing buffer's capacity upon allocation.
|
|
|
* @type {number}
|
|
|
* @expose
|
|
|
* @see ByteBuffer#flip
|
|
|
* @see ByteBuffer#clear
|
|
|
*/
|
|
|
this.limit = capacity;
|
|
|
|
|
|
/**
|
|
|
* Whether to use little endian byte order, defaults to `false` for big endian.
|
|
|
* @type {boolean}
|
|
|
* @expose
|
|
|
*/
|
|
|
this.littleEndian = typeof littleEndian !== 'undefined' ? !!littleEndian : false;
|
|
|
|
|
|
/**
|
|
|
* Whether to skip assertions of offsets and values, defaults to `false`.
|
|
|
* @type {boolean}
|
|
|
* @expose
|
|
|
*/
|
|
|
this.noAssert = !!noAssert;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* ByteBuffer version.
|
|
|
* @type {string}
|
|
|
* @const
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBuffer.VERSION = "3.5.4";
|
|
|
|
|
|
/**
|
|
|
* Little endian constant that can be used instead of its boolean value. Evaluates to `true`.
|
|
|
* @type {boolean}
|
|
|
* @const
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBuffer.LITTLE_ENDIAN = true;
|
|
|
|
|
|
/**
|
|
|
* Big endian constant that can be used instead of its boolean value. Evaluates to `false`.
|
|
|
* @type {boolean}
|
|
|
* @const
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBuffer.BIG_ENDIAN = false;
|
|
|
|
|
|
/**
|
|
|
* Default initial capacity of `16`.
|
|
|
* @type {number}
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBuffer.DEFAULT_CAPACITY = 16;
|
|
|
|
|
|
/**
|
|
|
* Default endianess of `false` for big endian.
|
|
|
* @type {boolean}
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBuffer.DEFAULT_ENDIAN = ByteBuffer.BIG_ENDIAN;
|
|
|
|
|
|
/**
|
|
|
* Default no assertions flag of `false`.
|
|
|
* @type {boolean}
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBuffer.DEFAULT_NOASSERT = false;
|
|
|
|
|
|
/**
|
|
|
* A `Long` class for representing a 64-bit two's-complement integer value. May be `null` if Long.js has not been loaded
|
|
|
* and int64 support is not available.
|
|
|
* @type {?Long}
|
|
|
* @const
|
|
|
* @see https://github.com/dcodeIO/Long.js
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBuffer.Long = Long || null;
|
|
|
|
|
|
/**
|
|
|
* @alias ByteBuffer.prototype
|
|
|
* @inner
|
|
|
*/
|
|
|
var ByteBufferPrototype = ByteBuffer.prototype;
|
|
|
|
|
|
// helpers
|
|
|
|
|
|
/**
|
|
|
* @type {!ArrayBuffer}
|
|
|
* @inner
|
|
|
*/
|
|
|
var EMPTY_BUFFER = new ArrayBuffer(0);
|
|
|
|
|
|
/**
|
|
|
* String.fromCharCode reference for compile-time renaming.
|
|
|
* @type {function(...number):string}
|
|
|
* @inner
|
|
|
*/
|
|
|
var stringFromCharCode = String.fromCharCode;
|
|
|
|
|
|
/**
|
|
|
* Creates a source function for a string.
|
|
|
* @param {string} s String to read from
|
|
|
* @returns {function():number|null} Source function returning the next char code respectively `null` if there are
|
|
|
* no more characters left.
|
|
|
* @throws {TypeError} If the argument is invalid
|
|
|
* @inner
|
|
|
*/
|
|
|
function stringSource(s) {
|
|
|
var i=0; return function() {
|
|
|
return i < s.length ? s.charCodeAt(i++) : null;
|
|
|
};
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* Creates a destination function for a string.
|
|
|
* @returns {function(number=):undefined|string} Destination function successively called with the next char code.
|
|
|
* Returns the final string when called without arguments.
|
|
|
* @inner
|
|
|
*/
|
|
|
function stringDestination() {
|
|
|
var cs = [], ps = []; return function() {
|
|
|
if (arguments.length === 0)
|
|
|
return ps.join('')+stringFromCharCode.apply(String, cs);
|
|
|
if (cs.length + arguments.length > 1024)
|
|
|
ps.push(stringFromCharCode.apply(String, cs)),
|
|
|
cs.length = 0;
|
|
|
Array.prototype.push.apply(cs, arguments);
|
|
|
};
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* Allocates a new ByteBuffer backed by a buffer of the specified capacity.
|
|
|
* @param {number=} capacity Initial capacity. Defaults to {@link ByteBuffer.DEFAULT_CAPACITY}.
|
|
|
* @param {boolean=} littleEndian Whether to use little or big endian byte order. Defaults to
|
|
|
* {@link ByteBuffer.DEFAULT_ENDIAN}.
|
|
|
* @param {boolean=} noAssert Whether to skip assertions of offsets and values. Defaults to
|
|
|
* {@link ByteBuffer.DEFAULT_NOASSERT}.
|
|
|
* @returns {!ByteBuffer}
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBuffer.allocate = function(capacity, littleEndian, noAssert) {
|
|
|
return new ByteBuffer(capacity, littleEndian, noAssert);
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Concatenates multiple ByteBuffers into one.
|
|
|
* @param {!Array.<!ByteBuffer|!ArrayBuffer|!Uint8Array|string>} buffers Buffers to concatenate
|
|
|
* @param {(string|boolean)=} encoding String encoding if `buffers` contains a string ("base64", "hex", "binary",
|
|
|
* defaults to "utf8")
|
|
|
* @param {boolean=} littleEndian Whether to use little or big endian byte order for the resulting ByteBuffer. Defaults
|
|
|
* to {@link ByteBuffer.DEFAULT_ENDIAN}.
|
|
|
* @param {boolean=} noAssert Whether to skip assertions of offsets and values for the resulting ByteBuffer. Defaults to
|
|
|
* {@link ByteBuffer.DEFAULT_NOASSERT}.
|
|
|
* @returns {!ByteBuffer} Concatenated ByteBuffer
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBuffer.concat = function(buffers, encoding, littleEndian, noAssert) {
|
|
|
if (typeof encoding === 'boolean' || typeof encoding !== 'string') {
|
|
|
noAssert = littleEndian;
|
|
|
littleEndian = encoding;
|
|
|
encoding = undefined;
|
|
|
}
|
|
|
var capacity = 0;
|
|
|
for (var i=0, k=buffers.length, length; i<k; ++i) {
|
|
|
if (!ByteBuffer.isByteBuffer(buffers[i]))
|
|
|
buffers[i] = ByteBuffer.wrap(buffers[i], encoding);
|
|
|
length = buffers[i].limit - buffers[i].offset;
|
|
|
if (length > 0) capacity += length;
|
|
|
}
|
|
|
if (capacity === 0)
|
|
|
return new ByteBuffer(0, littleEndian, noAssert);
|
|
|
var bb = new ByteBuffer(capacity, littleEndian, noAssert),
|
|
|
bi;
|
|
|
var view = new Uint8Array(bb.buffer);
|
|
|
i=0; while (i<k) {
|
|
|
bi = buffers[i++];
|
|
|
length = bi.limit - bi.offset;
|
|
|
if (length <= 0) continue;
|
|
|
view.set(new Uint8Array(bi.buffer).subarray(bi.offset, bi.limit), bb.offset);
|
|
|
bb.offset += length;
|
|
|
}
|
|
|
bb.limit = bb.offset;
|
|
|
bb.offset = 0;
|
|
|
return bb;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Tests if the specified type is a ByteBuffer.
|
|
|
* @param {*} bb ByteBuffer to test
|
|
|
* @returns {boolean} `true` if it is a ByteBuffer, otherwise `false`
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBuffer.isByteBuffer = function(bb) {
|
|
|
return (bb && bb instanceof ByteBuffer) === true;
|
|
|
};
|
|
|
/**
|
|
|
* Gets the backing buffer type.
|
|
|
* @returns {Function} `Buffer` for NB builds, `ArrayBuffer` for AB builds (classes)
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBuffer.type = function() {
|
|
|
return ArrayBuffer;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Wraps a buffer or a string. Sets the allocated ByteBuffer's {@link ByteBuffer#offset} to `0` and its
|
|
|
* {@link ByteBuffer#limit} to the length of the wrapped data.
|
|
|
* @param {!ByteBuffer|!ArrayBuffer|!Uint8Array|string|!Array.<number>} buffer Anything that can be wrapped
|
|
|
* @param {(string|boolean)=} encoding String encoding if `buffer` is a string ("base64", "hex", "binary", defaults to
|
|
|
* "utf8")
|
|
|
* @param {boolean=} littleEndian Whether to use little or big endian byte order. Defaults to
|
|
|
* {@link ByteBuffer.DEFAULT_ENDIAN}.
|
|
|
* @param {boolean=} noAssert Whether to skip assertions of offsets and values. Defaults to
|
|
|
* {@link ByteBuffer.DEFAULT_NOASSERT}.
|
|
|
* @returns {!ByteBuffer} A ByteBuffer wrapping `buffer`
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBuffer.wrap = function(buffer, encoding, littleEndian, noAssert) {
|
|
|
if (typeof encoding !== 'string') {
|
|
|
noAssert = littleEndian;
|
|
|
littleEndian = encoding;
|
|
|
encoding = undefined;
|
|
|
}
|
|
|
if (typeof buffer === 'string') {
|
|
|
if (typeof encoding === 'undefined')
|
|
|
encoding = "utf8";
|
|
|
switch (encoding) {
|
|
|
case "base64":
|
|
|
return ByteBuffer.fromBase64(buffer, littleEndian);
|
|
|
case "hex":
|
|
|
return ByteBuffer.fromHex(buffer, littleEndian);
|
|
|
case "binary":
|
|
|
return ByteBuffer.fromBinary(buffer, littleEndian);
|
|
|
case "utf8":
|
|
|
return ByteBuffer.fromUTF8(buffer, littleEndian);
|
|
|
case "debug":
|
|
|
return ByteBuffer.fromDebug(buffer, littleEndian);
|
|
|
default:
|
|
|
throw Error("Unsupported encoding: "+encoding);
|
|
|
}
|
|
|
}
|
|
|
if (buffer === null || typeof buffer !== 'object')
|
|
|
throw TypeError("Illegal buffer");
|
|
|
var bb;
|
|
|
if (ByteBuffer.isByteBuffer(buffer)) {
|
|
|
bb = ByteBufferPrototype.clone.call(buffer);
|
|
|
bb.markedOffset = -1;
|
|
|
return bb;
|
|
|
}
|
|
|
if (buffer instanceof Uint8Array) { // Extract ArrayBuffer from Uint8Array
|
|
|
bb = new ByteBuffer(0, littleEndian, noAssert);
|
|
|
if (buffer.length > 0) { // Avoid references to more than one EMPTY_BUFFER
|
|
|
bb.buffer = buffer.buffer;
|
|
|
bb.offset = buffer.byteOffset;
|
|
|
bb.limit = buffer.byteOffset + buffer.length;
|
|
|
bb.view = buffer.length > 0 ? new DataView(buffer.buffer) : null;
|
|
|
}
|
|
|
} else if (buffer instanceof ArrayBuffer) { // Reuse ArrayBuffer
|
|
|
bb = new ByteBuffer(0, littleEndian, noAssert);
|
|
|
if (buffer.byteLength > 0) {
|
|
|
bb.buffer = buffer;
|
|
|
bb.offset = 0;
|
|
|
bb.limit = buffer.byteLength;
|
|
|
bb.view = buffer.byteLength > 0 ? new DataView(buffer) : null;
|
|
|
}
|
|
|
} else if (Object.prototype.toString.call(buffer) === "[object Array]") { // Create from octets
|
|
|
bb = new ByteBuffer(buffer.length, littleEndian, noAssert);
|
|
|
bb.limit = buffer.length;
|
|
|
for (i=0; i<buffer.length; ++i)
|
|
|
bb.view.setUint8(i, buffer[i]);
|
|
|
} else
|
|
|
throw TypeError("Illegal buffer"); // Otherwise fail
|
|
|
return bb;
|
|
|
};
|
|
|
|
|
|
// types/ints/int8
|
|
|
|
|
|
/**
|
|
|
* Writes an 8bit signed integer.
|
|
|
* @param {number} value Value to write
|
|
|
* @param {number=} offset Offset to write to. Will use and advance {@link ByteBuffer#offset} by `1` if omitted.
|
|
|
* @returns {!ByteBuffer} this
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBufferPrototype.writeInt8 = function(value, offset) {
|
|
|
var relative = typeof offset === 'undefined';
|
|
|
if (relative) offset = this.offset;
|
|
|
if (!this.noAssert) {
|
|
|
if (typeof value !== 'number' || value % 1 !== 0)
|
|
|
throw TypeError("Illegal value: "+value+" (not an integer)");
|
|
|
value |= 0;
|
|
|
if (typeof offset !== 'number' || offset % 1 !== 0)
|
|
|
throw TypeError("Illegal offset: "+offset+" (not an integer)");
|
|
|
offset >>>= 0;
|
|
|
if (offset < 0 || offset + 0 > this.buffer.byteLength)
|
|
|
throw RangeError("Illegal offset: 0 <= "+offset+" (+"+0+") <= "+this.buffer.byteLength);
|
|
|
}
|
|
|
offset += 1;
|
|
|
var capacity0 = this.buffer.byteLength;
|
|
|
if (offset > capacity0)
|
|
|
this.resize((capacity0 *= 2) > offset ? capacity0 : offset);
|
|
|
offset -= 1;
|
|
|
this.view.setInt8(offset, value);
|
|
|
if (relative) this.offset += 1;
|
|
|
return this;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Writes an 8bit signed integer. This is an alias of {@link ByteBuffer#writeInt8}.
|
|
|
* @function
|
|
|
* @param {number} value Value to write
|
|
|
* @param {number=} offset Offset to write to. Will use and advance {@link ByteBuffer#offset} by `1` if omitted.
|
|
|
* @returns {!ByteBuffer} this
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBufferPrototype.writeByte = ByteBufferPrototype.writeInt8;
|
|
|
|
|
|
/**
|
|
|
* Reads an 8bit signed integer.
|
|
|
* @param {number=} offset Offset to read from. Will use and advance {@link ByteBuffer#offset} by `1` if omitted.
|
|
|
* @returns {number} Value read
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBufferPrototype.readInt8 = function(offset) {
|
|
|
var relative = typeof offset === 'undefined';
|
|
|
if (relative) offset = this.offset;
|
|
|
if (!this.noAssert) {
|
|
|
if (typeof offset !== 'number' || offset % 1 !== 0)
|
|
|
throw TypeError("Illegal offset: "+offset+" (not an integer)");
|
|
|
offset >>>= 0;
|
|
|
if (offset < 0 || offset + 1 > this.buffer.byteLength)
|
|
|
throw RangeError("Illegal offset: 0 <= "+offset+" (+"+1+") <= "+this.buffer.byteLength);
|
|
|
}
|
|
|
var value = this.view.getInt8(offset);
|
|
|
if (relative) this.offset += 1;
|
|
|
return value;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Reads an 8bit signed integer. This is an alias of {@link ByteBuffer#readInt8}.
|
|
|
* @function
|
|
|
* @param {number=} offset Offset to read from. Will use and advance {@link ByteBuffer#offset} by `1` if omitted.
|
|
|
* @returns {number} Value read
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBufferPrototype.readByte = ByteBufferPrototype.readInt8;
|
|
|
|
|
|
/**
|
|
|
* Writes an 8bit unsigned integer.
|
|
|
* @param {number} value Value to write
|
|
|
* @param {number=} offset Offset to write to. Will use and advance {@link ByteBuffer#offset} by `1` if omitted.
|
|
|
* @returns {!ByteBuffer} this
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBufferPrototype.writeUint8 = function(value, offset) {
|
|
|
var relative = typeof offset === 'undefined';
|
|
|
if (relative) offset = this.offset;
|
|
|
if (!this.noAssert) {
|
|
|
if (typeof value !== 'number' || value % 1 !== 0)
|
|
|
throw TypeError("Illegal value: "+value+" (not an integer)");
|
|
|
value >>>= 0;
|
|
|
if (typeof offset !== 'number' || offset % 1 !== 0)
|
|
|
throw TypeError("Illegal offset: "+offset+" (not an integer)");
|
|
|
offset >>>= 0;
|
|
|
if (offset < 0 || offset + 0 > this.buffer.byteLength)
|
|
|
throw RangeError("Illegal offset: 0 <= "+offset+" (+"+0+") <= "+this.buffer.byteLength);
|
|
|
}
|
|
|
offset += 1;
|
|
|
var capacity1 = this.buffer.byteLength;
|
|
|
if (offset > capacity1)
|
|
|
this.resize((capacity1 *= 2) > offset ? capacity1 : offset);
|
|
|
offset -= 1;
|
|
|
this.view.setUint8(offset, value);
|
|
|
if (relative) this.offset += 1;
|
|
|
return this;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Reads an 8bit unsigned integer.
|
|
|
* @param {number=} offset Offset to read from. Will use and advance {@link ByteBuffer#offset} by `1` if omitted.
|
|
|
* @returns {number} Value read
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBufferPrototype.readUint8 = function(offset) {
|
|
|
var relative = typeof offset === 'undefined';
|
|
|
if (relative) offset = this.offset;
|
|
|
if (!this.noAssert) {
|
|
|
if (typeof offset !== 'number' || offset % 1 !== 0)
|
|
|
throw TypeError("Illegal offset: "+offset+" (not an integer)");
|
|
|
offset >>>= 0;
|
|
|
if (offset < 0 || offset + 1 > this.buffer.byteLength)
|
|
|
throw RangeError("Illegal offset: 0 <= "+offset+" (+"+1+") <= "+this.buffer.byteLength);
|
|
|
}
|
|
|
var value = this.view.getUint8(offset);
|
|
|
if (relative) this.offset += 1;
|
|
|
return value;
|
|
|
};
|
|
|
|
|
|
// types/ints/int16
|
|
|
|
|
|
/**
|
|
|
* Writes a 16bit signed integer.
|
|
|
* @param {number} value Value to write
|
|
|
* @param {number=} offset Offset to write to. Will use and advance {@link ByteBuffer#offset} by `2` if omitted.
|
|
|
* @throws {TypeError} If `offset` or `value` is not a valid number
|
|
|
* @throws {RangeError} If `offset` is out of bounds
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBufferPrototype.writeInt16 = function(value, offset) {
|
|
|
var relative = typeof offset === 'undefined';
|
|
|
if (relative) offset = this.offset;
|
|
|
if (!this.noAssert) {
|
|
|
if (typeof value !== 'number' || value % 1 !== 0)
|
|
|
throw TypeError("Illegal value: "+value+" (not an integer)");
|
|
|
value |= 0;
|
|
|
if (typeof offset !== 'number' || offset % 1 !== 0)
|
|
|
throw TypeError("Illegal offset: "+offset+" (not an integer)");
|
|
|
offset >>>= 0;
|
|
|
if (offset < 0 || offset + 0 > this.buffer.byteLength)
|
|
|
throw RangeError("Illegal offset: 0 <= "+offset+" (+"+0+") <= "+this.buffer.byteLength);
|
|
|
}
|
|
|
offset += 2;
|
|
|
var capacity2 = this.buffer.byteLength;
|
|
|
if (offset > capacity2)
|
|
|
this.resize((capacity2 *= 2) > offset ? capacity2 : offset);
|
|
|
offset -= 2;
|
|
|
this.view.setInt16(offset, value, this.littleEndian);
|
|
|
if (relative) this.offset += 2;
|
|
|
return this;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Writes a 16bit signed integer. This is an alias of {@link ByteBuffer#writeInt16}.
|
|
|
* @function
|
|
|
* @param {number} value Value to write
|
|
|
* @param {number=} offset Offset to write to. Will use and advance {@link ByteBuffer#offset} by `2` if omitted.
|
|
|
* @throws {TypeError} If `offset` or `value` is not a valid number
|
|
|
* @throws {RangeError} If `offset` is out of bounds
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBufferPrototype.writeShort = ByteBufferPrototype.writeInt16;
|
|
|
|
|
|
/**
|
|
|
* Reads a 16bit signed integer.
|
|
|
* @param {number=} offset Offset to read from. Will use and advance {@link ByteBuffer#offset} by `2` if omitted.
|
|
|
* @returns {number} Value read
|
|
|
* @throws {TypeError} If `offset` is not a valid number
|
|
|
* @throws {RangeError} If `offset` is out of bounds
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBufferPrototype.readInt16 = function(offset) {
|
|
|
var relative = typeof offset === 'undefined';
|
|
|
if (relative) offset = this.offset;
|
|
|
if (!this.noAssert) {
|
|
|
if (typeof offset !== 'number' || offset % 1 !== 0)
|
|
|
throw TypeError("Illegal offset: "+offset+" (not an integer)");
|
|
|
offset >>>= 0;
|
|
|
if (offset < 0 || offset + 2 > this.buffer.byteLength)
|
|
|
throw RangeError("Illegal offset: 0 <= "+offset+" (+"+2+") <= "+this.buffer.byteLength);
|
|
|
}
|
|
|
var value = this.view.getInt16(offset, this.littleEndian);
|
|
|
if (relative) this.offset += 2;
|
|
|
return value;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Reads a 16bit signed integer. This is an alias of {@link ByteBuffer#readInt16}.
|
|
|
* @function
|
|
|
* @param {number=} offset Offset to read from. Will use and advance {@link ByteBuffer#offset} by `2` if omitted.
|
|
|
* @returns {number} Value read
|
|
|
* @throws {TypeError} If `offset` is not a valid number
|
|
|
* @throws {RangeError} If `offset` is out of bounds
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBufferPrototype.readShort = ByteBufferPrototype.readInt16;
|
|
|
|
|
|
/**
|
|
|
* Writes a 16bit unsigned integer.
|
|
|
* @param {number} value Value to write
|
|
|
* @param {number=} offset Offset to write to. Will use and advance {@link ByteBuffer#offset} by `2` if omitted.
|
|
|
* @throws {TypeError} If `offset` or `value` is not a valid number
|
|
|
* @throws {RangeError} If `offset` is out of bounds
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBufferPrototype.writeUint16 = function(value, offset) {
|
|
|
var relative = typeof offset === 'undefined';
|
|
|
if (relative) offset = this.offset;
|
|
|
if (!this.noAssert) {
|
|
|
if (typeof value !== 'number' || value % 1 !== 0)
|
|
|
throw TypeError("Illegal value: "+value+" (not an integer)");
|
|
|
value >>>= 0;
|
|
|
if (typeof offset !== 'number' || offset % 1 !== 0)
|
|
|
throw TypeError("Illegal offset: "+offset+" (not an integer)");
|
|
|
offset >>>= 0;
|
|
|
if (offset < 0 || offset + 0 > this.buffer.byteLength)
|
|
|
throw RangeError("Illegal offset: 0 <= "+offset+" (+"+0+") <= "+this.buffer.byteLength);
|
|
|
}
|
|
|
offset += 2;
|
|
|
var capacity3 = this.buffer.byteLength;
|
|
|
if (offset > capacity3)
|
|
|
this.resize((capacity3 *= 2) > offset ? capacity3 : offset);
|
|
|
offset -= 2;
|
|
|
this.view.setUint16(offset, value, this.littleEndian);
|
|
|
if (relative) this.offset += 2;
|
|
|
return this;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Reads a 16bit unsigned integer.
|
|
|
* @param {number=} offset Offset to read from. Will use and advance {@link ByteBuffer#offset} by `2` if omitted.
|
|
|
* @returns {number} Value read
|
|
|
* @throws {TypeError} If `offset` is not a valid number
|
|
|
* @throws {RangeError} If `offset` is out of bounds
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBufferPrototype.readUint16 = function(offset) {
|
|
|
var relative = typeof offset === 'undefined';
|
|
|
if (relative) offset = this.offset;
|
|
|
if (!this.noAssert) {
|
|
|
if (typeof offset !== 'number' || offset % 1 !== 0)
|
|
|
throw TypeError("Illegal offset: "+offset+" (not an integer)");
|
|
|
offset >>>= 0;
|
|
|
if (offset < 0 || offset + 2 > this.buffer.byteLength)
|
|
|
throw RangeError("Illegal offset: 0 <= "+offset+" (+"+2+") <= "+this.buffer.byteLength);
|
|
|
}
|
|
|
var value = this.view.getUint16(offset, this.littleEndian);
|
|
|
if (relative) this.offset += 2;
|
|
|
return value;
|
|
|
};
|
|
|
|
|
|
// types/ints/int32
|
|
|
|
|
|
/**
|
|
|
* Writes a 32bit signed integer.
|
|
|
* @param {number} value Value to write
|
|
|
* @param {number=} offset Offset to write to. Will use and increase {@link ByteBuffer#offset} by `4` if omitted.
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBufferPrototype.writeInt32 = function(value, offset) {
|
|
|
var relative = typeof offset === 'undefined';
|
|
|
if (relative) offset = this.offset;
|
|
|
if (!this.noAssert) {
|
|
|
if (typeof value !== 'number' || value % 1 !== 0)
|
|
|
throw TypeError("Illegal value: "+value+" (not an integer)");
|
|
|
value |= 0;
|
|
|
if (typeof offset !== 'number' || offset % 1 !== 0)
|
|
|
throw TypeError("Illegal offset: "+offset+" (not an integer)");
|
|
|
offset >>>= 0;
|
|
|
if (offset < 0 || offset + 0 > this.buffer.byteLength)
|
|
|
throw RangeError("Illegal offset: 0 <= "+offset+" (+"+0+") <= "+this.buffer.byteLength);
|
|
|
}
|
|
|
offset += 4;
|
|
|
var capacity4 = this.buffer.byteLength;
|
|
|
if (offset > capacity4)
|
|
|
this.resize((capacity4 *= 2) > offset ? capacity4 : offset);
|
|
|
offset -= 4;
|
|
|
this.view.setInt32(offset, value, this.littleEndian);
|
|
|
if (relative) this.offset += 4;
|
|
|
return this;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Writes a 32bit signed integer. This is an alias of {@link ByteBuffer#writeInt32}.
|
|
|
* @param {number} value Value to write
|
|
|
* @param {number=} offset Offset to write to. Will use and increase {@link ByteBuffer#offset} by `4` if omitted.
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBufferPrototype.writeInt = ByteBufferPrototype.writeInt32;
|
|
|
|
|
|
/**
|
|
|
* Reads a 32bit signed integer.
|
|
|
* @param {number=} offset Offset to read from. Will use and increase {@link ByteBuffer#offset} by `4` if omitted.
|
|
|
* @returns {number} Value read
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBufferPrototype.readInt32 = function(offset) {
|
|
|
var relative = typeof offset === 'undefined';
|
|
|
if (relative) offset = this.offset;
|
|
|
if (!this.noAssert) {
|
|
|
if (typeof offset !== 'number' || offset % 1 !== 0)
|
|
|
throw TypeError("Illegal offset: "+offset+" (not an integer)");
|
|
|
offset >>>= 0;
|
|
|
if (offset < 0 || offset + 4 > this.buffer.byteLength)
|
|
|
throw RangeError("Illegal offset: 0 <= "+offset+" (+"+4+") <= "+this.buffer.byteLength);
|
|
|
}
|
|
|
var value = this.view.getInt32(offset, this.littleEndian);
|
|
|
if (relative) this.offset += 4;
|
|
|
return value;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Reads a 32bit signed integer. This is an alias of {@link ByteBuffer#readInt32}.
|
|
|
* @param {number=} offset Offset to read from. Will use and advance {@link ByteBuffer#offset} by `4` if omitted.
|
|
|
* @returns {number} Value read
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBufferPrototype.readInt = ByteBufferPrototype.readInt32;
|
|
|
|
|
|
/**
|
|
|
* Writes a 32bit unsigned integer.
|
|
|
* @param {number} value Value to write
|
|
|
* @param {number=} offset Offset to write to. Will use and increase {@link ByteBuffer#offset} by `4` if omitted.
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBufferPrototype.writeUint32 = function(value, offset) {
|
|
|
var relative = typeof offset === 'undefined';
|
|
|
if (relative) offset = this.offset;
|
|
|
if (!this.noAssert) {
|
|
|
if (typeof value !== 'number' || value % 1 !== 0)
|
|
|
throw TypeError("Illegal value: "+value+" (not an integer)");
|
|
|
value >>>= 0;
|
|
|
if (typeof offset !== 'number' || offset % 1 !== 0)
|
|
|
throw TypeError("Illegal offset: "+offset+" (not an integer)");
|
|
|
offset >>>= 0;
|
|
|
if (offset < 0 || offset + 0 > this.buffer.byteLength)
|
|
|
throw RangeError("Illegal offset: 0 <= "+offset+" (+"+0+") <= "+this.buffer.byteLength);
|
|
|
}
|
|
|
offset += 4;
|
|
|
var capacity5 = this.buffer.byteLength;
|
|
|
if (offset > capacity5)
|
|
|
this.resize((capacity5 *= 2) > offset ? capacity5 : offset);
|
|
|
offset -= 4;
|
|
|
this.view.setUint32(offset, value, this.littleEndian);
|
|
|
if (relative) this.offset += 4;
|
|
|
return this;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Reads a 32bit unsigned integer.
|
|
|
* @param {number=} offset Offset to read from. Will use and increase {@link ByteBuffer#offset} by `4` if omitted.
|
|
|
* @returns {number} Value read
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBufferPrototype.readUint32 = function(offset) {
|
|
|
var relative = typeof offset === 'undefined';
|
|
|
if (relative) offset = this.offset;
|
|
|
if (!this.noAssert) {
|
|
|
if (typeof offset !== 'number' || offset % 1 !== 0)
|
|
|
throw TypeError("Illegal offset: "+offset+" (not an integer)");
|
|
|
offset >>>= 0;
|
|
|
if (offset < 0 || offset + 4 > this.buffer.byteLength)
|
|
|
throw RangeError("Illegal offset: 0 <= "+offset+" (+"+4+") <= "+this.buffer.byteLength);
|
|
|
}
|
|
|
var value = this.view.getUint32(offset, this.littleEndian);
|
|
|
if (relative) this.offset += 4;
|
|
|
return value;
|
|
|
};
|
|
|
|
|
|
// types/ints/int64
|
|
|
|
|
|
if (Long) {
|
|
|
|
|
|
/**
|
|
|
* Writes a 64bit signed integer.
|
|
|
* @param {number|!Long} value Value to write
|
|
|
* @param {number=} offset Offset to write to. Will use and increase {@link ByteBuffer#offset} by `8` if omitted.
|
|
|
* @returns {!ByteBuffer} this
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBufferPrototype.writeInt64 = function(value, offset) {
|
|
|
var relative = typeof offset === 'undefined';
|
|
|
if (relative) offset = this.offset;
|
|
|
if (!this.noAssert) {
|
|
|
if (typeof value === 'number')
|
|
|
value = Long.fromNumber(value);
|
|
|
else if (!(value && value instanceof Long))
|
|
|
throw TypeError("Illegal value: "+value+" (not an integer or Long)");
|
|
|
if (typeof offset !== 'number' || offset % 1 !== 0)
|
|
|
throw TypeError("Illegal offset: "+offset+" (not an integer)");
|
|
|
offset >>>= 0;
|
|
|
if (offset < 0 || offset + 0 > this.buffer.byteLength)
|
|
|
throw RangeError("Illegal offset: 0 <= "+offset+" (+"+0+") <= "+this.buffer.byteLength);
|
|
|
}
|
|
|
if (typeof value === 'number')
|
|
|
value = Long.fromNumber(value);
|
|
|
offset += 8;
|
|
|
var capacity6 = this.buffer.byteLength;
|
|
|
if (offset > capacity6)
|
|
|
this.resize((capacity6 *= 2) > offset ? capacity6 : offset);
|
|
|
offset -= 8;
|
|
|
if (this.littleEndian) {
|
|
|
this.view.setInt32(offset , value.low , true);
|
|
|
this.view.setInt32(offset+4, value.high, true);
|
|
|
} else {
|
|
|
this.view.setInt32(offset , value.high, false);
|
|
|
this.view.setInt32(offset+4, value.low , false);
|
|
|
}
|
|
|
if (relative) this.offset += 8;
|
|
|
return this;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Writes a 64bit signed integer. This is an alias of {@link ByteBuffer#writeInt64}.
|
|
|
* @param {number|!Long} value Value to write
|
|
|
* @param {number=} offset Offset to write to. Will use and increase {@link ByteBuffer#offset} by `8` if omitted.
|
|
|
* @returns {!ByteBuffer} this
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBufferPrototype.writeLong = ByteBufferPrototype.writeInt64;
|
|
|
|
|
|
/**
|
|
|
* Reads a 64bit signed integer.
|
|
|
* @param {number=} offset Offset to read from. Will use and increase {@link ByteBuffer#offset} by `8` if omitted.
|
|
|
* @returns {!Long}
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBufferPrototype.readInt64 = function(offset) {
|
|
|
var relative = typeof offset === 'undefined';
|
|
|
if (relative) offset = this.offset;
|
|
|
if (!this.noAssert) {
|
|
|
if (typeof offset !== 'number' || offset % 1 !== 0)
|
|
|
throw TypeError("Illegal offset: "+offset+" (not an integer)");
|
|
|
offset >>>= 0;
|
|
|
if (offset < 0 || offset + 8 > this.buffer.byteLength)
|
|
|
throw RangeError("Illegal offset: 0 <= "+offset+" (+"+8+") <= "+this.buffer.byteLength);
|
|
|
}
|
|
|
var value = this.littleEndian
|
|
|
? new Long(this.view.getInt32(offset , true ), this.view.getInt32(offset+4, true ), false)
|
|
|
: new Long(this.view.getInt32(offset+4, false), this.view.getInt32(offset , false), false);
|
|
|
if (relative) this.offset += 8;
|
|
|
return value;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Reads a 64bit signed integer. This is an alias of {@link ByteBuffer#readInt64}.
|
|
|
* @param {number=} offset Offset to read from. Will use and increase {@link ByteBuffer#offset} by `8` if omitted.
|
|
|
* @returns {!Long}
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBufferPrototype.readLong = ByteBufferPrototype.readInt64;
|
|
|
|
|
|
/**
|
|
|
* Writes a 64bit unsigned integer.
|
|
|
* @param {number|!Long} value Value to write
|
|
|
* @param {number=} offset Offset to write to. Will use and increase {@link ByteBuffer#offset} by `8` if omitted.
|
|
|
* @returns {!ByteBuffer} this
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBufferPrototype.writeUint64 = function(value, offset) {
|
|
|
var relative = typeof offset === 'undefined';
|
|
|
if (relative) offset = this.offset;
|
|
|
if (!this.noAssert) {
|
|
|
if (typeof value === 'number')
|
|
|
value = Long.fromNumber(value);
|
|
|
else if (!(value && value instanceof Long))
|
|
|
throw TypeError("Illegal value: "+value+" (not an integer or Long)");
|
|
|
if (typeof offset !== 'number' || offset % 1 !== 0)
|
|
|
throw TypeError("Illegal offset: "+offset+" (not an integer)");
|
|
|
offset >>>= 0;
|
|
|
if (offset < 0 || offset + 0 > this.buffer.byteLength)
|
|
|
throw RangeError("Illegal offset: 0 <= "+offset+" (+"+0+") <= "+this.buffer.byteLength);
|
|
|
}
|
|
|
if (typeof value === 'number')
|
|
|
value = Long.fromNumber(value);
|
|
|
offset += 8;
|
|
|
var capacity7 = this.buffer.byteLength;
|
|
|
if (offset > capacity7)
|
|
|
this.resize((capacity7 *= 2) > offset ? capacity7 : offset);
|
|
|
offset -= 8;
|
|
|
if (this.littleEndian) {
|
|
|
this.view.setInt32(offset , value.low , true);
|
|
|
this.view.setInt32(offset+4, value.high, true);
|
|
|
} else {
|
|
|
this.view.setInt32(offset , value.high, false);
|
|
|
this.view.setInt32(offset+4, value.low , false);
|
|
|
}
|
|
|
if (relative) this.offset += 8;
|
|
|
return this;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Reads a 64bit unsigned integer.
|
|
|
* @param {number=} offset Offset to read from. Will use and increase {@link ByteBuffer#offset} by `8` if omitted.
|
|
|
* @returns {!Long}
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBufferPrototype.readUint64 = function(offset) {
|
|
|
var relative = typeof offset === 'undefined';
|
|
|
if (relative) offset = this.offset;
|
|
|
if (!this.noAssert) {
|
|
|
if (typeof offset !== 'number' || offset % 1 !== 0)
|
|
|
throw TypeError("Illegal offset: "+offset+" (not an integer)");
|
|
|
offset >>>= 0;
|
|
|
if (offset < 0 || offset + 8 > this.buffer.byteLength)
|
|
|
throw RangeError("Illegal offset: 0 <= "+offset+" (+"+8+") <= "+this.buffer.byteLength);
|
|
|
}
|
|
|
var value = this.littleEndian
|
|
|
? new Long(this.view.getInt32(offset , true ), this.view.getInt32(offset+4, true ), true)
|
|
|
: new Long(this.view.getInt32(offset+4, false), this.view.getInt32(offset , false), true);
|
|
|
if (relative) this.offset += 8;
|
|
|
return value;
|
|
|
};
|
|
|
|
|
|
} // Long
|
|
|
|
|
|
|
|
|
// types/floats/float32
|
|
|
|
|
|
/**
|
|
|
* Writes a 32bit float.
|
|
|
* @param {number} value Value to write
|
|
|
* @param {number=} offset Offset to write to. Will use and increase {@link ByteBuffer#offset} by `4` if omitted.
|
|
|
* @returns {!ByteBuffer} this
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBufferPrototype.writeFloat32 = function(value, offset) {
|
|
|
var relative = typeof offset === 'undefined';
|
|
|
if (relative) offset = this.offset;
|
|
|
if (!this.noAssert) {
|
|
|
if (typeof value !== 'number')
|
|
|
throw TypeError("Illegal value: "+value+" (not a number)");
|
|
|
if (typeof offset !== 'number' || offset % 1 !== 0)
|
|
|
throw TypeError("Illegal offset: "+offset+" (not an integer)");
|
|
|
offset >>>= 0;
|
|
|
if (offset < 0 || offset + 0 > this.buffer.byteLength)
|
|
|
throw RangeError("Illegal offset: 0 <= "+offset+" (+"+0+") <= "+this.buffer.byteLength);
|
|
|
}
|
|
|
offset += 4;
|
|
|
var capacity8 = this.buffer.byteLength;
|
|
|
if (offset > capacity8)
|
|
|
this.resize((capacity8 *= 2) > offset ? capacity8 : offset);
|
|
|
offset -= 4;
|
|
|
this.view.setFloat32(offset, value, this.littleEndian);
|
|
|
if (relative) this.offset += 4;
|
|
|
return this;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Writes a 32bit float. This is an alias of {@link ByteBuffer#writeFloat32}.
|
|
|
* @function
|
|
|
* @param {number} value Value to write
|
|
|
* @param {number=} offset Offset to write to. Will use and increase {@link ByteBuffer#offset} by `4` if omitted.
|
|
|
* @returns {!ByteBuffer} this
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBufferPrototype.writeFloat = ByteBufferPrototype.writeFloat32;
|
|
|
|
|
|
/**
|
|
|
* Reads a 32bit float.
|
|
|
* @param {number=} offset Offset to read from. Will use and increase {@link ByteBuffer#offset} by `4` if omitted.
|
|
|
* @returns {number}
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBufferPrototype.readFloat32 = function(offset) {
|
|
|
var relative = typeof offset === 'undefined';
|
|
|
if (relative) offset = this.offset;
|
|
|
if (!this.noAssert) {
|
|
|
if (typeof offset !== 'number' || offset % 1 !== 0)
|
|
|
throw TypeError("Illegal offset: "+offset+" (not an integer)");
|
|
|
offset >>>= 0;
|
|
|
if (offset < 0 || offset + 4 > this.buffer.byteLength)
|
|
|
throw RangeError("Illegal offset: 0 <= "+offset+" (+"+4+") <= "+this.buffer.byteLength);
|
|
|
}
|
|
|
var value = this.view.getFloat32(offset, this.littleEndian);
|
|
|
if (relative) this.offset += 4;
|
|
|
return value;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Reads a 32bit float. This is an alias of {@link ByteBuffer#readFloat32}.
|
|
|
* @function
|
|
|
* @param {number=} offset Offset to read from. Will use and increase {@link ByteBuffer#offset} by `4` if omitted.
|
|
|
* @returns {number}
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBufferPrototype.readFloat = ByteBufferPrototype.readFloat32;
|
|
|
|
|
|
// types/floats/float64
|
|
|
|
|
|
/**
|
|
|
* Writes a 64bit float.
|
|
|
* @param {number} value Value to write
|
|
|
* @param {number=} offset Offset to write to. Will use and increase {@link ByteBuffer#offset} by `8` if omitted.
|
|
|
* @returns {!ByteBuffer} this
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBufferPrototype.writeFloat64 = function(value, offset) {
|
|
|
var relative = typeof offset === 'undefined';
|
|
|
if (relative) offset = this.offset;
|
|
|
if (!this.noAssert) {
|
|
|
if (typeof value !== 'number')
|
|
|
throw TypeError("Illegal value: "+value+" (not a number)");
|
|
|
if (typeof offset !== 'number' || offset % 1 !== 0)
|
|
|
throw TypeError("Illegal offset: "+offset+" (not an integer)");
|
|
|
offset >>>= 0;
|
|
|
if (offset < 0 || offset + 0 > this.buffer.byteLength)
|
|
|
throw RangeError("Illegal offset: 0 <= "+offset+" (+"+0+") <= "+this.buffer.byteLength);
|
|
|
}
|
|
|
offset += 8;
|
|
|
var capacity9 = this.buffer.byteLength;
|
|
|
if (offset > capacity9)
|
|
|
this.resize((capacity9 *= 2) > offset ? capacity9 : offset);
|
|
|
offset -= 8;
|
|
|
this.view.setFloat64(offset, value, this.littleEndian);
|
|
|
if (relative) this.offset += 8;
|
|
|
return this;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Writes a 64bit float. This is an alias of {@link ByteBuffer#writeFloat64}.
|
|
|
* @function
|
|
|
* @param {number} value Value to write
|
|
|
* @param {number=} offset Offset to write to. Will use and increase {@link ByteBuffer#offset} by `8` if omitted.
|
|
|
* @returns {!ByteBuffer} this
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBufferPrototype.writeDouble = ByteBufferPrototype.writeFloat64;
|
|
|
|
|
|
/**
|
|
|
* Reads a 64bit float.
|
|
|
* @param {number=} offset Offset to read from. Will use and increase {@link ByteBuffer#offset} by `8` if omitted.
|
|
|
* @returns {number}
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBufferPrototype.readFloat64 = function(offset) {
|
|
|
var relative = typeof offset === 'undefined';
|
|
|
if (relative) offset = this.offset;
|
|
|
if (!this.noAssert) {
|
|
|
if (typeof offset !== 'number' || offset % 1 !== 0)
|
|
|
throw TypeError("Illegal offset: "+offset+" (not an integer)");
|
|
|
offset >>>= 0;
|
|
|
if (offset < 0 || offset + 8 > this.buffer.byteLength)
|
|
|
throw RangeError("Illegal offset: 0 <= "+offset+" (+"+8+") <= "+this.buffer.byteLength);
|
|
|
}
|
|
|
var value = this.view.getFloat64(offset, this.littleEndian);
|
|
|
if (relative) this.offset += 8;
|
|
|
return value;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Reads a 64bit float. This is an alias of {@link ByteBuffer#readFloat64}.
|
|
|
* @function
|
|
|
* @param {number=} offset Offset to read from. Will use and increase {@link ByteBuffer#offset} by `8` if omitted.
|
|
|
* @returns {number}
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBufferPrototype.readDouble = ByteBufferPrototype.readFloat64;
|
|
|
|
|
|
|
|
|
// types/varints/varint32
|
|
|
|
|
|
/**
|
|
|
* Maximum number of bytes required to store a 32bit base 128 variable-length integer.
|
|
|
* @type {number}
|
|
|
* @const
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBuffer.MAX_VARINT32_BYTES = 5;
|
|
|
|
|
|
/**
|
|
|
* Calculates the actual number of bytes required to store a 32bit base 128 variable-length integer.
|
|
|
* @param {number} value Value to encode
|
|
|
* @returns {number} Number of bytes required. Capped to {@link ByteBuffer.MAX_VARINT32_BYTES}
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBuffer.calculateVarint32 = function(value) {
|
|
|
// ref: src/google/protobuf/io/coded_stream.cc
|
|
|
value = value >>> 0;
|
|
|
if (value < 1 << 7 ) return 1;
|
|
|
else if (value < 1 << 14) return 2;
|
|
|
else if (value < 1 << 21) return 3;
|
|
|
else if (value < 1 << 28) return 4;
|
|
|
else return 5;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Zigzag encodes a signed 32bit integer so that it can be effectively used with varint encoding.
|
|
|
* @param {number} n Signed 32bit integer
|
|
|
* @returns {number} Unsigned zigzag encoded 32bit integer
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBuffer.zigZagEncode32 = function(n) {
|
|
|
return (((n |= 0) << 1) ^ (n >> 31)) >>> 0; // ref: src/google/protobuf/wire_format_lite.h
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Decodes a zigzag encoded signed 32bit integer.
|
|
|
* @param {number} n Unsigned zigzag encoded 32bit integer
|
|
|
* @returns {number} Signed 32bit integer
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBuffer.zigZagDecode32 = function(n) {
|
|
|
return ((n >>> 1) ^ -(n & 1)) | 0; // // ref: src/google/protobuf/wire_format_lite.h
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Writes a 32bit base 128 variable-length integer.
|
|
|
* @param {number} value Value to write
|
|
|
* @param {number=} offset Offset to write to. Will use and increase {@link ByteBuffer#offset} by the number of bytes
|
|
|
* written if omitted.
|
|
|
* @returns {!ByteBuffer|number} this if `offset` is omitted, else the actual number of bytes written
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBufferPrototype.writeVarint32 = function(value, offset) {
|
|
|
var relative = typeof offset === 'undefined';
|
|
|
if (relative) offset = this.offset;
|
|
|
if (!this.noAssert) {
|
|
|
if (typeof value !== 'number' || value % 1 !== 0)
|
|
|
throw TypeError("Illegal value: "+value+" (not an integer)");
|
|
|
value |= 0;
|
|
|
if (typeof offset !== 'number' || offset % 1 !== 0)
|
|
|
throw TypeError("Illegal offset: "+offset+" (not an integer)");
|
|
|
offset >>>= 0;
|
|
|
if (offset < 0 || offset + 0 > this.buffer.byteLength)
|
|
|
throw RangeError("Illegal offset: 0 <= "+offset+" (+"+0+") <= "+this.buffer.byteLength);
|
|
|
}
|
|
|
var size = ByteBuffer.calculateVarint32(value),
|
|
|
b;
|
|
|
offset += size;
|
|
|
var capacity10 = this.buffer.byteLength;
|
|
|
if (offset > capacity10)
|
|
|
this.resize((capacity10 *= 2) > offset ? capacity10 : offset);
|
|
|
offset -= size;
|
|
|
// ref: http://code.google.com/searchframe#WTeibokF6gE/trunk/src/google/protobuf/io/coded_stream.cc
|
|
|
this.view.setUint8(offset, b = value | 0x80);
|
|
|
value >>>= 0;
|
|
|
if (value >= 1 << 7) {
|
|
|
b = (value >> 7) | 0x80;
|
|
|
this.view.setUint8(offset+1, b);
|
|
|
if (value >= 1 << 14) {
|
|
|
b = (value >> 14) | 0x80;
|
|
|
this.view.setUint8(offset+2, b);
|
|
|
if (value >= 1 << 21) {
|
|
|
b = (value >> 21) | 0x80;
|
|
|
this.view.setUint8(offset+3, b);
|
|
|
if (value >= 1 << 28) {
|
|
|
this.view.setUint8(offset+4, (value >> 28) & 0x0F);
|
|
|
size = 5;
|
|
|
} else {
|
|
|
this.view.setUint8(offset+3, b & 0x7F);
|
|
|
size = 4;
|
|
|
}
|
|
|
} else {
|
|
|
this.view.setUint8(offset+2, b & 0x7F);
|
|
|
size = 3;
|
|
|
}
|
|
|
} else {
|
|
|
this.view.setUint8(offset+1, b & 0x7F);
|
|
|
size = 2;
|
|
|
}
|
|
|
} else {
|
|
|
this.view.setUint8(offset, b & 0x7F);
|
|
|
size = 1;
|
|
|
}
|
|
|
if (relative) {
|
|
|
this.offset += size;
|
|
|
return this;
|
|
|
}
|
|
|
return size;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Writes a zig-zag encoded 32bit base 128 variable-length integer.
|
|
|
* @param {number} value Value to write
|
|
|
* @param {number=} offset Offset to write to. Will use and increase {@link ByteBuffer#offset} by the number of bytes
|
|
|
* written if omitted.
|
|
|
* @returns {!ByteBuffer|number} this if `offset` is omitted, else the actual number of bytes written
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBufferPrototype.writeVarint32ZigZag = function(value, offset) {
|
|
|
return this.writeVarint32(ByteBuffer.zigZagEncode32(value), offset);
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Reads a 32bit base 128 variable-length integer.
|
|
|
* @param {number=} offset Offset to read from. Will use and increase {@link ByteBuffer#offset} by the number of bytes
|
|
|
* written if omitted.
|
|
|
* @returns {number|!{value: number, length: number}} The value read if offset is omitted, else the value read
|
|
|
* and the actual number of bytes read.
|
|
|
* @throws {Error} If it's not a valid varint. Has a property `truncated = true` if there is not enough data available
|
|
|
* to fully decode the varint.
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBufferPrototype.readVarint32 = function(offset) {
|
|
|
var relative = typeof offset === 'undefined';
|
|
|
if (relative) offset = this.offset;
|
|
|
if (!this.noAssert) {
|
|
|
if (typeof offset !== 'number' || offset % 1 !== 0)
|
|
|
throw TypeError("Illegal offset: "+offset+" (not an integer)");
|
|
|
offset >>>= 0;
|
|
|
if (offset < 0 || offset + 1 > this.buffer.byteLength)
|
|
|
throw RangeError("Illegal offset: 0 <= "+offset+" (+"+1+") <= "+this.buffer.byteLength);
|
|
|
}
|
|
|
// ref: src/google/protobuf/io/coded_stream.cc
|
|
|
var size = 0,
|
|
|
value = 0 >>> 0,
|
|
|
temp,
|
|
|
ioffset;
|
|
|
do {
|
|
|
ioffset = offset+size;
|
|
|
if (!this.noAssert && ioffset > this.limit) {
|
|
|
var err = Error("Truncated");
|
|
|
err['truncated'] = true;
|
|
|
throw err;
|
|
|
}
|
|
|
temp = this.view.getUint8(ioffset);
|
|
|
if (size < 5)
|
|
|
value |= ((temp&0x7F)<<(7*size)) >>> 0;
|
|
|
++size;
|
|
|
} while ((temp & 0x80) === 0x80);
|
|
|
value = value | 0; // Make sure to discard the higher order bits
|
|
|
if (relative) {
|
|
|
this.offset += size;
|
|
|
return value;
|
|
|
}
|
|
|
return {
|
|
|
"value": value,
|
|
|
"length": size
|
|
|
};
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Reads a zig-zag encoded 32bit base 128 variable-length integer.
|
|
|
* @param {number=} offset Offset to read from. Will use and increase {@link ByteBuffer#offset} by the number of bytes
|
|
|
* written if omitted.
|
|
|
* @returns {number|!{value: number, length: number}} The value read if offset is omitted, else the value read
|
|
|
* and the actual number of bytes read.
|
|
|
* @throws {Error} If it's not a valid varint
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBufferPrototype.readVarint32ZigZag = function(offset) {
|
|
|
var val = this.readVarint32(offset);
|
|
|
if (typeof val === 'object')
|
|
|
val["value"] = ByteBuffer.zigZagDecode32(val["value"]);
|
|
|
else
|
|
|
val = ByteBuffer.zigZagDecode32(val);
|
|
|
return val;
|
|
|
};
|
|
|
|
|
|
// types/varints/varint64
|
|
|
|
|
|
if (Long) {
|
|
|
|
|
|
/**
|
|
|
* Maximum number of bytes required to store a 64bit base 128 variable-length integer.
|
|
|
* @type {number}
|
|
|
* @const
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBuffer.MAX_VARINT64_BYTES = 10;
|
|
|
|
|
|
/**
|
|
|
* Calculates the actual number of bytes required to store a 64bit base 128 variable-length integer.
|
|
|
* @param {number|!Long} value Value to encode
|
|
|
* @returns {number} Number of bytes required. Capped to {@link ByteBuffer.MAX_VARINT64_BYTES}
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBuffer.calculateVarint64 = function(value) {
|
|
|
if (typeof value === 'number')
|
|
|
value = Long.fromNumber(value);
|
|
|
// ref: src/google/protobuf/io/coded_stream.cc
|
|
|
var part0 = value.toInt() >>> 0,
|
|
|
part1 = value.shiftRightUnsigned(28).toInt() >>> 0,
|
|
|
part2 = value.shiftRightUnsigned(56).toInt() >>> 0;
|
|
|
if (part2 == 0) {
|
|
|
if (part1 == 0) {
|
|
|
if (part0 < 1 << 14)
|
|
|
return part0 < 1 << 7 ? 1 : 2;
|
|
|
else
|
|
|
return part0 < 1 << 21 ? 3 : 4;
|
|
|
} else {
|
|
|
if (part1 < 1 << 14)
|
|
|
return part1 < 1 << 7 ? 5 : 6;
|
|
|
else
|
|
|
return part1 < 1 << 21 ? 7 : 8;
|
|
|
}
|
|
|
} else
|
|
|
return part2 < 1 << 7 ? 9 : 10;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Zigzag encodes a signed 64bit integer so that it can be effectively used with varint encoding.
|
|
|
* @param {number|!Long} value Signed long
|
|
|
* @returns {!Long} Unsigned zigzag encoded long
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBuffer.zigZagEncode64 = function(value) {
|
|
|
if (typeof value === 'number')
|
|
|
value = Long.fromNumber(value, false);
|
|
|
else if (value.unsigned !== false) value = value.toSigned();
|
|
|
// ref: src/google/protobuf/wire_format_lite.h
|
|
|
return value.shiftLeft(1).xor(value.shiftRight(63)).toUnsigned();
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Decodes a zigzag encoded signed 64bit integer.
|
|
|
* @param {!Long|number} value Unsigned zigzag encoded long or JavaScript number
|
|
|
* @returns {!Long} Signed long
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBuffer.zigZagDecode64 = function(value) {
|
|
|
if (typeof value === 'number')
|
|
|
value = Long.fromNumber(value, false);
|
|
|
else if (value.unsigned !== false) value = value.toSigned();
|
|
|
// ref: src/google/protobuf/wire_format_lite.h
|
|
|
return value.shiftRightUnsigned(1).xor(value.and(Long.ONE).toSigned().negate()).toSigned();
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Writes a 64bit base 128 variable-length integer.
|
|
|
* @param {number|Long} value Value to write
|
|
|
* @param {number=} offset Offset to write to. Will use and increase {@link ByteBuffer#offset} by the number of bytes
|
|
|
* written if omitted.
|
|
|
* @returns {!ByteBuffer|number} `this` if offset is omitted, else the actual number of bytes written.
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBufferPrototype.writeVarint64 = function(value, offset) {
|
|
|
var relative = typeof offset === 'undefined';
|
|
|
if (relative) offset = this.offset;
|
|
|
if (!this.noAssert) {
|
|
|
if (typeof value === 'number')
|
|
|
value = Long.fromNumber(value);
|
|
|
else if (!(value && value instanceof Long))
|
|
|
throw TypeError("Illegal value: "+value+" (not an integer or Long)");
|
|
|
if (typeof offset !== 'number' || offset % 1 !== 0)
|
|
|
throw TypeError("Illegal offset: "+offset+" (not an integer)");
|
|
|
offset >>>= 0;
|
|
|
if (offset < 0 || offset + 0 > this.buffer.byteLength)
|
|
|
throw RangeError("Illegal offset: 0 <= "+offset+" (+"+0+") <= "+this.buffer.byteLength);
|
|
|
}
|
|
|
if (typeof value === 'number')
|
|
|
value = Long.fromNumber(value, false);
|
|
|
else if (value.unsigned !== false) value = value.toSigned();
|
|
|
var size = ByteBuffer.calculateVarint64(value),
|
|
|
part0 = value.toInt() >>> 0,
|
|
|
part1 = value.shiftRightUnsigned(28).toInt() >>> 0,
|
|
|
part2 = value.shiftRightUnsigned(56).toInt() >>> 0;
|
|
|
offset += size;
|
|
|
var capacity11 = this.buffer.byteLength;
|
|
|
if (offset > capacity11)
|
|
|
this.resize((capacity11 *= 2) > offset ? capacity11 : offset);
|
|
|
offset -= size;
|
|
|
switch (size) {
|
|
|
case 10: this.view.setUint8(offset+9, (part2 >>> 7) & 0x01);
|
|
|
case 9 : this.view.setUint8(offset+8, size !== 9 ? (part2 ) | 0x80 : (part2 ) & 0x7F);
|
|
|
case 8 : this.view.setUint8(offset+7, size !== 8 ? (part1 >>> 21) | 0x80 : (part1 >>> 21) & 0x7F);
|
|
|
case 7 : this.view.setUint8(offset+6, size !== 7 ? (part1 >>> 14) | 0x80 : (part1 >>> 14) & 0x7F);
|
|
|
case 6 : this.view.setUint8(offset+5, size !== 6 ? (part1 >>> 7) | 0x80 : (part1 >>> 7) & 0x7F);
|
|
|
case 5 : this.view.setUint8(offset+4, size !== 5 ? (part1 ) | 0x80 : (part1 ) & 0x7F);
|
|
|
case 4 : this.view.setUint8(offset+3, size !== 4 ? (part0 >>> 21) | 0x80 : (part0 >>> 21) & 0x7F);
|
|
|
case 3 : this.view.setUint8(offset+2, size !== 3 ? (part0 >>> 14) | 0x80 : (part0 >>> 14) & 0x7F);
|
|
|
case 2 : this.view.setUint8(offset+1, size !== 2 ? (part0 >>> 7) | 0x80 : (part0 >>> 7) & 0x7F);
|
|
|
case 1 : this.view.setUint8(offset , size !== 1 ? (part0 ) | 0x80 : (part0 ) & 0x7F);
|
|
|
}
|
|
|
if (relative) {
|
|
|
this.offset += size;
|
|
|
return this;
|
|
|
} else {
|
|
|
return size;
|
|
|
}
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Writes a zig-zag encoded 64bit base 128 variable-length integer.
|
|
|
* @param {number|Long} value Value to write
|
|
|
* @param {number=} offset Offset to write to. Will use and increase {@link ByteBuffer#offset} by the number of bytes
|
|
|
* written if omitted.
|
|
|
* @returns {!ByteBuffer|number} `this` if offset is omitted, else the actual number of bytes written.
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBufferPrototype.writeVarint64ZigZag = function(value, offset) {
|
|
|
return this.writeVarint64(ByteBuffer.zigZagEncode64(value), offset);
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Reads a 64bit base 128 variable-length integer. Requires Long.js.
|
|
|
* @param {number=} offset Offset to read from. Will use and increase {@link ByteBuffer#offset} by the number of bytes
|
|
|
* read if omitted.
|
|
|
* @returns {!Long|!{value: Long, length: number}} The value read if offset is omitted, else the value read and
|
|
|
* the actual number of bytes read.
|
|
|
* @throws {Error} If it's not a valid varint
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBufferPrototype.readVarint64 = function(offset) {
|
|
|
var relative = typeof offset === 'undefined';
|
|
|
if (relative) offset = this.offset;
|
|
|
if (!this.noAssert) {
|
|
|
if (typeof offset !== 'number' || offset % 1 !== 0)
|
|
|
throw TypeError("Illegal offset: "+offset+" (not an integer)");
|
|
|
offset >>>= 0;
|
|
|
if (offset < 0 || offset + 1 > this.buffer.byteLength)
|
|
|
throw RangeError("Illegal offset: 0 <= "+offset+" (+"+1+") <= "+this.buffer.byteLength);
|
|
|
}
|
|
|
// ref: src/google/protobuf/io/coded_stream.cc
|
|
|
var start = offset,
|
|
|
part0 = 0,
|
|
|
part1 = 0,
|
|
|
part2 = 0,
|
|
|
b = 0;
|
|
|
b = this.view.getUint8(offset++); part0 = (b & 0x7F) ; if (b & 0x80) {
|
|
|
b = this.view.getUint8(offset++); part0 |= (b & 0x7F) << 7; if (b & 0x80) {
|
|
|
b = this.view.getUint8(offset++); part0 |= (b & 0x7F) << 14; if (b & 0x80) {
|
|
|
b = this.view.getUint8(offset++); part0 |= (b & 0x7F) << 21; if (b & 0x80) {
|
|
|
b = this.view.getUint8(offset++); part1 = (b & 0x7F) ; if (b & 0x80) {
|
|
|
b = this.view.getUint8(offset++); part1 |= (b & 0x7F) << 7; if (b & 0x80) {
|
|
|
b = this.view.getUint8(offset++); part1 |= (b & 0x7F) << 14; if (b & 0x80) {
|
|
|
b = this.view.getUint8(offset++); part1 |= (b & 0x7F) << 21; if (b & 0x80) {
|
|
|
b = this.view.getUint8(offset++); part2 = (b & 0x7F) ; if (b & 0x80) {
|
|
|
b = this.view.getUint8(offset++); part2 |= (b & 0x7F) << 7; if (b & 0x80) {
|
|
|
throw Error("Buffer overrun"); }}}}}}}}}}
|
|
|
var value = Long.fromBits(part0 | (part1 << 28), (part1 >>> 4) | (part2) << 24, false);
|
|
|
if (relative) {
|
|
|
this.offset = offset;
|
|
|
return value;
|
|
|
} else {
|
|
|
return {
|
|
|
'value': value,
|
|
|
'length': offset-start
|
|
|
};
|
|
|
}
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Reads a zig-zag encoded 64bit base 128 variable-length integer. Requires Long.js.
|
|
|
* @param {number=} offset Offset to read from. Will use and increase {@link ByteBuffer#offset} by the number of bytes
|
|
|
* read if omitted.
|
|
|
* @returns {!Long|!{value: Long, length: number}} The value read if offset is omitted, else the value read and
|
|
|
* the actual number of bytes read.
|
|
|
* @throws {Error} If it's not a valid varint
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBufferPrototype.readVarint64ZigZag = function(offset) {
|
|
|
var val = this.readVarint64(offset);
|
|
|
if (val && val['value'] instanceof Long)
|
|
|
val["value"] = ByteBuffer.zigZagDecode64(val["value"]);
|
|
|
else
|
|
|
val = ByteBuffer.zigZagDecode64(val);
|
|
|
return val;
|
|
|
};
|
|
|
|
|
|
} // Long
|
|
|
|
|
|
|
|
|
// types/strings/cstring
|
|
|
|
|
|
/**
|
|
|
* Writes a NULL-terminated UTF8 encoded string. For this to work the specified string must not contain any NULL
|
|
|
* characters itself.
|
|
|
* @param {string} str String to write
|
|
|
* @param {number=} offset Offset to write to. Will use and increase {@link ByteBuffer#offset} by the number of bytes
|
|
|
* contained in `str` + 1 if omitted.
|
|
|
* @returns {!ByteBuffer|number} this if offset is omitted, else the actual number of bytes written
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBufferPrototype.writeCString = function(str, offset) {
|
|
|
var relative = typeof offset === 'undefined';
|
|
|
if (relative) offset = this.offset;
|
|
|
var i,
|
|
|
k = str.length;
|
|
|
if (!this.noAssert) {
|
|
|
if (typeof str !== 'string')
|
|
|
throw TypeError("Illegal str: Not a string");
|
|
|
for (i=0; i<k; ++i) {
|
|
|
if (str.charCodeAt(i) === 0)
|
|
|
throw RangeError("Illegal str: Contains NULL-characters");
|
|
|
}
|
|
|
if (typeof offset !== 'number' || offset % 1 !== 0)
|
|
|
throw TypeError("Illegal offset: "+offset+" (not an integer)");
|
|
|
offset >>>= 0;
|
|
|
if (offset < 0 || offset + 0 > this.buffer.byteLength)
|
|
|
throw RangeError("Illegal offset: 0 <= "+offset+" (+"+0+") <= "+this.buffer.byteLength);
|
|
|
}
|
|
|
var start = offset;
|
|
|
// UTF8 strings do not contain zero bytes in between except for the zero character, so:
|
|
|
k = utfx.calculateUTF16asUTF8(stringSource(str))[1];
|
|
|
offset += k+1;
|
|
|
var capacity12 = this.buffer.byteLength;
|
|
|
if (offset > capacity12)
|
|
|
this.resize((capacity12 *= 2) > offset ? capacity12 : offset);
|
|
|
offset -= k+1;
|
|
|
utfx.encodeUTF16toUTF8(stringSource(str), function(b) {
|
|
|
this.view.setUint8(offset++, b);
|
|
|
}.bind(this));
|
|
|
this.view.setUint8(offset++, 0);
|
|
|
if (relative) {
|
|
|
this.offset = offset - start;
|
|
|
return this;
|
|
|
}
|
|
|
return k;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Reads a NULL-terminated UTF8 encoded string. For this to work the string read must not contain any NULL characters
|
|
|
* itself.
|
|
|
* @param {number=} offset Offset to read from. Will use and increase {@link ByteBuffer#offset} by the number of bytes
|
|
|
* read if omitted.
|
|
|
* @returns {string|!{string: string, length: number}} The string read if offset is omitted, else the string
|
|
|
* read and the actual number of bytes read.
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBufferPrototype.readCString = function(offset) {
|
|
|
var relative = typeof offset === 'undefined';
|
|
|
if (relative) offset = this.offset;
|
|
|
if (!this.noAssert) {
|
|
|
if (typeof offset !== 'number' || offset % 1 !== 0)
|
|
|
throw TypeError("Illegal offset: "+offset+" (not an integer)");
|
|
|
offset >>>= 0;
|
|
|
if (offset < 0 || offset + 1 > this.buffer.byteLength)
|
|
|
throw RangeError("Illegal offset: 0 <= "+offset+" (+"+1+") <= "+this.buffer.byteLength);
|
|
|
}
|
|
|
var start = offset,
|
|
|
temp;
|
|
|
// UTF8 strings do not contain zero bytes in between except for the zero character itself, so:
|
|
|
var sd, b = -1;
|
|
|
utfx.decodeUTF8toUTF16(function() {
|
|
|
if (b === 0) return null;
|
|
|
if (offset >= this.limit)
|
|
|
throw RangeError("Illegal range: Truncated data, "+offset+" < "+this.limit);
|
|
|
return (b = this.view.getUint8(offset++)) === 0 ? null : b;
|
|
|
}.bind(this), sd = stringDestination(), true);
|
|
|
if (relative) {
|
|
|
this.offset = offset;
|
|
|
return sd();
|
|
|
} else {
|
|
|
return {
|
|
|
"string": sd(),
|
|
|
"length": offset - start
|
|
|
};
|
|
|
}
|
|
|
};
|
|
|
|
|
|
// types/strings/istring
|
|
|
|
|
|
/**
|
|
|
* Writes a length as uint32 prefixed UTF8 encoded string.
|
|
|
* @param {string} str String to write
|
|
|
* @param {number=} offset Offset to write to. Will use and increase {@link ByteBuffer#offset} by the number of bytes
|
|
|
* written if omitted.
|
|
|
* @returns {!ByteBuffer|number} `this` if `offset` is omitted, else the actual number of bytes written
|
|
|
* @expose
|
|
|
* @see ByteBuffer#writeVarint32
|
|
|
*/
|
|
|
ByteBufferPrototype.writeIString = function(str, offset) {
|
|
|
var relative = typeof offset === 'undefined';
|
|
|
if (relative) offset = this.offset;
|
|
|
if (!this.noAssert) {
|
|
|
if (typeof str !== 'string')
|
|
|
throw TypeError("Illegal str: Not a string");
|
|
|
if (typeof offset !== 'number' || offset % 1 !== 0)
|
|
|
throw TypeError("Illegal offset: "+offset+" (not an integer)");
|
|
|
offset >>>= 0;
|
|
|
if (offset < 0 || offset + 0 > this.buffer.byteLength)
|
|
|
throw RangeError("Illegal offset: 0 <= "+offset+" (+"+0+") <= "+this.buffer.byteLength);
|
|
|
}
|
|
|
var start = offset,
|
|
|
k;
|
|
|
k = utfx.calculateUTF16asUTF8(stringSource(str), this.noAssert)[1];
|
|
|
offset += 4+k;
|
|
|
var capacity13 = this.buffer.byteLength;
|
|
|
if (offset > capacity13)
|
|
|
this.resize((capacity13 *= 2) > offset ? capacity13 : offset);
|
|
|
offset -= 4+k;
|
|
|
this.view.setUint32(offset, k, this.littleEndian);
|
|
|
offset += 4;
|
|
|
utfx.encodeUTF16toUTF8(stringSource(str), function(b) {
|
|
|
this.view.setUint8(offset++, b);
|
|
|
}.bind(this));
|
|
|
if (offset !== start + 4 + k)
|
|
|
throw RangeError("Illegal range: Truncated data, "+offset+" == "+(offset+4+k));
|
|
|
if (relative) {
|
|
|
this.offset = offset;
|
|
|
return this;
|
|
|
}
|
|
|
return offset - start;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Reads a length as uint32 prefixed UTF8 encoded string.
|
|
|
* @param {number=} offset Offset to read from. Will use and increase {@link ByteBuffer#offset} by the number of bytes
|
|
|
* read if omitted.
|
|
|
* @returns {string|!{string: string, length: number}} The string read if offset is omitted, else the string
|
|
|
* read and the actual number of bytes read.
|
|
|
* @expose
|
|
|
* @see ByteBuffer#readVarint32
|
|
|
*/
|
|
|
ByteBufferPrototype.readIString = function(offset) {
|
|
|
var relative = typeof offset === 'undefined';
|
|
|
if (relative) offset = this.offset;
|
|
|
if (!this.noAssert) {
|
|
|
if (typeof offset !== 'number' || offset % 1 !== 0)
|
|
|
throw TypeError("Illegal offset: "+offset+" (not an integer)");
|
|
|
offset >>>= 0;
|
|
|
if (offset < 0 || offset + 4 > this.buffer.byteLength)
|
|
|
throw RangeError("Illegal offset: 0 <= "+offset+" (+"+4+") <= "+this.buffer.byteLength);
|
|
|
}
|
|
|
var temp = 0,
|
|
|
start = offset,
|
|
|
str;
|
|
|
temp = this.view.getUint32(offset, this.littleEndian);
|
|
|
offset += 4;
|
|
|
var k = offset + temp,
|
|
|
sd;
|
|
|
utfx.decodeUTF8toUTF16(function() {
|
|
|
return offset < k ? this.view.getUint8(offset++) : null;
|
|
|
}.bind(this), sd = stringDestination(), this.noAssert);
|
|
|
str = sd();
|
|
|
if (relative) {
|
|
|
this.offset = offset;
|
|
|
return str;
|
|
|
} else {
|
|
|
return {
|
|
|
'string': str,
|
|
|
'length': offset - start
|
|
|
};
|
|
|
}
|
|
|
};
|
|
|
|
|
|
// types/strings/utf8string
|
|
|
|
|
|
/**
|
|
|
* Metrics representing number of UTF8 characters. Evaluates to `c`.
|
|
|
* @type {string}
|
|
|
* @const
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBuffer.METRICS_CHARS = 'c';
|
|
|
|
|
|
/**
|
|
|
* Metrics representing number of bytes. Evaluates to `b`.
|
|
|
* @type {string}
|
|
|
* @const
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBuffer.METRICS_BYTES = 'b';
|
|
|
|
|
|
/**
|
|
|
* Writes an UTF8 encoded string.
|
|
|
* @param {string} str String to write
|
|
|
* @param {number=} offset Offset to write to. Will use and increase {@link ByteBuffer#offset} if omitted.
|
|
|
* @returns {!ByteBuffer|number} this if offset is omitted, else the actual number of bytes written.
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBufferPrototype.writeUTF8String = function(str, offset) {
|
|
|
var relative = typeof offset === 'undefined';
|
|
|
if (relative) offset = this.offset;
|
|
|
if (!this.noAssert) {
|
|
|
if (typeof offset !== 'number' || offset % 1 !== 0)
|
|
|
throw TypeError("Illegal offset: "+offset+" (not an integer)");
|
|
|
offset >>>= 0;
|
|
|
if (offset < 0 || offset + 0 > this.buffer.byteLength)
|
|
|
throw RangeError("Illegal offset: 0 <= "+offset+" (+"+0+") <= "+this.buffer.byteLength);
|
|
|
}
|
|
|
var k;
|
|
|
var start = offset;
|
|
|
k = utfx.calculateUTF16asUTF8(stringSource(str))[1];
|
|
|
offset += k;
|
|
|
var capacity14 = this.buffer.byteLength;
|
|
|
if (offset > capacity14)
|
|
|
this.resize((capacity14 *= 2) > offset ? capacity14 : offset);
|
|
|
offset -= k;
|
|
|
utfx.encodeUTF16toUTF8(stringSource(str), function(b) {
|
|
|
this.view.setUint8(offset++, b);
|
|
|
}.bind(this));
|
|
|
if (relative) {
|
|
|
this.offset = offset;
|
|
|
return this;
|
|
|
}
|
|
|
return offset - start;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Writes an UTF8 encoded string. This is an alias of {@link ByteBuffer#writeUTF8String}.
|
|
|
* @function
|
|
|
* @param {string} str String to write
|
|
|
* @param {number=} offset Offset to write to. Will use and increase {@link ByteBuffer#offset} if omitted.
|
|
|
* @returns {!ByteBuffer|number} this if offset is omitted, else the actual number of bytes written.
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBufferPrototype.writeString = ByteBufferPrototype.writeUTF8String;
|
|
|
|
|
|
/**
|
|
|
* Calculates the number of UTF8 characters of a string. JavaScript itself uses UTF-16, so that a string's
|
|
|
* `length` property does not reflect its actual UTF8 size if it contains code points larger than 0xFFFF.
|
|
|
* @function
|
|
|
* @param {string} str String to calculate
|
|
|
* @returns {number} Number of UTF8 characters
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBuffer.calculateUTF8Chars = function(str) {
|
|
|
return utfx.calculateUTF16asUTF8(stringSource(str))[0];
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Calculates the number of UTF8 bytes of a string.
|
|
|
* @function
|
|
|
* @param {string} str String to calculate
|
|
|
* @returns {number} Number of UTF8 bytes
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBuffer.calculateUTF8Bytes = function(str) {
|
|
|
return utfx.calculateUTF16asUTF8(stringSource(str))[1];
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Reads an UTF8 encoded string.
|
|
|
* @param {number} length Number of characters or bytes to read.
|
|
|
* @param {string=} metrics Metrics specifying what `length` is meant to count. Defaults to
|
|
|
* {@link ByteBuffer.METRICS_CHARS}.
|
|
|
* @param {number=} offset Offset to read from. Will use and increase {@link ByteBuffer#offset} by the number of bytes
|
|
|
* read if omitted.
|
|
|
* @returns {string|!{string: string, length: number}} The string read if offset is omitted, else the string
|
|
|
* read and the actual number of bytes read.
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBufferPrototype.readUTF8String = function(length, metrics, offset) {
|
|
|
if (typeof metrics === 'number') {
|
|
|
offset = metrics;
|
|
|
metrics = undefined;
|
|
|
}
|
|
|
var relative = typeof offset === 'undefined';
|
|
|
if (relative) offset = this.offset;
|
|
|
if (typeof metrics === 'undefined') metrics = ByteBuffer.METRICS_CHARS;
|
|
|
if (!this.noAssert) {
|
|
|
if (typeof length !== 'number' || length % 1 !== 0)
|
|
|
throw TypeError("Illegal length: "+length+" (not an integer)");
|
|
|
length |= 0;
|
|
|
if (typeof offset !== 'number' || offset % 1 !== 0)
|
|
|
throw TypeError("Illegal offset: "+offset+" (not an integer)");
|
|
|
offset >>>= 0;
|
|
|
if (offset < 0 || offset + 0 > this.buffer.byteLength)
|
|
|
throw RangeError("Illegal offset: 0 <= "+offset+" (+"+0+") <= "+this.buffer.byteLength);
|
|
|
}
|
|
|
var i = 0,
|
|
|
start = offset,
|
|
|
sd;
|
|
|
if (metrics === ByteBuffer.METRICS_CHARS) { // The same for node and the browser
|
|
|
sd = stringDestination();
|
|
|
utfx.decodeUTF8(function() {
|
|
|
return i < length && offset < this.limit ? this.view.getUint8(offset++) : null;
|
|
|
}.bind(this), function(cp) {
|
|
|
++i; utfx.UTF8toUTF16(cp, sd);
|
|
|
}.bind(this));
|
|
|
if (i !== length)
|
|
|
throw RangeError("Illegal range: Truncated data, "+i+" == "+length);
|
|
|
if (relative) {
|
|
|
this.offset = offset;
|
|
|
return sd();
|
|
|
} else {
|
|
|
return {
|
|
|
"string": sd(),
|
|
|
"length": offset - start
|
|
|
};
|
|
|
}
|
|
|
} else if (metrics === ByteBuffer.METRICS_BYTES) {
|
|
|
if (!this.noAssert) {
|
|
|
if (typeof offset !== 'number' || offset % 1 !== 0)
|
|
|
throw TypeError("Illegal offset: "+offset+" (not an integer)");
|
|
|
offset >>>= 0;
|
|
|
if (offset < 0 || offset + length > this.buffer.byteLength)
|
|
|
throw RangeError("Illegal offset: 0 <= "+offset+" (+"+length+") <= "+this.buffer.byteLength);
|
|
|
}
|
|
|
var k = offset + length;
|
|
|
utfx.decodeUTF8toUTF16(function() {
|
|
|
return offset < k ? this.view.getUint8(offset++) : null;
|
|
|
}.bind(this), sd = stringDestination(), this.noAssert);
|
|
|
if (offset !== k)
|
|
|
throw RangeError("Illegal range: Truncated data, "+offset+" == "+k);
|
|
|
if (relative) {
|
|
|
this.offset = offset;
|
|
|
return sd();
|
|
|
} else {
|
|
|
return {
|
|
|
'string': sd(),
|
|
|
'length': offset - start
|
|
|
};
|
|
|
}
|
|
|
} else
|
|
|
throw TypeError("Unsupported metrics: "+metrics);
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Reads an UTF8 encoded string. This is an alias of {@link ByteBuffer#readUTF8String}.
|
|
|
* @function
|
|
|
* @param {number} length Number of characters or bytes to read
|
|
|
* @param {number=} metrics Metrics specifying what `n` is meant to count. Defaults to
|
|
|
* {@link ByteBuffer.METRICS_CHARS}.
|
|
|
* @param {number=} offset Offset to read from. Will use and increase {@link ByteBuffer#offset} by the number of bytes
|
|
|
* read if omitted.
|
|
|
* @returns {string|!{string: string, length: number}} The string read if offset is omitted, else the string
|
|
|
* read and the actual number of bytes read.
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBufferPrototype.readString = ByteBufferPrototype.readUTF8String;
|
|
|
|
|
|
// types/strings/vstring
|
|
|
|
|
|
/**
|
|
|
* Writes a length as varint32 prefixed UTF8 encoded string.
|
|
|
* @param {string} str String to write
|
|
|
* @param {number=} offset Offset to write to. Will use and increase {@link ByteBuffer#offset} by the number of bytes
|
|
|
* written if omitted.
|
|
|
* @returns {!ByteBuffer|number} `this` if `offset` is omitted, else the actual number of bytes written
|
|
|
* @expose
|
|
|
* @see ByteBuffer#writeVarint32
|
|
|
*/
|
|
|
ByteBufferPrototype.writeVString = function(str, offset) {
|
|
|
var relative = typeof offset === 'undefined';
|
|
|
if (relative) offset = this.offset;
|
|
|
if (!this.noAssert) {
|
|
|
if (typeof str !== 'string')
|
|
|
throw TypeError("Illegal str: Not a string");
|
|
|
if (typeof offset !== 'number' || offset % 1 !== 0)
|
|
|
throw TypeError("Illegal offset: "+offset+" (not an integer)");
|
|
|
offset >>>= 0;
|
|
|
if (offset < 0 || offset + 0 > this.buffer.byteLength)
|
|
|
throw RangeError("Illegal offset: 0 <= "+offset+" (+"+0+") <= "+this.buffer.byteLength);
|
|
|
}
|
|
|
var start = offset,
|
|
|
k, l;
|
|
|
k = utfx.calculateUTF16asUTF8(stringSource(str), this.noAssert)[1];
|
|
|
l = ByteBuffer.calculateVarint32(k);
|
|
|
offset += l+k;
|
|
|
var capacity15 = this.buffer.byteLength;
|
|
|
if (offset > capacity15)
|
|
|
this.resize((capacity15 *= 2) > offset ? capacity15 : offset);
|
|
|
offset -= l+k;
|
|
|
offset += this.writeVarint32(k, offset);
|
|
|
utfx.encodeUTF16toUTF8(stringSource(str), function(b) {
|
|
|
this.view.setUint8(offset++, b);
|
|
|
}.bind(this));
|
|
|
if (offset !== start+k+l)
|
|
|
throw RangeError("Illegal range: Truncated data, "+offset+" == "+(offset+k+l));
|
|
|
if (relative) {
|
|
|
this.offset = offset;
|
|
|
return this;
|
|
|
}
|
|
|
return offset - start;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Reads a length as varint32 prefixed UTF8 encoded string.
|
|
|
* @param {number=} offset Offset to read from. Will use and increase {@link ByteBuffer#offset} by the number of bytes
|
|
|
* read if omitted.
|
|
|
* @returns {string|!{string: string, length: number}} The string read if offset is omitted, else the string
|
|
|
* read and the actual number of bytes read.
|
|
|
* @expose
|
|
|
* @see ByteBuffer#readVarint32
|
|
|
*/
|
|
|
ByteBufferPrototype.readVString = function(offset) {
|
|
|
var relative = typeof offset === 'undefined';
|
|
|
if (relative) offset = this.offset;
|
|
|
if (!this.noAssert) {
|
|
|
if (typeof offset !== 'number' || offset % 1 !== 0)
|
|
|
throw TypeError("Illegal offset: "+offset+" (not an integer)");
|
|
|
offset >>>= 0;
|
|
|
if (offset < 0 || offset + 1 > this.buffer.byteLength)
|
|
|
throw RangeError("Illegal offset: 0 <= "+offset+" (+"+1+") <= "+this.buffer.byteLength);
|
|
|
}
|
|
|
var temp = this.readVarint32(offset),
|
|
|
start = offset,
|
|
|
str;
|
|
|
offset += temp['length'];
|
|
|
temp = temp['value'];
|
|
|
var k = offset + temp,
|
|
|
sd = stringDestination();
|
|
|
utfx.decodeUTF8toUTF16(function() {
|
|
|
return offset < k ? this.view.getUint8(offset++) : null;
|
|
|
}.bind(this), sd, this.noAssert);
|
|
|
str = sd();
|
|
|
if (relative) {
|
|
|
this.offset = offset;
|
|
|
return str;
|
|
|
} else {
|
|
|
return {
|
|
|
'string': str,
|
|
|
'length': offset - start
|
|
|
};
|
|
|
}
|
|
|
};
|
|
|
|
|
|
|
|
|
/**
|
|
|
* Appends some data to this ByteBuffer. This will overwrite any contents behind the specified offset up to the appended
|
|
|
* data's length.
|
|
|
* @param {!ByteBuffer|!ArrayBuffer|!Uint8Array|string} source Data to append. If `source` is a ByteBuffer, its offsets
|
|
|
* will be modified according to the performed read operation.
|
|
|
* @param {(string|number)=} encoding Encoding if `data` is a string ("base64", "hex", "binary", defaults to "utf8")
|
|
|
* @param {number=} offset Offset to append at. Will use and increase {@link ByteBuffer#offset} by the number of bytes
|
|
|
* read if omitted.
|
|
|
* @returns {!ByteBuffer} this
|
|
|
* @expose
|
|
|
* @example A relative `<01 02>03.append(<04 05>)` will result in `<01 02 04 05>, 04 05|`
|
|
|
* @example An absolute `<01 02>03.append(04 05>, 1)` will result in `<01 04>05, 04 05|`
|
|
|
*/
|
|
|
ByteBufferPrototype.append = function(source, encoding, offset) {
|
|
|
if (typeof encoding === 'number' || typeof encoding !== 'string') {
|
|
|
offset = encoding;
|
|
|
encoding = undefined;
|
|
|
}
|
|
|
var relative = typeof offset === 'undefined';
|
|
|
if (relative) offset = this.offset;
|
|
|
if (!this.noAssert) {
|
|
|
if (typeof offset !== 'number' || offset % 1 !== 0)
|
|
|
throw TypeError("Illegal offset: "+offset+" (not an integer)");
|
|
|
offset >>>= 0;
|
|
|
if (offset < 0 || offset + 0 > this.buffer.byteLength)
|
|
|
throw RangeError("Illegal offset: 0 <= "+offset+" (+"+0+") <= "+this.buffer.byteLength);
|
|
|
}
|
|
|
if (!(source instanceof ByteBuffer))
|
|
|
source = ByteBuffer.wrap(source, encoding);
|
|
|
var length = source.limit - source.offset;
|
|
|
if (length <= 0) return this; // Nothing to append
|
|
|
offset += length;
|
|
|
var capacity16 = this.buffer.byteLength;
|
|
|
if (offset > capacity16)
|
|
|
this.resize((capacity16 *= 2) > offset ? capacity16 : offset);
|
|
|
offset -= length;
|
|
|
new Uint8Array(this.buffer, offset).set(new Uint8Array(source.buffer).subarray(source.offset, source.limit));
|
|
|
source.offset += length;
|
|
|
if (relative) this.offset += length;
|
|
|
return this;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Appends this ByteBuffer's contents to another ByteBuffer. This will overwrite any contents behind the specified
|
|
|
* offset up to the length of this ByteBuffer's data.
|
|
|
* @param {!ByteBuffer} target Target ByteBuffer
|
|
|
* @param {number=} offset Offset to append to. Will use and increase {@link ByteBuffer#offset} by the number of bytes
|
|
|
* read if omitted.
|
|
|
* @returns {!ByteBuffer} this
|
|
|
* @expose
|
|
|
* @see ByteBuffer#append
|
|
|
*/
|
|
|
ByteBufferPrototype.appendTo = function(target, offset) {
|
|
|
target.append(this, offset);
|
|
|
return this;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Enables or disables assertions of argument types and offsets. Assertions are enabled by default but you can opt to
|
|
|
* disable them if your code already makes sure that everything is valid.
|
|
|
* @param {boolean} assert `true` to enable assertions, otherwise `false`
|
|
|
* @returns {!ByteBuffer} this
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBufferPrototype.assert = function(assert) {
|
|
|
this.noAssert = !assert;
|
|
|
return this;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Gets the capacity of this ByteBuffer's backing buffer.
|
|
|
* @returns {number} Capacity of the backing buffer
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBufferPrototype.capacity = function() {
|
|
|
return this.buffer.byteLength;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Clears this ByteBuffer's offsets by setting {@link ByteBuffer#offset} to `0` and {@link ByteBuffer#limit} to the
|
|
|
* backing buffer's capacity. Discards {@link ByteBuffer#markedOffset}.
|
|
|
* @returns {!ByteBuffer} this
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBufferPrototype.clear = function() {
|
|
|
this.offset = 0;
|
|
|
this.limit = this.buffer.byteLength;
|
|
|
this.markedOffset = -1;
|
|
|
return this;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Creates a cloned instance of this ByteBuffer, preset with this ByteBuffer's values for {@link ByteBuffer#offset},
|
|
|
* {@link ByteBuffer#markedOffset} and {@link ByteBuffer#limit}.
|
|
|
* @param {boolean=} copy Whether to copy the backing buffer or to return another view on the same, defaults to `false`
|
|
|
* @returns {!ByteBuffer} Cloned instance
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBufferPrototype.clone = function(copy) {
|
|
|
var bb = new ByteBuffer(0, this.littleEndian, this.noAssert);
|
|
|
if (copy) {
|
|
|
var buffer = new ArrayBuffer(this.buffer.byteLength);
|
|
|
new Uint8Array(buffer).set(this.buffer);
|
|
|
bb.buffer = buffer;
|
|
|
bb.view = new DataView(buffer);
|
|
|
} else {
|
|
|
bb.buffer = this.buffer;
|
|
|
bb.view = this.view;
|
|
|
}
|
|
|
bb.offset = this.offset;
|
|
|
bb.markedOffset = this.markedOffset;
|
|
|
bb.limit = this.limit;
|
|
|
return bb;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Compacts this ByteBuffer to be backed by a {@link ByteBuffer#buffer} of its contents' length. Contents are the bytes
|
|
|
* between {@link ByteBuffer#offset} and {@link ByteBuffer#limit}. Will set `offset = 0` and `limit = capacity` and
|
|
|
* adapt {@link ByteBuffer#markedOffset} to the same relative position if set.
|
|
|
* @param {number=} begin Offset to start at, defaults to {@link ByteBuffer#offset}
|
|
|
* @param {number=} end Offset to end at, defaults to {@link ByteBuffer#limit}
|
|
|
* @returns {!ByteBuffer} this
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBufferPrototype.compact = function(begin, end) {
|
|
|
if (typeof begin === 'undefined') begin = this.offset;
|
|
|
if (typeof end === 'undefined') end = this.limit;
|
|
|
if (!this.noAssert) {
|
|
|
if (typeof begin !== 'number' || begin % 1 !== 0)
|
|
|
throw TypeError("Illegal begin: Not an integer");
|
|
|
begin >>>= 0;
|
|
|
if (typeof end !== 'number' || end % 1 !== 0)
|
|
|
throw TypeError("Illegal end: Not an integer");
|
|
|
end >>>= 0;
|
|
|
if (begin < 0 || begin > end || end > this.buffer.byteLength)
|
|
|
throw RangeError("Illegal range: 0 <= "+begin+" <= "+end+" <= "+this.buffer.byteLength);
|
|
|
}
|
|
|
if (begin === 0 && end === this.buffer.byteLength)
|
|
|
return this; // Already compacted
|
|
|
var len = end - begin;
|
|
|
if (len === 0) {
|
|
|
this.buffer = EMPTY_BUFFER;
|
|
|
this.view = null;
|
|
|
if (this.markedOffset >= 0) this.markedOffset -= begin;
|
|
|
this.offset = 0;
|
|
|
this.limit = 0;
|
|
|
return this;
|
|
|
}
|
|
|
var buffer = new ArrayBuffer(len);
|
|
|
new Uint8Array(buffer).set(new Uint8Array(this.buffer).subarray(begin, end));
|
|
|
this.buffer = buffer;
|
|
|
this.view = new DataView(buffer);
|
|
|
if (this.markedOffset >= 0) this.markedOffset -= begin;
|
|
|
this.offset = 0;
|
|
|
this.limit = len;
|
|
|
return this;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Creates a copy of this ByteBuffer's contents. Contents are the bytes between {@link ByteBuffer#offset} and
|
|
|
* {@link ByteBuffer#limit}.
|
|
|
* @param {number=} begin Begin offset, defaults to {@link ByteBuffer#offset}.
|
|
|
* @param {number=} end End offset, defaults to {@link ByteBuffer#limit}.
|
|
|
* @returns {!ByteBuffer} Copy
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBufferPrototype.copy = function(begin, end) {
|
|
|
if (typeof begin === 'undefined') begin = this.offset;
|
|
|
if (typeof end === 'undefined') end = this.limit;
|
|
|
if (!this.noAssert) {
|
|
|
if (typeof begin !== 'number' || begin % 1 !== 0)
|
|
|
throw TypeError("Illegal begin: Not an integer");
|
|
|
begin >>>= 0;
|
|
|
if (typeof end !== 'number' || end % 1 !== 0)
|
|
|
throw TypeError("Illegal end: Not an integer");
|
|
|
end >>>= 0;
|
|
|
if (begin < 0 || begin > end || end > this.buffer.byteLength)
|
|
|
throw RangeError("Illegal range: 0 <= "+begin+" <= "+end+" <= "+this.buffer.byteLength);
|
|
|
}
|
|
|
if (begin === end)
|
|
|
return new ByteBuffer(0, this.littleEndian, this.noAssert);
|
|
|
var capacity = end - begin,
|
|
|
bb = new ByteBuffer(capacity, this.littleEndian, this.noAssert);
|
|
|
bb.offset = 0;
|
|
|
bb.limit = capacity;
|
|
|
if (bb.markedOffset >= 0) bb.markedOffset -= begin;
|
|
|
this.copyTo(bb, 0, begin, end);
|
|
|
return bb;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Copies this ByteBuffer's contents to another ByteBuffer. Contents are the bytes between {@link ByteBuffer#offset} and
|
|
|
* {@link ByteBuffer#limit}.
|
|
|
* @param {!ByteBuffer} target Target ByteBuffer
|
|
|
* @param {number=} targetOffset Offset to copy to. Will use and increase the target's {@link ByteBuffer#offset}
|
|
|
* by the number of bytes copied if omitted.
|
|
|
* @param {number=} sourceOffset Offset to start copying from. Will use and increase {@link ByteBuffer#offset} by the
|
|
|
* number of bytes copied if omitted.
|
|
|
* @param {number=} sourceLimit Offset to end copying from, defaults to {@link ByteBuffer#limit}
|
|
|
* @returns {!ByteBuffer} this
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBufferPrototype.copyTo = function(target, targetOffset, sourceOffset, sourceLimit) {
|
|
|
var relative,
|
|
|
targetRelative;
|
|
|
if (!this.noAssert) {
|
|
|
if (!ByteBuffer.isByteBuffer(target))
|
|
|
throw TypeError("Illegal target: Not a ByteBuffer");
|
|
|
}
|
|
|
targetOffset = (targetRelative = typeof targetOffset === 'undefined') ? target.offset : targetOffset | 0;
|
|
|
sourceOffset = (relative = typeof sourceOffset === 'undefined') ? this.offset : sourceOffset | 0;
|
|
|
sourceLimit = typeof sourceLimit === 'undefined' ? this.limit : sourceLimit | 0;
|
|
|
|
|
|
if (targetOffset < 0 || targetOffset > target.buffer.byteLength)
|
|
|
throw RangeError("Illegal target range: 0 <= "+targetOffset+" <= "+target.buffer.byteLength);
|
|
|
if (sourceOffset < 0 || sourceLimit > this.buffer.byteLength)
|
|
|
throw RangeError("Illegal source range: 0 <= "+sourceOffset+" <= "+this.buffer.byteLength);
|
|
|
|
|
|
var len = sourceLimit - sourceOffset;
|
|
|
if (len === 0)
|
|
|
return target; // Nothing to copy
|
|
|
|
|
|
target.ensureCapacity(targetOffset + len);
|
|
|
|
|
|
new Uint8Array(target.buffer).set(new Uint8Array(this.buffer).subarray(sourceOffset, sourceLimit), targetOffset);
|
|
|
|
|
|
if (relative) this.offset += len;
|
|
|
if (targetRelative) target.offset += len;
|
|
|
|
|
|
return this;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Makes sure that this ByteBuffer is backed by a {@link ByteBuffer#buffer} of at least the specified capacity. If the
|
|
|
* current capacity is exceeded, it will be doubled. If double the current capacity is less than the required capacity,
|
|
|
* the required capacity will be used instead.
|
|
|
* @param {number} capacity Required capacity
|
|
|
* @returns {!ByteBuffer} this
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBufferPrototype.ensureCapacity = function(capacity) {
|
|
|
var current = this.buffer.byteLength;
|
|
|
if (current < capacity)
|
|
|
return this.resize((current *= 2) > capacity ? current : capacity);
|
|
|
return this;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Overwrites this ByteBuffer's contents with the specified value. Contents are the bytes between
|
|
|
* {@link ByteBuffer#offset} and {@link ByteBuffer#limit}.
|
|
|
* @param {number|string} value Byte value to fill with. If given as a string, the first character is used.
|
|
|
* @param {number=} begin Begin offset. Will use and increase {@link ByteBuffer#offset} by the number of bytes
|
|
|
* written if omitted. defaults to {@link ByteBuffer#offset}.
|
|
|
* @param {number=} end End offset, defaults to {@link ByteBuffer#limit}.
|
|
|
* @returns {!ByteBuffer} this
|
|
|
* @expose
|
|
|
* @example `someByteBuffer.clear().fill(0)` fills the entire backing buffer with zeroes
|
|
|
*/
|
|
|
ByteBufferPrototype.fill = function(value, begin, end) {
|
|
|
var relative = typeof begin === 'undefined';
|
|
|
if (relative) begin = this.offset;
|
|
|
if (typeof value === 'string' && value.length > 0)
|
|
|
value = value.charCodeAt(0);
|
|
|
if (typeof begin === 'undefined') begin = this.offset;
|
|
|
if (typeof end === 'undefined') end = this.limit;
|
|
|
if (!this.noAssert) {
|
|
|
if (typeof value !== 'number' || value % 1 !== 0)
|
|
|
throw TypeError("Illegal value: "+value+" (not an integer)");
|
|
|
value |= 0;
|
|
|
if (typeof begin !== 'number' || begin % 1 !== 0)
|
|
|
throw TypeError("Illegal begin: Not an integer");
|
|
|
begin >>>= 0;
|
|
|
if (typeof end !== 'number' || end % 1 !== 0)
|
|
|
throw TypeError("Illegal end: Not an integer");
|
|
|
end >>>= 0;
|
|
|
if (begin < 0 || begin > end || end > this.buffer.byteLength)
|
|
|
throw RangeError("Illegal range: 0 <= "+begin+" <= "+end+" <= "+this.buffer.byteLength);
|
|
|
}
|
|
|
if (begin >= end)
|
|
|
return this; // Nothing to fill
|
|
|
while (begin < end) this.view.setUint8(begin++, value);
|
|
|
if (relative) this.offset = begin;
|
|
|
return this;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Makes this ByteBuffer ready for a new sequence of write or relative read operations. Sets `limit = offset` and
|
|
|
* `offset = 0`. Make sure always to flip a ByteBuffer when all relative read or write operations are complete.
|
|
|
* @returns {!ByteBuffer} this
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBufferPrototype.flip = function() {
|
|
|
this.limit = this.offset;
|
|
|
this.offset = 0;
|
|
|
return this;
|
|
|
};
|
|
|
/**
|
|
|
* Marks an offset on this ByteBuffer to be used later.
|
|
|
* @param {number=} offset Offset to mark. Defaults to {@link ByteBuffer#offset}.
|
|
|
* @returns {!ByteBuffer} this
|
|
|
* @throws {TypeError} If `offset` is not a valid number
|
|
|
* @throws {RangeError} If `offset` is out of bounds
|
|
|
* @see ByteBuffer#reset
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBufferPrototype.mark = function(offset) {
|
|
|
offset = typeof offset === 'undefined' ? this.offset : offset;
|
|
|
if (!this.noAssert) {
|
|
|
if (typeof offset !== 'number' || offset % 1 !== 0)
|
|
|
throw TypeError("Illegal offset: "+offset+" (not an integer)");
|
|
|
offset >>>= 0;
|
|
|
if (offset < 0 || offset + 0 > this.buffer.byteLength)
|
|
|
throw RangeError("Illegal offset: 0 <= "+offset+" (+"+0+") <= "+this.buffer.byteLength);
|
|
|
}
|
|
|
this.markedOffset = offset;
|
|
|
return this;
|
|
|
};
|
|
|
/**
|
|
|
* Sets the byte order.
|
|
|
* @param {boolean} littleEndian `true` for little endian byte order, `false` for big endian
|
|
|
* @returns {!ByteBuffer} this
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBufferPrototype.order = function(littleEndian) {
|
|
|
if (!this.noAssert) {
|
|
|
if (typeof littleEndian !== 'boolean')
|
|
|
throw TypeError("Illegal littleEndian: Not a boolean");
|
|
|
}
|
|
|
this.littleEndian = !!littleEndian;
|
|
|
return this;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Switches (to) little endian byte order.
|
|
|
* @param {boolean=} littleEndian Defaults to `true`, otherwise uses big endian
|
|
|
* @returns {!ByteBuffer} this
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBufferPrototype.LE = function(littleEndian) {
|
|
|
this.littleEndian = typeof littleEndian !== 'undefined' ? !!littleEndian : true;
|
|
|
return this;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Switches (to) big endian byte order.
|
|
|
* @param {boolean=} bigEndian Defaults to `true`, otherwise uses little endian
|
|
|
* @returns {!ByteBuffer} this
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBufferPrototype.BE = function(bigEndian) {
|
|
|
this.littleEndian = typeof bigEndian !== 'undefined' ? !bigEndian : false;
|
|
|
return this;
|
|
|
};
|
|
|
/**
|
|
|
* Prepends some data to this ByteBuffer. This will overwrite any contents before the specified offset up to the
|
|
|
* prepended data's length. If there is not enough space available before the specified `offset`, the backing buffer
|
|
|
* will be resized and its contents moved accordingly.
|
|
|
* @param {!ByteBuffer|string|!ArrayBuffer} source Data to prepend. If `source` is a ByteBuffer, its offset will be
|
|
|
* modified according to the performed read operation.
|
|
|
* @param {(string|number)=} encoding Encoding if `data` is a string ("base64", "hex", "binary", defaults to "utf8")
|
|
|
* @param {number=} offset Offset to prepend at. Will use and decrease {@link ByteBuffer#offset} by the number of bytes
|
|
|
* prepended if omitted.
|
|
|
* @returns {!ByteBuffer} this
|
|
|
* @expose
|
|
|
* @example A relative `00<01 02 03>.prepend(<04 05>)` results in `<04 05 01 02 03>, 04 05|`
|
|
|
* @example An absolute `00<01 02 03>.prepend(<04 05>, 2)` results in `04<05 02 03>, 04 05|`
|
|
|
*/
|
|
|
ByteBufferPrototype.prepend = function(source, encoding, offset) {
|
|
|
if (typeof encoding === 'number' || typeof encoding !== 'string') {
|
|
|
offset = encoding;
|
|
|
encoding = undefined;
|
|
|
}
|
|
|
var relative = typeof offset === 'undefined';
|
|
|
if (relative) offset = this.offset;
|
|
|
if (!this.noAssert) {
|
|
|
if (typeof offset !== 'number' || offset % 1 !== 0)
|
|
|
throw TypeError("Illegal offset: "+offset+" (not an integer)");
|
|
|
offset >>>= 0;
|
|
|
if (offset < 0 || offset + 0 > this.buffer.byteLength)
|
|
|
throw RangeError("Illegal offset: 0 <= "+offset+" (+"+0+") <= "+this.buffer.byteLength);
|
|
|
}
|
|
|
if (!(source instanceof ByteBuffer))
|
|
|
source = ByteBuffer.wrap(source, encoding);
|
|
|
var len = source.limit - source.offset;
|
|
|
if (len <= 0) return this; // Nothing to prepend
|
|
|
var diff = len - offset;
|
|
|
var arrayView;
|
|
|
if (diff > 0) { // Not enough space before offset, so resize + move
|
|
|
var buffer = new ArrayBuffer(this.buffer.byteLength + diff);
|
|
|
arrayView = new Uint8Array(buffer);
|
|
|
arrayView.set(new Uint8Array(this.buffer).subarray(offset, this.buffer.byteLength), len);
|
|
|
this.buffer = buffer;
|
|
|
this.view = new DataView(buffer);
|
|
|
this.offset += diff;
|
|
|
if (this.markedOffset >= 0) this.markedOffset += diff;
|
|
|
this.limit += diff;
|
|
|
offset += diff;
|
|
|
} else {
|
|
|
arrayView = new Uint8Array(this.buffer);
|
|
|
}
|
|
|
arrayView.set(new Uint8Array(source.buffer).subarray(source.offset, source.limit), offset - len);
|
|
|
source.offset = source.limit;
|
|
|
if (relative)
|
|
|
this.offset -= len;
|
|
|
return this;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Prepends this ByteBuffer to another ByteBuffer. This will overwrite any contents before the specified offset up to the
|
|
|
* prepended data's length. If there is not enough space available before the specified `offset`, the backing buffer
|
|
|
* will be resized and its contents moved accordingly.
|
|
|
* @param {!ByteBuffer} target Target ByteBuffer
|
|
|
* @param {number=} offset Offset to prepend at. Will use and decrease {@link ByteBuffer#offset} by the number of bytes
|
|
|
* prepended if omitted.
|
|
|
* @returns {!ByteBuffer} this
|
|
|
* @expose
|
|
|
* @see ByteBuffer#prepend
|
|
|
*/
|
|
|
ByteBufferPrototype.prependTo = function(target, offset) {
|
|
|
target.prepend(this, offset);
|
|
|
return this;
|
|
|
};
|
|
|
/**
|
|
|
* Prints debug information about this ByteBuffer's contents.
|
|
|
* @param {function(string)=} out Output function to call, defaults to console.log
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBufferPrototype.printDebug = function(out) {
|
|
|
if (typeof out !== 'function') out = console.log.bind(console);
|
|
|
out(
|
|
|
this.toString()+"\n"+
|
|
|
"-------------------------------------------------------------------\n"+
|
|
|
this.toDebug(/* columns */ true)
|
|
|
);
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Gets the number of remaining readable bytes. Contents are the bytes between {@link ByteBuffer#offset} and
|
|
|
* {@link ByteBuffer#limit}, so this returns `limit - offset`.
|
|
|
* @returns {number} Remaining readable bytes. May be negative if `offset > limit`.
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBufferPrototype.remaining = function() {
|
|
|
return this.limit - this.offset;
|
|
|
};
|
|
|
/**
|
|
|
* Resets this ByteBuffer's {@link ByteBuffer#offset}. If an offset has been marked through {@link ByteBuffer#mark}
|
|
|
* before, `offset` will be set to {@link ByteBuffer#markedOffset}, which will then be discarded. If no offset has been
|
|
|
* marked, sets `offset = 0`.
|
|
|
* @returns {!ByteBuffer} this
|
|
|
* @see ByteBuffer#mark
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBufferPrototype.reset = function() {
|
|
|
if (this.markedOffset >= 0) {
|
|
|
this.offset = this.markedOffset;
|
|
|
this.markedOffset = -1;
|
|
|
} else {
|
|
|
this.offset = 0;
|
|
|
}
|
|
|
return this;
|
|
|
};
|
|
|
/**
|
|
|
* Resizes this ByteBuffer to be backed by a buffer of at least the given capacity. Will do nothing if already that
|
|
|
* large or larger.
|
|
|
* @param {number} capacity Capacity required
|
|
|
* @returns {!ByteBuffer} this
|
|
|
* @throws {TypeError} If `capacity` is not a number
|
|
|
* @throws {RangeError} If `capacity < 0`
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBufferPrototype.resize = function(capacity) {
|
|
|
if (!this.noAssert) {
|
|
|
if (typeof capacity !== 'number' || capacity % 1 !== 0)
|
|
|
throw TypeError("Illegal capacity: "+capacity+" (not an integer)");
|
|
|
capacity |= 0;
|
|
|
if (capacity < 0)
|
|
|
throw RangeError("Illegal capacity: 0 <= "+capacity);
|
|
|
}
|
|
|
if (this.buffer.byteLength < capacity) {
|
|
|
var buffer = new ArrayBuffer(capacity);
|
|
|
new Uint8Array(buffer).set(new Uint8Array(this.buffer));
|
|
|
this.buffer = buffer;
|
|
|
this.view = new DataView(buffer);
|
|
|
}
|
|
|
return this;
|
|
|
};
|
|
|
/**
|
|
|
* Reverses this ByteBuffer's contents.
|
|
|
* @param {number=} begin Offset to start at, defaults to {@link ByteBuffer#offset}
|
|
|
* @param {number=} end Offset to end at, defaults to {@link ByteBuffer#limit}
|
|
|
* @returns {!ByteBuffer} this
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBufferPrototype.reverse = function(begin, end) {
|
|
|
if (typeof begin === 'undefined') begin = this.offset;
|
|
|
if (typeof end === 'undefined') end = this.limit;
|
|
|
if (!this.noAssert) {
|
|
|
if (typeof begin !== 'number' || begin % 1 !== 0)
|
|
|
throw TypeError("Illegal begin: Not an integer");
|
|
|
begin >>>= 0;
|
|
|
if (typeof end !== 'number' || end % 1 !== 0)
|
|
|
throw TypeError("Illegal end: Not an integer");
|
|
|
end >>>= 0;
|
|
|
if (begin < 0 || begin > end || end > this.buffer.byteLength)
|
|
|
throw RangeError("Illegal range: 0 <= "+begin+" <= "+end+" <= "+this.buffer.byteLength);
|
|
|
}
|
|
|
if (begin === end)
|
|
|
return this; // Nothing to reverse
|
|
|
Array.prototype.reverse.call(new Uint8Array(this.buffer).subarray(begin, end));
|
|
|
this.view = new DataView(this.buffer); // FIXME: Why exactly is this necessary?
|
|
|
return this;
|
|
|
};
|
|
|
/**
|
|
|
* Skips the next `length` bytes. This will just advance
|
|
|
* @param {number} length Number of bytes to skip. May also be negative to move the offset back.
|
|
|
* @returns {!ByteBuffer} this
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBufferPrototype.skip = function(length) {
|
|
|
if (!this.noAssert) {
|
|
|
if (typeof length !== 'number' || length % 1 !== 0)
|
|
|
throw TypeError("Illegal length: "+length+" (not an integer)");
|
|
|
length |= 0;
|
|
|
}
|
|
|
var offset = this.offset + length;
|
|
|
if (!this.noAssert) {
|
|
|
if (offset < 0 || offset > this.buffer.byteLength)
|
|
|
throw RangeError("Illegal length: 0 <= "+this.offset+" + "+length+" <= "+this.buffer.byteLength);
|
|
|
}
|
|
|
this.offset = offset;
|
|
|
return this;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Slices this ByteBuffer by creating a cloned instance with `offset = begin` and `limit = end`.
|
|
|
* @param {number=} begin Begin offset, defaults to {@link ByteBuffer#offset}.
|
|
|
* @param {number=} end End offset, defaults to {@link ByteBuffer#limit}.
|
|
|
* @returns {!ByteBuffer} Clone of this ByteBuffer with slicing applied, backed by the same {@link ByteBuffer#buffer}
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBufferPrototype.slice = function(begin, end) {
|
|
|
if (typeof begin === 'undefined') begin = this.offset;
|
|
|
if (typeof end === 'undefined') end = this.limit;
|
|
|
if (!this.noAssert) {
|
|
|
if (typeof begin !== 'number' || begin % 1 !== 0)
|
|
|
throw TypeError("Illegal begin: Not an integer");
|
|
|
begin >>>= 0;
|
|
|
if (typeof end !== 'number' || end % 1 !== 0)
|
|
|
throw TypeError("Illegal end: Not an integer");
|
|
|
end >>>= 0;
|
|
|
if (begin < 0 || begin > end || end > this.buffer.byteLength)
|
|
|
throw RangeError("Illegal range: 0 <= "+begin+" <= "+end+" <= "+this.buffer.byteLength);
|
|
|
}
|
|
|
var bb = this.clone();
|
|
|
bb.offset = begin;
|
|
|
bb.limit = end;
|
|
|
return bb;
|
|
|
};
|
|
|
/**
|
|
|
* Returns a copy of the backing buffer that contains this ByteBuffer's contents. Contents are the bytes between
|
|
|
* {@link ByteBuffer#offset} and {@link ByteBuffer#limit}. Will transparently {@link ByteBuffer#flip} this
|
|
|
* ByteBuffer if `offset > limit` but the actual offsets remain untouched.
|
|
|
* @param {boolean=} forceCopy If `true` returns a copy, otherwise returns a view referencing the same memory if
|
|
|
* possible. Defaults to `false`
|
|
|
* @returns {!ArrayBuffer} Contents as an ArrayBuffer
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBufferPrototype.toBuffer = function(forceCopy) {
|
|
|
var offset = this.offset,
|
|
|
limit = this.limit;
|
|
|
if (offset > limit) {
|
|
|
var t = offset;
|
|
|
offset = limit;
|
|
|
limit = t;
|
|
|
}
|
|
|
if (!this.noAssert) {
|
|
|
if (typeof offset !== 'number' || offset % 1 !== 0)
|
|
|
throw TypeError("Illegal offset: Not an integer");
|
|
|
offset >>>= 0;
|
|
|
if (typeof limit !== 'number' || limit % 1 !== 0)
|
|
|
throw TypeError("Illegal limit: Not an integer");
|
|
|
limit >>>= 0;
|
|
|
if (offset < 0 || offset > limit || limit > this.buffer.byteLength)
|
|
|
throw RangeError("Illegal range: 0 <= "+offset+" <= "+limit+" <= "+this.buffer.byteLength);
|
|
|
}
|
|
|
// NOTE: It's not possible to have another ArrayBuffer reference the same memory as the backing buffer. This is
|
|
|
// possible with Uint8Array#subarray only, but we have to return an ArrayBuffer by contract. So:
|
|
|
if (!forceCopy && offset === 0 && limit === this.buffer.byteLength) {
|
|
|
return this.buffer;
|
|
|
}
|
|
|
if (offset === limit) {
|
|
|
return EMPTY_BUFFER;
|
|
|
}
|
|
|
var buffer = new ArrayBuffer(limit - offset);
|
|
|
new Uint8Array(buffer).set(new Uint8Array(this.buffer).subarray(offset, limit), 0);
|
|
|
return buffer;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Returns a raw buffer compacted to contain this ByteBuffer's contents. Contents are the bytes between
|
|
|
* {@link ByteBuffer#offset} and {@link ByteBuffer#limit}. Will transparently {@link ByteBuffer#flip} this
|
|
|
* ByteBuffer if `offset > limit` but the actual offsets remain untouched. This is an alias of
|
|
|
* {@link ByteBuffer#toBuffer}.
|
|
|
* @function
|
|
|
* @param {boolean=} forceCopy If `true` returns a copy, otherwise returns a view referencing the same memory.
|
|
|
* Defaults to `false`
|
|
|
* @returns {!ArrayBuffer} Contents as an ArrayBuffer
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBufferPrototype.toArrayBuffer = ByteBufferPrototype.toBuffer;
|
|
|
|
|
|
|
|
|
/**
|
|
|
* Converts the ByteBuffer's contents to a string.
|
|
|
* @param {string=} encoding Output encoding. Returns an informative string representation if omitted but also allows
|
|
|
* direct conversion to "utf8", "hex", "base64" and "binary" encoding. "debug" returns a hex representation with
|
|
|
* highlighted offsets.
|
|
|
* @param {number=} begin Offset to begin at, defaults to {@link ByteBuffer#offset}
|
|
|
* @param {number=} end Offset to end at, defaults to {@link ByteBuffer#limit}
|
|
|
* @returns {string} String representation
|
|
|
* @throws {Error} If `encoding` is invalid
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBufferPrototype.toString = function(encoding, begin, end) {
|
|
|
if (typeof encoding === 'undefined')
|
|
|
return "ByteBufferAB(offset="+this.offset+",markedOffset="+this.markedOffset+",limit="+this.limit+",capacity="+this.capacity()+")";
|
|
|
if (typeof encoding === 'number')
|
|
|
encoding = "utf8",
|
|
|
begin = encoding,
|
|
|
end = begin;
|
|
|
switch (encoding) {
|
|
|
case "utf8":
|
|
|
return this.toUTF8(begin, end);
|
|
|
case "base64":
|
|
|
return this.toBase64(begin, end);
|
|
|
case "hex":
|
|
|
return this.toHex(begin, end);
|
|
|
case "binary":
|
|
|
return this.toBinary(begin, end);
|
|
|
case "debug":
|
|
|
return this.toDebug();
|
|
|
case "columns":
|
|
|
return this.toColumns();
|
|
|
default:
|
|
|
throw Error("Unsupported encoding: "+encoding);
|
|
|
}
|
|
|
};
|
|
|
|
|
|
// lxiv-embeddable
|
|
|
|
|
|
/**
|
|
|
* lxiv-embeddable (c) 2014 Daniel Wirtz <dcode@dcode.io>
|
|
|
* Released under the Apache License, Version 2.0
|
|
|
* see: https://github.com/dcodeIO/lxiv for details
|
|
|
*/
|
|
|
var lxiv = function() {
|
|
|
"use strict";
|
|
|
|
|
|
/**
|
|
|
* lxiv namespace.
|
|
|
* @type {!Object.<string,*>}
|
|
|
* @exports lxiv
|
|
|
*/
|
|
|
var lxiv = {};
|
|
|
|
|
|
/**
|
|
|
* Character codes for output.
|
|
|
* @type {!Array.<number>}
|
|
|
* @inner
|
|
|
*/
|
|
|
var aout = [
|
|
|
65, 66, 67, 68, 69, 70, 71, 72, 73, 74, 75, 76, 77, 78, 79, 80,
|
|
|
81, 82, 83, 84, 85, 86, 87, 88, 89, 90, 97, 98, 99, 100, 101, 102,
|
|
|
103, 104, 105, 106, 107, 108, 109, 110, 111, 112, 113, 114, 115, 116, 117, 118,
|
|
|
119, 120, 121, 122, 48, 49, 50, 51, 52, 53, 54, 55, 56, 57, 43, 47
|
|
|
];
|
|
|
|
|
|
/**
|
|
|
* Character codes for input.
|
|
|
* @type {!Array.<number>}
|
|
|
* @inner
|
|
|
*/
|
|
|
var ain = [];
|
|
|
for (var i=0, k=aout.length; i<k; ++i)
|
|
|
ain[aout[i]] = i;
|
|
|
|
|
|
/**
|
|
|
* Encodes bytes to base64 char codes.
|
|
|
* @param {!function():number|null} src Bytes source as a function returning the next byte respectively `null` if
|
|
|
* there are no more bytes left.
|
|
|
* @param {!function(number)} dst Characters destination as a function successively called with each encoded char
|
|
|
* code.
|
|
|
*/
|
|
|
lxiv.encode = function(src, dst) {
|
|
|
var b, t;
|
|
|
while ((b = src()) !== null) {
|
|
|
dst(aout[(b>>2)&0x3f]);
|
|
|
t = (b&0x3)<<4;
|
|
|
if ((b = src()) !== null) {
|
|
|
t |= (b>>4)&0xf;
|
|
|
dst(aout[(t|((b>>4)&0xf))&0x3f]);
|
|
|
t = (b&0xf)<<2;
|
|
|
if ((b = src()) !== null)
|
|
|
dst(aout[(t|((b>>6)&0x3))&0x3f]),
|
|
|
dst(aout[b&0x3f]);
|
|
|
else
|
|
|
dst(aout[t&0x3f]),
|
|
|
dst(61);
|
|
|
} else
|
|
|
dst(aout[t&0x3f]),
|
|
|
dst(61),
|
|
|
dst(61);
|
|
|
}
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Decodes base64 char codes to bytes.
|
|
|
* @param {!function():number|null} src Characters source as a function returning the next char code respectively
|
|
|
* `null` if there are no more characters left.
|
|
|
* @param {!function(number)} dst Bytes destination as a function successively called with the next byte.
|
|
|
* @throws {Error} If a character code is invalid
|
|
|
*/
|
|
|
lxiv.decode = function(src, dst) {
|
|
|
var c, t1, t2;
|
|
|
function fail(c) {
|
|
|
throw Error("Illegal character code: "+c);
|
|
|
}
|
|
|
while ((c = src()) !== null) {
|
|
|
t1 = ain[c];
|
|
|
if (typeof t1 === 'undefined') fail(c);
|
|
|
if ((c = src()) !== null) {
|
|
|
t2 = ain[c];
|
|
|
if (typeof t2 === 'undefined') fail(c);
|
|
|
dst((t1<<2)>>>0|(t2&0x30)>>4);
|
|
|
if ((c = src()) !== null) {
|
|
|
t1 = ain[c];
|
|
|
if (typeof t1 === 'undefined')
|
|
|
if (c === 61) break; else fail(c);
|
|
|
dst(((t2&0xf)<<4)>>>0|(t1&0x3c)>>2);
|
|
|
if ((c = src()) !== null) {
|
|
|
t2 = ain[c];
|
|
|
if (typeof t2 === 'undefined')
|
|
|
if (c === 61) break; else fail(c);
|
|
|
dst(((t1&0x3)<<6)>>>0|t2);
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Tests if a string is valid base64.
|
|
|
* @param {string} str String to test
|
|
|
* @returns {boolean} `true` if valid, otherwise `false`
|
|
|
*/
|
|
|
lxiv.test = function(str) {
|
|
|
return /^(?:[A-Za-z0-9+/]{4})*(?:[A-Za-z0-9+/]{2}==|[A-Za-z0-9+/]{3}=)?$/.test(str);
|
|
|
};
|
|
|
|
|
|
return lxiv;
|
|
|
}();
|
|
|
|
|
|
// encodings/base64
|
|
|
|
|
|
/**
|
|
|
* Encodes this ByteBuffer's contents to a base64 encoded string.
|
|
|
* @param {number=} begin Offset to begin at, defaults to {@link ByteBuffer#offset}.
|
|
|
* @param {number=} end Offset to end at, defaults to {@link ByteBuffer#limit}.
|
|
|
* @returns {string} Base64 encoded string
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBufferPrototype.toBase64 = function(begin, end) {
|
|
|
if (typeof begin === 'undefined')
|
|
|
begin = this.offset;
|
|
|
if (typeof end === 'undefined')
|
|
|
end = this.limit;
|
|
|
if (!this.noAssert) {
|
|
|
if (typeof begin !== 'number' || begin % 1 !== 0)
|
|
|
throw TypeError("Illegal begin: Not an integer");
|
|
|
begin >>>= 0;
|
|
|
if (typeof end !== 'number' || end % 1 !== 0)
|
|
|
throw TypeError("Illegal end: Not an integer");
|
|
|
end >>>= 0;
|
|
|
if (begin < 0 || begin > end || end > this.buffer.byteLength)
|
|
|
throw RangeError("Illegal range: 0 <= "+begin+" <= "+end+" <= "+this.buffer.byteLength);
|
|
|
}
|
|
|
var sd; lxiv.encode(function() {
|
|
|
return begin < end ? this.view.getUint8(begin++) : null;
|
|
|
}.bind(this), sd = stringDestination());
|
|
|
return sd();
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Decodes a base64 encoded string to a ByteBuffer.
|
|
|
* @param {string} str String to decode
|
|
|
* @param {boolean=} littleEndian Whether to use little or big endian byte order. Defaults to
|
|
|
* {@link ByteBuffer.DEFAULT_ENDIAN}.
|
|
|
* @param {boolean=} noAssert Whether to skip assertions of offsets and values. Defaults to
|
|
|
* {@link ByteBuffer.DEFAULT_NOASSERT}.
|
|
|
* @returns {!ByteBuffer} ByteBuffer
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBuffer.fromBase64 = function(str, littleEndian, noAssert) {
|
|
|
if (!noAssert) {
|
|
|
if (typeof str !== 'string')
|
|
|
throw TypeError("Illegal str: Not a string");
|
|
|
if (str.length % 4 !== 0)
|
|
|
throw TypeError("Illegal str: Length not a multiple of 4");
|
|
|
}
|
|
|
var bb = new ByteBuffer(str.length/4*3, littleEndian, noAssert),
|
|
|
i = 0;
|
|
|
lxiv.decode(stringSource(str), function(b) {
|
|
|
bb.view.setUint8(i++, b);
|
|
|
});
|
|
|
bb.limit = i;
|
|
|
return bb;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Encodes a binary string to base64 like `window.btoa` does.
|
|
|
* @param {string} str Binary string
|
|
|
* @returns {string} Base64 encoded string
|
|
|
* @see https://developer.mozilla.org/en-US/docs/Web/API/Window.btoa
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBuffer.btoa = function(str) {
|
|
|
return ByteBuffer.fromBinary(str).toBase64();
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Decodes a base64 encoded string to binary like `window.atob` does.
|
|
|
* @param {string} b64 Base64 encoded string
|
|
|
* @returns {string} Binary string
|
|
|
* @see https://developer.mozilla.org/en-US/docs/Web/API/Window.atob
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBuffer.atob = function(b64) {
|
|
|
return ByteBuffer.fromBase64(b64).toBinary();
|
|
|
};
|
|
|
|
|
|
// encodings/binary
|
|
|
|
|
|
/**
|
|
|
* Encodes this ByteBuffer to a binary encoded string, that is using only characters 0x00-0xFF as bytes.
|
|
|
* @param {number=} begin Offset to begin at. Defaults to {@link ByteBuffer#offset}.
|
|
|
* @param {number=} end Offset to end at. Defaults to {@link ByteBuffer#limit}.
|
|
|
* @returns {string} Binary encoded string
|
|
|
* @throws {RangeError} If `offset > limit`
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBufferPrototype.toBinary = function(begin, end) {
|
|
|
begin = typeof begin === 'undefined' ? this.offset : begin;
|
|
|
end = typeof end === 'undefined' ? this.limit : end;
|
|
|
if (!this.noAssert) {
|
|
|
if (typeof begin !== 'number' || begin % 1 !== 0)
|
|
|
throw TypeError("Illegal begin: Not an integer");
|
|
|
begin >>>= 0;
|
|
|
if (typeof end !== 'number' || end % 1 !== 0)
|
|
|
throw TypeError("Illegal end: Not an integer");
|
|
|
end >>>= 0;
|
|
|
if (begin < 0 || begin > end || end > this.buffer.byteLength)
|
|
|
throw RangeError("Illegal range: 0 <= "+begin+" <= "+end+" <= "+this.buffer.byteLength);
|
|
|
}
|
|
|
if (begin === end)
|
|
|
return "";
|
|
|
var cc = [], pt = [];
|
|
|
while (begin < end) {
|
|
|
cc.push(this.view.getUint8(begin++));
|
|
|
if (cc.length >= 1024)
|
|
|
pt.push(String.fromCharCode.apply(String, cc)),
|
|
|
cc = [];
|
|
|
}
|
|
|
return pt.join('') + String.fromCharCode.apply(String, cc);
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Decodes a binary encoded string, that is using only characters 0x00-0xFF as bytes, to a ByteBuffer.
|
|
|
* @param {string} str String to decode
|
|
|
* @param {boolean=} littleEndian Whether to use little or big endian byte order. Defaults to
|
|
|
* {@link ByteBuffer.DEFAULT_ENDIAN}.
|
|
|
* @param {boolean=} noAssert Whether to skip assertions of offsets and values. Defaults to
|
|
|
* {@link ByteBuffer.DEFAULT_NOASSERT}.
|
|
|
* @returns {!ByteBuffer} ByteBuffer
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBuffer.fromBinary = function(str, littleEndian, noAssert) {
|
|
|
if (!noAssert) {
|
|
|
if (typeof str !== 'string')
|
|
|
throw TypeError("Illegal str: Not a string");
|
|
|
}
|
|
|
var i = 0, k = str.length, charCode,
|
|
|
bb = new ByteBuffer(k, littleEndian, noAssert);
|
|
|
while (i<k) {
|
|
|
charCode = str.charCodeAt(i);
|
|
|
if (!noAssert && charCode > 255)
|
|
|
throw RangeError("Illegal charCode at "+i+": 0 <= "+charCode+" <= 255");
|
|
|
bb.view.setUint8(i++, charCode);
|
|
|
}
|
|
|
bb.limit = k;
|
|
|
return bb;
|
|
|
};
|
|
|
|
|
|
// encodings/debug
|
|
|
|
|
|
/**
|
|
|
* Encodes this ByteBuffer to a hex encoded string with marked offsets. Offset symbols are:
|
|
|
* * `<` : offset,
|
|
|
* * `'` : markedOffset,
|
|
|
* * `>` : limit,
|
|
|
* * `|` : offset and limit,
|
|
|
* * `[` : offset and markedOffset,
|
|
|
* * `]` : markedOffset and limit,
|
|
|
* * `!` : offset, markedOffset and limit
|
|
|
* @param {boolean=} columns If `true` returns two columns hex + ascii, defaults to `false`
|
|
|
* @returns {string|!Array.<string>} Debug string or array of lines if `asArray = true`
|
|
|
* @expose
|
|
|
* @example `>00'01 02<03` contains four bytes with `limit=0, markedOffset=1, offset=3`
|
|
|
* @example `00[01 02 03>` contains four bytes with `offset=markedOffset=1, limit=4`
|
|
|
* @example `00|01 02 03` contains four bytes with `offset=limit=1, markedOffset=-1`
|
|
|
* @example `|` contains zero bytes with `offset=limit=0, markedOffset=-1`
|
|
|
*/
|
|
|
ByteBufferPrototype.toDebug = function(columns) {
|
|
|
var i = -1,
|
|
|
k = this.buffer.byteLength,
|
|
|
b,
|
|
|
hex = "",
|
|
|
asc = "",
|
|
|
out = "";
|
|
|
while (i<k) {
|
|
|
if (i !== -1) {
|
|
|
b = this.view.getUint8(i);
|
|
|
if (b < 0x10) hex += "0"+b.toString(16).toUpperCase();
|
|
|
else hex += b.toString(16).toUpperCase();
|
|
|
if (columns) {
|
|
|
asc += b > 32 && b < 127 ? String.fromCharCode(b) : '.';
|
|
|
}
|
|
|
}
|
|
|
++i;
|
|
|
if (columns) {
|
|
|
if (i > 0 && i % 16 === 0 && i !== k) {
|
|
|
while (hex.length < 3*16+3) hex += " ";
|
|
|
out += hex+asc+"\n";
|
|
|
hex = asc = "";
|
|
|
}
|
|
|
}
|
|
|
if (i === this.offset && i === this.limit)
|
|
|
hex += i === this.markedOffset ? "!" : "|";
|
|
|
else if (i === this.offset)
|
|
|
hex += i === this.markedOffset ? "[" : "<";
|
|
|
else if (i === this.limit)
|
|
|
hex += i === this.markedOffset ? "]" : ">";
|
|
|
else
|
|
|
hex += i === this.markedOffset ? "'" : (columns || (i !== 0 && i !== k) ? " " : "");
|
|
|
}
|
|
|
if (columns && hex !== " ") {
|
|
|
while (hex.length < 3*16+3) hex += " ";
|
|
|
out += hex+asc+"\n";
|
|
|
}
|
|
|
return columns ? out : hex;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Decodes a hex encoded string with marked offsets to a ByteBuffer.
|
|
|
* @param {string} str Debug string to decode (not be generated with `columns = true`)
|
|
|
* @param {boolean=} littleEndian Whether to use little or big endian byte order. Defaults to
|
|
|
* {@link ByteBuffer.DEFAULT_ENDIAN}.
|
|
|
* @param {boolean=} noAssert Whether to skip assertions of offsets and values. Defaults to
|
|
|
* {@link ByteBuffer.DEFAULT_NOASSERT}.
|
|
|
* @returns {!ByteBuffer} ByteBuffer
|
|
|
* @expose
|
|
|
* @see ByteBuffer#toDebug
|
|
|
*/
|
|
|
ByteBuffer.fromDebug = function(str, littleEndian, noAssert) {
|
|
|
var k = str.length,
|
|
|
bb = new ByteBuffer(((k+1)/3)|0, littleEndian, noAssert);
|
|
|
var i = 0, j = 0, ch, b,
|
|
|
rs = false, // Require symbol next
|
|
|
ho = false, hm = false, hl = false, // Already has offset, markedOffset, limit?
|
|
|
fail = false;
|
|
|
while (i<k) {
|
|
|
switch (ch = str.charAt(i++)) {
|
|
|
case '!':
|
|
|
if (!noAssert) {
|
|
|
if (ho || hm || hl) {
|
|
|
fail = true; break;
|
|
|
}
|
|
|
ho = hm = hl = true;
|
|
|
}
|
|
|
bb.offset = bb.markedOffset = bb.limit = j;
|
|
|
rs = false;
|
|
|
break;
|
|
|
case '|':
|
|
|
if (!noAssert) {
|
|
|
if (ho || hl) {
|
|
|
fail = true; break;
|
|
|
}
|
|
|
ho = hl = true;
|
|
|
}
|
|
|
bb.offset = bb.limit = j;
|
|
|
rs = false;
|
|
|
break;
|
|
|
case '[':
|
|
|
if (!noAssert) {
|
|
|
if (ho || hm) {
|
|
|
fail = true; break;
|
|
|
}
|
|
|
ho = hm = true;
|
|
|
}
|
|
|
bb.offset = bb.markedOffset = j;
|
|
|
rs = false;
|
|
|
break;
|
|
|
case '<':
|
|
|
if (!noAssert) {
|
|
|
if (ho) {
|
|
|
fail = true; break;
|
|
|
}
|
|
|
ho = true;
|
|
|
}
|
|
|
bb.offset = j;
|
|
|
rs = false;
|
|
|
break;
|
|
|
case ']':
|
|
|
if (!noAssert) {
|
|
|
if (hl || hm) {
|
|
|
fail = true; break;
|
|
|
}
|
|
|
hl = hm = true;
|
|
|
}
|
|
|
bb.limit = bb.markedOffset = j;
|
|
|
rs = false;
|
|
|
break;
|
|
|
case '>':
|
|
|
if (!noAssert) {
|
|
|
if (hl) {
|
|
|
fail = true; break;
|
|
|
}
|
|
|
hl = true;
|
|
|
}
|
|
|
bb.limit = j;
|
|
|
rs = false;
|
|
|
break;
|
|
|
case "'":
|
|
|
if (!noAssert) {
|
|
|
if (hm) {
|
|
|
fail = true; break;
|
|
|
}
|
|
|
hm = true;
|
|
|
}
|
|
|
bb.markedOffset = j;
|
|
|
rs = false;
|
|
|
break;
|
|
|
case ' ':
|
|
|
rs = false;
|
|
|
break;
|
|
|
default:
|
|
|
if (!noAssert) {
|
|
|
if (rs) {
|
|
|
fail = true; break;
|
|
|
}
|
|
|
}
|
|
|
b = parseInt(ch+str.charAt(i++), 16);
|
|
|
if (!noAssert) {
|
|
|
if (isNaN(b) || b < 0 || b > 255)
|
|
|
throw TypeError("Illegal str: Not a debug encoded string");
|
|
|
}
|
|
|
bb.view.setUint8(j++, b);
|
|
|
rs = true;
|
|
|
}
|
|
|
if (fail)
|
|
|
throw TypeError("Illegal str: Invalid symbol at "+i);
|
|
|
}
|
|
|
if (!noAssert) {
|
|
|
if (!ho || !hl)
|
|
|
throw TypeError("Illegal str: Missing offset or limit");
|
|
|
if (j<bb.buffer.byteLength)
|
|
|
throw TypeError("Illegal str: Not a debug encoded string (is it hex?) "+j+" < "+k);
|
|
|
}
|
|
|
return bb;
|
|
|
};
|
|
|
|
|
|
// encodings/hex
|
|
|
|
|
|
/**
|
|
|
* Encodes this ByteBuffer's contents to a hex encoded string.
|
|
|
* @param {number=} begin Offset to begin at. Defaults to {@link ByteBuffer#offset}.
|
|
|
* @param {number=} end Offset to end at. Defaults to {@link ByteBuffer#limit}.
|
|
|
* @returns {string} Hex encoded string
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBufferPrototype.toHex = function(begin, end) {
|
|
|
begin = typeof begin === 'undefined' ? this.offset : begin;
|
|
|
end = typeof end === 'undefined' ? this.limit : end;
|
|
|
if (!this.noAssert) {
|
|
|
if (typeof begin !== 'number' || begin % 1 !== 0)
|
|
|
throw TypeError("Illegal begin: Not an integer");
|
|
|
begin >>>= 0;
|
|
|
if (typeof end !== 'number' || end % 1 !== 0)
|
|
|
throw TypeError("Illegal end: Not an integer");
|
|
|
end >>>= 0;
|
|
|
if (begin < 0 || begin > end || end > this.buffer.byteLength)
|
|
|
throw RangeError("Illegal range: 0 <= "+begin+" <= "+end+" <= "+this.buffer.byteLength);
|
|
|
}
|
|
|
var out = new Array(end - begin),
|
|
|
b;
|
|
|
while (begin < end) {
|
|
|
b = this.view.getUint8(begin++);
|
|
|
if (b < 0x10)
|
|
|
out.push("0", b.toString(16));
|
|
|
else out.push(b.toString(16));
|
|
|
}
|
|
|
return out.join('');
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Decodes a hex encoded string to a ByteBuffer.
|
|
|
* @param {string} str String to decode
|
|
|
* @param {boolean=} littleEndian Whether to use little or big endian byte order. Defaults to
|
|
|
* {@link ByteBuffer.DEFAULT_ENDIAN}.
|
|
|
* @param {boolean=} noAssert Whether to skip assertions of offsets and values. Defaults to
|
|
|
* {@link ByteBuffer.DEFAULT_NOASSERT}.
|
|
|
* @returns {!ByteBuffer} ByteBuffer
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBuffer.fromHex = function(str, littleEndian, noAssert) {
|
|
|
if (!noAssert) {
|
|
|
if (typeof str !== 'string')
|
|
|
throw TypeError("Illegal str: Not a string");
|
|
|
if (str.length % 2 !== 0)
|
|
|
throw TypeError("Illegal str: Length not a multiple of 2");
|
|
|
}
|
|
|
var k = str.length,
|
|
|
bb = new ByteBuffer((k / 2) | 0, littleEndian),
|
|
|
b;
|
|
|
for (var i=0, j=0; i<k; i+=2) {
|
|
|
b = parseInt(str.substring(i, i+2), 16);
|
|
|
if (!noAssert)
|
|
|
if (!isFinite(b) || b < 0 || b > 255)
|
|
|
throw TypeError("Illegal str: Contains non-hex characters");
|
|
|
bb.view.setUint8(j++, b);
|
|
|
}
|
|
|
bb.limit = j;
|
|
|
return bb;
|
|
|
};
|
|
|
|
|
|
// utfx-embeddable
|
|
|
|
|
|
/**
|
|
|
* utfx-embeddable (c) 2014 Daniel Wirtz <dcode@dcode.io>
|
|
|
* Released under the Apache License, Version 2.0
|
|
|
* see: https://github.com/dcodeIO/utfx for details
|
|
|
*/
|
|
|
var utfx = function() {
|
|
|
"use strict";
|
|
|
|
|
|
/**
|
|
|
* utfx namespace.
|
|
|
* @inner
|
|
|
* @type {!Object.<string,*>}
|
|
|
*/
|
|
|
var utfx = {};
|
|
|
|
|
|
/**
|
|
|
* Maximum valid code point.
|
|
|
* @type {number}
|
|
|
* @const
|
|
|
*/
|
|
|
utfx.MAX_CODEPOINT = 0x10FFFF;
|
|
|
|
|
|
/**
|
|
|
* Encodes UTF8 code points to UTF8 bytes.
|
|
|
* @param {(!function():number|null) | number} src Code points source, either as a function returning the next code point
|
|
|
* respectively `null` if there are no more code points left or a single numeric code point.
|
|
|
* @param {!function(number)} dst Bytes destination as a function successively called with the next byte
|
|
|
*/
|
|
|
utfx.encodeUTF8 = function(src, dst) {
|
|
|
var cp = null;
|
|
|
if (typeof src === 'number')
|
|
|
cp = src,
|
|
|
src = function() { return null; };
|
|
|
while (cp !== null || (cp = src()) !== null) {
|
|
|
if (cp < 0x80)
|
|
|
dst(cp&0x7F);
|
|
|
else if (cp < 0x800)
|
|
|
dst(((cp>>6)&0x1F)|0xC0),
|
|
|
dst((cp&0x3F)|0x80);
|
|
|
else if (cp < 0x10000)
|
|
|
dst(((cp>>12)&0x0F)|0xE0),
|
|
|
dst(((cp>>6)&0x3F)|0x80),
|
|
|
dst((cp&0x3F)|0x80);
|
|
|
else
|
|
|
dst(((cp>>18)&0x07)|0xF0),
|
|
|
dst(((cp>>12)&0x3F)|0x80),
|
|
|
dst(((cp>>6)&0x3F)|0x80),
|
|
|
dst((cp&0x3F)|0x80);
|
|
|
cp = null;
|
|
|
}
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Decodes UTF8 bytes to UTF8 code points.
|
|
|
* @param {!function():number|null} src Bytes source as a function returning the next byte respectively `null` if there
|
|
|
* are no more bytes left.
|
|
|
* @param {!function(number)} dst Code points destination as a function successively called with each decoded code point.
|
|
|
* @throws {RangeError} If a starting byte is invalid in UTF8
|
|
|
* @throws {Error} If the last sequence is truncated. Has an array property `bytes` holding the
|
|
|
* remaining bytes.
|
|
|
*/
|
|
|
utfx.decodeUTF8 = function(src, dst) {
|
|
|
var a, b, c, d, fail = function(b) {
|
|
|
b = b.slice(0, b.indexOf(null));
|
|
|
var err = Error(b.toString());
|
|
|
err.name = "TruncatedError";
|
|
|
err['bytes'] = b;
|
|
|
throw err;
|
|
|
};
|
|
|
while ((a = src()) !== null) {
|
|
|
if ((a&0x80) === 0)
|
|
|
dst(a);
|
|
|
else if ((a&0xE0) === 0xC0)
|
|
|
((b = src()) === null) && fail([a, b]),
|
|
|
dst(((a&0x1F)<<6) | (b&0x3F));
|
|
|
else if ((a&0xF0) === 0xE0)
|
|
|
((b=src()) === null || (c=src()) === null) && fail([a, b, c]),
|
|
|
dst(((a&0x0F)<<12) | ((b&0x3F)<<6) | (c&0x3F));
|
|
|
else if ((a&0xF8) === 0xF0)
|
|
|
((b=src()) === null || (c=src()) === null || (d=src()) === null) && fail([a, b, c ,d]),
|
|
|
dst(((a&0x07)<<18) | ((b&0x3F)<<12) | ((c&0x3F)<<6) | (d&0x3F));
|
|
|
else throw RangeError("Illegal starting byte: "+a);
|
|
|
}
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Converts UTF16 characters to UTF8 code points.
|
|
|
* @param {!function():number|null} src Characters source as a function returning the next char code respectively
|
|
|
* `null` if there are no more characters left.
|
|
|
* @param {!function(number)} dst Code points destination as a function successively called with each converted code
|
|
|
* point.
|
|
|
*/
|
|
|
utfx.UTF16toUTF8 = function(src, dst) {
|
|
|
var c1, c2 = null;
|
|
|
while (true) {
|
|
|
if ((c1 = c2 !== null ? c2 : src()) === null)
|
|
|
break;
|
|
|
if (c1 >= 0xD800 && c1 <= 0xDFFF) {
|
|
|
if ((c2 = src()) !== null) {
|
|
|
if (c2 >= 0xDC00 && c2 <= 0xDFFF) {
|
|
|
dst((c1-0xD800)*0x400+c2-0xDC00+0x10000);
|
|
|
c2 = null; continue;
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
dst(c1);
|
|
|
}
|
|
|
if (c2 !== null) dst(c2);
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Converts UTF8 code points to UTF16 characters.
|
|
|
* @param {(!function():number|null) | number} src Code points source, either as a function returning the next code point
|
|
|
* respectively `null` if there are no more code points left or a single numeric code point.
|
|
|
* @param {!function(number)} dst Characters destination as a function successively called with each converted char code.
|
|
|
* @throws {RangeError} If a code point is out of range
|
|
|
*/
|
|
|
utfx.UTF8toUTF16 = function(src, dst) {
|
|
|
var cp = null;
|
|
|
if (typeof src === 'number')
|
|
|
cp = src, src = function() { return null; };
|
|
|
while (cp !== null || (cp = src()) !== null) {
|
|
|
if (cp <= 0xFFFF)
|
|
|
dst(cp);
|
|
|
else
|
|
|
cp -= 0x10000,
|
|
|
dst((cp>>10)+0xD800),
|
|
|
dst((cp%0x400)+0xDC00);
|
|
|
cp = null;
|
|
|
}
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Converts and encodes UTF16 characters to UTF8 bytes.
|
|
|
* @param {!function():number|null} src Characters source as a function returning the next char code respectively `null`
|
|
|
* if there are no more characters left.
|
|
|
* @param {!function(number)} dst Bytes destination as a function successively called with the next byte.
|
|
|
*/
|
|
|
utfx.encodeUTF16toUTF8 = function(src, dst) {
|
|
|
utfx.UTF16toUTF8(src, function(cp) {
|
|
|
utfx.encodeUTF8(cp, dst);
|
|
|
});
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Decodes and converts UTF8 bytes to UTF16 characters.
|
|
|
* @param {!function():number|null} src Bytes source as a function returning the next byte respectively `null` if there
|
|
|
* are no more bytes left.
|
|
|
* @param {!function(number)} dst Characters destination as a function successively called with each converted char code.
|
|
|
* @throws {RangeError} If a starting byte is invalid in UTF8
|
|
|
* @throws {Error} If the last sequence is truncated. Has an array property `bytes` holding the remaining bytes.
|
|
|
*/
|
|
|
utfx.decodeUTF8toUTF16 = function(src, dst) {
|
|
|
utfx.decodeUTF8(src, function(cp) {
|
|
|
utfx.UTF8toUTF16(cp, dst);
|
|
|
});
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Calculates the byte length of an UTF8 code point.
|
|
|
* @param {number} cp UTF8 code point
|
|
|
* @returns {number} Byte length
|
|
|
*/
|
|
|
utfx.calculateCodePoint = function(cp) {
|
|
|
return (cp < 0x80) ? 1 : (cp < 0x800) ? 2 : (cp < 0x10000) ? 3 : 4;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Calculates the number of UTF8 bytes required to store UTF8 code points.
|
|
|
* @param {(!function():number|null)} src Code points source as a function returning the next code point respectively
|
|
|
* `null` if there are no more code points left.
|
|
|
* @returns {number} The number of UTF8 bytes required
|
|
|
*/
|
|
|
utfx.calculateUTF8 = function(src) {
|
|
|
var cp, l=0;
|
|
|
while ((cp = src()) !== null)
|
|
|
l += utfx.calculateCodePoint(cp);
|
|
|
return l;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Calculates the number of UTF8 code points respectively UTF8 bytes required to store UTF16 char codes.
|
|
|
* @param {(!function():number|null)} src Characters source as a function returning the next char code respectively
|
|
|
* `null` if there are no more characters left.
|
|
|
* @returns {!Array.<number>} The number of UTF8 code points at index 0 and the number of UTF8 bytes required at index 1.
|
|
|
*/
|
|
|
utfx.calculateUTF16asUTF8 = function(src) {
|
|
|
var n=0, l=0;
|
|
|
utfx.UTF16toUTF8(src, function(cp) {
|
|
|
++n; l += utfx.calculateCodePoint(cp);
|
|
|
});
|
|
|
return [n,l];
|
|
|
};
|
|
|
|
|
|
return utfx;
|
|
|
}();
|
|
|
|
|
|
// encodings/utf8
|
|
|
|
|
|
/**
|
|
|
* Encodes this ByteBuffer's contents between {@link ByteBuffer#offset} and {@link ByteBuffer#limit} to an UTF8 encoded
|
|
|
* string.
|
|
|
* @returns {string} Hex encoded string
|
|
|
* @throws {RangeError} If `offset > limit`
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBufferPrototype.toUTF8 = function(begin, end) {
|
|
|
if (typeof begin === 'undefined') begin = this.offset;
|
|
|
if (typeof end === 'undefined') end = this.limit;
|
|
|
if (!this.noAssert) {
|
|
|
if (typeof begin !== 'number' || begin % 1 !== 0)
|
|
|
throw TypeError("Illegal begin: Not an integer");
|
|
|
begin >>>= 0;
|
|
|
if (typeof end !== 'number' || end % 1 !== 0)
|
|
|
throw TypeError("Illegal end: Not an integer");
|
|
|
end >>>= 0;
|
|
|
if (begin < 0 || begin > end || end > this.buffer.byteLength)
|
|
|
throw RangeError("Illegal range: 0 <= "+begin+" <= "+end+" <= "+this.buffer.byteLength);
|
|
|
}
|
|
|
var sd; try {
|
|
|
utfx.decodeUTF8toUTF16(function() {
|
|
|
return begin < end ? this.view.getUint8(begin++) : null;
|
|
|
}.bind(this), sd = stringDestination());
|
|
|
} catch (e) {
|
|
|
if (begin !== end)
|
|
|
throw RangeError("Illegal range: Truncated data, "+begin+" != "+end);
|
|
|
}
|
|
|
return sd();
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Decodes an UTF8 encoded string to a ByteBuffer.
|
|
|
* @param {string} str String to decode
|
|
|
* @param {boolean=} littleEndian Whether to use little or big endian byte order. Defaults to
|
|
|
* {@link ByteBuffer.DEFAULT_ENDIAN}.
|
|
|
* @param {boolean=} noAssert Whether to skip assertions of offsets and values. Defaults to
|
|
|
* {@link ByteBuffer.DEFAULT_NOASSERT}.
|
|
|
* @returns {!ByteBuffer} ByteBuffer
|
|
|
* @expose
|
|
|
*/
|
|
|
ByteBuffer.fromUTF8 = function(str, littleEndian, noAssert) {
|
|
|
if (!noAssert)
|
|
|
if (typeof str !== 'string')
|
|
|
throw TypeError("Illegal str: Not a string");
|
|
|
var bb = new ByteBuffer(utfx.calculateUTF16asUTF8(stringSource(str), true)[1], littleEndian, noAssert),
|
|
|
i = 0;
|
|
|
utfx.encodeUTF16toUTF8(stringSource(str), function(b) {
|
|
|
bb.view.setUint8(i++, b);
|
|
|
});
|
|
|
bb.limit = i;
|
|
|
return bb;
|
|
|
};
|
|
|
|
|
|
|
|
|
return ByteBuffer;
|
|
|
}
|
|
|
|
|
|
/* CommonJS */ if (typeof require === 'function' && typeof module === 'object' && module && typeof exports === 'object' && exports)
|
|
|
module['exports'] = (function() {
|
|
|
var Long; try { Long = require("long"); } catch (e) {}
|
|
|
return loadByteBuffer(Long);
|
|
|
})();
|
|
|
/* AMD */ else if (typeof define === 'function' && define["amd"])
|
|
|
define("ByteBuffer", ["Long"], function(Long) { return loadByteBuffer(Long); });
|
|
|
/* Global */ else
|
|
|
(global["dcodeIO"] = global["dcodeIO"] || {})["ByteBuffer"] = loadByteBuffer(global["dcodeIO"]["Long"]);
|
|
|
|
|
|
})(this);
|
|
|
|
|
|
/*
|
|
|
Copyright 2013 Daniel Wirtz <dcode@dcode.io>
|
|
|
|
|
|
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.
|
|
|
*/
|
|
|
|
|
|
/**
|
|
|
* @license ProtoBuf.js (c) 2013 Daniel Wirtz <dcode@dcode.io>
|
|
|
* Released under the Apache License, Version 2.0
|
|
|
* see: https://github.com/dcodeIO/ProtoBuf.js for details
|
|
|
*/
|
|
|
(function(global) {
|
|
|
"use strict";
|
|
|
|
|
|
function init(ByteBuffer) {
|
|
|
|
|
|
/**
|
|
|
* The ProtoBuf namespace.
|
|
|
* @exports ProtoBuf
|
|
|
* @namespace
|
|
|
* @expose
|
|
|
*/
|
|
|
var ProtoBuf = {};
|
|
|
|
|
|
/**
|
|
|
* ProtoBuf.js version.
|
|
|
* @type {string}
|
|
|
* @const
|
|
|
* @expose
|
|
|
*/
|
|
|
ProtoBuf.VERSION = "3.8.0";
|
|
|
|
|
|
/**
|
|
|
* Wire types.
|
|
|
* @type {Object.<string,number>}
|
|
|
* @const
|
|
|
* @expose
|
|
|
*/
|
|
|
ProtoBuf.WIRE_TYPES = {};
|
|
|
|
|
|
/**
|
|
|
* Varint wire type.
|
|
|
* @type {number}
|
|
|
* @expose
|
|
|
*/
|
|
|
ProtoBuf.WIRE_TYPES.VARINT = 0;
|
|
|
|
|
|
/**
|
|
|
* Fixed 64 bits wire type.
|
|
|
* @type {number}
|
|
|
* @const
|
|
|
* @expose
|
|
|
*/
|
|
|
ProtoBuf.WIRE_TYPES.BITS64 = 1;
|
|
|
|
|
|
/**
|
|
|
* Length delimited wire type.
|
|
|
* @type {number}
|
|
|
* @const
|
|
|
* @expose
|
|
|
*/
|
|
|
ProtoBuf.WIRE_TYPES.LDELIM = 2;
|
|
|
|
|
|
/**
|
|
|
* Start group wire type.
|
|
|
* @type {number}
|
|
|
* @const
|
|
|
* @expose
|
|
|
*/
|
|
|
ProtoBuf.WIRE_TYPES.STARTGROUP = 3;
|
|
|
|
|
|
/**
|
|
|
* End group wire type.
|
|
|
* @type {number}
|
|
|
* @const
|
|
|
* @expose
|
|
|
*/
|
|
|
ProtoBuf.WIRE_TYPES.ENDGROUP = 4;
|
|
|
|
|
|
/**
|
|
|
* Fixed 32 bits wire type.
|
|
|
* @type {number}
|
|
|
* @const
|
|
|
* @expose
|
|
|
*/
|
|
|
ProtoBuf.WIRE_TYPES.BITS32 = 5;
|
|
|
|
|
|
/**
|
|
|
* Packable wire types.
|
|
|
* @type {!Array.<number>}
|
|
|
* @const
|
|
|
* @expose
|
|
|
*/
|
|
|
ProtoBuf.PACKABLE_WIRE_TYPES = [
|
|
|
ProtoBuf.WIRE_TYPES.VARINT,
|
|
|
ProtoBuf.WIRE_TYPES.BITS64,
|
|
|
ProtoBuf.WIRE_TYPES.BITS32
|
|
|
];
|
|
|
|
|
|
/**
|
|
|
* Types.
|
|
|
* @dict
|
|
|
* @type {Object.<string,{name: string, wireType: number}>}
|
|
|
* @const
|
|
|
* @expose
|
|
|
*/
|
|
|
ProtoBuf.TYPES = {
|
|
|
// According to the protobuf spec.
|
|
|
"int32": {
|
|
|
name: "int32",
|
|
|
wireType: ProtoBuf.WIRE_TYPES.VARINT
|
|
|
},
|
|
|
"uint32": {
|
|
|
name: "uint32",
|
|
|
wireType: ProtoBuf.WIRE_TYPES.VARINT
|
|
|
},
|
|
|
"sint32": {
|
|
|
name: "sint32",
|
|
|
wireType: ProtoBuf.WIRE_TYPES.VARINT
|
|
|
},
|
|
|
"int64": {
|
|
|
name: "int64",
|
|
|
wireType: ProtoBuf.WIRE_TYPES.VARINT
|
|
|
},
|
|
|
"uint64": {
|
|
|
name: "uint64",
|
|
|
wireType: ProtoBuf.WIRE_TYPES.VARINT
|
|
|
},
|
|
|
"sint64": {
|
|
|
name: "sint64",
|
|
|
wireType: ProtoBuf.WIRE_TYPES.VARINT
|
|
|
},
|
|
|
"bool": {
|
|
|
name: "bool",
|
|
|
wireType: ProtoBuf.WIRE_TYPES.VARINT
|
|
|
},
|
|
|
"double": {
|
|
|
name: "double",
|
|
|
wireType: ProtoBuf.WIRE_TYPES.BITS64
|
|
|
},
|
|
|
"string": {
|
|
|
name: "string",
|
|
|
wireType: ProtoBuf.WIRE_TYPES.LDELIM
|
|
|
},
|
|
|
"bytes": {
|
|
|
name: "bytes",
|
|
|
wireType: ProtoBuf.WIRE_TYPES.LDELIM
|
|
|
},
|
|
|
"fixed32": {
|
|
|
name: "fixed32",
|
|
|
wireType: ProtoBuf.WIRE_TYPES.BITS32
|
|
|
},
|
|
|
"sfixed32": {
|
|
|
name: "sfixed32",
|
|
|
wireType: ProtoBuf.WIRE_TYPES.BITS32
|
|
|
},
|
|
|
"fixed64": {
|
|
|
name: "fixed64",
|
|
|
wireType: ProtoBuf.WIRE_TYPES.BITS64
|
|
|
},
|
|
|
"sfixed64": {
|
|
|
name: "sfixed64",
|
|
|
wireType: ProtoBuf.WIRE_TYPES.BITS64
|
|
|
},
|
|
|
"float": {
|
|
|
name: "float",
|
|
|
wireType: ProtoBuf.WIRE_TYPES.BITS32
|
|
|
},
|
|
|
"enum": {
|
|
|
name: "enum",
|
|
|
wireType: ProtoBuf.WIRE_TYPES.VARINT
|
|
|
},
|
|
|
"message": {
|
|
|
name: "message",
|
|
|
wireType: ProtoBuf.WIRE_TYPES.LDELIM
|
|
|
},
|
|
|
"group": {
|
|
|
name: "group",
|
|
|
wireType: ProtoBuf.WIRE_TYPES.STARTGROUP
|
|
|
}
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Minimum field id.
|
|
|
* @type {number}
|
|
|
* @const
|
|
|
* @expose
|
|
|
*/
|
|
|
ProtoBuf.ID_MIN = 1;
|
|
|
|
|
|
/**
|
|
|
* Maximum field id.
|
|
|
* @type {number}
|
|
|
* @const
|
|
|
* @expose
|
|
|
*/
|
|
|
ProtoBuf.ID_MAX = 0x1FFFFFFF;
|
|
|
|
|
|
/**
|
|
|
* @type {!function(new: ByteBuffer, ...[*])}
|
|
|
* @expose
|
|
|
*/
|
|
|
ProtoBuf.ByteBuffer = ByteBuffer;
|
|
|
|
|
|
/**
|
|
|
* @type {?function(new: Long, ...[*])}
|
|
|
* @expose
|
|
|
*/
|
|
|
ProtoBuf.Long = ByteBuffer.Long || null;
|
|
|
|
|
|
/**
|
|
|
* If set to `true`, field names will be converted from underscore notation to camel case. Defaults to `false`.
|
|
|
* Must be set prior to parsing.
|
|
|
* @type {boolean}
|
|
|
* @expose
|
|
|
*/
|
|
|
ProtoBuf.convertFieldsToCamelCase = false;
|
|
|
|
|
|
/**
|
|
|
* By default, messages are populated with (setX, set_x) accessors for each field. This can be disabled by
|
|
|
* setting this to `false` prior to building messages.
|
|
|
* @type {boolean}
|
|
|
* @expose
|
|
|
*/
|
|
|
ProtoBuf.populateAccessors = true;
|
|
|
|
|
|
/**
|
|
|
* @alias ProtoBuf.Util
|
|
|
* @expose
|
|
|
*/
|
|
|
ProtoBuf.Util = (function() {
|
|
|
"use strict";
|
|
|
|
|
|
// Object.create polyfill
|
|
|
// ref: https://developer.mozilla.org/de/docs/JavaScript/Reference/Global_Objects/Object/create
|
|
|
if (!Object.create)
|
|
|
/** @expose */
|
|
|
Object.create = function (o) {
|
|
|
if (arguments.length > 1)
|
|
|
throw Error('Object.create polyfill only accepts the first parameter.');
|
|
|
function F() {}
|
|
|
F.prototype = o;
|
|
|
return new F();
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* ProtoBuf utilities.
|
|
|
* @exports ProtoBuf.Util
|
|
|
* @namespace
|
|
|
*/
|
|
|
var Util = {};
|
|
|
|
|
|
/**
|
|
|
* Flag if running in node (fs is available) or not.
|
|
|
* @type {boolean}
|
|
|
* @const
|
|
|
* @expose
|
|
|
*/
|
|
|
Util.IS_NODE = false;
|
|
|
try {
|
|
|
// There is no reliable way to detect node.js as an environment, so our
|
|
|
// best bet is to feature-detect what we actually need.
|
|
|
Util.IS_NODE =
|
|
|
typeof require === 'function' &&
|
|
|
typeof require("fs").readFileSync === 'function' &&
|
|
|
typeof require("path").resolve === 'function';
|
|
|
} catch (e) {}
|
|
|
|
|
|
/**
|
|
|
* Constructs a XMLHttpRequest object.
|
|
|
* @return {XMLHttpRequest}
|
|
|
* @throws {Error} If XMLHttpRequest is not supported
|
|
|
* @expose
|
|
|
*/
|
|
|
Util.XHR = function() {
|
|
|
// No dependencies please, ref: http://www.quirksmode.org/js/xmlhttp.html
|
|
|
var XMLHttpFactories = [
|
|
|
function () {return new XMLHttpRequest()},
|
|
|
function () {return new ActiveXObject("Msxml2.XMLHTTP")},
|
|
|
function () {return new ActiveXObject("Msxml3.XMLHTTP")},
|
|
|
function () {return new ActiveXObject("Microsoft.XMLHTTP")}
|
|
|
];
|
|
|
/** @type {?XMLHttpRequest} */
|
|
|
var xhr = null;
|
|
|
for (var i=0;i<XMLHttpFactories.length;i++) {
|
|
|
try { xhr = XMLHttpFactories[i](); }
|
|
|
catch (e) { continue; }
|
|
|
break;
|
|
|
}
|
|
|
if (!xhr)
|
|
|
throw Error("XMLHttpRequest is not supported");
|
|
|
return xhr;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Fetches a resource.
|
|
|
* @param {string} path Resource path
|
|
|
* @param {function(?string)=} callback Callback receiving the resource's contents. If omitted the resource will
|
|
|
* be fetched synchronously. If the request failed, contents will be null.
|
|
|
* @return {?string|undefined} Resource contents if callback is omitted (null if the request failed), else undefined.
|
|
|
* @expose
|
|
|
*/
|
|
|
Util.fetch = function(path, callback) {
|
|
|
if (callback && typeof callback != 'function')
|
|
|
callback = null;
|
|
|
if (Util.IS_NODE) {
|
|
|
if (callback) {
|
|
|
require("fs").readFile(path, function(err, data) {
|
|
|
if (err)
|
|
|
callback(null);
|
|
|
else
|
|
|
callback(""+data);
|
|
|
});
|
|
|
} else
|
|
|
try {
|
|
|
return require("fs").readFileSync(path);
|
|
|
} catch (e) {
|
|
|
return null;
|
|
|
}
|
|
|
} else {
|
|
|
var xhr = Util.XHR();
|
|
|
xhr.open('GET', path, callback ? true : false);
|
|
|
// xhr.setRequestHeader('User-Agent', 'XMLHTTP/1.0');
|
|
|
xhr.setRequestHeader('Accept', 'text/plain');
|
|
|
if (typeof xhr.overrideMimeType === 'function') xhr.overrideMimeType('text/plain');
|
|
|
if (callback) {
|
|
|
xhr.onreadystatechange = function() {
|
|
|
if (xhr.readyState != 4) return;
|
|
|
if (/* remote */ xhr.status == 200 || /* local */ (xhr.status == 0 && typeof xhr.responseText === 'string'))
|
|
|
callback(xhr.responseText);
|
|
|
else
|
|
|
callback(null);
|
|
|
};
|
|
|
if (xhr.readyState == 4)
|
|
|
return;
|
|
|
xhr.send(null);
|
|
|
} else {
|
|
|
xhr.send(null);
|
|
|
if (/* remote */ xhr.status == 200 || /* local */ (xhr.status == 0 && typeof xhr.responseText === 'string'))
|
|
|
return xhr.responseText;
|
|
|
return null;
|
|
|
}
|
|
|
}
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Tests if an object is an array.
|
|
|
* @function
|
|
|
* @param {*} obj Object to test
|
|
|
* @returns {boolean} true if it is an array, else false
|
|
|
* @expose
|
|
|
*/
|
|
|
Util.isArray = Array.isArray || function(obj) {
|
|
|
return Object.prototype.toString.call(obj) === "[object Array]";
|
|
|
};
|
|
|
|
|
|
return Util;
|
|
|
})();
|
|
|
|
|
|
/**
|
|
|
* Language expressions.
|
|
|
* @type {!Object.<string,string|!RegExp>}
|
|
|
* @expose
|
|
|
*/
|
|
|
ProtoBuf.Lang = {
|
|
|
OPEN: "{",
|
|
|
CLOSE: "}",
|
|
|
OPTOPEN: "[",
|
|
|
OPTCLOSE: "]",
|
|
|
OPTEND: ",",
|
|
|
EQUAL: "=",
|
|
|
END: ";",
|
|
|
STRINGOPEN: '"',
|
|
|
STRINGCLOSE: '"',
|
|
|
STRINGOPEN_SQ: "'",
|
|
|
STRINGCLOSE_SQ: "'",
|
|
|
COPTOPEN: '(',
|
|
|
COPTCLOSE: ')',
|
|
|
DELIM: /[\s\{\}=;\[\],'"\(\)]/g,
|
|
|
// KEYWORD: /^(?:package|option|import|message|enum|extend|service|syntax|extensions|group)$/,
|
|
|
RULE: /^(?:required|optional|repeated)$/,
|
|
|
TYPE: /^(?:double|float|int32|uint32|sint32|int64|uint64|sint64|fixed32|sfixed32|fixed64|sfixed64|bool|string|bytes)$/,
|
|
|
NAME: /^[a-zA-Z_][a-zA-Z_0-9]*$/,
|
|
|
TYPEDEF: /^[a-zA-Z][a-zA-Z_0-9]*$/,
|
|
|
TYPEREF: /^(?:\.?[a-zA-Z_][a-zA-Z_0-9]*)+$/,
|
|
|
FQTYPEREF: /^(?:\.[a-zA-Z][a-zA-Z_0-9]*)+$/,
|
|
|
NUMBER: /^-?(?:[1-9][0-9]*|0|0x[0-9a-fA-F]+|0[0-7]+|([0-9]*\.[0-9]+([Ee][+-]?[0-9]+)?))$/,
|
|
|
NUMBER_DEC: /^(?:[1-9][0-9]*|0)$/,
|
|
|
NUMBER_HEX: /^0x[0-9a-fA-F]+$/,
|
|
|
NUMBER_OCT: /^0[0-7]+$/,
|
|
|
NUMBER_FLT: /^[0-9]*\.[0-9]+([Ee][+-]?[0-9]+)?$/,
|
|
|
ID: /^(?:[1-9][0-9]*|0|0x[0-9a-fA-F]+|0[0-7]+)$/,
|
|
|
NEGID: /^\-?(?:[1-9][0-9]*|0|0x[0-9a-fA-F]+|0[0-7]+)$/,
|
|
|
WHITESPACE: /\s/,
|
|
|
STRING: /['"]([^'"\\]*(\\.[^"\\]*)*)['"]/g,
|
|
|
BOOL: /^(?:true|false)$/i
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* @alias ProtoBuf.DotProto
|
|
|
* @expose
|
|
|
*/
|
|
|
ProtoBuf.DotProto = (function(ProtoBuf, Lang) {
|
|
|
"use strict";
|
|
|
|
|
|
/**
|
|
|
* Utilities to parse .proto files.
|
|
|
* @exports ProtoBuf.DotProto
|
|
|
* @namespace
|
|
|
*/
|
|
|
var DotProto = {};
|
|
|
|
|
|
/**
|
|
|
* Constructs a new Tokenizer.
|
|
|
* @exports ProtoBuf.DotProto.Tokenizer
|
|
|
* @class prototype tokenizer
|
|
|
* @param {string} proto Proto to tokenize
|
|
|
* @constructor
|
|
|
*/
|
|
|
var Tokenizer = function(proto) {
|
|
|
|
|
|
/**
|
|
|
* Source to parse.
|
|
|
* @type {string}
|
|
|
* @expose
|
|
|
*/
|
|
|
this.source = ""+proto; // In case it's a buffer
|
|
|
|
|
|
/**
|
|
|
* Current index.
|
|
|
* @type {number}
|
|
|
* @expose
|
|
|
*/
|
|
|
this.index = 0;
|
|
|
|
|
|
/**
|
|
|
* Current line.
|
|
|
* @type {number}
|
|
|
* @expose
|
|
|
*/
|
|
|
this.line = 1;
|
|
|
|
|
|
/**
|
|
|
* Stacked values.
|
|
|
* @type {Array}
|
|
|
* @expose
|
|
|
*/
|
|
|
this.stack = [];
|
|
|
|
|
|
/**
|
|
|
* Whether currently reading a string or not.
|
|
|
* @type {boolean}
|
|
|
* @expose
|
|
|
*/
|
|
|
this.readingString = false;
|
|
|
|
|
|
/**
|
|
|
* Whatever character ends the string. Either a single or double quote character.
|
|
|
* @type {string}
|
|
|
* @expose
|
|
|
*/
|
|
|
this.stringEndsWith = Lang.STRINGCLOSE;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* @alias ProtoBuf.DotProto.Tokenizer.prototype
|
|
|
* @inner
|
|
|
*/
|
|
|
var TokenizerPrototype = Tokenizer.prototype;
|
|
|
|
|
|
/**
|
|
|
* Reads a string beginning at the current index.
|
|
|
* @return {string} The string
|
|
|
* @throws {Error} If it's not a valid string
|
|
|
* @private
|
|
|
*/
|
|
|
TokenizerPrototype._readString = function() {
|
|
|
Lang.STRING.lastIndex = this.index-1; // Include the open quote
|
|
|
var match;
|
|
|
if ((match = Lang.STRING.exec(this.source)) !== null) {
|
|
|
var s = match[1];
|
|
|
this.index = Lang.STRING.lastIndex;
|
|
|
this.stack.push(this.stringEndsWith);
|
|
|
return s;
|
|
|
}
|
|
|
throw Error("Unterminated string at line "+this.line+", index "+this.index);
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Gets the next token and advances by one.
|
|
|
* @return {?string} Token or `null` on EOF
|
|
|
* @throws {Error} If it's not a valid proto file
|
|
|
* @expose
|
|
|
*/
|
|
|
TokenizerPrototype.next = function() {
|
|
|
if (this.stack.length > 0)
|
|
|
return this.stack.shift();
|
|
|
if (this.index >= this.source.length)
|
|
|
return null; // No more tokens
|
|
|
if (this.readingString) {
|
|
|
this.readingString = false;
|
|
|
return this._readString();
|
|
|
}
|
|
|
var repeat, last;
|
|
|
do {
|
|
|
repeat = false;
|
|
|
// Strip white spaces
|
|
|
while (Lang.WHITESPACE.test(last = this.source.charAt(this.index))) {
|
|
|
this.index++;
|
|
|
if (last === "\n")
|
|
|
this.line++;
|
|
|
if (this.index === this.source.length)
|
|
|
return null;
|
|
|
}
|
|
|
// Strip comments
|
|
|
if (this.source.charAt(this.index) === '/') {
|
|
|
if (this.source.charAt(++this.index) === '/') { // Single line
|
|
|
while (this.source.charAt(this.index) !== "\n") {
|
|
|
this.index++;
|
|
|
if (this.index == this.source.length)
|
|
|
return null;
|
|
|
}
|
|
|
this.index++;
|
|
|
this.line++;
|
|
|
repeat = true;
|
|
|
} else if (this.source.charAt(this.index) === '*') { /* Block */
|
|
|
last = '';
|
|
|
while (last+(last=this.source.charAt(this.index)) !== '*/') {
|
|
|
this.index++;
|
|
|
if (last === "\n")
|
|
|
this.line++;
|
|
|
if (this.index === this.source.length)
|
|
|
return null;
|
|
|
}
|
|
|
this.index++;
|
|
|
repeat = true;
|
|
|
} else
|
|
|
throw Error("Unterminated comment at line "+this.line+": /"+this.source.charAt(this.index));
|
|
|
}
|
|
|
} while (repeat);
|
|
|
if (this.index === this.source.length) return null;
|
|
|
|
|
|
// Read the next token
|
|
|
var end = this.index;
|
|
|
Lang.DELIM.lastIndex = 0;
|
|
|
var delim = Lang.DELIM.test(this.source.charAt(end));
|
|
|
if (!delim) {
|
|
|
++end;
|
|
|
while(end < this.source.length && !Lang.DELIM.test(this.source.charAt(end)))
|
|
|
end++;
|
|
|
} else
|
|
|
++end;
|
|
|
var token = this.source.substring(this.index, this.index = end);
|
|
|
if (token === Lang.STRINGOPEN)
|
|
|
this.readingString = true,
|
|
|
this.stringEndsWith = Lang.STRINGCLOSE;
|
|
|
else if (token === Lang.STRINGOPEN_SQ)
|
|
|
this.readingString = true,
|
|
|
this.stringEndsWith = Lang.STRINGCLOSE_SQ;
|
|
|
return token;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Peeks for the next token.
|
|
|
* @return {?string} Token or `null` on EOF
|
|
|
* @throws {Error} If it's not a valid proto file
|
|
|
* @expose
|
|
|
*/
|
|
|
TokenizerPrototype.peek = function() {
|
|
|
if (this.stack.length === 0) {
|
|
|
var token = this.next();
|
|
|
if (token === null)
|
|
|
return null;
|
|
|
this.stack.push(token);
|
|
|
}
|
|
|
return this.stack[0];
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Returns a string representation of this object.
|
|
|
* @return {string} String representation as of "Tokenizer(index/length)"
|
|
|
* @expose
|
|
|
*/
|
|
|
TokenizerPrototype.toString = function() {
|
|
|
return "Tokenizer("+this.index+"/"+this.source.length+" at line "+this.line+")";
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* @alias ProtoBuf.DotProto.Tokenizer
|
|
|
* @expose
|
|
|
*/
|
|
|
DotProto.Tokenizer = Tokenizer;
|
|
|
|
|
|
/**
|
|
|
* Constructs a new Parser.
|
|
|
* @exports ProtoBuf.DotProto.Parser
|
|
|
* @class prototype parser
|
|
|
* @param {string} proto Protocol source
|
|
|
* @constructor
|
|
|
*/
|
|
|
var Parser = function(proto) {
|
|
|
|
|
|
/**
|
|
|
* Tokenizer.
|
|
|
* @type {ProtoBuf.DotProto.Tokenizer}
|
|
|
* @expose
|
|
|
*/
|
|
|
this.tn = new Tokenizer(proto);
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* @alias ProtoBuf.DotProto.Parser.prototype
|
|
|
* @inner
|
|
|
*/
|
|
|
var ParserPrototype = Parser.prototype;
|
|
|
|
|
|
/**
|
|
|
* Runs the parser.
|
|
|
* @return {{package: string|null, messages: Array.<object>, enums: Array.<object>, imports: Array.<string>, options: object<string,*>}}
|
|
|
* @throws {Error} If the source cannot be parsed
|
|
|
* @expose
|
|
|
*/
|
|
|
ParserPrototype.parse = function() {
|
|
|
var topLevel = {
|
|
|
"name": "[ROOT]", // temporary
|
|
|
"package": null,
|
|
|
"messages": [],
|
|
|
"enums": [],
|
|
|
"imports": [],
|
|
|
"options": {},
|
|
|
"services": []
|
|
|
};
|
|
|
var token, head = true;
|
|
|
while(token = this.tn.next()) {
|
|
|
switch (token) {
|
|
|
case 'package':
|
|
|
if (!head || topLevel["package"] !== null)
|
|
|
throw Error("Unexpected package at line "+this.tn.line);
|
|
|
topLevel["package"] = this._parsePackage(token);
|
|
|
break;
|
|
|
case 'import':
|
|
|
if (!head)
|
|
|
throw Error("Unexpected import at line "+this.tn.line);
|
|
|
topLevel.imports.push(this._parseImport(token));
|
|
|
break;
|
|
|
case 'message':
|
|
|
this._parseMessage(topLevel, null, token);
|
|
|
head = false;
|
|
|
break;
|
|
|
case 'enum':
|
|
|
this._parseEnum(topLevel, token);
|
|
|
head = false;
|
|
|
break;
|
|
|
case 'option':
|
|
|
if (!head)
|
|
|
throw Error("Unexpected option at line "+this.tn.line);
|
|
|
this._parseOption(topLevel, token);
|
|
|
break;
|
|
|
case 'service':
|
|
|
this._parseService(topLevel, token);
|
|
|
break;
|
|
|
case 'extend':
|
|
|
this._parseExtend(topLevel, token);
|
|
|
break;
|
|
|
case 'syntax':
|
|
|
this._parseIgnoredStatement(topLevel, token);
|
|
|
break;
|
|
|
default:
|
|
|
throw Error("Unexpected token at line "+this.tn.line+": "+token);
|
|
|
}
|
|
|
}
|
|
|
delete topLevel["name"];
|
|
|
return topLevel;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Parses a number value.
|
|
|
* @param {string} val Number value to parse
|
|
|
* @return {number} Number
|
|
|
* @throws {Error} If the number value is invalid
|
|
|
* @private
|
|
|
*/
|
|
|
ParserPrototype._parseNumber = function(val) {
|
|
|
var sign = 1;
|
|
|
if (val.charAt(0) == '-')
|
|
|
sign = -1,
|
|
|
val = val.substring(1);
|
|
|
if (Lang.NUMBER_DEC.test(val))
|
|
|
return sign*parseInt(val, 10);
|
|
|
else if (Lang.NUMBER_HEX.test(val))
|
|
|
return sign*parseInt(val.substring(2), 16);
|
|
|
else if (Lang.NUMBER_OCT.test(val))
|
|
|
return sign*parseInt(val.substring(1), 8);
|
|
|
else if (Lang.NUMBER_FLT.test(val))
|
|
|
return sign*parseFloat(val);
|
|
|
throw Error("Illegal number at line "+this.tn.line+": "+(sign < 0 ? '-' : '')+val);
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Parses a (possibly multiline) string.
|
|
|
* @returns {string}
|
|
|
* @private
|
|
|
*/
|
|
|
ParserPrototype._parseString = function() {
|
|
|
var value = "", token;
|
|
|
do {
|
|
|
token = this.tn.next(); // Known to be = this.tn.stringEndsWith
|
|
|
value += this.tn.next();
|
|
|
token = this.tn.next();
|
|
|
if (token !== this.tn.stringEndsWith)
|
|
|
throw Error("Illegal end of string at line "+this.tn.line+": "+token);
|
|
|
token = this.tn.peek();
|
|
|
} while (token === Lang.STRINGOPEN || token === Lang.STRINGOPEN_SQ);
|
|
|
return value;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Parses an ID value.
|
|
|
* @param {string} val ID value to parse
|
|
|
* @param {boolean=} neg Whether the ID may be negative, defaults to `false`
|
|
|
* @returns {number} ID
|
|
|
* @throws {Error} If the ID value is invalid
|
|
|
* @private
|
|
|
*/
|
|
|
ParserPrototype._parseId = function(val, neg) {
|
|
|
var id = -1;
|
|
|
var sign = 1;
|
|
|
if (val.charAt(0) == '-')
|
|
|
sign = -1,
|
|
|
val = val.substring(1);
|
|
|
if (Lang.NUMBER_DEC.test(val))
|
|
|
id = parseInt(val);
|
|
|
else if (Lang.NUMBER_HEX.test(val))
|
|
|
id = parseInt(val.substring(2), 16);
|
|
|
else if (Lang.NUMBER_OCT.test(val))
|
|
|
id = parseInt(val.substring(1), 8);
|
|
|
else
|
|
|
throw Error("Illegal id at line "+this.tn.line+": "+(sign < 0 ? '-' : '')+val);
|
|
|
id = (sign*id)|0; // Force to 32bit
|
|
|
if (!neg && id < 0)
|
|
|
throw Error("Illegal id at line "+this.tn.line+": "+(sign < 0 ? '-' : '')+val);
|
|
|
return id;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Parses the package definition.
|
|
|
* @param {string} token Initial token
|
|
|
* @return {string} Package name
|
|
|
* @throws {Error} If the package definition cannot be parsed
|
|
|
* @private
|
|
|
*/
|
|
|
ParserPrototype._parsePackage = function(token) {
|
|
|
token = this.tn.next();
|
|
|
if (!Lang.TYPEREF.test(token))
|
|
|
throw Error("Illegal package name at line "+this.tn.line+": "+token);
|
|
|
var pkg = token;
|
|
|
token = this.tn.next();
|
|
|
if (token != Lang.END)
|
|
|
throw Error("Illegal end of package at line "+this.tn.line+": "+token);
|
|
|
return pkg;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Parses an import definition.
|
|
|
* @param {string} token Initial token
|
|
|
* @return {string} Import file name
|
|
|
* @throws {Error} If the import definition cannot be parsed
|
|
|
* @private
|
|
|
*/
|
|
|
ParserPrototype._parseImport = function(token) {
|
|
|
token = this.tn.peek();
|
|
|
if (token === "public")
|
|
|
this.tn.next(),
|
|
|
token = this.tn.peek();
|
|
|
if (token !== Lang.STRINGOPEN && token !== Lang.STRINGOPEN_SQ)
|
|
|
throw Error("Illegal start of import at line "+this.tn.line+": "+token);
|
|
|
var imported = this._parseString();
|
|
|
token = this.tn.next();
|
|
|
if (token !== Lang.END)
|
|
|
throw Error("Illegal end of import at line "+this.tn.line+": "+token);
|
|
|
return imported;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Parses a namespace option.
|
|
|
* @param {Object} parent Parent definition
|
|
|
* @param {string} token Initial token
|
|
|
* @throws {Error} If the option cannot be parsed
|
|
|
* @private
|
|
|
*/
|
|
|
ParserPrototype._parseOption = function(parent, token) {
|
|
|
token = this.tn.next();
|
|
|
var custom = false;
|
|
|
if (token == Lang.COPTOPEN)
|
|
|
custom = true,
|
|
|
token = this.tn.next();
|
|
|
if (!Lang.TYPEREF.test(token))
|
|
|
// we can allow options of the form google.protobuf.* since they will just get ignored anyways
|
|
|
if (!/google\.protobuf\./.test(token))
|
|
|
throw Error("Illegal option name in message "+parent.name+" at line "+this.tn.line+": "+token);
|
|
|
var name = token;
|
|
|
token = this.tn.next();
|
|
|
if (custom) { // (my_method_option).foo, (my_method_option), some_method_option, (foo.my_option).bar
|
|
|
if (token !== Lang.COPTCLOSE)
|
|
|
throw Error("Illegal end in message "+parent.name+", option "+name+" at line "+this.tn.line+": "+token);
|
|
|
name = '('+name+')';
|
|
|
token = this.tn.next();
|
|
|
if (Lang.FQTYPEREF.test(token))
|
|
|
name += token,
|
|
|
token = this.tn.next();
|
|
|
}
|
|
|
if (token !== Lang.EQUAL)
|
|
|
throw Error("Illegal operator in message "+parent.name+", option "+name+" at line "+this.tn.line+": "+token);
|
|
|
var value;
|
|
|
token = this.tn.peek();
|
|
|
if (token === Lang.STRINGOPEN || token === Lang.STRINGOPEN_SQ)
|
|
|
value = this._parseString();
|
|
|
else {
|
|
|
this.tn.next();
|
|
|
if (Lang.NUMBER.test(token))
|
|
|
value = this._parseNumber(token, true);
|
|
|
else if (Lang.BOOL.test(token))
|
|
|
value = token === 'true';
|
|
|
else if (Lang.TYPEREF.test(token))
|
|
|
value = token;
|
|
|
else
|
|
|
throw Error("Illegal option value in message "+parent.name+", option "+name+" at line "+this.tn.line+": "+token);
|
|
|
}
|
|
|
token = this.tn.next();
|
|
|
if (token !== Lang.END)
|
|
|
throw Error("Illegal end of option in message "+parent.name+", option "+name+" at line "+this.tn.line+": "+token);
|
|
|
parent["options"][name] = value;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Parses an ignored statement of the form ['keyword', ..., ';'].
|
|
|
* @param {Object} parent Parent definition
|
|
|
* @param {string} keyword Initial token
|
|
|
* @throws {Error} If the directive cannot be parsed
|
|
|
* @private
|
|
|
*/
|
|
|
ParserPrototype._parseIgnoredStatement = function(parent, keyword) {
|
|
|
var token;
|
|
|
do {
|
|
|
token = this.tn.next();
|
|
|
if (token === null)
|
|
|
throw Error("Unexpected EOF in "+parent.name+", "+keyword+" at line "+this.tn.line);
|
|
|
if (token === Lang.END)
|
|
|
break;
|
|
|
} while (true);
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Parses a service definition.
|
|
|
* @param {Object} parent Parent definition
|
|
|
* @param {string} token Initial token
|
|
|
* @throws {Error} If the service cannot be parsed
|
|
|
* @private
|
|
|
*/
|
|
|
ParserPrototype._parseService = function(parent, token) {
|
|
|
token = this.tn.next();
|
|
|
if (!Lang.NAME.test(token))
|
|
|
throw Error("Illegal service name at line "+this.tn.line+": "+token);
|
|
|
var name = token;
|
|
|
var svc = {
|
|
|
"name": name,
|
|
|
"rpc": {},
|
|
|
"options": {}
|
|
|
};
|
|
|
token = this.tn.next();
|
|
|
if (token !== Lang.OPEN)
|
|
|
throw Error("Illegal start of service "+name+" at line "+this.tn.line+": "+token);
|
|
|
do {
|
|
|
token = this.tn.next();
|
|
|
if (token === "option")
|
|
|
this._parseOption(svc, token);
|
|
|
else if (token === 'rpc')
|
|
|
this._parseServiceRPC(svc, token);
|
|
|
else if (token !== Lang.CLOSE)
|
|
|
throw Error("Illegal type of service "+name+" at line "+this.tn.line+": "+token);
|
|
|
} while (token !== Lang.CLOSE);
|
|
|
parent["services"].push(svc);
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Parses a RPC service definition of the form ['rpc', name, (request), 'returns', (response)].
|
|
|
* @param {Object} svc Parent definition
|
|
|
* @param {string} token Initial token
|
|
|
* @private
|
|
|
*/
|
|
|
ParserPrototype._parseServiceRPC = function(svc, token) {
|
|
|
var type = token;
|
|
|
token = this.tn.next();
|
|
|
if (!Lang.NAME.test(token))
|
|
|
throw Error("Illegal method name in service "+svc["name"]+" at line "+this.tn.line+": "+token);
|
|
|
var name = token;
|
|
|
var method = {
|
|
|
"request": null,
|
|
|
"response": null,
|
|
|
"options": {}
|
|
|
};
|
|
|
token = this.tn.next();
|
|
|
if (token !== Lang.COPTOPEN)
|
|
|
throw Error("Illegal start of request type in service "+svc["name"]+"#"+name+" at line "+this.tn.line+": "+token);
|
|
|
token = this.tn.next();
|
|
|
if (!Lang.TYPEREF.test(token))
|
|
|
throw Error("Illegal request type in service "+svc["name"]+"#"+name+" at line "+this.tn.line+": "+token);
|
|
|
method["request"] = token;
|
|
|
token = this.tn.next();
|
|
|
if (token != Lang.COPTCLOSE)
|
|
|
throw Error("Illegal end of request type in service "+svc["name"]+"#"+name+" at line "+this.tn.line+": "+token);
|
|
|
token = this.tn.next();
|
|
|
if (token.toLowerCase() !== "returns")
|
|
|
throw Error("Illegal delimiter in service "+svc["name"]+"#"+name+" at line "+this.tn.line+": "+token);
|
|
|
token = this.tn.next();
|
|
|
if (token != Lang.COPTOPEN)
|
|
|
throw Error("Illegal start of response type in service "+svc["name"]+"#"+name+" at line "+this.tn.line+": "+token);
|
|
|
token = this.tn.next();
|
|
|
method["response"] = token;
|
|
|
token = this.tn.next();
|
|
|
if (token !== Lang.COPTCLOSE)
|
|
|
throw Error("Illegal end of response type in service "+svc["name"]+"#"+name+" at line "+this.tn.line+": "+token);
|
|
|
token = this.tn.next();
|
|
|
if (token === Lang.OPEN) {
|
|
|
do {
|
|
|
token = this.tn.next();
|
|
|
if (token === 'option')
|
|
|
this._parseOption(method, token); // <- will fail for the custom-options example
|
|
|
else if (token !== Lang.CLOSE)
|
|
|
throw Error("Illegal start of option inservice "+svc["name"]+"#"+name+" at line "+this.tn.line+": "+token);
|
|
|
} while (token !== Lang.CLOSE);
|
|
|
if (this.tn.peek() === Lang.END)
|
|
|
this.tn.next();
|
|
|
} else if (token !== Lang.END)
|
|
|
throw Error("Illegal delimiter in service "+svc["name"]+"#"+name+" at line "+this.tn.line+": "+token);
|
|
|
if (typeof svc[type] === 'undefined')
|
|
|
svc[type] = {};
|
|
|
svc[type][name] = method;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Parses a message definition.
|
|
|
* @param {Object} parent Parent definition
|
|
|
* @param {Object} fld Field definition if this is a group, otherwise `null`
|
|
|
* @param {string} token First token
|
|
|
* @return {Object}
|
|
|
* @throws {Error} If the message cannot be parsed
|
|
|
* @private
|
|
|
*/
|
|
|
ParserPrototype._parseMessage = function(parent, fld, token) {
|
|
|
/** @dict */
|
|
|
var msg = {}; // Note: At some point we might want to exclude the parser, so we need a dict.
|
|
|
var isGroup = token === "group";
|
|
|
token = this.tn.next();
|
|
|
if (!Lang.NAME.test(token))
|
|
|
throw Error("Illegal "+(isGroup ? "group" : "message")+" name"+(parent ? " in message "+parent["name"] : "")+" at line "+this.tn.line+": "+token);
|
|
|
msg["name"] = token;
|
|
|
if (isGroup) {
|
|
|
token = this.tn.next();
|
|
|
if (token !== Lang.EQUAL)
|
|
|
throw Error("Illegal id assignment after group "+msg.name+" at line "+this.tn.line+": "+token);
|
|
|
token = this.tn.next();
|
|
|
try {
|
|
|
fld["id"] = this._parseId(token);
|
|
|
} catch (e) {
|
|
|
throw Error("Illegal field id value for group "+msg.name+"#"+fld.name+" at line "+this.tn.line+": "+token);
|
|
|
}
|
|
|
msg["isGroup"] = true;
|
|
|
}
|
|
|
msg["fields"] = []; // Note: Using arrays to support also browser that cannot preserve order of object keys.
|
|
|
msg["enums"] = [];
|
|
|
msg["messages"] = [];
|
|
|
msg["options"] = {};
|
|
|
msg["oneofs"] = {};
|
|
|
token = this.tn.next();
|
|
|
if (token === Lang.OPTOPEN && fld)
|
|
|
this._parseFieldOptions(msg, fld, token),
|
|
|
token = this.tn.next();
|
|
|
if (token !== Lang.OPEN)
|
|
|
throw Error("Illegal start of "+(isGroup ? "group" : "message")+" "+msg.name+" at line "+this.tn.line+": "+token);
|
|
|
// msg["extensions"] = undefined
|
|
|
do {
|
|
|
token = this.tn.next();
|
|
|
if (token === Lang.CLOSE) {
|
|
|
token = this.tn.peek();
|
|
|
if (token === Lang.END)
|
|
|
this.tn.next();
|
|
|
break;
|
|
|
} else if (Lang.RULE.test(token))
|
|
|
this._parseMessageField(msg, token);
|
|
|
else if (token === "oneof")
|
|
|
this._parseMessageOneOf(msg, token);
|
|
|
else if (token === "enum")
|
|
|
this._parseEnum(msg, token);
|
|
|
else if (token === "message")
|
|
|
this._parseMessage(msg, null, token);
|
|
|
else if (token === "option")
|
|
|
this._parseOption(msg, token);
|
|
|
else if (token === "extensions")
|
|
|
msg["extensions"] = this._parseExtensions(msg, token);
|
|
|
else if (token === "extend")
|
|
|
this._parseExtend(msg, token);
|
|
|
else
|
|
|
throw Error("Illegal token in message "+msg.name+" at line "+this.tn.line+": "+token);
|
|
|
} while (true);
|
|
|
parent["messages"].push(msg);
|
|
|
return msg;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Parses a message field.
|
|
|
* @param {Object} msg Message definition
|
|
|
* @param {string} token Initial token
|
|
|
* @returns {!Object} Field descriptor
|
|
|
* @throws {Error} If the message field cannot be parsed
|
|
|
* @private
|
|
|
*/
|
|
|
ParserPrototype._parseMessageField = function(msg, token) {
|
|
|
/** @dict */
|
|
|
var fld = {}, grp = null;
|
|
|
fld["rule"] = token;
|
|
|
/** @dict */
|
|
|
fld["options"] = {};
|
|
|
token = this.tn.next();
|
|
|
if (token === "group") {
|
|
|
// "A [legacy] group simply combines a nested message type and a field into a single declaration. In your
|
|
|
// code, you can treat this message just as if it had a Result type field called result (the latter name is
|
|
|
// converted to lower-case so that it does not conflict with the former)."
|
|
|
grp = this._parseMessage(msg, fld, token);
|
|
|
if (!/^[A-Z]/.test(grp["name"]))
|
|
|
throw Error('Group names must start with a capital letter');
|
|
|
fld["type"] = grp["name"];
|
|
|
fld["name"] = grp["name"].toLowerCase();
|
|
|
token = this.tn.peek();
|
|
|
if (token === Lang.END)
|
|
|
this.tn.next();
|
|
|
} else {
|
|
|
if (!Lang.TYPE.test(token) && !Lang.TYPEREF.test(token))
|
|
|
throw Error("Illegal field type in message "+msg.name+" at line "+this.tn.line+": "+token);
|
|
|
fld["type"] = token;
|
|
|
token = this.tn.next();
|
|
|
if (!Lang.NAME.test(token))
|
|
|
throw Error("Illegal field name in message "+msg.name+" at line "+this.tn.line+": "+token);
|
|
|
fld["name"] = token;
|
|
|
token = this.tn.next();
|
|
|
if (token !== Lang.EQUAL)
|
|
|
throw Error("Illegal token in field "+msg.name+"#"+fld.name+" at line "+this.tn.line+": "+token);
|
|
|
token = this.tn.next();
|
|
|
try {
|
|
|
fld["id"] = this._parseId(token);
|
|
|
} catch (e) {
|
|
|
throw Error("Illegal field id in message "+msg.name+"#"+fld.name+" at line "+this.tn.line+": "+token);
|
|
|
}
|
|
|
token = this.tn.next();
|
|
|
if (token === Lang.OPTOPEN)
|
|
|
this._parseFieldOptions(msg, fld, token),
|
|
|
token = this.tn.next();
|
|
|
if (token !== Lang.END)
|
|
|
throw Error("Illegal delimiter in message "+msg.name+"#"+fld.name+" at line "+this.tn.line+": "+token);
|
|
|
}
|
|
|
msg["fields"].push(fld);
|
|
|
return fld;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Parses a message oneof.
|
|
|
* @param {Object} msg Message definition
|
|
|
* @param {string} token Initial token
|
|
|
* @throws {Error} If the message oneof cannot be parsed
|
|
|
* @private
|
|
|
*/
|
|
|
ParserPrototype._parseMessageOneOf = function(msg, token) {
|
|
|
token = this.tn.next();
|
|
|
if (!Lang.NAME.test(token))
|
|
|
throw Error("Illegal oneof name in message "+msg.name+" at line "+this.tn.line+": "+token);
|
|
|
var name = token,
|
|
|
fld;
|
|
|
var fields = [];
|
|
|
token = this.tn.next();
|
|
|
if (token !== Lang.OPEN)
|
|
|
throw Error("Illegal start of oneof "+name+" at line "+this.tn.line+": "+token);
|
|
|
while (this.tn.peek() !== Lang.CLOSE) {
|
|
|
fld = this._parseMessageField(msg, "optional");
|
|
|
fld["oneof"] = name;
|
|
|
fields.push(fld["id"]);
|
|
|
}
|
|
|
this.tn.next();
|
|
|
msg["oneofs"][name] = fields;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Parses a set of field option definitions.
|
|
|
* @param {Object} msg Message definition
|
|
|
* @param {Object} fld Field definition
|
|
|
* @param {string} token Initial token
|
|
|
* @throws {Error} If the message field options cannot be parsed
|
|
|
* @private
|
|
|
*/
|
|
|
ParserPrototype._parseFieldOptions = function(msg, fld, token) {
|
|
|
var first = true;
|
|
|
do {
|
|
|
token = this.tn.next();
|
|
|
if (token === Lang.OPTCLOSE)
|
|
|
break;
|
|
|
else if (token === Lang.OPTEND) {
|
|
|
if (first)
|
|
|
throw Error("Illegal start of options in message "+msg.name+"#"+fld.name+" at line "+this.tn.line+": "+token);
|
|
|
token = this.tn.next();
|
|
|
}
|
|
|
this._parseFieldOption(msg, fld, token);
|
|
|
first = false;
|
|
|
} while (true);
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Parses a single field option.
|
|
|
* @param {Object} msg Message definition
|
|
|
* @param {Object} fld Field definition
|
|
|
* @param {string} token Initial token
|
|
|
* @throws {Error} If the mesage field option cannot be parsed
|
|
|
* @private
|
|
|
*/
|
|
|
ParserPrototype._parseFieldOption = function(msg, fld, token) {
|
|
|
var custom = false;
|
|
|
if (token === Lang.COPTOPEN)
|
|
|
token = this.tn.next(),
|
|
|
custom = true;
|
|
|
if (!Lang.TYPEREF.test(token))
|
|
|
throw Error("Illegal field option in "+msg.name+"#"+fld.name+" at line "+this.tn.line+": "+token);
|
|
|
var name = token;
|
|
|
token = this.tn.next();
|
|
|
if (custom) {
|
|
|
if (token !== Lang.COPTCLOSE)
|
|
|
throw Error("Illegal delimiter in "+msg.name+"#"+fld.name+" at line "+this.tn.line+": "+token);
|
|
|
name = '('+name+')';
|
|
|
token = this.tn.next();
|
|
|
if (Lang.FQTYPEREF.test(token))
|
|
|
name += token,
|
|
|
token = this.tn.next();
|
|
|
}
|
|
|
if (token !== Lang.EQUAL)
|
|
|
throw Error("Illegal token in "+msg.name+"#"+fld.name+" at line "+this.tn.line+": "+token);
|
|
|
var value;
|
|
|
token = this.tn.peek();
|
|
|
if (token === Lang.STRINGOPEN || token === Lang.STRINGOPEN_SQ) {
|
|
|
value = this._parseString();
|
|
|
} else if (Lang.NUMBER.test(token, true))
|
|
|
value = this._parseNumber(this.tn.next(), true);
|
|
|
else if (Lang.BOOL.test(token))
|
|
|
value = this.tn.next().toLowerCase() === 'true';
|
|
|
else if (Lang.TYPEREF.test(token))
|
|
|
value = this.tn.next(); // TODO: Resolve?
|
|
|
else
|
|
|
throw Error("Illegal value in message "+msg.name+"#"+fld.name+", option "+name+" at line "+this.tn.line+": "+token);
|
|
|
fld["options"][name] = value;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Parses an enum.
|
|
|
* @param {Object} msg Message definition
|
|
|
* @param {string} token Initial token
|
|
|
* @throws {Error} If the enum cannot be parsed
|
|
|
* @private
|
|
|
*/
|
|
|
ParserPrototype._parseEnum = function(msg, token) {
|
|
|
/** @dict */
|
|
|
var enm = {};
|
|
|
token = this.tn.next();
|
|
|
if (!Lang.NAME.test(token))
|
|
|
throw Error("Illegal enum name in message "+msg.name+" at line "+this.tn.line+": "+token);
|
|
|
enm["name"] = token;
|
|
|
token = this.tn.next();
|
|
|
if (token !== Lang.OPEN)
|
|
|
throw Error("Illegal start of enum "+enm.name+" at line "+this.tn.line+": "+token);
|
|
|
enm["values"] = [];
|
|
|
enm["options"] = {};
|
|
|
do {
|
|
|
token = this.tn.next();
|
|
|
if (token === Lang.CLOSE) {
|
|
|
token = this.tn.peek();
|
|
|
if (token === Lang.END)
|
|
|
this.tn.next();
|
|
|
break;
|
|
|
}
|
|
|
if (token == 'option')
|
|
|
this._parseOption(enm, token);
|
|
|
else {
|
|
|
if (!Lang.NAME.test(token))
|
|
|
throw Error("Illegal name in enum "+enm.name+" at line "+this.tn.line+": "+token);
|
|
|
this._parseEnumValue(enm, token);
|
|
|
}
|
|
|
} while (true);
|
|
|
msg["enums"].push(enm);
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Parses an enum value.
|
|
|
* @param {Object} enm Enum definition
|
|
|
* @param {string} token Initial token
|
|
|
* @throws {Error} If the enum value cannot be parsed
|
|
|
* @private
|
|
|
*/
|
|
|
ParserPrototype._parseEnumValue = function(enm, token) {
|
|
|
/** @dict */
|
|
|
var val = {};
|
|
|
val["name"] = token;
|
|
|
token = this.tn.next();
|
|
|
if (token !== Lang.EQUAL)
|
|
|
throw Error("Illegal token in enum "+enm.name+" at line "+this.tn.line+": "+token);
|
|
|
token = this.tn.next();
|
|
|
try {
|
|
|
val["id"] = this._parseId(token, true);
|
|
|
} catch (e) {
|
|
|
throw Error("Illegal id in enum "+enm.name+" at line "+this.tn.line+": "+token);
|
|
|
}
|
|
|
enm["values"].push(val);
|
|
|
token = this.tn.next();
|
|
|
if (token === Lang.OPTOPEN) {
|
|
|
var opt = { 'options' : {} }; // TODO: Actually expose them somehow.
|
|
|
this._parseFieldOptions(enm, opt, token);
|
|
|
token = this.tn.next();
|
|
|
}
|
|
|
if (token !== Lang.END)
|
|
|
throw Error("Illegal delimiter in enum "+enm.name+" at line "+this.tn.line+": "+token);
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Parses an extensions statement.
|
|
|
* @param {Object} msg Message object
|
|
|
* @param {string} token Initial token
|
|
|
* @throws {Error} If the extensions statement cannot be parsed
|
|
|
* @private
|
|
|
*/
|
|
|
ParserPrototype._parseExtensions = function(msg, token) {
|
|
|
/** @type {Array.<number>} */
|
|
|
var range = [];
|
|
|
token = this.tn.next();
|
|
|
if (token === "min") // FIXME: Does the official implementation support this?
|
|
|
range.push(ProtoBuf.ID_MIN);
|
|
|
else if (token === "max")
|
|
|
range.push(ProtoBuf.ID_MAX);
|
|
|
else
|
|
|
range.push(this._parseNumber(token));
|
|
|
token = this.tn.next();
|
|
|
if (token !== 'to')
|
|
|
throw Error("Illegal extensions delimiter in message "+msg.name+" at line "+this.tn.line+": "+token);
|
|
|
token = this.tn.next();
|
|
|
if (token === "min")
|
|
|
range.push(ProtoBuf.ID_MIN);
|
|
|
else if (token === "max")
|
|
|
range.push(ProtoBuf.ID_MAX);
|
|
|
else
|
|
|
range.push(this._parseNumber(token));
|
|
|
token = this.tn.next();
|
|
|
if (token !== Lang.END)
|
|
|
throw Error("Illegal extensions delimiter in message "+msg.name+" at line "+this.tn.line+": "+token);
|
|
|
return range;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Parses an extend block.
|
|
|
* @param {Object} parent Parent object
|
|
|
* @param {string} token Initial token
|
|
|
* @throws {Error} If the extend block cannot be parsed
|
|
|
* @private
|
|
|
*/
|
|
|
ParserPrototype._parseExtend = function(parent, token) {
|
|
|
token = this.tn.next();
|
|
|
if (!Lang.TYPEREF.test(token))
|
|
|
throw Error("Illegal message name at line "+this.tn.line+": "+token);
|
|
|
/** @dict */
|
|
|
var ext = {};
|
|
|
ext["ref"] = token;
|
|
|
ext["fields"] = [];
|
|
|
token = this.tn.next();
|
|
|
if (token !== Lang.OPEN)
|
|
|
throw Error("Illegal start of extend "+ext.name+" at line "+this.tn.line+": "+token);
|
|
|
do {
|
|
|
token = this.tn.next();
|
|
|
if (token === Lang.CLOSE) {
|
|
|
token = this.tn.peek();
|
|
|
if (token == Lang.END)
|
|
|
this.tn.next();
|
|
|
break;
|
|
|
} else if (Lang.RULE.test(token))
|
|
|
this._parseMessageField(ext, token);
|
|
|
else
|
|
|
throw Error("Illegal token in extend "+ext.name+" at line "+this.tn.line+": "+token);
|
|
|
} while (true);
|
|
|
parent["messages"].push(ext);
|
|
|
return ext;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Returns a string representation of this object.
|
|
|
* @returns {string} String representation as of "Parser"
|
|
|
*/
|
|
|
ParserPrototype.toString = function() {
|
|
|
return "Parser";
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* @alias ProtoBuf.DotProto.Parser
|
|
|
* @expose
|
|
|
*/
|
|
|
DotProto.Parser = Parser;
|
|
|
|
|
|
return DotProto;
|
|
|
|
|
|
})(ProtoBuf, ProtoBuf.Lang);
|
|
|
|
|
|
/**
|
|
|
* @alias ProtoBuf.Reflect
|
|
|
* @expose
|
|
|
*/
|
|
|
ProtoBuf.Reflect = (function(ProtoBuf) {
|
|
|
"use strict";
|
|
|
|
|
|
/**
|
|
|
* Reflection types.
|
|
|
* @exports ProtoBuf.Reflect
|
|
|
* @namespace
|
|
|
*/
|
|
|
var Reflect = {};
|
|
|
|
|
|
/**
|
|
|
* Constructs a Reflect base class.
|
|
|
* @exports ProtoBuf.Reflect.T
|
|
|
* @constructor
|
|
|
* @abstract
|
|
|
* @param {!ProtoBuf.Builder} builder Builder reference
|
|
|
* @param {?ProtoBuf.Reflect.T} parent Parent object
|
|
|
* @param {string} name Object name
|
|
|
*/
|
|
|
var T = function(builder, parent, name) {
|
|
|
|
|
|
/**
|
|
|
* Builder reference.
|
|
|
* @type {!ProtoBuf.Builder}
|
|
|
* @expose
|
|
|
*/
|
|
|
this.builder = builder;
|
|
|
|
|
|
/**
|
|
|
* Parent object.
|
|
|
* @type {?ProtoBuf.Reflect.T}
|
|
|
* @expose
|
|
|
*/
|
|
|
this.parent = parent;
|
|
|
|
|
|
/**
|
|
|
* Object name in namespace.
|
|
|
* @type {string}
|
|
|
* @expose
|
|
|
*/
|
|
|
this.name = name;
|
|
|
|
|
|
/**
|
|
|
* Fully qualified class name
|
|
|
* @type {string}
|
|
|
* @expose
|
|
|
*/
|
|
|
this.className;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* @alias ProtoBuf.Reflect.T.prototype
|
|
|
* @inner
|
|
|
*/
|
|
|
var TPrototype = T.prototype;
|
|
|
|
|
|
/**
|
|
|
* Returns the fully qualified name of this object.
|
|
|
* @returns {string} Fully qualified name as of ".PATH.TO.THIS"
|
|
|
* @expose
|
|
|
*/
|
|
|
TPrototype.fqn = function() {
|
|
|
var name = this.name,
|
|
|
ptr = this;
|
|
|
do {
|
|
|
ptr = ptr.parent;
|
|
|
if (ptr == null)
|
|
|
break;
|
|
|
name = ptr.name+"."+name;
|
|
|
} while (true);
|
|
|
return name;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Returns a string representation of this Reflect object (its fully qualified name).
|
|
|
* @param {boolean=} includeClass Set to true to include the class name. Defaults to false.
|
|
|
* @return String representation
|
|
|
* @expose
|
|
|
*/
|
|
|
TPrototype.toString = function(includeClass) {
|
|
|
return (includeClass ? this.className + " " : "") + this.fqn();
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Builds this type.
|
|
|
* @throws {Error} If this type cannot be built directly
|
|
|
* @expose
|
|
|
*/
|
|
|
TPrototype.build = function() {
|
|
|
throw Error(this.toString(true)+" cannot be built directly");
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* @alias ProtoBuf.Reflect.T
|
|
|
* @expose
|
|
|
*/
|
|
|
Reflect.T = T;
|
|
|
|
|
|
/**
|
|
|
* Constructs a new Namespace.
|
|
|
* @exports ProtoBuf.Reflect.Namespace
|
|
|
* @param {!ProtoBuf.Builder} builder Builder reference
|
|
|
* @param {?ProtoBuf.Reflect.Namespace} parent Namespace parent
|
|
|
* @param {string} name Namespace name
|
|
|
* @param {Object.<string,*>=} options Namespace options
|
|
|
* @constructor
|
|
|
* @extends ProtoBuf.Reflect.T
|
|
|
*/
|
|
|
var Namespace = function(builder, parent, name, options) {
|
|
|
T.call(this, builder, parent, name);
|
|
|
|
|
|
/**
|
|
|
* @override
|
|
|
*/
|
|
|
this.className = "Namespace";
|
|
|
|
|
|
/**
|
|
|
* Children inside the namespace.
|
|
|
* @type {!Array.<ProtoBuf.Reflect.T>}
|
|
|
*/
|
|
|
this.children = [];
|
|
|
|
|
|
/**
|
|
|
* Options.
|
|
|
* @type {!Object.<string, *>}
|
|
|
*/
|
|
|
this.options = options || {};
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* @alias ProtoBuf.Reflect.Namespace.prototype
|
|
|
* @inner
|
|
|
*/
|
|
|
var NamespacePrototype = Namespace.prototype = Object.create(T.prototype);
|
|
|
|
|
|
/**
|
|
|
* Returns an array of the namespace's children.
|
|
|
* @param {ProtoBuf.Reflect.T=} type Filter type (returns instances of this type only). Defaults to null (all children).
|
|
|
* @return {Array.<ProtoBuf.Reflect.T>}
|
|
|
* @expose
|
|
|
*/
|
|
|
NamespacePrototype.getChildren = function(type) {
|
|
|
type = type || null;
|
|
|
if (type == null)
|
|
|
return this.children.slice();
|
|
|
var children = [];
|
|
|
for (var i=0, k=this.children.length; i<k; ++i)
|
|
|
if (this.children[i] instanceof type)
|
|
|
children.push(this.children[i]);
|
|
|
return children;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Adds a child to the namespace.
|
|
|
* @param {ProtoBuf.Reflect.T} child Child
|
|
|
* @throws {Error} If the child cannot be added (duplicate)
|
|
|
* @expose
|
|
|
*/
|
|
|
NamespacePrototype.addChild = function(child) {
|
|
|
var other;
|
|
|
if (other = this.getChild(child.name)) {
|
|
|
// Try to revert camelcase transformation on collision
|
|
|
if (other instanceof Message.Field && other.name !== other.originalName && this.getChild(other.originalName) === null)
|
|
|
other.name = other.originalName; // Revert previous first (effectively keeps both originals)
|
|
|
else if (child instanceof Message.Field && child.name !== child.originalName && this.getChild(child.originalName) === null)
|
|
|
child.name = child.originalName;
|
|
|
else
|
|
|
throw Error("Duplicate name in namespace "+this.toString(true)+": "+child.name);
|
|
|
}
|
|
|
this.children.push(child);
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Gets a child by its name or id.
|
|
|
* @param {string|number} nameOrId Child name or id
|
|
|
* @return {?ProtoBuf.Reflect.T} The child or null if not found
|
|
|
* @expose
|
|
|
*/
|
|
|
NamespacePrototype.getChild = function(nameOrId) {
|
|
|
var key = typeof nameOrId === 'number' ? 'id' : 'name';
|
|
|
for (var i=0, k=this.children.length; i<k; ++i)
|
|
|
if (this.children[i][key] === nameOrId)
|
|
|
return this.children[i];
|
|
|
return null;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Resolves a reflect object inside of this namespace.
|
|
|
* @param {string} qn Qualified name to resolve
|
|
|
* @param {boolean=} excludeFields Excludes fields, defaults to `false`
|
|
|
* @return {?ProtoBuf.Reflect.Namespace} The resolved type or null if not found
|
|
|
* @expose
|
|
|
*/
|
|
|
NamespacePrototype.resolve = function(qn, excludeFields) {
|
|
|
var part = qn.split("."),
|
|
|
ptr = this,
|
|
|
i = 0;
|
|
|
if (part[i] === "") { // Fully qualified name, e.g. ".My.Message'
|
|
|
while (ptr.parent !== null)
|
|
|
ptr = ptr.parent;
|
|
|
i++;
|
|
|
}
|
|
|
var child;
|
|
|
do {
|
|
|
do {
|
|
|
child = ptr.getChild(part[i]);
|
|
|
if (!child || !(child instanceof Reflect.T) || (excludeFields && child instanceof Reflect.Message.Field)) {
|
|
|
ptr = null;
|
|
|
break;
|
|
|
}
|
|
|
ptr = child; i++;
|
|
|
} while (i < part.length);
|
|
|
if (ptr != null)
|
|
|
break; // Found
|
|
|
// Else search the parent
|
|
|
if (this.parent !== null) {
|
|
|
return this.parent.resolve(qn, excludeFields);
|
|
|
}
|
|
|
} while (ptr != null);
|
|
|
return ptr;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Builds the namespace and returns the runtime counterpart.
|
|
|
* @return {Object.<string,Function|Object>} Runtime namespace
|
|
|
* @expose
|
|
|
*/
|
|
|
NamespacePrototype.build = function() {
|
|
|
/** @dict */
|
|
|
var ns = {};
|
|
|
var children = this.children;
|
|
|
for (var i=0, k=children.length, child; i<k; ++i) {
|
|
|
child = children[i];
|
|
|
if (child instanceof Namespace)
|
|
|
ns[child.name] = child.build();
|
|
|
}
|
|
|
if (Object.defineProperty)
|
|
|
Object.defineProperty(ns, "$options", { "value": this.buildOpt() });
|
|
|
return ns;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Builds the namespace's '$options' property.
|
|
|
* @return {Object.<string,*>}
|
|
|
*/
|
|
|
NamespacePrototype.buildOpt = function() {
|
|
|
var opt = {},
|
|
|
keys = Object.keys(this.options);
|
|
|
for (var i=0, k=keys.length; i<k; ++i) {
|
|
|
var key = keys[i],
|
|
|
val = this.options[keys[i]];
|
|
|
// TODO: Options are not resolved, yet.
|
|
|
// if (val instanceof Namespace) {
|
|
|
// opt[key] = val.build();
|
|
|
// } else {
|
|
|
opt[key] = val;
|
|
|
// }
|
|
|
}
|
|
|
return opt;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Gets the value assigned to the option with the specified name.
|
|
|
* @param {string=} name Returns the option value if specified, otherwise all options are returned.
|
|
|
* @return {*|Object.<string,*>}null} Option value or NULL if there is no such option
|
|
|
*/
|
|
|
NamespacePrototype.getOption = function(name) {
|
|
|
if (typeof name === 'undefined')
|
|
|
return this.options;
|
|
|
return typeof this.options[name] !== 'undefined' ? this.options[name] : null;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* @alias ProtoBuf.Reflect.Namespace
|
|
|
* @expose
|
|
|
*/
|
|
|
Reflect.Namespace = Namespace;
|
|
|
|
|
|
/**
|
|
|
* Constructs a new Message.
|
|
|
* @exports ProtoBuf.Reflect.Message
|
|
|
* @param {!ProtoBuf.Builder} builder Builder reference
|
|
|
* @param {!ProtoBuf.Reflect.Namespace} parent Parent message or namespace
|
|
|
* @param {string} name Message name
|
|
|
* @param {Object.<string,*>=} options Message options
|
|
|
* @param {boolean=} isGroup `true` if this is a legacy group
|
|
|
* @constructor
|
|
|
* @extends ProtoBuf.Reflect.Namespace
|
|
|
*/
|
|
|
var Message = function(builder, parent, name, options, isGroup) {
|
|
|
Namespace.call(this, builder, parent, name, options);
|
|
|
|
|
|
/**
|
|
|
* @override
|
|
|
*/
|
|
|
this.className = "Message";
|
|
|
|
|
|
/**
|
|
|
* Extensions range.
|
|
|
* @type {!Array.<number>}
|
|
|
* @expose
|
|
|
*/
|
|
|
this.extensions = [ProtoBuf.ID_MIN, ProtoBuf.ID_MAX];
|
|
|
|
|
|
/**
|
|
|
* Runtime message class.
|
|
|
* @type {?function(new:ProtoBuf.Builder.Message)}
|
|
|
* @expose
|
|
|
*/
|
|
|
this.clazz = null;
|
|
|
|
|
|
/**
|
|
|
* Whether this is a legacy group or not.
|
|
|
* @type {boolean}
|
|
|
* @expose
|
|
|
*/
|
|
|
this.isGroup = !!isGroup;
|
|
|
|
|
|
// The following cached collections are used to efficiently iterate over or look up fields when decoding.
|
|
|
|
|
|
/**
|
|
|
* Cached fields.
|
|
|
* @type {?Array.<!ProtoBuf.Reflect.Message.Field>}
|
|
|
* @private
|
|
|
*/
|
|
|
this._fields = null;
|
|
|
|
|
|
/**
|
|
|
* Cached fields by id.
|
|
|
* @type {?Object.<number,!ProtoBuf.Reflect.Message.Field>}
|
|
|
* @private
|
|
|
*/
|
|
|
this._fieldsById = null;
|
|
|
|
|
|
/**
|
|
|
* Cached fields by name.
|
|
|
* @type {?Object.<string,!ProtoBuf.Reflect.Message.Field>}
|
|
|
* @private
|
|
|
*/
|
|
|
this._fieldsByName = null;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* @alias ProtoBuf.Reflect.Message.prototype
|
|
|
* @inner
|
|
|
*/
|
|
|
var MessagePrototype = Message.prototype = Object.create(Namespace.prototype);
|
|
|
|
|
|
/**
|
|
|
* Builds the message and returns the runtime counterpart, which is a fully functional class.
|
|
|
* @see ProtoBuf.Builder.Message
|
|
|
* @param {boolean=} rebuild Whether to rebuild or not, defaults to false
|
|
|
* @return {ProtoBuf.Reflect.Message} Message class
|
|
|
* @throws {Error} If the message cannot be built
|
|
|
* @expose
|
|
|
*/
|
|
|
MessagePrototype.build = function(rebuild) {
|
|
|
if (this.clazz && !rebuild)
|
|
|
return this.clazz;
|
|
|
|
|
|
// Create the runtime Message class in its own scope
|
|
|
var clazz = (function(ProtoBuf, T) {
|
|
|
|
|
|
var fields = T.getChildren(ProtoBuf.Reflect.Message.Field),
|
|
|
oneofs = T.getChildren(ProtoBuf.Reflect.Message.OneOf);
|
|
|
|
|
|
/**
|
|
|
* Constructs a new runtime Message.
|
|
|
* @name ProtoBuf.Builder.Message
|
|
|
* @class Barebone of all runtime messages.
|
|
|
* @param {!Object.<string,*>|string} values Preset values
|
|
|
* @param {...string} var_args
|
|
|
* @constructor
|
|
|
* @throws {Error} If the message cannot be created
|
|
|
*/
|
|
|
var Message = function(values, var_args) {
|
|
|
ProtoBuf.Builder.Message.call(this);
|
|
|
|
|
|
// Create virtual oneof properties
|
|
|
for (var i=0, k=oneofs.length; i<k; ++i)
|
|
|
this[oneofs[i].name] = null;
|
|
|
// Create fields and set default values
|
|
|
for (i=0, k=fields.length; i<k; ++i) {
|
|
|
var field = fields[i];
|
|
|
this[field.name] = field.repeated ? [] : null;
|
|
|
if (field.required && field.defaultValue !== null)
|
|
|
this[field.name] = field.defaultValue;
|
|
|
}
|
|
|
|
|
|
if (arguments.length > 0) {
|
|
|
// Set field values from a values object
|
|
|
if (arguments.length === 1 && typeof values === 'object' &&
|
|
|
/* not another Message */ typeof values.encode !== 'function' &&
|
|
|
/* not a repeated field */ !ProtoBuf.Util.isArray(values) &&
|
|
|
/* not a ByteBuffer */ !(values instanceof ByteBuffer) &&
|
|
|
/* not an ArrayBuffer */ !(values instanceof ArrayBuffer) &&
|
|
|
/* not a Long */ !(ProtoBuf.Long && values instanceof ProtoBuf.Long)) {
|
|
|
var keys = Object.keys(values);
|
|
|
for (i=0, k=keys.length; i<k; ++i)
|
|
|
this.$set(keys[i], values[keys[i]]); // May throw
|
|
|
} else // Set field values from arguments, in declaration order
|
|
|
for (i=0, k=arguments.length; i<k; ++i)
|
|
|
this.$set(fields[i].name, arguments[i]); // May throw
|
|
|
}
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* @alias ProtoBuf.Builder.Message.prototype
|
|
|
* @inner
|
|
|
*/
|
|
|
var MessagePrototype = Message.prototype = Object.create(ProtoBuf.Builder.Message.prototype);
|
|
|
|
|
|
/**
|
|
|
* Adds a value to a repeated field.
|
|
|
* @name ProtoBuf.Builder.Message#add
|
|
|
* @function
|
|
|
* @param {string} key Field name
|
|
|
* @param {*} value Value to add
|
|
|
* @param {boolean=} noAssert Whether to assert the value or not (asserts by default)
|
|
|
* @throws {Error} If the value cannot be added
|
|
|
* @expose
|
|
|
*/
|
|
|
MessagePrototype.add = function(key, value, noAssert) {
|
|
|
var field = T._fieldsByName[key];
|
|
|
if (!noAssert) {
|
|
|
if (!field)
|
|
|
throw Error(this+"#"+key+" is undefined");
|
|
|
if (!(field instanceof ProtoBuf.Reflect.Message.Field))
|
|
|
throw Error(this+"#"+key+" is not a field: "+field.toString(true)); // May throw if it's an enum or embedded message
|
|
|
if (!field.repeated)
|
|
|
throw Error(this+"#"+key+" is not a repeated field");
|
|
|
}
|
|
|
if (this[field.name] === null)
|
|
|
this[field.name] = [];
|
|
|
this[field.name].push(noAssert ? value : field.verifyValue(value, true));
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Adds a value to a repeated field. This is an alias for {@link ProtoBuf.Builder.Message#add}.
|
|
|
* @name ProtoBuf.Builder.Message#$add
|
|
|
* @function
|
|
|
* @param {string} key Field name
|
|
|
* @param {*} value Value to add
|
|
|
* @param {boolean=} noAssert Whether to assert the value or not (asserts by default)
|
|
|
* @throws {Error} If the value cannot be added
|
|
|
* @expose
|
|
|
*/
|
|
|
MessagePrototype.$add = MessagePrototype.add;
|
|
|
|
|
|
/**
|
|
|
* Sets a field's value.
|
|
|
* @name ProtoBuf.Builder.Message#set
|
|
|
* @function
|
|
|
* @param {string} key Key
|
|
|
* @param {*} value Value to set
|
|
|
* @param {boolean=} noAssert Whether to not assert for an actual field / proper value type, defaults to `false`
|
|
|
* @returns {!ProtoBuf.Builder.Message} this
|
|
|
* @throws {Error} If the value cannot be set
|
|
|
* @expose
|
|
|
*/
|
|
|
MessagePrototype.set = function(key, value, noAssert) {
|
|
|
if (key && typeof key === 'object') {
|
|
|
for (var i in key)
|
|
|
if (key.hasOwnProperty(i))
|
|
|
this.$set(i, key[i], noAssert);
|
|
|
return this;
|
|
|
}
|
|
|
var field = T._fieldsByName[key];
|
|
|
if (!noAssert) {
|
|
|
if (!field)
|
|
|
throw Error(this+"#"+key+" is not a field: undefined");
|
|
|
if (!(field instanceof ProtoBuf.Reflect.Message.Field))
|
|
|
throw Error(this+"#"+key+" is not a field: "+field.toString(true));
|
|
|
this[field.name] = (value = field.verifyValue(value)); // May throw
|
|
|
} else {
|
|
|
this[field.name] = value;
|
|
|
}
|
|
|
if (field.oneof) {
|
|
|
if (value !== null) {
|
|
|
if (this[field.oneof.name] !== null)
|
|
|
this[this[field.oneof.name]] = null; // Unset the previous (field name is the oneof field's value)
|
|
|
this[field.oneof.name] = field.name;
|
|
|
} else if (field.oneof.name === key)
|
|
|
this[field.oneof.name] = null;
|
|
|
}
|
|
|
return this;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Sets a field's value. This is an alias for [@link ProtoBuf.Builder.Message#set}.
|
|
|
* @name ProtoBuf.Builder.Message#$set
|
|
|
* @function
|
|
|
* @param {string} key Key
|
|
|
* @param {*} value Value to set
|
|
|
* @param {boolean=} noAssert Whether to not assert the value, defaults to `false`
|
|
|
* @throws {Error} If the value cannot be set
|
|
|
* @expose
|
|
|
*/
|
|
|
MessagePrototype.$set = MessagePrototype.set;
|
|
|
|
|
|
/**
|
|
|
* Gets a field's value.
|
|
|
* @name ProtoBuf.Builder.Message#get
|
|
|
* @function
|
|
|
* @param {string} key Key
|
|
|
* @param {boolean=} noAssert Whether to not assert for an actual field, defaults to `false`
|
|
|
* @return {*} Value
|
|
|
* @throws {Error} If there is no such field
|
|
|
* @expose
|
|
|
*/
|
|
|
MessagePrototype.get = function(key, noAssert) {
|
|
|
if (noAssert)
|
|
|
return this[key];
|
|
|
var field = T._fieldsByName[key];
|
|
|
if (!field || !(field instanceof ProtoBuf.Reflect.Message.Field))
|
|
|
throw Error(this+"#"+key+" is not a field: undefined");
|
|
|
if (!(field instanceof ProtoBuf.Reflect.Message.Field))
|
|
|
throw Error(this+"#"+key+" is not a field: "+field.toString(true));
|
|
|
return this[field.name];
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Gets a field's value. This is an alias for {@link ProtoBuf.Builder.Message#$get}.
|
|
|
* @name ProtoBuf.Builder.Message#$get
|
|
|
* @function
|
|
|
* @param {string} key Key
|
|
|
* @return {*} Value
|
|
|
* @throws {Error} If there is no such field
|
|
|
* @expose
|
|
|
*/
|
|
|
MessagePrototype.$get = MessagePrototype.get;
|
|
|
|
|
|
// Getters and setters
|
|
|
|
|
|
for (var i=0; i<fields.length; i++) {
|
|
|
var field = fields[i];
|
|
|
// no setters for extension fields as these are named by their fqn
|
|
|
if (field instanceof ProtoBuf.Reflect.Message.ExtensionField)
|
|
|
continue;
|
|
|
|
|
|
if (T.builder.options['populateAccessors'])
|
|
|
(function(field) {
|
|
|
// set/get[SomeValue]
|
|
|
var Name = field.originalName.replace(/(_[a-zA-Z])/g, function(match) {
|
|
|
return match.toUpperCase().replace('_','');
|
|
|
});
|
|
|
Name = Name.substring(0,1).toUpperCase() + Name.substring(1);
|
|
|
|
|
|
// set/get_[some_value] FIXME: Do we really need these?
|
|
|
var name = field.originalName.replace(/([A-Z])/g, function(match) {
|
|
|
return "_"+match;
|
|
|
});
|
|
|
|
|
|
/**
|
|
|
* The current field's unbound setter function.
|
|
|
* @function
|
|
|
* @param {*} value
|
|
|
* @param {boolean=} noAssert
|
|
|
* @returns {!ProtoBuf.Builder.Message}
|
|
|
* @inner
|
|
|
*/
|
|
|
var setter = function(value, noAssert) {
|
|
|
this[field.name] = noAssert ? value : field.verifyValue(value);
|
|
|
return this;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* The current field's unbound getter function.
|
|
|
* @function
|
|
|
* @returns {*}
|
|
|
* @inner
|
|
|
*/
|
|
|
var getter = function() {
|
|
|
return this[field.name];
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Sets a value. This method is present for each field, but only if there is no name conflict with
|
|
|
* another field.
|
|
|
* @name ProtoBuf.Builder.Message#set[SomeField]
|
|
|
* @function
|
|
|
* @param {*} value Value to set
|
|
|
* @param {boolean=} noAssert Whether to not assert the value, defaults to `false`
|
|
|
* @returns {!ProtoBuf.Builder.Message} this
|
|
|
* @abstract
|
|
|
* @throws {Error} If the value cannot be set
|
|
|
*/
|
|
|
if (T.getChild("set"+Name) === null)
|
|
|
MessagePrototype["set"+Name] = setter;
|
|
|
|
|
|
/**
|
|
|
* Sets a value. This method is present for each field, but only if there is no name conflict with
|
|
|
* another field.
|
|
|
* @name ProtoBuf.Builder.Message#set_[some_field]
|
|
|
* @function
|
|
|
* @param {*} value Value to set
|
|
|
* @param {boolean=} noAssert Whether to not assert the value, defaults to `false`
|
|
|
* @returns {!ProtoBuf.Builder.Message} this
|
|
|
* @abstract
|
|
|
* @throws {Error} If the value cannot be set
|
|
|
*/
|
|
|
if (T.getChild("set_"+name) === null)
|
|
|
MessagePrototype["set_"+name] = setter;
|
|
|
|
|
|
/**
|
|
|
* Gets a value. This method is present for each field, but only if there is no name conflict with
|
|
|
* another field.
|
|
|
* @name ProtoBuf.Builder.Message#get[SomeField]
|
|
|
* @function
|
|
|
* @abstract
|
|
|
* @return {*} The value
|
|
|
*/
|
|
|
if (T.getChild("get"+Name) === null)
|
|
|
MessagePrototype["get"+Name] = getter;
|
|
|
|
|
|
/**
|
|
|
* Gets a value. This method is present for each field, but only if there is no name conflict with
|
|
|
* another field.
|
|
|
* @name ProtoBuf.Builder.Message#get_[some_field]
|
|
|
* @function
|
|
|
* @return {*} The value
|
|
|
* @abstract
|
|
|
*/
|
|
|
if (T.getChild("get_"+name) === null)
|
|
|
MessagePrototype["get_"+name] = getter;
|
|
|
|
|
|
})(field);
|
|
|
}
|
|
|
|
|
|
// En-/decoding
|
|
|
|
|
|
/**
|
|
|
* Encodes the message.
|
|
|
* @name ProtoBuf.Builder.Message#$encode
|
|
|
* @function
|
|
|
* @param {(!ByteBuffer|boolean)=} buffer ByteBuffer to encode to. Will create a new one and flip it if omitted.
|
|
|
* @param {boolean=} noVerify Whether to not verify field values, defaults to `false`
|
|
|
* @return {!ByteBuffer} Encoded message as a ByteBuffer
|
|
|
* @throws {Error} If the message cannot be encoded or if required fields are missing. The later still
|
|
|
* returns the encoded ByteBuffer in the `encoded` property on the error.
|
|
|
* @expose
|
|
|
* @see ProtoBuf.Builder.Message#encode64
|
|
|
* @see ProtoBuf.Builder.Message#encodeHex
|
|
|
* @see ProtoBuf.Builder.Message#encodeAB
|
|
|
*/
|
|
|
MessagePrototype.encode = function(buffer, noVerify) {
|
|
|
if (typeof buffer === 'boolean')
|
|
|
noVerify = buffer,
|
|
|
buffer = undefined;
|
|
|
var isNew = false;
|
|
|
if (!buffer)
|
|
|
buffer = new ByteBuffer(),
|
|
|
isNew = true;
|
|
|
var le = buffer.littleEndian;
|
|
|
try {
|
|
|
T.encode(this, buffer.LE(), noVerify);
|
|
|
return (isNew ? buffer.flip() : buffer).LE(le);
|
|
|
} catch (e) {
|
|
|
buffer.LE(le);
|
|
|
throw(e);
|
|
|
}
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Calculates the byte length of the message.
|
|
|
* @name ProtoBuf.Builder.Message#calculate
|
|
|
* @function
|
|
|
* @returns {number} Byte length
|
|
|
* @throws {Error} If the message cannot be calculated or if required fields are missing.
|
|
|
* @expose
|
|
|
*/
|
|
|
MessagePrototype.calculate = function() {
|
|
|
return T.calculate(this);
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Encodes the varint32 length-delimited message.
|
|
|
* @name ProtoBuf.Builder.Message#encodeDelimited
|
|
|
* @function
|
|
|
* @param {(!ByteBuffer|boolean)=} buffer ByteBuffer to encode to. Will create a new one and flip it if omitted.
|
|
|
* @return {!ByteBuffer} Encoded message as a ByteBuffer
|
|
|
* @throws {Error} If the message cannot be encoded or if required fields are missing. The later still
|
|
|
* returns the encoded ByteBuffer in the `encoded` property on the error.
|
|
|
* @expose
|
|
|
*/
|
|
|
MessagePrototype.encodeDelimited = function(buffer) {
|
|
|
var isNew = false;
|
|
|
if (!buffer)
|
|
|
buffer = new ByteBuffer(),
|
|
|
isNew = true;
|
|
|
var enc = new ByteBuffer().LE();
|
|
|
T.encode(this, enc).flip();
|
|
|
buffer.writeVarint32(enc.remaining());
|
|
|
buffer.append(enc);
|
|
|
return isNew ? buffer.flip() : buffer;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Directly encodes the message to an ArrayBuffer.
|
|
|
* @name ProtoBuf.Builder.Message#encodeAB
|
|
|
* @function
|
|
|
* @return {ArrayBuffer} Encoded message as ArrayBuffer
|
|
|
* @throws {Error} If the message cannot be encoded or if required fields are missing. The later still
|
|
|
* returns the encoded ArrayBuffer in the `encoded` property on the error.
|
|
|
* @expose
|
|
|
*/
|
|
|
MessagePrototype.encodeAB = function() {
|
|
|
try {
|
|
|
return this.encode().toArrayBuffer();
|
|
|
} catch (e) {
|
|
|
if (e["encoded"]) e["encoded"] = e["encoded"].toArrayBuffer();
|
|
|
throw(e);
|
|
|
}
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Returns the message as an ArrayBuffer. This is an alias for {@link ProtoBuf.Builder.Message#encodeAB}.
|
|
|
* @name ProtoBuf.Builder.Message#toArrayBuffer
|
|
|
* @function
|
|
|
* @return {ArrayBuffer} Encoded message as ArrayBuffer
|
|
|
* @throws {Error} If the message cannot be encoded or if required fields are missing. The later still
|
|
|
* returns the encoded ArrayBuffer in the `encoded` property on the error.
|
|
|
* @expose
|
|
|
*/
|
|
|
MessagePrototype.toArrayBuffer = MessagePrototype.encodeAB;
|
|
|
|
|
|
/**
|
|
|
* Directly encodes the message to a node Buffer.
|
|
|
* @name ProtoBuf.Builder.Message#encodeNB
|
|
|
* @function
|
|
|
* @return {!Buffer}
|
|
|
* @throws {Error} If the message cannot be encoded, not running under node.js or if required fields are
|
|
|
* missing. The later still returns the encoded node Buffer in the `encoded` property on the error.
|
|
|
* @expose
|
|
|
*/
|
|
|
MessagePrototype.encodeNB = function() {
|
|
|
try {
|
|
|
return this.encode().toBuffer();
|
|
|
} catch (e) {
|
|
|
if (e["encoded"]) e["encoded"] = e["encoded"].toBuffer();
|
|
|
throw(e);
|
|
|
}
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Returns the message as a node Buffer. This is an alias for {@link ProtoBuf.Builder.Message#encodeNB}.
|
|
|
* @name ProtoBuf.Builder.Message#toBuffer
|
|
|
* @function
|
|
|
* @return {!Buffer}
|
|
|
* @throws {Error} If the message cannot be encoded or if required fields are missing. The later still
|
|
|
* returns the encoded node Buffer in the `encoded` property on the error.
|
|
|
* @expose
|
|
|
*/
|
|
|
MessagePrototype.toBuffer = MessagePrototype.encodeNB;
|
|
|
|
|
|
/**
|
|
|
* Directly encodes the message to a base64 encoded string.
|
|
|
* @name ProtoBuf.Builder.Message#encode64
|
|
|
* @function
|
|
|
* @return {string} Base64 encoded string
|
|
|
* @throws {Error} If the underlying buffer cannot be encoded or if required fields are missing. The later
|
|
|
* still returns the encoded base64 string in the `encoded` property on the error.
|
|
|
* @expose
|
|
|
*/
|
|
|
MessagePrototype.encode64 = function() {
|
|
|
try {
|
|
|
return this.encode().toBase64();
|
|
|
} catch (e) {
|
|
|
if (e["encoded"]) e["encoded"] = e["encoded"].toBase64();
|
|
|
throw(e);
|
|
|
}
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Returns the message as a base64 encoded string. This is an alias for {@link ProtoBuf.Builder.Message#encode64}.
|
|
|
* @name ProtoBuf.Builder.Message#toBase64
|
|
|
* @function
|
|
|
* @return {string} Base64 encoded string
|
|
|
* @throws {Error} If the message cannot be encoded or if required fields are missing. The later still
|
|
|
* returns the encoded base64 string in the `encoded` property on the error.
|
|
|
* @expose
|
|
|
*/
|
|
|
MessagePrototype.toBase64 = MessagePrototype.encode64;
|
|
|
|
|
|
/**
|
|
|
* Directly encodes the message to a hex encoded string.
|
|
|
* @name ProtoBuf.Builder.Message#encodeHex
|
|
|
* @function
|
|
|
* @return {string} Hex encoded string
|
|
|
* @throws {Error} If the underlying buffer cannot be encoded or if required fields are missing. The later
|
|
|
* still returns the encoded hex string in the `encoded` property on the error.
|
|
|
* @expose
|
|
|
*/
|
|
|
MessagePrototype.encodeHex = function() {
|
|
|
try {
|
|
|
return this.encode().toHex();
|
|
|
} catch (e) {
|
|
|
if (e["encoded"]) e["encoded"] = e["encoded"].toHex();
|
|
|
throw(e);
|
|
|
}
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Returns the message as a hex encoded string. This is an alias for {@link ProtoBuf.Builder.Message#encodeHex}.
|
|
|
* @name ProtoBuf.Builder.Message#toHex
|
|
|
* @function
|
|
|
* @return {string} Hex encoded string
|
|
|
* @throws {Error} If the message cannot be encoded or if required fields are missing. The later still
|
|
|
* returns the encoded hex string in the `encoded` property on the error.
|
|
|
* @expose
|
|
|
*/
|
|
|
MessagePrototype.toHex = MessagePrototype.encodeHex;
|
|
|
|
|
|
/**
|
|
|
* Clones a message object to a raw object.
|
|
|
* @param {*} obj Object to clone
|
|
|
* @param {boolean} includeBinaryAsBase64 Whether to include binary data as base64 strings or not
|
|
|
* @returns {*} Cloned object
|
|
|
* @inner
|
|
|
*/
|
|
|
function cloneRaw(obj, includeBinaryAsBase64) {
|
|
|
var clone = {};
|
|
|
for (var i in obj)
|
|
|
if (obj.hasOwnProperty(i)) {
|
|
|
if (obj[i] === null || typeof obj[i] !== 'object')
|
|
|
clone[i] = obj[i];
|
|
|
else if (obj[i] instanceof ByteBuffer) {
|
|
|
if (includeBinaryAsBase64)
|
|
|
clone[i] = obj[i].toBase64();
|
|
|
} else // is a non-null object
|
|
|
clone[i] = cloneRaw(obj[i], includeBinaryAsBase64);
|
|
|
}
|
|
|
return clone;
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* Returns the message's raw payload.
|
|
|
* @param {boolean=} includeBinaryAsBase64 Whether to include binary data as base64 strings or not, defaults to `false`
|
|
|
* @returns {Object.<string,*>} Raw payload
|
|
|
* @expose
|
|
|
*/
|
|
|
MessagePrototype.toRaw = function(includeBinaryAsBase64) {
|
|
|
return cloneRaw(this, !!includeBinaryAsBase64);
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Decodes a message from the specified buffer or string.
|
|
|
* @name ProtoBuf.Builder.Message.decode
|
|
|
* @function
|
|
|
* @param {!ByteBuffer|!ArrayBuffer|!Buffer|string} buffer Buffer to decode from
|
|
|
* @param {string=} enc Encoding if buffer is a string: hex, utf8 (not recommended), defaults to base64
|
|
|
* @return {!ProtoBuf.Builder.Message} Decoded message
|
|
|
* @throws {Error} If the message cannot be decoded or if required fields are missing. The later still
|
|
|
* returns the decoded message with missing fields in the `decoded` property on the error.
|
|
|
* @expose
|
|
|
* @see ProtoBuf.Builder.Message.decode64
|
|
|
* @see ProtoBuf.Builder.Message.decodeHex
|
|
|
*/
|
|
|
Message.decode = function(buffer, enc) {
|
|
|
if (typeof buffer === 'string')
|
|
|
buffer = ByteBuffer.wrap(buffer, enc ? enc : "base64");
|
|
|
buffer = buffer instanceof ByteBuffer ? buffer : ByteBuffer.wrap(buffer); // May throw
|
|
|
var le = buffer.littleEndian;
|
|
|
try {
|
|
|
var msg = T.decode(buffer.LE());
|
|
|
buffer.LE(le);
|
|
|
return msg;
|
|
|
} catch (e) {
|
|
|
buffer.LE(le);
|
|
|
throw(e);
|
|
|
}
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Decodes a varint32 length-delimited message from the specified buffer or string.
|
|
|
* @name ProtoBuf.Builder.Message.decodeDelimited
|
|
|
* @function
|
|
|
* @param {!ByteBuffer|!ArrayBuffer|!Buffer|string} buffer Buffer to decode from
|
|
|
* @param {string=} enc Encoding if buffer is a string: hex, utf8 (not recommended), defaults to base64
|
|
|
* @return {ProtoBuf.Builder.Message} Decoded message or `null` if not enough bytes are available yet
|
|
|
* @throws {Error} If the message cannot be decoded or if required fields are missing. The later still
|
|
|
* returns the decoded message with missing fields in the `decoded` property on the error.
|
|
|
* @expose
|
|
|
*/
|
|
|
Message.decodeDelimited = function(buffer, enc) {
|
|
|
if (typeof buffer === 'string')
|
|
|
buffer = ByteBuffer.wrap(buffer, enc ? enc : "base64");
|
|
|
buffer = buffer instanceof ByteBuffer ? buffer : ByteBuffer.wrap(buffer); // May throw
|
|
|
if (buffer.remaining() < 1)
|
|
|
return null;
|
|
|
var off = buffer.offset,
|
|
|
len = buffer.readVarint32();
|
|
|
if (buffer.remaining() < len) {
|
|
|
buffer.offset = off;
|
|
|
return null;
|
|
|
}
|
|
|
try {
|
|
|
var msg = T.decode(buffer.slice(buffer.offset, buffer.offset + len).LE());
|
|
|
buffer.offset += len;
|
|
|
return msg;
|
|
|
} catch (err) {
|
|
|
buffer.offset += len;
|
|
|
throw err;
|
|
|
}
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Decodes the message from the specified base64 encoded string.
|
|
|
* @name ProtoBuf.Builder.Message.decode64
|
|
|
* @function
|
|
|
* @param {string} str String to decode from
|
|
|
* @return {!ProtoBuf.Builder.Message} Decoded message
|
|
|
* @throws {Error} If the message cannot be decoded or if required fields are missing. The later still
|
|
|
* returns the decoded message with missing fields in the `decoded` property on the error.
|
|
|
* @expose
|
|
|
*/
|
|
|
Message.decode64 = function(str) {
|
|
|
return Message.decode(str, "base64");
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Decodes the message from the specified hex encoded string.
|
|
|
* @name ProtoBuf.Builder.Message.decodeHex
|
|
|
* @function
|
|
|
* @param {string} str String to decode from
|
|
|
* @return {!ProtoBuf.Builder.Message} Decoded message
|
|
|
* @throws {Error} If the message cannot be decoded or if required fields are missing. The later still
|
|
|
* returns the decoded message with missing fields in the `decoded` property on the error.
|
|
|
* @expose
|
|
|
*/
|
|
|
Message.decodeHex = function(str) {
|
|
|
return Message.decode(str, "hex");
|
|
|
};
|
|
|
|
|
|
// Utility
|
|
|
|
|
|
/**
|
|
|
* Returns a string representation of this Message.
|
|
|
* @name ProtoBuf.Builder.Message#toString
|
|
|
* @function
|
|
|
* @return {string} String representation as of ".Fully.Qualified.MessageName"
|
|
|
* @expose
|
|
|
*/
|
|
|
MessagePrototype.toString = function() {
|
|
|
return T.toString();
|
|
|
};
|
|
|
|
|
|
// Properties
|
|
|
|
|
|
/**
|
|
|
* Options.
|
|
|
* @name ProtoBuf.Builder.Message.$options
|
|
|
* @type {Object.<string,*>}
|
|
|
* @expose
|
|
|
*/
|
|
|
var $options; // cc
|
|
|
|
|
|
/**
|
|
|
* Reflection type.
|
|
|
* @name ProtoBuf.Builder.Message#$type
|
|
|
* @type {!ProtoBuf.Reflect.Message}
|
|
|
* @expose
|
|
|
*/
|
|
|
var $type; // cc
|
|
|
|
|
|
if (Object.defineProperty)
|
|
|
Object.defineProperty(Message, '$options', { "value": T.buildOpt() }),
|
|
|
Object.defineProperty(MessagePrototype, "$type", {
|
|
|
get: function() { return T; }
|
|
|
});
|
|
|
|
|
|
return Message;
|
|
|
|
|
|
})(ProtoBuf, this);
|
|
|
|
|
|
// Static enums and prototyped sub-messages / cached collections
|
|
|
this._fields = [];
|
|
|
this._fieldsById = {};
|
|
|
this._fieldsByName = {};
|
|
|
for (var i=0, k=this.children.length, child; i<k; i++) {
|
|
|
child = this.children[i];
|
|
|
if (child instanceof Enum)
|
|
|
clazz[child.name] = child.build();
|
|
|
else if (child instanceof Message)
|
|
|
clazz[child.name] = child.build();
|
|
|
else if (child instanceof Message.Field)
|
|
|
child.build(),
|
|
|
this._fields.push(child),
|
|
|
this._fieldsById[child.id] = child,
|
|
|
this._fieldsByName[child.name] = child;
|
|
|
else if (!(child instanceof Message.OneOf) && !(child instanceof Extension)) // Not built
|
|
|
throw Error("Illegal reflect child of "+this.toString(true)+": "+children[i].toString(true));
|
|
|
}
|
|
|
|
|
|
return this.clazz = clazz;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Encodes a runtime message's contents to the specified buffer.
|
|
|
* @param {!ProtoBuf.Builder.Message} message Runtime message to encode
|
|
|
* @param {ByteBuffer} buffer ByteBuffer to write to
|
|
|
* @param {boolean=} noVerify Whether to not verify field values, defaults to `false`
|
|
|
* @return {ByteBuffer} The ByteBuffer for chaining
|
|
|
* @throws {Error} If required fields are missing or the message cannot be encoded for another reason
|
|
|
* @expose
|
|
|
*/
|
|
|
MessagePrototype.encode = function(message, buffer, noVerify) {
|
|
|
var fieldMissing = null,
|
|
|
field;
|
|
|
for (var i=0, k=this._fields.length, val; i<k; ++i) {
|
|
|
field = this._fields[i];
|
|
|
val = message[field.name];
|
|
|
if (field.required && val === null) {
|
|
|
if (fieldMissing === null)
|
|
|
fieldMissing = field;
|
|
|
} else
|
|
|
field.encode(noVerify ? val : field.verifyValue(val), buffer);
|
|
|
}
|
|
|
if (fieldMissing !== null) {
|
|
|
var err = Error("Missing at least one required field for "+this.toString(true)+": "+fieldMissing);
|
|
|
err["encoded"] = buffer; // Still expose what we got
|
|
|
throw(err);
|
|
|
}
|
|
|
return buffer;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Calculates a runtime message's byte length.
|
|
|
* @param {!ProtoBuf.Builder.Message} message Runtime message to encode
|
|
|
* @returns {number} Byte length
|
|
|
* @throws {Error} If required fields are missing or the message cannot be calculated for another reason
|
|
|
* @expose
|
|
|
*/
|
|
|
MessagePrototype.calculate = function(message) {
|
|
|
for (var n=0, i=0, k=this._fields.length, field, val; i<k; ++i) {
|
|
|
field = this._fields[i];
|
|
|
val = message[field.name];
|
|
|
if (field.required && val === null)
|
|
|
throw Error("Missing at least one required field for "+this.toString(true)+": "+field);
|
|
|
else
|
|
|
n += field.calculate(val);
|
|
|
}
|
|
|
return n;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Skips all data until the end of the specified group has been reached.
|
|
|
* @param {number} expectedId Expected GROUPEND id
|
|
|
* @param {!ByteBuffer} buf ByteBuffer
|
|
|
* @returns {boolean} `true` if a value as been skipped, `false` if the end has been reached
|
|
|
* @throws {Error} If it wasn't possible to find the end of the group (buffer overrun or end tag mismatch)
|
|
|
* @inner
|
|
|
*/
|
|
|
function skipTillGroupEnd(expectedId, buf) {
|
|
|
var tag = buf.readVarint32(), // Throws on OOB
|
|
|
wireType = tag & 0x07,
|
|
|
id = tag >> 3;
|
|
|
switch (wireType) {
|
|
|
case ProtoBuf.WIRE_TYPES.VARINT:
|
|
|
do tag = buf.readUint8();
|
|
|
while ((tag & 0x80) === 0x80);
|
|
|
break;
|
|
|
case ProtoBuf.WIRE_TYPES.BITS64:
|
|
|
buf.offset += 8;
|
|
|
break;
|
|
|
case ProtoBuf.WIRE_TYPES.LDELIM:
|
|
|
tag = buf.readVarint32(); // reads the varint
|
|
|
buf.offset += tag; // skips n bytes
|
|
|
break;
|
|
|
case ProtoBuf.WIRE_TYPES.STARTGROUP:
|
|
|
skipTillGroupEnd(id, buf);
|
|
|
break;
|
|
|
case ProtoBuf.WIRE_TYPES.ENDGROUP:
|
|
|
if (id === expectedId)
|
|
|
return false;
|
|
|
else
|
|
|
throw Error("Illegal GROUPEND after unknown group: "+id+" ("+expectedId+" expected)");
|
|
|
case ProtoBuf.WIRE_TYPES.BITS32:
|
|
|
buf.offset += 4;
|
|
|
break;
|
|
|
default:
|
|
|
throw Error("Illegal wire type in unknown group "+expectedId+": "+wireType);
|
|
|
}
|
|
|
return true;
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* Decodes an encoded message and returns the decoded message.
|
|
|
* @param {ByteBuffer} buffer ByteBuffer to decode from
|
|
|
* @param {number=} length Message length. Defaults to decode all the available data.
|
|
|
* @param {number=} expectedGroupEndId Expected GROUPEND id if this is a legacy group
|
|
|
* @return {ProtoBuf.Builder.Message} Decoded message
|
|
|
* @throws {Error} If the message cannot be decoded
|
|
|
* @expose
|
|
|
*/
|
|
|
MessagePrototype.decode = function(buffer, length, expectedGroupEndId) {
|
|
|
length = typeof length === 'number' ? length : -1;
|
|
|
var start = buffer.offset,
|
|
|
msg = new (this.clazz)(),
|
|
|
tag, wireType, id, field;
|
|
|
while (buffer.offset < start+length || (length === -1 && buffer.remaining() > 0)) {
|
|
|
tag = buffer.readVarint32();
|
|
|
wireType = tag & 0x07;
|
|
|
id = tag >> 3;
|
|
|
if (wireType === ProtoBuf.WIRE_TYPES.ENDGROUP) {
|
|
|
if (id !== expectedGroupEndId)
|
|
|
throw Error("Illegal group end indicator for "+this.toString(true)+": "+id+" ("+(expectedGroupEndId ? expectedGroupEndId+" expected" : "not a group")+")");
|
|
|
break;
|
|
|
}
|
|
|
if (!(field = this._fieldsById[id])) {
|
|
|
// "messages created by your new code can be parsed by your old code: old binaries simply ignore the new field when parsing."
|
|
|
switch (wireType) {
|
|
|
case ProtoBuf.WIRE_TYPES.VARINT:
|
|
|
buffer.readVarint32();
|
|
|
break;
|
|
|
case ProtoBuf.WIRE_TYPES.BITS32:
|
|
|
buffer.offset += 4;
|
|
|
break;
|
|
|
case ProtoBuf.WIRE_TYPES.BITS64:
|
|
|
buffer.offset += 8;
|
|
|
break;
|
|
|
case ProtoBuf.WIRE_TYPES.LDELIM:
|
|
|
var len = buffer.readVarint32();
|
|
|
buffer.offset += len;
|
|
|
break;
|
|
|
case ProtoBuf.WIRE_TYPES.STARTGROUP:
|
|
|
while (skipTillGroupEnd(id, buffer)) {}
|
|
|
break;
|
|
|
default:
|
|
|
throw Error("Illegal wire type for unknown field "+id+" in "+this.toString(true)+"#decode: "+wireType);
|
|
|
}
|
|
|
continue;
|
|
|
}
|
|
|
if (field.repeated && !field.options["packed"])
|
|
|
msg[field.name].push(field.decode(wireType, buffer));
|
|
|
else {
|
|
|
msg[field.name] = field.decode(wireType, buffer);
|
|
|
if (field.oneof) {
|
|
|
if (this[field.oneof.name] !== null)
|
|
|
this[this[field.oneof.name]] = null;
|
|
|
msg[field.oneof.name] = field.name;
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
|
|
|
// Check if all required fields are present and set default values for optional fields that are not
|
|
|
for (var i=0, k=this._fields.length; i<k; ++i) {
|
|
|
field = this._fields[i];
|
|
|
if (msg[field.name] === null)
|
|
|
if (field.required) {
|
|
|
var err = Error("Missing at least one required field for "+this.toString(true)+": "+field.name);
|
|
|
err["decoded"] = msg; // Still expose what we got
|
|
|
throw(err);
|
|
|
} else if (field.defaultValue !== null)
|
|
|
msg[field.name] = field.defaultValue;
|
|
|
}
|
|
|
return msg;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* @alias ProtoBuf.Reflect.Message
|
|
|
* @expose
|
|
|
*/
|
|
|
Reflect.Message = Message;
|
|
|
|
|
|
/**
|
|
|
* Constructs a new Message Field.
|
|
|
* @exports ProtoBuf.Reflect.Message.Field
|
|
|
* @param {!ProtoBuf.Builder} builder Builder reference
|
|
|
* @param {!ProtoBuf.Reflect.Message} message Message reference
|
|
|
* @param {string} rule Rule, one of requried, optional, repeated
|
|
|
* @param {string} type Data type, e.g. int32
|
|
|
* @param {string} name Field name
|
|
|
* @param {number} id Unique field id
|
|
|
* @param {Object.<string,*>=} options Options
|
|
|
* @param {!ProtoBuf.Reflect.Message.OneOf=} oneof Enclosing OneOf
|
|
|
* @constructor
|
|
|
* @extends ProtoBuf.Reflect.T
|
|
|
*/
|
|
|
var Field = function(builder, message, rule, type, name, id, options, oneof) {
|
|
|
T.call(this, builder, message, name);
|
|
|
|
|
|
/**
|
|
|
* @override
|
|
|
*/
|
|
|
this.className = "Message.Field";
|
|
|
|
|
|
/**
|
|
|
* Message field required flag.
|
|
|
* @type {boolean}
|
|
|
* @expose
|
|
|
*/
|
|
|
this.required = rule === "required";
|
|
|
|
|
|
/**
|
|
|
* Message field repeated flag.
|
|
|
* @type {boolean}
|
|
|
* @expose
|
|
|
*/
|
|
|
this.repeated = rule === "repeated";
|
|
|
|
|
|
/**
|
|
|
* Message field type. Type reference string if unresolved, protobuf type if resolved.
|
|
|
* @type {string|{name: string, wireType: number}}
|
|
|
* @expose
|
|
|
*/
|
|
|
this.type = type;
|
|
|
|
|
|
/**
|
|
|
* Resolved type reference inside the global namespace.
|
|
|
* @type {ProtoBuf.Reflect.T|null}
|
|
|
* @expose
|
|
|
*/
|
|
|
this.resolvedType = null;
|
|
|
|
|
|
/**
|
|
|
* Unique message field id.
|
|
|
* @type {number}
|
|
|
* @expose
|
|
|
*/
|
|
|
this.id = id;
|
|
|
|
|
|
/**
|
|
|
* Message field options.
|
|
|
* @type {!Object.<string,*>}
|
|
|
* @dict
|
|
|
* @expose
|
|
|
*/
|
|
|
this.options = options || {};
|
|
|
|
|
|
/**
|
|
|
* Default value.
|
|
|
* @type {*}
|
|
|
* @expose
|
|
|
*/
|
|
|
this.defaultValue = null;
|
|
|
|
|
|
/**
|
|
|
* Enclosing OneOf.
|
|
|
* @type {?ProtoBuf.Reflect.Message.OneOf}
|
|
|
* @expose
|
|
|
*/
|
|
|
this.oneof = oneof || null;
|
|
|
|
|
|
/**
|
|
|
* Original field name.
|
|
|
* @type {string}
|
|
|
* @expose
|
|
|
*/
|
|
|
this.originalName = this.name; // Used to revert camelcase transformation on naming collisions
|
|
|
|
|
|
// Convert field names to camel case notation if the override is set
|
|
|
if (this.builder.options['convertFieldsToCamelCase'] && !(this instanceof Message.ExtensionField))
|
|
|
this.name = Field._toCamelCase(this.name);
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Converts a field name to camel case.
|
|
|
* @param {string} name Likely underscore notated name
|
|
|
* @returns {string} Camel case notated name
|
|
|
* @private
|
|
|
*/
|
|
|
Field._toCamelCase = function(name) {
|
|
|
return name.replace(/_([a-zA-Z])/g, function($0, $1) {
|
|
|
return $1.toUpperCase();
|
|
|
});
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* @alias ProtoBuf.Reflect.Message.Field.prototype
|
|
|
* @inner
|
|
|
*/
|
|
|
var FieldPrototype = Field.prototype = Object.create(T.prototype);
|
|
|
|
|
|
/**
|
|
|
* Builds the field.
|
|
|
* @override
|
|
|
* @expose
|
|
|
*/
|
|
|
FieldPrototype.build = function() {
|
|
|
this.defaultValue = typeof this.options['default'] !== 'undefined'
|
|
|
? this.verifyValue(this.options['default']) : null;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Makes a Long from a value.
|
|
|
* @param {{low: number, high: number, unsigned: boolean}|string|number} value Value
|
|
|
* @param {boolean=} unsigned Whether unsigned or not, defaults to reuse it from Long-like objects or to signed for
|
|
|
* strings and numbers
|
|
|
* @returns {!Long}
|
|
|
* @throws {Error} If the value cannot be converted to a Long
|
|
|
* @inner
|
|
|
*/
|
|
|
function mkLong(value, unsigned) {
|
|
|
if (value && typeof value.low === 'number' && typeof value.high === 'number' && typeof value.unsigned === 'boolean'
|
|
|
&& value.low === value.low && value.high === value.high)
|
|
|
return new ProtoBuf.Long(value.low, value.high, typeof unsigned === 'undefined' ? value.unsigned : unsigned);
|
|
|
if (typeof value === 'string')
|
|
|
return ProtoBuf.Long.fromString(value, unsigned || false, 10);
|
|
|
if (typeof value === 'number')
|
|
|
return ProtoBuf.Long.fromNumber(value, unsigned || false);
|
|
|
throw Error("not convertible to Long");
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* Checks if the given value can be set for this field.
|
|
|
* @param {*} value Value to check
|
|
|
* @param {boolean=} skipRepeated Whether to skip the repeated value check or not. Defaults to false.
|
|
|
* @return {*} Verified, maybe adjusted, value
|
|
|
* @throws {Error} If the value cannot be set for this field
|
|
|
* @expose
|
|
|
*/
|
|
|
FieldPrototype.verifyValue = function(value, skipRepeated) {
|
|
|
skipRepeated = skipRepeated || false;
|
|
|
var fail = function(val, msg) {
|
|
|
throw Error("Illegal value for "+this.toString(true)+" of type "+this.type.name+": "+val+" ("+msg+")");
|
|
|
}.bind(this);
|
|
|
if (value === null) { // NULL values for optional fields
|
|
|
if (this.required)
|
|
|
fail(typeof value, "required");
|
|
|
return null;
|
|
|
}
|
|
|
var i;
|
|
|
if (this.repeated && !skipRepeated) { // Repeated values as arrays
|
|
|
if (!ProtoBuf.Util.isArray(value))
|
|
|
value = [value];
|
|
|
var res = [];
|
|
|
for (i=0; i<value.length; i++)
|
|
|
res.push(this.verifyValue(value[i], true));
|
|
|
return res;
|
|
|
}
|
|
|
// All non-repeated fields expect no array
|
|
|
if (!this.repeated && ProtoBuf.Util.isArray(value))
|
|
|
fail(typeof value, "no array expected");
|
|
|
|
|
|
switch (this.type) {
|
|
|
// Signed 32bit
|
|
|
case ProtoBuf.TYPES["int32"]:
|
|
|
case ProtoBuf.TYPES["sint32"]:
|
|
|
case ProtoBuf.TYPES["sfixed32"]:
|
|
|
// Account for !NaN: value === value
|
|
|
if (typeof value !== 'number' || (value === value && value % 1 !== 0))
|
|
|
fail(typeof value, "not an integer");
|
|
|
return value > 4294967295 ? value | 0 : value;
|
|
|
|
|
|
// Unsigned 32bit
|
|
|
case ProtoBuf.TYPES["uint32"]:
|
|
|
case ProtoBuf.TYPES["fixed32"]:
|
|
|
if (typeof value !== 'number' || (value === value && value % 1 !== 0))
|
|
|
fail(typeof value, "not an integer");
|
|
|
return value < 0 ? value >>> 0 : value;
|
|
|
|
|
|
// Signed 64bit
|
|
|
case ProtoBuf.TYPES["int64"]:
|
|
|
case ProtoBuf.TYPES["sint64"]:
|
|
|
case ProtoBuf.TYPES["sfixed64"]: {
|
|
|
if (ProtoBuf.Long)
|
|
|
try {
|
|
|
return mkLong(value, false);
|
|
|
} catch (e) {
|
|
|
fail(typeof value, e.message);
|
|
|
}
|
|
|
else
|
|
|
fail(typeof value, "requires Long.js");
|
|
|
}
|
|
|
|
|
|
// Unsigned 64bit
|
|
|
case ProtoBuf.TYPES["uint64"]:
|
|
|
case ProtoBuf.TYPES["fixed64"]: {
|
|
|
if (ProtoBuf.Long)
|
|
|
try {
|
|
|
return mkLong(value, true);
|
|
|
} catch (e) {
|
|
|
fail(typeof value, e.message);
|
|
|
}
|
|
|
else
|
|
|
fail(typeof value, "requires Long.js");
|
|
|
}
|
|
|
|
|
|
// Bool
|
|
|
case ProtoBuf.TYPES["bool"]:
|
|
|
if (typeof value !== 'boolean')
|
|
|
fail(typeof value, "not a boolean");
|
|
|
return value;
|
|
|
|
|
|
// Float
|
|
|
case ProtoBuf.TYPES["float"]:
|
|
|
case ProtoBuf.TYPES["double"]:
|
|
|
if (typeof value !== 'number')
|
|
|
fail(typeof value, "not a number");
|
|
|
return value;
|
|
|
|
|
|
// Length-delimited string
|
|
|
case ProtoBuf.TYPES["string"]:
|
|
|
if (typeof value !== 'string' && !(value && value instanceof String))
|
|
|
fail(typeof value, "not a string");
|
|
|
return ""+value; // Convert String object to string
|
|
|
|
|
|
// Length-delimited bytes
|
|
|
case ProtoBuf.TYPES["bytes"]:
|
|
|
if (ByteBuffer.isByteBuffer(value))
|
|
|
return value;
|
|
|
return ByteBuffer.wrap(value, "base64");
|
|
|
|
|
|
// Constant enum value
|
|
|
case ProtoBuf.TYPES["enum"]: {
|
|
|
var values = this.resolvedType.getChildren(Enum.Value);
|
|
|
for (i=0; i<values.length; i++)
|
|
|
if (values[i].name == value)
|
|
|
return values[i].id;
|
|
|
else if (values[i].id == value)
|
|
|
return values[i].id;
|
|
|
fail(value, "not a valid enum value");
|
|
|
}
|
|
|
// Embedded message
|
|
|
case ProtoBuf.TYPES["group"]:
|
|
|
case ProtoBuf.TYPES["message"]: {
|
|
|
if (!value || typeof value !== 'object')
|
|
|
fail(typeof value, "object expected");
|
|
|
if (value instanceof this.resolvedType.clazz)
|
|
|
return value;
|
|
|
if (value instanceof ProtoBuf.Builder.Message) {
|
|
|
// Mismatched type: Convert to object (see: https://github.com/dcodeIO/ProtoBuf.js/issues/180)
|
|
|
var obj = {};
|
|
|
for (var i in value)
|
|
|
if (value.hasOwnProperty(i))
|
|
|
obj[i] = value[i];
|
|
|
value = obj;
|
|
|
}
|
|
|
// Else let's try to construct one from a key-value object
|
|
|
return new (this.resolvedType.clazz)(value); // May throw for a hundred of reasons
|
|
|
}
|
|
|
}
|
|
|
|
|
|
// We should never end here
|
|
|
throw Error("[INTERNAL] Illegal value for "+this.toString(true)+": "+value+" (undefined type "+this.type+")");
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Encodes the specified field value to the specified buffer.
|
|
|
* @param {*} value Verified field value
|
|
|
* @param {ByteBuffer} buffer ByteBuffer to encode to
|
|
|
* @return {ByteBuffer} The ByteBuffer for chaining
|
|
|
* @throws {Error} If the field cannot be encoded
|
|
|
* @expose
|
|
|
*/
|
|
|
FieldPrototype.encode = function(value, buffer) {
|
|
|
if (this.type === null || typeof this.type !== 'object')
|
|
|
throw Error("[INTERNAL] Unresolved type in "+this.toString(true)+": "+this.type);
|
|
|
if (value === null || (this.repeated && value.length == 0))
|
|
|
return buffer; // Optional omitted
|
|
|
try {
|
|
|
if (this.repeated) {
|
|
|
var i;
|
|
|
// "Only repeated fields of primitive numeric types (types which use the varint, 32-bit, or 64-bit wire
|
|
|
// types) can be declared 'packed'."
|
|
|
if (this.options["packed"] && ProtoBuf.PACKABLE_WIRE_TYPES.indexOf(this.type.wireType) >= 0) {
|
|
|
// "All of the elements of the field are packed into a single key-value pair with wire type 2
|
|
|
// (length-delimited). Each element is encoded the same way it would be normally, except without a
|
|
|
// tag preceding it."
|
|
|
buffer.writeVarint32((this.id << 3) | ProtoBuf.WIRE_TYPES.LDELIM);
|
|
|
buffer.ensureCapacity(buffer.offset += 1); // We do not know the length yet, so let's assume a varint of length 1
|
|
|
var start = buffer.offset; // Remember where the contents begin
|
|
|
for (i=0; i<value.length; i++)
|
|
|
this.encodeValue(value[i], buffer);
|
|
|
var len = buffer.offset-start,
|
|
|
varintLen = ByteBuffer.calculateVarint32(len);
|
|
|
if (varintLen > 1) { // We need to move the contents
|
|
|
var contents = buffer.slice(start, buffer.offset);
|
|
|
start += varintLen-1;
|
|
|
buffer.offset = start;
|
|
|
buffer.append(contents);
|
|
|
}
|
|
|
buffer.writeVarint32(len, start-varintLen);
|
|
|
} else {
|
|
|
// "If your message definition has repeated elements (without the [packed=true] option), the encoded
|
|
|
// message has zero or more key-value pairs with the same tag number"
|
|
|
for (i=0; i<value.length; i++)
|
|
|
buffer.writeVarint32((this.id << 3) | this.type.wireType),
|
|
|
this.encodeValue(value[i], buffer);
|
|
|
}
|
|
|
} else
|
|
|
buffer.writeVarint32((this.id << 3) | this.type.wireType),
|
|
|
this.encodeValue(value, buffer);
|
|
|
} catch (e) {
|
|
|
throw Error("Illegal value for "+this.toString(true)+": "+value+" ("+e+")");
|
|
|
}
|
|
|
return buffer;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Encodes a value to the specified buffer. Does not encode the key.
|
|
|
* @param {*} value Field value
|
|
|
* @param {ByteBuffer} buffer ByteBuffer to encode to
|
|
|
* @return {ByteBuffer} The ByteBuffer for chaining
|
|
|
* @throws {Error} If the value cannot be encoded
|
|
|
* @expose
|
|
|
*/
|
|
|
FieldPrototype.encodeValue = function(value, buffer) {
|
|
|
if (value === null) return buffer; // Nothing to encode
|
|
|
// Tag has already been written
|
|
|
|
|
|
switch (this.type) {
|
|
|
// 32bit signed varint
|
|
|
case ProtoBuf.TYPES["int32"]:
|
|
|
// "If you use int32 or int64 as the type for a negative number, the resulting varint is always ten bytes
|
|
|
// long – it is, effectively, treated like a very large unsigned integer." (see #122)
|
|
|
if (value < 0)
|
|
|
buffer.writeVarint64(value);
|
|
|
else
|
|
|
buffer.writeVarint32(value);
|
|
|
break;
|
|
|
|
|
|
// 32bit unsigned varint
|
|
|
case ProtoBuf.TYPES["uint32"]:
|
|
|
buffer.writeVarint32(value);
|
|
|
break;
|
|
|
|
|
|
// 32bit varint zig-zag
|
|
|
case ProtoBuf.TYPES["sint32"]:
|
|
|
buffer.writeVarint32ZigZag(value);
|
|
|
break;
|
|
|
|
|
|
// Fixed unsigned 32bit
|
|
|
case ProtoBuf.TYPES["fixed32"]:
|
|
|
buffer.writeUint32(value);
|
|
|
break;
|
|
|
|
|
|
// Fixed signed 32bit
|
|
|
case ProtoBuf.TYPES["sfixed32"]:
|
|
|
buffer.writeInt32(value);
|
|
|
break;
|
|
|
|
|
|
// 64bit varint as-is
|
|
|
case ProtoBuf.TYPES["int64"]:
|
|
|
case ProtoBuf.TYPES["uint64"]:
|
|
|
buffer.writeVarint64(value); // throws
|
|
|
break;
|
|
|
|
|
|
// 64bit varint zig-zag
|
|
|
case ProtoBuf.TYPES["sint64"]:
|
|
|
buffer.writeVarint64ZigZag(value); // throws
|
|
|
break;
|
|
|
|
|
|
// Fixed unsigned 64bit
|
|
|
case ProtoBuf.TYPES["fixed64"]:
|
|
|
buffer.writeUint64(value); // throws
|
|
|
break;
|
|
|
|
|
|
// Fixed signed 64bit
|
|
|
case ProtoBuf.TYPES["sfixed64"]:
|
|
|
buffer.writeInt64(value); // throws
|
|
|
break;
|
|
|
|
|
|
// Bool
|
|
|
case ProtoBuf.TYPES["bool"]:
|
|
|
if (typeof value === 'string')
|
|
|
buffer.writeVarint32(value.toLowerCase() === 'false' ? 0 : !!value);
|
|
|
else
|
|
|
buffer.writeVarint32(value ? 1 : 0);
|
|
|
break;
|
|
|
|
|
|
// Constant enum value
|
|
|
case ProtoBuf.TYPES["enum"]:
|
|
|
buffer.writeVarint32(value);
|
|
|
break;
|
|
|
|
|
|
// 32bit float
|
|
|
case ProtoBuf.TYPES["float"]:
|
|
|
buffer.writeFloat32(value);
|
|
|
break;
|
|
|
|
|
|
// 64bit float
|
|
|
case ProtoBuf.TYPES["double"]:
|
|
|
buffer.writeFloat64(value);
|
|
|
break;
|
|
|
|
|
|
// Length-delimited string
|
|
|
case ProtoBuf.TYPES["string"]:
|
|
|
buffer.writeVString(value);
|
|
|
break;
|
|
|
|
|
|
// Length-delimited bytes
|
|
|
case ProtoBuf.TYPES["bytes"]:
|
|
|
if (value.remaining() < 0)
|
|
|
throw Error("Illegal value for "+this.toString(true)+": "+value.remaining()+" bytes remaining");
|
|
|
var prevOffset = value.offset;
|
|
|
buffer.writeVarint32(value.remaining());
|
|
|
buffer.append(value);
|
|
|
value.offset = prevOffset;
|
|
|
break;
|
|
|
|
|
|
// Embedded message
|
|
|
case ProtoBuf.TYPES["message"]:
|
|
|
var bb = new ByteBuffer().LE();
|
|
|
this.resolvedType.encode(value, bb);
|
|
|
buffer.writeVarint32(bb.offset);
|
|
|
buffer.append(bb.flip());
|
|
|
break;
|
|
|
|
|
|
// Legacy group
|
|
|
case ProtoBuf.TYPES["group"]:
|
|
|
this.resolvedType.encode(value, buffer);
|
|
|
buffer.writeVarint32((this.id << 3) | ProtoBuf.WIRE_TYPES.ENDGROUP);
|
|
|
break;
|
|
|
|
|
|
default:
|
|
|
// We should never end here
|
|
|
throw Error("[INTERNAL] Illegal value to encode in "+this.toString(true)+": "+value+" (unknown type)");
|
|
|
}
|
|
|
return buffer;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Calculates the length of this field's value on the network level.
|
|
|
* @param {*} value Field value
|
|
|
* @returns {number} Byte length
|
|
|
* @expose
|
|
|
*/
|
|
|
FieldPrototype.calculate = function(value) {
|
|
|
value = this.verifyValue(value); // May throw
|
|
|
if (this.type === null || typeof this.type !== 'object')
|
|
|
throw Error("[INTERNAL] Unresolved type in "+this.toString(true)+": "+this.type);
|
|
|
if (value === null || (this.repeated && value.length == 0))
|
|
|
return 0; // Optional omitted
|
|
|
var n = 0;
|
|
|
try {
|
|
|
if (this.repeated) {
|
|
|
var i, ni;
|
|
|
if (this.options["packed"] && ProtoBuf.PACKABLE_WIRE_TYPES.indexOf(this.type.wireType) >= 0) {
|
|
|
n += ByteBuffer.calculateVarint32((this.id << 3) | ProtoBuf.WIRE_TYPES.LDELIM);
|
|
|
ni = 0;
|
|
|
for (i=0; i<value.length; i++)
|
|
|
ni += this.calculateValue(value[i]);
|
|
|
n += ByteBuffer.calculateVarint32(ni);
|
|
|
n += ni;
|
|
|
} else {
|
|
|
for (i=0; i<value.length; i++)
|
|
|
n += ByteBuffer.calculateVarint32((this.id << 3) | this.type.wireType),
|
|
|
n += this.calculateValue(value[i]);
|
|
|
}
|
|
|
} else {
|
|
|
n += ByteBuffer.calculateVarint32((this.id << 3) | this.type.wireType);
|
|
|
n += this.calculateValue(value);
|
|
|
}
|
|
|
} catch (e) {
|
|
|
throw Error("Illegal value for "+this.toString(true)+": "+value+" ("+e+")");
|
|
|
}
|
|
|
return n;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Calculates the byte length of a value.
|
|
|
* @param {*} value Field value
|
|
|
* @returns {number} Byte length
|
|
|
* @throws {Error} If the value cannot be calculated
|
|
|
* @expose
|
|
|
*/
|
|
|
FieldPrototype.calculateValue = function(value) {
|
|
|
if (value === null) return 0; // Nothing to encode
|
|
|
// Tag has already been written
|
|
|
var n;
|
|
|
switch (this.type) {
|
|
|
case ProtoBuf.TYPES["int32"]:
|
|
|
return value < 0 ? ByteBuffer.calculateVarint64(value) : ByteBuffer.calculateVarint32(value);
|
|
|
case ProtoBuf.TYPES["uint32"]:
|
|
|
return ByteBuffer.calculateVarint32(value);
|
|
|
case ProtoBuf.TYPES["sint32"]:
|
|
|
return ByteBuffer.calculateVarint32(ByteBuffer.zigZagEncode32(value));
|
|
|
case ProtoBuf.TYPES["fixed32"]:
|
|
|
case ProtoBuf.TYPES["sfixed32"]:
|
|
|
case ProtoBuf.TYPES["float"]:
|
|
|
return 4;
|
|
|
case ProtoBuf.TYPES["int64"]:
|
|
|
case ProtoBuf.TYPES["uint64"]:
|
|
|
return ByteBuffer.calculateVarint64(value);
|
|
|
case ProtoBuf.TYPES["sint64"]:
|
|
|
return ByteBuffer.calculateVarint64(ByteBuffer.zigZagEncode64(value));
|
|
|
case ProtoBuf.TYPES["fixed64"]:
|
|
|
case ProtoBuf.TYPES["sfixed64"]:
|
|
|
return 8;
|
|
|
case ProtoBuf.TYPES["bool"]:
|
|
|
return 1;
|
|
|
case ProtoBuf.TYPES["enum"]:
|
|
|
return ByteBuffer.calculateVarint32(value);
|
|
|
case ProtoBuf.TYPES["double"]:
|
|
|
return 8;
|
|
|
case ProtoBuf.TYPES["string"]:
|
|
|
n = ByteBuffer.calculateUTF8Bytes(value);
|
|
|
return ByteBuffer.calculateVarint32(n) + n;
|
|
|
case ProtoBuf.TYPES["bytes"]:
|
|
|
if (value.remaining() < 0)
|
|
|
throw Error("Illegal value for "+this.toString(true)+": "+value.remaining()+" bytes remaining");
|
|
|
return ByteBuffer.calculateVarint32(value.remaining()) + value.remaining();
|
|
|
case ProtoBuf.TYPES["message"]:
|
|
|
n = this.resolvedType.calculate(value);
|
|
|
return ByteBuffer.calculateVarint32(n) + n;
|
|
|
case ProtoBuf.TYPES["group"]:
|
|
|
n = this.resolvedType.calculate(value);
|
|
|
return n + ByteBuffer.calculateVarint32((this.id << 3) | ProtoBuf.WIRE_TYPES.ENDGROUP);
|
|
|
}
|
|
|
// We should never end here
|
|
|
throw Error("[INTERNAL] Illegal value to encode in "+this.toString(true)+": "+value+" (unknown type)");
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Decode the field value from the specified buffer.
|
|
|
* @param {number} wireType Leading wire type
|
|
|
* @param {ByteBuffer} buffer ByteBuffer to decode from
|
|
|
* @param {boolean=} skipRepeated Whether to skip the repeated check or not. Defaults to false.
|
|
|
* @return {*} Decoded value
|
|
|
* @throws {Error} If the field cannot be decoded
|
|
|
* @expose
|
|
|
*/
|
|
|
FieldPrototype.decode = function(wireType, buffer, skipRepeated) {
|
|
|
var value, nBytes;
|
|
|
if (wireType != this.type.wireType && (skipRepeated || (wireType != ProtoBuf.WIRE_TYPES.LDELIM || !this.repeated)))
|
|
|
throw Error("Illegal wire type for field "+this.toString(true)+": "+wireType+" ("+this.type.wireType+" expected)");
|
|
|
if (wireType == ProtoBuf.WIRE_TYPES.LDELIM && this.repeated && this.options["packed"] && ProtoBuf.PACKABLE_WIRE_TYPES.indexOf(this.type.wireType) >= 0) {
|
|
|
if (!skipRepeated) {
|
|
|
nBytes = buffer.readVarint32();
|
|
|
nBytes = buffer.offset + nBytes; // Limit
|
|
|
var values = [];
|
|
|
while (buffer.offset < nBytes)
|
|
|
values.push(this.decode(this.type.wireType, buffer, true));
|
|
|
return values;
|
|
|
}
|
|
|
// Read the next value otherwise...
|
|
|
}
|
|
|
switch (this.type) {
|
|
|
// 32bit signed varint
|
|
|
case ProtoBuf.TYPES["int32"]:
|
|
|
return buffer.readVarint32() | 0;
|
|
|
|
|
|
// 32bit unsigned varint
|
|
|
case ProtoBuf.TYPES["uint32"]:
|
|
|
return buffer.readVarint32() >>> 0;
|
|
|
|
|
|
// 32bit signed varint zig-zag
|
|
|
case ProtoBuf.TYPES["sint32"]:
|
|
|
return buffer.readVarint32ZigZag() | 0;
|
|
|
|
|
|
// Fixed 32bit unsigned
|
|
|
case ProtoBuf.TYPES["fixed32"]:
|
|
|
return buffer.readUint32() >>> 0;
|
|
|
|
|
|
case ProtoBuf.TYPES["sfixed32"]:
|
|
|
return buffer.readInt32() | 0;
|
|
|
|
|
|
// 64bit signed varint
|
|
|
case ProtoBuf.TYPES["int64"]:
|
|
|
return buffer.readVarint64();
|
|
|
|
|
|
// 64bit unsigned varint
|
|
|
case ProtoBuf.TYPES["uint64"]:
|
|
|
return buffer.readVarint64().toUnsigned();
|
|
|
|
|
|
// 64bit signed varint zig-zag
|
|
|
case ProtoBuf.TYPES["sint64"]:
|
|
|
return buffer.readVarint64ZigZag();
|
|
|
|
|
|
// Fixed 64bit unsigned
|
|
|
case ProtoBuf.TYPES["fixed64"]:
|
|
|
return buffer.readUint64();
|
|
|
|
|
|
// Fixed 64bit signed
|
|
|
case ProtoBuf.TYPES["sfixed64"]:
|
|
|
return buffer.readInt64();
|
|
|
|
|
|
// Bool varint
|
|
|
case ProtoBuf.TYPES["bool"]:
|
|
|
return !!buffer.readVarint32();
|
|
|
|
|
|
// Constant enum value (varint)
|
|
|
case ProtoBuf.TYPES["enum"]:
|
|
|
// The following Builder.Message#set will already throw
|
|
|
return buffer.readVarint32();
|
|
|
|
|
|
// 32bit float
|
|
|
case ProtoBuf.TYPES["float"]:
|
|
|
return buffer.readFloat();
|
|
|
|
|
|
// 64bit float
|
|
|
case ProtoBuf.TYPES["double"]:
|
|
|
return buffer.readDouble();
|
|
|
|
|
|
// Length-delimited string
|
|
|
case ProtoBuf.TYPES["string"]:
|
|
|
return buffer.readVString();
|
|
|
|
|
|
// Length-delimited bytes
|
|
|
case ProtoBuf.TYPES["bytes"]: {
|
|
|
nBytes = buffer.readVarint32();
|
|
|
if (buffer.remaining() < nBytes)
|
|
|
throw Error("Illegal number of bytes for "+this.toString(true)+": "+nBytes+" required but got only "+buffer.remaining());
|
|
|
value = buffer.clone(); // Offset already set
|
|
|
value.limit = value.offset+nBytes;
|
|
|
buffer.offset += nBytes;
|
|
|
return value;
|
|
|
}
|
|
|
|
|
|
// Length-delimited embedded message
|
|
|
case ProtoBuf.TYPES["message"]: {
|
|
|
nBytes = buffer.readVarint32();
|
|
|
return this.resolvedType.decode(buffer, nBytes);
|
|
|
}
|
|
|
|
|
|
// Legacy group
|
|
|
case ProtoBuf.TYPES["group"]:
|
|
|
return this.resolvedType.decode(buffer, -1, this.id);
|
|
|
}
|
|
|
|
|
|
// We should never end here
|
|
|
throw Error("[INTERNAL] Illegal wire type for "+this.toString(true)+": "+wireType);
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* @alias ProtoBuf.Reflect.Message.Field
|
|
|
* @expose
|
|
|
*/
|
|
|
Reflect.Message.Field = Field;
|
|
|
|
|
|
/**
|
|
|
* Constructs a new Message ExtensionField.
|
|
|
* @exports ProtoBuf.Reflect.Message.ExtensionField
|
|
|
* @param {!ProtoBuf.Builder} builder Builder reference
|
|
|
* @param {!ProtoBuf.Reflect.Message} message Message reference
|
|
|
* @param {string} rule Rule, one of requried, optional, repeated
|
|
|
* @param {string} type Data type, e.g. int32
|
|
|
* @param {string} name Field name
|
|
|
* @param {number} id Unique field id
|
|
|
* @param {Object.<string,*>=} options Options
|
|
|
* @constructor
|
|
|
* @extends ProtoBuf.Reflect.Message.Field
|
|
|
*/
|
|
|
var ExtensionField = function(builder, message, rule, type, name, id, options) {
|
|
|
Field.call(this, builder, message, rule, type, name, id, options);
|
|
|
|
|
|
/**
|
|
|
* Extension reference.
|
|
|
* @type {!ProtoBuf.Reflect.Extension}
|
|
|
* @expose
|
|
|
*/
|
|
|
this.extension;
|
|
|
};
|
|
|
|
|
|
// Extends Field
|
|
|
ExtensionField.prototype = Object.create(Field.prototype);
|
|
|
|
|
|
/**
|
|
|
* @alias ProtoBuf.Reflect.Message.ExtensionField
|
|
|
* @expose
|
|
|
*/
|
|
|
Reflect.Message.ExtensionField = ExtensionField;
|
|
|
|
|
|
/**
|
|
|
* Constructs a new Message OneOf.
|
|
|
* @exports ProtoBuf.Reflect.Message.OneOf
|
|
|
* @param {!ProtoBuf.Builder} builder Builder reference
|
|
|
* @param {!ProtoBuf.Reflect.Message} message Message reference
|
|
|
* @param {string} name OneOf name
|
|
|
* @constructor
|
|
|
* @extends ProtoBuf.Reflect.T
|
|
|
*/
|
|
|
var OneOf = function(builder, message, name) {
|
|
|
T.call(this, builder, message, name);
|
|
|
|
|
|
/**
|
|
|
* Enclosed fields.
|
|
|
* @type {!Array.<!ProtoBuf.Reflect.Message.Field>}
|
|
|
* @expose
|
|
|
*/
|
|
|
this.fields = [];
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* @alias ProtoBuf.Reflect.Message.OneOf
|
|
|
* @expose
|
|
|
*/
|
|
|
Reflect.Message.OneOf = OneOf;
|
|
|
|
|
|
/**
|
|
|
* Constructs a new Enum.
|
|
|
* @exports ProtoBuf.Reflect.Enum
|
|
|
* @param {!ProtoBuf.Builder} builder Builder reference
|
|
|
* @param {!ProtoBuf.Reflect.T} parent Parent Reflect object
|
|
|
* @param {string} name Enum name
|
|
|
* @param {Object.<string,*>=} options Enum options
|
|
|
* @constructor
|
|
|
* @extends ProtoBuf.Reflect.Namespace
|
|
|
*/
|
|
|
var Enum = function(builder, parent, name, options) {
|
|
|
Namespace.call(this, builder, parent, name, options);
|
|
|
|
|
|
/**
|
|
|
* @override
|
|
|
*/
|
|
|
this.className = "Enum";
|
|
|
|
|
|
/**
|
|
|
* Runtime enum object.
|
|
|
* @type {Object.<string,number>|null}
|
|
|
* @expose
|
|
|
*/
|
|
|
this.object = null;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* @alias ProtoBuf.Reflect.Enum.prototype
|
|
|
* @inner
|
|
|
*/
|
|
|
var EnumPrototype = Enum.prototype = Object.create(Namespace.prototype);
|
|
|
|
|
|
/**
|
|
|
* Builds this enum and returns the runtime counterpart.
|
|
|
* @return {Object<string,*>}
|
|
|
* @expose
|
|
|
*/
|
|
|
EnumPrototype.build = function() {
|
|
|
var enm = {},
|
|
|
values = this.getChildren(Enum.Value);
|
|
|
for (var i=0, k=values.length; i<k; ++i)
|
|
|
enm[values[i]['name']] = values[i]['id'];
|
|
|
if (Object.defineProperty)
|
|
|
Object.defineProperty(enm, '$options', { "value": this.buildOpt() });
|
|
|
return this.object = enm;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* @alias ProtoBuf.Reflect.Enum
|
|
|
* @expose
|
|
|
*/
|
|
|
Reflect.Enum = Enum;
|
|
|
|
|
|
/**
|
|
|
* Constructs a new Enum Value.
|
|
|
* @exports ProtoBuf.Reflect.Enum.Value
|
|
|
* @param {!ProtoBuf.Builder} builder Builder reference
|
|
|
* @param {!ProtoBuf.Reflect.Enum} enm Enum reference
|
|
|
* @param {string} name Field name
|
|
|
* @param {number} id Unique field id
|
|
|
* @constructor
|
|
|
* @extends ProtoBuf.Reflect.T
|
|
|
*/
|
|
|
var Value = function(builder, enm, name, id) {
|
|
|
T.call(this, builder, enm, name);
|
|
|
|
|
|
/**
|
|
|
* @override
|
|
|
*/
|
|
|
this.className = "Enum.Value";
|
|
|
|
|
|
/**
|
|
|
* Unique enum value id.
|
|
|
* @type {number}
|
|
|
* @expose
|
|
|
*/
|
|
|
this.id = id;
|
|
|
};
|
|
|
|
|
|
// Extends T
|
|
|
Value.prototype = Object.create(T.prototype);
|
|
|
|
|
|
/**
|
|
|
* @alias ProtoBuf.Reflect.Enum.Value
|
|
|
* @expose
|
|
|
*/
|
|
|
Reflect.Enum.Value = Value;
|
|
|
|
|
|
/**
|
|
|
* An extension (field).
|
|
|
* @exports ProtoBuf.Reflect.Extension
|
|
|
* @constructor
|
|
|
* @param {!ProtoBuf.Builder} builder Builder reference
|
|
|
* @param {!ProtoBuf.Reflect.T} parent Parent object
|
|
|
* @param {string} name Object name
|
|
|
* @param {!ProtoBuf.Reflect.Message.Field} field Extension field
|
|
|
*/
|
|
|
var Extension = function(builder, parent, name, field) {
|
|
|
T.call(this, builder, parent, name);
|
|
|
|
|
|
/**
|
|
|
* Extended message field.
|
|
|
* @type {!ProtoBuf.Reflect.Message.Field}
|
|
|
* @expose
|
|
|
*/
|
|
|
this.field = field;
|
|
|
};
|
|
|
|
|
|
// Extends T
|
|
|
Extension.prototype = Object.create(T.prototype);
|
|
|
|
|
|
/**
|
|
|
* @alias ProtoBuf.Reflect.Extension
|
|
|
* @expose
|
|
|
*/
|
|
|
Reflect.Extension = Extension;
|
|
|
|
|
|
/**
|
|
|
* Constructs a new Service.
|
|
|
* @exports ProtoBuf.Reflect.Service
|
|
|
* @param {!ProtoBuf.Builder} builder Builder reference
|
|
|
* @param {!ProtoBuf.Reflect.Namespace} root Root
|
|
|
* @param {string} name Service name
|
|
|
* @param {Object.<string,*>=} options Options
|
|
|
* @constructor
|
|
|
* @extends ProtoBuf.Reflect.Namespace
|
|
|
*/
|
|
|
var Service = function(builder, root, name, options) {
|
|
|
Namespace.call(this, builder, root, name, options);
|
|
|
|
|
|
/**
|
|
|
* @override
|
|
|
*/
|
|
|
this.className = "Service";
|
|
|
|
|
|
/**
|
|
|
* Built runtime service class.
|
|
|
* @type {?function(new:ProtoBuf.Builder.Service)}
|
|
|
*/
|
|
|
this.clazz = null;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* @alias ProtoBuf.Reflect.Service.prototype
|
|
|
* @inner
|
|
|
*/
|
|
|
var ServicePrototype = Service.prototype = Object.create(Namespace.prototype);
|
|
|
|
|
|
/**
|
|
|
* Builds the service and returns the runtime counterpart, which is a fully functional class.
|
|
|
* @see ProtoBuf.Builder.Service
|
|
|
* @param {boolean=} rebuild Whether to rebuild or not
|
|
|
* @return {Function} Service class
|
|
|
* @throws {Error} If the message cannot be built
|
|
|
* @expose
|
|
|
*/
|
|
|
ServicePrototype.build = function(rebuild) {
|
|
|
if (this.clazz && !rebuild)
|
|
|
return this.clazz;
|
|
|
|
|
|
// Create the runtime Service class in its own scope
|
|
|
return this.clazz = (function(ProtoBuf, T) {
|
|
|
|
|
|
/**
|
|
|
* Constructs a new runtime Service.
|
|
|
* @name ProtoBuf.Builder.Service
|
|
|
* @param {function(string, ProtoBuf.Builder.Message, function(Error, ProtoBuf.Builder.Message=))=} rpcImpl RPC implementation receiving the method name and the message
|
|
|
* @class Barebone of all runtime services.
|
|
|
* @constructor
|
|
|
* @throws {Error} If the service cannot be created
|
|
|
*/
|
|
|
var Service = function(rpcImpl) {
|
|
|
ProtoBuf.Builder.Service.call(this);
|
|
|
|
|
|
/**
|
|
|
* Service implementation.
|
|
|
* @name ProtoBuf.Builder.Service#rpcImpl
|
|
|
* @type {!function(string, ProtoBuf.Builder.Message, function(Error, ProtoBuf.Builder.Message=))}
|
|
|
* @expose
|
|
|
*/
|
|
|
this.rpcImpl = rpcImpl || function(name, msg, callback) {
|
|
|
// This is what a user has to implement: A function receiving the method name, the actual message to
|
|
|
// send (type checked) and the callback that's either provided with the error as its first
|
|
|
// argument or null and the actual response message.
|
|
|
setTimeout(callback.bind(this, Error("Not implemented, see: https://github.com/dcodeIO/ProtoBuf.js/wiki/Services")), 0); // Must be async!
|
|
|
};
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* @alias ProtoBuf.Builder.Service.prototype
|
|
|
* @inner
|
|
|
*/
|
|
|
var ServicePrototype = Service.prototype = Object.create(ProtoBuf.Builder.Service.prototype);
|
|
|
|
|
|
if (Object.defineProperty)
|
|
|
Object.defineProperty(Service, "$options", { "value": T.buildOpt() }),
|
|
|
Object.defineProperty(ServicePrototype, "$options", { "value": Service["$options"] });
|
|
|
|
|
|
/**
|
|
|
* Asynchronously performs an RPC call using the given RPC implementation.
|
|
|
* @name ProtoBuf.Builder.Service.[Method]
|
|
|
* @function
|
|
|
* @param {!function(string, ProtoBuf.Builder.Message, function(Error, ProtoBuf.Builder.Message=))} rpcImpl RPC implementation
|
|
|
* @param {ProtoBuf.Builder.Message} req Request
|
|
|
* @param {function(Error, (ProtoBuf.Builder.Message|ByteBuffer|Buffer|string)=)} callback Callback receiving
|
|
|
* the error if any and the response either as a pre-parsed message or as its raw bytes
|
|
|
* @abstract
|
|
|
*/
|
|
|
|
|
|
/**
|
|
|
* Asynchronously performs an RPC call using the instance's RPC implementation.
|
|
|
* @name ProtoBuf.Builder.Service#[Method]
|
|
|
* @function
|
|
|
* @param {ProtoBuf.Builder.Message} req Request
|
|
|
* @param {function(Error, (ProtoBuf.Builder.Message|ByteBuffer|Buffer|string)=)} callback Callback receiving
|
|
|
* the error if any and the response either as a pre-parsed message or as its raw bytes
|
|
|
* @abstract
|
|
|
*/
|
|
|
|
|
|
var rpc = T.getChildren(ProtoBuf.Reflect.Service.RPCMethod);
|
|
|
for (var i=0; i<rpc.length; i++) {
|
|
|
(function(method) {
|
|
|
|
|
|
// service#Method(message, callback)
|
|
|
ServicePrototype[method.name] = function(req, callback) {
|
|
|
try {
|
|
|
if (!req || !(req instanceof method.resolvedRequestType.clazz)) {
|
|
|
setTimeout(callback.bind(this, Error("Illegal request type provided to service method "+T.name+"#"+method.name)), 0);
|
|
|
return;
|
|
|
}
|
|
|
this.rpcImpl(method.fqn(), req, function(err, res) { // Assumes that this is properly async
|
|
|
if (err) {
|
|
|
callback(err);
|
|
|
return;
|
|
|
}
|
|
|
try { res = method.resolvedResponseType.clazz.decode(res); } catch (notABuffer) {}
|
|
|
if (!res || !(res instanceof method.resolvedResponseType.clazz)) {
|
|
|
callback(Error("Illegal response type received in service method "+ T.name+"#"+method.name));
|
|
|
return;
|
|
|
}
|
|
|
callback(null, res);
|
|
|
});
|
|
|
} catch (err) {
|
|
|
setTimeout(callback.bind(this, err), 0);
|
|
|
}
|
|
|
};
|
|
|
|
|
|
// Service.Method(rpcImpl, message, callback)
|
|
|
Service[method.name] = function(rpcImpl, req, callback) {
|
|
|
new Service(rpcImpl)[method.name](req, callback);
|
|
|
};
|
|
|
|
|
|
if (Object.defineProperty)
|
|
|
Object.defineProperty(Service[method.name], "$options", { "value": method.buildOpt() }),
|
|
|
Object.defineProperty(ServicePrototype[method.name], "$options", { "value": Service[method.name]["$options"] });
|
|
|
})(rpc[i]);
|
|
|
}
|
|
|
|
|
|
return Service;
|
|
|
|
|
|
})(ProtoBuf, this);
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* @alias ProtoBuf.Reflect.Service
|
|
|
* @expose
|
|
|
*/
|
|
|
Reflect.Service = Service;
|
|
|
|
|
|
/**
|
|
|
* Abstract service method.
|
|
|
* @exports ProtoBuf.Reflect.Service.Method
|
|
|
* @param {!ProtoBuf.Builder} builder Builder reference
|
|
|
* @param {!ProtoBuf.Reflect.Service} svc Service
|
|
|
* @param {string} name Method name
|
|
|
* @param {Object.<string,*>=} options Options
|
|
|
* @constructor
|
|
|
* @extends ProtoBuf.Reflect.T
|
|
|
*/
|
|
|
var Method = function(builder, svc, name, options) {
|
|
|
T.call(this, builder, svc, name);
|
|
|
|
|
|
/**
|
|
|
* @override
|
|
|
*/
|
|
|
this.className = "Service.Method";
|
|
|
|
|
|
/**
|
|
|
* Options.
|
|
|
* @type {Object.<string, *>}
|
|
|
* @expose
|
|
|
*/
|
|
|
this.options = options || {};
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* @alias ProtoBuf.Reflect.Service.Method.prototype
|
|
|
* @inner
|
|
|
*/
|
|
|
var MethodPrototype = Method.prototype = Object.create(T.prototype);
|
|
|
|
|
|
/**
|
|
|
* Builds the method's '$options' property.
|
|
|
* @name ProtoBuf.Reflect.Service.Method#buildOpt
|
|
|
* @function
|
|
|
* @return {Object.<string,*>}
|
|
|
*/
|
|
|
MethodPrototype.buildOpt = NamespacePrototype.buildOpt;
|
|
|
|
|
|
/**
|
|
|
* @alias ProtoBuf.Reflect.Service.Method
|
|
|
* @expose
|
|
|
*/
|
|
|
Reflect.Service.Method = Method;
|
|
|
|
|
|
/**
|
|
|
* RPC service method.
|
|
|
* @exports ProtoBuf.Reflect.Service.RPCMethod
|
|
|
* @param {!ProtoBuf.Builder} builder Builder reference
|
|
|
* @param {!ProtoBuf.Reflect.Service} svc Service
|
|
|
* @param {string} name Method name
|
|
|
* @param {string} request Request message name
|
|
|
* @param {string} response Response message name
|
|
|
* @param {Object.<string,*>=} options Options
|
|
|
* @constructor
|
|
|
* @extends ProtoBuf.Reflect.Service.Method
|
|
|
*/
|
|
|
var RPCMethod = function(builder, svc, name, request, response, options) {
|
|
|
Method.call(this, builder, svc, name, options);
|
|
|
|
|
|
/**
|
|
|
* @override
|
|
|
*/
|
|
|
this.className = "Service.RPCMethod";
|
|
|
|
|
|
/**
|
|
|
* Request message name.
|
|
|
* @type {string}
|
|
|
* @expose
|
|
|
*/
|
|
|
this.requestName = request;
|
|
|
|
|
|
/**
|
|
|
* Response message name.
|
|
|
* @type {string}
|
|
|
* @expose
|
|
|
*/
|
|
|
this.responseName = response;
|
|
|
|
|
|
/**
|
|
|
* Resolved request message type.
|
|
|
* @type {ProtoBuf.Reflect.Message}
|
|
|
* @expose
|
|
|
*/
|
|
|
this.resolvedRequestType = null;
|
|
|
|
|
|
/**
|
|
|
* Resolved response message type.
|
|
|
* @type {ProtoBuf.Reflect.Message}
|
|
|
* @expose
|
|
|
*/
|
|
|
this.resolvedResponseType = null;
|
|
|
};
|
|
|
|
|
|
// Extends Method
|
|
|
RPCMethod.prototype = Object.create(Method.prototype);
|
|
|
|
|
|
/**
|
|
|
* @alias ProtoBuf.Reflect.Service.RPCMethod
|
|
|
* @expose
|
|
|
*/
|
|
|
Reflect.Service.RPCMethod = RPCMethod;
|
|
|
|
|
|
return Reflect;
|
|
|
|
|
|
})(ProtoBuf);
|
|
|
|
|
|
/**
|
|
|
* @alias ProtoBuf.Builder
|
|
|
* @expose
|
|
|
*/
|
|
|
ProtoBuf.Builder = (function(ProtoBuf, Lang, Reflect) {
|
|
|
"use strict";
|
|
|
|
|
|
/**
|
|
|
* Constructs a new Builder.
|
|
|
* @exports ProtoBuf.Builder
|
|
|
* @class Provides the functionality to build protocol messages.
|
|
|
* @param {Object.<string,*>=} options Options
|
|
|
* @constructor
|
|
|
*/
|
|
|
var Builder = function(options) {
|
|
|
|
|
|
/**
|
|
|
* Namespace.
|
|
|
* @type {ProtoBuf.Reflect.Namespace}
|
|
|
* @expose
|
|
|
*/
|
|
|
this.ns = new Reflect.Namespace(this, null, ""); // Global namespace
|
|
|
|
|
|
/**
|
|
|
* Namespace pointer.
|
|
|
* @type {ProtoBuf.Reflect.T}
|
|
|
* @expose
|
|
|
*/
|
|
|
this.ptr = this.ns;
|
|
|
|
|
|
/**
|
|
|
* Resolved flag.
|
|
|
* @type {boolean}
|
|
|
* @expose
|
|
|
*/
|
|
|
this.resolved = false;
|
|
|
|
|
|
/**
|
|
|
* The current building result.
|
|
|
* @type {Object.<string,ProtoBuf.Builder.Message|Object>|null}
|
|
|
* @expose
|
|
|
*/
|
|
|
this.result = null;
|
|
|
|
|
|
/**
|
|
|
* Imported files.
|
|
|
* @type {Array.<string>}
|
|
|
* @expose
|
|
|
*/
|
|
|
this.files = {};
|
|
|
|
|
|
/**
|
|
|
* Import root override.
|
|
|
* @type {?string}
|
|
|
* @expose
|
|
|
*/
|
|
|
this.importRoot = null;
|
|
|
|
|
|
/**
|
|
|
* Options.
|
|
|
* @type {!Object.<string, *>}
|
|
|
* @expose
|
|
|
*/
|
|
|
this.options = options || {};
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* @alias ProtoBuf.Builder.prototype
|
|
|
* @inner
|
|
|
*/
|
|
|
var BuilderPrototype = Builder.prototype;
|
|
|
|
|
|
/**
|
|
|
* Resets the pointer to the root namespace.
|
|
|
* @expose
|
|
|
*/
|
|
|
BuilderPrototype.reset = function() {
|
|
|
this.ptr = this.ns;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Defines a package on top of the current pointer position and places the pointer on it.
|
|
|
* @param {string} pkg
|
|
|
* @param {Object.<string,*>=} options
|
|
|
* @return {ProtoBuf.Builder} this
|
|
|
* @throws {Error} If the package name is invalid
|
|
|
* @expose
|
|
|
*/
|
|
|
BuilderPrototype.define = function(pkg, options) {
|
|
|
if (typeof pkg !== 'string' || !Lang.TYPEREF.test(pkg))
|
|
|
throw Error("Illegal package: "+pkg);
|
|
|
var part = pkg.split("."), i;
|
|
|
for (i=0; i<part.length; i++) // To be absolutely sure
|
|
|
if (!Lang.NAME.test(part[i]))
|
|
|
throw Error("Illegal package: "+part[i]);
|
|
|
for (i=0; i<part.length; i++) {
|
|
|
if (this.ptr.getChild(part[i]) === null) // Keep existing namespace
|
|
|
this.ptr.addChild(new Reflect.Namespace(this, this.ptr, part[i], options));
|
|
|
this.ptr = this.ptr.getChild(part[i]);
|
|
|
}
|
|
|
return this;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Tests if a definition is a valid message definition.
|
|
|
* @param {Object.<string,*>} def Definition
|
|
|
* @return {boolean} true if valid, else false
|
|
|
* @expose
|
|
|
*/
|
|
|
Builder.isValidMessage = function(def) {
|
|
|
// Messages require a string name
|
|
|
if (typeof def["name"] !== 'string' || !Lang.NAME.test(def["name"]))
|
|
|
return false;
|
|
|
// Messages must not contain values (that'd be an enum) or methods (that'd be a service)
|
|
|
if (typeof def["values"] !== 'undefined' || typeof def["rpc"] !== 'undefined')
|
|
|
return false;
|
|
|
// Fields, enums and messages are arrays if provided
|
|
|
var i;
|
|
|
if (typeof def["fields"] !== 'undefined') {
|
|
|
if (!ProtoBuf.Util.isArray(def["fields"]))
|
|
|
return false;
|
|
|
var ids = [], id; // IDs must be unique
|
|
|
for (i=0; i<def["fields"].length; i++) {
|
|
|
if (!Builder.isValidMessageField(def["fields"][i]))
|
|
|
return false;
|
|
|
id = parseInt(def["fields"][i]["id"], 10);
|
|
|
if (ids.indexOf(id) >= 0)
|
|
|
return false;
|
|
|
ids.push(id);
|
|
|
}
|
|
|
ids = null;
|
|
|
}
|
|
|
if (typeof def["enums"] !== 'undefined') {
|
|
|
if (!ProtoBuf.Util.isArray(def["enums"]))
|
|
|
return false;
|
|
|
for (i=0; i<def["enums"].length; i++)
|
|
|
if (!Builder.isValidEnum(def["enums"][i]))
|
|
|
return false;
|
|
|
}
|
|
|
if (typeof def["messages"] !== 'undefined') {
|
|
|
if (!ProtoBuf.Util.isArray(def["messages"]))
|
|
|
return false;
|
|
|
for (i=0; i<def["messages"].length; i++)
|
|
|
if (!Builder.isValidMessage(def["messages"][i]) && !Builder.isValidExtend(def["messages"][i]))
|
|
|
return false;
|
|
|
}
|
|
|
if (typeof def["extensions"] !== 'undefined')
|
|
|
if (!ProtoBuf.Util.isArray(def["extensions"]) || def["extensions"].length !== 2 || typeof def["extensions"][0] !== 'number' || typeof def["extensions"][1] !== 'number')
|
|
|
return false;
|
|
|
return true;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Tests if a definition is a valid message field definition.
|
|
|
* @param {Object} def Definition
|
|
|
* @return {boolean} true if valid, else false
|
|
|
* @expose
|
|
|
*/
|
|
|
Builder.isValidMessageField = function(def) {
|
|
|
// Message fields require a string rule, name and type and an id
|
|
|
if (typeof def["rule"] !== 'string' || typeof def["name"] !== 'string' || typeof def["type"] !== 'string' || typeof def["id"] === 'undefined')
|
|
|
return false;
|
|
|
if (!Lang.RULE.test(def["rule"]) || !Lang.NAME.test(def["name"]) || !Lang.TYPEREF.test(def["type"]) || !Lang.ID.test(""+def["id"]))
|
|
|
return false;
|
|
|
if (typeof def["options"] !== 'undefined') {
|
|
|
// Options are objects
|
|
|
if (typeof def["options"] !== 'object')
|
|
|
return false;
|
|
|
// Options are <string,string|number|boolean>
|
|
|
var keys = Object.keys(def["options"]);
|
|
|
for (var i=0, key; i<keys.length; i++)
|
|
|
if (typeof (key = keys[i]) !== 'string' || (typeof def["options"][key] !== 'string' && typeof def["options"][key] !== 'number' && typeof def["options"][key] !== 'boolean'))
|
|
|
return false;
|
|
|
}
|
|
|
return true;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Tests if a definition is a valid enum definition.
|
|
|
* @param {Object} def Definition
|
|
|
* @return {boolean} true if valid, else false
|
|
|
* @expose
|
|
|
*/
|
|
|
Builder.isValidEnum = function(def) {
|
|
|
// Enums require a string name
|
|
|
if (typeof def["name"] !== 'string' || !Lang.NAME.test(def["name"]))
|
|
|
return false;
|
|
|
// Enums require at least one value
|
|
|
if (typeof def["values"] === 'undefined' || !ProtoBuf.Util.isArray(def["values"]) || def["values"].length == 0)
|
|
|
return false;
|
|
|
for (var i=0; i<def["values"].length; i++) {
|
|
|
// Values are objects
|
|
|
if (typeof def["values"][i] != "object")
|
|
|
return false;
|
|
|
// Values require a string name and an id
|
|
|
if (typeof def["values"][i]["name"] !== 'string' || typeof def["values"][i]["id"] === 'undefined')
|
|
|
return false;
|
|
|
if (!Lang.NAME.test(def["values"][i]["name"]) || !Lang.NEGID.test(""+def["values"][i]["id"]))
|
|
|
return false;
|
|
|
}
|
|
|
// It's not important if there are other fields because ["values"] is already unique
|
|
|
return true;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Creates ths specified protocol types at the current pointer position.
|
|
|
* @param {Array.<Object.<string,*>>} defs Messages, enums or services to create
|
|
|
* @return {ProtoBuf.Builder} this
|
|
|
* @throws {Error} If a message definition is invalid
|
|
|
* @expose
|
|
|
*/
|
|
|
BuilderPrototype.create = function(defs) {
|
|
|
if (!defs)
|
|
|
return this; // Nothing to create
|
|
|
if (!ProtoBuf.Util.isArray(defs))
|
|
|
defs = [defs];
|
|
|
if (defs.length == 0)
|
|
|
return this;
|
|
|
|
|
|
// It's quite hard to keep track of scopes and memory here, so let's do this iteratively.
|
|
|
var stack = [];
|
|
|
stack.push(defs); // One level [a, b, c]
|
|
|
while (stack.length > 0) {
|
|
|
defs = stack.pop();
|
|
|
if (ProtoBuf.Util.isArray(defs)) { // Stack always contains entire namespaces
|
|
|
while (defs.length > 0) {
|
|
|
var def = defs.shift(); // Namespace always contains an array of messages, enums and services
|
|
|
if (Builder.isValidMessage(def)) {
|
|
|
var obj = new Reflect.Message(this, this.ptr, def["name"], def["options"], def["isGroup"]);
|
|
|
// Create OneOfs
|
|
|
var oneofs = {};
|
|
|
if (def["oneofs"]) {
|
|
|
var keys = Object.keys(def["oneofs"]);
|
|
|
for (var i=0, k=keys.length; i<k; ++i)
|
|
|
obj.addChild(oneofs[keys[i]] = new Reflect.Message.OneOf(this, obj, keys[i]));
|
|
|
}
|
|
|
// Create fields
|
|
|
if (def["fields"] && def["fields"].length > 0) {
|
|
|
for (i=0, k=def["fields"].length; i<k; ++i) { // i:k=Fields
|
|
|
var fld = def['fields'][i];
|
|
|
if (obj.getChild(fld['id']) !== null)
|
|
|
throw Error("Duplicate field id in message "+obj.name+": "+fld['id']);
|
|
|
if (fld["options"]) {
|
|
|
var opts = Object.keys(fld["options"]);
|
|
|
for (var j= 0,l=opts.length; j<l; ++j) { // j:l=Option names
|
|
|
if (typeof opts[j] !== 'string')
|
|
|
throw Error("Illegal field option name in message "+obj.name+"#"+fld["name"]+": "+opts[j]);
|
|
|
if (typeof fld["options"][opts[j]] !== 'string' && typeof fld["options"][opts[j]] !== 'number' && typeof fld["options"][opts[j]] !== 'boolean')
|
|
|
throw Error("Illegal field option value in message "+obj.name+"#"+fld["name"]+"#"+opts[j]+": "+fld["options"][opts[j]]);
|
|
|
}
|
|
|
}
|
|
|
var oneof = null;
|
|
|
if (typeof fld["oneof"] === 'string') {
|
|
|
oneof = oneofs[fld["oneof"]];
|
|
|
if (typeof oneof === 'undefined')
|
|
|
throw Error("Illegal oneof in message "+obj.name+"#"+fld["name"]+": "+fld["oneof"]);
|
|
|
}
|
|
|
fld = new Reflect.Message.Field(this, obj, fld["rule"], fld["type"], fld["name"], fld["id"], fld["options"], oneof);
|
|
|
if (oneof)
|
|
|
oneof.fields.push(fld);
|
|
|
obj.addChild(fld);
|
|
|
}
|
|
|
}
|
|
|
// Push enums and messages to stack
|
|
|
var subObj = [];
|
|
|
if (typeof def["enums"] !== 'undefined' && def['enums'].length > 0)
|
|
|
for (i=0; i<def["enums"].length; i++)
|
|
|
subObj.push(def["enums"][i]);
|
|
|
if (def["messages"] && def["messages"].length > 0)
|
|
|
for (i=0; i<def["messages"].length; i++)
|
|
|
subObj.push(def["messages"][i]);
|
|
|
// Set extension range
|
|
|
if (def["extensions"]) {
|
|
|
obj.extensions = def["extensions"];
|
|
|
if (obj.extensions[0] < ProtoBuf.ID_MIN)
|
|
|
obj.extensions[0] = ProtoBuf.ID_MIN;
|
|
|
if (obj.extensions[1] > ProtoBuf.ID_MAX)
|
|
|
obj.extensions[1] = ProtoBuf.ID_MAX;
|
|
|
}
|
|
|
this.ptr.addChild(obj); // Add to current namespace
|
|
|
if (subObj.length > 0) {
|
|
|
stack.push(defs); // Push the current level back
|
|
|
defs = subObj; // Continue processing sub level
|
|
|
subObj = null;
|
|
|
this.ptr = obj; // And move the pointer to this namespace
|
|
|
obj = null;
|
|
|
continue;
|
|
|
}
|
|
|
subObj = null;
|
|
|
obj = null;
|
|
|
} else if (Builder.isValidEnum(def)) {
|
|
|
obj = new Reflect.Enum(this, this.ptr, def["name"], def["options"]);
|
|
|
for (i=0; i<def["values"].length; i++)
|
|
|
obj.addChild(new Reflect.Enum.Value(this, obj, def["values"][i]["name"], def["values"][i]["id"]));
|
|
|
this.ptr.addChild(obj);
|
|
|
obj = null;
|
|
|
} else if (Builder.isValidService(def)) {
|
|
|
obj = new Reflect.Service(this, this.ptr, def["name"], def["options"]);
|
|
|
for (i in def["rpc"])
|
|
|
if (def["rpc"].hasOwnProperty(i))
|
|
|
obj.addChild(new Reflect.Service.RPCMethod(this, obj, i, def["rpc"][i]["request"], def["rpc"][i]["response"], def["rpc"][i]["options"]));
|
|
|
this.ptr.addChild(obj);
|
|
|
obj = null;
|
|
|
} else if (Builder.isValidExtend(def)) {
|
|
|
obj = this.ptr.resolve(def["ref"]);
|
|
|
if (obj) {
|
|
|
for (i=0; i<def["fields"].length; i++) { // i=Fields
|
|
|
if (obj.getChild(def['fields'][i]['id']) !== null)
|
|
|
throw Error("Duplicate extended field id in message "+obj.name+": "+def['fields'][i]['id']);
|
|
|
if (def['fields'][i]['id'] < obj.extensions[0] || def['fields'][i]['id'] > obj.extensions[1])
|
|
|
throw Error("Illegal extended field id in message "+obj.name+": "+def['fields'][i]['id']+" ("+obj.extensions.join(' to ')+" expected)");
|
|
|
// Convert extension field names to camel case notation if the override is set
|
|
|
var name = def["fields"][i]["name"];
|
|
|
if (this.options['convertFieldsToCamelCase'])
|
|
|
name = Reflect.Message.Field._toCamelCase(def["fields"][i]["name"]);
|
|
|
// see #161: Extensions use their fully qualified name as their runtime key and...
|
|
|
fld = new Reflect.Message.ExtensionField(this, obj, def["fields"][i]["rule"], def["fields"][i]["type"], this.ptr.fqn()+'.'+name, def["fields"][i]["id"], def["fields"][i]["options"]);
|
|
|
// ...are added on top of the current namespace as an extension which is used for
|
|
|
// resolving their type later on (the extension always keeps the original name to
|
|
|
// prevent naming collisions)
|
|
|
var ext = new Reflect.Extension(this, this.ptr, def["fields"][i]["name"], fld);
|
|
|
fld.extension = ext;
|
|
|
this.ptr.addChild(ext);
|
|
|
obj.addChild(fld);
|
|
|
}
|
|
|
} else if (!/\.?google\.protobuf\./.test(def["ref"])) // Silently skip internal extensions
|
|
|
throw Error("Extended message "+def["ref"]+" is not defined");
|
|
|
} else
|
|
|
throw Error("Not a valid definition: "+JSON.stringify(def));
|
|
|
def = null;
|
|
|
}
|
|
|
// Break goes here
|
|
|
} else
|
|
|
throw Error("Not a valid namespace: "+JSON.stringify(defs));
|
|
|
defs = null;
|
|
|
this.ptr = this.ptr.parent; // This namespace is s done
|
|
|
}
|
|
|
this.resolved = false; // Require re-resolve
|
|
|
this.result = null; // Require re-build
|
|
|
return this;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Imports another definition into this builder.
|
|
|
* @param {Object.<string,*>} json Parsed import
|
|
|
* @param {(string|{root: string, file: string})=} filename Imported file name
|
|
|
* @return {ProtoBuf.Builder} this
|
|
|
* @throws {Error} If the definition or file cannot be imported
|
|
|
* @expose
|
|
|
*/
|
|
|
BuilderPrototype["import"] = function(json, filename) {
|
|
|
if (typeof filename === 'string') {
|
|
|
if (ProtoBuf.Util.IS_NODE)
|
|
|
filename = require("path")['resolve'](filename);
|
|
|
if (this.files[filename] === true) {
|
|
|
this.reset();
|
|
|
return this; // Skip duplicate imports
|
|
|
}
|
|
|
this.files[filename] = true;
|
|
|
}
|
|
|
if (!!json['imports'] && json['imports'].length > 0) {
|
|
|
var importRoot, delim = '/', resetRoot = false;
|
|
|
if (typeof filename === 'object') { // If an import root is specified, override
|
|
|
this.importRoot = filename["root"]; resetRoot = true; // ... and reset afterwards
|
|
|
importRoot = this.importRoot;
|
|
|
filename = filename["file"];
|
|
|
if (importRoot.indexOf("\\") >= 0 || filename.indexOf("\\") >= 0) delim = '\\';
|
|
|
} else if (typeof filename === 'string') {
|
|
|
if (this.importRoot) // If import root is overridden, use it
|
|
|
importRoot = this.importRoot;
|
|
|
else { // Otherwise compute from filename
|
|
|
if (filename.indexOf("/") >= 0) { // Unix
|
|
|
importRoot = filename.replace(/\/[^\/]*$/, "");
|
|
|
if (/* /file.proto */ importRoot === "")
|
|
|
importRoot = "/";
|
|
|
} else if (filename.indexOf("\\") >= 0) { // Windows
|
|
|
importRoot = filename.replace(/\\[^\\]*$/, "");
|
|
|
delim = '\\';
|
|
|
} else
|
|
|
importRoot = ".";
|
|
|
}
|
|
|
} else
|
|
|
importRoot = null;
|
|
|
|
|
|
for (var i=0; i<json['imports'].length; i++) {
|
|
|
if (typeof json['imports'][i] === 'string') { // Import file
|
|
|
if (!importRoot)
|
|
|
throw Error("Cannot determine import root: File name is unknown");
|
|
|
var importFilename = json['imports'][i];
|
|
|
if (/^google\/protobuf\//.test(importFilename))
|
|
|
continue; // Not needed and therefore not used
|
|
|
importFilename = importRoot+delim+importFilename;
|
|
|
if (this.files[importFilename] === true)
|
|
|
continue; // Already imported
|
|
|
if (/\.proto$/i.test(importFilename) && !ProtoBuf.DotProto) // If this is a NOPARSE build
|
|
|
importFilename = importFilename.replace(/\.proto$/, ".json"); // always load the JSON file
|
|
|
var contents = ProtoBuf.Util.fetch(importFilename);
|
|
|
if (contents === null)
|
|
|
throw Error("Failed to import '"+importFilename+"' in '"+filename+"': File not found");
|
|
|
if (/\.json$/i.test(importFilename)) // Always possible
|
|
|
this["import"](JSON.parse(contents+""), importFilename); // May throw
|
|
|
else
|
|
|
this["import"]((new ProtoBuf.DotProto.Parser(contents+"")).parse(), importFilename); // May throw
|
|
|
} else // Import structure
|
|
|
if (!filename)
|
|
|
this["import"](json['imports'][i]);
|
|
|
else if (/\.(\w+)$/.test(filename)) // With extension: Append _importN to the name portion to make it unique
|
|
|
this["import"](json['imports'][i], filename.replace(/^(.+)\.(\w+)$/, function($0, $1, $2) { return $1+"_import"+i+"."+$2; }));
|
|
|
else // Without extension: Append _importN to make it unique
|
|
|
this["import"](json['imports'][i], filename+"_import"+i);
|
|
|
}
|
|
|
if (resetRoot) // Reset import root override when all imports are done
|
|
|
this.importRoot = null;
|
|
|
}
|
|
|
if (json['messages']) {
|
|
|
if (json['package'])
|
|
|
this.define(json['package'], json["options"]);
|
|
|
this.create(json['messages']);
|
|
|
this.reset();
|
|
|
}
|
|
|
if (json['enums']) {
|
|
|
if (json['package'])
|
|
|
this.define(json['package'], json["options"]);
|
|
|
this.create(json['enums']);
|
|
|
this.reset();
|
|
|
}
|
|
|
if (json['services']) {
|
|
|
if (json['package'])
|
|
|
this.define(json['package'], json["options"]);
|
|
|
this.create(json['services']);
|
|
|
this.reset();
|
|
|
}
|
|
|
if (json['extends']) {
|
|
|
if (json['package'])
|
|
|
this.define(json['package'], json["options"]);
|
|
|
this.create(json['extends']);
|
|
|
this.reset();
|
|
|
}
|
|
|
return this;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Tests if a definition is a valid service definition.
|
|
|
* @param {Object} def Definition
|
|
|
* @return {boolean} true if valid, else false
|
|
|
* @expose
|
|
|
*/
|
|
|
Builder.isValidService = function(def) {
|
|
|
// Services require a string name and an rpc object
|
|
|
return !(typeof def["name"] !== 'string' || !Lang.NAME.test(def["name"]) || typeof def["rpc"] !== 'object');
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Tests if a definition is a valid extension.
|
|
|
* @param {Object} def Definition
|
|
|
* @returns {boolean} true if valid, else false
|
|
|
* @expose
|
|
|
*/
|
|
|
Builder.isValidExtend = function(def) {
|
|
|
if (typeof def["ref"] !== 'string' || !Lang.TYPEREF.test(def["ref"]))
|
|
|
return false;
|
|
|
var i;
|
|
|
if (typeof def["fields"] !== 'undefined') {
|
|
|
if (!ProtoBuf.Util.isArray(def["fields"]))
|
|
|
return false;
|
|
|
var ids = [], id; // IDs must be unique (does not yet test for the extended message's ids)
|
|
|
for (i=0; i<def["fields"].length; i++) {
|
|
|
if (!Builder.isValidMessageField(def["fields"][i]))
|
|
|
return false;
|
|
|
id = parseInt(def["id"], 10);
|
|
|
if (ids.indexOf(id) >= 0)
|
|
|
return false;
|
|
|
ids.push(id);
|
|
|
}
|
|
|
ids = null;
|
|
|
}
|
|
|
return true;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Resolves all namespace objects.
|
|
|
* @throws {Error} If a type cannot be resolved
|
|
|
* @expose
|
|
|
*/
|
|
|
BuilderPrototype.resolveAll = function() {
|
|
|
// Resolve all reflected objects
|
|
|
var res;
|
|
|
if (this.ptr == null || typeof this.ptr.type === 'object')
|
|
|
return; // Done (already resolved)
|
|
|
if (this.ptr instanceof Reflect.Namespace) {
|
|
|
// Build all children
|
|
|
var children = this.ptr.children;
|
|
|
for (var i= 0, k=children.length; i<k; ++i)
|
|
|
this.ptr = children[i],
|
|
|
this.resolveAll();
|
|
|
} else if (this.ptr instanceof Reflect.Message.Field) {
|
|
|
if (!Lang.TYPE.test(this.ptr.type)) { // Resolve type...
|
|
|
if (!Lang.TYPEREF.test(this.ptr.type))
|
|
|
throw Error("Illegal type reference in "+this.ptr.toString(true)+": "+this.ptr.type);
|
|
|
res = (this.ptr instanceof Reflect.Message.ExtensionField ? this.ptr.extension.parent : this.ptr.parent).resolve(this.ptr.type, true);
|
|
|
if (!res)
|
|
|
throw Error("Unresolvable type reference in "+this.ptr.toString(true)+": "+this.ptr.type);
|
|
|
this.ptr.resolvedType = res;
|
|
|
if (res instanceof Reflect.Enum)
|
|
|
this.ptr.type = ProtoBuf.TYPES["enum"];
|
|
|
else if (res instanceof Reflect.Message)
|
|
|
this.ptr.type = res.isGroup ? ProtoBuf.TYPES["group"] : ProtoBuf.TYPES["message"];
|
|
|
else
|
|
|
throw Error("Illegal type reference in "+this.ptr.toString(true)+": "+this.ptr.type);
|
|
|
} else
|
|
|
this.ptr.type = ProtoBuf.TYPES[this.ptr.type];
|
|
|
} else if (this.ptr instanceof ProtoBuf.Reflect.Enum.Value) {
|
|
|
// No need to build enum values (built in enum)
|
|
|
} else if (this.ptr instanceof ProtoBuf.Reflect.Service.Method) {
|
|
|
if (this.ptr instanceof ProtoBuf.Reflect.Service.RPCMethod) {
|
|
|
res = this.ptr.parent.resolve(this.ptr.requestName);
|
|
|
if (!res || !(res instanceof ProtoBuf.Reflect.Message))
|
|
|
throw Error("Illegal type reference in "+this.ptr.toString(true)+": "+this.ptr.requestName);
|
|
|
this.ptr.resolvedRequestType = res;
|
|
|
res = this.ptr.parent.resolve(this.ptr.responseName);
|
|
|
if (!res || !(res instanceof ProtoBuf.Reflect.Message))
|
|
|
throw Error("Illegal type reference in "+this.ptr.toString(true)+": "+this.ptr.responseName);
|
|
|
this.ptr.resolvedResponseType = res;
|
|
|
} else {
|
|
|
// Should not happen as nothing else is implemented
|
|
|
throw Error("Illegal service type in "+this.ptr.toString(true));
|
|
|
}
|
|
|
} else if (!(this.ptr instanceof ProtoBuf.Reflect.Message.OneOf) && !(this.ptr instanceof ProtoBuf.Reflect.Extension))
|
|
|
throw Error("Illegal object in namespace: "+typeof(this.ptr)+":"+this.ptr);
|
|
|
this.reset();
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Builds the protocol. This will first try to resolve all definitions and, if this has been successful,
|
|
|
* return the built package.
|
|
|
* @param {string=} path Specifies what to return. If omitted, the entire namespace will be returned.
|
|
|
* @return {ProtoBuf.Builder.Message|Object.<string,*>}
|
|
|
* @throws {Error} If a type could not be resolved
|
|
|
* @expose
|
|
|
*/
|
|
|
BuilderPrototype.build = function(path) {
|
|
|
this.reset();
|
|
|
if (!this.resolved)
|
|
|
this.resolveAll(),
|
|
|
this.resolved = true,
|
|
|
this.result = null; // Require re-build
|
|
|
if (this.result == null) // (Re-)Build
|
|
|
this.result = this.ns.build();
|
|
|
if (!path)
|
|
|
return this.result;
|
|
|
else {
|
|
|
var part = path.split(".");
|
|
|
var ptr = this.result; // Build namespace pointer (no hasChild etc.)
|
|
|
for (var i=0; i<part.length; i++)
|
|
|
if (ptr[part[i]])
|
|
|
ptr = ptr[part[i]];
|
|
|
else {
|
|
|
ptr = null;
|
|
|
break;
|
|
|
}
|
|
|
return ptr;
|
|
|
}
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Similar to {@link ProtoBuf.Builder#build}, but looks up the internal reflection descriptor.
|
|
|
* @param {string=} path Specifies what to return. If omitted, the entire namespace wiil be returned.
|
|
|
* @return {ProtoBuf.Reflect.T} Reflection descriptor or `null` if not found
|
|
|
*/
|
|
|
BuilderPrototype.lookup = function(path) {
|
|
|
return path ? this.ns.resolve(path) : this.ns;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Returns a string representation of this object.
|
|
|
* @return {string} String representation as of "Builder"
|
|
|
* @expose
|
|
|
*/
|
|
|
BuilderPrototype.toString = function() {
|
|
|
return "Builder";
|
|
|
};
|
|
|
|
|
|
// Pseudo types documented in Reflect.js.
|
|
|
// Exist for the sole purpose of being able to "... instanceof ProtoBuf.Builder.Message" etc.
|
|
|
Builder.Message = function() {};
|
|
|
Builder.Service = function() {};
|
|
|
|
|
|
return Builder;
|
|
|
|
|
|
})(ProtoBuf, ProtoBuf.Lang, ProtoBuf.Reflect);
|
|
|
|
|
|
|
|
|
/**
|
|
|
* Loads a .proto string and returns the Builder.
|
|
|
* @param {string} proto .proto file contents
|
|
|
* @param {(ProtoBuf.Builder|string|{root: string, file: string})=} builder Builder to append to. Will create a new one if omitted.
|
|
|
* @param {(string|{root: string, file: string})=} filename The corresponding file name if known. Must be specified for imports.
|
|
|
* @return {ProtoBuf.Builder} Builder to create new messages
|
|
|
* @throws {Error} If the definition cannot be parsed or built
|
|
|
* @expose
|
|
|
*/
|
|
|
ProtoBuf.loadProto = function(proto, builder, filename) {
|
|
|
if (typeof builder === 'string' || (builder && typeof builder["file"] === 'string' && typeof builder["root"] === 'string'))
|
|
|
filename = builder,
|
|
|
builder = undefined;
|
|
|
return ProtoBuf.loadJson((new ProtoBuf.DotProto.Parser(proto)).parse(), builder, filename);
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Loads a .proto string and returns the Builder. This is an alias of {@link ProtoBuf.loadProto}.
|
|
|
* @function
|
|
|
* @param {string} proto .proto file contents
|
|
|
* @param {(ProtoBuf.Builder|string)=} builder Builder to append to. Will create a new one if omitted.
|
|
|
* @param {(string|{root: string, file: string})=} filename The corresponding file name if known. Must be specified for imports.
|
|
|
* @return {ProtoBuf.Builder} Builder to create new messages
|
|
|
* @throws {Error} If the definition cannot be parsed or built
|
|
|
* @expose
|
|
|
*/
|
|
|
ProtoBuf.protoFromString = ProtoBuf.loadProto; // Legacy
|
|
|
|
|
|
/**
|
|
|
* Loads a .proto file and returns the Builder.
|
|
|
* @param {string|{root: string, file: string}} filename Path to proto file or an object specifying 'file' with
|
|
|
* an overridden 'root' path for all imported files.
|
|
|
* @param {function(?Error, !ProtoBuf.Builder=)=} callback Callback that will receive `null` as the first and
|
|
|
* the Builder as its second argument on success, otherwise the error as its first argument. If omitted, the
|
|
|
* file will be read synchronously and this function will return the Builder.
|
|
|
* @param {ProtoBuf.Builder=} builder Builder to append to. Will create a new one if omitted.
|
|
|
* @return {?ProtoBuf.Builder|undefined} The Builder if synchronous (no callback specified, will be NULL if the
|
|
|
* request has failed), else undefined
|
|
|
* @expose
|
|
|
*/
|
|
|
ProtoBuf.loadProtoFile = function(filename, callback, builder) {
|
|
|
if (callback && typeof callback === 'object')
|
|
|
builder = callback,
|
|
|
callback = null;
|
|
|
else if (!callback || typeof callback !== 'function')
|
|
|
callback = null;
|
|
|
if (callback)
|
|
|
return ProtoBuf.Util.fetch(typeof filename === 'string' ? filename : filename["root"]+"/"+filename["file"], function(contents) {
|
|
|
if (contents === null) {
|
|
|
callback(Error("Failed to fetch file"));
|
|
|
return;
|
|
|
}
|
|
|
try {
|
|
|
callback(null, ProtoBuf.loadProto(contents, builder, filename));
|
|
|
} catch (e) {
|
|
|
callback(e);
|
|
|
}
|
|
|
});
|
|
|
var contents = ProtoBuf.Util.fetch(typeof filename === 'object' ? filename["root"]+"/"+filename["file"] : filename);
|
|
|
return contents === null ? null : ProtoBuf.loadProto(contents, builder, filename);
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Loads a .proto file and returns the Builder. This is an alias of {@link ProtoBuf.loadProtoFile}.
|
|
|
* @function
|
|
|
* @param {string|{root: string, file: string}} filename Path to proto file or an object specifying 'file' with
|
|
|
* an overridden 'root' path for all imported files.
|
|
|
* @param {function(?Error, !ProtoBuf.Builder=)=} callback Callback that will receive `null` as the first and
|
|
|
* the Builder as its second argument on success, otherwise the error as its first argument. If omitted, the
|
|
|
* file will be read synchronously and this function will return the Builder.
|
|
|
* @param {ProtoBuf.Builder=} builder Builder to append to. Will create a new one if omitted.
|
|
|
* @return {!ProtoBuf.Builder|undefined} The Builder if synchronous (no callback specified, will be NULL if the
|
|
|
* request has failed), else undefined
|
|
|
* @expose
|
|
|
*/
|
|
|
ProtoBuf.protoFromFile = ProtoBuf.loadProtoFile; // Legacy
|
|
|
|
|
|
|
|
|
/**
|
|
|
* Constructs a new empty Builder.
|
|
|
* @param {Object.<string,*>=} options Builder options, defaults to global options set on ProtoBuf
|
|
|
* @return {!ProtoBuf.Builder} Builder
|
|
|
* @expose
|
|
|
*/
|
|
|
ProtoBuf.newBuilder = function(options) {
|
|
|
options = options || {};
|
|
|
if (typeof options['convertFieldsToCamelCase'] === 'undefined')
|
|
|
options['convertFieldsToCamelCase'] = ProtoBuf.convertFieldsToCamelCase;
|
|
|
if (typeof options['populateAccessors'] === 'undefined')
|
|
|
options['populateAccessors'] = ProtoBuf.populateAccessors;
|
|
|
return new ProtoBuf.Builder(options);
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Loads a .json definition and returns the Builder.
|
|
|
* @param {!*|string} json JSON definition
|
|
|
* @param {(ProtoBuf.Builder|string|{root: string, file: string})=} builder Builder to append to. Will create a new one if omitted.
|
|
|
* @param {(string|{root: string, file: string})=} filename The corresponding file name if known. Must be specified for imports.
|
|
|
* @return {ProtoBuf.Builder} Builder to create new messages
|
|
|
* @throws {Error} If the definition cannot be parsed or built
|
|
|
* @expose
|
|
|
*/
|
|
|
ProtoBuf.loadJson = function(json, builder, filename) {
|
|
|
if (typeof builder === 'string' || (builder && typeof builder["file"] === 'string' && typeof builder["root"] === 'string'))
|
|
|
filename = builder,
|
|
|
builder = null;
|
|
|
if (!builder || typeof builder !== 'object')
|
|
|
builder = ProtoBuf.newBuilder();
|
|
|
if (typeof json === 'string')
|
|
|
json = JSON.parse(json);
|
|
|
builder["import"](json, filename);
|
|
|
builder.resolveAll();
|
|
|
builder.build();
|
|
|
return builder;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Loads a .json file and returns the Builder.
|
|
|
* @param {string|!{root: string, file: string}} filename Path to json file or an object specifying 'file' with
|
|
|
* an overridden 'root' path for all imported files.
|
|
|
* @param {function(?Error, !ProtoBuf.Builder=)=} callback Callback that will receive `null` as the first and
|
|
|
* the Builder as its second argument on success, otherwise the error as its first argument. If omitted, the
|
|
|
* file will be read synchronously and this function will return the Builder.
|
|
|
* @param {ProtoBuf.Builder=} builder Builder to append to. Will create a new one if omitted.
|
|
|
* @return {?ProtoBuf.Builder|undefined} The Builder if synchronous (no callback specified, will be NULL if the
|
|
|
* request has failed), else undefined
|
|
|
* @expose
|
|
|
*/
|
|
|
ProtoBuf.loadJsonFile = function(filename, callback, builder) {
|
|
|
if (callback && typeof callback === 'object')
|
|
|
builder = callback,
|
|
|
callback = null;
|
|
|
else if (!callback || typeof callback !== 'function')
|
|
|
callback = null;
|
|
|
if (callback)
|
|
|
return ProtoBuf.Util.fetch(typeof filename === 'string' ? filename : filename["root"]+"/"+filename["file"], function(contents) {
|
|
|
if (contents === null) {
|
|
|
callback(Error("Failed to fetch file"));
|
|
|
return;
|
|
|
}
|
|
|
try {
|
|
|
callback(null, ProtoBuf.loadJson(JSON.parse(contents), builder, filename));
|
|
|
} catch (e) {
|
|
|
callback(e);
|
|
|
}
|
|
|
});
|
|
|
var contents = ProtoBuf.Util.fetch(typeof filename === 'object' ? filename["root"]+"/"+filename["file"] : filename);
|
|
|
return contents === null ? null : ProtoBuf.loadJson(JSON.parse(contents), builder, filename);
|
|
|
};
|
|
|
|
|
|
return ProtoBuf;
|
|
|
}
|
|
|
|
|
|
/* CommonJS */ if (typeof require === 'function' && typeof module === 'object' && module && typeof exports === 'object' && exports)
|
|
|
module['exports'] = init(require("bytebuffer"));
|
|
|
/* AMD */ else if (typeof define === 'function' && define["amd"])
|
|
|
define(["ByteBuffer"], init);
|
|
|
/* Global */ else
|
|
|
(global["dcodeIO"] = global["dcodeIO"] || {})["ProtoBuf"] = init(global["dcodeIO"]["ByteBuffer"]);
|
|
|
|
|
|
})(this);
|
|
|
|
|
|
/*!
|
|
|
* mustache.js - Logic-less {{mustache}} templates with JavaScript
|
|
|
* http://github.com/janl/mustache.js
|
|
|
*/
|
|
|
|
|
|
/*global define: false*/
|
|
|
|
|
|
(function (root, factory) {
|
|
|
if (typeof exports === "object" && exports) {
|
|
|
factory(exports); // CommonJS
|
|
|
} else {
|
|
|
var mustache = {};
|
|
|
factory(mustache);
|
|
|
if (typeof define === "function" && define.amd) {
|
|
|
define(mustache); // AMD
|
|
|
} else {
|
|
|
root.Mustache = mustache; // <script>
|
|
|
}
|
|
|
}
|
|
|
}(this, function (mustache) {
|
|
|
|
|
|
// Workaround for https://issues.apache.org/jira/browse/COUCHDB-577
|
|
|
// See https://github.com/janl/mustache.js/issues/189
|
|
|
var RegExp_test = RegExp.prototype.test;
|
|
|
function testRegExp(re, string) {
|
|
|
return RegExp_test.call(re, string);
|
|
|
}
|
|
|
|
|
|
var nonSpaceRe = /\S/;
|
|
|
function isWhitespace(string) {
|
|
|
return !testRegExp(nonSpaceRe, string);
|
|
|
}
|
|
|
|
|
|
var Object_toString = Object.prototype.toString;
|
|
|
var isArray = Array.isArray || function (object) {
|
|
|
return Object_toString.call(object) === '[object Array]';
|
|
|
};
|
|
|
|
|
|
function isFunction(object) {
|
|
|
return typeof object === 'function';
|
|
|
}
|
|
|
|
|
|
function escapeRegExp(string) {
|
|
|
return string.replace(/[\-\[\]{}()*+?.,\\\^$|#\s]/g, "\\$&");
|
|
|
}
|
|
|
|
|
|
var entityMap = {
|
|
|
"&": "&",
|
|
|
"<": "<",
|
|
|
">": ">",
|
|
|
'"': '"',
|
|
|
"'": ''',
|
|
|
"/": '/'
|
|
|
};
|
|
|
|
|
|
function escapeHtml(string) {
|
|
|
return String(string).replace(/[&<>"'\/]/g, function (s) {
|
|
|
return entityMap[s];
|
|
|
});
|
|
|
}
|
|
|
|
|
|
function escapeTags(tags) {
|
|
|
if (!isArray(tags) || tags.length !== 2) {
|
|
|
throw new Error('Invalid tags: ' + tags);
|
|
|
}
|
|
|
|
|
|
return [
|
|
|
new RegExp(escapeRegExp(tags[0]) + "\\s*"),
|
|
|
new RegExp("\\s*" + escapeRegExp(tags[1]))
|
|
|
];
|
|
|
}
|
|
|
|
|
|
var whiteRe = /\s*/;
|
|
|
var spaceRe = /\s+/;
|
|
|
var equalsRe = /\s*=/;
|
|
|
var curlyRe = /\s*\}/;
|
|
|
var tagRe = /#|\^|\/|>|\{|&|=|!/;
|
|
|
|
|
|
/**
|
|
|
* Breaks up the given `template` string into a tree of tokens. If the `tags`
|
|
|
* argument is given here it must be an array with two string values: the
|
|
|
* opening and closing tags used in the template (e.g. [ "<%", "%>" ]). Of
|
|
|
* course, the default is to use mustaches (i.e. mustache.tags).
|
|
|
*
|
|
|
* A token is an array with at least 4 elements. The first element is the
|
|
|
* mustache symbol that was used inside the tag, e.g. "#" or "&". If the tag
|
|
|
* did not contain a symbol (i.e. {{myValue}}) this element is "name". For
|
|
|
* all text that appears outside a symbol this element is "text".
|
|
|
*
|
|
|
* The second element of a token is its "value". For mustache tags this is
|
|
|
* whatever else was inside the tag besides the opening symbol. For text tokens
|
|
|
* this is the text itself.
|
|
|
*
|
|
|
* The third and fourth elements of the token are the start and end indices,
|
|
|
* respectively, of the token in the original template.
|
|
|
*
|
|
|
* Tokens that are the root node of a subtree contain two more elements: 1) an
|
|
|
* array of tokens in the subtree and 2) the index in the original template at
|
|
|
* which the closing tag for that section begins.
|
|
|
*/
|
|
|
function parseTemplate(template, tags) {
|
|
|
tags = tags || mustache.tags;
|
|
|
template = template || '';
|
|
|
|
|
|
if (typeof tags === 'string') {
|
|
|
tags = tags.split(spaceRe);
|
|
|
}
|
|
|
|
|
|
var tagRes = escapeTags(tags);
|
|
|
var scanner = new Scanner(template);
|
|
|
|
|
|
var sections = []; // Stack to hold section tokens
|
|
|
var tokens = []; // Buffer to hold the tokens
|
|
|
var spaces = []; // Indices of whitespace tokens on the current line
|
|
|
var hasTag = false; // Is there a {{tag}} on the current line?
|
|
|
var nonSpace = false; // Is there a non-space char on the current line?
|
|
|
|
|
|
// Strips all whitespace tokens array for the current line
|
|
|
// if there was a {{#tag}} on it and otherwise only space.
|
|
|
function stripSpace() {
|
|
|
if (hasTag && !nonSpace) {
|
|
|
while (spaces.length) {
|
|
|
delete tokens[spaces.pop()];
|
|
|
}
|
|
|
} else {
|
|
|
spaces = [];
|
|
|
}
|
|
|
|
|
|
hasTag = false;
|
|
|
nonSpace = false;
|
|
|
}
|
|
|
|
|
|
var start, type, value, chr, token, openSection;
|
|
|
while (!scanner.eos()) {
|
|
|
start = scanner.pos;
|
|
|
|
|
|
// Match any text between tags.
|
|
|
value = scanner.scanUntil(tagRes[0]);
|
|
|
if (value) {
|
|
|
for (var i = 0, len = value.length; i < len; ++i) {
|
|
|
chr = value.charAt(i);
|
|
|
|
|
|
if (isWhitespace(chr)) {
|
|
|
spaces.push(tokens.length);
|
|
|
} else {
|
|
|
nonSpace = true;
|
|
|
}
|
|
|
|
|
|
tokens.push(['text', chr, start, start + 1]);
|
|
|
start += 1;
|
|
|
|
|
|
// Check for whitespace on the current line.
|
|
|
if (chr === '\n') {
|
|
|
stripSpace();
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
|
|
|
// Match the opening tag.
|
|
|
if (!scanner.scan(tagRes[0])) break;
|
|
|
hasTag = true;
|
|
|
|
|
|
// Get the tag type.
|
|
|
type = scanner.scan(tagRe) || 'name';
|
|
|
scanner.scan(whiteRe);
|
|
|
|
|
|
// Get the tag value.
|
|
|
if (type === '=') {
|
|
|
value = scanner.scanUntil(equalsRe);
|
|
|
scanner.scan(equalsRe);
|
|
|
scanner.scanUntil(tagRes[1]);
|
|
|
} else if (type === '{') {
|
|
|
value = scanner.scanUntil(new RegExp('\\s*' + escapeRegExp('}' + tags[1])));
|
|
|
scanner.scan(curlyRe);
|
|
|
scanner.scanUntil(tagRes[1]);
|
|
|
type = '&';
|
|
|
} else {
|
|
|
value = scanner.scanUntil(tagRes[1]);
|
|
|
}
|
|
|
|
|
|
// Match the closing tag.
|
|
|
if (!scanner.scan(tagRes[1])) {
|
|
|
throw new Error('Unclosed tag at ' + scanner.pos);
|
|
|
}
|
|
|
|
|
|
token = [ type, value, start, scanner.pos ];
|
|
|
tokens.push(token);
|
|
|
|
|
|
if (type === '#' || type === '^') {
|
|
|
sections.push(token);
|
|
|
} else if (type === '/') {
|
|
|
// Check section nesting.
|
|
|
openSection = sections.pop();
|
|
|
|
|
|
if (!openSection) {
|
|
|
throw new Error('Unopened section "' + value + '" at ' + start);
|
|
|
}
|
|
|
if (openSection[1] !== value) {
|
|
|
throw new Error('Unclosed section "' + openSection[1] + '" at ' + start);
|
|
|
}
|
|
|
} else if (type === 'name' || type === '{' || type === '&') {
|
|
|
nonSpace = true;
|
|
|
} else if (type === '=') {
|
|
|
// Set the tags for the next time around.
|
|
|
tagRes = escapeTags(tags = value.split(spaceRe));
|
|
|
}
|
|
|
}
|
|
|
|
|
|
// Make sure there are no open sections when we're done.
|
|
|
openSection = sections.pop();
|
|
|
if (openSection) {
|
|
|
throw new Error('Unclosed section "' + openSection[1] + '" at ' + scanner.pos);
|
|
|
}
|
|
|
|
|
|
return nestTokens(squashTokens(tokens));
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* Combines the values of consecutive text tokens in the given `tokens` array
|
|
|
* to a single token.
|
|
|
*/
|
|
|
function squashTokens(tokens) {
|
|
|
var squashedTokens = [];
|
|
|
|
|
|
var token, lastToken;
|
|
|
for (var i = 0, len = tokens.length; i < len; ++i) {
|
|
|
token = tokens[i];
|
|
|
|
|
|
if (token) {
|
|
|
if (token[0] === 'text' && lastToken && lastToken[0] === 'text') {
|
|
|
lastToken[1] += token[1];
|
|
|
lastToken[3] = token[3];
|
|
|
} else {
|
|
|
squashedTokens.push(token);
|
|
|
lastToken = token;
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
|
|
|
return squashedTokens;
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* Forms the given array of `tokens` into a nested tree structure where
|
|
|
* tokens that represent a section have two additional items: 1) an array of
|
|
|
* all tokens that appear in that section and 2) the index in the original
|
|
|
* template that represents the end of that section.
|
|
|
*/
|
|
|
function nestTokens(tokens) {
|
|
|
var nestedTokens = [];
|
|
|
var collector = nestedTokens;
|
|
|
var sections = [];
|
|
|
|
|
|
var token, section;
|
|
|
for (var i = 0, len = tokens.length; i < len; ++i) {
|
|
|
token = tokens[i];
|
|
|
|
|
|
switch (token[0]) {
|
|
|
case '#':
|
|
|
case '^':
|
|
|
collector.push(token);
|
|
|
sections.push(token);
|
|
|
collector = token[4] = [];
|
|
|
break;
|
|
|
case '/':
|
|
|
section = sections.pop();
|
|
|
section[5] = token[2];
|
|
|
collector = sections.length > 0 ? sections[sections.length - 1][4] : nestedTokens;
|
|
|
break;
|
|
|
default:
|
|
|
collector.push(token);
|
|
|
}
|
|
|
}
|
|
|
|
|
|
return nestedTokens;
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* A simple string scanner that is used by the template parser to find
|
|
|
* tokens in template strings.
|
|
|
*/
|
|
|
function Scanner(string) {
|
|
|
this.string = string;
|
|
|
this.tail = string;
|
|
|
this.pos = 0;
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* Returns `true` if the tail is empty (end of string).
|
|
|
*/
|
|
|
Scanner.prototype.eos = function () {
|
|
|
return this.tail === "";
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Tries to match the given regular expression at the current position.
|
|
|
* Returns the matched text if it can match, the empty string otherwise.
|
|
|
*/
|
|
|
Scanner.prototype.scan = function (re) {
|
|
|
var match = this.tail.match(re);
|
|
|
|
|
|
if (match && match.index === 0) {
|
|
|
var string = match[0];
|
|
|
this.tail = this.tail.substring(string.length);
|
|
|
this.pos += string.length;
|
|
|
return string;
|
|
|
}
|
|
|
|
|
|
return "";
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Skips all text until the given regular expression can be matched. Returns
|
|
|
* the skipped string, which is the entire tail if no match can be made.
|
|
|
*/
|
|
|
Scanner.prototype.scanUntil = function (re) {
|
|
|
var index = this.tail.search(re), match;
|
|
|
|
|
|
switch (index) {
|
|
|
case -1:
|
|
|
match = this.tail;
|
|
|
this.tail = "";
|
|
|
break;
|
|
|
case 0:
|
|
|
match = "";
|
|
|
break;
|
|
|
default:
|
|
|
match = this.tail.substring(0, index);
|
|
|
this.tail = this.tail.substring(index);
|
|
|
}
|
|
|
|
|
|
this.pos += match.length;
|
|
|
|
|
|
return match;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Represents a rendering context by wrapping a view object and
|
|
|
* maintaining a reference to the parent context.
|
|
|
*/
|
|
|
function Context(view, parentContext) {
|
|
|
this.view = view == null ? {} : view;
|
|
|
this.cache = { '.': this.view };
|
|
|
this.parent = parentContext;
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* Creates a new context using the given view with this context
|
|
|
* as the parent.
|
|
|
*/
|
|
|
Context.prototype.push = function (view) {
|
|
|
return new Context(view, this);
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Returns the value of the given name in this context, traversing
|
|
|
* up the context hierarchy if the value is absent in this context's view.
|
|
|
*/
|
|
|
Context.prototype.lookup = function (name) {
|
|
|
var value;
|
|
|
if (name in this.cache) {
|
|
|
value = this.cache[name];
|
|
|
} else {
|
|
|
var context = this;
|
|
|
|
|
|
while (context) {
|
|
|
if (name.indexOf('.') > 0) {
|
|
|
value = context.view;
|
|
|
|
|
|
var names = name.split('.'), i = 0;
|
|
|
while (value != null && i < names.length) {
|
|
|
value = value[names[i++]];
|
|
|
}
|
|
|
} else {
|
|
|
value = context.view[name];
|
|
|
}
|
|
|
|
|
|
if (value != null) break;
|
|
|
|
|
|
context = context.parent;
|
|
|
}
|
|
|
|
|
|
this.cache[name] = value;
|
|
|
}
|
|
|
|
|
|
if (isFunction(value)) {
|
|
|
value = value.call(this.view);
|
|
|
}
|
|
|
|
|
|
return value;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* A Writer knows how to take a stream of tokens and render them to a
|
|
|
* string, given a context. It also maintains a cache of templates to
|
|
|
* avoid the need to parse the same template twice.
|
|
|
*/
|
|
|
function Writer() {
|
|
|
this.cache = {};
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* Clears all cached templates in this writer.
|
|
|
*/
|
|
|
Writer.prototype.clearCache = function () {
|
|
|
this.cache = {};
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Parses and caches the given `template` and returns the array of tokens
|
|
|
* that is generated from the parse.
|
|
|
*/
|
|
|
Writer.prototype.parse = function (template, tags) {
|
|
|
var cache = this.cache;
|
|
|
var tokens = cache[template];
|
|
|
|
|
|
if (tokens == null) {
|
|
|
tokens = cache[template] = parseTemplate(template, tags);
|
|
|
}
|
|
|
|
|
|
return tokens;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* High-level method that is used to render the given `template` with
|
|
|
* the given `view`.
|
|
|
*
|
|
|
* The optional `partials` argument may be an object that contains the
|
|
|
* names and templates of partials that are used in the template. It may
|
|
|
* also be a function that is used to load partial templates on the fly
|
|
|
* that takes a single argument: the name of the partial.
|
|
|
*/
|
|
|
Writer.prototype.render = function (template, view, partials) {
|
|
|
var tokens = this.parse(template);
|
|
|
var context = (view instanceof Context) ? view : new Context(view);
|
|
|
return this.renderTokens(tokens, context, partials, template);
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Low-level method that renders the given array of `tokens` using
|
|
|
* the given `context` and `partials`.
|
|
|
*
|
|
|
* Note: The `originalTemplate` is only ever used to extract the portion
|
|
|
* of the original template that was contained in a higher-order section.
|
|
|
* If the template doesn't use higher-order sections, this argument may
|
|
|
* be omitted.
|
|
|
*/
|
|
|
Writer.prototype.renderTokens = function (tokens, context, partials, originalTemplate) {
|
|
|
var buffer = '';
|
|
|
|
|
|
// This function is used to render an arbitrary template
|
|
|
// in the current context by higher-order sections.
|
|
|
var self = this;
|
|
|
function subRender(template) {
|
|
|
return self.render(template, context, partials);
|
|
|
}
|
|
|
|
|
|
var token, value;
|
|
|
for (var i = 0, len = tokens.length; i < len; ++i) {
|
|
|
token = tokens[i];
|
|
|
|
|
|
switch (token[0]) {
|
|
|
case '#':
|
|
|
value = context.lookup(token[1]);
|
|
|
if (!value) continue;
|
|
|
|
|
|
if (isArray(value)) {
|
|
|
for (var j = 0, jlen = value.length; j < jlen; ++j) {
|
|
|
buffer += this.renderTokens(token[4], context.push(value[j]), partials, originalTemplate);
|
|
|
}
|
|
|
} else if (typeof value === 'object' || typeof value === 'string') {
|
|
|
buffer += this.renderTokens(token[4], context.push(value), partials, originalTemplate);
|
|
|
} else if (isFunction(value)) {
|
|
|
if (typeof originalTemplate !== 'string') {
|
|
|
throw new Error('Cannot use higher-order sections without the original template');
|
|
|
}
|
|
|
|
|
|
// Extract the portion of the original template that the section contains.
|
|
|
value = value.call(context.view, originalTemplate.slice(token[3], token[5]), subRender);
|
|
|
|
|
|
if (value != null) buffer += value;
|
|
|
} else {
|
|
|
buffer += this.renderTokens(token[4], context, partials, originalTemplate);
|
|
|
}
|
|
|
|
|
|
break;
|
|
|
case '^':
|
|
|
value = context.lookup(token[1]);
|
|
|
|
|
|
// Use JavaScript's definition of falsy. Include empty arrays.
|
|
|
// See https://github.com/janl/mustache.js/issues/186
|
|
|
if (!value || (isArray(value) && value.length === 0)) {
|
|
|
buffer += this.renderTokens(token[4], context, partials, originalTemplate);
|
|
|
}
|
|
|
|
|
|
break;
|
|
|
case '>':
|
|
|
if (!partials) continue;
|
|
|
value = isFunction(partials) ? partials(token[1]) : partials[token[1]];
|
|
|
if (value != null) buffer += this.renderTokens(this.parse(value), context, partials, value);
|
|
|
break;
|
|
|
case '&':
|
|
|
value = context.lookup(token[1]);
|
|
|
if (value != null) buffer += value;
|
|
|
break;
|
|
|
case 'name':
|
|
|
value = context.lookup(token[1]);
|
|
|
if (value != null) buffer += mustache.escape(value);
|
|
|
break;
|
|
|
case 'text':
|
|
|
buffer += token[1];
|
|
|
break;
|
|
|
}
|
|
|
}
|
|
|
|
|
|
return buffer;
|
|
|
};
|
|
|
|
|
|
mustache.name = "mustache.js";
|
|
|
mustache.version = "0.8.1";
|
|
|
mustache.tags = [ "{{", "}}" ];
|
|
|
|
|
|
// All high-level mustache.* functions use this writer.
|
|
|
var defaultWriter = new Writer();
|
|
|
|
|
|
/**
|
|
|
* Clears all cached templates in the default writer.
|
|
|
*/
|
|
|
mustache.clearCache = function () {
|
|
|
return defaultWriter.clearCache();
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Parses and caches the given template in the default writer and returns the
|
|
|
* array of tokens it contains. Doing this ahead of time avoids the need to
|
|
|
* parse templates on the fly as they are rendered.
|
|
|
*/
|
|
|
mustache.parse = function (template, tags) {
|
|
|
return defaultWriter.parse(template, tags);
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Renders the `template` with the given `view` and `partials` using the
|
|
|
* default writer.
|
|
|
*/
|
|
|
mustache.render = function (template, view, partials) {
|
|
|
return defaultWriter.render(template, view, partials);
|
|
|
};
|
|
|
|
|
|
// This is here for backwards compatibility with 0.4.x.
|
|
|
mustache.to_html = function (template, view, partials, send) {
|
|
|
var result = mustache.render(template, view, partials);
|
|
|
|
|
|
if (isFunction(send)) {
|
|
|
send(result);
|
|
|
} else {
|
|
|
return result;
|
|
|
}
|
|
|
};
|
|
|
|
|
|
// Export the escaping function so that the user may override it.
|
|
|
// See https://github.com/janl/mustache.js/issues/244
|
|
|
mustache.escape = escapeHtml;
|
|
|
|
|
|
// Export these mainly for testing, but also for advanced usage.
|
|
|
mustache.Scanner = Scanner;
|
|
|
mustache.Context = Context;
|
|
|
mustache.Writer = Writer;
|
|
|
|
|
|
}));
|
|
|
|
|
|
// Underscore.js 1.7.0
|
|
|
// http://underscorejs.org
|
|
|
// (c) 2009-2014 Jeremy Ashkenas, DocumentCloud and Investigative Reporters & Editors
|
|
|
// Underscore may be freely distributed under the MIT license.
|
|
|
|
|
|
(function() {
|
|
|
|
|
|
// Baseline setup
|
|
|
// --------------
|
|
|
|
|
|
// Establish the root object, `window` in the browser, or `exports` on the server.
|
|
|
var root = this;
|
|
|
|
|
|
// Save the previous value of the `_` variable.
|
|
|
var previousUnderscore = root._;
|
|
|
|
|
|
// Save bytes in the minified (but not gzipped) version:
|
|
|
var ArrayProto = Array.prototype, ObjProto = Object.prototype, FuncProto = Function.prototype;
|
|
|
|
|
|
// Create quick reference variables for speed access to core prototypes.
|
|
|
var
|
|
|
push = ArrayProto.push,
|
|
|
slice = ArrayProto.slice,
|
|
|
concat = ArrayProto.concat,
|
|
|
toString = ObjProto.toString,
|
|
|
hasOwnProperty = ObjProto.hasOwnProperty;
|
|
|
|
|
|
// All **ECMAScript 5** native function implementations that we hope to use
|
|
|
// are declared here.
|
|
|
var
|
|
|
nativeIsArray = Array.isArray,
|
|
|
nativeKeys = Object.keys,
|
|
|
nativeBind = FuncProto.bind;
|
|
|
|
|
|
// Create a safe reference to the Underscore object for use below.
|
|
|
var _ = function(obj) {
|
|
|
if (obj instanceof _) return obj;
|
|
|
if (!(this instanceof _)) return new _(obj);
|
|
|
this._wrapped = obj;
|
|
|
};
|
|
|
|
|
|
// Export the Underscore object for **Node.js**, with
|
|
|
// backwards-compatibility for the old `require()` API. If we're in
|
|
|
// the browser, add `_` as a global object.
|
|
|
if (typeof exports !== 'undefined') {
|
|
|
if (typeof module !== 'undefined' && module.exports) {
|
|
|
exports = module.exports = _;
|
|
|
}
|
|
|
exports._ = _;
|
|
|
} else {
|
|
|
root._ = _;
|
|
|
}
|
|
|
|
|
|
// Current version.
|
|
|
_.VERSION = '1.7.0';
|
|
|
|
|
|
// Internal function that returns an efficient (for current engines) version
|
|
|
// of the passed-in callback, to be repeatedly applied in other Underscore
|
|
|
// functions.
|
|
|
var createCallback = function(func, context, argCount) {
|
|
|
if (context === void 0) return func;
|
|
|
switch (argCount == null ? 3 : argCount) {
|
|
|
case 1: return function(value) {
|
|
|
return func.call(context, value);
|
|
|
};
|
|
|
case 2: return function(value, other) {
|
|
|
return func.call(context, value, other);
|
|
|
};
|
|
|
case 3: return function(value, index, collection) {
|
|
|
return func.call(context, value, index, collection);
|
|
|
};
|
|
|
case 4: return function(accumulator, value, index, collection) {
|
|
|
return func.call(context, accumulator, value, index, collection);
|
|
|
};
|
|
|
}
|
|
|
return function() {
|
|
|
return func.apply(context, arguments);
|
|
|
};
|
|
|
};
|
|
|
|
|
|
// A mostly-internal function to generate callbacks that can be applied
|
|
|
// to each element in a collection, returning the desired result — either
|
|
|
// identity, an arbitrary callback, a property matcher, or a property accessor.
|
|
|
_.iteratee = function(value, context, argCount) {
|
|
|
if (value == null) return _.identity;
|
|
|
if (_.isFunction(value)) return createCallback(value, context, argCount);
|
|
|
if (_.isObject(value)) return _.matches(value);
|
|
|
return _.property(value);
|
|
|
};
|
|
|
|
|
|
// Collection Functions
|
|
|
// --------------------
|
|
|
|
|
|
// The cornerstone, an `each` implementation, aka `forEach`.
|
|
|
// Handles raw objects in addition to array-likes. Treats all
|
|
|
// sparse array-likes as if they were dense.
|
|
|
_.each = _.forEach = function(obj, iteratee, context) {
|
|
|
if (obj == null) return obj;
|
|
|
iteratee = createCallback(iteratee, context);
|
|
|
var i, length = obj.length;
|
|
|
if (length === +length) {
|
|
|
for (i = 0; i < length; i++) {
|
|
|
iteratee(obj[i], i, obj);
|
|
|
}
|
|
|
} else {
|
|
|
var keys = _.keys(obj);
|
|
|
for (i = 0, length = keys.length; i < length; i++) {
|
|
|
iteratee(obj[keys[i]], keys[i], obj);
|
|
|
}
|
|
|
}
|
|
|
return obj;
|
|
|
};
|
|
|
|
|
|
// Return the results of applying the iteratee to each element.
|
|
|
_.map = _.collect = function(obj, iteratee, context) {
|
|
|
if (obj == null) return [];
|
|
|
iteratee = _.iteratee(iteratee, context);
|
|
|
var keys = obj.length !== +obj.length && _.keys(obj),
|
|
|
length = (keys || obj).length,
|
|
|
results = Array(length),
|
|
|
currentKey;
|
|
|
for (var index = 0; index < length; index++) {
|
|
|
currentKey = keys ? keys[index] : index;
|
|
|
results[index] = iteratee(obj[currentKey], currentKey, obj);
|
|
|
}
|
|
|
return results;
|
|
|
};
|
|
|
|
|
|
var reduceError = 'Reduce of empty array with no initial value';
|
|
|
|
|
|
// **Reduce** builds up a single result from a list of values, aka `inject`,
|
|
|
// or `foldl`.
|
|
|
_.reduce = _.foldl = _.inject = function(obj, iteratee, memo, context) {
|
|
|
if (obj == null) obj = [];
|
|
|
iteratee = createCallback(iteratee, context, 4);
|
|
|
var keys = obj.length !== +obj.length && _.keys(obj),
|
|
|
length = (keys || obj).length,
|
|
|
index = 0, currentKey;
|
|
|
if (arguments.length < 3) {
|
|
|
if (!length) throw new TypeError(reduceError);
|
|
|
memo = obj[keys ? keys[index++] : index++];
|
|
|
}
|
|
|
for (; index < length; index++) {
|
|
|
currentKey = keys ? keys[index] : index;
|
|
|
memo = iteratee(memo, obj[currentKey], currentKey, obj);
|
|
|
}
|
|
|
return memo;
|
|
|
};
|
|
|
|
|
|
// The right-associative version of reduce, also known as `foldr`.
|
|
|
_.reduceRight = _.foldr = function(obj, iteratee, memo, context) {
|
|
|
if (obj == null) obj = [];
|
|
|
iteratee = createCallback(iteratee, context, 4);
|
|
|
var keys = obj.length !== + obj.length && _.keys(obj),
|
|
|
index = (keys || obj).length,
|
|
|
currentKey;
|
|
|
if (arguments.length < 3) {
|
|
|
if (!index) throw new TypeError(reduceError);
|
|
|
memo = obj[keys ? keys[--index] : --index];
|
|
|
}
|
|
|
while (index--) {
|
|
|
currentKey = keys ? keys[index] : index;
|
|
|
memo = iteratee(memo, obj[currentKey], currentKey, obj);
|
|
|
}
|
|
|
return memo;
|
|
|
};
|
|
|
|
|
|
// Return the first value which passes a truth test. Aliased as `detect`.
|
|
|
_.find = _.detect = function(obj, predicate, context) {
|
|
|
var result;
|
|
|
predicate = _.iteratee(predicate, context);
|
|
|
_.some(obj, function(value, index, list) {
|
|
|
if (predicate(value, index, list)) {
|
|
|
result = value;
|
|
|
return true;
|
|
|
}
|
|
|
});
|
|
|
return result;
|
|
|
};
|
|
|
|
|
|
// Return all the elements that pass a truth test.
|
|
|
// Aliased as `select`.
|
|
|
_.filter = _.select = function(obj, predicate, context) {
|
|
|
var results = [];
|
|
|
if (obj == null) return results;
|
|
|
predicate = _.iteratee(predicate, context);
|
|
|
_.each(obj, function(value, index, list) {
|
|
|
if (predicate(value, index, list)) results.push(value);
|
|
|
});
|
|
|
return results;
|
|
|
};
|
|
|
|
|
|
// Return all the elements for which a truth test fails.
|
|
|
_.reject = function(obj, predicate, context) {
|
|
|
return _.filter(obj, _.negate(_.iteratee(predicate)), context);
|
|
|
};
|
|
|
|
|
|
// Determine whether all of the elements match a truth test.
|
|
|
// Aliased as `all`.
|
|
|
_.every = _.all = function(obj, predicate, context) {
|
|
|
if (obj == null) return true;
|
|
|
predicate = _.iteratee(predicate, context);
|
|
|
var keys = obj.length !== +obj.length && _.keys(obj),
|
|
|
length = (keys || obj).length,
|
|
|
index, currentKey;
|
|
|
for (index = 0; index < length; index++) {
|
|
|
currentKey = keys ? keys[index] : index;
|
|
|
if (!predicate(obj[currentKey], currentKey, obj)) return false;
|
|
|
}
|
|
|
return true;
|
|
|
};
|
|
|
|
|
|
// Determine if at least one element in the object matches a truth test.
|
|
|
// Aliased as `any`.
|
|
|
_.some = _.any = function(obj, predicate, context) {
|
|
|
if (obj == null) return false;
|
|
|
predicate = _.iteratee(predicate, context);
|
|
|
var keys = obj.length !== +obj.length && _.keys(obj),
|
|
|
length = (keys || obj).length,
|
|
|
index, currentKey;
|
|
|
for (index = 0; index < length; index++) {
|
|
|
currentKey = keys ? keys[index] : index;
|
|
|
if (predicate(obj[currentKey], currentKey, obj)) return true;
|
|
|
}
|
|
|
return false;
|
|
|
};
|
|
|
|
|
|
// Determine if the array or object contains a given value (using `===`).
|
|
|
// Aliased as `include`.
|
|
|
_.contains = _.include = function(obj, target) {
|
|
|
if (obj == null) return false;
|
|
|
if (obj.length !== +obj.length) obj = _.values(obj);
|
|
|
return _.indexOf(obj, target) >= 0;
|
|
|
};
|
|
|
|
|
|
// Invoke a method (with arguments) on every item in a collection.
|
|
|
_.invoke = function(obj, method) {
|
|
|
var args = slice.call(arguments, 2);
|
|
|
var isFunc = _.isFunction(method);
|
|
|
return _.map(obj, function(value) {
|
|
|
return (isFunc ? method : value[method]).apply(value, args);
|
|
|
});
|
|
|
};
|
|
|
|
|
|
// Convenience version of a common use case of `map`: fetching a property.
|
|
|
_.pluck = function(obj, key) {
|
|
|
return _.map(obj, _.property(key));
|
|
|
};
|
|
|
|
|
|
// Convenience version of a common use case of `filter`: selecting only objects
|
|
|
// containing specific `key:value` pairs.
|
|
|
_.where = function(obj, attrs) {
|
|
|
return _.filter(obj, _.matches(attrs));
|
|
|
};
|
|
|
|
|
|
// Convenience version of a common use case of `find`: getting the first object
|
|
|
// containing specific `key:value` pairs.
|
|
|
_.findWhere = function(obj, attrs) {
|
|
|
return _.find(obj, _.matches(attrs));
|
|
|
};
|
|
|
|
|
|
// Return the maximum element (or element-based computation).
|
|
|
_.max = function(obj, iteratee, context) {
|
|
|
var result = -Infinity, lastComputed = -Infinity,
|
|
|
value, computed;
|
|
|
if (iteratee == null && obj != null) {
|
|
|
obj = obj.length === +obj.length ? obj : _.values(obj);
|
|
|
for (var i = 0, length = obj.length; i < length; i++) {
|
|
|
value = obj[i];
|
|
|
if (value > result) {
|
|
|
result = value;
|
|
|
}
|
|
|
}
|
|
|
} else {
|
|
|
iteratee = _.iteratee(iteratee, context);
|
|
|
_.each(obj, function(value, index, list) {
|
|
|
computed = iteratee(value, index, list);
|
|
|
if (computed > lastComputed || computed === -Infinity && result === -Infinity) {
|
|
|
result = value;
|
|
|
lastComputed = computed;
|
|
|
}
|
|
|
});
|
|
|
}
|
|
|
return result;
|
|
|
};
|
|
|
|
|
|
// Return the minimum element (or element-based computation).
|
|
|
_.min = function(obj, iteratee, context) {
|
|
|
var result = Infinity, lastComputed = Infinity,
|
|
|
value, computed;
|
|
|
if (iteratee == null && obj != null) {
|
|
|
obj = obj.length === +obj.length ? obj : _.values(obj);
|
|
|
for (var i = 0, length = obj.length; i < length; i++) {
|
|
|
value = obj[i];
|
|
|
if (value < result) {
|
|
|
result = value;
|
|
|
}
|
|
|
}
|
|
|
} else {
|
|
|
iteratee = _.iteratee(iteratee, context);
|
|
|
_.each(obj, function(value, index, list) {
|
|
|
computed = iteratee(value, index, list);
|
|
|
if (computed < lastComputed || computed === Infinity && result === Infinity) {
|
|
|
result = value;
|
|
|
lastComputed = computed;
|
|
|
}
|
|
|
});
|
|
|
}
|
|
|
return result;
|
|
|
};
|
|
|
|
|
|
// Shuffle a collection, using the modern version of the
|
|
|
// [Fisher-Yates shuffle](http://en.wikipedia.org/wiki/Fisher–Yates_shuffle).
|
|
|
_.shuffle = function(obj) {
|
|
|
var set = obj && obj.length === +obj.length ? obj : _.values(obj);
|
|
|
var length = set.length;
|
|
|
var shuffled = Array(length);
|
|
|
for (var index = 0, rand; index < length; index++) {
|
|
|
rand = _.random(0, index);
|
|
|
if (rand !== index) shuffled[index] = shuffled[rand];
|
|
|
shuffled[rand] = set[index];
|
|
|
}
|
|
|
return shuffled;
|
|
|
};
|
|
|
|
|
|
// Sample **n** random values from a collection.
|
|
|
// If **n** is not specified, returns a single random element.
|
|
|
// The internal `guard` argument allows it to work with `map`.
|
|
|
_.sample = function(obj, n, guard) {
|
|
|
if (n == null || guard) {
|
|
|
if (obj.length !== +obj.length) obj = _.values(obj);
|
|
|
return obj[_.random(obj.length - 1)];
|
|
|
}
|
|
|
return _.shuffle(obj).slice(0, Math.max(0, n));
|
|
|
};
|
|
|
|
|
|
// Sort the object's values by a criterion produced by an iteratee.
|
|
|
_.sortBy = function(obj, iteratee, context) {
|
|
|
iteratee = _.iteratee(iteratee, context);
|
|
|
return _.pluck(_.map(obj, function(value, index, list) {
|
|
|
return {
|
|
|
value: value,
|
|
|
index: index,
|
|
|
criteria: iteratee(value, index, list)
|
|
|
};
|
|
|
}).sort(function(left, right) {
|
|
|
var a = left.criteria;
|
|
|
var b = right.criteria;
|
|
|
if (a !== b) {
|
|
|
if (a > b || a === void 0) return 1;
|
|
|
if (a < b || b === void 0) return -1;
|
|
|
}
|
|
|
return left.index - right.index;
|
|
|
}), 'value');
|
|
|
};
|
|
|
|
|
|
// An internal function used for aggregate "group by" operations.
|
|
|
var group = function(behavior) {
|
|
|
return function(obj, iteratee, context) {
|
|
|
var result = {};
|
|
|
iteratee = _.iteratee(iteratee, context);
|
|
|
_.each(obj, function(value, index) {
|
|
|
var key = iteratee(value, index, obj);
|
|
|
behavior(result, value, key);
|
|
|
});
|
|
|
return result;
|
|
|
};
|
|
|
};
|
|
|
|
|
|
// Groups the object's values by a criterion. Pass either a string attribute
|
|
|
// to group by, or a function that returns the criterion.
|
|
|
_.groupBy = group(function(result, value, key) {
|
|
|
if (_.has(result, key)) result[key].push(value); else result[key] = [value];
|
|
|
});
|
|
|
|
|
|
// Indexes the object's values by a criterion, similar to `groupBy`, but for
|
|
|
// when you know that your index values will be unique.
|
|
|
_.indexBy = group(function(result, value, key) {
|
|
|
result[key] = value;
|
|
|
});
|
|
|
|
|
|
// Counts instances of an object that group by a certain criterion. Pass
|
|
|
// either a string attribute to count by, or a function that returns the
|
|
|
// criterion.
|
|
|
_.countBy = group(function(result, value, key) {
|
|
|
if (_.has(result, key)) result[key]++; else result[key] = 1;
|
|
|
});
|
|
|
|
|
|
// Use a comparator function to figure out the smallest index at which
|
|
|
// an object should be inserted so as to maintain order. Uses binary search.
|
|
|
_.sortedIndex = function(array, obj, iteratee, context) {
|
|
|
iteratee = _.iteratee(iteratee, context, 1);
|
|
|
var value = iteratee(obj);
|
|
|
var low = 0, high = array.length;
|
|
|
while (low < high) {
|
|
|
var mid = low + high >>> 1;
|
|
|
if (iteratee(array[mid]) < value) low = mid + 1; else high = mid;
|
|
|
}
|
|
|
return low;
|
|
|
};
|
|
|
|
|
|
// Safely create a real, live array from anything iterable.
|
|
|
_.toArray = function(obj) {
|
|
|
if (!obj) return [];
|
|
|
if (_.isArray(obj)) return slice.call(obj);
|
|
|
if (obj.length === +obj.length) return _.map(obj, _.identity);
|
|
|
return _.values(obj);
|
|
|
};
|
|
|
|
|
|
// Return the number of elements in an object.
|
|
|
_.size = function(obj) {
|
|
|
if (obj == null) return 0;
|
|
|
return obj.length === +obj.length ? obj.length : _.keys(obj).length;
|
|
|
};
|
|
|
|
|
|
// Split a collection into two arrays: one whose elements all satisfy the given
|
|
|
// predicate, and one whose elements all do not satisfy the predicate.
|
|
|
_.partition = function(obj, predicate, context) {
|
|
|
predicate = _.iteratee(predicate, context);
|
|
|
var pass = [], fail = [];
|
|
|
_.each(obj, function(value, key, obj) {
|
|
|
(predicate(value, key, obj) ? pass : fail).push(value);
|
|
|
});
|
|
|
return [pass, fail];
|
|
|
};
|
|
|
|
|
|
// Array Functions
|
|
|
// ---------------
|
|
|
|
|
|
// Get the first element of an array. Passing **n** will return the first N
|
|
|
// values in the array. Aliased as `head` and `take`. The **guard** check
|
|
|
// allows it to work with `_.map`.
|
|
|
_.first = _.head = _.take = function(array, n, guard) {
|
|
|
if (array == null) return void 0;
|
|
|
if (n == null || guard) return array[0];
|
|
|
if (n < 0) return [];
|
|
|
return slice.call(array, 0, n);
|
|
|
};
|
|
|
|
|
|
// Returns everything but the last entry of the array. Especially useful on
|
|
|
// the arguments object. Passing **n** will return all the values in
|
|
|
// the array, excluding the last N. The **guard** check allows it to work with
|
|
|
// `_.map`.
|
|
|
_.initial = function(array, n, guard) {
|
|
|
return slice.call(array, 0, Math.max(0, array.length - (n == null || guard ? 1 : n)));
|
|
|
};
|
|
|
|
|
|
// Get the last element of an array. Passing **n** will return the last N
|
|
|
// values in the array. The **guard** check allows it to work with `_.map`.
|
|
|
_.last = function(array, n, guard) {
|
|
|
if (array == null) return void 0;
|
|
|
if (n == null || guard) return array[array.length - 1];
|
|
|
return slice.call(array, Math.max(array.length - n, 0));
|
|
|
};
|
|
|
|
|
|
// Returns everything but the first entry of the array. Aliased as `tail` and `drop`.
|
|
|
// Especially useful on the arguments object. Passing an **n** will return
|
|
|
// the rest N values in the array. The **guard**
|
|
|
// check allows it to work with `_.map`.
|
|
|
_.rest = _.tail = _.drop = function(array, n, guard) {
|
|
|
return slice.call(array, n == null || guard ? 1 : n);
|
|
|
};
|
|
|
|
|
|
// Trim out all falsy values from an array.
|
|
|
_.compact = function(array) {
|
|
|
return _.filter(array, _.identity);
|
|
|
};
|
|
|
|
|
|
// Internal implementation of a recursive `flatten` function.
|
|
|
var flatten = function(input, shallow, strict, output) {
|
|
|
if (shallow && _.every(input, _.isArray)) {
|
|
|
return concat.apply(output, input);
|
|
|
}
|
|
|
for (var i = 0, length = input.length; i < length; i++) {
|
|
|
var value = input[i];
|
|
|
if (!_.isArray(value) && !_.isArguments(value)) {
|
|
|
if (!strict) output.push(value);
|
|
|
} else if (shallow) {
|
|
|
push.apply(output, value);
|
|
|
} else {
|
|
|
flatten(value, shallow, strict, output);
|
|
|
}
|
|
|
}
|
|
|
return output;
|
|
|
};
|
|
|
|
|
|
// Flatten out an array, either recursively (by default), or just one level.
|
|
|
_.flatten = function(array, shallow) {
|
|
|
return flatten(array, shallow, false, []);
|
|
|
};
|
|
|
|
|
|
// Return a version of the array that does not contain the specified value(s).
|
|
|
_.without = function(array) {
|
|
|
return _.difference(array, slice.call(arguments, 1));
|
|
|
};
|
|
|
|
|
|
// Produce a duplicate-free version of the array. If the array has already
|
|
|
// been sorted, you have the option of using a faster algorithm.
|
|
|
// Aliased as `unique`.
|
|
|
_.uniq = _.unique = function(array, isSorted, iteratee, context) {
|
|
|
if (array == null) return [];
|
|
|
if (!_.isBoolean(isSorted)) {
|
|
|
context = iteratee;
|
|
|
iteratee = isSorted;
|
|
|
isSorted = false;
|
|
|
}
|
|
|
if (iteratee != null) iteratee = _.iteratee(iteratee, context);
|
|
|
var result = [];
|
|
|
var seen = [];
|
|
|
for (var i = 0, length = array.length; i < length; i++) {
|
|
|
var value = array[i];
|
|
|
if (isSorted) {
|
|
|
if (!i || seen !== value) result.push(value);
|
|
|
seen = value;
|
|
|
} else if (iteratee) {
|
|
|
var computed = iteratee(value, i, array);
|
|
|
if (_.indexOf(seen, computed) < 0) {
|
|
|
seen.push(computed);
|
|
|
result.push(value);
|
|
|
}
|
|
|
} else if (_.indexOf(result, value) < 0) {
|
|
|
result.push(value);
|
|
|
}
|
|
|
}
|
|
|
return result;
|
|
|
};
|
|
|
|
|
|
// Produce an array that contains the union: each distinct element from all of
|
|
|
// the passed-in arrays.
|
|
|
_.union = function() {
|
|
|
return _.uniq(flatten(arguments, true, true, []));
|
|
|
};
|
|
|
|
|
|
// Produce an array that contains every item shared between all the
|
|
|
// passed-in arrays.
|
|
|
_.intersection = function(array) {
|
|
|
if (array == null) return [];
|
|
|
var result = [];
|
|
|
var argsLength = arguments.length;
|
|
|
for (var i = 0, length = array.length; i < length; i++) {
|
|
|
var item = array[i];
|
|
|
if (_.contains(result, item)) continue;
|
|
|
for (var j = 1; j < argsLength; j++) {
|
|
|
if (!_.contains(arguments[j], item)) break;
|
|
|
}
|
|
|
if (j === argsLength) result.push(item);
|
|
|
}
|
|
|
return result;
|
|
|
};
|
|
|
|
|
|
// Take the difference between one array and a number of other arrays.
|
|
|
// Only the elements present in just the first array will remain.
|
|
|
_.difference = function(array) {
|
|
|
var rest = flatten(slice.call(arguments, 1), true, true, []);
|
|
|
return _.filter(array, function(value){
|
|
|
return !_.contains(rest, value);
|
|
|
});
|
|
|
};
|
|
|
|
|
|
// Zip together multiple lists into a single array -- elements that share
|
|
|
// an index go together.
|
|
|
_.zip = function(array) {
|
|
|
if (array == null) return [];
|
|
|
var length = _.max(arguments, 'length').length;
|
|
|
var results = Array(length);
|
|
|
for (var i = 0; i < length; i++) {
|
|
|
results[i] = _.pluck(arguments, i);
|
|
|
}
|
|
|
return results;
|
|
|
};
|
|
|
|
|
|
// Converts lists into objects. Pass either a single array of `[key, value]`
|
|
|
// pairs, or two parallel arrays of the same length -- one of keys, and one of
|
|
|
// the corresponding values.
|
|
|
_.object = function(list, values) {
|
|
|
if (list == null) return {};
|
|
|
var result = {};
|
|
|
for (var i = 0, length = list.length; i < length; i++) {
|
|
|
if (values) {
|
|
|
result[list[i]] = values[i];
|
|
|
} else {
|
|
|
result[list[i][0]] = list[i][1];
|
|
|
}
|
|
|
}
|
|
|
return result;
|
|
|
};
|
|
|
|
|
|
// Return the position of the first occurrence of an item in an array,
|
|
|
// or -1 if the item is not included in the array.
|
|
|
// If the array is large and already in sort order, pass `true`
|
|
|
// for **isSorted** to use binary search.
|
|
|
_.indexOf = function(array, item, isSorted) {
|
|
|
if (array == null) return -1;
|
|
|
var i = 0, length = array.length;
|
|
|
if (isSorted) {
|
|
|
if (typeof isSorted == 'number') {
|
|
|
i = isSorted < 0 ? Math.max(0, length + isSorted) : isSorted;
|
|
|
} else {
|
|
|
i = _.sortedIndex(array, item);
|
|
|
return array[i] === item ? i : -1;
|
|
|
}
|
|
|
}
|
|
|
for (; i < length; i++) if (array[i] === item) return i;
|
|
|
return -1;
|
|
|
};
|
|
|
|
|
|
_.lastIndexOf = function(array, item, from) {
|
|
|
if (array == null) return -1;
|
|
|
var idx = array.length;
|
|
|
if (typeof from == 'number') {
|
|
|
idx = from < 0 ? idx + from + 1 : Math.min(idx, from + 1);
|
|
|
}
|
|
|
while (--idx >= 0) if (array[idx] === item) return idx;
|
|
|
return -1;
|
|
|
};
|
|
|
|
|
|
// Generate an integer Array containing an arithmetic progression. A port of
|
|
|
// the native Python `range()` function. See
|
|
|
// [the Python documentation](http://docs.python.org/library/functions.html#range).
|
|
|
_.range = function(start, stop, step) {
|
|
|
if (arguments.length <= 1) {
|
|
|
stop = start || 0;
|
|
|
start = 0;
|
|
|
}
|
|
|
step = step || 1;
|
|
|
|
|
|
var length = Math.max(Math.ceil((stop - start) / step), 0);
|
|
|
var range = Array(length);
|
|
|
|
|
|
for (var idx = 0; idx < length; idx++, start += step) {
|
|
|
range[idx] = start;
|
|
|
}
|
|
|
|
|
|
return range;
|
|
|
};
|
|
|
|
|
|
// Function (ahem) Functions
|
|
|
// ------------------
|
|
|
|
|
|
// Reusable constructor function for prototype setting.
|
|
|
var Ctor = function(){};
|
|
|
|
|
|
// Create a function bound to a given object (assigning `this`, and arguments,
|
|
|
// optionally). Delegates to **ECMAScript 5**'s native `Function.bind` if
|
|
|
// available.
|
|
|
_.bind = function(func, context) {
|
|
|
var args, bound;
|
|
|
if (nativeBind && func.bind === nativeBind) return nativeBind.apply(func, slice.call(arguments, 1));
|
|
|
if (!_.isFunction(func)) throw new TypeError('Bind must be called on a function');
|
|
|
args = slice.call(arguments, 2);
|
|
|
bound = function() {
|
|
|
if (!(this instanceof bound)) return func.apply(context, args.concat(slice.call(arguments)));
|
|
|
Ctor.prototype = func.prototype;
|
|
|
var self = new Ctor;
|
|
|
Ctor.prototype = null;
|
|
|
var result = func.apply(self, args.concat(slice.call(arguments)));
|
|
|
if (_.isObject(result)) return result;
|
|
|
return self;
|
|
|
};
|
|
|
return bound;
|
|
|
};
|
|
|
|
|
|
// Partially apply a function by creating a version that has had some of its
|
|
|
// arguments pre-filled, without changing its dynamic `this` context. _ acts
|
|
|
// as a placeholder, allowing any combination of arguments to be pre-filled.
|
|
|
_.partial = function(func) {
|
|
|
var boundArgs = slice.call(arguments, 1);
|
|
|
return function() {
|
|
|
var position = 0;
|
|
|
var args = boundArgs.slice();
|
|
|
for (var i = 0, length = args.length; i < length; i++) {
|
|
|
if (args[i] === _) args[i] = arguments[position++];
|
|
|
}
|
|
|
while (position < arguments.length) args.push(arguments[position++]);
|
|
|
return func.apply(this, args);
|
|
|
};
|
|
|
};
|
|
|
|
|
|
// Bind a number of an object's methods to that object. Remaining arguments
|
|
|
// are the method names to be bound. Useful for ensuring that all callbacks
|
|
|
// defined on an object belong to it.
|
|
|
_.bindAll = function(obj) {
|
|
|
var i, length = arguments.length, key;
|
|
|
if (length <= 1) throw new Error('bindAll must be passed function names');
|
|
|
for (i = 1; i < length; i++) {
|
|
|
key = arguments[i];
|
|
|
obj[key] = _.bind(obj[key], obj);
|
|
|
}
|
|
|
return obj;
|
|
|
};
|
|
|
|
|
|
// Memoize an expensive function by storing its results.
|
|
|
_.memoize = function(func, hasher) {
|
|
|
var memoize = function(key) {
|
|
|
var cache = memoize.cache;
|
|
|
var address = hasher ? hasher.apply(this, arguments) : key;
|
|
|
if (!_.has(cache, address)) cache[address] = func.apply(this, arguments);
|
|
|
return cache[address];
|
|
|
};
|
|
|
memoize.cache = {};
|
|
|
return memoize;
|
|
|
};
|
|
|
|
|
|
// Delays a function for the given number of milliseconds, and then calls
|
|
|
// it with the arguments supplied.
|
|
|
_.delay = function(func, wait) {
|
|
|
var args = slice.call(arguments, 2);
|
|
|
return setTimeout(function(){
|
|
|
return func.apply(null, args);
|
|
|
}, wait);
|
|
|
};
|
|
|
|
|
|
// Defers a function, scheduling it to run after the current call stack has
|
|
|
// cleared.
|
|
|
_.defer = function(func) {
|
|
|
return _.delay.apply(_, [func, 1].concat(slice.call(arguments, 1)));
|
|
|
};
|
|
|
|
|
|
// Returns a function, that, when invoked, will only be triggered at most once
|
|
|
// during a given window of time. Normally, the throttled function will run
|
|
|
// as much as it can, without ever going more than once per `wait` duration;
|
|
|
// but if you'd like to disable the execution on the leading edge, pass
|
|
|
// `{leading: false}`. To disable execution on the trailing edge, ditto.
|
|
|
_.throttle = function(func, wait, options) {
|
|
|
var context, args, result;
|
|
|
var timeout = null;
|
|
|
var previous = 0;
|
|
|
if (!options) options = {};
|
|
|
var later = function() {
|
|
|
previous = options.leading === false ? 0 : _.now();
|
|
|
timeout = null;
|
|
|
result = func.apply(context, args);
|
|
|
if (!timeout) context = args = null;
|
|
|
};
|
|
|
return function() {
|
|
|
var now = _.now();
|
|
|
if (!previous && options.leading === false) previous = now;
|
|
|
var remaining = wait - (now - previous);
|
|
|
context = this;
|
|
|
args = arguments;
|
|
|
if (remaining <= 0 || remaining > wait) {
|
|
|
clearTimeout(timeout);
|
|
|
timeout = null;
|
|
|
previous = now;
|
|
|
result = func.apply(context, args);
|
|
|
if (!timeout) context = args = null;
|
|
|
} else if (!timeout && options.trailing !== false) {
|
|
|
timeout = setTimeout(later, remaining);
|
|
|
}
|
|
|
return result;
|
|
|
};
|
|
|
};
|
|
|
|
|
|
// Returns a function, that, as long as it continues to be invoked, will not
|
|
|
// be triggered. The function will be called after it stops being called for
|
|
|
// N milliseconds. If `immediate` is passed, trigger the function on the
|
|
|
// leading edge, instead of the trailing.
|
|
|
_.debounce = function(func, wait, immediate) {
|
|
|
var timeout, args, context, timestamp, result;
|
|
|
|
|
|
var later = function() {
|
|
|
var last = _.now() - timestamp;
|
|
|
|
|
|
if (last < wait && last > 0) {
|
|
|
timeout = setTimeout(later, wait - last);
|
|
|
} else {
|
|
|
timeout = null;
|
|
|
if (!immediate) {
|
|
|
result = func.apply(context, args);
|
|
|
if (!timeout) context = args = null;
|
|
|
}
|
|
|
}
|
|
|
};
|
|
|
|
|
|
return function() {
|
|
|
context = this;
|
|
|
args = arguments;
|
|
|
timestamp = _.now();
|
|
|
var callNow = immediate && !timeout;
|
|
|
if (!timeout) timeout = setTimeout(later, wait);
|
|
|
if (callNow) {
|
|
|
result = func.apply(context, args);
|
|
|
context = args = null;
|
|
|
}
|
|
|
|
|
|
return result;
|
|
|
};
|
|
|
};
|
|
|
|
|
|
// Returns the first function passed as an argument to the second,
|
|
|
// allowing you to adjust arguments, run code before and after, and
|
|
|
// conditionally execute the original function.
|
|
|
_.wrap = function(func, wrapper) {
|
|
|
return _.partial(wrapper, func);
|
|
|
};
|
|
|
|
|
|
// Returns a negated version of the passed-in predicate.
|
|
|
_.negate = function(predicate) {
|
|
|
return function() {
|
|
|
return !predicate.apply(this, arguments);
|
|
|
};
|
|
|
};
|
|
|
|
|
|
// Returns a function that is the composition of a list of functions, each
|
|
|
// consuming the return value of the function that follows.
|
|
|
_.compose = function() {
|
|
|
var args = arguments;
|
|
|
var start = args.length - 1;
|
|
|
return function() {
|
|
|
var i = start;
|
|
|
var result = args[start].apply(this, arguments);
|
|
|
while (i--) result = args[i].call(this, result);
|
|
|
return result;
|
|
|
};
|
|
|
};
|
|
|
|
|
|
// Returns a function that will only be executed after being called N times.
|
|
|
_.after = function(times, func) {
|
|
|
return function() {
|
|
|
if (--times < 1) {
|
|
|
return func.apply(this, arguments);
|
|
|
}
|
|
|
};
|
|
|
};
|
|
|
|
|
|
// Returns a function that will only be executed before being called N times.
|
|
|
_.before = function(times, func) {
|
|
|
var memo;
|
|
|
return function() {
|
|
|
if (--times > 0) {
|
|
|
memo = func.apply(this, arguments);
|
|
|
} else {
|
|
|
func = null;
|
|
|
}
|
|
|
return memo;
|
|
|
};
|
|
|
};
|
|
|
|
|
|
// Returns a function that will be executed at most one time, no matter how
|
|
|
// often you call it. Useful for lazy initialization.
|
|
|
_.once = _.partial(_.before, 2);
|
|
|
|
|
|
// Object Functions
|
|
|
// ----------------
|
|
|
|
|
|
// Retrieve the names of an object's properties.
|
|
|
// Delegates to **ECMAScript 5**'s native `Object.keys`
|
|
|
_.keys = function(obj) {
|
|
|
if (!_.isObject(obj)) return [];
|
|
|
if (nativeKeys) return nativeKeys(obj);
|
|
|
var keys = [];
|
|
|
for (var key in obj) if (_.has(obj, key)) keys.push(key);
|
|
|
return keys;
|
|
|
};
|
|
|
|
|
|
// Retrieve the values of an object's properties.
|
|
|
_.values = function(obj) {
|
|
|
var keys = _.keys(obj);
|
|
|
var length = keys.length;
|
|
|
var values = Array(length);
|
|
|
for (var i = 0; i < length; i++) {
|
|
|
values[i] = obj[keys[i]];
|
|
|
}
|
|
|
return values;
|
|
|
};
|
|
|
|
|
|
// Convert an object into a list of `[key, value]` pairs.
|
|
|
_.pairs = function(obj) {
|
|
|
var keys = _.keys(obj);
|
|
|
var length = keys.length;
|
|
|
var pairs = Array(length);
|
|
|
for (var i = 0; i < length; i++) {
|
|
|
pairs[i] = [keys[i], obj[keys[i]]];
|
|
|
}
|
|
|
return pairs;
|
|
|
};
|
|
|
|
|
|
// Invert the keys and values of an object. The values must be serializable.
|
|
|
_.invert = function(obj) {
|
|
|
var result = {};
|
|
|
var keys = _.keys(obj);
|
|
|
for (var i = 0, length = keys.length; i < length; i++) {
|
|
|
result[obj[keys[i]]] = keys[i];
|
|
|
}
|
|
|
return result;
|
|
|
};
|
|
|
|
|
|
// Return a sorted list of the function names available on the object.
|
|
|
// Aliased as `methods`
|
|
|
_.functions = _.methods = function(obj) {
|
|
|
var names = [];
|
|
|
for (var key in obj) {
|
|
|
if (_.isFunction(obj[key])) names.push(key);
|
|
|
}
|
|
|
return names.sort();
|
|
|
};
|
|
|
|
|
|
// Extend a given object with all the properties in passed-in object(s).
|
|
|
_.extend = function(obj) {
|
|
|
if (!_.isObject(obj)) return obj;
|
|
|
var source, prop;
|
|
|
for (var i = 1, length = arguments.length; i < length; i++) {
|
|
|
source = arguments[i];
|
|
|
for (prop in source) {
|
|
|
if (hasOwnProperty.call(source, prop)) {
|
|
|
obj[prop] = source[prop];
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
return obj;
|
|
|
};
|
|
|
|
|
|
// Return a copy of the object only containing the whitelisted properties.
|
|
|
_.pick = function(obj, iteratee, context) {
|
|
|
var result = {}, key;
|
|
|
if (obj == null) return result;
|
|
|
if (_.isFunction(iteratee)) {
|
|
|
iteratee = createCallback(iteratee, context);
|
|
|
for (key in obj) {
|
|
|
var value = obj[key];
|
|
|
if (iteratee(value, key, obj)) result[key] = value;
|
|
|
}
|
|
|
} else {
|
|
|
var keys = concat.apply([], slice.call(arguments, 1));
|
|
|
obj = new Object(obj);
|
|
|
for (var i = 0, length = keys.length; i < length; i++) {
|
|
|
key = keys[i];
|
|
|
if (key in obj) result[key] = obj[key];
|
|
|
}
|
|
|
}
|
|
|
return result;
|
|
|
};
|
|
|
|
|
|
// Return a copy of the object without the blacklisted properties.
|
|
|
_.omit = function(obj, iteratee, context) {
|
|
|
if (_.isFunction(iteratee)) {
|
|
|
iteratee = _.negate(iteratee);
|
|
|
} else {
|
|
|
var keys = _.map(concat.apply([], slice.call(arguments, 1)), String);
|
|
|
iteratee = function(value, key) {
|
|
|
return !_.contains(keys, key);
|
|
|
};
|
|
|
}
|
|
|
return _.pick(obj, iteratee, context);
|
|
|
};
|
|
|
|
|
|
// Fill in a given object with default properties.
|
|
|
_.defaults = function(obj) {
|
|
|
if (!_.isObject(obj)) return obj;
|
|
|
for (var i = 1, length = arguments.length; i < length; i++) {
|
|
|
var source = arguments[i];
|
|
|
for (var prop in source) {
|
|
|
if (obj[prop] === void 0) obj[prop] = source[prop];
|
|
|
}
|
|
|
}
|
|
|
return obj;
|
|
|
};
|
|
|
|
|
|
// Create a (shallow-cloned) duplicate of an object.
|
|
|
_.clone = function(obj) {
|
|
|
if (!_.isObject(obj)) return obj;
|
|
|
return _.isArray(obj) ? obj.slice() : _.extend({}, obj);
|
|
|
};
|
|
|
|
|
|
// Invokes interceptor with the obj, and then returns obj.
|
|
|
// The primary purpose of this method is to "tap into" a method chain, in
|
|
|
// order to perform operations on intermediate results within the chain.
|
|
|
_.tap = function(obj, interceptor) {
|
|
|
interceptor(obj);
|
|
|
return obj;
|
|
|
};
|
|
|
|
|
|
// Internal recursive comparison function for `isEqual`.
|
|
|
var eq = function(a, b, aStack, bStack) {
|
|
|
// Identical objects are equal. `0 === -0`, but they aren't identical.
|
|
|
// See the [Harmony `egal` proposal](http://wiki.ecmascript.org/doku.php?id=harmony:egal).
|
|
|
if (a === b) return a !== 0 || 1 / a === 1 / b;
|
|
|
// A strict comparison is necessary because `null == undefined`.
|
|
|
if (a == null || b == null) return a === b;
|
|
|
// Unwrap any wrapped objects.
|
|
|
if (a instanceof _) a = a._wrapped;
|
|
|
if (b instanceof _) b = b._wrapped;
|
|
|
// Compare `[[Class]]` names.
|
|
|
var className = toString.call(a);
|
|
|
if (className !== toString.call(b)) return false;
|
|
|
switch (className) {
|
|
|
// Strings, numbers, regular expressions, dates, and booleans are compared by value.
|
|
|
case '[object RegExp]':
|
|
|
// RegExps are coerced to strings for comparison (Note: '' + /a/i === '/a/i')
|
|
|
case '[object String]':
|
|
|
// Primitives and their corresponding object wrappers are equivalent; thus, `"5"` is
|
|
|
// equivalent to `new String("5")`.
|
|
|
return '' + a === '' + b;
|
|
|
case '[object Number]':
|
|
|
// `NaN`s are equivalent, but non-reflexive.
|
|
|
// Object(NaN) is equivalent to NaN
|
|
|
if (+a !== +a) return +b !== +b;
|
|
|
// An `egal` comparison is performed for other numeric values.
|
|
|
return +a === 0 ? 1 / +a === 1 / b : +a === +b;
|
|
|
case '[object Date]':
|
|
|
case '[object Boolean]':
|
|
|
// Coerce dates and booleans to numeric primitive values. Dates are compared by their
|
|
|
// millisecond representations. Note that invalid dates with millisecond representations
|
|
|
// of `NaN` are not equivalent.
|
|
|
return +a === +b;
|
|
|
}
|
|
|
if (typeof a != 'object' || typeof b != 'object') return false;
|
|
|
// Assume equality for cyclic structures. The algorithm for detecting cyclic
|
|
|
// structures is adapted from ES 5.1 section 15.12.3, abstract operation `JO`.
|
|
|
var length = aStack.length;
|
|
|
while (length--) {
|
|
|
// Linear search. Performance is inversely proportional to the number of
|
|
|
// unique nested structures.
|
|
|
if (aStack[length] === a) return bStack[length] === b;
|
|
|
}
|
|
|
// Objects with different constructors are not equivalent, but `Object`s
|
|
|
// from different frames are.
|
|
|
var aCtor = a.constructor, bCtor = b.constructor;
|
|
|
if (
|
|
|
aCtor !== bCtor &&
|
|
|
// Handle Object.create(x) cases
|
|
|
'constructor' in a && 'constructor' in b &&
|
|
|
!(_.isFunction(aCtor) && aCtor instanceof aCtor &&
|
|
|
_.isFunction(bCtor) && bCtor instanceof bCtor)
|
|
|
) {
|
|
|
return false;
|
|
|
}
|
|
|
// Add the first object to the stack of traversed objects.
|
|
|
aStack.push(a);
|
|
|
bStack.push(b);
|
|
|
var size, result;
|
|
|
// Recursively compare objects and arrays.
|
|
|
if (className === '[object Array]') {
|
|
|
// Compare array lengths to determine if a deep comparison is necessary.
|
|
|
size = a.length;
|
|
|
result = size === b.length;
|
|
|
if (result) {
|
|
|
// Deep compare the contents, ignoring non-numeric properties.
|
|
|
while (size--) {
|
|
|
if (!(result = eq(a[size], b[size], aStack, bStack))) break;
|
|
|
}
|
|
|
}
|
|
|
} else {
|
|
|
// Deep compare objects.
|
|
|
var keys = _.keys(a), key;
|
|
|
size = keys.length;
|
|
|
// Ensure that both objects contain the same number of properties before comparing deep equality.
|
|
|
result = _.keys(b).length === size;
|
|
|
if (result) {
|
|
|
while (size--) {
|
|
|
// Deep compare each member
|
|
|
key = keys[size];
|
|
|
if (!(result = _.has(b, key) && eq(a[key], b[key], aStack, bStack))) break;
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
// Remove the first object from the stack of traversed objects.
|
|
|
aStack.pop();
|
|
|
bStack.pop();
|
|
|
return result;
|
|
|
};
|
|
|
|
|
|
// Perform a deep comparison to check if two objects are equal.
|
|
|
_.isEqual = function(a, b) {
|
|
|
return eq(a, b, [], []);
|
|
|
};
|
|
|
|
|
|
// Is a given array, string, or object empty?
|
|
|
// An "empty" object has no enumerable own-properties.
|
|
|
_.isEmpty = function(obj) {
|
|
|
if (obj == null) return true;
|
|
|
if (_.isArray(obj) || _.isString(obj) || _.isArguments(obj)) return obj.length === 0;
|
|
|
for (var key in obj) if (_.has(obj, key)) return false;
|
|
|
return true;
|
|
|
};
|
|
|
|
|
|
// Is a given value a DOM element?
|
|
|
_.isElement = function(obj) {
|
|
|
return !!(obj && obj.nodeType === 1);
|
|
|
};
|
|
|
|
|
|
// Is a given value an array?
|
|
|
// Delegates to ECMA5's native Array.isArray
|
|
|
_.isArray = nativeIsArray || function(obj) {
|
|
|
return toString.call(obj) === '[object Array]';
|
|
|
};
|
|
|
|
|
|
// Is a given variable an object?
|
|
|
_.isObject = function(obj) {
|
|
|
var type = typeof obj;
|
|
|
return type === 'function' || type === 'object' && !!obj;
|
|
|
};
|
|
|
|
|
|
// Add some isType methods: isArguments, isFunction, isString, isNumber, isDate, isRegExp.
|
|
|
_.each(['Arguments', 'Function', 'String', 'Number', 'Date', 'RegExp'], function(name) {
|
|
|
_['is' + name] = function(obj) {
|
|
|
return toString.call(obj) === '[object ' + name + ']';
|
|
|
};
|
|
|
});
|
|
|
|
|
|
// Define a fallback version of the method in browsers (ahem, IE), where
|
|
|
// there isn't any inspectable "Arguments" type.
|
|
|
if (!_.isArguments(arguments)) {
|
|
|
_.isArguments = function(obj) {
|
|
|
return _.has(obj, 'callee');
|
|
|
};
|
|
|
}
|
|
|
|
|
|
// Optimize `isFunction` if appropriate. Work around an IE 11 bug.
|
|
|
if (typeof /./ !== 'function') {
|
|
|
_.isFunction = function(obj) {
|
|
|
return typeof obj == 'function' || false;
|
|
|
};
|
|
|
}
|
|
|
|
|
|
// Is a given object a finite number?
|
|
|
_.isFinite = function(obj) {
|
|
|
return isFinite(obj) && !isNaN(parseFloat(obj));
|
|
|
};
|
|
|
|
|
|
// Is the given value `NaN`? (NaN is the only number which does not equal itself).
|
|
|
_.isNaN = function(obj) {
|
|
|
return _.isNumber(obj) && obj !== +obj;
|
|
|
};
|
|
|
|
|
|
// Is a given value a boolean?
|
|
|
_.isBoolean = function(obj) {
|
|
|
return obj === true || obj === false || toString.call(obj) === '[object Boolean]';
|
|
|
};
|
|
|
|
|
|
// Is a given value equal to null?
|
|
|
_.isNull = function(obj) {
|
|
|
return obj === null;
|
|
|
};
|
|
|
|
|
|
// Is a given variable undefined?
|
|
|
_.isUndefined = function(obj) {
|
|
|
return obj === void 0;
|
|
|
};
|
|
|
|
|
|
// Shortcut function for checking if an object has a given property directly
|
|
|
// on itself (in other words, not on a prototype).
|
|
|
_.has = function(obj, key) {
|
|
|
return obj != null && hasOwnProperty.call(obj, key);
|
|
|
};
|
|
|
|
|
|
// Utility Functions
|
|
|
// -----------------
|
|
|
|
|
|
// Run Underscore.js in *noConflict* mode, returning the `_` variable to its
|
|
|
// previous owner. Returns a reference to the Underscore object.
|
|
|
_.noConflict = function() {
|
|
|
root._ = previousUnderscore;
|
|
|
return this;
|
|
|
};
|
|
|
|
|
|
// Keep the identity function around for default iteratees.
|
|
|
_.identity = function(value) {
|
|
|
return value;
|
|
|
};
|
|
|
|
|
|
_.constant = function(value) {
|
|
|
return function() {
|
|
|
return value;
|
|
|
};
|
|
|
};
|
|
|
|
|
|
_.noop = function(){};
|
|
|
|
|
|
_.property = function(key) {
|
|
|
return function(obj) {
|
|
|
return obj[key];
|
|
|
};
|
|
|
};
|
|
|
|
|
|
// Returns a predicate for checking whether an object has a given set of `key:value` pairs.
|
|
|
_.matches = function(attrs) {
|
|
|
var pairs = _.pairs(attrs), length = pairs.length;
|
|
|
return function(obj) {
|
|
|
if (obj == null) return !length;
|
|
|
obj = new Object(obj);
|
|
|
for (var i = 0; i < length; i++) {
|
|
|
var pair = pairs[i], key = pair[0];
|
|
|
if (pair[1] !== obj[key] || !(key in obj)) return false;
|
|
|
}
|
|
|
return true;
|
|
|
};
|
|
|
};
|
|
|
|
|
|
// Run a function **n** times.
|
|
|
_.times = function(n, iteratee, context) {
|
|
|
var accum = Array(Math.max(0, n));
|
|
|
iteratee = createCallback(iteratee, context, 1);
|
|
|
for (var i = 0; i < n; i++) accum[i] = iteratee(i);
|
|
|
return accum;
|
|
|
};
|
|
|
|
|
|
// Return a random integer between min and max (inclusive).
|
|
|
_.random = function(min, max) {
|
|
|
if (max == null) {
|
|
|
max = min;
|
|
|
min = 0;
|
|
|
}
|
|
|
return min + Math.floor(Math.random() * (max - min + 1));
|
|
|
};
|
|
|
|
|
|
// A (possibly faster) way to get the current timestamp as an integer.
|
|
|
_.now = Date.now || function() {
|
|
|
return new Date().getTime();
|
|
|
};
|
|
|
|
|
|
// List of HTML entities for escaping.
|
|
|
var escapeMap = {
|
|
|
'&': '&',
|
|
|
'<': '<',
|
|
|
'>': '>',
|
|
|
'"': '"',
|
|
|
"'": ''',
|
|
|
'`': '`'
|
|
|
};
|
|
|
var unescapeMap = _.invert(escapeMap);
|
|
|
|
|
|
// Functions for escaping and unescaping strings to/from HTML interpolation.
|
|
|
var createEscaper = function(map) {
|
|
|
var escaper = function(match) {
|
|
|
return map[match];
|
|
|
};
|
|
|
// Regexes for identifying a key that needs to be escaped
|
|
|
var source = '(?:' + _.keys(map).join('|') + ')';
|
|
|
var testRegexp = RegExp(source);
|
|
|
var replaceRegexp = RegExp(source, 'g');
|
|
|
return function(string) {
|
|
|
string = string == null ? '' : '' + string;
|
|
|
return testRegexp.test(string) ? string.replace(replaceRegexp, escaper) : string;
|
|
|
};
|
|
|
};
|
|
|
_.escape = createEscaper(escapeMap);
|
|
|
_.unescape = createEscaper(unescapeMap);
|
|
|
|
|
|
// If the value of the named `property` is a function then invoke it with the
|
|
|
// `object` as context; otherwise, return it.
|
|
|
_.result = function(object, property) {
|
|
|
if (object == null) return void 0;
|
|
|
var value = object[property];
|
|
|
return _.isFunction(value) ? object[property]() : value;
|
|
|
};
|
|
|
|
|
|
// Generate a unique integer id (unique within the entire client session).
|
|
|
// Useful for temporary DOM ids.
|
|
|
var idCounter = 0;
|
|
|
_.uniqueId = function(prefix) {
|
|
|
var id = ++idCounter + '';
|
|
|
return prefix ? prefix + id : id;
|
|
|
};
|
|
|
|
|
|
// By default, Underscore uses ERB-style template delimiters, change the
|
|
|
// following template settings to use alternative delimiters.
|
|
|
_.templateSettings = {
|
|
|
evaluate : /<%([\s\S]+?)%>/g,
|
|
|
interpolate : /<%=([\s\S]+?)%>/g,
|
|
|
escape : /<%-([\s\S]+?)%>/g
|
|
|
};
|
|
|
|
|
|
// When customizing `templateSettings`, if you don't want to define an
|
|
|
// interpolation, evaluation or escaping regex, we need one that is
|
|
|
// guaranteed not to match.
|
|
|
var noMatch = /(.)^/;
|
|
|
|
|
|
// Certain characters need to be escaped so that they can be put into a
|
|
|
// string literal.
|
|
|
var escapes = {
|
|
|
"'": "'",
|
|
|
'\\': '\\',
|
|
|
'\r': 'r',
|
|
|
'\n': 'n',
|
|
|
'\u2028': 'u2028',
|
|
|
'\u2029': 'u2029'
|
|
|
};
|
|
|
|
|
|
var escaper = /\\|'|\r|\n|\u2028|\u2029/g;
|
|
|
|
|
|
var escapeChar = function(match) {
|
|
|
return '\\' + escapes[match];
|
|
|
};
|
|
|
|
|
|
// JavaScript micro-templating, similar to John Resig's implementation.
|
|
|
// Underscore templating handles arbitrary delimiters, preserves whitespace,
|
|
|
// and correctly escapes quotes within interpolated code.
|
|
|
// NB: `oldSettings` only exists for backwards compatibility.
|
|
|
_.template = function(text, settings, oldSettings) {
|
|
|
if (!settings && oldSettings) settings = oldSettings;
|
|
|
settings = _.defaults({}, settings, _.templateSettings);
|
|
|
|
|
|
// Combine delimiters into one regular expression via alternation.
|
|
|
var matcher = RegExp([
|
|
|
(settings.escape || noMatch).source,
|
|
|
(settings.interpolate || noMatch).source,
|
|
|
(settings.evaluate || noMatch).source
|
|
|
].join('|') + '|$', 'g');
|
|
|
|
|
|
// Compile the template source, escaping string literals appropriately.
|
|
|
var index = 0;
|
|
|
var source = "__p+='";
|
|
|
text.replace(matcher, function(match, escape, interpolate, evaluate, offset) {
|
|
|
source += text.slice(index, offset).replace(escaper, escapeChar);
|
|
|
index = offset + match.length;
|
|
|
|
|
|
if (escape) {
|
|
|
source += "'+\n((__t=(" + escape + "))==null?'':_.escape(__t))+\n'";
|
|
|
} else if (interpolate) {
|
|
|
source += "'+\n((__t=(" + interpolate + "))==null?'':__t)+\n'";
|
|
|
} else if (evaluate) {
|
|
|
source += "';\n" + evaluate + "\n__p+='";
|
|
|
}
|
|
|
|
|
|
// Adobe VMs need the match returned to produce the correct offest.
|
|
|
return match;
|
|
|
});
|
|
|
source += "';\n";
|
|
|
|
|
|
// If a variable is not specified, place data values in local scope.
|
|
|
if (!settings.variable) source = 'with(obj||{}){\n' + source + '}\n';
|
|
|
|
|
|
source = "var __t,__p='',__j=Array.prototype.join," +
|
|
|
"print=function(){__p+=__j.call(arguments,'');};\n" +
|
|
|
source + 'return __p;\n';
|
|
|
|
|
|
try {
|
|
|
var render = new Function(settings.variable || 'obj', '_', source);
|
|
|
} catch (e) {
|
|
|
e.source = source;
|
|
|
throw e;
|
|
|
}
|
|
|
|
|
|
var template = function(data) {
|
|
|
return render.call(this, data, _);
|
|
|
};
|
|
|
|
|
|
// Provide the compiled source as a convenience for precompilation.
|
|
|
var argument = settings.variable || 'obj';
|
|
|
template.source = 'function(' + argument + '){\n' + source + '}';
|
|
|
|
|
|
return template;
|
|
|
};
|
|
|
|
|
|
// Add a "chain" function. Start chaining a wrapped Underscore object.
|
|
|
_.chain = function(obj) {
|
|
|
var instance = _(obj);
|
|
|
instance._chain = true;
|
|
|
return instance;
|
|
|
};
|
|
|
|
|
|
// OOP
|
|
|
// ---------------
|
|
|
// If Underscore is called as a function, it returns a wrapped object that
|
|
|
// can be used OO-style. This wrapper holds altered versions of all the
|
|
|
// underscore functions. Wrapped objects may be chained.
|
|
|
|
|
|
// Helper function to continue chaining intermediate results.
|
|
|
var result = function(obj) {
|
|
|
return this._chain ? _(obj).chain() : obj;
|
|
|
};
|
|
|
|
|
|
// Add your own custom functions to the Underscore object.
|
|
|
_.mixin = function(obj) {
|
|
|
_.each(_.functions(obj), function(name) {
|
|
|
var func = _[name] = obj[name];
|
|
|
_.prototype[name] = function() {
|
|
|
var args = [this._wrapped];
|
|
|
push.apply(args, arguments);
|
|
|
return result.call(this, func.apply(_, args));
|
|
|
};
|
|
|
});
|
|
|
};
|
|
|
|
|
|
// Add all of the Underscore functions to the wrapper object.
|
|
|
_.mixin(_);
|
|
|
|
|
|
// Add all mutator Array functions to the wrapper.
|
|
|
_.each(['pop', 'push', 'reverse', 'shift', 'sort', 'splice', 'unshift'], function(name) {
|
|
|
var method = ArrayProto[name];
|
|
|
_.prototype[name] = function() {
|
|
|
var obj = this._wrapped;
|
|
|
method.apply(obj, arguments);
|
|
|
if ((name === 'shift' || name === 'splice') && obj.length === 0) delete obj[0];
|
|
|
return result.call(this, obj);
|
|
|
};
|
|
|
});
|
|
|
|
|
|
// Add all accessor Array functions to the wrapper.
|
|
|
_.each(['concat', 'join', 'slice'], function(name) {
|
|
|
var method = ArrayProto[name];
|
|
|
_.prototype[name] = function() {
|
|
|
return result.call(this, method.apply(this._wrapped, arguments));
|
|
|
};
|
|
|
});
|
|
|
|
|
|
// Extracts the result from a wrapped and chained object.
|
|
|
_.prototype.value = function() {
|
|
|
return this._wrapped;
|
|
|
};
|
|
|
|
|
|
// AMD registration happens at the end for compatibility with AMD loaders
|
|
|
// that may not enforce next-turn semantics on modules. Even though general
|
|
|
// practice for AMD registration is to be anonymous, underscore registers
|
|
|
// as a named module because, like jQuery, it is a base library that is
|
|
|
// popular enough to be bundled in a third party lib, but not be part of
|
|
|
// an AMD load request. Those cases could generate an error when an
|
|
|
// anonymous define() is called outside of a loader request.
|
|
|
if (typeof define === 'function' && define.amd) {
|
|
|
define('underscore', [], function() {
|
|
|
return _;
|
|
|
});
|
|
|
}
|
|
|
}.call(this));
|
|
|
|
|
|
// Backbone.js 1.1.2
|
|
|
|
|
|
// (c) 2010-2014 Jeremy Ashkenas, DocumentCloud and Investigative Reporters & Editors
|
|
|
// Backbone may be freely distributed under the MIT license.
|
|
|
// For all details and documentation:
|
|
|
// http://backbonejs.org
|
|
|
|
|
|
(function(root, factory) {
|
|
|
|
|
|
// Set up Backbone appropriately for the environment. Start with AMD.
|
|
|
if (typeof define === 'function' && define.amd) {
|
|
|
define(['underscore', 'jquery', 'exports'], function(_, $, exports) {
|
|
|
// Export global even in AMD case in case this script is loaded with
|
|
|
// others that may still expect a global Backbone.
|
|
|
root.Backbone = factory(root, exports, _, $);
|
|
|
});
|
|
|
|
|
|
// Next for Node.js or CommonJS. jQuery may not be needed as a module.
|
|
|
} else if (typeof exports !== 'undefined') {
|
|
|
var _ = require('underscore');
|
|
|
factory(root, exports, _);
|
|
|
|
|
|
// Finally, as a browser global.
|
|
|
} else {
|
|
|
root.Backbone = factory(root, {}, root._, (root.jQuery || root.Zepto || root.ender || root.$));
|
|
|
}
|
|
|
|
|
|
}(this, function(root, Backbone, _, $) {
|
|
|
|
|
|
// Initial Setup
|
|
|
// -------------
|
|
|
|
|
|
// Save the previous value of the `Backbone` variable, so that it can be
|
|
|
// restored later on, if `noConflict` is used.
|
|
|
var previousBackbone = root.Backbone;
|
|
|
|
|
|
// Create local references to array methods we'll want to use later.
|
|
|
var array = [];
|
|
|
var push = array.push;
|
|
|
var slice = array.slice;
|
|
|
var splice = array.splice;
|
|
|
|
|
|
// Current version of the library. Keep in sync with `package.json`.
|
|
|
Backbone.VERSION = '1.1.2';
|
|
|
|
|
|
// For Backbone's purposes, jQuery, Zepto, Ender, or My Library (kidding) owns
|
|
|
// the `$` variable.
|
|
|
Backbone.$ = $;
|
|
|
|
|
|
// Runs Backbone.js in *noConflict* mode, returning the `Backbone` variable
|
|
|
// to its previous owner. Returns a reference to this Backbone object.
|
|
|
Backbone.noConflict = function() {
|
|
|
root.Backbone = previousBackbone;
|
|
|
return this;
|
|
|
};
|
|
|
|
|
|
// Turn on `emulateHTTP` to support legacy HTTP servers. Setting this option
|
|
|
// will fake `"PATCH"`, `"PUT"` and `"DELETE"` requests via the `_method` parameter and
|
|
|
// set a `X-Http-Method-Override` header.
|
|
|
Backbone.emulateHTTP = false;
|
|
|
|
|
|
// Turn on `emulateJSON` to support legacy servers that can't deal with direct
|
|
|
// `application/json` requests ... will encode the body as
|
|
|
// `application/x-www-form-urlencoded` instead and will send the model in a
|
|
|
// form param named `model`.
|
|
|
Backbone.emulateJSON = false;
|
|
|
|
|
|
// Backbone.Events
|
|
|
// ---------------
|
|
|
|
|
|
// A module that can be mixed in to *any object* in order to provide it with
|
|
|
// custom events. You may bind with `on` or remove with `off` callback
|
|
|
// functions to an event; `trigger`-ing an event fires all callbacks in
|
|
|
// succession.
|
|
|
//
|
|
|
// var object = {};
|
|
|
// _.extend(object, Backbone.Events);
|
|
|
// object.on('expand', function(){ alert('expanded'); });
|
|
|
// object.trigger('expand');
|
|
|
//
|
|
|
var Events = Backbone.Events = {
|
|
|
|
|
|
// Bind an event to a `callback` function. Passing `"all"` will bind
|
|
|
// the callback to all events fired.
|
|
|
on: function(name, callback, context) {
|
|
|
if (!eventsApi(this, 'on', name, [callback, context]) || !callback) return this;
|
|
|
this._events || (this._events = {});
|
|
|
var events = this._events[name] || (this._events[name] = []);
|
|
|
events.push({callback: callback, context: context, ctx: context || this});
|
|
|
return this;
|
|
|
},
|
|
|
|
|
|
// Bind an event to only be triggered a single time. After the first time
|
|
|
// the callback is invoked, it will be removed.
|
|
|
once: function(name, callback, context) {
|
|
|
if (!eventsApi(this, 'once', name, [callback, context]) || !callback) return this;
|
|
|
var self = this;
|
|
|
var once = _.once(function() {
|
|
|
self.off(name, once);
|
|
|
callback.apply(this, arguments);
|
|
|
});
|
|
|
once._callback = callback;
|
|
|
return this.on(name, once, context);
|
|
|
},
|
|
|
|
|
|
// Remove one or many callbacks. If `context` is null, removes all
|
|
|
// callbacks with that function. If `callback` is null, removes all
|
|
|
// callbacks for the event. If `name` is null, removes all bound
|
|
|
// callbacks for all events.
|
|
|
off: function(name, callback, context) {
|
|
|
var retain, ev, events, names, i, l, j, k;
|
|
|
if (!this._events || !eventsApi(this, 'off', name, [callback, context])) return this;
|
|
|
if (!name && !callback && !context) {
|
|
|
this._events = void 0;
|
|
|
return this;
|
|
|
}
|
|
|
names = name ? [name] : _.keys(this._events);
|
|
|
for (i = 0, l = names.length; i < l; i++) {
|
|
|
name = names[i];
|
|
|
if (events = this._events[name]) {
|
|
|
this._events[name] = retain = [];
|
|
|
if (callback || context) {
|
|
|
for (j = 0, k = events.length; j < k; j++) {
|
|
|
ev = events[j];
|
|
|
if ((callback && callback !== ev.callback && callback !== ev.callback._callback) ||
|
|
|
(context && context !== ev.context)) {
|
|
|
retain.push(ev);
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
if (!retain.length) delete this._events[name];
|
|
|
}
|
|
|
}
|
|
|
|
|
|
return this;
|
|
|
},
|
|
|
|
|
|
// Trigger one or many events, firing all bound callbacks. Callbacks are
|
|
|
// passed the same arguments as `trigger` is, apart from the event name
|
|
|
// (unless you're listening on `"all"`, which will cause your callback to
|
|
|
// receive the true name of the event as the first argument).
|
|
|
trigger: function(name) {
|
|
|
if (!this._events) return this;
|
|
|
var args = slice.call(arguments, 1);
|
|
|
if (!eventsApi(this, 'trigger', name, args)) return this;
|
|
|
var events = this._events[name];
|
|
|
var allEvents = this._events.all;
|
|
|
if (events) triggerEvents(events, args);
|
|
|
if (allEvents) triggerEvents(allEvents, arguments);
|
|
|
return this;
|
|
|
},
|
|
|
|
|
|
// Tell this object to stop listening to either specific events ... or
|
|
|
// to every object it's currently listening to.
|
|
|
stopListening: function(obj, name, callback) {
|
|
|
var listeningTo = this._listeningTo;
|
|
|
if (!listeningTo) return this;
|
|
|
var remove = !name && !callback;
|
|
|
if (!callback && typeof name === 'object') callback = this;
|
|
|
if (obj) (listeningTo = {})[obj._listenId] = obj;
|
|
|
for (var id in listeningTo) {
|
|
|
obj = listeningTo[id];
|
|
|
obj.off(name, callback, this);
|
|
|
if (remove || _.isEmpty(obj._events)) delete this._listeningTo[id];
|
|
|
}
|
|
|
return this;
|
|
|
}
|
|
|
|
|
|
};
|
|
|
|
|
|
// Regular expression used to split event strings.
|
|
|
var eventSplitter = /\s+/;
|
|
|
|
|
|
// Implement fancy features of the Events API such as multiple event
|
|
|
// names `"change blur"` and jQuery-style event maps `{change: action}`
|
|
|
// in terms of the existing API.
|
|
|
var eventsApi = function(obj, action, name, rest) {
|
|
|
if (!name) return true;
|
|
|
|
|
|
// Handle event maps.
|
|
|
if (typeof name === 'object') {
|
|
|
for (var key in name) {
|
|
|
obj[action].apply(obj, [key, name[key]].concat(rest));
|
|
|
}
|
|
|
return false;
|
|
|
}
|
|
|
|
|
|
// Handle space separated event names.
|
|
|
if (eventSplitter.test(name)) {
|
|
|
var names = name.split(eventSplitter);
|
|
|
for (var i = 0, l = names.length; i < l; i++) {
|
|
|
obj[action].apply(obj, [names[i]].concat(rest));
|
|
|
}
|
|
|
return false;
|
|
|
}
|
|
|
|
|
|
return true;
|
|
|
};
|
|
|
|
|
|
// A difficult-to-believe, but optimized internal dispatch function for
|
|
|
// triggering events. Tries to keep the usual cases speedy (most internal
|
|
|
// Backbone events have 3 arguments).
|
|
|
var triggerEvents = function(events, args) {
|
|
|
var ev, i = -1, l = events.length, a1 = args[0], a2 = args[1], a3 = args[2];
|
|
|
switch (args.length) {
|
|
|
case 0: while (++i < l) (ev = events[i]).callback.call(ev.ctx); return;
|
|
|
case 1: while (++i < l) (ev = events[i]).callback.call(ev.ctx, a1); return;
|
|
|
case 2: while (++i < l) (ev = events[i]).callback.call(ev.ctx, a1, a2); return;
|
|
|
case 3: while (++i < l) (ev = events[i]).callback.call(ev.ctx, a1, a2, a3); return;
|
|
|
default: while (++i < l) (ev = events[i]).callback.apply(ev.ctx, args); return;
|
|
|
}
|
|
|
};
|
|
|
|
|
|
var listenMethods = {listenTo: 'on', listenToOnce: 'once'};
|
|
|
|
|
|
// Inversion-of-control versions of `on` and `once`. Tell *this* object to
|
|
|
// listen to an event in another object ... keeping track of what it's
|
|
|
// listening to.
|
|
|
_.each(listenMethods, function(implementation, method) {
|
|
|
Events[method] = function(obj, name, callback) {
|
|
|
var listeningTo = this._listeningTo || (this._listeningTo = {});
|
|
|
var id = obj._listenId || (obj._listenId = _.uniqueId('l'));
|
|
|
listeningTo[id] = obj;
|
|
|
if (!callback && typeof name === 'object') callback = this;
|
|
|
obj[implementation](name, callback, this);
|
|
|
return this;
|
|
|
};
|
|
|
});
|
|
|
|
|
|
// Aliases for backwards compatibility.
|
|
|
Events.bind = Events.on;
|
|
|
Events.unbind = Events.off;
|
|
|
|
|
|
// Allow the `Backbone` object to serve as a global event bus, for folks who
|
|
|
// want global "pubsub" in a convenient place.
|
|
|
_.extend(Backbone, Events);
|
|
|
|
|
|
// Backbone.Model
|
|
|
// --------------
|
|
|
|
|
|
// Backbone **Models** are the basic data object in the framework --
|
|
|
// frequently representing a row in a table in a database on your server.
|
|
|
// A discrete chunk of data and a bunch of useful, related methods for
|
|
|
// performing computations and transformations on that data.
|
|
|
|
|
|
// Create a new model with the specified attributes. A client id (`cid`)
|
|
|
// is automatically generated and assigned for you.
|
|
|
var Model = Backbone.Model = function(attributes, options) {
|
|
|
var attrs = attributes || {};
|
|
|
options || (options = {});
|
|
|
this.cid = _.uniqueId('c');
|
|
|
this.attributes = {};
|
|
|
if (options.collection) this.collection = options.collection;
|
|
|
if (options.parse) attrs = this.parse(attrs, options) || {};
|
|
|
attrs = _.defaults({}, attrs, _.result(this, 'defaults'));
|
|
|
this.set(attrs, options);
|
|
|
this.changed = {};
|
|
|
this.initialize.apply(this, arguments);
|
|
|
};
|
|
|
|
|
|
// Attach all inheritable methods to the Model prototype.
|
|
|
_.extend(Model.prototype, Events, {
|
|
|
|
|
|
// A hash of attributes whose current and previous value differ.
|
|
|
changed: null,
|
|
|
|
|
|
// The value returned during the last failed validation.
|
|
|
validationError: null,
|
|
|
|
|
|
// The default name for the JSON `id` attribute is `"id"`. MongoDB and
|
|
|
// CouchDB users may want to set this to `"_id"`.
|
|
|
idAttribute: 'id',
|
|
|
|
|
|
// Initialize is an empty function by default. Override it with your own
|
|
|
// initialization logic.
|
|
|
initialize: function(){},
|
|
|
|
|
|
// Return a copy of the model's `attributes` object.
|
|
|
toJSON: function(options) {
|
|
|
return _.clone(this.attributes);
|
|
|
},
|
|
|
|
|
|
// Proxy `Backbone.sync` by default -- but override this if you need
|
|
|
// custom syncing semantics for *this* particular model.
|
|
|
sync: function() {
|
|
|
return Backbone.sync.apply(this, arguments);
|
|
|
},
|
|
|
|
|
|
// Get the value of an attribute.
|
|
|
get: function(attr) {
|
|
|
return this.attributes[attr];
|
|
|
},
|
|
|
|
|
|
// Get the HTML-escaped value of an attribute.
|
|
|
escape: function(attr) {
|
|
|
return _.escape(this.get(attr));
|
|
|
},
|
|
|
|
|
|
// Returns `true` if the attribute contains a value that is not null
|
|
|
// or undefined.
|
|
|
has: function(attr) {
|
|
|
return this.get(attr) != null;
|
|
|
},
|
|
|
|
|
|
// Set a hash of model attributes on the object, firing `"change"`. This is
|
|
|
// the core primitive operation of a model, updating the data and notifying
|
|
|
// anyone who needs to know about the change in state. The heart of the beast.
|
|
|
set: function(key, val, options) {
|
|
|
var attr, attrs, unset, changes, silent, changing, prev, current;
|
|
|
if (key == null) return this;
|
|
|
|
|
|
// Handle both `"key", value` and `{key: value}` -style arguments.
|
|
|
if (typeof key === 'object') {
|
|
|
attrs = key;
|
|
|
options = val;
|
|
|
} else {
|
|
|
(attrs = {})[key] = val;
|
|
|
}
|
|
|
|
|
|
options || (options = {});
|
|
|
|
|
|
// Run validation.
|
|
|
if (!this._validate(attrs, options)) return false;
|
|
|
|
|
|
// Extract attributes and options.
|
|
|
unset = options.unset;
|
|
|
silent = options.silent;
|
|
|
changes = [];
|
|
|
changing = this._changing;
|
|
|
this._changing = true;
|
|
|
|
|
|
if (!changing) {
|
|
|
this._previousAttributes = _.clone(this.attributes);
|
|
|
this.changed = {};
|
|
|
}
|
|
|
current = this.attributes, prev = this._previousAttributes;
|
|
|
|
|
|
// Check for changes of `id`.
|
|
|
if (this.idAttribute in attrs) this.id = attrs[this.idAttribute];
|
|
|
|
|
|
// For each `set` attribute, update or delete the current value.
|
|
|
for (attr in attrs) {
|
|
|
val = attrs[attr];
|
|
|
if (!_.isEqual(current[attr], val)) changes.push(attr);
|
|
|
if (!_.isEqual(prev[attr], val)) {
|
|
|
this.changed[attr] = val;
|
|
|
} else {
|
|
|
delete this.changed[attr];
|
|
|
}
|
|
|
unset ? delete current[attr] : current[attr] = val;
|
|
|
}
|
|
|
|
|
|
// Trigger all relevant attribute changes.
|
|
|
if (!silent) {
|
|
|
if (changes.length) this._pending = options;
|
|
|
for (var i = 0, l = changes.length; i < l; i++) {
|
|
|
this.trigger('change:' + changes[i], this, current[changes[i]], options);
|
|
|
}
|
|
|
}
|
|
|
|
|
|
// You might be wondering why there's a `while` loop here. Changes can
|
|
|
// be recursively nested within `"change"` events.
|
|
|
if (changing) return this;
|
|
|
if (!silent) {
|
|
|
while (this._pending) {
|
|
|
options = this._pending;
|
|
|
this._pending = false;
|
|
|
this.trigger('change', this, options);
|
|
|
}
|
|
|
}
|
|
|
this._pending = false;
|
|
|
this._changing = false;
|
|
|
return this;
|
|
|
},
|
|
|
|
|
|
// Remove an attribute from the model, firing `"change"`. `unset` is a noop
|
|
|
// if the attribute doesn't exist.
|
|
|
unset: function(attr, options) {
|
|
|
return this.set(attr, void 0, _.extend({}, options, {unset: true}));
|
|
|
},
|
|
|
|
|
|
// Clear all attributes on the model, firing `"change"`.
|
|
|
clear: function(options) {
|
|
|
var attrs = {};
|
|
|
for (var key in this.attributes) attrs[key] = void 0;
|
|
|
return this.set(attrs, _.extend({}, options, {unset: true}));
|
|
|
},
|
|
|
|
|
|
// Determine if the model has changed since the last `"change"` event.
|
|
|
// If you specify an attribute name, determine if that attribute has changed.
|
|
|
hasChanged: function(attr) {
|
|
|
if (attr == null) return !_.isEmpty(this.changed);
|
|
|
return _.has(this.changed, attr);
|
|
|
},
|
|
|
|
|
|
// Return an object containing all the attributes that have changed, or
|
|
|
// false if there are no changed attributes. Useful for determining what
|
|
|
// parts of a view need to be updated and/or what attributes need to be
|
|
|
// persisted to the server. Unset attributes will be set to undefined.
|
|
|
// You can also pass an attributes object to diff against the model,
|
|
|
// determining if there *would be* a change.
|
|
|
changedAttributes: function(diff) {
|
|
|
if (!diff) return this.hasChanged() ? _.clone(this.changed) : false;
|
|
|
var val, changed = false;
|
|
|
var old = this._changing ? this._previousAttributes : this.attributes;
|
|
|
for (var attr in diff) {
|
|
|
if (_.isEqual(old[attr], (val = diff[attr]))) continue;
|
|
|
(changed || (changed = {}))[attr] = val;
|
|
|
}
|
|
|
return changed;
|
|
|
},
|
|
|
|
|
|
// Get the previous value of an attribute, recorded at the time the last
|
|
|
// `"change"` event was fired.
|
|
|
previous: function(attr) {
|
|
|
if (attr == null || !this._previousAttributes) return null;
|
|
|
return this._previousAttributes[attr];
|
|
|
},
|
|
|
|
|
|
// Get all of the attributes of the model at the time of the previous
|
|
|
// `"change"` event.
|
|
|
previousAttributes: function() {
|
|
|
return _.clone(this._previousAttributes);
|
|
|
},
|
|
|
|
|
|
// Fetch the model from the server. If the server's representation of the
|
|
|
// model differs from its current attributes, they will be overridden,
|
|
|
// triggering a `"change"` event.
|
|
|
fetch: function(options) {
|
|
|
options = options ? _.clone(options) : {};
|
|
|
if (options.parse === void 0) options.parse = true;
|
|
|
var model = this;
|
|
|
var success = options.success;
|
|
|
options.success = function(resp) {
|
|
|
if (!model.set(model.parse(resp, options), options)) return false;
|
|
|
if (success) success(model, resp, options);
|
|
|
model.trigger('sync', model, resp, options);
|
|
|
};
|
|
|
wrapError(this, options);
|
|
|
return this.sync('read', this, options);
|
|
|
},
|
|
|
|
|
|
// Set a hash of model attributes, and sync the model to the server.
|
|
|
// If the server returns an attributes hash that differs, the model's
|
|
|
// state will be `set` again.
|
|
|
save: function(key, val, options) {
|
|
|
var attrs, method, xhr, attributes = this.attributes;
|
|
|
|
|
|
// Handle both `"key", value` and `{key: value}` -style arguments.
|
|
|
if (key == null || typeof key === 'object') {
|
|
|
attrs = key;
|
|
|
options = val;
|
|
|
} else {
|
|
|
(attrs = {})[key] = val;
|
|
|
}
|
|
|
|
|
|
options = _.extend({validate: true}, options);
|
|
|
|
|
|
// If we're not waiting and attributes exist, save acts as
|
|
|
// `set(attr).save(null, opts)` with validation. Otherwise, check if
|
|
|
// the model will be valid when the attributes, if any, are set.
|
|
|
if (attrs && !options.wait) {
|
|
|
if (!this.set(attrs, options)) return false;
|
|
|
} else {
|
|
|
if (!this._validate(attrs, options)) return false;
|
|
|
}
|
|
|
|
|
|
// Set temporary attributes if `{wait: true}`.
|
|
|
if (attrs && options.wait) {
|
|
|
this.attributes = _.extend({}, attributes, attrs);
|
|
|
}
|
|
|
|
|
|
// After a successful server-side save, the client is (optionally)
|
|
|
// updated with the server-side state.
|
|
|
if (options.parse === void 0) options.parse = true;
|
|
|
var model = this;
|
|
|
var success = options.success;
|
|
|
options.success = function(resp) {
|
|
|
// Ensure attributes are restored during synchronous saves.
|
|
|
model.attributes = attributes;
|
|
|
var serverAttrs = model.parse(resp, options);
|
|
|
if (options.wait) serverAttrs = _.extend(attrs || {}, serverAttrs);
|
|
|
if (_.isObject(serverAttrs) && !model.set(serverAttrs, options)) {
|
|
|
return false;
|
|
|
}
|
|
|
if (success) success(model, resp, options);
|
|
|
model.trigger('sync', model, resp, options);
|
|
|
};
|
|
|
wrapError(this, options);
|
|
|
|
|
|
method = this.isNew() ? 'create' : (options.patch ? 'patch' : 'update');
|
|
|
if (method === 'patch') options.attrs = attrs;
|
|
|
xhr = this.sync(method, this, options);
|
|
|
|
|
|
// Restore attributes.
|
|
|
if (attrs && options.wait) this.attributes = attributes;
|
|
|
|
|
|
return xhr;
|
|
|
},
|
|
|
|
|
|
// Destroy this model on the server if it was already persisted.
|
|
|
// Optimistically removes the model from its collection, if it has one.
|
|
|
// If `wait: true` is passed, waits for the server to respond before removal.
|
|
|
destroy: function(options) {
|
|
|
options = options ? _.clone(options) : {};
|
|
|
var model = this;
|
|
|
var success = options.success;
|
|
|
|
|
|
var destroy = function() {
|
|
|
model.trigger('destroy', model, model.collection, options);
|
|
|
};
|
|
|
|
|
|
options.success = function(resp) {
|
|
|
if (options.wait || model.isNew()) destroy();
|
|
|
if (success) success(model, resp, options);
|
|
|
if (!model.isNew()) model.trigger('sync', model, resp, options);
|
|
|
};
|
|
|
|
|
|
if (this.isNew()) {
|
|
|
options.success();
|
|
|
return false;
|
|
|
}
|
|
|
wrapError(this, options);
|
|
|
|
|
|
var xhr = this.sync('delete', this, options);
|
|
|
if (!options.wait) destroy();
|
|
|
return xhr;
|
|
|
},
|
|
|
|
|
|
// Default URL for the model's representation on the server -- if you're
|
|
|
// using Backbone's restful methods, override this to change the endpoint
|
|
|
// that will be called.
|
|
|
url: function() {
|
|
|
var base =
|
|
|
_.result(this, 'urlRoot') ||
|
|
|
_.result(this.collection, 'url') ||
|
|
|
urlError();
|
|
|
if (this.isNew()) return base;
|
|
|
return base.replace(/([^\/])$/, '$1/') + encodeURIComponent(this.id);
|
|
|
},
|
|
|
|
|
|
// **parse** converts a response into the hash of attributes to be `set` on
|
|
|
// the model. The default implementation is just to pass the response along.
|
|
|
parse: function(resp, options) {
|
|
|
return resp;
|
|
|
},
|
|
|
|
|
|
// Create a new model with identical attributes to this one.
|
|
|
clone: function() {
|
|
|
return new this.constructor(this.attributes);
|
|
|
},
|
|
|
|
|
|
// A model is new if it has never been saved to the server, and lacks an id.
|
|
|
isNew: function() {
|
|
|
return !this.has(this.idAttribute);
|
|
|
},
|
|
|
|
|
|
// Check if the model is currently in a valid state.
|
|
|
isValid: function(options) {
|
|
|
return this._validate({}, _.extend(options || {}, { validate: true }));
|
|
|
},
|
|
|
|
|
|
// Run validation against the next complete set of model attributes,
|
|
|
// returning `true` if all is well. Otherwise, fire an `"invalid"` event.
|
|
|
_validate: function(attrs, options) {
|
|
|
if (!options.validate || !this.validate) return true;
|
|
|
attrs = _.extend({}, this.attributes, attrs);
|
|
|
var error = this.validationError = this.validate(attrs, options) || null;
|
|
|
if (!error) return true;
|
|
|
this.trigger('invalid', this, error, _.extend(options, {validationError: error}));
|
|
|
return false;
|
|
|
}
|
|
|
|
|
|
});
|
|
|
|
|
|
// Underscore methods that we want to implement on the Model.
|
|
|
var modelMethods = ['keys', 'values', 'pairs', 'invert', 'pick', 'omit'];
|
|
|
|
|
|
// Mix in each Underscore method as a proxy to `Model#attributes`.
|
|
|
_.each(modelMethods, function(method) {
|
|
|
Model.prototype[method] = function() {
|
|
|
var args = slice.call(arguments);
|
|
|
args.unshift(this.attributes);
|
|
|
return _[method].apply(_, args);
|
|
|
};
|
|
|
});
|
|
|
|
|
|
// Backbone.Collection
|
|
|
// -------------------
|
|
|
|
|
|
// If models tend to represent a single row of data, a Backbone Collection is
|
|
|
// more analagous to a table full of data ... or a small slice or page of that
|
|
|
// table, or a collection of rows that belong together for a particular reason
|
|
|
// -- all of the messages in this particular folder, all of the documents
|
|
|
// belonging to this particular author, and so on. Collections maintain
|
|
|
// indexes of their models, both in order, and for lookup by `id`.
|
|
|
|
|
|
// Create a new **Collection**, perhaps to contain a specific type of `model`.
|
|
|
// If a `comparator` is specified, the Collection will maintain
|
|
|
// its models in sort order, as they're added and removed.
|
|
|
var Collection = Backbone.Collection = function(models, options) {
|
|
|
options || (options = {});
|
|
|
if (options.model) this.model = options.model;
|
|
|
if (options.comparator !== void 0) this.comparator = options.comparator;
|
|
|
this._reset();
|
|
|
this.initialize.apply(this, arguments);
|
|
|
if (models) this.reset(models, _.extend({silent: true}, options));
|
|
|
};
|
|
|
|
|
|
// Default options for `Collection#set`.
|
|
|
var setOptions = {add: true, remove: true, merge: true};
|
|
|
var addOptions = {add: true, remove: false};
|
|
|
|
|
|
// Define the Collection's inheritable methods.
|
|
|
_.extend(Collection.prototype, Events, {
|
|
|
|
|
|
// The default model for a collection is just a **Backbone.Model**.
|
|
|
// This should be overridden in most cases.
|
|
|
model: Model,
|
|
|
|
|
|
// Initialize is an empty function by default. Override it with your own
|
|
|
// initialization logic.
|
|
|
initialize: function(){},
|
|
|
|
|
|
// The JSON representation of a Collection is an array of the
|
|
|
// models' attributes.
|
|
|
toJSON: function(options) {
|
|
|
return this.map(function(model){ return model.toJSON(options); });
|
|
|
},
|
|
|
|
|
|
// Proxy `Backbone.sync` by default.
|
|
|
sync: function() {
|
|
|
return Backbone.sync.apply(this, arguments);
|
|
|
},
|
|
|
|
|
|
// Add a model, or list of models to the set.
|
|
|
add: function(models, options) {
|
|
|
return this.set(models, _.extend({merge: false}, options, addOptions));
|
|
|
},
|
|
|
|
|
|
// Remove a model, or a list of models from the set.
|
|
|
remove: function(models, options) {
|
|
|
var singular = !_.isArray(models);
|
|
|
models = singular ? [models] : _.clone(models);
|
|
|
options || (options = {});
|
|
|
var i, l, index, model;
|
|
|
for (i = 0, l = models.length; i < l; i++) {
|
|
|
model = models[i] = this.get(models[i]);
|
|
|
if (!model) continue;
|
|
|
delete this._byId[model.id];
|
|
|
delete this._byId[model.cid];
|
|
|
index = this.indexOf(model);
|
|
|
this.models.splice(index, 1);
|
|
|
this.length--;
|
|
|
if (!options.silent) {
|
|
|
options.index = index;
|
|
|
model.trigger('remove', model, this, options);
|
|
|
}
|
|
|
this._removeReference(model, options);
|
|
|
}
|
|
|
return singular ? models[0] : models;
|
|
|
},
|
|
|
|
|
|
// Update a collection by `set`-ing a new list of models, adding new ones,
|
|
|
// removing models that are no longer present, and merging models that
|
|
|
// already exist in the collection, as necessary. Similar to **Model#set**,
|
|
|
// the core operation for updating the data contained by the collection.
|
|
|
set: function(models, options) {
|
|
|
options = _.defaults({}, options, setOptions);
|
|
|
if (options.parse) models = this.parse(models, options);
|
|
|
var singular = !_.isArray(models);
|
|
|
models = singular ? (models ? [models] : []) : _.clone(models);
|
|
|
var i, l, id, model, attrs, existing, sort;
|
|
|
var at = options.at;
|
|
|
var targetModel = this.model;
|
|
|
var sortable = this.comparator && (at == null) && options.sort !== false;
|
|
|
var sortAttr = _.isString(this.comparator) ? this.comparator : null;
|
|
|
var toAdd = [], toRemove = [], modelMap = {};
|
|
|
var add = options.add, merge = options.merge, remove = options.remove;
|
|
|
var order = !sortable && add && remove ? [] : false;
|
|
|
|
|
|
// Turn bare objects into model references, and prevent invalid models
|
|
|
// from being added.
|
|
|
for (i = 0, l = models.length; i < l; i++) {
|
|
|
attrs = models[i] || {};
|
|
|
if (attrs instanceof Model) {
|
|
|
id = model = attrs;
|
|
|
} else {
|
|
|
id = attrs[targetModel.prototype.idAttribute || 'id'];
|
|
|
}
|
|
|
|
|
|
// If a duplicate is found, prevent it from being added and
|
|
|
// optionally merge it into the existing model.
|
|
|
if (existing = this.get(id)) {
|
|
|
if (remove) modelMap[existing.cid] = true;
|
|
|
if (merge) {
|
|
|
attrs = attrs === model ? model.attributes : attrs;
|
|
|
if (options.parse) attrs = existing.parse(attrs, options);
|
|
|
existing.set(attrs, options);
|
|
|
if (sortable && !sort && existing.hasChanged(sortAttr)) sort = true;
|
|
|
}
|
|
|
models[i] = existing;
|
|
|
|
|
|
// If this is a new, valid model, push it to the `toAdd` list.
|
|
|
} else if (add) {
|
|
|
model = models[i] = this._prepareModel(attrs, options);
|
|
|
if (!model) continue;
|
|
|
toAdd.push(model);
|
|
|
this._addReference(model, options);
|
|
|
}
|
|
|
|
|
|
// Do not add multiple models with the same `id`.
|
|
|
model = existing || model;
|
|
|
if (order && (model.isNew() || !modelMap[model.id])) order.push(model);
|
|
|
modelMap[model.id] = true;
|
|
|
}
|
|
|
|
|
|
// Remove nonexistent models if appropriate.
|
|
|
if (remove) {
|
|
|
for (i = 0, l = this.length; i < l; ++i) {
|
|
|
if (!modelMap[(model = this.models[i]).cid]) toRemove.push(model);
|
|
|
}
|
|
|
if (toRemove.length) this.remove(toRemove, options);
|
|
|
}
|
|
|
|
|
|
// See if sorting is needed, update `length` and splice in new models.
|
|
|
if (toAdd.length || (order && order.length)) {
|
|
|
if (sortable) sort = true;
|
|
|
this.length += toAdd.length;
|
|
|
if (at != null) {
|
|
|
for (i = 0, l = toAdd.length; i < l; i++) {
|
|
|
this.models.splice(at + i, 0, toAdd[i]);
|
|
|
}
|
|
|
} else {
|
|
|
if (order) this.models.length = 0;
|
|
|
var orderedModels = order || toAdd;
|
|
|
for (i = 0, l = orderedModels.length; i < l; i++) {
|
|
|
this.models.push(orderedModels[i]);
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
|
|
|
// Silently sort the collection if appropriate.
|
|
|
if (sort) this.sort({silent: true});
|
|
|
|
|
|
// Unless silenced, it's time to fire all appropriate add/sort events.
|
|
|
if (!options.silent) {
|
|
|
for (i = 0, l = toAdd.length; i < l; i++) {
|
|
|
(model = toAdd[i]).trigger('add', model, this, options);
|
|
|
}
|
|
|
if (sort || (order && order.length)) this.trigger('sort', this, options);
|
|
|
}
|
|
|
|
|
|
// Return the added (or merged) model (or models).
|
|
|
return singular ? models[0] : models;
|
|
|
},
|
|
|
|
|
|
// When you have more items than you want to add or remove individually,
|
|
|
// you can reset the entire set with a new list of models, without firing
|
|
|
// any granular `add` or `remove` events. Fires `reset` when finished.
|
|
|
// Useful for bulk operations and optimizations.
|
|
|
reset: function(models, options) {
|
|
|
options || (options = {});
|
|
|
for (var i = 0, l = this.models.length; i < l; i++) {
|
|
|
this._removeReference(this.models[i], options);
|
|
|
}
|
|
|
options.previousModels = this.models;
|
|
|
this._reset();
|
|
|
models = this.add(models, _.extend({silent: true}, options));
|
|
|
if (!options.silent) this.trigger('reset', this, options);
|
|
|
return models;
|
|
|
},
|
|
|
|
|
|
// Add a model to the end of the collection.
|
|
|
push: function(model, options) {
|
|
|
return this.add(model, _.extend({at: this.length}, options));
|
|
|
},
|
|
|
|
|
|
// Remove a model from the end of the collection.
|
|
|
pop: function(options) {
|
|
|
var model = this.at(this.length - 1);
|
|
|
this.remove(model, options);
|
|
|
return model;
|
|
|
},
|
|
|
|
|
|
// Add a model to the beginning of the collection.
|
|
|
unshift: function(model, options) {
|
|
|
return this.add(model, _.extend({at: 0}, options));
|
|
|
},
|
|
|
|
|
|
// Remove a model from the beginning of the collection.
|
|
|
shift: function(options) {
|
|
|
var model = this.at(0);
|
|
|
this.remove(model, options);
|
|
|
return model;
|
|
|
},
|
|
|
|
|
|
// Slice out a sub-array of models from the collection.
|
|
|
slice: function() {
|
|
|
return slice.apply(this.models, arguments);
|
|
|
},
|
|
|
|
|
|
// Get a model from the set by id.
|
|
|
get: function(obj) {
|
|
|
if (obj == null) return void 0;
|
|
|
return this._byId[obj] || this._byId[obj.id] || this._byId[obj.cid];
|
|
|
},
|
|
|
|
|
|
// Get the model at the given index.
|
|
|
at: function(index) {
|
|
|
return this.models[index];
|
|
|
},
|
|
|
|
|
|
// Return models with matching attributes. Useful for simple cases of
|
|
|
// `filter`.
|
|
|
where: function(attrs, first) {
|
|
|
if (_.isEmpty(attrs)) return first ? void 0 : [];
|
|
|
return this[first ? 'find' : 'filter'](function(model) {
|
|
|
for (var key in attrs) {
|
|
|
if (attrs[key] !== model.get(key)) return false;
|
|
|
}
|
|
|
return true;
|
|
|
});
|
|
|
},
|
|
|
|
|
|
// Return the first model with matching attributes. Useful for simple cases
|
|
|
// of `find`.
|
|
|
findWhere: function(attrs) {
|
|
|
return this.where(attrs, true);
|
|
|
},
|
|
|
|
|
|
// Force the collection to re-sort itself. You don't need to call this under
|
|
|
// normal circumstances, as the set will maintain sort order as each item
|
|
|
// is added.
|
|
|
sort: function(options) {
|
|
|
if (!this.comparator) throw new Error('Cannot sort a set without a comparator');
|
|
|
options || (options = {});
|
|
|
|
|
|
// Run sort based on type of `comparator`.
|
|
|
if (_.isString(this.comparator) || this.comparator.length === 1) {
|
|
|
this.models = this.sortBy(this.comparator, this);
|
|
|
} else {
|
|
|
this.models.sort(_.bind(this.comparator, this));
|
|
|
}
|
|
|
|
|
|
if (!options.silent) this.trigger('sort', this, options);
|
|
|
return this;
|
|
|
},
|
|
|
|
|
|
// Pluck an attribute from each model in the collection.
|
|
|
pluck: function(attr) {
|
|
|
return _.invoke(this.models, 'get', attr);
|
|
|
},
|
|
|
|
|
|
// Fetch the default set of models for this collection, resetting the
|
|
|
// collection when they arrive. If `reset: true` is passed, the response
|
|
|
// data will be passed through the `reset` method instead of `set`.
|
|
|
fetch: function(options) {
|
|
|
options = options ? _.clone(options) : {};
|
|
|
if (options.parse === void 0) options.parse = true;
|
|
|
var success = options.success;
|
|
|
var collection = this;
|
|
|
options.success = function(resp) {
|
|
|
var method = options.reset ? 'reset' : 'set';
|
|
|
collection[method](resp, options);
|
|
|
if (success) success(collection, resp, options);
|
|
|
collection.trigger('sync', collection, resp, options);
|
|
|
};
|
|
|
wrapError(this, options);
|
|
|
return this.sync('read', this, options);
|
|
|
},
|
|
|
|
|
|
// Create a new instance of a model in this collection. Add the model to the
|
|
|
// collection immediately, unless `wait: true` is passed, in which case we
|
|
|
// wait for the server to agree.
|
|
|
create: function(model, options) {
|
|
|
options = options ? _.clone(options) : {};
|
|
|
if (!(model = this._prepareModel(model, options))) return false;
|
|
|
if (!options.wait) this.add(model, options);
|
|
|
var collection = this;
|
|
|
var success = options.success;
|
|
|
options.success = function(model, resp) {
|
|
|
if (options.wait) collection.add(model, options);
|
|
|
if (success) success(model, resp, options);
|
|
|
};
|
|
|
model.save(null, options);
|
|
|
return model;
|
|
|
},
|
|
|
|
|
|
// **parse** converts a response into a list of models to be added to the
|
|
|
// collection. The default implementation is just to pass it through.
|
|
|
parse: function(resp, options) {
|
|
|
return resp;
|
|
|
},
|
|
|
|
|
|
// Create a new collection with an identical list of models as this one.
|
|
|
clone: function() {
|
|
|
return new this.constructor(this.models);
|
|
|
},
|
|
|
|
|
|
// Private method to reset all internal state. Called when the collection
|
|
|
// is first initialized or reset.
|
|
|
_reset: function() {
|
|
|
this.length = 0;
|
|
|
this.models = [];
|
|
|
this._byId = {};
|
|
|
},
|
|
|
|
|
|
// Prepare a hash of attributes (or other model) to be added to this
|
|
|
// collection.
|
|
|
_prepareModel: function(attrs, options) {
|
|
|
if (attrs instanceof Model) return attrs;
|
|
|
options = options ? _.clone(options) : {};
|
|
|
options.collection = this;
|
|
|
var model = new this.model(attrs, options);
|
|
|
if (!model.validationError) return model;
|
|
|
this.trigger('invalid', this, model.validationError, options);
|
|
|
return false;
|
|
|
},
|
|
|
|
|
|
// Internal method to create a model's ties to a collection.
|
|
|
_addReference: function(model, options) {
|
|
|
this._byId[model.cid] = model;
|
|
|
if (model.id != null) this._byId[model.id] = model;
|
|
|
if (!model.collection) model.collection = this;
|
|
|
model.on('all', this._onModelEvent, this);
|
|
|
},
|
|
|
|
|
|
// Internal method to sever a model's ties to a collection.
|
|
|
_removeReference: function(model, options) {
|
|
|
if (this === model.collection) delete model.collection;
|
|
|
model.off('all', this._onModelEvent, this);
|
|
|
},
|
|
|
|
|
|
// Internal method called every time a model in the set fires an event.
|
|
|
// Sets need to update their indexes when models change ids. All other
|
|
|
// events simply proxy through. "add" and "remove" events that originate
|
|
|
// in other collections are ignored.
|
|
|
_onModelEvent: function(event, model, collection, options) {
|
|
|
if ((event === 'add' || event === 'remove') && collection !== this) return;
|
|
|
if (event === 'destroy') this.remove(model, options);
|
|
|
if (model && event === 'change:' + model.idAttribute) {
|
|
|
delete this._byId[model.previous(model.idAttribute)];
|
|
|
if (model.id != null) this._byId[model.id] = model;
|
|
|
}
|
|
|
this.trigger.apply(this, arguments);
|
|
|
}
|
|
|
|
|
|
});
|
|
|
|
|
|
// Underscore methods that we want to implement on the Collection.
|
|
|
// 90% of the core usefulness of Backbone Collections is actually implemented
|
|
|
// right here:
|
|
|
var methods = ['forEach', 'each', 'map', 'collect', 'reduce', 'foldl',
|
|
|
'inject', 'reduceRight', 'foldr', 'find', 'detect', 'filter', 'select',
|
|
|
'reject', 'every', 'all', 'some', 'any', 'include', 'contains', 'invoke',
|
|
|
'max', 'min', 'toArray', 'size', 'first', 'head', 'take', 'initial', 'rest',
|
|
|
'tail', 'drop', 'last', 'without', 'difference', 'indexOf', 'shuffle',
|
|
|
'lastIndexOf', 'isEmpty', 'chain', 'sample'];
|
|
|
|
|
|
// Mix in each Underscore method as a proxy to `Collection#models`.
|
|
|
_.each(methods, function(method) {
|
|
|
Collection.prototype[method] = function() {
|
|
|
var args = slice.call(arguments);
|
|
|
args.unshift(this.models);
|
|
|
return _[method].apply(_, args);
|
|
|
};
|
|
|
});
|
|
|
|
|
|
// Underscore methods that take a property name as an argument.
|
|
|
var attributeMethods = ['groupBy', 'countBy', 'sortBy', 'indexBy'];
|
|
|
|
|
|
// Use attributes instead of properties.
|
|
|
_.each(attributeMethods, function(method) {
|
|
|
Collection.prototype[method] = function(value, context) {
|
|
|
var iterator = _.isFunction(value) ? value : function(model) {
|
|
|
return model.get(value);
|
|
|
};
|
|
|
return _[method](this.models, iterator, context);
|
|
|
};
|
|
|
});
|
|
|
|
|
|
// Backbone.View
|
|
|
// -------------
|
|
|
|
|
|
// Backbone Views are almost more convention than they are actual code. A View
|
|
|
// is simply a JavaScript object that represents a logical chunk of UI in the
|
|
|
// DOM. This might be a single item, an entire list, a sidebar or panel, or
|
|
|
// even the surrounding frame which wraps your whole app. Defining a chunk of
|
|
|
// UI as a **View** allows you to define your DOM events declaratively, without
|
|
|
// having to worry about render order ... and makes it easy for the view to
|
|
|
// react to specific changes in the state of your models.
|
|
|
|
|
|
// Creating a Backbone.View creates its initial element outside of the DOM,
|
|
|
// if an existing element is not provided...
|
|
|
var View = Backbone.View = function(options) {
|
|
|
this.cid = _.uniqueId('view');
|
|
|
options || (options = {});
|
|
|
_.extend(this, _.pick(options, viewOptions));
|
|
|
this._ensureElement();
|
|
|
this.initialize.apply(this, arguments);
|
|
|
this.delegateEvents();
|
|
|
};
|
|
|
|
|
|
// Cached regex to split keys for `delegate`.
|
|
|
var delegateEventSplitter = /^(\S+)\s*(.*)$/;
|
|
|
|
|
|
// List of view options to be merged as properties.
|
|
|
var viewOptions = ['model', 'collection', 'el', 'id', 'attributes', 'className', 'tagName', 'events'];
|
|
|
|
|
|
// Set up all inheritable **Backbone.View** properties and methods.
|
|
|
_.extend(View.prototype, Events, {
|
|
|
|
|
|
// The default `tagName` of a View's element is `"div"`.
|
|
|
tagName: 'div',
|
|
|
|
|
|
// jQuery delegate for element lookup, scoped to DOM elements within the
|
|
|
// current view. This should be preferred to global lookups where possible.
|
|
|
$: function(selector) {
|
|
|
return this.$el.find(selector);
|
|
|
},
|
|
|
|
|
|
// Initialize is an empty function by default. Override it with your own
|
|
|
// initialization logic.
|
|
|
initialize: function(){},
|
|
|
|
|
|
// **render** is the core function that your view should override, in order
|
|
|
// to populate its element (`this.el`), with the appropriate HTML. The
|
|
|
// convention is for **render** to always return `this`.
|
|
|
render: function() {
|
|
|
return this;
|
|
|
},
|
|
|
|
|
|
// Remove this view by taking the element out of the DOM, and removing any
|
|
|
// applicable Backbone.Events listeners.
|
|
|
remove: function() {
|
|
|
this.$el.remove();
|
|
|
this.stopListening();
|
|
|
return this;
|
|
|
},
|
|
|
|
|
|
// Change the view's element (`this.el` property), including event
|
|
|
// re-delegation.
|
|
|
setElement: function(element, delegate) {
|
|
|
if (this.$el) this.undelegateEvents();
|
|
|
this.$el = element instanceof Backbone.$ ? element : Backbone.$(element);
|
|
|
this.el = this.$el[0];
|
|
|
if (delegate !== false) this.delegateEvents();
|
|
|
return this;
|
|
|
},
|
|
|
|
|
|
// Set callbacks, where `this.events` is a hash of
|
|
|
//
|
|
|
// *{"event selector": "callback"}*
|
|
|
//
|
|
|
// {
|
|
|
// 'mousedown .title': 'edit',
|
|
|
// 'click .button': 'save',
|
|
|
// 'click .open': function(e) { ... }
|
|
|
// }
|
|
|
//
|
|
|
// pairs. Callbacks will be bound to the view, with `this` set properly.
|
|
|
// Uses event delegation for efficiency.
|
|
|
// Omitting the selector binds the event to `this.el`.
|
|
|
// This only works for delegate-able events: not `focus`, `blur`, and
|
|
|
// not `change`, `submit`, and `reset` in Internet Explorer.
|
|
|
delegateEvents: function(events) {
|
|
|
if (!(events || (events = _.result(this, 'events')))) return this;
|
|
|
this.undelegateEvents();
|
|
|
for (var key in events) {
|
|
|
var method = events[key];
|
|
|
if (!_.isFunction(method)) method = this[events[key]];
|
|
|
if (!method) continue;
|
|
|
|
|
|
var match = key.match(delegateEventSplitter);
|
|
|
var eventName = match[1], selector = match[2];
|
|
|
method = _.bind(method, this);
|
|
|
eventName += '.delegateEvents' + this.cid;
|
|
|
if (selector === '') {
|
|
|
this.$el.on(eventName, method);
|
|
|
} else {
|
|
|
this.$el.on(eventName, selector, method);
|
|
|
}
|
|
|
}
|
|
|
return this;
|
|
|
},
|
|
|
|
|
|
// Clears all callbacks previously bound to the view with `delegateEvents`.
|
|
|
// You usually don't need to use this, but may wish to if you have multiple
|
|
|
// Backbone views attached to the same DOM element.
|
|
|
undelegateEvents: function() {
|
|
|
this.$el.off('.delegateEvents' + this.cid);
|
|
|
return this;
|
|
|
},
|
|
|
|
|
|
// Ensure that the View has a DOM element to render into.
|
|
|
// If `this.el` is a string, pass it through `$()`, take the first
|
|
|
// matching element, and re-assign it to `el`. Otherwise, create
|
|
|
// an element from the `id`, `className` and `tagName` properties.
|
|
|
_ensureElement: function() {
|
|
|
if (!this.el) {
|
|
|
var attrs = _.extend({}, _.result(this, 'attributes'));
|
|
|
if (this.id) attrs.id = _.result(this, 'id');
|
|
|
if (this.className) attrs['class'] = _.result(this, 'className');
|
|
|
var $el = Backbone.$('<' + _.result(this, 'tagName') + '>').attr(attrs);
|
|
|
this.setElement($el, false);
|
|
|
} else {
|
|
|
this.setElement(_.result(this, 'el'), false);
|
|
|
}
|
|
|
}
|
|
|
|
|
|
});
|
|
|
|
|
|
// Backbone.sync
|
|
|
// -------------
|
|
|
|
|
|
// Override this function to change the manner in which Backbone persists
|
|
|
// models to the server. You will be passed the type of request, and the
|
|
|
// model in question. By default, makes a RESTful Ajax request
|
|
|
// to the model's `url()`. Some possible customizations could be:
|
|
|
//
|
|
|
// * Use `setTimeout` to batch rapid-fire updates into a single request.
|
|
|
// * Send up the models as XML instead of JSON.
|
|
|
// * Persist models via WebSockets instead of Ajax.
|
|
|
//
|
|
|
// Turn on `Backbone.emulateHTTP` in order to send `PUT` and `DELETE` requests
|
|
|
// as `POST`, with a `_method` parameter containing the true HTTP method,
|
|
|
// as well as all requests with the body as `application/x-www-form-urlencoded`
|
|
|
// instead of `application/json` with the model in a param named `model`.
|
|
|
// Useful when interfacing with server-side languages like **PHP** that make
|
|
|
// it difficult to read the body of `PUT` requests.
|
|
|
Backbone.sync = function(method, model, options) {
|
|
|
var type = methodMap[method];
|
|
|
|
|
|
// Default options, unless specified.
|
|
|
_.defaults(options || (options = {}), {
|
|
|
emulateHTTP: Backbone.emulateHTTP,
|
|
|
emulateJSON: Backbone.emulateJSON
|
|
|
});
|
|
|
|
|
|
// Default JSON-request options.
|
|
|
var params = {type: type, dataType: 'json'};
|
|
|
|
|
|
// Ensure that we have a URL.
|
|
|
if (!options.url) {
|
|
|
params.url = _.result(model, 'url') || urlError();
|
|
|
}
|
|
|
|
|
|
// Ensure that we have the appropriate request data.
|
|
|
if (options.data == null && model && (method === 'create' || method === 'update' || method === 'patch')) {
|
|
|
params.contentType = 'application/json';
|
|
|
params.data = JSON.stringify(options.attrs || model.toJSON(options));
|
|
|
}
|
|
|
|
|
|
// For older servers, emulate JSON by encoding the request into an HTML-form.
|
|
|
if (options.emulateJSON) {
|
|
|
params.contentType = 'application/x-www-form-urlencoded';
|
|
|
params.data = params.data ? {model: params.data} : {};
|
|
|
}
|
|
|
|
|
|
// For older servers, emulate HTTP by mimicking the HTTP method with `_method`
|
|
|
// And an `X-HTTP-Method-Override` header.
|
|
|
if (options.emulateHTTP && (type === 'PUT' || type === 'DELETE' || type === 'PATCH')) {
|
|
|
params.type = 'POST';
|
|
|
if (options.emulateJSON) params.data._method = type;
|
|
|
var beforeSend = options.beforeSend;
|
|
|
options.beforeSend = function(xhr) {
|
|
|
xhr.setRequestHeader('X-HTTP-Method-Override', type);
|
|
|
if (beforeSend) return beforeSend.apply(this, arguments);
|
|
|
};
|
|
|
}
|
|
|
|
|
|
// Don't process data on a non-GET request.
|
|
|
if (params.type !== 'GET' && !options.emulateJSON) {
|
|
|
params.processData = false;
|
|
|
}
|
|
|
|
|
|
// If we're sending a `PATCH` request, and we're in an old Internet Explorer
|
|
|
// that still has ActiveX enabled by default, override jQuery to use that
|
|
|
// for XHR instead. Remove this line when jQuery supports `PATCH` on IE8.
|
|
|
if (params.type === 'PATCH' && noXhrPatch) {
|
|
|
params.xhr = function() {
|
|
|
return new ActiveXObject("Microsoft.XMLHTTP");
|
|
|
};
|
|
|
}
|
|
|
|
|
|
// Make the request, allowing the user to override any Ajax options.
|
|
|
var xhr = options.xhr = Backbone.ajax(_.extend(params, options));
|
|
|
model.trigger('request', model, xhr, options);
|
|
|
return xhr;
|
|
|
};
|
|
|
|
|
|
var noXhrPatch =
|
|
|
typeof window !== 'undefined' && !!window.ActiveXObject &&
|
|
|
!(window.XMLHttpRequest && (new XMLHttpRequest).dispatchEvent);
|
|
|
|
|
|
// Map from CRUD to HTTP for our default `Backbone.sync` implementation.
|
|
|
var methodMap = {
|
|
|
'create': 'POST',
|
|
|
'update': 'PUT',
|
|
|
'patch': 'PATCH',
|
|
|
'delete': 'DELETE',
|
|
|
'read': 'GET'
|
|
|
};
|
|
|
|
|
|
// Set the default implementation of `Backbone.ajax` to proxy through to `$`.
|
|
|
// Override this if you'd like to use a different library.
|
|
|
Backbone.ajax = function() {
|
|
|
return Backbone.$.ajax.apply(Backbone.$, arguments);
|
|
|
};
|
|
|
|
|
|
// Backbone.Router
|
|
|
// ---------------
|
|
|
|
|
|
// Routers map faux-URLs to actions, and fire events when routes are
|
|
|
// matched. Creating a new one sets its `routes` hash, if not set statically.
|
|
|
var Router = Backbone.Router = function(options) {
|
|
|
options || (options = {});
|
|
|
if (options.routes) this.routes = options.routes;
|
|
|
this._bindRoutes();
|
|
|
this.initialize.apply(this, arguments);
|
|
|
};
|
|
|
|
|
|
// Cached regular expressions for matching named param parts and splatted
|
|
|
// parts of route strings.
|
|
|
var optionalParam = /\((.*?)\)/g;
|
|
|
var namedParam = /(\(\?)?:\w+/g;
|
|
|
var splatParam = /\*\w+/g;
|
|
|
var escapeRegExp = /[\-{}\[\]+?.,\\\^$|#\s]/g;
|
|
|
|
|
|
// Set up all inheritable **Backbone.Router** properties and methods.
|
|
|
_.extend(Router.prototype, Events, {
|
|
|
|
|
|
// Initialize is an empty function by default. Override it with your own
|
|
|
// initialization logic.
|
|
|
initialize: function(){},
|
|
|
|
|
|
// Manually bind a single named route to a callback. For example:
|
|
|
//
|
|
|
// this.route('search/:query/p:num', 'search', function(query, num) {
|
|
|
// ...
|
|
|
// });
|
|
|
//
|
|
|
route: function(route, name, callback) {
|
|
|
if (!_.isRegExp(route)) route = this._routeToRegExp(route);
|
|
|
if (_.isFunction(name)) {
|
|
|
callback = name;
|
|
|
name = '';
|
|
|
}
|
|
|
if (!callback) callback = this[name];
|
|
|
var router = this;
|
|
|
Backbone.history.route(route, function(fragment) {
|
|
|
var args = router._extractParameters(route, fragment);
|
|
|
router.execute(callback, args);
|
|
|
router.trigger.apply(router, ['route:' + name].concat(args));
|
|
|
router.trigger('route', name, args);
|
|
|
Backbone.history.trigger('route', router, name, args);
|
|
|
});
|
|
|
return this;
|
|
|
},
|
|
|
|
|
|
// Execute a route handler with the provided parameters. This is an
|
|
|
// excellent place to do pre-route setup or post-route cleanup.
|
|
|
execute: function(callback, args) {
|
|
|
if (callback) callback.apply(this, args);
|
|
|
},
|
|
|
|
|
|
// Simple proxy to `Backbone.history` to save a fragment into the history.
|
|
|
navigate: function(fragment, options) {
|
|
|
Backbone.history.navigate(fragment, options);
|
|
|
return this;
|
|
|
},
|
|
|
|
|
|
// Bind all defined routes to `Backbone.history`. We have to reverse the
|
|
|
// order of the routes here to support behavior where the most general
|
|
|
// routes can be defined at the bottom of the route map.
|
|
|
_bindRoutes: function() {
|
|
|
if (!this.routes) return;
|
|
|
this.routes = _.result(this, 'routes');
|
|
|
var route, routes = _.keys(this.routes);
|
|
|
while ((route = routes.pop()) != null) {
|
|
|
this.route(route, this.routes[route]);
|
|
|
}
|
|
|
},
|
|
|
|
|
|
// Convert a route string into a regular expression, suitable for matching
|
|
|
// against the current location hash.
|
|
|
_routeToRegExp: function(route) {
|
|
|
route = route.replace(escapeRegExp, '\\$&')
|
|
|
.replace(optionalParam, '(?:$1)?')
|
|
|
.replace(namedParam, function(match, optional) {
|
|
|
return optional ? match : '([^/?]+)';
|
|
|
})
|
|
|
.replace(splatParam, '([^?]*?)');
|
|
|
return new RegExp('^' + route + '(?:\\?([\\s\\S]*))?$');
|
|
|
},
|
|
|
|
|
|
// Given a route, and a URL fragment that it matches, return the array of
|
|
|
// extracted decoded parameters. Empty or unmatched parameters will be
|
|
|
// treated as `null` to normalize cross-browser behavior.
|
|
|
_extractParameters: function(route, fragment) {
|
|
|
var params = route.exec(fragment).slice(1);
|
|
|
return _.map(params, function(param, i) {
|
|
|
// Don't decode the search params.
|
|
|
if (i === params.length - 1) return param || null;
|
|
|
return param ? decodeURIComponent(param) : null;
|
|
|
});
|
|
|
}
|
|
|
|
|
|
});
|
|
|
|
|
|
// Backbone.History
|
|
|
// ----------------
|
|
|
|
|
|
// Handles cross-browser history management, based on either
|
|
|
// [pushState](http://diveintohtml5.info/history.html) and real URLs, or
|
|
|
// [onhashchange](https://developer.mozilla.org/en-US/docs/DOM/window.onhashchange)
|
|
|
// and URL fragments. If the browser supports neither (old IE, natch),
|
|
|
// falls back to polling.
|
|
|
var History = Backbone.History = function() {
|
|
|
this.handlers = [];
|
|
|
_.bindAll(this, 'checkUrl');
|
|
|
|
|
|
// Ensure that `History` can be used outside of the browser.
|
|
|
if (typeof window !== 'undefined') {
|
|
|
this.location = window.location;
|
|
|
this.history = window.history;
|
|
|
}
|
|
|
};
|
|
|
|
|
|
// Cached regex for stripping a leading hash/slash and trailing space.
|
|
|
var routeStripper = /^[#\/]|\s+$/g;
|
|
|
|
|
|
// Cached regex for stripping leading and trailing slashes.
|
|
|
var rootStripper = /^\/+|\/+$/g;
|
|
|
|
|
|
// Cached regex for detecting MSIE.
|
|
|
var isExplorer = /msie [\w.]+/;
|
|
|
|
|
|
// Cached regex for removing a trailing slash.
|
|
|
var trailingSlash = /\/$/;
|
|
|
|
|
|
// Cached regex for stripping urls of hash.
|
|
|
var pathStripper = /#.*$/;
|
|
|
|
|
|
// Has the history handling already been started?
|
|
|
History.started = false;
|
|
|
|
|
|
// Set up all inheritable **Backbone.History** properties and methods.
|
|
|
_.extend(History.prototype, Events, {
|
|
|
|
|
|
// The default interval to poll for hash changes, if necessary, is
|
|
|
// twenty times a second.
|
|
|
interval: 50,
|
|
|
|
|
|
// Are we at the app root?
|
|
|
atRoot: function() {
|
|
|
return this.location.pathname.replace(/[^\/]$/, '$&/') === this.root;
|
|
|
},
|
|
|
|
|
|
// Gets the true hash value. Cannot use location.hash directly due to bug
|
|
|
// in Firefox where location.hash will always be decoded.
|
|
|
getHash: function(window) {
|
|
|
var match = (window || this).location.href.match(/#(.*)$/);
|
|
|
return match ? match[1] : '';
|
|
|
},
|
|
|
|
|
|
// Get the cross-browser normalized URL fragment, either from the URL,
|
|
|
// the hash, or the override.
|
|
|
getFragment: function(fragment, forcePushState) {
|
|
|
if (fragment == null) {
|
|
|
if (this._hasPushState || !this._wantsHashChange || forcePushState) {
|
|
|
fragment = decodeURI(this.location.pathname + this.location.search);
|
|
|
var root = this.root.replace(trailingSlash, '');
|
|
|
if (!fragment.indexOf(root)) fragment = fragment.slice(root.length);
|
|
|
} else {
|
|
|
fragment = this.getHash();
|
|
|
}
|
|
|
}
|
|
|
return fragment.replace(routeStripper, '');
|
|
|
},
|
|
|
|
|
|
// Start the hash change handling, returning `true` if the current URL matches
|
|
|
// an existing route, and `false` otherwise.
|
|
|
start: function(options) {
|
|
|
if (History.started) throw new Error("Backbone.history has already been started");
|
|
|
History.started = true;
|
|
|
|
|
|
// Figure out the initial configuration. Do we need an iframe?
|
|
|
// Is pushState desired ... is it available?
|
|
|
this.options = _.extend({root: '/'}, this.options, options);
|
|
|
this.root = this.options.root;
|
|
|
this._wantsHashChange = this.options.hashChange !== false;
|
|
|
this._wantsPushState = !!this.options.pushState;
|
|
|
this._hasPushState = !!(this.options.pushState && this.history && this.history.pushState);
|
|
|
var fragment = this.getFragment();
|
|
|
var docMode = document.documentMode;
|
|
|
var oldIE = (isExplorer.exec(navigator.userAgent.toLowerCase()) && (!docMode || docMode <= 7));
|
|
|
|
|
|
// Normalize root to always include a leading and trailing slash.
|
|
|
this.root = ('/' + this.root + '/').replace(rootStripper, '/');
|
|
|
|
|
|
if (oldIE && this._wantsHashChange) {
|
|
|
var frame = Backbone.$('<iframe src="javascript:0" tabindex="-1">');
|
|
|
this.iframe = frame.hide().appendTo('body')[0].contentWindow;
|
|
|
this.navigate(fragment);
|
|
|
}
|
|
|
|
|
|
// Depending on whether we're using pushState or hashes, and whether
|
|
|
// 'onhashchange' is supported, determine how we check the URL state.
|
|
|
if (this._hasPushState) {
|
|
|
Backbone.$(window).on('popstate', this.checkUrl);
|
|
|
} else if (this._wantsHashChange && ('onhashchange' in window) && !oldIE) {
|
|
|
Backbone.$(window).on('hashchange', this.checkUrl);
|
|
|
} else if (this._wantsHashChange) {
|
|
|
this._checkUrlInterval = setInterval(this.checkUrl, this.interval);
|
|
|
}
|
|
|
|
|
|
// Determine if we need to change the base url, for a pushState link
|
|
|
// opened by a non-pushState browser.
|
|
|
this.fragment = fragment;
|
|
|
var loc = this.location;
|
|
|
|
|
|
// Transition from hashChange to pushState or vice versa if both are
|
|
|
// requested.
|
|
|
if (this._wantsHashChange && this._wantsPushState) {
|
|
|
|
|
|
// If we've started off with a route from a `pushState`-enabled
|
|
|
// browser, but we're currently in a browser that doesn't support it...
|
|
|
if (!this._hasPushState && !this.atRoot()) {
|
|
|
this.fragment = this.getFragment(null, true);
|
|
|
this.location.replace(this.root + '#' + this.fragment);
|
|
|
// Return immediately as browser will do redirect to new url
|
|
|
return true;
|
|
|
|
|
|
// Or if we've started out with a hash-based route, but we're currently
|
|
|
// in a browser where it could be `pushState`-based instead...
|
|
|
} else if (this._hasPushState && this.atRoot() && loc.hash) {
|
|
|
this.fragment = this.getHash().replace(routeStripper, '');
|
|
|
this.history.replaceState({}, document.title, this.root + this.fragment);
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
if (!this.options.silent) return this.loadUrl();
|
|
|
},
|
|
|
|
|
|
// Disable Backbone.history, perhaps temporarily. Not useful in a real app,
|
|
|
// but possibly useful for unit testing Routers.
|
|
|
stop: function() {
|
|
|
Backbone.$(window).off('popstate', this.checkUrl).off('hashchange', this.checkUrl);
|
|
|
if (this._checkUrlInterval) clearInterval(this._checkUrlInterval);
|
|
|
History.started = false;
|
|
|
},
|
|
|
|
|
|
// Add a route to be tested when the fragment changes. Routes added later
|
|
|
// may override previous routes.
|
|
|
route: function(route, callback) {
|
|
|
this.handlers.unshift({route: route, callback: callback});
|
|
|
},
|
|
|
|
|
|
// Checks the current URL to see if it has changed, and if it has,
|
|
|
// calls `loadUrl`, normalizing across the hidden iframe.
|
|
|
checkUrl: function(e) {
|
|
|
var current = this.getFragment();
|
|
|
if (current === this.fragment && this.iframe) {
|
|
|
current = this.getFragment(this.getHash(this.iframe));
|
|
|
}
|
|
|
if (current === this.fragment) return false;
|
|
|
if (this.iframe) this.navigate(current);
|
|
|
this.loadUrl();
|
|
|
},
|
|
|
|
|
|
// Attempt to load the current URL fragment. If a route succeeds with a
|
|
|
// match, returns `true`. If no defined routes matches the fragment,
|
|
|
// returns `false`.
|
|
|
loadUrl: function(fragment) {
|
|
|
fragment = this.fragment = this.getFragment(fragment);
|
|
|
return _.any(this.handlers, function(handler) {
|
|
|
if (handler.route.test(fragment)) {
|
|
|
handler.callback(fragment);
|
|
|
return true;
|
|
|
}
|
|
|
});
|
|
|
},
|
|
|
|
|
|
// Save a fragment into the hash history, or replace the URL state if the
|
|
|
// 'replace' option is passed. You are responsible for properly URL-encoding
|
|
|
// the fragment in advance.
|
|
|
//
|
|
|
// The options object can contain `trigger: true` if you wish to have the
|
|
|
// route callback be fired (not usually desirable), or `replace: true`, if
|
|
|
// you wish to modify the current URL without adding an entry to the history.
|
|
|
navigate: function(fragment, options) {
|
|
|
if (!History.started) return false;
|
|
|
if (!options || options === true) options = {trigger: !!options};
|
|
|
|
|
|
var url = this.root + (fragment = this.getFragment(fragment || ''));
|
|
|
|
|
|
// Strip the hash for matching.
|
|
|
fragment = fragment.replace(pathStripper, '');
|
|
|
|
|
|
if (this.fragment === fragment) return;
|
|
|
this.fragment = fragment;
|
|
|
|
|
|
// Don't include a trailing slash on the root.
|
|
|
if (fragment === '' && url !== '/') url = url.slice(0, -1);
|
|
|
|
|
|
// If pushState is available, we use it to set the fragment as a real URL.
|
|
|
if (this._hasPushState) {
|
|
|
this.history[options.replace ? 'replaceState' : 'pushState']({}, document.title, url);
|
|
|
|
|
|
// If hash changes haven't been explicitly disabled, update the hash
|
|
|
// fragment to store history.
|
|
|
} else if (this._wantsHashChange) {
|
|
|
this._updateHash(this.location, fragment, options.replace);
|
|
|
if (this.iframe && (fragment !== this.getFragment(this.getHash(this.iframe)))) {
|
|
|
// Opening and closing the iframe tricks IE7 and earlier to push a
|
|
|
// history entry on hash-tag change. When replace is true, we don't
|
|
|
// want this.
|
|
|
if(!options.replace) this.iframe.document.open().close();
|
|
|
this._updateHash(this.iframe.location, fragment, options.replace);
|
|
|
}
|
|
|
|
|
|
// If you've told us that you explicitly don't want fallback hashchange-
|
|
|
// based history, then `navigate` becomes a page refresh.
|
|
|
} else {
|
|
|
return this.location.assign(url);
|
|
|
}
|
|
|
if (options.trigger) return this.loadUrl(fragment);
|
|
|
},
|
|
|
|
|
|
// Update the hash location, either replacing the current entry, or adding
|
|
|
// a new one to the browser history.
|
|
|
_updateHash: function(location, fragment, replace) {
|
|
|
if (replace) {
|
|
|
var href = location.href.replace(/(javascript:|#).*$/, '');
|
|
|
location.replace(href + '#' + fragment);
|
|
|
} else {
|
|
|
// Some browsers require that `hash` contains a leading #.
|
|
|
location.hash = '#' + fragment;
|
|
|
}
|
|
|
}
|
|
|
|
|
|
});
|
|
|
|
|
|
// Create the default Backbone.history.
|
|
|
Backbone.history = new History;
|
|
|
|
|
|
// Helpers
|
|
|
// -------
|
|
|
|
|
|
// Helper function to correctly set up the prototype chain, for subclasses.
|
|
|
// Similar to `goog.inherits`, but uses a hash of prototype properties and
|
|
|
// class properties to be extended.
|
|
|
var extend = function(protoProps, staticProps) {
|
|
|
var parent = this;
|
|
|
var child;
|
|
|
|
|
|
// The constructor function for the new subclass is either defined by you
|
|
|
// (the "constructor" property in your `extend` definition), or defaulted
|
|
|
// by us to simply call the parent's constructor.
|
|
|
if (protoProps && _.has(protoProps, 'constructor')) {
|
|
|
child = protoProps.constructor;
|
|
|
} else {
|
|
|
child = function(){ return parent.apply(this, arguments); };
|
|
|
}
|
|
|
|
|
|
// Add static properties to the constructor function, if supplied.
|
|
|
_.extend(child, parent, staticProps);
|
|
|
|
|
|
// Set the prototype chain to inherit from `parent`, without calling
|
|
|
// `parent`'s constructor function.
|
|
|
var Surrogate = function(){ this.constructor = child; };
|
|
|
Surrogate.prototype = parent.prototype;
|
|
|
child.prototype = new Surrogate;
|
|
|
|
|
|
// Add prototype properties (instance properties) to the subclass,
|
|
|
// if supplied.
|
|
|
if (protoProps) _.extend(child.prototype, protoProps);
|
|
|
|
|
|
// Set a convenience property in case the parent's prototype is needed
|
|
|
// later.
|
|
|
child.__super__ = parent.prototype;
|
|
|
|
|
|
return child;
|
|
|
};
|
|
|
|
|
|
// Set up inheritance for the model, collection, router, view and history.
|
|
|
Model.extend = Collection.extend = Router.extend = View.extend = History.extend = extend;
|
|
|
|
|
|
// Throw an error when a URL is needed, and none is supplied.
|
|
|
var urlError = function() {
|
|
|
throw new Error('A "url" property or function must be specified');
|
|
|
};
|
|
|
|
|
|
// Wrap an optional error callback with a fallback error event.
|
|
|
var wrapError = function(model, options) {
|
|
|
var error = options.error;
|
|
|
options.error = function(resp) {
|
|
|
if (error) error(model, resp, options);
|
|
|
model.trigger('error', model, resp, options);
|
|
|
};
|
|
|
};
|
|
|
|
|
|
return Backbone;
|
|
|
|
|
|
}));
|
|
|
|
|
|
(function (root, factory) {
|
|
|
if (typeof define === 'function' && define.amd) {
|
|
|
// AMD. Register as an anonymous module.
|
|
|
define(['backbone', 'underscore'], factory);
|
|
|
} 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('backbone'), require('underscore'));
|
|
|
} else {
|
|
|
// Browser globals (root is window)
|
|
|
root.returnExports = factory(root.Backbone, root._);
|
|
|
}
|
|
|
}(this, function (Backbone, _) {
|
|
|
|
|
|
// Generate four random hex digits.
|
|
|
function S4() {
|
|
|
return (((1 + Math.random()) * 0x10000) | 0).toString(16).substring(1);
|
|
|
}
|
|
|
|
|
|
// Generate a pseudo-GUID by concatenating random hexadecimal.
|
|
|
function guid() {
|
|
|
return (S4() + S4() + "-" + S4() + "-" + S4() + "-" + S4() + "-" + S4() + S4() + S4());
|
|
|
}
|
|
|
|
|
|
if ( _(indexedDB).isUndefined() ) { return; }
|
|
|
|
|
|
// Driver object
|
|
|
// That's the interesting part.
|
|
|
// There is a driver for each schema provided. The schema is a te combination of name (for the database), a version as well as migrations to reach that
|
|
|
// version of the database.
|
|
|
function Driver(schema, ready, nolog, onerror) {
|
|
|
this.schema = schema;
|
|
|
this.ready = ready;
|
|
|
this.error = null;
|
|
|
this.transactions = []; // Used to list all transactions and keep track of active ones.
|
|
|
this.db = null;
|
|
|
this.nolog = nolog;
|
|
|
this.onerror = onerror;
|
|
|
var lastMigrationPathVersion = _.last(this.schema.migrations).version;
|
|
|
if (!this.nolog) debugLog("opening database " + this.schema.id + " in version #" + lastMigrationPathVersion);
|
|
|
this.dbRequest = indexedDB.open(this.schema.id,lastMigrationPathVersion); //schema version need to be an unsigned long
|
|
|
|
|
|
this.launchMigrationPath = function(dbVersion) {
|
|
|
var transaction = this.dbRequest.transaction;
|
|
|
var clonedMigrations = _.clone(schema.migrations);
|
|
|
this.migrate(transaction, clonedMigrations, dbVersion, {
|
|
|
error: function (event) {
|
|
|
this.error = "Database not up to date. " + dbVersion + " expected was " + lastMigrationPathVersion;
|
|
|
}.bind(this)
|
|
|
});
|
|
|
};
|
|
|
|
|
|
this.dbRequest.onblocked = function(event){
|
|
|
if (!this.nolog) debugLog("connection to database blocked");
|
|
|
}
|
|
|
|
|
|
this.dbRequest.onsuccess = function (e) {
|
|
|
this.db = e.target.result; // Attach the connection ot the queue.
|
|
|
var currentIntDBVersion = (parseInt(this.db.version) || 0); // we need convert beacuse chrome store in integer and ie10 DP4+ in int;
|
|
|
var lastMigrationInt = (parseInt(lastMigrationPathVersion) || 0); // And make sure we compare numbers with numbers.
|
|
|
|
|
|
if (currentIntDBVersion === lastMigrationInt) { //if support new event onupgradeneeded will trigger the ready function
|
|
|
// No migration to perform!
|
|
|
this.ready();
|
|
|
} else if (currentIntDBVersion < lastMigrationInt ) {
|
|
|
// We need to migrate up to the current migration defined in the database
|
|
|
this.launchMigrationPath(currentIntDBVersion);
|
|
|
} else {
|
|
|
// Looks like the IndexedDB is at a higher version than the current driver schema.
|
|
|
this.error = "Database version is greater than current code " + currentIntDBVersion + " expected was " + lastMigrationInt;
|
|
|
}
|
|
|
}.bind(this);
|
|
|
|
|
|
|
|
|
|
|
|
this.dbRequest.onerror = function (e) {
|
|
|
// Failed to open the database
|
|
|
this.error = "Couldn't not connect to the database"
|
|
|
if (!this.nolog) debugLog("Couldn't not connect to the database");
|
|
|
this.onerror();
|
|
|
}.bind(this);
|
|
|
|
|
|
this.dbRequest.onabort = function (e) {
|
|
|
// Failed to open the database
|
|
|
this.error = "Connection to the database aborted"
|
|
|
if (!this.nolog) debugLog("Connection to the database aborted");
|
|
|
this.onerror();
|
|
|
}.bind(this);
|
|
|
|
|
|
|
|
|
|
|
|
this.dbRequest.onupgradeneeded = function(iDBVersionChangeEvent){
|
|
|
this.db =iDBVersionChangeEvent.target.result;
|
|
|
|
|
|
var newVersion = iDBVersionChangeEvent.newVersion;
|
|
|
var oldVersion = iDBVersionChangeEvent.oldVersion;
|
|
|
|
|
|
// Fix Safari 8 and iOS 8 bug
|
|
|
// at the first connection oldVersion is equal to 9223372036854776000
|
|
|
// but the real value is 0
|
|
|
if (oldVersion > 99999999999)
|
|
|
oldVersion = 0;
|
|
|
|
|
|
if (!this.nolog) debugLog("onupgradeneeded = " + oldVersion + " => " + newVersion);
|
|
|
this.launchMigrationPath(oldVersion);
|
|
|
}.bind(this);
|
|
|
}
|
|
|
|
|
|
function debugLog(str) {
|
|
|
if (typeof window !== "undefined" && typeof window.console !== "undefined" && typeof window.console.log !== "undefined") {
|
|
|
window.console.log(str);
|
|
|
}
|
|
|
else if(console.log !== "undefined") {
|
|
|
console.log(str)
|
|
|
}
|
|
|
}
|
|
|
|
|
|
// Driver Prototype
|
|
|
Driver.prototype = {
|
|
|
|
|
|
// Tracks transactions. Mostly for debugging purposes. TO-IMPROVE
|
|
|
_track_transaction: function(transaction) {
|
|
|
this.transactions.push(transaction);
|
|
|
function removeIt() {
|
|
|
var idx = this.transactions.indexOf(transaction);
|
|
|
if (idx !== -1) {this.transactions.splice(idx); }
|
|
|
};
|
|
|
transaction.oncomplete = removeIt.bind(this);
|
|
|
transaction.onabort = removeIt.bind(this);
|
|
|
transaction.onerror = removeIt.bind(this);
|
|
|
},
|
|
|
|
|
|
// Performs all the migrations to reach the right version of the database.
|
|
|
migrate: function (transaction, migrations, version, options) {
|
|
|
transaction.onerror = options.error;
|
|
|
transaction.onabort = options.error;
|
|
|
|
|
|
if (!this.nolog) debugLog("migrate begin version from #" + version);
|
|
|
var that = this;
|
|
|
var migration = migrations.shift();
|
|
|
if (migration) {
|
|
|
if (!version || version < migration.version) {
|
|
|
// We need to apply this migration-
|
|
|
if (typeof migration.before == "undefined") {
|
|
|
migration.before = function (next) {
|
|
|
next();
|
|
|
};
|
|
|
}
|
|
|
if (typeof migration.after == "undefined") {
|
|
|
migration.after = function (next) {
|
|
|
next();
|
|
|
};
|
|
|
}
|
|
|
// First, let's run the before script
|
|
|
if (!this.nolog) debugLog("migrate begin before version #" + migration.version);
|
|
|
migration.before(function () {
|
|
|
if (!this.nolog) debugLog("migrate done before version #" + migration.version);
|
|
|
|
|
|
if (!this.nolog) debugLog("migrate begin migrate version #" + migration.version);
|
|
|
|
|
|
migration.migrate(transaction, function () {
|
|
|
if (!this.nolog) debugLog("migrate done migrate version #" + migration.version);
|
|
|
// Migration successfully appliedn let's go to the next one!
|
|
|
if (!this.nolog) debugLog("migrate begin after version #" + migration.version);
|
|
|
migration.after(function () {
|
|
|
if (!this.nolog) debugLog("migrate done after version #" + migration.version);
|
|
|
if (!this.nolog) debugLog("Migrated to " + migration.version);
|
|
|
|
|
|
//last modification occurred, need finish
|
|
|
if(migrations.length ==0) {
|
|
|
if (!this.nolog) {
|
|
|
debugLog("migrate setting transaction.oncomplete to finish version #" + migration.version);
|
|
|
transaction.oncomplete = function() {
|
|
|
debugLog("migrate done transaction.oncomplete version #" + migration.version);
|
|
|
debugLog("Done migrating");
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
else
|
|
|
{
|
|
|
if (!this.nolog) debugLog("migrate end from version #" + version + " to " + migration.version);
|
|
|
that.migrate(transaction, migrations, version, options);
|
|
|
}
|
|
|
|
|
|
}.bind(this));
|
|
|
}.bind(this));
|
|
|
}.bind(this));
|
|
|
} else {
|
|
|
// No need to apply this migration
|
|
|
if (!this.nolog) debugLog("Skipping migration " + migration.version);
|
|
|
this.migrate(transaction, migrations, version, options);
|
|
|
}
|
|
|
}
|
|
|
},
|
|
|
|
|
|
// This is the main method, called by the ExecutionQueue when the driver is ready (database open and migration performed)
|
|
|
execute: function (storeName, method, object, options) {
|
|
|
if (!this.nolog) debugLog("execute : " + method + " on " + storeName + " for " + object.id);
|
|
|
switch (method) {
|
|
|
case "create":
|
|
|
this.create(storeName, object, options);
|
|
|
break;
|
|
|
case "read":
|
|
|
if (object.id || object.cid) {
|
|
|
this.read(storeName, object, options); // It's a model
|
|
|
} else {
|
|
|
this.query(storeName, object, options); // It's a collection
|
|
|
}
|
|
|
break;
|
|
|
case "update":
|
|
|
this.update(storeName, object, options); // We may want to check that this is not a collection. TOFIX
|
|
|
break;
|
|
|
case "delete":
|
|
|
if (object.id || object.cid) {
|
|
|
this.delete(storeName, object, options);
|
|
|
} else {
|
|
|
this.clear(storeName, object, options);
|
|
|
}
|
|
|
break;
|
|
|
default:
|
|
|
// Hum what?
|
|
|
}
|
|
|
},
|
|
|
|
|
|
// Writes the json to the storeName in db. It is a create operations, which means it will fail if the key already exists
|
|
|
// options are just success and error callbacks.
|
|
|
create: function (storeName, object, options) {
|
|
|
var writeTransaction = this.db.transaction([storeName], 'readwrite');
|
|
|
//this._track_transaction(writeTransaction);
|
|
|
var store = writeTransaction.objectStore(storeName);
|
|
|
var json = object.toJSON();
|
|
|
var idAttribute = _.result(object, 'idAttribute');
|
|
|
var writeRequest;
|
|
|
|
|
|
if (json[idAttribute] === undefined && !store.autoIncrement) json[idAttribute] = guid();
|
|
|
|
|
|
writeTransaction.onerror = function (e) {
|
|
|
options.error(e);
|
|
|
};
|
|
|
writeTransaction.oncomplete = function (e) {
|
|
|
options.success(json);
|
|
|
};
|
|
|
|
|
|
if (!store.keyPath)
|
|
|
writeRequest = store.add(json, json[idAttribute]);
|
|
|
else
|
|
|
writeRequest = store.add(json);
|
|
|
},
|
|
|
|
|
|
// Writes the json to the storeName in db. It is an update operation, which means it will overwrite the value if the key already exist
|
|
|
// options are just success and error callbacks.
|
|
|
update: function (storeName, object, options) {
|
|
|
var writeTransaction = this.db.transaction([storeName], 'readwrite');
|
|
|
//this._track_transaction(writeTransaction);
|
|
|
var store = writeTransaction.objectStore(storeName);
|
|
|
var json = object.toJSON();
|
|
|
var idAttribute = _.result(object, 'idAttribute');
|
|
|
var writeRequest;
|
|
|
|
|
|
if (!json[idAttribute]) json[idAttribute] = guid();
|
|
|
|
|
|
if (!store.keyPath)
|
|
|
writeRequest = store.put(json, json[idAttribute]);
|
|
|
else
|
|
|
writeRequest = store.put(json);
|
|
|
|
|
|
writeRequest.onerror = function (e) {
|
|
|
options.error(e);
|
|
|
};
|
|
|
writeTransaction.oncomplete = function (e) {
|
|
|
options.success(json);
|
|
|
};
|
|
|
},
|
|
|
|
|
|
// Reads from storeName in db with json.id if it's there of with any json.xxxx as long as xxx is an index in storeName
|
|
|
read: function (storeName, object, options) {
|
|
|
var readTransaction = this.db.transaction([storeName], "readonly");
|
|
|
this._track_transaction(readTransaction);
|
|
|
|
|
|
var store = readTransaction.objectStore(storeName);
|
|
|
var json = object.toJSON();
|
|
|
var idAttribute = _.result(object, 'idAttribute');
|
|
|
|
|
|
var getRequest = null;
|
|
|
if (json[idAttribute]) {
|
|
|
getRequest = store.get(json[idAttribute]);
|
|
|
} else if(options.index) {
|
|
|
var index = store.index(options.index.name);
|
|
|
getRequest = index.get(options.index.value);
|
|
|
} else {
|
|
|
// We need to find which index we have
|
|
|
var cardinality = 0; // try to fit the index with most matches
|
|
|
_.each(store.indexNames, function (key, index) {
|
|
|
index = store.index(key);
|
|
|
if(typeof index.keyPath === 'string' && 1 > cardinality) {
|
|
|
// simple index
|
|
|
if (json[index.keyPath] !== undefined) {
|
|
|
getRequest = index.get(json[index.keyPath]);
|
|
|
cardinality = 1;
|
|
|
}
|
|
|
} else if(typeof index.keyPath === 'object' && index.keyPath.length > cardinality) {
|
|
|
// compound index
|
|
|
var valid = true;
|
|
|
var keyValue = _.map(index.keyPath, function(keyPart) {
|
|
|
valid = valid && json[keyPart] !== undefined;
|
|
|
return json[keyPart];
|
|
|
});
|
|
|
if(valid) {
|
|
|
getRequest = index.get(keyValue);
|
|
|
cardinality = index.keyPath.length;
|
|
|
}
|
|
|
}
|
|
|
});
|
|
|
}
|
|
|
if (getRequest) {
|
|
|
getRequest.onsuccess = function (event) {
|
|
|
if (event.target.result) {
|
|
|
options.success(event.target.result);
|
|
|
} else {
|
|
|
options.error("Not Found");
|
|
|
}
|
|
|
};
|
|
|
getRequest.onerror = function () {
|
|
|
options.error("Not Found"); // We couldn't find the record.
|
|
|
}
|
|
|
} else {
|
|
|
options.error("Not Found"); // We couldn't even look for it, as we don't have enough data.
|
|
|
}
|
|
|
},
|
|
|
|
|
|
// Deletes the json.id key and value in storeName from db.
|
|
|
delete: function (storeName, object, options) {
|
|
|
var deleteTransaction = this.db.transaction([storeName], 'readwrite');
|
|
|
//this._track_transaction(deleteTransaction);
|
|
|
|
|
|
var store = deleteTransaction.objectStore(storeName);
|
|
|
var json = object.toJSON();
|
|
|
var idAttribute = _.result(object, 'idAttribute');
|
|
|
|
|
|
var deleteRequest = store.delete(json[idAttribute]);
|
|
|
|
|
|
deleteTransaction.oncomplete = function (event) {
|
|
|
options.success(null);
|
|
|
};
|
|
|
deleteRequest.onerror = function (event) {
|
|
|
options.error("Not Deleted");
|
|
|
};
|
|
|
},
|
|
|
|
|
|
// Clears all records for storeName from db.
|
|
|
clear: function (storeName, object, options) {
|
|
|
var deleteTransaction = this.db.transaction([storeName], "readwrite");
|
|
|
//this._track_transaction(deleteTransaction);
|
|
|
|
|
|
var store = deleteTransaction.objectStore(storeName);
|
|
|
|
|
|
var deleteRequest = store.clear();
|
|
|
deleteRequest.onsuccess = function (event) {
|
|
|
options.success(null);
|
|
|
};
|
|
|
deleteRequest.onerror = function (event) {
|
|
|
options.error("Not Cleared");
|
|
|
};
|
|
|
},
|
|
|
|
|
|
// Performs a query on storeName in db.
|
|
|
// options may include :
|
|
|
// - conditions : value of an index, or range for an index
|
|
|
// - range : range for the primary key
|
|
|
// - limit : max number of elements to be yielded
|
|
|
// - offset : skipped items.
|
|
|
query: function (storeName, collection, options) {
|
|
|
var elements = [];
|
|
|
var skipped = 0, processed = 0;
|
|
|
var queryTransaction = this.db.transaction([storeName], "readonly");
|
|
|
//this._track_transaction(queryTransaction);
|
|
|
|
|
|
var idAttribute = _.result(collection.model.prototype, 'idAttribute');
|
|
|
var readCursor = null;
|
|
|
var store = queryTransaction.objectStore(storeName);
|
|
|
var index = null,
|
|
|
lower = null,
|
|
|
upper = null,
|
|
|
bounds = null;
|
|
|
|
|
|
if (options.conditions) {
|
|
|
// We have a condition, we need to use it for the cursor
|
|
|
_.each(store.indexNames, function (key) {
|
|
|
if (!readCursor) {
|
|
|
index = store.index(key);
|
|
|
if (options.conditions[index.keyPath] instanceof Array) {
|
|
|
lower = options.conditions[index.keyPath][0] > options.conditions[index.keyPath][1] ? options.conditions[index.keyPath][1] : options.conditions[index.keyPath][0];
|
|
|
upper = options.conditions[index.keyPath][0] > options.conditions[index.keyPath][1] ? options.conditions[index.keyPath][0] : options.conditions[index.keyPath][1];
|
|
|
bounds = IDBKeyRange.bound(lower, upper, true, true);
|
|
|
|
|
|
if (options.conditions[index.keyPath][0] > options.conditions[index.keyPath][1]) {
|
|
|
// Looks like we want the DESC order
|
|
|
readCursor = index.openCursor(bounds, window.IDBCursor.PREV || "prev");
|
|
|
} else {
|
|
|
// We want ASC order
|
|
|
readCursor = index.openCursor(bounds, window.IDBCursor.NEXT || "next");
|
|
|
}
|
|
|
} else if (typeof options.conditions[index.keyPath] === 'object' && ('$gt' in options.conditions[index.keyPath] || '$gte' in options.conditions[index.keyPath])) {
|
|
|
if('$gt' in options.conditions[index.keyPath])
|
|
|
bounds = IDBKeyRange.lowerBound(options.conditions[index.keyPath]['$gt'], true);
|
|
|
else
|
|
|
bounds = IDBKeyRange.lowerBound(options.conditions[index.keyPath]['$gte']);
|
|
|
readCursor = index.openCursor(bounds, window.IDBCursor.NEXT || "next");
|
|
|
} else if (typeof options.conditions[index.keyPath] === 'object' && ('$lt' in options.conditions[index.keyPath] || '$lte' in options.conditions[index.keyPath])) {
|
|
|
if('$lt' in options.conditions[index.keyPath])
|
|
|
bounds = IDBKeyRange.upperBound(options.conditions[index.keyPath]['$lt'], true);
|
|
|
else
|
|
|
bounds = IDBKeyRange.upperBound(options.conditions[index.keyPath]['$lte']);
|
|
|
readCursor = index.openCursor(bounds, window.IDBCursor.NEXT || "next");
|
|
|
} else if (options.conditions[index.keyPath] != undefined) {
|
|
|
bounds = IDBKeyRange.only(options.conditions[index.keyPath]);
|
|
|
readCursor = index.openCursor(bounds);
|
|
|
}
|
|
|
}
|
|
|
});
|
|
|
} else if (options.index) {
|
|
|
index = store.index(options.index.name);
|
|
|
if (index) {
|
|
|
if (options.index.lower && options.index.upper) {
|
|
|
bounds = IDBKeyRange.bound(options.index.lower, options.index.upper);
|
|
|
} else if (options.index.lower) {
|
|
|
bounds = IDBKeyRange.lowerBound(options.index.lower);
|
|
|
} else if (options.index.upper) {
|
|
|
bounds = IDBKeyRange.upperBound(options.index.upper);
|
|
|
} else if (options.index.only) {
|
|
|
bounds = IDBKeyRange.only(options.index.only);
|
|
|
}
|
|
|
|
|
|
if (typeof options.index.order === 'string' && options.index.order.toLowerCase() === 'desc') {
|
|
|
readCursor = index.openCursor(bounds, window.IDBCursor.PREV || "prev");
|
|
|
} else {
|
|
|
readCursor = index.openCursor(bounds, window.IDBCursor.NEXT || "next");
|
|
|
}
|
|
|
}
|
|
|
} else {
|
|
|
// No conditions, use the index
|
|
|
if (options.range) {
|
|
|
lower = options.range[0] > options.range[1] ? options.range[1] : options.range[0];
|
|
|
upper = options.range[0] > options.range[1] ? options.range[0] : options.range[1];
|
|
|
bounds = IDBKeyRange.bound(lower, upper);
|
|
|
if (options.range[0] > options.range[1]) {
|
|
|
readCursor = store.openCursor(bounds, window.IDBCursor.PREV || "prev");
|
|
|
} else {
|
|
|
readCursor = store.openCursor(bounds, window.IDBCursor.NEXT || "next");
|
|
|
}
|
|
|
} else {
|
|
|
readCursor = store.openCursor();
|
|
|
}
|
|
|
}
|
|
|
|
|
|
if (typeof (readCursor) == "undefined" || !readCursor) {
|
|
|
options.error("No Cursor");
|
|
|
} else {
|
|
|
readCursor.onerror = function(e){
|
|
|
options.error("readCursor error", e);
|
|
|
};
|
|
|
// Setup a handler for the cursor’s `success` event:
|
|
|
readCursor.onsuccess = function (e) {
|
|
|
var cursor = e.target.result;
|
|
|
if (!cursor) {
|
|
|
if (options.addIndividually || options.clear) {
|
|
|
// nothing!
|
|
|
// We need to indicate that we're done. But, how?
|
|
|
collection.trigger("reset");
|
|
|
} else {
|
|
|
options.success(elements); // We're done. No more elements.
|
|
|
}
|
|
|
}
|
|
|
else {
|
|
|
// Cursor is not over yet.
|
|
|
if (options.limit && processed >= options.limit) {
|
|
|
// Yet, we have processed enough elements. So, let's just skip.
|
|
|
if (bounds && options.conditions[index.keyPath]) {
|
|
|
cursor.continue(options.conditions[index.keyPath][1] + 1); /* We need to 'terminate' the cursor cleany, by moving to the end */
|
|
|
} else {
|
|
|
cursor.continue(); /* We need to 'terminate' the cursor cleany, by moving to the end */
|
|
|
}
|
|
|
}
|
|
|
else if (options.offset && options.offset > skipped) {
|
|
|
skipped++;
|
|
|
cursor.continue(); /* We need to Moving the cursor forward */
|
|
|
} else {
|
|
|
// This time, it looks like it's good!
|
|
|
if (options.addIndividually) {
|
|
|
collection.add(cursor.value);
|
|
|
} else if (options.clear) {
|
|
|
var deleteRequest = store.delete(cursor.value[idAttribute]);
|
|
|
deleteRequest.onsuccess = function (event) {
|
|
|
elements.push(cursor.value);
|
|
|
};
|
|
|
deleteRequest.onerror = function (event) {
|
|
|
elements.push(cursor.value);
|
|
|
};
|
|
|
|
|
|
} else {
|
|
|
elements.push(cursor.value);
|
|
|
}
|
|
|
processed++;
|
|
|
cursor.continue();
|
|
|
}
|
|
|
}
|
|
|
};
|
|
|
}
|
|
|
},
|
|
|
close :function(){
|
|
|
if(this.db){
|
|
|
this.db.close();
|
|
|
}
|
|
|
}
|
|
|
};
|
|
|
|
|
|
// ExecutionQueue object
|
|
|
// The execution queue is an abstraction to buffer up requests to the database.
|
|
|
// It holds a "driver". When the driver is ready, it just fires up the queue and executes in sync.
|
|
|
function ExecutionQueue(schema,next,nolog) {
|
|
|
this.driver = new Driver(schema, this.ready.bind(this), nolog, this.error.bind(this));
|
|
|
this.started = false;
|
|
|
this.failed = false;
|
|
|
this.stack = [];
|
|
|
this.version = _.last(schema.migrations).version;
|
|
|
this.next = next;
|
|
|
}
|
|
|
|
|
|
// ExecutionQueue Prototype
|
|
|
ExecutionQueue.prototype = {
|
|
|
// Called when the driver is ready
|
|
|
// It just loops over the elements in the queue and executes them.
|
|
|
ready: function () {
|
|
|
this.started = true;
|
|
|
_.each(this.stack, function (message) {
|
|
|
this.execute(message);
|
|
|
}.bind(this));
|
|
|
this.stack = []; // fix memory leak
|
|
|
this.next();
|
|
|
},
|
|
|
|
|
|
error: function() {
|
|
|
this.failed = true;
|
|
|
_.each(this.stack, function (message) {
|
|
|
this.execute(message);
|
|
|
}.bind(this));
|
|
|
this.stack = [];
|
|
|
this.next();
|
|
|
},
|
|
|
|
|
|
// Executes a given command on the driver. If not started, just stacks up one more element.
|
|
|
execute: function (message) {
|
|
|
if (this.started) {
|
|
|
this.driver.execute(message[2].storeName || message[1].storeName, message[0], message[1], message[2]); // Upon messages, we execute the query
|
|
|
} else if (this.failed) {
|
|
|
message[2].error();
|
|
|
} else {
|
|
|
this.stack.push(message);
|
|
|
}
|
|
|
},
|
|
|
|
|
|
close : function(){
|
|
|
this.driver.close();
|
|
|
}
|
|
|
};
|
|
|
|
|
|
// Method used by Backbone for sync of data with data store. It was initially designed to work with "server side" APIs, This wrapper makes
|
|
|
// it work with the local indexedDB stuff. It uses the schema attribute provided by the object.
|
|
|
// The wrapper keeps an active Executuon Queue for each "schema", and executes querues agains it, based on the object type (collection or
|
|
|
// single model), but also the method... etc.
|
|
|
// Keeps track of the connections
|
|
|
var Databases = {};
|
|
|
|
|
|
function sync(method, object, options) {
|
|
|
|
|
|
if(method == "closeall"){
|
|
|
_.each(Databases,function(database){
|
|
|
database.close();
|
|
|
});
|
|
|
// Clean up active databases object.
|
|
|
Databases = {};
|
|
|
return Backbone.$.Deferred().resolve();
|
|
|
}
|
|
|
|
|
|
// If a model or a collection does not define a database, fall back on ajaxSync
|
|
|
if (!object || !_.isObject(object.database)) {
|
|
|
return Backbone.ajaxSync(method, object, options);
|
|
|
}
|
|
|
|
|
|
var schema = object.database;
|
|
|
if (Databases[schema.id]) {
|
|
|
if(Databases[schema.id].version != _.last(schema.migrations).version){
|
|
|
Databases[schema.id].close();
|
|
|
delete Databases[schema.id];
|
|
|
}
|
|
|
}
|
|
|
|
|
|
var promise;
|
|
|
|
|
|
if (typeof Backbone.$ === 'undefined' || typeof Backbone.$.Deferred === 'undefined') {
|
|
|
var noop = function() {};
|
|
|
var resolve = noop;
|
|
|
var reject = noop;
|
|
|
} else {
|
|
|
var dfd = Backbone.$.Deferred();
|
|
|
var resolve = dfd.resolve;
|
|
|
var reject = dfd.reject;
|
|
|
|
|
|
promise = dfd.promise();
|
|
|
}
|
|
|
|
|
|
var success = options.success;
|
|
|
options.success = function(resp) {
|
|
|
if (success) success(resp);
|
|
|
resolve();
|
|
|
object.trigger('sync', object, resp, options);
|
|
|
};
|
|
|
|
|
|
var error = options.error;
|
|
|
options.error = function(resp) {
|
|
|
if (error) error(resp);
|
|
|
reject();
|
|
|
object.trigger('error', object, resp, options);
|
|
|
};
|
|
|
|
|
|
var next = function(){
|
|
|
Databases[schema.id].execute([method, object, options]);
|
|
|
};
|
|
|
|
|
|
if (!Databases[schema.id]) {
|
|
|
Databases[schema.id] = new ExecutionQueue(schema,next,schema.nolog);
|
|
|
} else {
|
|
|
next();
|
|
|
}
|
|
|
|
|
|
return promise;
|
|
|
};
|
|
|
|
|
|
Backbone.ajaxSync = Backbone.sync;
|
|
|
Backbone.sync = sync;
|
|
|
|
|
|
return { sync: sync, debugLog: debugLog};
|
|
|
}));
|
|
|
|
|
|
/**
|
|
|
* @fileoverview
|
|
|
* - Using the 'QRCode for Javascript library'
|
|
|
* - Fixed dataset of 'QRCode for Javascript library' for support full-spec.
|
|
|
* - this library has no dependencies.
|
|
|
*
|
|
|
* @author davidshimjs
|
|
|
* @see <a href="http://www.d-project.com/" target="_blank">http://www.d-project.com/</a>
|
|
|
* @see <a href="http://jeromeetienne.github.com/jquery-qrcode/" target="_blank">http://jeromeetienne.github.com/jquery-qrcode/</a>
|
|
|
*/
|
|
|
var QRCode;
|
|
|
|
|
|
(function () {
|
|
|
//---------------------------------------------------------------------
|
|
|
// QRCode for JavaScript
|
|
|
//
|
|
|
// Copyright (c) 2009 Kazuhiko Arase
|
|
|
//
|
|
|
// URL: http://www.d-project.com/
|
|
|
//
|
|
|
// Licensed under the MIT license:
|
|
|
// http://www.opensource.org/licenses/mit-license.php
|
|
|
//
|
|
|
// The word "QR Code" is registered trademark of
|
|
|
// DENSO WAVE INCORPORATED
|
|
|
// http://www.denso-wave.com/qrcode/faqpatent-e.html
|
|
|
//
|
|
|
//---------------------------------------------------------------------
|
|
|
function QR8bitByte(data) {
|
|
|
this.mode = QRMode.MODE_8BIT_BYTE;
|
|
|
this.data = data;
|
|
|
this.parsedData = [];
|
|
|
|
|
|
// Added to support UTF-8 Characters
|
|
|
for (var i = 0, l = this.data.length; i < l; i++) {
|
|
|
var byteArray = [];
|
|
|
var code = this.data.charCodeAt(i);
|
|
|
|
|
|
if (code > 0x10000) {
|
|
|
byteArray[0] = 0xF0 | ((code & 0x1C0000) >>> 18);
|
|
|
byteArray[1] = 0x80 | ((code & 0x3F000) >>> 12);
|
|
|
byteArray[2] = 0x80 | ((code & 0xFC0) >>> 6);
|
|
|
byteArray[3] = 0x80 | (code & 0x3F);
|
|
|
} else if (code > 0x800) {
|
|
|
byteArray[0] = 0xE0 | ((code & 0xF000) >>> 12);
|
|
|
byteArray[1] = 0x80 | ((code & 0xFC0) >>> 6);
|
|
|
byteArray[2] = 0x80 | (code & 0x3F);
|
|
|
} else if (code > 0x80) {
|
|
|
byteArray[0] = 0xC0 | ((code & 0x7C0) >>> 6);
|
|
|
byteArray[1] = 0x80 | (code & 0x3F);
|
|
|
} else {
|
|
|
byteArray[0] = code;
|
|
|
}
|
|
|
|
|
|
this.parsedData.push(byteArray);
|
|
|
}
|
|
|
|
|
|
this.parsedData = Array.prototype.concat.apply([], this.parsedData);
|
|
|
|
|
|
if (this.parsedData.length != this.data.length) {
|
|
|
this.parsedData.unshift(191);
|
|
|
this.parsedData.unshift(187);
|
|
|
this.parsedData.unshift(239);
|
|
|
}
|
|
|
}
|
|
|
|
|
|
QR8bitByte.prototype = {
|
|
|
getLength: function (buffer) {
|
|
|
return this.parsedData.length;
|
|
|
},
|
|
|
write: function (buffer) {
|
|
|
for (var i = 0, l = this.parsedData.length; i < l; i++) {
|
|
|
buffer.put(this.parsedData[i], 8);
|
|
|
}
|
|
|
}
|
|
|
};
|
|
|
|
|
|
function QRCodeModel(typeNumber, errorCorrectLevel) {
|
|
|
this.typeNumber = typeNumber;
|
|
|
this.errorCorrectLevel = errorCorrectLevel;
|
|
|
this.modules = null;
|
|
|
this.moduleCount = 0;
|
|
|
this.dataCache = null;
|
|
|
this.dataList = [];
|
|
|
}
|
|
|
|
|
|
QRCodeModel.prototype={addData:function(data){var newData=new QR8bitByte(data);this.dataList.push(newData);this.dataCache=null;},isDark:function(row,col){if(row<0||this.moduleCount<=row||col<0||this.moduleCount<=col){throw new Error(row+","+col);}
|
|
|
return this.modules[row][col];},getModuleCount:function(){return this.moduleCount;},make:function(){this.makeImpl(false,this.getBestMaskPattern());},makeImpl:function(test,maskPattern){this.moduleCount=this.typeNumber*4+17;this.modules=new Array(this.moduleCount);for(var row=0;row<this.moduleCount;row++){this.modules[row]=new Array(this.moduleCount);for(var col=0;col<this.moduleCount;col++){this.modules[row][col]=null;}}
|
|
|
this.setupPositionProbePattern(0,0);this.setupPositionProbePattern(this.moduleCount-7,0);this.setupPositionProbePattern(0,this.moduleCount-7);this.setupPositionAdjustPattern();this.setupTimingPattern();this.setupTypeInfo(test,maskPattern);if(this.typeNumber>=7){this.setupTypeNumber(test);}
|
|
|
if(this.dataCache==null){this.dataCache=QRCodeModel.createData(this.typeNumber,this.errorCorrectLevel,this.dataList);}
|
|
|
this.mapData(this.dataCache,maskPattern);},setupPositionProbePattern:function(row,col){for(var r=-1;r<=7;r++){if(row+r<=-1||this.moduleCount<=row+r)continue;for(var c=-1;c<=7;c++){if(col+c<=-1||this.moduleCount<=col+c)continue;if((0<=r&&r<=6&&(c==0||c==6))||(0<=c&&c<=6&&(r==0||r==6))||(2<=r&&r<=4&&2<=c&&c<=4)){this.modules[row+r][col+c]=true;}else{this.modules[row+r][col+c]=false;}}}},getBestMaskPattern:function(){var minLostPoint=0;var pattern=0;for(var i=0;i<8;i++){this.makeImpl(true,i);var lostPoint=QRUtil.getLostPoint(this);if(i==0||minLostPoint>lostPoint){minLostPoint=lostPoint;pattern=i;}}
|
|
|
return pattern;},createMovieClip:function(target_mc,instance_name,depth){var qr_mc=target_mc.createEmptyMovieClip(instance_name,depth);var cs=1;this.make();for(var row=0;row<this.modules.length;row++){var y=row*cs;for(var col=0;col<this.modules[row].length;col++){var x=col*cs;var dark=this.modules[row][col];if(dark){qr_mc.beginFill(0,100);qr_mc.moveTo(x,y);qr_mc.lineTo(x+cs,y);qr_mc.lineTo(x+cs,y+cs);qr_mc.lineTo(x,y+cs);qr_mc.endFill();}}}
|
|
|
return qr_mc;},setupTimingPattern:function(){for(var r=8;r<this.moduleCount-8;r++){if(this.modules[r][6]!=null){continue;}
|
|
|
this.modules[r][6]=(r%2==0);}
|
|
|
for(var c=8;c<this.moduleCount-8;c++){if(this.modules[6][c]!=null){continue;}
|
|
|
this.modules[6][c]=(c%2==0);}},setupPositionAdjustPattern:function(){var pos=QRUtil.getPatternPosition(this.typeNumber);for(var i=0;i<pos.length;i++){for(var j=0;j<pos.length;j++){var row=pos[i];var col=pos[j];if(this.modules[row][col]!=null){continue;}
|
|
|
for(var r=-2;r<=2;r++){for(var c=-2;c<=2;c++){if(r==-2||r==2||c==-2||c==2||(r==0&&c==0)){this.modules[row+r][col+c]=true;}else{this.modules[row+r][col+c]=false;}}}}}},setupTypeNumber:function(test){var bits=QRUtil.getBCHTypeNumber(this.typeNumber);for(var i=0;i<18;i++){var mod=(!test&&((bits>>i)&1)==1);this.modules[Math.floor(i/3)][i%3+this.moduleCount-8-3]=mod;}
|
|
|
for(var i=0;i<18;i++){var mod=(!test&&((bits>>i)&1)==1);this.modules[i%3+this.moduleCount-8-3][Math.floor(i/3)]=mod;}},setupTypeInfo:function(test,maskPattern){var data=(this.errorCorrectLevel<<3)|maskPattern;var bits=QRUtil.getBCHTypeInfo(data);for(var i=0;i<15;i++){var mod=(!test&&((bits>>i)&1)==1);if(i<6){this.modules[i][8]=mod;}else if(i<8){this.modules[i+1][8]=mod;}else{this.modules[this.moduleCount-15+i][8]=mod;}}
|
|
|
for(var i=0;i<15;i++){var mod=(!test&&((bits>>i)&1)==1);if(i<8){this.modules[8][this.moduleCount-i-1]=mod;}else if(i<9){this.modules[8][15-i-1+1]=mod;}else{this.modules[8][15-i-1]=mod;}}
|
|
|
this.modules[this.moduleCount-8][8]=(!test);},mapData:function(data,maskPattern){var inc=-1;var row=this.moduleCount-1;var bitIndex=7;var byteIndex=0;for(var col=this.moduleCount-1;col>0;col-=2){if(col==6)col--;while(true){for(var c=0;c<2;c++){if(this.modules[row][col-c]==null){var dark=false;if(byteIndex<data.length){dark=(((data[byteIndex]>>>bitIndex)&1)==1);}
|
|
|
var mask=QRUtil.getMask(maskPattern,row,col-c);if(mask){dark=!dark;}
|
|
|
this.modules[row][col-c]=dark;bitIndex--;if(bitIndex==-1){byteIndex++;bitIndex=7;}}}
|
|
|
row+=inc;if(row<0||this.moduleCount<=row){row-=inc;inc=-inc;break;}}}}};QRCodeModel.PAD0=0xEC;QRCodeModel.PAD1=0x11;QRCodeModel.createData=function(typeNumber,errorCorrectLevel,dataList){var rsBlocks=QRRSBlock.getRSBlocks(typeNumber,errorCorrectLevel);var buffer=new QRBitBuffer();for(var i=0;i<dataList.length;i++){var data=dataList[i];buffer.put(data.mode,4);buffer.put(data.getLength(),QRUtil.getLengthInBits(data.mode,typeNumber));data.write(buffer);}
|
|
|
var totalDataCount=0;for(var i=0;i<rsBlocks.length;i++){totalDataCount+=rsBlocks[i].dataCount;}
|
|
|
if(buffer.getLengthInBits()>totalDataCount*8){throw new Error("code length overflow. ("
|
|
|
+buffer.getLengthInBits()
|
|
|
+">"
|
|
|
+totalDataCount*8
|
|
|
+")");}
|
|
|
if(buffer.getLengthInBits()+4<=totalDataCount*8){buffer.put(0,4);}
|
|
|
while(buffer.getLengthInBits()%8!=0){buffer.putBit(false);}
|
|
|
while(true){if(buffer.getLengthInBits()>=totalDataCount*8){break;}
|
|
|
buffer.put(QRCodeModel.PAD0,8);if(buffer.getLengthInBits()>=totalDataCount*8){break;}
|
|
|
buffer.put(QRCodeModel.PAD1,8);}
|
|
|
return QRCodeModel.createBytes(buffer,rsBlocks);};QRCodeModel.createBytes=function(buffer,rsBlocks){var offset=0;var maxDcCount=0;var maxEcCount=0;var dcdata=new Array(rsBlocks.length);var ecdata=new Array(rsBlocks.length);for(var r=0;r<rsBlocks.length;r++){var dcCount=rsBlocks[r].dataCount;var ecCount=rsBlocks[r].totalCount-dcCount;maxDcCount=Math.max(maxDcCount,dcCount);maxEcCount=Math.max(maxEcCount,ecCount);dcdata[r]=new Array(dcCount);for(var i=0;i<dcdata[r].length;i++){dcdata[r][i]=0xff&buffer.buffer[i+offset];}
|
|
|
offset+=dcCount;var rsPoly=QRUtil.getErrorCorrectPolynomial(ecCount);var rawPoly=new QRPolynomial(dcdata[r],rsPoly.getLength()-1);var modPoly=rawPoly.mod(rsPoly);ecdata[r]=new Array(rsPoly.getLength()-1);for(var i=0;i<ecdata[r].length;i++){var modIndex=i+modPoly.getLength()-ecdata[r].length;ecdata[r][i]=(modIndex>=0)?modPoly.get(modIndex):0;}}
|
|
|
var totalCodeCount=0;for(var i=0;i<rsBlocks.length;i++){totalCodeCount+=rsBlocks[i].totalCount;}
|
|
|
var data=new Array(totalCodeCount);var index=0;for(var i=0;i<maxDcCount;i++){for(var r=0;r<rsBlocks.length;r++){if(i<dcdata[r].length){data[index++]=dcdata[r][i];}}}
|
|
|
for(var i=0;i<maxEcCount;i++){for(var r=0;r<rsBlocks.length;r++){if(i<ecdata[r].length){data[index++]=ecdata[r][i];}}}
|
|
|
return data;};var QRMode={MODE_NUMBER:1<<0,MODE_ALPHA_NUM:1<<1,MODE_8BIT_BYTE:1<<2,MODE_KANJI:1<<3};var QRErrorCorrectLevel={L:1,M:0,Q:3,H:2};var QRMaskPattern={PATTERN000:0,PATTERN001:1,PATTERN010:2,PATTERN011:3,PATTERN100:4,PATTERN101:5,PATTERN110:6,PATTERN111:7};var QRUtil={PATTERN_POSITION_TABLE:[[],[6,18],[6,22],[6,26],[6,30],[6,34],[6,22,38],[6,24,42],[6,26,46],[6,28,50],[6,30,54],[6,32,58],[6,34,62],[6,26,46,66],[6,26,48,70],[6,26,50,74],[6,30,54,78],[6,30,56,82],[6,30,58,86],[6,34,62,90],[6,28,50,72,94],[6,26,50,74,98],[6,30,54,78,102],[6,28,54,80,106],[6,32,58,84,110],[6,30,58,86,114],[6,34,62,90,118],[6,26,50,74,98,122],[6,30,54,78,102,126],[6,26,52,78,104,130],[6,30,56,82,108,134],[6,34,60,86,112,138],[6,30,58,86,114,142],[6,34,62,90,118,146],[6,30,54,78,102,126,150],[6,24,50,76,102,128,154],[6,28,54,80,106,132,158],[6,32,58,84,110,136,162],[6,26,54,82,110,138,166],[6,30,58,86,114,142,170]],G15:(1<<10)|(1<<8)|(1<<5)|(1<<4)|(1<<2)|(1<<1)|(1<<0),G18:(1<<12)|(1<<11)|(1<<10)|(1<<9)|(1<<8)|(1<<5)|(1<<2)|(1<<0),G15_MASK:(1<<14)|(1<<12)|(1<<10)|(1<<4)|(1<<1),getBCHTypeInfo:function(data){var d=data<<10;while(QRUtil.getBCHDigit(d)-QRUtil.getBCHDigit(QRUtil.G15)>=0){d^=(QRUtil.G15<<(QRUtil.getBCHDigit(d)-QRUtil.getBCHDigit(QRUtil.G15)));}
|
|
|
return((data<<10)|d)^QRUtil.G15_MASK;},getBCHTypeNumber:function(data){var d=data<<12;while(QRUtil.getBCHDigit(d)-QRUtil.getBCHDigit(QRUtil.G18)>=0){d^=(QRUtil.G18<<(QRUtil.getBCHDigit(d)-QRUtil.getBCHDigit(QRUtil.G18)));}
|
|
|
return(data<<12)|d;},getBCHDigit:function(data){var digit=0;while(data!=0){digit++;data>>>=1;}
|
|
|
return digit;},getPatternPosition:function(typeNumber){return QRUtil.PATTERN_POSITION_TABLE[typeNumber-1];},getMask:function(maskPattern,i,j){switch(maskPattern){case QRMaskPattern.PATTERN000:return(i+j)%2==0;case QRMaskPattern.PATTERN001:return i%2==0;case QRMaskPattern.PATTERN010:return j%3==0;case QRMaskPattern.PATTERN011:return(i+j)%3==0;case QRMaskPattern.PATTERN100:return(Math.floor(i/2)+Math.floor(j/3))%2==0;case QRMaskPattern.PATTERN101:return(i*j)%2+(i*j)%3==0;case QRMaskPattern.PATTERN110:return((i*j)%2+(i*j)%3)%2==0;case QRMaskPattern.PATTERN111:return((i*j)%3+(i+j)%2)%2==0;default:throw new Error("bad maskPattern:"+maskPattern);}},getErrorCorrectPolynomial:function(errorCorrectLength){var a=new QRPolynomial([1],0);for(var i=0;i<errorCorrectLength;i++){a=a.multiply(new QRPolynomial([1,QRMath.gexp(i)],0));}
|
|
|
return a;},getLengthInBits:function(mode,type){if(1<=type&&type<10){switch(mode){case QRMode.MODE_NUMBER:return 10;case QRMode.MODE_ALPHA_NUM:return 9;case QRMode.MODE_8BIT_BYTE:return 8;case QRMode.MODE_KANJI:return 8;default:throw new Error("mode:"+mode);}}else if(type<27){switch(mode){case QRMode.MODE_NUMBER:return 12;case QRMode.MODE_ALPHA_NUM:return 11;case QRMode.MODE_8BIT_BYTE:return 16;case QRMode.MODE_KANJI:return 10;default:throw new Error("mode:"+mode);}}else if(type<41){switch(mode){case QRMode.MODE_NUMBER:return 14;case QRMode.MODE_ALPHA_NUM:return 13;case QRMode.MODE_8BIT_BYTE:return 16;case QRMode.MODE_KANJI:return 12;default:throw new Error("mode:"+mode);}}else{throw new Error("type:"+type);}},getLostPoint:function(qrCode){var moduleCount=qrCode.getModuleCount();var lostPoint=0;for(var row=0;row<moduleCount;row++){for(var col=0;col<moduleCount;col++){var sameCount=0;var dark=qrCode.isDark(row,col);for(var r=-1;r<=1;r++){if(row+r<0||moduleCount<=row+r){continue;}
|
|
|
for(var c=-1;c<=1;c++){if(col+c<0||moduleCount<=col+c){continue;}
|
|
|
if(r==0&&c==0){continue;}
|
|
|
if(dark==qrCode.isDark(row+r,col+c)){sameCount++;}}}
|
|
|
if(sameCount>5){lostPoint+=(3+sameCount-5);}}}
|
|
|
for(var row=0;row<moduleCount-1;row++){for(var col=0;col<moduleCount-1;col++){var count=0;if(qrCode.isDark(row,col))count++;if(qrCode.isDark(row+1,col))count++;if(qrCode.isDark(row,col+1))count++;if(qrCode.isDark(row+1,col+1))count++;if(count==0||count==4){lostPoint+=3;}}}
|
|
|
for(var row=0;row<moduleCount;row++){for(var col=0;col<moduleCount-6;col++){if(qrCode.isDark(row,col)&&!qrCode.isDark(row,col+1)&&qrCode.isDark(row,col+2)&&qrCode.isDark(row,col+3)&&qrCode.isDark(row,col+4)&&!qrCode.isDark(row,col+5)&&qrCode.isDark(row,col+6)){lostPoint+=40;}}}
|
|
|
for(var col=0;col<moduleCount;col++){for(var row=0;row<moduleCount-6;row++){if(qrCode.isDark(row,col)&&!qrCode.isDark(row+1,col)&&qrCode.isDark(row+2,col)&&qrCode.isDark(row+3,col)&&qrCode.isDark(row+4,col)&&!qrCode.isDark(row+5,col)&&qrCode.isDark(row+6,col)){lostPoint+=40;}}}
|
|
|
var darkCount=0;for(var col=0;col<moduleCount;col++){for(var row=0;row<moduleCount;row++){if(qrCode.isDark(row,col)){darkCount++;}}}
|
|
|
var ratio=Math.abs(100*darkCount/moduleCount/moduleCount-50)/5;lostPoint+=ratio*10;return lostPoint;}};var QRMath={glog:function(n){if(n<1){throw new Error("glog("+n+")");}
|
|
|
return QRMath.LOG_TABLE[n];},gexp:function(n){while(n<0){n+=255;}
|
|
|
while(n>=256){n-=255;}
|
|
|
return QRMath.EXP_TABLE[n];},EXP_TABLE:new Array(256),LOG_TABLE:new Array(256)};for(var i=0;i<8;i++){QRMath.EXP_TABLE[i]=1<<i;}
|
|
|
for(var i=8;i<256;i++){QRMath.EXP_TABLE[i]=QRMath.EXP_TABLE[i-4]^QRMath.EXP_TABLE[i-5]^QRMath.EXP_TABLE[i-6]^QRMath.EXP_TABLE[i-8];}
|
|
|
for(var i=0;i<255;i++){QRMath.LOG_TABLE[QRMath.EXP_TABLE[i]]=i;}
|
|
|
function QRPolynomial(num,shift){if(num.length==undefined){throw new Error(num.length+"/"+shift);}
|
|
|
var offset=0;while(offset<num.length&&num[offset]==0){offset++;}
|
|
|
this.num=new Array(num.length-offset+shift);for(var i=0;i<num.length-offset;i++){this.num[i]=num[i+offset];}}
|
|
|
QRPolynomial.prototype={get:function(index){return this.num[index];},getLength:function(){return this.num.length;},multiply:function(e){var num=new Array(this.getLength()+e.getLength()-1);for(var i=0;i<this.getLength();i++){for(var j=0;j<e.getLength();j++){num[i+j]^=QRMath.gexp(QRMath.glog(this.get(i))+QRMath.glog(e.get(j)));}}
|
|
|
return new QRPolynomial(num,0);},mod:function(e){if(this.getLength()-e.getLength()<0){return this;}
|
|
|
var ratio=QRMath.glog(this.get(0))-QRMath.glog(e.get(0));var num=new Array(this.getLength());for(var i=0;i<this.getLength();i++){num[i]=this.get(i);}
|
|
|
for(var i=0;i<e.getLength();i++){num[i]^=QRMath.gexp(QRMath.glog(e.get(i))+ratio);}
|
|
|
return new QRPolynomial(num,0).mod(e);}};function QRRSBlock(totalCount,dataCount){this.totalCount=totalCount;this.dataCount=dataCount;}
|
|
|
QRRSBlock.RS_BLOCK_TABLE=[[1,26,19],[1,26,16],[1,26,13],[1,26,9],[1,44,34],[1,44,28],[1,44,22],[1,44,16],[1,70,55],[1,70,44],[2,35,17],[2,35,13],[1,100,80],[2,50,32],[2,50,24],[4,25,9],[1,134,108],[2,67,43],[2,33,15,2,34,16],[2,33,11,2,34,12],[2,86,68],[4,43,27],[4,43,19],[4,43,15],[2,98,78],[4,49,31],[2,32,14,4,33,15],[4,39,13,1,40,14],[2,121,97],[2,60,38,2,61,39],[4,40,18,2,41,19],[4,40,14,2,41,15],[2,146,116],[3,58,36,2,59,37],[4,36,16,4,37,17],[4,36,12,4,37,13],[2,86,68,2,87,69],[4,69,43,1,70,44],[6,43,19,2,44,20],[6,43,15,2,44,16],[4,101,81],[1,80,50,4,81,51],[4,50,22,4,51,23],[3,36,12,8,37,13],[2,116,92,2,117,93],[6,58,36,2,59,37],[4,46,20,6,47,21],[7,42,14,4,43,15],[4,133,107],[8,59,37,1,60,38],[8,44,20,4,45,21],[12,33,11,4,34,12],[3,145,115,1,146,116],[4,64,40,5,65,41],[11,36,16,5,37,17],[11,36,12,5,37,13],[5,109,87,1,110,88],[5,65,41,5,66,42],[5,54,24,7,55,25],[11,36,12],[5,122,98,1,123,99],[7,73,45,3,74,46],[15,43,19,2,44,20],[3,45,15,13,46,16],[1,135,107,5,136,108],[10,74,46,1,75,47],[1,50,22,15,51,23],[2,42,14,17,43,15],[5,150,120,1,151,121],[9,69,43,4,70,44],[17,50,22,1,51,23],[2,42,14,19,43,15],[3,141,113,4,142,114],[3,70,44,11,71,45],[17,47,21,4,48,22],[9,39,13,16,40,14],[3,135,107,5,136,108],[3,67,41,13,68,42],[15,54,24,5,55,25],[15,43,15,10,44,16],[4,144,116,4,145,117],[17,68,42],[17,50,22,6,51,23],[19,46,16,6,47,17],[2,139,111,7,140,112],[17,74,46],[7,54,24,16,55,25],[34,37,13],[4,151,121,5,152,122],[4,75,47,14,76,48],[11,54,24,14,55,25],[16,45,15,14,46,16],[6,147,117,4,148,118],[6,73,45,14,74,46],[11,54,24,16,55,25],[30,46,16,2,47,17],[8,132,106,4,133,107],[8,75,47,13,76,48],[7,54,24,22,55,25],[22,45,15,13,46,16],[10,142,114,2,143,115],[19,74,46,4,75,47],[28,50,22,6,51,23],[33,46,16,4,47,17],[8,152,122,4,153,123],[22,73,45,3,74,46],[8,53,23,26,54,24],[12,45,15,28,46,16],[3,147,117,10,148,118],[3,73,45,23,74,46],[4,54,24,31,55,25],[11,45,15,31,46,16],[7,146,116,7,147,117],[21,73,45,7,74,46],[1,53,23,37,54,24],[19,45,15,26,46,16],[5,145,115,10,146,116],[19,75,47,10,76,48],[15,54,24,25,55,25],[23,45,15,25,46,16],[13,145,115,3,146,116],[2,74,46,29,75,47],[42,54,24,1,55,25],[23,45,15,28,46,16],[17,145,115],[10,74,46,23,75,47],[10,54,24,35,55,25],[19,45,15,35,46,16],[17,145,115,1,146,116],[14,74,46,21,75,47],[29,54,24,19,55,25],[11,45,15,46,46,16],[13,145,115,6,146,116],[14,74,46,23,75,47],[44,54,24,7,55,25],[59,46,16,1,47,17],[12,151,121,7,152,122],[12,75,47,26,76,48],[39,54,24,14,55,25],[22,45,15,41,46,16],[6,151,121,14,152,122],[6,75,47,34,76,48],[46,54,24,10,55,25],[2,45,15,64,46,16],[17,152,122,4,153,123],[29,74,46,14,75,47],[49,54,24,10,55,25],[24,45,15,46,46,16],[4,152,122,18,153,123],[13,74,46,32,75,47],[48,54,24,14,55,25],[42,45,15,32,46,16],[20,147,117,4,148,118],[40,75,47,7,76,48],[43,54,24,22,55,25],[10,45,15,67,46,16],[19,148,118,6,149,119],[18,75,47,31,76,48],[34,54,24,34,55,25],[20,45,15,61,46,16]];QRRSBlock.getRSBlocks=function(typeNumber,errorCorrectLevel){var rsBlock=QRRSBlock.getRsBlockTable(typeNumber,errorCorrectLevel);if(rsBlock==undefined){throw new Error("bad rs block @ typeNumber:"+typeNumber+"/errorCorrectLevel:"+errorCorrectLevel);}
|
|
|
var length=rsBlock.length/3;var list=[];for(var i=0;i<length;i++){var count=rsBlock[i*3+0];var totalCount=rsBlock[i*3+1];var dataCount=rsBlock[i*3+2];for(var j=0;j<count;j++){list.push(new QRRSBlock(totalCount,dataCount));}}
|
|
|
return list;};QRRSBlock.getRsBlockTable=function(typeNumber,errorCorrectLevel){switch(errorCorrectLevel){case QRErrorCorrectLevel.L:return QRRSBlock.RS_BLOCK_TABLE[(typeNumber-1)*4+0];case QRErrorCorrectLevel.M:return QRRSBlock.RS_BLOCK_TABLE[(typeNumber-1)*4+1];case QRErrorCorrectLevel.Q:return QRRSBlock.RS_BLOCK_TABLE[(typeNumber-1)*4+2];case QRErrorCorrectLevel.H:return QRRSBlock.RS_BLOCK_TABLE[(typeNumber-1)*4+3];default:return undefined;}};function QRBitBuffer(){this.buffer=[];this.length=0;}
|
|
|
QRBitBuffer.prototype={get:function(index){var bufIndex=Math.floor(index/8);return((this.buffer[bufIndex]>>>(7-index%8))&1)==1;},put:function(num,length){for(var i=0;i<length;i++){this.putBit(((num>>>(length-i-1))&1)==1);}},getLengthInBits:function(){return this.length;},putBit:function(bit){var bufIndex=Math.floor(this.length/8);if(this.buffer.length<=bufIndex){this.buffer.push(0);}
|
|
|
if(bit){this.buffer[bufIndex]|=(0x80>>>(this.length%8));}
|
|
|
this.length++;}};var QRCodeLimitLength=[[17,14,11,7],[32,26,20,14],[53,42,32,24],[78,62,46,34],[106,84,60,44],[134,106,74,58],[154,122,86,64],[192,152,108,84],[230,180,130,98],[271,213,151,119],[321,251,177,137],[367,287,203,155],[425,331,241,177],[458,362,258,194],[520,412,292,220],[586,450,322,250],[644,504,364,280],[718,560,394,310],[792,624,442,338],[858,666,482,382],[929,711,509,403],[1003,779,565,439],[1091,857,611,461],[1171,911,661,511],[1273,997,715,535],[1367,1059,751,593],[1465,1125,805,625],[1528,1190,868,658],[1628,1264,908,698],[1732,1370,982,742],[1840,1452,1030,790],[1952,1538,1112,842],[2068,1628,1168,898],[2188,1722,1228,958],[2303,1809,1283,983],[2431,1911,1351,1051],[2563,1989,1423,1093],[2699,2099,1499,1139],[2809,2213,1579,1219],[2953,2331,1663,1273]];
|
|
|
|
|
|
function _isSupportCanvas() {
|
|
|
return typeof CanvasRenderingContext2D != "undefined";
|
|
|
}
|
|
|
|
|
|
// android 2.x doesn't support Data-URI spec
|
|
|
function _getAndroid() {
|
|
|
var android = false;
|
|
|
var sAgent = navigator.userAgent;
|
|
|
|
|
|
if (/android/i.test(sAgent)) { // android
|
|
|
android = true;
|
|
|
aMat = sAgent.toString().match(/android ([0-9]\.[0-9])/i);
|
|
|
|
|
|
if (aMat && aMat[1]) {
|
|
|
android = parseFloat(aMat[1]);
|
|
|
}
|
|
|
}
|
|
|
|
|
|
return android;
|
|
|
}
|
|
|
|
|
|
var svgDrawer = (function() {
|
|
|
|
|
|
var Drawing = function (el, htOption) {
|
|
|
this._el = el;
|
|
|
this._htOption = htOption;
|
|
|
};
|
|
|
|
|
|
Drawing.prototype.draw = function (oQRCode) {
|
|
|
var _htOption = this._htOption;
|
|
|
var _el = this._el;
|
|
|
var nCount = oQRCode.getModuleCount();
|
|
|
var nWidth = Math.floor(_htOption.width / nCount);
|
|
|
var nHeight = Math.floor(_htOption.height / nCount);
|
|
|
|
|
|
this.clear();
|
|
|
|
|
|
function makeSVG(tag, attrs) {
|
|
|
var el = document.createElementNS('http://www.w3.org/2000/svg', tag);
|
|
|
for (var k in attrs)
|
|
|
if (attrs.hasOwnProperty(k)) el.setAttribute(k, attrs[k]);
|
|
|
return el;
|
|
|
}
|
|
|
|
|
|
var svg = makeSVG("svg" , {'viewBox': '0 0 ' + String(nCount) + " " + String(nCount), 'width': '100%', 'height': '100%', 'fill': _htOption.colorLight});
|
|
|
svg.setAttributeNS("http://www.w3.org/2000/xmlns/", "xmlns:xlink", "http://www.w3.org/1999/xlink");
|
|
|
_el.appendChild(svg);
|
|
|
|
|
|
svg.appendChild(makeSVG("rect", {"fill": _htOption.colorDark, "width": "1", "height": "1", "id": "template"}));
|
|
|
|
|
|
for (var row = 0; row < nCount; row++) {
|
|
|
for (var col = 0; col < nCount; col++) {
|
|
|
if (oQRCode.isDark(row, col)) {
|
|
|
var child = makeSVG("use", {"x": String(row), "y": String(col)});
|
|
|
child.setAttributeNS("http://www.w3.org/1999/xlink", "href", "#template")
|
|
|
svg.appendChild(child);
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
};
|
|
|
Drawing.prototype.clear = function () {
|
|
|
while (this._el.hasChildNodes())
|
|
|
this._el.removeChild(this._el.lastChild);
|
|
|
};
|
|
|
return Drawing;
|
|
|
})();
|
|
|
|
|
|
var useSVG = document.documentElement.tagName.toLowerCase() === "svg";
|
|
|
|
|
|
// Drawing in DOM by using Table tag
|
|
|
var Drawing = useSVG ? svgDrawer : !_isSupportCanvas() ? (function () {
|
|
|
var Drawing = function (el, htOption) {
|
|
|
this._el = el;
|
|
|
this._htOption = htOption;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Draw the QRCode
|
|
|
*
|
|
|
* @param {QRCode} oQRCode
|
|
|
*/
|
|
|
Drawing.prototype.draw = function (oQRCode) {
|
|
|
var _htOption = this._htOption;
|
|
|
var _el = this._el;
|
|
|
var nCount = oQRCode.getModuleCount();
|
|
|
var nWidth = Math.floor(_htOption.width / nCount);
|
|
|
var nHeight = Math.floor(_htOption.height / nCount);
|
|
|
var aHTML = ['<table style="border:0;border-collapse:collapse;">'];
|
|
|
|
|
|
for (var row = 0; row < nCount; row++) {
|
|
|
aHTML.push('<tr>');
|
|
|
|
|
|
for (var col = 0; col < nCount; col++) {
|
|
|
aHTML.push('<td style="border:0;border-collapse:collapse;padding:0;margin:0;width:' + nWidth + 'px;height:' + nHeight + 'px;background-color:' + (oQRCode.isDark(row, col) ? _htOption.colorDark : _htOption.colorLight) + ';"></td>');
|
|
|
}
|
|
|
|
|
|
aHTML.push('</tr>');
|
|
|
}
|
|
|
|
|
|
aHTML.push('</table>');
|
|
|
_el.innerHTML = aHTML.join('');
|
|
|
|
|
|
// Fix the margin values as real size.
|
|
|
var elTable = _el.childNodes[0];
|
|
|
var nLeftMarginTable = (_htOption.width - elTable.offsetWidth) / 2;
|
|
|
var nTopMarginTable = (_htOption.height - elTable.offsetHeight) / 2;
|
|
|
|
|
|
if (nLeftMarginTable > 0 && nTopMarginTable > 0) {
|
|
|
elTable.style.margin = nTopMarginTable + "px " + nLeftMarginTable + "px";
|
|
|
}
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Clear the QRCode
|
|
|
*/
|
|
|
Drawing.prototype.clear = function () {
|
|
|
this._el.innerHTML = '';
|
|
|
};
|
|
|
|
|
|
return Drawing;
|
|
|
})() : (function () { // Drawing in Canvas
|
|
|
function _onMakeImage() {
|
|
|
this._elImage.src = this._elCanvas.toDataURL("image/png");
|
|
|
this._elImage.style.display = "block";
|
|
|
this._elCanvas.style.display = "none";
|
|
|
}
|
|
|
|
|
|
// Android 2.1 bug workaround
|
|
|
// http://code.google.com/p/android/issues/detail?id=5141
|
|
|
if (this._android && this._android <= 2.1) {
|
|
|
var factor = 1 / window.devicePixelRatio;
|
|
|
var drawImage = CanvasRenderingContext2D.prototype.drawImage;
|
|
|
CanvasRenderingContext2D.prototype.drawImage = function (image, sx, sy, sw, sh, dx, dy, dw, dh) {
|
|
|
if (("nodeName" in image) && /img/i.test(image.nodeName)) {
|
|
|
for (var i = arguments.length - 1; i >= 1; i--) {
|
|
|
arguments[i] = arguments[i] * factor;
|
|
|
}
|
|
|
} else if (typeof dw == "undefined") {
|
|
|
arguments[1] *= factor;
|
|
|
arguments[2] *= factor;
|
|
|
arguments[3] *= factor;
|
|
|
arguments[4] *= factor;
|
|
|
}
|
|
|
|
|
|
drawImage.apply(this, arguments);
|
|
|
};
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* Check whether the user's browser supports Data URI or not
|
|
|
*
|
|
|
* @private
|
|
|
* @param {Function} fSuccess Occurs if it supports Data URI
|
|
|
* @param {Function} fFail Occurs if it doesn't support Data URI
|
|
|
*/
|
|
|
function _safeSetDataURI(fSuccess, fFail) {
|
|
|
var self = this;
|
|
|
self._fFail = fFail;
|
|
|
self._fSuccess = fSuccess;
|
|
|
|
|
|
// Check it just once
|
|
|
if (self._bSupportDataURI === null) {
|
|
|
var el = document.createElement("img");
|
|
|
var fOnError = function() {
|
|
|
self._bSupportDataURI = false;
|
|
|
|
|
|
if (self._fFail) {
|
|
|
_fFail.call(self);
|
|
|
}
|
|
|
};
|
|
|
var fOnSuccess = function() {
|
|
|
self._bSupportDataURI = true;
|
|
|
|
|
|
if (self._fSuccess) {
|
|
|
self._fSuccess.call(self);
|
|
|
}
|
|
|
};
|
|
|
|
|
|
el.onabort = fOnError;
|
|
|
el.onerror = fOnError;
|
|
|
el.onload = fOnSuccess;
|
|
|
el.src = ""; // the Image contains 1px data.
|
|
|
return;
|
|
|
} else if (self._bSupportDataURI === true && self._fSuccess) {
|
|
|
self._fSuccess.call(self);
|
|
|
} else if (self._bSupportDataURI === false && self._fFail) {
|
|
|
self._fFail.call(self);
|
|
|
}
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Drawing QRCode by using canvas
|
|
|
*
|
|
|
* @constructor
|
|
|
* @param {HTMLElement} el
|
|
|
* @param {Object} htOption QRCode Options
|
|
|
*/
|
|
|
var Drawing = function (el, htOption) {
|
|
|
this._bIsPainted = false;
|
|
|
this._android = _getAndroid();
|
|
|
|
|
|
this._htOption = htOption;
|
|
|
this._elCanvas = document.createElement("canvas");
|
|
|
this._elCanvas.width = htOption.width;
|
|
|
this._elCanvas.height = htOption.height;
|
|
|
el.appendChild(this._elCanvas);
|
|
|
this._el = el;
|
|
|
this._oContext = this._elCanvas.getContext("2d");
|
|
|
this._bIsPainted = false;
|
|
|
this._elImage = document.createElement("img");
|
|
|
this._elImage.alt = "Scan me!";
|
|
|
this._elImage.style.display = "none";
|
|
|
this._el.appendChild(this._elImage);
|
|
|
this._bSupportDataURI = null;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Draw the QRCode
|
|
|
*
|
|
|
* @param {QRCode} oQRCode
|
|
|
*/
|
|
|
Drawing.prototype.draw = function (oQRCode) {
|
|
|
var _elImage = this._elImage;
|
|
|
var _oContext = this._oContext;
|
|
|
var _htOption = this._htOption;
|
|
|
|
|
|
var nCount = oQRCode.getModuleCount();
|
|
|
var nWidth = _htOption.width / nCount;
|
|
|
var nHeight = _htOption.height / nCount;
|
|
|
var nRoundedWidth = Math.round(nWidth);
|
|
|
var nRoundedHeight = Math.round(nHeight);
|
|
|
|
|
|
_elImage.style.display = "none";
|
|
|
this.clear();
|
|
|
|
|
|
for (var row = 0; row < nCount; row++) {
|
|
|
for (var col = 0; col < nCount; col++) {
|
|
|
var bIsDark = oQRCode.isDark(row, col);
|
|
|
var nLeft = col * nWidth;
|
|
|
var nTop = row * nHeight;
|
|
|
_oContext.strokeStyle = bIsDark ? _htOption.colorDark : _htOption.colorLight;
|
|
|
_oContext.lineWidth = 1;
|
|
|
_oContext.fillStyle = bIsDark ? _htOption.colorDark : _htOption.colorLight;
|
|
|
_oContext.fillRect(nLeft, nTop, nWidth, nHeight);
|
|
|
|
|
|
// 안티 앨리어싱 방지 처리
|
|
|
_oContext.strokeRect(
|
|
|
Math.floor(nLeft) + 0.5,
|
|
|
Math.floor(nTop) + 0.5,
|
|
|
nRoundedWidth,
|
|
|
nRoundedHeight
|
|
|
);
|
|
|
|
|
|
_oContext.strokeRect(
|
|
|
Math.ceil(nLeft) - 0.5,
|
|
|
Math.ceil(nTop) - 0.5,
|
|
|
nRoundedWidth,
|
|
|
nRoundedHeight
|
|
|
);
|
|
|
}
|
|
|
}
|
|
|
|
|
|
this._bIsPainted = true;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Make the image from Canvas if the browser supports Data URI.
|
|
|
*/
|
|
|
Drawing.prototype.makeImage = function () {
|
|
|
if (this._bIsPainted) {
|
|
|
_safeSetDataURI.call(this, _onMakeImage);
|
|
|
}
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Return whether the QRCode is painted or not
|
|
|
*
|
|
|
* @return {Boolean}
|
|
|
*/
|
|
|
Drawing.prototype.isPainted = function () {
|
|
|
return this._bIsPainted;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Clear the QRCode
|
|
|
*/
|
|
|
Drawing.prototype.clear = function () {
|
|
|
this._oContext.clearRect(0, 0, this._elCanvas.width, this._elCanvas.height);
|
|
|
this._bIsPainted = false;
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* @private
|
|
|
* @param {Number} nNumber
|
|
|
*/
|
|
|
Drawing.prototype.round = function (nNumber) {
|
|
|
if (!nNumber) {
|
|
|
return nNumber;
|
|
|
}
|
|
|
|
|
|
return Math.floor(nNumber * 1000) / 1000;
|
|
|
};
|
|
|
|
|
|
return Drawing;
|
|
|
})();
|
|
|
|
|
|
/**
|
|
|
* Get the type by string length
|
|
|
*
|
|
|
* @private
|
|
|
* @param {String} sText
|
|
|
* @param {Number} nCorrectLevel
|
|
|
* @return {Number} type
|
|
|
*/
|
|
|
function _getTypeNumber(sText, nCorrectLevel) {
|
|
|
var nType = 1;
|
|
|
var length = _getUTF8Length(sText);
|
|
|
|
|
|
for (var i = 0, len = QRCodeLimitLength.length; i <= len; i++) {
|
|
|
var nLimit = 0;
|
|
|
|
|
|
switch (nCorrectLevel) {
|
|
|
case QRErrorCorrectLevel.L :
|
|
|
nLimit = QRCodeLimitLength[i][0];
|
|
|
break;
|
|
|
case QRErrorCorrectLevel.M :
|
|
|
nLimit = QRCodeLimitLength[i][1];
|
|
|
break;
|
|
|
case QRErrorCorrectLevel.Q :
|
|
|
nLimit = QRCodeLimitLength[i][2];
|
|
|
break;
|
|
|
case QRErrorCorrectLevel.H :
|
|
|
nLimit = QRCodeLimitLength[i][3];
|
|
|
break;
|
|
|
}
|
|
|
|
|
|
if (length <= nLimit) {
|
|
|
break;
|
|
|
} else {
|
|
|
nType++;
|
|
|
}
|
|
|
}
|
|
|
|
|
|
if (nType > QRCodeLimitLength.length) {
|
|
|
throw new Error("Too long data");
|
|
|
}
|
|
|
|
|
|
return nType;
|
|
|
}
|
|
|
|
|
|
function _getUTF8Length(sText) {
|
|
|
var replacedText = encodeURI(sText).toString().replace(/\%[0-9a-fA-F]{2}/g, 'a');
|
|
|
return replacedText.length + (replacedText.length != sText ? 3 : 0);
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* @class QRCode
|
|
|
* @constructor
|
|
|
* @example
|
|
|
* new QRCode(document.getElementById("test"), "http://jindo.dev.naver.com/collie");
|
|
|
*
|
|
|
* @example
|
|
|
* var oQRCode = new QRCode("test", {
|
|
|
* text : "http://naver.com",
|
|
|
* width : 128,
|
|
|
* height : 128
|
|
|
* });
|
|
|
*
|
|
|
* oQRCode.clear(); // Clear the QRCode.
|
|
|
* oQRCode.makeCode("http://map.naver.com"); // Re-create the QRCode.
|
|
|
*
|
|
|
* @param {HTMLElement|String} el target element or 'id' attribute of element.
|
|
|
* @param {Object|String} vOption
|
|
|
* @param {String} vOption.text QRCode link data
|
|
|
* @param {Number} [vOption.width=256]
|
|
|
* @param {Number} [vOption.height=256]
|
|
|
* @param {String} [vOption.colorDark="#000000"]
|
|
|
* @param {String} [vOption.colorLight="#ffffff"]
|
|
|
* @param {QRCode.CorrectLevel} [vOption.correctLevel=QRCode.CorrectLevel.H] [L|M|Q|H]
|
|
|
*/
|
|
|
QRCode = function (el, vOption) {
|
|
|
this._htOption = {
|
|
|
width : 256,
|
|
|
height : 256,
|
|
|
typeNumber : 4,
|
|
|
colorDark : "#000000",
|
|
|
colorLight : "#ffffff",
|
|
|
correctLevel : QRErrorCorrectLevel.H
|
|
|
};
|
|
|
|
|
|
if (typeof vOption === 'string') {
|
|
|
vOption = {
|
|
|
text : vOption
|
|
|
};
|
|
|
}
|
|
|
|
|
|
// Overwrites options
|
|
|
if (vOption) {
|
|
|
for (var i in vOption) {
|
|
|
this._htOption[i] = vOption[i];
|
|
|
}
|
|
|
}
|
|
|
|
|
|
if (typeof el == "string") {
|
|
|
el = document.getElementById(el);
|
|
|
}
|
|
|
|
|
|
this._android = _getAndroid();
|
|
|
this._el = el;
|
|
|
this._oQRCode = null;
|
|
|
this._oDrawing = new Drawing(this._el, this._htOption);
|
|
|
|
|
|
if (this._htOption.text) {
|
|
|
this.makeCode(this._htOption.text);
|
|
|
}
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Make the QRCode
|
|
|
*
|
|
|
* @param {String} sText link data
|
|
|
*/
|
|
|
QRCode.prototype.makeCode = function (sText) {
|
|
|
this._oQRCode = new QRCodeModel(_getTypeNumber(sText, this._htOption.correctLevel), this._htOption.correctLevel);
|
|
|
this._oQRCode.addData(sText);
|
|
|
this._oQRCode.make();
|
|
|
this._el.title = sText;
|
|
|
this._oDrawing.draw(this._oQRCode);
|
|
|
this.makeImage();
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Make the Image from Canvas element
|
|
|
* - It occurs automatically
|
|
|
* - Android below 3 doesn't support Data-URI spec.
|
|
|
*
|
|
|
* @private
|
|
|
*/
|
|
|
QRCode.prototype.makeImage = function () {
|
|
|
if (typeof this._oDrawing.makeImage == "function" && (!this._android || this._android >= 3)) {
|
|
|
this._oDrawing.makeImage();
|
|
|
}
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Clear the QRCode
|
|
|
*/
|
|
|
QRCode.prototype.clear = function () {
|
|
|
this._oDrawing.clear();
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* @name QRCode.CorrectLevel
|
|
|
*/
|
|
|
QRCode.CorrectLevel = QRErrorCorrectLevel;
|
|
|
})();
|
|
|
|
|
|
(function(){var COMPILED=!0,goog=goog||{};goog.global=this;goog.isDef=function(a){return void 0!==a};goog.exportPath_=function(a,b,c){a=a.split(".");c=c||goog.global;a[0]in c||!c.execScript||c.execScript("var "+a[0]);for(var d;a.length&&(d=a.shift());)!a.length&&goog.isDef(b)?c[d]=b:c=c[d]?c[d]:c[d]={}};
|
|
|
goog.define=function(a,b){var c=b;COMPILED||(goog.global.CLOSURE_UNCOMPILED_DEFINES&&Object.prototype.hasOwnProperty.call(goog.global.CLOSURE_UNCOMPILED_DEFINES,a)?c=goog.global.CLOSURE_UNCOMPILED_DEFINES[a]:goog.global.CLOSURE_DEFINES&&Object.prototype.hasOwnProperty.call(goog.global.CLOSURE_DEFINES,a)&&(c=goog.global.CLOSURE_DEFINES[a]));goog.exportPath_(a,c)};goog.DEBUG=!0;goog.LOCALE="en";goog.TRUSTED_SITE=!0;goog.STRICT_MODE_COMPATIBLE=!1;
|
|
|
goog.provide=function(a){if(!COMPILED&&goog.isProvided_(a))throw Error('Namespace "'+a+'" already declared.');goog.constructNamespace_(a)};goog.constructNamespace_=function(a,b){if(!COMPILED){delete goog.implicitNamespaces_[a];for(var c=a;(c=c.substring(0,c.lastIndexOf(".")))&&!goog.getObjectByName(c);)goog.implicitNamespaces_[c]=!0}goog.exportPath_(a,b)};
|
|
|
goog.module=function(a){if(!goog.isString(a)||!a)throw Error("Invalid module identifier");if(!goog.isInModuleLoader_())throw Error("Module "+a+" has been loaded incorrectly.");if(goog.moduleLoaderState_.moduleName)throw Error("goog.module may only be called once per module.");goog.moduleLoaderState_.moduleName=a;if(!COMPILED){if(goog.isProvided_(a))throw Error('Namespace "'+a+'" already declared.');delete goog.implicitNamespaces_[a]}};goog.module.get=function(a){return goog.module.getInternal_(a)};
|
|
|
goog.module.getInternal_=function(a){if(!COMPILED)return goog.isProvided_(a)?a in goog.loadedModules_?goog.loadedModules_[a]:goog.getObjectByName(a):null};goog.moduleLoaderState_=null;goog.isInModuleLoader_=function(){return null!=goog.moduleLoaderState_};goog.module.declareTestMethods=function(){if(!goog.isInModuleLoader_())throw Error("goog.module.declareTestMethods must be called from within a goog.module");goog.moduleLoaderState_.declareTestMethods=!0};
|
|
|
goog.module.declareLegacyNamespace=function(){if(!COMPILED&&!goog.isInModuleLoader_())throw Error("goog.module.declareLegacyNamespace must be called from within a goog.module");if(!COMPILED&&!goog.moduleLoaderState_.moduleName)throw Error("goog.module must be called prior to goog.module.declareLegacyNamespace.");goog.moduleLoaderState_.declareLegacyNamespace=!0};
|
|
|
goog.setTestOnly=function(a){if(COMPILED&&!goog.DEBUG)throw a=a||"",Error("Importing test-only code into non-debug environment"+(a?": "+a:"."));};goog.forwardDeclare=function(a){};COMPILED||(goog.isProvided_=function(a){return a in goog.loadedModules_||!goog.implicitNamespaces_[a]&&goog.isDefAndNotNull(goog.getObjectByName(a))},goog.implicitNamespaces_={"goog.module":!0});
|
|
|
goog.getObjectByName=function(a,b){for(var c=a.split("."),d=b||goog.global,e;e=c.shift();)if(goog.isDefAndNotNull(d[e]))d=d[e];else return null;return d};goog.globalize=function(a,b){var c=b||goog.global,d;for(d in a)c[d]=a[d]};goog.addDependency=function(a,b,c,d){if(goog.DEPENDENCIES_ENABLED){var e;a=a.replace(/\\/g,"/");for(var f=goog.dependencies_,g=0;e=b[g];g++)f.nameToPath[e]=a,f.pathIsModule[a]=!!d;for(d=0;b=c[d];d++)a in f.requires||(f.requires[a]={}),f.requires[a][b]=!0}};
|
|
|
goog.ENABLE_DEBUG_LOADER=!0;goog.logToConsole_=function(a){goog.global.console&&goog.global.console.error(a)};goog.require=function(a){if(!COMPILED){if(goog.isProvided_(a))return goog.isInModuleLoader_()?goog.module.getInternal_(a):null;if(goog.ENABLE_DEBUG_LOADER){var b=goog.getPathFromDeps_(a);if(b)return goog.included_[b]=!0,goog.writeScripts_(),null}a="goog.require could not find: "+a;goog.logToConsole_(a);throw Error(a);}};goog.basePath="";goog.nullFunction=function(){};
|
|
|
goog.identityFunction=function(a,b){return a};goog.abstractMethod=function(){throw Error("unimplemented abstract method");};goog.addSingletonGetter=function(a){a.getInstance=function(){if(a.instance_)return a.instance_;goog.DEBUG&&(goog.instantiatedSingletons_[goog.instantiatedSingletons_.length]=a);return a.instance_=new a}};goog.instantiatedSingletons_=[];goog.LOAD_MODULE_USING_EVAL=!0;goog.SEAL_MODULE_EXPORTS=goog.DEBUG;goog.loadedModules_={};goog.DEPENDENCIES_ENABLED=!COMPILED&&goog.ENABLE_DEBUG_LOADER;
|
|
|
goog.DEPENDENCIES_ENABLED&&(goog.included_={},goog.dependencies_={pathIsModule:{},nameToPath:{},requires:{},visited:{},written:{}},goog.inHtmlDocument_=function(){var a=goog.global.document;return"undefined"!=typeof a&&"write"in a},goog.findBasePath_=function(){if(goog.global.CLOSURE_BASE_PATH)goog.basePath=goog.global.CLOSURE_BASE_PATH;else if(goog.inHtmlDocument_())for(var a=goog.global.document.getElementsByTagName("script"),b=a.length-1;0<=b;--b){var c=a[b].src,d=c.lastIndexOf("?"),d=-1==d?c.length:
|
|
|
d;if("base.js"==c.substr(d-7,7)){goog.basePath=c.substr(0,d-7);break}}},goog.importScript_=function(a,b){(goog.global.CLOSURE_IMPORT_SCRIPT||goog.writeScriptTag_)(a,b)&&(goog.dependencies_.written[a]=!0)},goog.IS_OLD_IE_=goog.global.document&&goog.global.document.all&&!goog.global.atob,goog.importModule_=function(a){goog.importScript_("",'goog.retrieveAndExecModule_("'+a+'");')&&(goog.dependencies_.written[a]=!0)},goog.queuedModules_=[],goog.retrieveAndExecModule_=function(a){for(var b;-1!=(b=a.indexOf("/./"));)a=
|
|
|
a.substr(0,b)+a.substr(b+2);for(;-1!=(b=a.indexOf("/../"));){var c=a.lastIndexOf("/",b-1);a=a.substr(0,c)+a.substr(b+3)}b=goog.global.CLOSURE_IMPORT_SCRIPT||goog.writeScriptTag_;var d=null,c=new goog.global.XMLHttpRequest;c.onload=function(){d=this.responseText};c.open("get",a,!1);c.send();d=c.responseText;if(null!=d)c=goog.wrapModule_(a,d),goog.IS_OLD_IE_?goog.queuedModules_.push(c):b(a,c),goog.dependencies_.written[a]=!0;else throw Error("load of "+a+"failed");},goog.wrapModule_=function(a,b){return goog.LOAD_MODULE_USING_EVAL&&
|
|
|
goog.isDef(goog.global.JSON)?"goog.loadModule("+goog.global.JSON.stringify(b+"\n//# sourceURL="+a+"\n")+");":'goog.loadModule(function(exports) {"use strict";'+b+"\n;return exports});\n//# sourceURL="+a+"\n"},goog.loadQueuedModules_=function(){var a=goog.queuedModules_.length;if(0<a){var b=goog.queuedModules_;goog.queuedModules_=[];for(var c=0;c<a;c++)goog.globalEval(b[c])}},goog.loadModule=function(a){try{goog.moduleLoaderState_={moduleName:void 0,declareTestMethods:!1};var b;if(goog.isFunction(a))b=
|
|
|
a.call(goog.global,{});else if(goog.isString(a))b=goog.loadModuleFromSource_.call(goog.global,a);else throw Error("Invalid module definition");var c=goog.moduleLoaderState_.moduleName;if(!goog.isString(c)||!c)throw Error('Invalid module name "'+c+'"');goog.moduleLoaderState_.declareLegacyNamespace?goog.constructNamespace_(c,b):goog.SEAL_MODULE_EXPORTS&&Object.seal&&Object.seal(b);goog.loadedModules_[c]=b;if(goog.moduleLoaderState_.declareTestMethods)for(var d in b)if(0===d.indexOf("test",0)||"tearDown"==
|
|
|
d||"setUp"==d||"setUpPage"==d||"tearDownPage"==d)goog.global[d]=b[d]}finally{goog.moduleLoaderState_=null}},goog.loadModuleFromSource_=function(a){eval(a);return{}},goog.writeScriptTag_=function(a,b){if(goog.inHtmlDocument_()){var c=goog.global.document;if("complete"==c.readyState){if(/\bdeps.js$/.test(a))return!1;throw Error('Cannot write "'+a+'" after document load');}var d=goog.IS_OLD_IE_;void 0===b?d?(d=" onreadystatechange='goog.onScriptLoad_(this, "+ ++goog.lastNonModuleScriptIndex_+")' ",c.write('<script type="text/javascript" src="'+
|
|
|
a+'"'+d+">\x3c/script>")):c.write('<script type="text/javascript" src="'+a+'">\x3c/script>'):c.write('<script type="text/javascript">'+b+"\x3c/script>");return!0}return!1},goog.lastNonModuleScriptIndex_=0,goog.onScriptLoad_=function(a,b){"complete"==a.readyState&&goog.lastNonModuleScriptIndex_==b&&goog.loadQueuedModules_();return!0},goog.writeScripts_=function(){function a(e){if(!(e in d.written)){if(!(e in d.visited)&&(d.visited[e]=!0,e in d.requires))for(var f in d.requires[e])if(!goog.isProvided_(f))if(f in
|
|
|
d.nameToPath)a(d.nameToPath[f]);else throw Error("Undefined nameToPath for "+f);e in c||(c[e]=!0,b.push(e))}}var b=[],c={},d=goog.dependencies_,e;for(e in goog.included_)d.written[e]||a(e);for(var f=0;f<b.length;f++)e=b[f],goog.dependencies_.written[e]=!0;var g=goog.moduleLoaderState_;goog.moduleLoaderState_=null;for(f=0;f<b.length;f++)if(e=b[f])d.pathIsModule[e]?goog.importModule_(goog.basePath+e):goog.importScript_(goog.basePath+e);else throw goog.moduleLoaderState_=g,Error("Undefined script input");
|
|
|
goog.moduleLoaderState_=g},goog.getPathFromDeps_=function(a){return a in goog.dependencies_.nameToPath?goog.dependencies_.nameToPath[a]:null},goog.findBasePath_(),goog.global.CLOSURE_NO_DEPS||goog.importScript_(goog.basePath+"deps.js"));
|
|
|
goog.typeOf=function(a){var b=typeof a;if("object"==b)if(a){if(a instanceof Array)return"array";if(a instanceof Object)return b;var c=Object.prototype.toString.call(a);if("[object Window]"==c)return"object";if("[object Array]"==c||"number"==typeof a.length&&"undefined"!=typeof a.splice&&"undefined"!=typeof a.propertyIsEnumerable&&!a.propertyIsEnumerable("splice"))return"array";if("[object Function]"==c||"undefined"!=typeof a.call&&"undefined"!=typeof a.propertyIsEnumerable&&!a.propertyIsEnumerable("call"))return"function"}else return"null";
|
|
|
else if("function"==b&&"undefined"==typeof a.call)return"object";return b};goog.isNull=function(a){return null===a};goog.isDefAndNotNull=function(a){return null!=a};goog.isArray=function(a){return"array"==goog.typeOf(a)};goog.isArrayLike=function(a){var b=goog.typeOf(a);return"array"==b||"object"==b&&"number"==typeof a.length};goog.isDateLike=function(a){return goog.isObject(a)&&"function"==typeof a.getFullYear};goog.isString=function(a){return"string"==typeof a};
|
|
|
goog.isBoolean=function(a){return"boolean"==typeof a};goog.isNumber=function(a){return"number"==typeof a};goog.isFunction=function(a){return"function"==goog.typeOf(a)};goog.isObject=function(a){var b=typeof a;return"object"==b&&null!=a||"function"==b};goog.getUid=function(a){return a[goog.UID_PROPERTY_]||(a[goog.UID_PROPERTY_]=++goog.uidCounter_)};goog.hasUid=function(a){return!!a[goog.UID_PROPERTY_]};goog.removeUid=function(a){"removeAttribute"in a&&a.removeAttribute(goog.UID_PROPERTY_);try{delete a[goog.UID_PROPERTY_]}catch(b){}};
|
|
|
goog.UID_PROPERTY_="closure_uid_"+(1E9*Math.random()>>>0);goog.uidCounter_=0;goog.getHashCode=goog.getUid;goog.removeHashCode=goog.removeUid;goog.cloneObject=function(a){var b=goog.typeOf(a);if("object"==b||"array"==b){if(a.clone)return a.clone();var b="array"==b?[]:{},c;for(c in a)b[c]=goog.cloneObject(a[c]);return b}return a};goog.bindNative_=function(a,b,c){return a.call.apply(a.bind,arguments)};
|
|
|
goog.bindJs_=function(a,b,c){if(!a)throw Error();if(2<arguments.length){var d=Array.prototype.slice.call(arguments,2);return function(){var c=Array.prototype.slice.call(arguments);Array.prototype.unshift.apply(c,d);return a.apply(b,c)}}return function(){return a.apply(b,arguments)}};goog.bind=function(a,b,c){Function.prototype.bind&&-1!=Function.prototype.bind.toString().indexOf("native code")?goog.bind=goog.bindNative_:goog.bind=goog.bindJs_;return goog.bind.apply(null,arguments)};
|
|
|
goog.partial=function(a,b){var c=Array.prototype.slice.call(arguments,1);return function(){var b=c.slice();b.push.apply(b,arguments);return a.apply(this,b)}};goog.mixin=function(a,b){for(var c in b)a[c]=b[c]};goog.now=goog.TRUSTED_SITE&&Date.now||function(){return+new Date};
|
|
|
goog.globalEval=function(a){if(goog.global.execScript)goog.global.execScript(a,"JavaScript");else if(goog.global.eval)if(null==goog.evalWorksForGlobals_&&(goog.global.eval("var _et_ = 1;"),"undefined"!=typeof goog.global._et_?(delete goog.global._et_,goog.evalWorksForGlobals_=!0):goog.evalWorksForGlobals_=!1),goog.evalWorksForGlobals_)goog.global.eval(a);else{var b=goog.global.document,c=b.createElement("script");c.type="text/javascript";c.defer=!1;c.appendChild(b.createTextNode(a));b.body.appendChild(c);
|
|
|
b.body.removeChild(c)}else throw Error("goog.globalEval not available");};goog.evalWorksForGlobals_=null;goog.getCssName=function(a,b){var c=function(a){return goog.cssNameMapping_[a]||a},d=function(a){a=a.split("-");for(var b=[],d=0;d<a.length;d++)b.push(c(a[d]));return b.join("-")},d=goog.cssNameMapping_?"BY_WHOLE"==goog.cssNameMappingStyle_?c:d:function(a){return a};return b?a+"-"+d(b):d(a)};goog.setCssNameMapping=function(a,b){goog.cssNameMapping_=a;goog.cssNameMappingStyle_=b};
|
|
|
!COMPILED&&goog.global.CLOSURE_CSS_NAME_MAPPING&&(goog.cssNameMapping_=goog.global.CLOSURE_CSS_NAME_MAPPING);goog.getMsg=function(a,b){b&&(a=a.replace(/\{\$([^}]+)}/g,function(a,d){return d in b?b[d]:a}));return a};goog.getMsgWithFallback=function(a,b){return a};goog.exportSymbol=function(a,b,c){goog.exportPath_(a,b,c)};goog.exportProperty=function(a,b,c){a[b]=c};
|
|
|
goog.inherits=function(a,b){function c(){}c.prototype=b.prototype;a.superClass_=b.prototype;a.prototype=new c;a.prototype.constructor=a;a.base=function(a,c,f){var g=Array.prototype.slice.call(arguments,2);return b.prototype[c].apply(a,g)}};
|
|
|
goog.base=function(a,b,c){var d=arguments.callee.caller;if(goog.STRICT_MODE_COMPATIBLE||goog.DEBUG&&!d)throw Error("arguments.caller not defined. goog.base() cannot be used with strict mode code. See http://www.ecma-international.org/ecma-262/5.1/#sec-C");if(d.superClass_)return d.superClass_.constructor.apply(a,Array.prototype.slice.call(arguments,1));for(var e=Array.prototype.slice.call(arguments,2),f=!1,g=a.constructor;g;g=g.superClass_&&g.superClass_.constructor)if(g.prototype[b]===d)f=!0;else if(f)return g.prototype[b].apply(a,
|
|
|
e);if(a[b]===d)return a.constructor.prototype[b].apply(a,e);throw Error("goog.base called from a method of one name to a method of a different name");};goog.scope=function(a){a.call(goog.global)};COMPILED||(goog.global.COMPILED=COMPILED);
|
|
|
goog.defineClass=function(a,b){var c=b.constructor,d=b.statics;c&&c!=Object.prototype.constructor||(c=function(){throw Error("cannot instantiate an interface (no constructor defined).");});c=goog.defineClass.createSealingConstructor_(c,a);a&&goog.inherits(c,a);delete b.constructor;delete b.statics;goog.defineClass.applyProperties_(c.prototype,b);null!=d&&(d instanceof Function?d(c):goog.defineClass.applyProperties_(c,d));return c};goog.defineClass.SEAL_CLASS_INSTANCES=goog.DEBUG;
|
|
|
goog.defineClass.createSealingConstructor_=function(a,b){if(goog.defineClass.SEAL_CLASS_INSTANCES&&Object.seal instanceof Function){if(b&&b.prototype&&b.prototype[goog.UNSEALABLE_CONSTRUCTOR_PROPERTY_])return a;var c=function(){var b=a.apply(this,arguments)||this;b[goog.UID_PROPERTY_]=b[goog.UID_PROPERTY_];this.constructor===c&&Object.seal(b);return b};return c}return a};goog.defineClass.OBJECT_PROTOTYPE_FIELDS_="constructor hasOwnProperty isPrototypeOf propertyIsEnumerable toLocaleString toString valueOf".split(" ");
|
|
|
goog.defineClass.applyProperties_=function(a,b){for(var c in b)Object.prototype.hasOwnProperty.call(b,c)&&(a[c]=b[c]);for(var d=0;d<goog.defineClass.OBJECT_PROTOTYPE_FIELDS_.length;d++)c=goog.defineClass.OBJECT_PROTOTYPE_FIELDS_[d],Object.prototype.hasOwnProperty.call(b,c)&&(a[c]=b[c])};goog.tagUnsealableClass=function(a){!COMPILED&&goog.defineClass.SEAL_CLASS_INSTANCES&&(a.prototype[goog.UNSEALABLE_CONSTRUCTOR_PROPERTY_]=!0)};goog.UNSEALABLE_CONSTRUCTOR_PROPERTY_="goog_defineClass_legacy_unsealable";goog.debug={};goog.debug.Error=function(a){if(Error.captureStackTrace)Error.captureStackTrace(this,goog.debug.Error);else{var b=Error().stack;b&&(this.stack=b)}a&&(this.message=String(a))};goog.inherits(goog.debug.Error,Error);goog.debug.Error.prototype.name="CustomError";goog.dom={};goog.dom.NodeType={ELEMENT:1,ATTRIBUTE:2,TEXT:3,CDATA_SECTION:4,ENTITY_REFERENCE:5,ENTITY:6,PROCESSING_INSTRUCTION:7,COMMENT:8,DOCUMENT:9,DOCUMENT_TYPE:10,DOCUMENT_FRAGMENT:11,NOTATION:12};goog.string={};goog.string.DETECT_DOUBLE_ESCAPING=!1;goog.string.Unicode={NBSP:"\u00a0"};goog.string.startsWith=function(a,b){return 0==a.lastIndexOf(b,0)};goog.string.endsWith=function(a,b){var c=a.length-b.length;return 0<=c&&a.indexOf(b,c)==c};goog.string.caseInsensitiveStartsWith=function(a,b){return 0==goog.string.caseInsensitiveCompare(b,a.substr(0,b.length))};goog.string.caseInsensitiveEndsWith=function(a,b){return 0==goog.string.caseInsensitiveCompare(b,a.substr(a.length-b.length,b.length))};
|
|
|
goog.string.caseInsensitiveEquals=function(a,b){return a.toLowerCase()==b.toLowerCase()};goog.string.subs=function(a,b){for(var c=a.split("%s"),d="",e=Array.prototype.slice.call(arguments,1);e.length&&1<c.length;)d+=c.shift()+e.shift();return d+c.join("%s")};goog.string.collapseWhitespace=function(a){return a.replace(/[\s\xa0]+/g," ").replace(/^\s+|\s+$/g,"")};goog.string.isEmptyOrWhitespace=function(a){return/^[\s\xa0]*$/.test(a)};goog.string.isEmptyString=function(a){return 0==a.length};
|
|
|
goog.string.isEmpty=goog.string.isEmptyOrWhitespace;goog.string.isEmptyOrWhitespaceSafe=function(a){return goog.string.isEmptyOrWhitespace(goog.string.makeSafe(a))};goog.string.isEmptySafe=goog.string.isEmptyOrWhitespaceSafe;goog.string.isBreakingWhitespace=function(a){return!/[^\t\n\r ]/.test(a)};goog.string.isAlpha=function(a){return!/[^a-zA-Z]/.test(a)};goog.string.isNumeric=function(a){return!/[^0-9]/.test(a)};goog.string.isAlphaNumeric=function(a){return!/[^a-zA-Z0-9]/.test(a)};
|
|
|
goog.string.isSpace=function(a){return" "==a};goog.string.isUnicodeChar=function(a){return 1==a.length&&" "<=a&&"~">=a||"\u0080"<=a&&"\ufffd">=a};goog.string.stripNewlines=function(a){return a.replace(/(\r\n|\r|\n)+/g," ")};goog.string.canonicalizeNewlines=function(a){return a.replace(/(\r\n|\r|\n)/g,"\n")};goog.string.normalizeWhitespace=function(a){return a.replace(/\xa0|\s/g," ")};goog.string.normalizeSpaces=function(a){return a.replace(/\xa0|[ \t]+/g," ")};
|
|
|
goog.string.collapseBreakingSpaces=function(a){return a.replace(/[\t\r\n ]+/g," ").replace(/^[\t\r\n ]+|[\t\r\n ]+$/g,"")};goog.string.trim=goog.TRUSTED_SITE&&String.prototype.trim?function(a){return a.trim()}:function(a){return a.replace(/^[\s\xa0]+|[\s\xa0]+$/g,"")};goog.string.trimLeft=function(a){return a.replace(/^[\s\xa0]+/,"")};goog.string.trimRight=function(a){return a.replace(/[\s\xa0]+$/,"")};
|
|
|
goog.string.caseInsensitiveCompare=function(a,b){var c=String(a).toLowerCase(),d=String(b).toLowerCase();return c<d?-1:c==d?0:1};goog.string.numerateCompareRegExp_=/(\.\d+)|(\d+)|(\D+)/g;
|
|
|
goog.string.numerateCompare=function(a,b){if(a==b)return 0;if(!a)return-1;if(!b)return 1;for(var c=a.toLowerCase().match(goog.string.numerateCompareRegExp_),d=b.toLowerCase().match(goog.string.numerateCompareRegExp_),e=Math.min(c.length,d.length),f=0;f<e;f++){var g=c[f],h=d[f];if(g!=h)return c=parseInt(g,10),!isNaN(c)&&(d=parseInt(h,10),!isNaN(d)&&c-d)?c-d:g<h?-1:1}return c.length!=d.length?c.length-d.length:a<b?-1:1};goog.string.urlEncode=function(a){return encodeURIComponent(String(a))};
|
|
|
goog.string.urlDecode=function(a){return decodeURIComponent(a.replace(/\+/g," "))};goog.string.newLineToBr=function(a,b){return a.replace(/(\r\n|\r|\n)/g,b?"<br />":"<br>")};
|
|
|
goog.string.htmlEscape=function(a,b){if(b)a=a.replace(goog.string.AMP_RE_,"&").replace(goog.string.LT_RE_,"<").replace(goog.string.GT_RE_,">").replace(goog.string.QUOT_RE_,""").replace(goog.string.SINGLE_QUOTE_RE_,"'").replace(goog.string.NULL_RE_,"�"),goog.string.DETECT_DOUBLE_ESCAPING&&(a=a.replace(goog.string.E_RE_,"e"));else{if(!goog.string.ALL_RE_.test(a))return a;-1!=a.indexOf("&")&&(a=a.replace(goog.string.AMP_RE_,"&"));-1!=a.indexOf("<")&&(a=a.replace(goog.string.LT_RE_,
|
|
|
"<"));-1!=a.indexOf(">")&&(a=a.replace(goog.string.GT_RE_,">"));-1!=a.indexOf('"')&&(a=a.replace(goog.string.QUOT_RE_,"""));-1!=a.indexOf("'")&&(a=a.replace(goog.string.SINGLE_QUOTE_RE_,"'"));-1!=a.indexOf("\x00")&&(a=a.replace(goog.string.NULL_RE_,"�"));goog.string.DETECT_DOUBLE_ESCAPING&&-1!=a.indexOf("e")&&(a=a.replace(goog.string.E_RE_,"e"))}return a};goog.string.AMP_RE_=/&/g;goog.string.LT_RE_=/</g;goog.string.GT_RE_=/>/g;goog.string.QUOT_RE_=/"/g;
|
|
|
goog.string.SINGLE_QUOTE_RE_=/'/g;goog.string.NULL_RE_=/\x00/g;goog.string.E_RE_=/e/g;goog.string.ALL_RE_=goog.string.DETECT_DOUBLE_ESCAPING?/[\x00&<>"'e]/:/[\x00&<>"']/;goog.string.unescapeEntities=function(a){return goog.string.contains(a,"&")?"document"in goog.global?goog.string.unescapeEntitiesUsingDom_(a):goog.string.unescapePureXmlEntities_(a):a};goog.string.unescapeEntitiesWithDocument=function(a,b){return goog.string.contains(a,"&")?goog.string.unescapeEntitiesUsingDom_(a,b):a};
|
|
|
goog.string.unescapeEntitiesUsingDom_=function(a,b){var c={"&":"&","<":"<",">":">",""":'"'},d;d=b?b.createElement("div"):goog.global.document.createElement("div");return a.replace(goog.string.HTML_ENTITY_PATTERN_,function(a,b){var g=c[a];if(g)return g;if("#"==b.charAt(0)){var h=Number("0"+b.substr(1));isNaN(h)||(g=String.fromCharCode(h))}g||(d.innerHTML=a+" ",g=d.firstChild.nodeValue.slice(0,-1));return c[a]=g})};
|
|
|
goog.string.unescapePureXmlEntities_=function(a){return a.replace(/&([^;]+);/g,function(a,c){switch(c){case "amp":return"&";case "lt":return"<";case "gt":return">";case "quot":return'"';default:if("#"==c.charAt(0)){var d=Number("0"+c.substr(1));if(!isNaN(d))return String.fromCharCode(d)}return a}})};goog.string.HTML_ENTITY_PATTERN_=/&([^;\s<&]+);?/g;goog.string.whitespaceEscape=function(a,b){return goog.string.newLineToBr(a.replace(/ /g,"  "),b)};
|
|
|
goog.string.preserveSpaces=function(a){return a.replace(/(^|[\n ]) /g,"$1"+goog.string.Unicode.NBSP)};goog.string.stripQuotes=function(a,b){for(var c=b.length,d=0;d<c;d++){var e=1==c?b:b.charAt(d);if(a.charAt(0)==e&&a.charAt(a.length-1)==e)return a.substring(1,a.length-1)}return a};goog.string.truncate=function(a,b,c){c&&(a=goog.string.unescapeEntities(a));a.length>b&&(a=a.substring(0,b-3)+"...");c&&(a=goog.string.htmlEscape(a));return a};
|
|
|
goog.string.truncateMiddle=function(a,b,c,d){c&&(a=goog.string.unescapeEntities(a));if(d&&a.length>b){d>b&&(d=b);var e=a.length-d;a=a.substring(0,b-d)+"..."+a.substring(e)}else a.length>b&&(d=Math.floor(b/2),e=a.length-d,a=a.substring(0,d+b%2)+"..."+a.substring(e));c&&(a=goog.string.htmlEscape(a));return a};goog.string.specialEscapeChars_={"\x00":"\\0","\b":"\\b","\f":"\\f","\n":"\\n","\r":"\\r","\t":"\\t","\x0B":"\\x0B",'"':'\\"',"\\":"\\\\"};goog.string.jsEscapeCache_={"'":"\\'"};
|
|
|
goog.string.quote=function(a){a=String(a);if(a.quote)return a.quote();for(var b=['"'],c=0;c<a.length;c++){var d=a.charAt(c),e=d.charCodeAt(0);b[c+1]=goog.string.specialEscapeChars_[d]||(31<e&&127>e?d:goog.string.escapeChar(d))}b.push('"');return b.join("")};goog.string.escapeString=function(a){for(var b=[],c=0;c<a.length;c++)b[c]=goog.string.escapeChar(a.charAt(c));return b.join("")};
|
|
|
goog.string.escapeChar=function(a){if(a in goog.string.jsEscapeCache_)return goog.string.jsEscapeCache_[a];if(a in goog.string.specialEscapeChars_)return goog.string.jsEscapeCache_[a]=goog.string.specialEscapeChars_[a];var b=a,c=a.charCodeAt(0);if(31<c&&127>c)b=a;else{if(256>c){if(b="\\x",16>c||256<c)b+="0"}else b="\\u",4096>c&&(b+="0");b+=c.toString(16).toUpperCase()}return goog.string.jsEscapeCache_[a]=b};goog.string.contains=function(a,b){return-1!=a.indexOf(b)};
|
|
|
goog.string.caseInsensitiveContains=function(a,b){return goog.string.contains(a.toLowerCase(),b.toLowerCase())};goog.string.countOf=function(a,b){return a&&b?a.split(b).length-1:0};goog.string.removeAt=function(a,b,c){var d=a;0<=b&&b<a.length&&0<c&&(d=a.substr(0,b)+a.substr(b+c,a.length-b-c));return d};goog.string.remove=function(a,b){var c=new RegExp(goog.string.regExpEscape(b),"");return a.replace(c,"")};
|
|
|
goog.string.removeAll=function(a,b){var c=new RegExp(goog.string.regExpEscape(b),"g");return a.replace(c,"")};goog.string.regExpEscape=function(a){return String(a).replace(/([-()\[\]{}+?*.$\^|,:#<!\\])/g,"\\$1").replace(/\x08/g,"\\x08")};goog.string.repeat=function(a,b){return Array(b+1).join(a)};goog.string.padNumber=function(a,b,c){a=goog.isDef(c)?a.toFixed(c):String(a);c=a.indexOf(".");-1==c&&(c=a.length);return goog.string.repeat("0",Math.max(0,b-c))+a};
|
|
|
goog.string.makeSafe=function(a){return null==a?"":String(a)};goog.string.buildString=function(a){return Array.prototype.join.call(arguments,"")};goog.string.getRandomString=function(){return Math.floor(2147483648*Math.random()).toString(36)+Math.abs(Math.floor(2147483648*Math.random())^goog.now()).toString(36)};
|
|
|
goog.string.compareVersions=function(a,b){for(var c=0,d=goog.string.trim(String(a)).split("."),e=goog.string.trim(String(b)).split("."),f=Math.max(d.length,e.length),g=0;0==c&&g<f;g++){var h=d[g]||"",k=e[g]||"",l=RegExp("(\\d*)(\\D*)","g"),p=RegExp("(\\d*)(\\D*)","g");do{var m=l.exec(h)||["","",""],n=p.exec(k)||["","",""];if(0==m[0].length&&0==n[0].length)break;var c=0==m[1].length?0:parseInt(m[1],10),q=0==n[1].length?0:parseInt(n[1],10),c=goog.string.compareElements_(c,q)||goog.string.compareElements_(0==
|
|
|
m[2].length,0==n[2].length)||goog.string.compareElements_(m[2],n[2])}while(0==c)}return c};goog.string.compareElements_=function(a,b){return a<b?-1:a>b?1:0};goog.string.HASHCODE_MAX_=4294967296;goog.string.hashCode=function(a){for(var b=0,c=0;c<a.length;++c)b=31*b+a.charCodeAt(c),b%=goog.string.HASHCODE_MAX_;return b};goog.string.uniqueStringCounter_=2147483648*Math.random()|0;goog.string.createUniqueString=function(){return"goog_"+goog.string.uniqueStringCounter_++};
|
|
|
goog.string.toNumber=function(a){var b=Number(a);return 0==b&&goog.string.isEmpty(a)?NaN:b};goog.string.isLowerCamelCase=function(a){return/^[a-z]+([A-Z][a-z]*)*$/.test(a)};goog.string.isUpperCamelCase=function(a){return/^([A-Z][a-z]*)+$/.test(a)};goog.string.toCamelCase=function(a){return String(a).replace(/\-([a-z])/g,function(a,c){return c.toUpperCase()})};goog.string.toSelectorCase=function(a){return String(a).replace(/([A-Z])/g,"-$1").toLowerCase()};
|
|
|
goog.string.toTitleCase=function(a,b){var c=goog.isString(b)?goog.string.regExpEscape(b):"\\s";return a.replace(new RegExp("(^"+(c?"|["+c+"]+":"")+")([a-z])","g"),function(a,b,c){return b+c.toUpperCase()})};goog.string.parseInt=function(a){isFinite(a)&&(a=String(a));return goog.isString(a)?/^\s*-?0x/i.test(a)?parseInt(a,16):parseInt(a,10):NaN};goog.string.splitLimit=function(a,b,c){a=a.split(b);for(var d=[];0<c&&a.length;)d.push(a.shift()),c--;a.length&&d.push(a.join(b));return d};goog.asserts={};goog.asserts.ENABLE_ASSERTS=goog.DEBUG;goog.asserts.AssertionError=function(a,b){b.unshift(a);goog.debug.Error.call(this,goog.string.subs.apply(null,b));b.shift();this.messagePattern=a};goog.inherits(goog.asserts.AssertionError,goog.debug.Error);goog.asserts.AssertionError.prototype.name="AssertionError";goog.asserts.DEFAULT_ERROR_HANDLER=function(a){throw a;};goog.asserts.errorHandler_=goog.asserts.DEFAULT_ERROR_HANDLER;
|
|
|
goog.asserts.doAssertFailure_=function(a,b,c,d){var e="Assertion failed";if(c)var e=e+(": "+c),f=d;else a&&(e+=": "+a,f=b);a=new goog.asserts.AssertionError(""+e,f||[]);goog.asserts.errorHandler_(a)};goog.asserts.setErrorHandler=function(a){goog.asserts.ENABLE_ASSERTS&&(goog.asserts.errorHandler_=a)};goog.asserts.assert=function(a,b,c){goog.asserts.ENABLE_ASSERTS&&!a&&goog.asserts.doAssertFailure_("",null,b,Array.prototype.slice.call(arguments,2));return a};
|
|
|
goog.asserts.fail=function(a,b){goog.asserts.ENABLE_ASSERTS&&goog.asserts.errorHandler_(new goog.asserts.AssertionError("Failure"+(a?": "+a:""),Array.prototype.slice.call(arguments,1)))};goog.asserts.assertNumber=function(a,b,c){goog.asserts.ENABLE_ASSERTS&&!goog.isNumber(a)&&goog.asserts.doAssertFailure_("Expected number but got %s: %s.",[goog.typeOf(a),a],b,Array.prototype.slice.call(arguments,2));return a};
|
|
|
goog.asserts.assertString=function(a,b,c){goog.asserts.ENABLE_ASSERTS&&!goog.isString(a)&&goog.asserts.doAssertFailure_("Expected string but got %s: %s.",[goog.typeOf(a),a],b,Array.prototype.slice.call(arguments,2));return a};goog.asserts.assertFunction=function(a,b,c){goog.asserts.ENABLE_ASSERTS&&!goog.isFunction(a)&&goog.asserts.doAssertFailure_("Expected function but got %s: %s.",[goog.typeOf(a),a],b,Array.prototype.slice.call(arguments,2));return a};
|
|
|
goog.asserts.assertObject=function(a,b,c){goog.asserts.ENABLE_ASSERTS&&!goog.isObject(a)&&goog.asserts.doAssertFailure_("Expected object but got %s: %s.",[goog.typeOf(a),a],b,Array.prototype.slice.call(arguments,2));return a};goog.asserts.assertArray=function(a,b,c){goog.asserts.ENABLE_ASSERTS&&!goog.isArray(a)&&goog.asserts.doAssertFailure_("Expected array but got %s: %s.",[goog.typeOf(a),a],b,Array.prototype.slice.call(arguments,2));return a};
|
|
|
goog.asserts.assertBoolean=function(a,b,c){goog.asserts.ENABLE_ASSERTS&&!goog.isBoolean(a)&&goog.asserts.doAssertFailure_("Expected boolean but got %s: %s.",[goog.typeOf(a),a],b,Array.prototype.slice.call(arguments,2));return a};goog.asserts.assertElement=function(a,b,c){!goog.asserts.ENABLE_ASSERTS||goog.isObject(a)&&a.nodeType==goog.dom.NodeType.ELEMENT||goog.asserts.doAssertFailure_("Expected Element but got %s: %s.",[goog.typeOf(a),a],b,Array.prototype.slice.call(arguments,2));return a};
|
|
|
goog.asserts.assertInstanceof=function(a,b,c,d){!goog.asserts.ENABLE_ASSERTS||a instanceof b||goog.asserts.doAssertFailure_("Expected instanceof %s but got %s.",[goog.asserts.getType_(b),goog.asserts.getType_(a)],c,Array.prototype.slice.call(arguments,3));return a};goog.asserts.assertObjectPrototypeIsIntact=function(){for(var a in Object.prototype)goog.asserts.fail(a+" should not be enumerable in Object.prototype.")};
|
|
|
goog.asserts.getType_=function(a){return a instanceof Function?a.displayName||a.name||"unknown type name":a instanceof Object?a.constructor.displayName||a.constructor.name||Object.prototype.toString.call(a):null===a?"null":typeof a};goog.array={};goog.NATIVE_ARRAY_PROTOTYPES=goog.TRUSTED_SITE;goog.array.ASSUME_NATIVE_FUNCTIONS=!1;goog.array.peek=function(a){return a[a.length-1]};goog.array.last=goog.array.peek;goog.array.ARRAY_PROTOTYPE_=Array.prototype;
|
|
|
goog.array.indexOf=goog.NATIVE_ARRAY_PROTOTYPES&&(goog.array.ASSUME_NATIVE_FUNCTIONS||goog.array.ARRAY_PROTOTYPE_.indexOf)?function(a,b,c){goog.asserts.assert(null!=a.length);return goog.array.ARRAY_PROTOTYPE_.indexOf.call(a,b,c)}:function(a,b,c){c=null==c?0:0>c?Math.max(0,a.length+c):c;if(goog.isString(a))return goog.isString(b)&&1==b.length?a.indexOf(b,c):-1;for(;c<a.length;c++)if(c in a&&a[c]===b)return c;return-1};
|
|
|
goog.array.lastIndexOf=goog.NATIVE_ARRAY_PROTOTYPES&&(goog.array.ASSUME_NATIVE_FUNCTIONS||goog.array.ARRAY_PROTOTYPE_.lastIndexOf)?function(a,b,c){goog.asserts.assert(null!=a.length);return goog.array.ARRAY_PROTOTYPE_.lastIndexOf.call(a,b,null==c?a.length-1:c)}:function(a,b,c){c=null==c?a.length-1:c;0>c&&(c=Math.max(0,a.length+c));if(goog.isString(a))return goog.isString(b)&&1==b.length?a.lastIndexOf(b,c):-1;for(;0<=c;c--)if(c in a&&a[c]===b)return c;return-1};
|
|
|
goog.array.forEach=goog.NATIVE_ARRAY_PROTOTYPES&&(goog.array.ASSUME_NATIVE_FUNCTIONS||goog.array.ARRAY_PROTOTYPE_.forEach)?function(a,b,c){goog.asserts.assert(null!=a.length);goog.array.ARRAY_PROTOTYPE_.forEach.call(a,b,c)}:function(a,b,c){for(var d=a.length,e=goog.isString(a)?a.split(""):a,f=0;f<d;f++)f in e&&b.call(c,e[f],f,a)};goog.array.forEachRight=function(a,b,c){for(var d=a.length,e=goog.isString(a)?a.split(""):a,d=d-1;0<=d;--d)d in e&&b.call(c,e[d],d,a)};
|
|
|
goog.array.filter=goog.NATIVE_ARRAY_PROTOTYPES&&(goog.array.ASSUME_NATIVE_FUNCTIONS||goog.array.ARRAY_PROTOTYPE_.filter)?function(a,b,c){goog.asserts.assert(null!=a.length);return goog.array.ARRAY_PROTOTYPE_.filter.call(a,b,c)}:function(a,b,c){for(var d=a.length,e=[],f=0,g=goog.isString(a)?a.split(""):a,h=0;h<d;h++)if(h in g){var k=g[h];b.call(c,k,h,a)&&(e[f++]=k)}return e};
|
|
|
goog.array.map=goog.NATIVE_ARRAY_PROTOTYPES&&(goog.array.ASSUME_NATIVE_FUNCTIONS||goog.array.ARRAY_PROTOTYPE_.map)?function(a,b,c){goog.asserts.assert(null!=a.length);return goog.array.ARRAY_PROTOTYPE_.map.call(a,b,c)}:function(a,b,c){for(var d=a.length,e=Array(d),f=goog.isString(a)?a.split(""):a,g=0;g<d;g++)g in f&&(e[g]=b.call(c,f[g],g,a));return e};
|
|
|
goog.array.reduce=goog.NATIVE_ARRAY_PROTOTYPES&&(goog.array.ASSUME_NATIVE_FUNCTIONS||goog.array.ARRAY_PROTOTYPE_.reduce)?function(a,b,c,d){goog.asserts.assert(null!=a.length);d&&(b=goog.bind(b,d));return goog.array.ARRAY_PROTOTYPE_.reduce.call(a,b,c)}:function(a,b,c,d){var e=c;goog.array.forEach(a,function(c,g){e=b.call(d,e,c,g,a)});return e};
|
|
|
goog.array.reduceRight=goog.NATIVE_ARRAY_PROTOTYPES&&(goog.array.ASSUME_NATIVE_FUNCTIONS||goog.array.ARRAY_PROTOTYPE_.reduceRight)?function(a,b,c,d){goog.asserts.assert(null!=a.length);d&&(b=goog.bind(b,d));return goog.array.ARRAY_PROTOTYPE_.reduceRight.call(a,b,c)}:function(a,b,c,d){var e=c;goog.array.forEachRight(a,function(c,g){e=b.call(d,e,c,g,a)});return e};
|
|
|
goog.array.some=goog.NATIVE_ARRAY_PROTOTYPES&&(goog.array.ASSUME_NATIVE_FUNCTIONS||goog.array.ARRAY_PROTOTYPE_.some)?function(a,b,c){goog.asserts.assert(null!=a.length);return goog.array.ARRAY_PROTOTYPE_.some.call(a,b,c)}:function(a,b,c){for(var d=a.length,e=goog.isString(a)?a.split(""):a,f=0;f<d;f++)if(f in e&&b.call(c,e[f],f,a))return!0;return!1};
|
|
|
goog.array.every=goog.NATIVE_ARRAY_PROTOTYPES&&(goog.array.ASSUME_NATIVE_FUNCTIONS||goog.array.ARRAY_PROTOTYPE_.every)?function(a,b,c){goog.asserts.assert(null!=a.length);return goog.array.ARRAY_PROTOTYPE_.every.call(a,b,c)}:function(a,b,c){for(var d=a.length,e=goog.isString(a)?a.split(""):a,f=0;f<d;f++)if(f in e&&!b.call(c,e[f],f,a))return!1;return!0};goog.array.count=function(a,b,c){var d=0;goog.array.forEach(a,function(a,f,g){b.call(c,a,f,g)&&++d},c);return d};
|
|
|
goog.array.find=function(a,b,c){b=goog.array.findIndex(a,b,c);return 0>b?null:goog.isString(a)?a.charAt(b):a[b]};goog.array.findIndex=function(a,b,c){for(var d=a.length,e=goog.isString(a)?a.split(""):a,f=0;f<d;f++)if(f in e&&b.call(c,e[f],f,a))return f;return-1};goog.array.findRight=function(a,b,c){b=goog.array.findIndexRight(a,b,c);return 0>b?null:goog.isString(a)?a.charAt(b):a[b]};
|
|
|
goog.array.findIndexRight=function(a,b,c){for(var d=a.length,e=goog.isString(a)?a.split(""):a,d=d-1;0<=d;d--)if(d in e&&b.call(c,e[d],d,a))return d;return-1};goog.array.contains=function(a,b){return 0<=goog.array.indexOf(a,b)};goog.array.isEmpty=function(a){return 0==a.length};goog.array.clear=function(a){if(!goog.isArray(a))for(var b=a.length-1;0<=b;b--)delete a[b];a.length=0};goog.array.insert=function(a,b){goog.array.contains(a,b)||a.push(b)};
|
|
|
goog.array.insertAt=function(a,b,c){goog.array.splice(a,c,0,b)};goog.array.insertArrayAt=function(a,b,c){goog.partial(goog.array.splice,a,c,0).apply(null,b)};goog.array.insertBefore=function(a,b,c){var d;2==arguments.length||0>(d=goog.array.indexOf(a,c))?a.push(b):goog.array.insertAt(a,b,d)};goog.array.remove=function(a,b){var c=goog.array.indexOf(a,b),d;(d=0<=c)&&goog.array.removeAt(a,c);return d};
|
|
|
goog.array.removeAt=function(a,b){goog.asserts.assert(null!=a.length);return 1==goog.array.ARRAY_PROTOTYPE_.splice.call(a,b,1).length};goog.array.removeIf=function(a,b,c){b=goog.array.findIndex(a,b,c);return 0<=b?(goog.array.removeAt(a,b),!0):!1};goog.array.removeAllIf=function(a,b,c){var d=0;goog.array.forEachRight(a,function(e,f){b.call(c,e,f,a)&&goog.array.removeAt(a,f)&&d++});return d};goog.array.concat=function(a){return goog.array.ARRAY_PROTOTYPE_.concat.apply(goog.array.ARRAY_PROTOTYPE_,arguments)};
|
|
|
goog.array.join=function(a){return goog.array.ARRAY_PROTOTYPE_.concat.apply(goog.array.ARRAY_PROTOTYPE_,arguments)};goog.array.toArray=function(a){var b=a.length;if(0<b){for(var c=Array(b),d=0;d<b;d++)c[d]=a[d];return c}return[]};goog.array.clone=goog.array.toArray;
|
|
|
goog.array.extend=function(a,b){for(var c=1;c<arguments.length;c++){var d=arguments[c],e;if(goog.isArray(d)||(e=goog.isArrayLike(d))&&Object.prototype.hasOwnProperty.call(d,"callee"))a.push.apply(a,d);else if(e)for(var f=a.length,g=d.length,h=0;h<g;h++)a[f+h]=d[h];else a.push(d)}};goog.array.splice=function(a,b,c,d){goog.asserts.assert(null!=a.length);return goog.array.ARRAY_PROTOTYPE_.splice.apply(a,goog.array.slice(arguments,1))};
|
|
|
goog.array.slice=function(a,b,c){goog.asserts.assert(null!=a.length);return 2>=arguments.length?goog.array.ARRAY_PROTOTYPE_.slice.call(a,b):goog.array.ARRAY_PROTOTYPE_.slice.call(a,b,c)};goog.array.removeDuplicates=function(a,b,c){b=b||a;var d=function(a){return goog.isObject(g)?"o"+goog.getUid(g):(typeof g).charAt(0)+g};c=c||d;for(var d={},e=0,f=0;f<a.length;){var g=a[f++],h=c(g);Object.prototype.hasOwnProperty.call(d,h)||(d[h]=!0,b[e++]=g)}b.length=e};
|
|
|
goog.array.binarySearch=function(a,b,c){return goog.array.binarySearch_(a,c||goog.array.defaultCompare,!1,b)};goog.array.binarySelect=function(a,b,c){return goog.array.binarySearch_(a,b,!0,void 0,c)};goog.array.binarySearch_=function(a,b,c,d,e){for(var f=0,g=a.length,h;f<g;){var k=f+g>>1,l;l=c?b.call(e,a[k],k,a):b(d,a[k]);0<l?f=k+1:(g=k,h=!l)}return h?f:~f};goog.array.sort=function(a,b){a.sort(b||goog.array.defaultCompare)};
|
|
|
goog.array.stableSort=function(a,b){for(var c=0;c<a.length;c++)a[c]={index:c,value:a[c]};var d=b||goog.array.defaultCompare;goog.array.sort(a,function(a,b){return d(a.value,b.value)||a.index-b.index});for(c=0;c<a.length;c++)a[c]=a[c].value};goog.array.sortByKey=function(a,b,c){var d=c||goog.array.defaultCompare;goog.array.sort(a,function(a,c){return d(b(a),b(c))})};goog.array.sortObjectsByKey=function(a,b,c){goog.array.sortByKey(a,function(a){return a[b]},c)};
|
|
|
goog.array.isSorted=function(a,b,c){b=b||goog.array.defaultCompare;for(var d=1;d<a.length;d++){var e=b(a[d-1],a[d]);if(0<e||0==e&&c)return!1}return!0};goog.array.equals=function(a,b,c){if(!goog.isArrayLike(a)||!goog.isArrayLike(b)||a.length!=b.length)return!1;var d=a.length;c=c||goog.array.defaultCompareEquality;for(var e=0;e<d;e++)if(!c(a[e],b[e]))return!1;return!0};
|
|
|
goog.array.compare3=function(a,b,c){c=c||goog.array.defaultCompare;for(var d=Math.min(a.length,b.length),e=0;e<d;e++){var f=c(a[e],b[e]);if(0!=f)return f}return goog.array.defaultCompare(a.length,b.length)};goog.array.defaultCompare=function(a,b){return a>b?1:a<b?-1:0};goog.array.defaultCompareEquality=function(a,b){return a===b};goog.array.binaryInsert=function(a,b,c){c=goog.array.binarySearch(a,b,c);return 0>c?(goog.array.insertAt(a,b,-(c+1)),!0):!1};
|
|
|
goog.array.binaryRemove=function(a,b,c){b=goog.array.binarySearch(a,b,c);return 0<=b?goog.array.removeAt(a,b):!1};goog.array.bucket=function(a,b,c){for(var d={},e=0;e<a.length;e++){var f=a[e],g=b.call(c,f,e,a);goog.isDef(g)&&(d[g]||(d[g]=[])).push(f)}return d};goog.array.toObject=function(a,b,c){var d={};goog.array.forEach(a,function(e,f){d[b.call(c,e,f,a)]=e});return d};
|
|
|
goog.array.range=function(a,b,c){var d=[],e=0,f=a;c=c||1;void 0!==b&&(e=a,f=b);if(0>c*(f-e))return[];if(0<c)for(a=e;a<f;a+=c)d.push(a);else for(a=e;a>f;a+=c)d.push(a);return d};goog.array.repeat=function(a,b){for(var c=[],d=0;d<b;d++)c[d]=a;return c};goog.array.flatten=function(a){for(var b=[],c=0;c<arguments.length;c++){var d=arguments[c];if(goog.isArray(d))for(var e=0;e<d.length;e+=8192)for(var f=goog.array.slice(d,e,e+8192),f=goog.array.flatten.apply(null,f),g=0;g<f.length;g++)b.push(f[g]);else b.push(d)}return b};
|
|
|
goog.array.rotate=function(a,b){goog.asserts.assert(null!=a.length);a.length&&(b%=a.length,0<b?goog.array.ARRAY_PROTOTYPE_.unshift.apply(a,a.splice(-b,b)):0>b&&goog.array.ARRAY_PROTOTYPE_.push.apply(a,a.splice(0,-b)));return a};goog.array.moveItem=function(a,b,c){goog.asserts.assert(0<=b&&b<a.length);goog.asserts.assert(0<=c&&c<a.length);b=goog.array.ARRAY_PROTOTYPE_.splice.call(a,b,1);goog.array.ARRAY_PROTOTYPE_.splice.call(a,c,0,b[0])};
|
|
|
goog.array.zip=function(a){if(!arguments.length)return[];for(var b=[],c=0;;c++){for(var d=[],e=0;e<arguments.length;e++){var f=arguments[e];if(c>=f.length)return b;d.push(f[c])}b.push(d)}};goog.array.shuffle=function(a,b){for(var c=b||Math.random,d=a.length-1;0<d;d--){var e=Math.floor(c()*(d+1)),f=a[d];a[d]=a[e];a[e]=f}};goog.proto2={};
|
|
|
goog.proto2.FieldDescriptor=function(a,b,c){this.parent_=a;goog.asserts.assert(goog.string.isNumeric(b));this.tag_=b;this.name_=c.name;this.isRepeated_=!!c.repeated;this.isRequired_=!!c.required;this.fieldType_=c.fieldType;this.nativeType_=c.type;this.deserializationConversionPermitted_=!1;switch(this.fieldType_){case goog.proto2.FieldDescriptor.FieldType.INT64:case goog.proto2.FieldDescriptor.FieldType.UINT64:case goog.proto2.FieldDescriptor.FieldType.FIXED64:case goog.proto2.FieldDescriptor.FieldType.SFIXED64:case goog.proto2.FieldDescriptor.FieldType.SINT64:case goog.proto2.FieldDescriptor.FieldType.FLOAT:case goog.proto2.FieldDescriptor.FieldType.DOUBLE:this.deserializationConversionPermitted_=!0}this.defaultValue_=
|
|
|
c.defaultValue};goog.proto2.FieldDescriptor.FieldType={DOUBLE:1,FLOAT:2,INT64:3,UINT64:4,INT32:5,FIXED64:6,FIXED32:7,BOOL:8,STRING:9,GROUP:10,MESSAGE:11,BYTES:12,UINT32:13,ENUM:14,SFIXED32:15,SFIXED64:16,SINT32:17,SINT64:18};goog.proto2.FieldDescriptor.prototype.getTag=function(){return this.tag_};goog.proto2.FieldDescriptor.prototype.getContainingType=function(){return this.parent_.getDescriptor()};goog.proto2.FieldDescriptor.prototype.getName=function(){return this.name_};
|
|
|
goog.proto2.FieldDescriptor.prototype.getDefaultValue=function(){if(void 0===this.defaultValue_){var a=this.nativeType_;if(a===Boolean)this.defaultValue_=!1;else if(a===Number)this.defaultValue_=0;else if(a===String)this.defaultValue_=this.deserializationConversionPermitted_?"0":"";else return new a}return this.defaultValue_};goog.proto2.FieldDescriptor.prototype.getFieldType=function(){return this.fieldType_};goog.proto2.FieldDescriptor.prototype.getNativeType=function(){return this.nativeType_};
|
|
|
goog.proto2.FieldDescriptor.prototype.deserializationConversionPermitted=function(){return this.deserializationConversionPermitted_};goog.proto2.FieldDescriptor.prototype.getFieldMessageType=function(){return this.nativeType_.getDescriptor()};goog.proto2.FieldDescriptor.prototype.isCompositeType=function(){return this.fieldType_==goog.proto2.FieldDescriptor.FieldType.MESSAGE||this.fieldType_==goog.proto2.FieldDescriptor.FieldType.GROUP};goog.proto2.FieldDescriptor.prototype.isRepeated=function(){return this.isRepeated_};
|
|
|
goog.proto2.FieldDescriptor.prototype.isRequired=function(){return this.isRequired_};goog.proto2.FieldDescriptor.prototype.isOptional=function(){return!this.isRepeated_&&!this.isRequired_};goog.object={};goog.object.forEach=function(a,b,c){for(var d in a)b.call(c,a[d],d,a)};goog.object.filter=function(a,b,c){var d={},e;for(e in a)b.call(c,a[e],e,a)&&(d[e]=a[e]);return d};goog.object.map=function(a,b,c){var d={},e;for(e in a)d[e]=b.call(c,a[e],e,a);return d};goog.object.some=function(a,b,c){for(var d in a)if(b.call(c,a[d],d,a))return!0;return!1};goog.object.every=function(a,b,c){for(var d in a)if(!b.call(c,a[d],d,a))return!1;return!0};
|
|
|
goog.object.getCount=function(a){var b=0,c;for(c in a)b++;return b};goog.object.getAnyKey=function(a){for(var b in a)return b};goog.object.getAnyValue=function(a){for(var b in a)return a[b]};goog.object.contains=function(a,b){return goog.object.containsValue(a,b)};goog.object.getValues=function(a){var b=[],c=0,d;for(d in a)b[c++]=a[d];return b};goog.object.getKeys=function(a){var b=[],c=0,d;for(d in a)b[c++]=d;return b};
|
|
|
goog.object.getValueByKeys=function(a,b){for(var c=goog.isArrayLike(b),d=c?b:arguments,c=c?0:1;c<d.length&&(a=a[d[c]],goog.isDef(a));c++);return a};goog.object.containsKey=function(a,b){return b in a};goog.object.containsValue=function(a,b){for(var c in a)if(a[c]==b)return!0;return!1};goog.object.findKey=function(a,b,c){for(var d in a)if(b.call(c,a[d],d,a))return d};goog.object.findValue=function(a,b,c){return(b=goog.object.findKey(a,b,c))&&a[b]};
|
|
|
goog.object.isEmpty=function(a){for(var b in a)return!1;return!0};goog.object.clear=function(a){for(var b in a)delete a[b]};goog.object.remove=function(a,b){var c;(c=b in a)&&delete a[b];return c};goog.object.add=function(a,b,c){if(b in a)throw Error('The object already contains the key "'+b+'"');goog.object.set(a,b,c)};goog.object.get=function(a,b,c){return b in a?a[b]:c};goog.object.set=function(a,b,c){a[b]=c};goog.object.setIfUndefined=function(a,b,c){return b in a?a[b]:a[b]=c};
|
|
|
goog.object.equals=function(a,b){for(var c in a)if(!(c in b)||a[c]!==b[c])return!1;for(c in b)if(!(c in a))return!1;return!0};goog.object.clone=function(a){var b={},c;for(c in a)b[c]=a[c];return b};goog.object.unsafeClone=function(a){var b=goog.typeOf(a);if("object"==b||"array"==b){if(a.clone)return a.clone();var b="array"==b?[]:{},c;for(c in a)b[c]=goog.object.unsafeClone(a[c]);return b}return a};goog.object.transpose=function(a){var b={},c;for(c in a)b[a[c]]=c;return b};
|
|
|
goog.object.PROTOTYPE_FIELDS_="constructor hasOwnProperty isPrototypeOf propertyIsEnumerable toLocaleString toString valueOf".split(" ");goog.object.extend=function(a,b){for(var c,d,e=1;e<arguments.length;e++){d=arguments[e];for(c in d)a[c]=d[c];for(var f=0;f<goog.object.PROTOTYPE_FIELDS_.length;f++)c=goog.object.PROTOTYPE_FIELDS_[f],Object.prototype.hasOwnProperty.call(d,c)&&(a[c]=d[c])}};
|
|
|
goog.object.create=function(a){var b=arguments.length;if(1==b&&goog.isArray(arguments[0]))return goog.object.create.apply(null,arguments[0]);if(b%2)throw Error("Uneven number of arguments");for(var c={},d=0;d<b;d+=2)c[arguments[d]]=arguments[d+1];return c};goog.object.createSet=function(a){var b=arguments.length;if(1==b&&goog.isArray(arguments[0]))return goog.object.createSet.apply(null,arguments[0]);for(var c={},d=0;d<b;d++)c[arguments[d]]=!0;return c};
|
|
|
goog.object.createImmutableView=function(a){var b=a;Object.isFrozen&&!Object.isFrozen(a)&&(b=Object.create(a),Object.freeze(b));return b};goog.object.isImmutableView=function(a){return!!Object.isFrozen&&Object.isFrozen(a)};goog.proto2.Descriptor=function(a,b,c){this.messageType_=a;this.name_=b.name||null;this.fullName_=b.fullName||null;this.containingType_=b.containingType;this.fields_={};for(a=0;a<c.length;a++)b=c[a],this.fields_[b.getTag()]=b};goog.proto2.Descriptor.prototype.getName=function(){return this.name_};goog.proto2.Descriptor.prototype.getFullName=function(){return this.fullName_};
|
|
|
goog.proto2.Descriptor.prototype.getContainingType=function(){return this.containingType_?this.containingType_.getDescriptor():null};goog.proto2.Descriptor.prototype.getFields=function(){var a=goog.object.getValues(this.fields_);goog.array.sort(a,function(a,c){return a.getTag()-c.getTag()});return a};goog.proto2.Descriptor.prototype.getFieldsMap=function(){return this.fields_};
|
|
|
goog.proto2.Descriptor.prototype.findFieldByName=function(a){return goog.object.findValue(this.fields_,function(b,c,d){return b.getName()==a})||null};goog.proto2.Descriptor.prototype.findFieldByTag=function(a){goog.asserts.assert(goog.string.isNumeric(a));return this.fields_[parseInt(a,10)]||null};goog.proto2.Descriptor.prototype.createMessageInstance=function(){return new this.messageType_};goog.proto2.Message=function(){this.values_={};this.fields_=this.getDescriptor().getFieldsMap();this.deserializedFields_=this.lazyDeserializer_=null};goog.proto2.Message.FieldType={DOUBLE:1,FLOAT:2,INT64:3,UINT64:4,INT32:5,FIXED64:6,FIXED32:7,BOOL:8,STRING:9,GROUP:10,MESSAGE:11,BYTES:12,UINT32:13,ENUM:14,SFIXED32:15,SFIXED64:16,SINT32:17,SINT64:18};goog.proto2.Message.prototype.initializeForLazyDeserializer=function(a,b){this.lazyDeserializer_=a;this.values_=b;this.deserializedFields_={}};
|
|
|
goog.proto2.Message.prototype.setUnknown=function(a,b){goog.asserts.assert(!this.fields_[a],"Field is not unknown in this message");goog.asserts.assert(1<=a,"Tag is not valid");goog.asserts.assert(null!==b,"Value cannot be null");this.values_[a]=b;this.deserializedFields_&&delete this.deserializedFields_[a]};goog.proto2.Message.prototype.forEachUnknown=function(a,b){var c=b||this,d;for(d in this.values_){var e=Number(d);this.fields_[e]||a.call(c,e,this.values_[d])}};
|
|
|
goog.proto2.Message.prototype.getDescriptor=function(){var a=this.constructor;return a.descriptor_||(a.descriptor_=goog.proto2.Message.createDescriptor(a,a.descriptorObj_))};goog.proto2.Message.prototype.has=function(a){goog.asserts.assert(a.getContainingType()==this.getDescriptor(),"The current message does not contain the given field");return this.has$Value(a.getTag())};
|
|
|
goog.proto2.Message.prototype.arrayOf=function(a){goog.asserts.assert(a.getContainingType()==this.getDescriptor(),"The current message does not contain the given field");return this.array$Values(a.getTag())};goog.proto2.Message.prototype.countOf=function(a){goog.asserts.assert(a.getContainingType()==this.getDescriptor(),"The current message does not contain the given field");return this.count$Values(a.getTag())};
|
|
|
goog.proto2.Message.prototype.get=function(a,b){goog.asserts.assert(a.getContainingType()==this.getDescriptor(),"The current message does not contain the given field");return this.get$Value(a.getTag(),b)};goog.proto2.Message.prototype.getOrDefault=function(a,b){goog.asserts.assert(a.getContainingType()==this.getDescriptor(),"The current message does not contain the given field");return this.get$ValueOrDefault(a.getTag(),b)};
|
|
|
goog.proto2.Message.prototype.set=function(a,b){goog.asserts.assert(a.getContainingType()==this.getDescriptor(),"The current message does not contain the given field");this.set$Value(a.getTag(),b)};goog.proto2.Message.prototype.add=function(a,b){goog.asserts.assert(a.getContainingType()==this.getDescriptor(),"The current message does not contain the given field");this.add$Value(a.getTag(),b)};
|
|
|
goog.proto2.Message.prototype.clear=function(a){goog.asserts.assert(a.getContainingType()==this.getDescriptor(),"The current message does not contain the given field");this.clear$Field(a.getTag())};
|
|
|
goog.proto2.Message.prototype.equals=function(a){if(!a||this.constructor!=a.constructor)return!1;for(var b=this.getDescriptor().getFields(),c=0;c<b.length;c++){var d=b[c],e=d.getTag();if(this.has$Value(e)!=a.has$Value(e))return!1;if(this.has$Value(e)){var f=d.isCompositeType(),g=this.getValueForTag_(e),e=a.getValueForTag_(e);if(d.isRepeated()){if(g.length!=e.length)return!1;for(d=0;d<g.length;d++){var h=g[d],k=e[d];if(f?!h.equals(k):h!=k)return!1}}else if(f?!g.equals(e):g!=e)return!1}}return!0};
|
|
|
goog.proto2.Message.prototype.copyFrom=function(a){goog.asserts.assert(this.constructor==a.constructor,"The source message must have the same type.");this!=a&&(this.values_={},this.deserializedFields_&&(this.deserializedFields_={}),this.mergeFrom(a))};
|
|
|
goog.proto2.Message.prototype.mergeFrom=function(a){goog.asserts.assert(this.constructor==a.constructor,"The source message must have the same type.");for(var b=this.getDescriptor().getFields(),c=0;c<b.length;c++){var d=b[c],e=d.getTag();if(a.has$Value(e)){this.deserializedFields_&&delete this.deserializedFields_[d.getTag()];var f=d.isCompositeType();if(d.isRepeated())for(var d=a.array$Values(e),g=0;g<d.length;g++)this.add$Value(e,f?d[g].clone():d[g]);else d=a.getValueForTag_(e),f?(f=this.getValueForTag_(e))?
|
|
|
f.mergeFrom(d):this.set$Value(e,d.clone()):this.set$Value(e,d)}}};goog.proto2.Message.prototype.clone=function(){var a=new this.constructor;a.copyFrom(this);return a};
|
|
|
goog.proto2.Message.prototype.initDefaults=function(a){for(var b=this.getDescriptor().getFields(),c=0;c<b.length;c++){var d=b[c],e=d.getTag(),f=d.isCompositeType();this.has$Value(e)||d.isRepeated()||(f?this.values_[e]=new (d.getNativeType()):a&&(this.values_[e]=d.getDefaultValue()));if(f)if(d.isRepeated())for(d=this.array$Values(e),e=0;e<d.length;e++)d[e].initDefaults(a);else this.get$Value(e).initDefaults(a)}};goog.proto2.Message.prototype.has$Value=function(a){return null!=this.values_[a]};
|
|
|
goog.proto2.Message.prototype.getValueForTag_=function(a){var b=this.values_[a];return goog.isDefAndNotNull(b)?this.lazyDeserializer_?a in this.deserializedFields_?this.deserializedFields_[a]:(b=this.lazyDeserializer_.deserializeField(this,this.fields_[a],b),this.deserializedFields_[a]=b):b:null};
|
|
|
goog.proto2.Message.prototype.get$Value=function(a,b){var c=this.getValueForTag_(a);if(this.fields_[a].isRepeated()){var d=b||0;goog.asserts.assert(0<=d&&d<c.length,"Given index %s is out of bounds. Repeated field length: %s",d,c.length);return c[d]}return c};goog.proto2.Message.prototype.get$ValueOrDefault=function(a,b){return this.has$Value(a)?this.get$Value(a,b):this.fields_[a].getDefaultValue()};goog.proto2.Message.prototype.array$Values=function(a){return this.getValueForTag_(a)||[]};
|
|
|
goog.proto2.Message.prototype.count$Values=function(a){return this.fields_[a].isRepeated()?this.has$Value(a)?this.values_[a].length:0:this.has$Value(a)?1:0};goog.proto2.Message.prototype.set$Value=function(a,b){goog.asserts.ENABLE_ASSERTS&&this.checkFieldType_(this.fields_[a],b);this.values_[a]=b;this.deserializedFields_&&(this.deserializedFields_[a]=b)};
|
|
|
goog.proto2.Message.prototype.add$Value=function(a,b){goog.asserts.ENABLE_ASSERTS&&this.checkFieldType_(this.fields_[a],b);this.values_[a]||(this.values_[a]=[]);this.values_[a].push(b);this.deserializedFields_&&delete this.deserializedFields_[a]};goog.proto2.Message.prototype.checkFieldType_=function(a,b){a.getFieldType()==goog.proto2.FieldDescriptor.FieldType.ENUM?goog.asserts.assertNumber(b):goog.asserts.assert(b.constructor==a.getNativeType())};
|
|
|
goog.proto2.Message.prototype.clear$Field=function(a){delete this.values_[a];this.deserializedFields_&&delete this.deserializedFields_[a]};goog.proto2.Message.createDescriptor=function(a,b){var c=[],d=b[0],e;for(e in b)0!=e&&c.push(new goog.proto2.FieldDescriptor(a,e,b[e]));return new goog.proto2.Descriptor(a,d,c)};goog.proto2.Message.set$Metadata=function(a,b){a.descriptorObj_=b;a.getDescriptor=function(){return a.descriptor_||(new a).getDescriptor()}};goog.proto2.Serializer=function(){};goog.proto2.Serializer.DECODE_SYMBOLIC_ENUMS=!1;goog.proto2.Serializer.prototype.getSerializedValue=function(a,b){return a.isCompositeType()?this.serialize(b):goog.isNumber(b)&&!isFinite(b)?b.toString():b};goog.proto2.Serializer.prototype.deserialize=function(a,b){var c=a.createMessageInstance();this.deserializeTo(c,b);goog.asserts.assert(c instanceof goog.proto2.Message);return c};
|
|
|
goog.proto2.Serializer.prototype.getDeserializedValue=function(a,b){if(a.isCompositeType())return b instanceof goog.proto2.Message?b:this.deserialize(a.getFieldMessageType(),b);if(a.getFieldType()==goog.proto2.FieldDescriptor.FieldType.ENUM){if(goog.proto2.Serializer.DECODE_SYMBOLIC_ENUMS&&goog.isString(b)){var c=a.getNativeType();if(c.hasOwnProperty(b))return c[b]}return b}if(!a.deserializationConversionPermitted())return b;c=a.getNativeType();if(c===String){if(goog.isNumber(b))return String(b)}else if(c===
|
|
|
Number&&goog.isString(b)&&("Infinity"===b||"-Infinity"===b||"NaN"===b||/^-?[0-9]+$/.test(b)))return Number(b);return b};goog.proto2.LazyDeserializer=function(){};goog.inherits(goog.proto2.LazyDeserializer,goog.proto2.Serializer);goog.proto2.LazyDeserializer.prototype.deserialize=function(a,b){var c=a.createMessageInstance();c.initializeForLazyDeserializer(this,b);goog.asserts.assert(c instanceof goog.proto2.Message);return c};goog.proto2.LazyDeserializer.prototype.deserializeTo=function(a,b){throw Error("Unimplemented");};goog.proto2.PbLiteSerializer=function(){};goog.inherits(goog.proto2.PbLiteSerializer,goog.proto2.LazyDeserializer);goog.proto2.PbLiteSerializer.prototype.zeroIndexing_=!1;goog.proto2.PbLiteSerializer.prototype.setZeroIndexed=function(a){this.zeroIndexing_=a};
|
|
|
goog.proto2.PbLiteSerializer.prototype.serialize=function(a){for(var b=a.getDescriptor().getFields(),c=[],d=this.zeroIndexing_,e=0;e<b.length;e++){var f=b[e];if(a.has(f)){var g=f.getTag(),g=d?g-1:g;if(f.isRepeated()){c[g]=[];for(var h=0;h<a.countOf(f);h++)c[g][h]=this.getSerializedValue(f,a.get(f,h))}else c[g]=this.getSerializedValue(f,a.get(f))}}a.forEachUnknown(function(a,b){c[d?a-1:a]=b});return c};
|
|
|
goog.proto2.PbLiteSerializer.prototype.deserializeField=function(a,b,c){if(null==c)return c;if(b.isRepeated()){a=[];goog.asserts.assert(goog.isArray(c),"Value must be array: %s",c);for(var d=0;d<c.length;d++)a[d]=this.getDeserializedValue(b,c[d]);return a}return this.getDeserializedValue(b,c)};
|
|
|
goog.proto2.PbLiteSerializer.prototype.getSerializedValue=function(a,b){return a.getFieldType()==goog.proto2.FieldDescriptor.FieldType.BOOL?b?1:0:goog.proto2.Serializer.prototype.getSerializedValue.apply(this,arguments)};
|
|
|
goog.proto2.PbLiteSerializer.prototype.getDeserializedValue=function(a,b){return a.getFieldType()==goog.proto2.FieldDescriptor.FieldType.BOOL?(goog.asserts.assert(goog.isNumber(b)||goog.isBoolean(b),"Value is expected to be a number or boolean"),!!b):goog.proto2.Serializer.prototype.getDeserializedValue.apply(this,arguments)};
|
|
|
goog.proto2.PbLiteSerializer.prototype.deserialize=function(a,b){var c=b;if(this.zeroIndexing_){var c=[],d;for(d in b)c[parseInt(d,10)+1]=b[d]}return goog.proto2.PbLiteSerializer.superClass_.deserialize.call(this,a,c)};goog.string.StringBuffer=function(a,b){null!=a&&this.append.apply(this,arguments)};goog.string.StringBuffer.prototype.buffer_="";goog.string.StringBuffer.prototype.set=function(a){this.buffer_=""+a};goog.string.StringBuffer.prototype.append=function(a,b,c){this.buffer_+=a;if(null!=b)for(var d=1;d<arguments.length;d++)this.buffer_+=arguments[d];return this};goog.string.StringBuffer.prototype.clear=function(){this.buffer_=""};goog.string.StringBuffer.prototype.getLength=function(){return this.buffer_.length};
|
|
|
goog.string.StringBuffer.prototype.toString=function(){return this.buffer_};/*
|
|
|
|
|
|
Protocol Buffer 2 Copyright 2008 Google Inc.
|
|
|
All other code copyright its respective owners.
|
|
|
Copyright (C) 2010 The Libphonenumber Authors
|
|
|
|
|
|
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.
|
|
|
*/
|
|
|
var i18n={phonenumbers:{}};i18n.phonenumbers.NumberFormat=function(){goog.proto2.Message.apply(this)};goog.inherits(i18n.phonenumbers.NumberFormat,goog.proto2.Message);i18n.phonenumbers.NumberFormat.prototype.getPattern=function(){return this.get$Value(1)};i18n.phonenumbers.NumberFormat.prototype.getPatternOrDefault=function(){return this.get$ValueOrDefault(1)};i18n.phonenumbers.NumberFormat.prototype.setPattern=function(a){this.set$Value(1,a)};
|
|
|
i18n.phonenumbers.NumberFormat.prototype.hasPattern=function(){return this.has$Value(1)};i18n.phonenumbers.NumberFormat.prototype.patternCount=function(){return this.count$Values(1)};i18n.phonenumbers.NumberFormat.prototype.clearPattern=function(){this.clear$Field(1)};i18n.phonenumbers.NumberFormat.prototype.getFormat=function(){return this.get$Value(2)};i18n.phonenumbers.NumberFormat.prototype.getFormatOrDefault=function(){return this.get$ValueOrDefault(2)};
|
|
|
i18n.phonenumbers.NumberFormat.prototype.setFormat=function(a){this.set$Value(2,a)};i18n.phonenumbers.NumberFormat.prototype.hasFormat=function(){return this.has$Value(2)};i18n.phonenumbers.NumberFormat.prototype.formatCount=function(){return this.count$Values(2)};i18n.phonenumbers.NumberFormat.prototype.clearFormat=function(){this.clear$Field(2)};i18n.phonenumbers.NumberFormat.prototype.getLeadingDigitsPattern=function(a){return this.get$Value(3,a)};
|
|
|
i18n.phonenumbers.NumberFormat.prototype.getLeadingDigitsPatternOrDefault=function(a){return this.get$ValueOrDefault(3,a)};i18n.phonenumbers.NumberFormat.prototype.addLeadingDigitsPattern=function(a){this.add$Value(3,a)};i18n.phonenumbers.NumberFormat.prototype.leadingDigitsPatternArray=function(){return this.array$Values(3)};i18n.phonenumbers.NumberFormat.prototype.hasLeadingDigitsPattern=function(){return this.has$Value(3)};i18n.phonenumbers.NumberFormat.prototype.leadingDigitsPatternCount=function(){return this.count$Values(3)};
|
|
|
i18n.phonenumbers.NumberFormat.prototype.clearLeadingDigitsPattern=function(){this.clear$Field(3)};i18n.phonenumbers.NumberFormat.prototype.getNationalPrefixFormattingRule=function(){return this.get$Value(4)};i18n.phonenumbers.NumberFormat.prototype.getNationalPrefixFormattingRuleOrDefault=function(){return this.get$ValueOrDefault(4)};i18n.phonenumbers.NumberFormat.prototype.setNationalPrefixFormattingRule=function(a){this.set$Value(4,a)};
|
|
|
i18n.phonenumbers.NumberFormat.prototype.hasNationalPrefixFormattingRule=function(){return this.has$Value(4)};i18n.phonenumbers.NumberFormat.prototype.nationalPrefixFormattingRuleCount=function(){return this.count$Values(4)};i18n.phonenumbers.NumberFormat.prototype.clearNationalPrefixFormattingRule=function(){this.clear$Field(4)};i18n.phonenumbers.NumberFormat.prototype.getNationalPrefixOptionalWhenFormatting=function(){return this.get$Value(6)};
|
|
|
i18n.phonenumbers.NumberFormat.prototype.getNationalPrefixOptionalWhenFormattingOrDefault=function(){return this.get$ValueOrDefault(6)};i18n.phonenumbers.NumberFormat.prototype.setNationalPrefixOptionalWhenFormatting=function(a){this.set$Value(6,a)};i18n.phonenumbers.NumberFormat.prototype.hasNationalPrefixOptionalWhenFormatting=function(){return this.has$Value(6)};i18n.phonenumbers.NumberFormat.prototype.nationalPrefixOptionalWhenFormattingCount=function(){return this.count$Values(6)};
|
|
|
i18n.phonenumbers.NumberFormat.prototype.clearNationalPrefixOptionalWhenFormatting=function(){this.clear$Field(6)};i18n.phonenumbers.NumberFormat.prototype.getDomesticCarrierCodeFormattingRule=function(){return this.get$Value(5)};i18n.phonenumbers.NumberFormat.prototype.getDomesticCarrierCodeFormattingRuleOrDefault=function(){return this.get$ValueOrDefault(5)};i18n.phonenumbers.NumberFormat.prototype.setDomesticCarrierCodeFormattingRule=function(a){this.set$Value(5,a)};
|
|
|
i18n.phonenumbers.NumberFormat.prototype.hasDomesticCarrierCodeFormattingRule=function(){return this.has$Value(5)};i18n.phonenumbers.NumberFormat.prototype.domesticCarrierCodeFormattingRuleCount=function(){return this.count$Values(5)};i18n.phonenumbers.NumberFormat.prototype.clearDomesticCarrierCodeFormattingRule=function(){this.clear$Field(5)};i18n.phonenumbers.PhoneNumberDesc=function(){goog.proto2.Message.apply(this)};goog.inherits(i18n.phonenumbers.PhoneNumberDesc,goog.proto2.Message);
|
|
|
i18n.phonenumbers.PhoneNumberDesc.prototype.getNationalNumberPattern=function(){return this.get$Value(2)};i18n.phonenumbers.PhoneNumberDesc.prototype.getNationalNumberPatternOrDefault=function(){return this.get$ValueOrDefault(2)};i18n.phonenumbers.PhoneNumberDesc.prototype.setNationalNumberPattern=function(a){this.set$Value(2,a)};i18n.phonenumbers.PhoneNumberDesc.prototype.hasNationalNumberPattern=function(){return this.has$Value(2)};
|
|
|
i18n.phonenumbers.PhoneNumberDesc.prototype.nationalNumberPatternCount=function(){return this.count$Values(2)};i18n.phonenumbers.PhoneNumberDesc.prototype.clearNationalNumberPattern=function(){this.clear$Field(2)};i18n.phonenumbers.PhoneNumberDesc.prototype.getPossibleNumberPattern=function(){return this.get$Value(3)};i18n.phonenumbers.PhoneNumberDesc.prototype.getPossibleNumberPatternOrDefault=function(){return this.get$ValueOrDefault(3)};
|
|
|
i18n.phonenumbers.PhoneNumberDesc.prototype.setPossibleNumberPattern=function(a){this.set$Value(3,a)};i18n.phonenumbers.PhoneNumberDesc.prototype.hasPossibleNumberPattern=function(){return this.has$Value(3)};i18n.phonenumbers.PhoneNumberDesc.prototype.possibleNumberPatternCount=function(){return this.count$Values(3)};i18n.phonenumbers.PhoneNumberDesc.prototype.clearPossibleNumberPattern=function(){this.clear$Field(3)};i18n.phonenumbers.PhoneNumberDesc.prototype.getExampleNumber=function(){return this.get$Value(6)};
|
|
|
i18n.phonenumbers.PhoneNumberDesc.prototype.getExampleNumberOrDefault=function(){return this.get$ValueOrDefault(6)};i18n.phonenumbers.PhoneNumberDesc.prototype.setExampleNumber=function(a){this.set$Value(6,a)};i18n.phonenumbers.PhoneNumberDesc.prototype.hasExampleNumber=function(){return this.has$Value(6)};i18n.phonenumbers.PhoneNumberDesc.prototype.exampleNumberCount=function(){return this.count$Values(6)};i18n.phonenumbers.PhoneNumberDesc.prototype.clearExampleNumber=function(){this.clear$Field(6)};
|
|
|
i18n.phonenumbers.PhoneMetadata=function(){goog.proto2.Message.apply(this)};goog.inherits(i18n.phonenumbers.PhoneMetadata,goog.proto2.Message);i18n.phonenumbers.PhoneMetadata.prototype.getGeneralDesc=function(){return this.get$Value(1)};i18n.phonenumbers.PhoneMetadata.prototype.getGeneralDescOrDefault=function(){return this.get$ValueOrDefault(1)};i18n.phonenumbers.PhoneMetadata.prototype.setGeneralDesc=function(a){this.set$Value(1,a)};i18n.phonenumbers.PhoneMetadata.prototype.hasGeneralDesc=function(){return this.has$Value(1)};
|
|
|
i18n.phonenumbers.PhoneMetadata.prototype.generalDescCount=function(){return this.count$Values(1)};i18n.phonenumbers.PhoneMetadata.prototype.clearGeneralDesc=function(){this.clear$Field(1)};i18n.phonenumbers.PhoneMetadata.prototype.getFixedLine=function(){return this.get$Value(2)};i18n.phonenumbers.PhoneMetadata.prototype.getFixedLineOrDefault=function(){return this.get$ValueOrDefault(2)};i18n.phonenumbers.PhoneMetadata.prototype.setFixedLine=function(a){this.set$Value(2,a)};
|
|
|
i18n.phonenumbers.PhoneMetadata.prototype.hasFixedLine=function(){return this.has$Value(2)};i18n.phonenumbers.PhoneMetadata.prototype.fixedLineCount=function(){return this.count$Values(2)};i18n.phonenumbers.PhoneMetadata.prototype.clearFixedLine=function(){this.clear$Field(2)};i18n.phonenumbers.PhoneMetadata.prototype.getMobile=function(){return this.get$Value(3)};i18n.phonenumbers.PhoneMetadata.prototype.getMobileOrDefault=function(){return this.get$ValueOrDefault(3)};
|
|
|
i18n.phonenumbers.PhoneMetadata.prototype.setMobile=function(a){this.set$Value(3,a)};i18n.phonenumbers.PhoneMetadata.prototype.hasMobile=function(){return this.has$Value(3)};i18n.phonenumbers.PhoneMetadata.prototype.mobileCount=function(){return this.count$Values(3)};i18n.phonenumbers.PhoneMetadata.prototype.clearMobile=function(){this.clear$Field(3)};i18n.phonenumbers.PhoneMetadata.prototype.getTollFree=function(){return this.get$Value(4)};
|
|
|
i18n.phonenumbers.PhoneMetadata.prototype.getTollFreeOrDefault=function(){return this.get$ValueOrDefault(4)};i18n.phonenumbers.PhoneMetadata.prototype.setTollFree=function(a){this.set$Value(4,a)};i18n.phonenumbers.PhoneMetadata.prototype.hasTollFree=function(){return this.has$Value(4)};i18n.phonenumbers.PhoneMetadata.prototype.tollFreeCount=function(){return this.count$Values(4)};i18n.phonenumbers.PhoneMetadata.prototype.clearTollFree=function(){this.clear$Field(4)};
|
|
|
i18n.phonenumbers.PhoneMetadata.prototype.getPremiumRate=function(){return this.get$Value(5)};i18n.phonenumbers.PhoneMetadata.prototype.getPremiumRateOrDefault=function(){return this.get$ValueOrDefault(5)};i18n.phonenumbers.PhoneMetadata.prototype.setPremiumRate=function(a){this.set$Value(5,a)};i18n.phonenumbers.PhoneMetadata.prototype.hasPremiumRate=function(){return this.has$Value(5)};i18n.phonenumbers.PhoneMetadata.prototype.premiumRateCount=function(){return this.count$Values(5)};
|
|
|
i18n.phonenumbers.PhoneMetadata.prototype.clearPremiumRate=function(){this.clear$Field(5)};i18n.phonenumbers.PhoneMetadata.prototype.getSharedCost=function(){return this.get$Value(6)};i18n.phonenumbers.PhoneMetadata.prototype.getSharedCostOrDefault=function(){return this.get$ValueOrDefault(6)};i18n.phonenumbers.PhoneMetadata.prototype.setSharedCost=function(a){this.set$Value(6,a)};i18n.phonenumbers.PhoneMetadata.prototype.hasSharedCost=function(){return this.has$Value(6)};
|
|
|
i18n.phonenumbers.PhoneMetadata.prototype.sharedCostCount=function(){return this.count$Values(6)};i18n.phonenumbers.PhoneMetadata.prototype.clearSharedCost=function(){this.clear$Field(6)};i18n.phonenumbers.PhoneMetadata.prototype.getPersonalNumber=function(){return this.get$Value(7)};i18n.phonenumbers.PhoneMetadata.prototype.getPersonalNumberOrDefault=function(){return this.get$ValueOrDefault(7)};i18n.phonenumbers.PhoneMetadata.prototype.setPersonalNumber=function(a){this.set$Value(7,a)};
|
|
|
i18n.phonenumbers.PhoneMetadata.prototype.hasPersonalNumber=function(){return this.has$Value(7)};i18n.phonenumbers.PhoneMetadata.prototype.personalNumberCount=function(){return this.count$Values(7)};i18n.phonenumbers.PhoneMetadata.prototype.clearPersonalNumber=function(){this.clear$Field(7)};i18n.phonenumbers.PhoneMetadata.prototype.getVoip=function(){return this.get$Value(8)};i18n.phonenumbers.PhoneMetadata.prototype.getVoipOrDefault=function(){return this.get$ValueOrDefault(8)};
|
|
|
i18n.phonenumbers.PhoneMetadata.prototype.setVoip=function(a){this.set$Value(8,a)};i18n.phonenumbers.PhoneMetadata.prototype.hasVoip=function(){return this.has$Value(8)};i18n.phonenumbers.PhoneMetadata.prototype.voipCount=function(){return this.count$Values(8)};i18n.phonenumbers.PhoneMetadata.prototype.clearVoip=function(){this.clear$Field(8)};i18n.phonenumbers.PhoneMetadata.prototype.getPager=function(){return this.get$Value(21)};i18n.phonenumbers.PhoneMetadata.prototype.getPagerOrDefault=function(){return this.get$ValueOrDefault(21)};
|
|
|
i18n.phonenumbers.PhoneMetadata.prototype.setPager=function(a){this.set$Value(21,a)};i18n.phonenumbers.PhoneMetadata.prototype.hasPager=function(){return this.has$Value(21)};i18n.phonenumbers.PhoneMetadata.prototype.pagerCount=function(){return this.count$Values(21)};i18n.phonenumbers.PhoneMetadata.prototype.clearPager=function(){this.clear$Field(21)};i18n.phonenumbers.PhoneMetadata.prototype.getUan=function(){return this.get$Value(25)};i18n.phonenumbers.PhoneMetadata.prototype.getUanOrDefault=function(){return this.get$ValueOrDefault(25)};
|
|
|
i18n.phonenumbers.PhoneMetadata.prototype.setUan=function(a){this.set$Value(25,a)};i18n.phonenumbers.PhoneMetadata.prototype.hasUan=function(){return this.has$Value(25)};i18n.phonenumbers.PhoneMetadata.prototype.uanCount=function(){return this.count$Values(25)};i18n.phonenumbers.PhoneMetadata.prototype.clearUan=function(){this.clear$Field(25)};i18n.phonenumbers.PhoneMetadata.prototype.getEmergency=function(){return this.get$Value(27)};
|
|
|
i18n.phonenumbers.PhoneMetadata.prototype.getEmergencyOrDefault=function(){return this.get$ValueOrDefault(27)};i18n.phonenumbers.PhoneMetadata.prototype.setEmergency=function(a){this.set$Value(27,a)};i18n.phonenumbers.PhoneMetadata.prototype.hasEmergency=function(){return this.has$Value(27)};i18n.phonenumbers.PhoneMetadata.prototype.emergencyCount=function(){return this.count$Values(27)};i18n.phonenumbers.PhoneMetadata.prototype.clearEmergency=function(){this.clear$Field(27)};
|
|
|
i18n.phonenumbers.PhoneMetadata.prototype.getVoicemail=function(){return this.get$Value(28)};i18n.phonenumbers.PhoneMetadata.prototype.getVoicemailOrDefault=function(){return this.get$ValueOrDefault(28)};i18n.phonenumbers.PhoneMetadata.prototype.setVoicemail=function(a){this.set$Value(28,a)};i18n.phonenumbers.PhoneMetadata.prototype.hasVoicemail=function(){return this.has$Value(28)};i18n.phonenumbers.PhoneMetadata.prototype.voicemailCount=function(){return this.count$Values(28)};
|
|
|
i18n.phonenumbers.PhoneMetadata.prototype.clearVoicemail=function(){this.clear$Field(28)};i18n.phonenumbers.PhoneMetadata.prototype.getNoInternationalDialling=function(){return this.get$Value(24)};i18n.phonenumbers.PhoneMetadata.prototype.getNoInternationalDiallingOrDefault=function(){return this.get$ValueOrDefault(24)};i18n.phonenumbers.PhoneMetadata.prototype.setNoInternationalDialling=function(a){this.set$Value(24,a)};i18n.phonenumbers.PhoneMetadata.prototype.hasNoInternationalDialling=function(){return this.has$Value(24)};
|
|
|
i18n.phonenumbers.PhoneMetadata.prototype.noInternationalDiallingCount=function(){return this.count$Values(24)};i18n.phonenumbers.PhoneMetadata.prototype.clearNoInternationalDialling=function(){this.clear$Field(24)};i18n.phonenumbers.PhoneMetadata.prototype.getId=function(){return this.get$Value(9)};i18n.phonenumbers.PhoneMetadata.prototype.getIdOrDefault=function(){return this.get$ValueOrDefault(9)};i18n.phonenumbers.PhoneMetadata.prototype.setId=function(a){this.set$Value(9,a)};
|
|
|
i18n.phonenumbers.PhoneMetadata.prototype.hasId=function(){return this.has$Value(9)};i18n.phonenumbers.PhoneMetadata.prototype.idCount=function(){return this.count$Values(9)};i18n.phonenumbers.PhoneMetadata.prototype.clearId=function(){this.clear$Field(9)};i18n.phonenumbers.PhoneMetadata.prototype.getCountryCode=function(){return this.get$Value(10)};i18n.phonenumbers.PhoneMetadata.prototype.getCountryCodeOrDefault=function(){return this.get$ValueOrDefault(10)};
|
|
|
i18n.phonenumbers.PhoneMetadata.prototype.setCountryCode=function(a){this.set$Value(10,a)};i18n.phonenumbers.PhoneMetadata.prototype.hasCountryCode=function(){return this.has$Value(10)};i18n.phonenumbers.PhoneMetadata.prototype.countryCodeCount=function(){return this.count$Values(10)};i18n.phonenumbers.PhoneMetadata.prototype.clearCountryCode=function(){this.clear$Field(10)};i18n.phonenumbers.PhoneMetadata.prototype.getInternationalPrefix=function(){return this.get$Value(11)};
|
|
|
i18n.phonenumbers.PhoneMetadata.prototype.getInternationalPrefixOrDefault=function(){return this.get$ValueOrDefault(11)};i18n.phonenumbers.PhoneMetadata.prototype.setInternationalPrefix=function(a){this.set$Value(11,a)};i18n.phonenumbers.PhoneMetadata.prototype.hasInternationalPrefix=function(){return this.has$Value(11)};i18n.phonenumbers.PhoneMetadata.prototype.internationalPrefixCount=function(){return this.count$Values(11)};i18n.phonenumbers.PhoneMetadata.prototype.clearInternationalPrefix=function(){this.clear$Field(11)};
|
|
|
i18n.phonenumbers.PhoneMetadata.prototype.getPreferredInternationalPrefix=function(){return this.get$Value(17)};i18n.phonenumbers.PhoneMetadata.prototype.getPreferredInternationalPrefixOrDefault=function(){return this.get$ValueOrDefault(17)};i18n.phonenumbers.PhoneMetadata.prototype.setPreferredInternationalPrefix=function(a){this.set$Value(17,a)};i18n.phonenumbers.PhoneMetadata.prototype.hasPreferredInternationalPrefix=function(){return this.has$Value(17)};
|
|
|
i18n.phonenumbers.PhoneMetadata.prototype.preferredInternationalPrefixCount=function(){return this.count$Values(17)};i18n.phonenumbers.PhoneMetadata.prototype.clearPreferredInternationalPrefix=function(){this.clear$Field(17)};i18n.phonenumbers.PhoneMetadata.prototype.getNationalPrefix=function(){return this.get$Value(12)};i18n.phonenumbers.PhoneMetadata.prototype.getNationalPrefixOrDefault=function(){return this.get$ValueOrDefault(12)};
|
|
|
i18n.phonenumbers.PhoneMetadata.prototype.setNationalPrefix=function(a){this.set$Value(12,a)};i18n.phonenumbers.PhoneMetadata.prototype.hasNationalPrefix=function(){return this.has$Value(12)};i18n.phonenumbers.PhoneMetadata.prototype.nationalPrefixCount=function(){return this.count$Values(12)};i18n.phonenumbers.PhoneMetadata.prototype.clearNationalPrefix=function(){this.clear$Field(12)};i18n.phonenumbers.PhoneMetadata.prototype.getPreferredExtnPrefix=function(){return this.get$Value(13)};
|
|
|
i18n.phonenumbers.PhoneMetadata.prototype.getPreferredExtnPrefixOrDefault=function(){return this.get$ValueOrDefault(13)};i18n.phonenumbers.PhoneMetadata.prototype.setPreferredExtnPrefix=function(a){this.set$Value(13,a)};i18n.phonenumbers.PhoneMetadata.prototype.hasPreferredExtnPrefix=function(){return this.has$Value(13)};i18n.phonenumbers.PhoneMetadata.prototype.preferredExtnPrefixCount=function(){return this.count$Values(13)};i18n.phonenumbers.PhoneMetadata.prototype.clearPreferredExtnPrefix=function(){this.clear$Field(13)};
|
|
|
i18n.phonenumbers.PhoneMetadata.prototype.getNationalPrefixForParsing=function(){return this.get$Value(15)};i18n.phonenumbers.PhoneMetadata.prototype.getNationalPrefixForParsingOrDefault=function(){return this.get$ValueOrDefault(15)};i18n.phonenumbers.PhoneMetadata.prototype.setNationalPrefixForParsing=function(a){this.set$Value(15,a)};i18n.phonenumbers.PhoneMetadata.prototype.hasNationalPrefixForParsing=function(){return this.has$Value(15)};
|
|
|
i18n.phonenumbers.PhoneMetadata.prototype.nationalPrefixForParsingCount=function(){return this.count$Values(15)};i18n.phonenumbers.PhoneMetadata.prototype.clearNationalPrefixForParsing=function(){this.clear$Field(15)};i18n.phonenumbers.PhoneMetadata.prototype.getNationalPrefixTransformRule=function(){return this.get$Value(16)};i18n.phonenumbers.PhoneMetadata.prototype.getNationalPrefixTransformRuleOrDefault=function(){return this.get$ValueOrDefault(16)};
|
|
|
i18n.phonenumbers.PhoneMetadata.prototype.setNationalPrefixTransformRule=function(a){this.set$Value(16,a)};i18n.phonenumbers.PhoneMetadata.prototype.hasNationalPrefixTransformRule=function(){return this.has$Value(16)};i18n.phonenumbers.PhoneMetadata.prototype.nationalPrefixTransformRuleCount=function(){return this.count$Values(16)};i18n.phonenumbers.PhoneMetadata.prototype.clearNationalPrefixTransformRule=function(){this.clear$Field(16)};
|
|
|
i18n.phonenumbers.PhoneMetadata.prototype.getSameMobileAndFixedLinePattern=function(){return this.get$Value(18)};i18n.phonenumbers.PhoneMetadata.prototype.getSameMobileAndFixedLinePatternOrDefault=function(){return this.get$ValueOrDefault(18)};i18n.phonenumbers.PhoneMetadata.prototype.setSameMobileAndFixedLinePattern=function(a){this.set$Value(18,a)};i18n.phonenumbers.PhoneMetadata.prototype.hasSameMobileAndFixedLinePattern=function(){return this.has$Value(18)};
|
|
|
i18n.phonenumbers.PhoneMetadata.prototype.sameMobileAndFixedLinePatternCount=function(){return this.count$Values(18)};i18n.phonenumbers.PhoneMetadata.prototype.clearSameMobileAndFixedLinePattern=function(){this.clear$Field(18)};i18n.phonenumbers.PhoneMetadata.prototype.getNumberFormat=function(a){return this.get$Value(19,a)};i18n.phonenumbers.PhoneMetadata.prototype.getNumberFormatOrDefault=function(a){return this.get$ValueOrDefault(19,a)};
|
|
|
i18n.phonenumbers.PhoneMetadata.prototype.addNumberFormat=function(a){this.add$Value(19,a)};i18n.phonenumbers.PhoneMetadata.prototype.numberFormatArray=function(){return this.array$Values(19)};i18n.phonenumbers.PhoneMetadata.prototype.hasNumberFormat=function(){return this.has$Value(19)};i18n.phonenumbers.PhoneMetadata.prototype.numberFormatCount=function(){return this.count$Values(19)};i18n.phonenumbers.PhoneMetadata.prototype.clearNumberFormat=function(){this.clear$Field(19)};
|
|
|
i18n.phonenumbers.PhoneMetadata.prototype.getIntlNumberFormat=function(a){return this.get$Value(20,a)};i18n.phonenumbers.PhoneMetadata.prototype.getIntlNumberFormatOrDefault=function(a){return this.get$ValueOrDefault(20,a)};i18n.phonenumbers.PhoneMetadata.prototype.addIntlNumberFormat=function(a){this.add$Value(20,a)};i18n.phonenumbers.PhoneMetadata.prototype.intlNumberFormatArray=function(){return this.array$Values(20)};i18n.phonenumbers.PhoneMetadata.prototype.hasIntlNumberFormat=function(){return this.has$Value(20)};
|
|
|
i18n.phonenumbers.PhoneMetadata.prototype.intlNumberFormatCount=function(){return this.count$Values(20)};i18n.phonenumbers.PhoneMetadata.prototype.clearIntlNumberFormat=function(){this.clear$Field(20)};i18n.phonenumbers.PhoneMetadata.prototype.getMainCountryForCode=function(){return this.get$Value(22)};i18n.phonenumbers.PhoneMetadata.prototype.getMainCountryForCodeOrDefault=function(){return this.get$ValueOrDefault(22)};
|
|
|
i18n.phonenumbers.PhoneMetadata.prototype.setMainCountryForCode=function(a){this.set$Value(22,a)};i18n.phonenumbers.PhoneMetadata.prototype.hasMainCountryForCode=function(){return this.has$Value(22)};i18n.phonenumbers.PhoneMetadata.prototype.mainCountryForCodeCount=function(){return this.count$Values(22)};i18n.phonenumbers.PhoneMetadata.prototype.clearMainCountryForCode=function(){this.clear$Field(22)};i18n.phonenumbers.PhoneMetadata.prototype.getLeadingDigits=function(){return this.get$Value(23)};
|
|
|
i18n.phonenumbers.PhoneMetadata.prototype.getLeadingDigitsOrDefault=function(){return this.get$ValueOrDefault(23)};i18n.phonenumbers.PhoneMetadata.prototype.setLeadingDigits=function(a){this.set$Value(23,a)};i18n.phonenumbers.PhoneMetadata.prototype.hasLeadingDigits=function(){return this.has$Value(23)};i18n.phonenumbers.PhoneMetadata.prototype.leadingDigitsCount=function(){return this.count$Values(23)};i18n.phonenumbers.PhoneMetadata.prototype.clearLeadingDigits=function(){this.clear$Field(23)};
|
|
|
i18n.phonenumbers.PhoneMetadata.prototype.getLeadingZeroPossible=function(){return this.get$Value(26)};i18n.phonenumbers.PhoneMetadata.prototype.getLeadingZeroPossibleOrDefault=function(){return this.get$ValueOrDefault(26)};i18n.phonenumbers.PhoneMetadata.prototype.setLeadingZeroPossible=function(a){this.set$Value(26,a)};i18n.phonenumbers.PhoneMetadata.prototype.hasLeadingZeroPossible=function(){return this.has$Value(26)};i18n.phonenumbers.PhoneMetadata.prototype.leadingZeroPossibleCount=function(){return this.count$Values(26)};
|
|
|
i18n.phonenumbers.PhoneMetadata.prototype.clearLeadingZeroPossible=function(){this.clear$Field(26)};i18n.phonenumbers.PhoneMetadataCollection=function(){goog.proto2.Message.apply(this)};goog.inherits(i18n.phonenumbers.PhoneMetadataCollection,goog.proto2.Message);i18n.phonenumbers.PhoneMetadataCollection.prototype.getMetadata=function(a){return this.get$Value(1,a)};i18n.phonenumbers.PhoneMetadataCollection.prototype.getMetadataOrDefault=function(a){return this.get$ValueOrDefault(1,a)};
|
|
|
i18n.phonenumbers.PhoneMetadataCollection.prototype.addMetadata=function(a){this.add$Value(1,a)};i18n.phonenumbers.PhoneMetadataCollection.prototype.metadataArray=function(){return this.array$Values(1)};i18n.phonenumbers.PhoneMetadataCollection.prototype.hasMetadata=function(){return this.has$Value(1)};i18n.phonenumbers.PhoneMetadataCollection.prototype.metadataCount=function(){return this.count$Values(1)};i18n.phonenumbers.PhoneMetadataCollection.prototype.clearMetadata=function(){this.clear$Field(1)};
|
|
|
goog.proto2.Message.set$Metadata(i18n.phonenumbers.NumberFormat,{0:{name:"NumberFormat",fullName:"i18n.phonenumbers.NumberFormat"},1:{name:"pattern",required:!0,fieldType:goog.proto2.Message.FieldType.STRING,type:String},2:{name:"format",required:!0,fieldType:goog.proto2.Message.FieldType.STRING,type:String},3:{name:"leading_digits_pattern",repeated:!0,fieldType:goog.proto2.Message.FieldType.STRING,type:String},4:{name:"national_prefix_formatting_rule",fieldType:goog.proto2.Message.FieldType.STRING,
|
|
|
type:String},6:{name:"national_prefix_optional_when_formatting",fieldType:goog.proto2.Message.FieldType.BOOL,type:Boolean},5:{name:"domestic_carrier_code_formatting_rule",fieldType:goog.proto2.Message.FieldType.STRING,type:String}});
|
|
|
goog.proto2.Message.set$Metadata(i18n.phonenumbers.PhoneNumberDesc,{0:{name:"PhoneNumberDesc",fullName:"i18n.phonenumbers.PhoneNumberDesc"},2:{name:"national_number_pattern",fieldType:goog.proto2.Message.FieldType.STRING,type:String},3:{name:"possible_number_pattern",fieldType:goog.proto2.Message.FieldType.STRING,type:String},6:{name:"example_number",fieldType:goog.proto2.Message.FieldType.STRING,type:String}});
|
|
|
goog.proto2.Message.set$Metadata(i18n.phonenumbers.PhoneMetadata,{0:{name:"PhoneMetadata",fullName:"i18n.phonenumbers.PhoneMetadata"},1:{name:"general_desc",required:!0,fieldType:goog.proto2.Message.FieldType.MESSAGE,type:i18n.phonenumbers.PhoneNumberDesc},2:{name:"fixed_line",required:!0,fieldType:goog.proto2.Message.FieldType.MESSAGE,type:i18n.phonenumbers.PhoneNumberDesc},3:{name:"mobile",required:!0,fieldType:goog.proto2.Message.FieldType.MESSAGE,type:i18n.phonenumbers.PhoneNumberDesc},4:{name:"toll_free",
|
|
|
required:!0,fieldType:goog.proto2.Message.FieldType.MESSAGE,type:i18n.phonenumbers.PhoneNumberDesc},5:{name:"premium_rate",required:!0,fieldType:goog.proto2.Message.FieldType.MESSAGE,type:i18n.phonenumbers.PhoneNumberDesc},6:{name:"shared_cost",required:!0,fieldType:goog.proto2.Message.FieldType.MESSAGE,type:i18n.phonenumbers.PhoneNumberDesc},7:{name:"personal_number",required:!0,fieldType:goog.proto2.Message.FieldType.MESSAGE,type:i18n.phonenumbers.PhoneNumberDesc},8:{name:"voip",required:!0,fieldType:goog.proto2.Message.FieldType.MESSAGE,
|
|
|
type:i18n.phonenumbers.PhoneNumberDesc},21:{name:"pager",required:!0,fieldType:goog.proto2.Message.FieldType.MESSAGE,type:i18n.phonenumbers.PhoneNumberDesc},25:{name:"uan",required:!0,fieldType:goog.proto2.Message.FieldType.MESSAGE,type:i18n.phonenumbers.PhoneNumberDesc},27:{name:"emergency",required:!0,fieldType:goog.proto2.Message.FieldType.MESSAGE,type:i18n.phonenumbers.PhoneNumberDesc},28:{name:"voicemail",required:!0,fieldType:goog.proto2.Message.FieldType.MESSAGE,type:i18n.phonenumbers.PhoneNumberDesc},
|
|
|
24:{name:"no_international_dialling",required:!0,fieldType:goog.proto2.Message.FieldType.MESSAGE,type:i18n.phonenumbers.PhoneNumberDesc},9:{name:"id",required:!0,fieldType:goog.proto2.Message.FieldType.STRING,type:String},10:{name:"country_code",required:!0,fieldType:goog.proto2.Message.FieldType.INT32,type:Number},11:{name:"international_prefix",required:!0,fieldType:goog.proto2.Message.FieldType.STRING,type:String},17:{name:"preferred_international_prefix",fieldType:goog.proto2.Message.FieldType.STRING,
|
|
|
type:String},12:{name:"national_prefix",fieldType:goog.proto2.Message.FieldType.STRING,type:String},13:{name:"preferred_extn_prefix",fieldType:goog.proto2.Message.FieldType.STRING,type:String},15:{name:"national_prefix_for_parsing",fieldType:goog.proto2.Message.FieldType.STRING,type:String},16:{name:"national_prefix_transform_rule",fieldType:goog.proto2.Message.FieldType.STRING,type:String},18:{name:"same_mobile_and_fixed_line_pattern",fieldType:goog.proto2.Message.FieldType.BOOL,defaultValue:!1,
|
|
|
type:Boolean},19:{name:"number_format",repeated:!0,fieldType:goog.proto2.Message.FieldType.MESSAGE,type:i18n.phonenumbers.NumberFormat},20:{name:"intl_number_format",repeated:!0,fieldType:goog.proto2.Message.FieldType.MESSAGE,type:i18n.phonenumbers.NumberFormat},22:{name:"main_country_for_code",fieldType:goog.proto2.Message.FieldType.BOOL,defaultValue:!1,type:Boolean},23:{name:"leading_digits",fieldType:goog.proto2.Message.FieldType.STRING,type:String},26:{name:"leading_zero_possible",fieldType:goog.proto2.Message.FieldType.BOOL,
|
|
|
defaultValue:!1,type:Boolean}});goog.proto2.Message.set$Metadata(i18n.phonenumbers.PhoneMetadataCollection,{0:{name:"PhoneMetadataCollection",fullName:"i18n.phonenumbers.PhoneMetadataCollection"},1:{name:"metadata",repeated:!0,fieldType:goog.proto2.Message.FieldType.MESSAGE,type:i18n.phonenumbers.PhoneMetadata}});/*
|
|
|
|
|
|
Protocol Buffer 2 Copyright 2008 Google Inc.
|
|
|
All other code copyright its respective owners.
|
|
|
Copyright (C) 2010 The Libphonenumber Authors
|
|
|
|
|
|
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.
|
|
|
*/
|
|
|
i18n.phonenumbers.PhoneNumber=function(){goog.proto2.Message.apply(this)};goog.inherits(i18n.phonenumbers.PhoneNumber,goog.proto2.Message);i18n.phonenumbers.PhoneNumber.prototype.getCountryCode=function(){return this.get$Value(1)};i18n.phonenumbers.PhoneNumber.prototype.getCountryCodeOrDefault=function(){return this.get$ValueOrDefault(1)};i18n.phonenumbers.PhoneNumber.prototype.setCountryCode=function(a){this.set$Value(1,a)};i18n.phonenumbers.PhoneNumber.prototype.hasCountryCode=function(){return this.has$Value(1)};
|
|
|
i18n.phonenumbers.PhoneNumber.prototype.countryCodeCount=function(){return this.count$Values(1)};i18n.phonenumbers.PhoneNumber.prototype.clearCountryCode=function(){this.clear$Field(1)};i18n.phonenumbers.PhoneNumber.prototype.getNationalNumber=function(){return this.get$Value(2)};i18n.phonenumbers.PhoneNumber.prototype.getNationalNumberOrDefault=function(){return this.get$ValueOrDefault(2)};i18n.phonenumbers.PhoneNumber.prototype.setNationalNumber=function(a){this.set$Value(2,a)};
|
|
|
i18n.phonenumbers.PhoneNumber.prototype.hasNationalNumber=function(){return this.has$Value(2)};i18n.phonenumbers.PhoneNumber.prototype.nationalNumberCount=function(){return this.count$Values(2)};i18n.phonenumbers.PhoneNumber.prototype.clearNationalNumber=function(){this.clear$Field(2)};i18n.phonenumbers.PhoneNumber.prototype.getExtension=function(){return this.get$Value(3)};i18n.phonenumbers.PhoneNumber.prototype.getExtensionOrDefault=function(){return this.get$ValueOrDefault(3)};
|
|
|
i18n.phonenumbers.PhoneNumber.prototype.setExtension=function(a){this.set$Value(3,a)};i18n.phonenumbers.PhoneNumber.prototype.hasExtension=function(){return this.has$Value(3)};i18n.phonenumbers.PhoneNumber.prototype.extensionCount=function(){return this.count$Values(3)};i18n.phonenumbers.PhoneNumber.prototype.clearExtension=function(){this.clear$Field(3)};i18n.phonenumbers.PhoneNumber.prototype.getItalianLeadingZero=function(){return this.get$Value(4)};
|
|
|
i18n.phonenumbers.PhoneNumber.prototype.getItalianLeadingZeroOrDefault=function(){return this.get$ValueOrDefault(4)};i18n.phonenumbers.PhoneNumber.prototype.setItalianLeadingZero=function(a){this.set$Value(4,a)};i18n.phonenumbers.PhoneNumber.prototype.hasItalianLeadingZero=function(){return this.has$Value(4)};i18n.phonenumbers.PhoneNumber.prototype.italianLeadingZeroCount=function(){return this.count$Values(4)};i18n.phonenumbers.PhoneNumber.prototype.clearItalianLeadingZero=function(){this.clear$Field(4)};
|
|
|
i18n.phonenumbers.PhoneNumber.prototype.getNumberOfLeadingZeros=function(){return this.get$Value(8)};i18n.phonenumbers.PhoneNumber.prototype.getNumberOfLeadingZerosOrDefault=function(){return this.get$ValueOrDefault(8)};i18n.phonenumbers.PhoneNumber.prototype.setNumberOfLeadingZeros=function(a){this.set$Value(8,a)};i18n.phonenumbers.PhoneNumber.prototype.hasNumberOfLeadingZeros=function(){return this.has$Value(8)};i18n.phonenumbers.PhoneNumber.prototype.numberOfLeadingZerosCount=function(){return this.count$Values(8)};
|
|
|
i18n.phonenumbers.PhoneNumber.prototype.clearNumberOfLeadingZeros=function(){this.clear$Field(8)};i18n.phonenumbers.PhoneNumber.prototype.getRawInput=function(){return this.get$Value(5)};i18n.phonenumbers.PhoneNumber.prototype.getRawInputOrDefault=function(){return this.get$ValueOrDefault(5)};i18n.phonenumbers.PhoneNumber.prototype.setRawInput=function(a){this.set$Value(5,a)};i18n.phonenumbers.PhoneNumber.prototype.hasRawInput=function(){return this.has$Value(5)};
|
|
|
i18n.phonenumbers.PhoneNumber.prototype.rawInputCount=function(){return this.count$Values(5)};i18n.phonenumbers.PhoneNumber.prototype.clearRawInput=function(){this.clear$Field(5)};i18n.phonenumbers.PhoneNumber.prototype.getCountryCodeSource=function(){return this.get$Value(6)};i18n.phonenumbers.PhoneNumber.prototype.getCountryCodeSourceOrDefault=function(){return this.get$ValueOrDefault(6)};i18n.phonenumbers.PhoneNumber.prototype.setCountryCodeSource=function(a){this.set$Value(6,a)};
|
|
|
i18n.phonenumbers.PhoneNumber.prototype.hasCountryCodeSource=function(){return this.has$Value(6)};i18n.phonenumbers.PhoneNumber.prototype.countryCodeSourceCount=function(){return this.count$Values(6)};i18n.phonenumbers.PhoneNumber.prototype.clearCountryCodeSource=function(){this.clear$Field(6)};i18n.phonenumbers.PhoneNumber.prototype.getPreferredDomesticCarrierCode=function(){return this.get$Value(7)};i18n.phonenumbers.PhoneNumber.prototype.getPreferredDomesticCarrierCodeOrDefault=function(){return this.get$ValueOrDefault(7)};
|
|
|
i18n.phonenumbers.PhoneNumber.prototype.setPreferredDomesticCarrierCode=function(a){this.set$Value(7,a)};i18n.phonenumbers.PhoneNumber.prototype.hasPreferredDomesticCarrierCode=function(){return this.has$Value(7)};i18n.phonenumbers.PhoneNumber.prototype.preferredDomesticCarrierCodeCount=function(){return this.count$Values(7)};i18n.phonenumbers.PhoneNumber.prototype.clearPreferredDomesticCarrierCode=function(){this.clear$Field(7)};
|
|
|
i18n.phonenumbers.PhoneNumber.CountryCodeSource={FROM_NUMBER_WITH_PLUS_SIGN:1,FROM_NUMBER_WITH_IDD:5,FROM_NUMBER_WITHOUT_PLUS_SIGN:10,FROM_DEFAULT_COUNTRY:20};
|
|
|
goog.proto2.Message.set$Metadata(i18n.phonenumbers.PhoneNumber,{0:{name:"PhoneNumber",fullName:"i18n.phonenumbers.PhoneNumber"},1:{name:"country_code",required:!0,fieldType:goog.proto2.Message.FieldType.INT32,type:Number},2:{name:"national_number",required:!0,fieldType:goog.proto2.Message.FieldType.UINT64,type:Number},3:{name:"extension",fieldType:goog.proto2.Message.FieldType.STRING,type:String},4:{name:"italian_leading_zero",fieldType:goog.proto2.Message.FieldType.BOOL,type:Boolean},8:{name:"number_of_leading_zeros",
|
|
|
fieldType:goog.proto2.Message.FieldType.INT32,defaultValue:1,type:Number},5:{name:"raw_input",fieldType:goog.proto2.Message.FieldType.STRING,type:String},6:{name:"country_code_source",fieldType:goog.proto2.Message.FieldType.ENUM,defaultValue:i18n.phonenumbers.PhoneNumber.CountryCodeSource.FROM_NUMBER_WITH_PLUS_SIGN,type:i18n.phonenumbers.PhoneNumber.CountryCodeSource},7:{name:"preferred_domestic_carrier_code",fieldType:goog.proto2.Message.FieldType.STRING,type:String}});/*
|
|
|
|
|
|
Copyright (C) 2010 The Libphonenumber Authors
|
|
|
|
|
|
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.
|
|
|
*/
|
|
|
i18n.phonenumbers.metadata={};
|
|
|
i18n.phonenumbers.metadata.countryCodeToRegionCodeMap={1:"US AG AI AS BB BM BS CA DM DO GD GU JM KN KY LC MP MS PR SX TC TT VC VG VI".split(" "),7:["RU","KZ"],20:["EG"],27:["ZA"],30:["GR"],31:["NL"],32:["BE"],33:["FR"],34:["ES"],36:["HU"],39:["IT"],40:["RO"],41:["CH"],43:["AT"],44:["GB","GG","IM","JE"],45:["DK"],46:["SE"],47:["NO","SJ"],48:["PL"],49:["DE"],51:["PE"],52:["MX"],53:["CU"],54:["AR"],55:["BR"],56:["CL"],57:["CO"],58:["VE"],60:["MY"],61:["AU","CC","CX"],62:["ID"],63:["PH"],64:["NZ"],65:["SG"],
|
|
|
66:["TH"],81:["JP"],82:["KR"],84:["VN"],86:["CN"],90:["TR"],91:["IN"],92:["PK"],93:["AF"],94:["LK"],95:["MM"],98:["IR"],211:["SS"],212:["MA","EH"],213:["DZ"],216:["TN"],218:["LY"],220:["GM"],221:["SN"],222:["MR"],223:["ML"],224:["GN"],225:["CI"],226:["BF"],227:["NE"],228:["TG"],229:["BJ"],230:["MU"],231:["LR"],232:["SL"],233:["GH"],234:["NG"],235:["TD"],236:["CF"],237:["CM"],238:["CV"],239:["ST"],240:["GQ"],241:["GA"],242:["CG"],243:["CD"],244:["AO"],245:["GW"],246:["IO"],247:["AC"],248:["SC"],249:["SD"],
|
|
|
250:["RW"],251:["ET"],252:["SO"],253:["DJ"],254:["KE"],255:["TZ"],256:["UG"],257:["BI"],258:["MZ"],260:["ZM"],261:["MG"],262:["RE","YT"],263:["ZW"],264:["NA"],265:["MW"],266:["LS"],267:["BW"],268:["SZ"],269:["KM"],290:["SH","TA"],291:["ER"],297:["AW"],298:["FO"],299:["GL"],350:["GI"],351:["PT"],352:["LU"],353:["IE"],354:["IS"],355:["AL"],356:["MT"],357:["CY"],358:["FI","AX"],359:["BG"],370:["LT"],371:["LV"],372:["EE"],373:["MD"],374:["AM"],375:["BY"],376:["AD"],377:["MC"],378:["SM"],379:["VA"],380:["UA"],
|
|
|
381:["RS"],382:["ME"],385:["HR"],386:["SI"],387:["BA"],389:["MK"],420:["CZ"],421:["SK"],423:["LI"],500:["FK"],501:["BZ"],502:["GT"],503:["SV"],504:["HN"],505:["NI"],506:["CR"],507:["PA"],508:["PM"],509:["HT"],590:["GP","BL","MF"],591:["BO"],592:["GY"],593:["EC"],594:["GF"],595:["PY"],596:["MQ"],597:["SR"],598:["UY"],599:["CW","BQ"],670:["TL"],672:["NF"],673:["BN"],674:["NR"],675:["PG"],676:["TO"],677:["SB"],678:["VU"],679:["FJ"],680:["PW"],681:["WF"],682:["CK"],683:["NU"],685:["WS"],686:["KI"],687:["NC"],
|
|
|
688:["TV"],689:["PF"],690:["TK"],691:["FM"],692:["MH"],800:["001"],808:["001"],850:["KP"],852:["HK"],853:["MO"],855:["KH"],856:["LA"],870:["001"],878:["001"],880:["BD"],881:["001"],882:["001"],883:["001"],886:["TW"],888:["001"],960:["MV"],961:["LB"],962:["JO"],963:["SY"],964:["IQ"],965:["KW"],966:["SA"],967:["YE"],968:["OM"],970:["PS"],971:["AE"],972:["IL"],973:["BH"],974:["QA"],975:["BT"],976:["MN"],977:["NP"],979:["001"],992:["TJ"],993:["TM"],994:["AZ"],995:["GE"],996:["KG"],998:["UZ"]};
|
|
|
i18n.phonenumbers.metadata.countryToMetadata={AC:[,[,,"[2-7]\\d{3,5}","\\d{4,6}"],[,,"(?:[267]\\d|3[0-5]|4[4-69])\\d{2}","\\d{4}",,,"6889"],[,,"5\\d{5}","\\d{6}",,,"501234"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"AC",247,"00",,,,,,,,,,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],AD:[,[,,"(?:[346-9]|180)\\d{5}","\\d{6,8}"],[,,"[78]\\d{5}","\\d{6}",,,"712345"],[,,"[346]\\d{5}","\\d{6}",,,"312345"],[,,"180[02]\\d{4}","\\d{8}",,,"18001234"],[,,"9\\d{5}",
|
|
|
"\\d{6}",,,"912345"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"AD",376,"00",,,,,,,,[[,"(\\d{3})(\\d{3})","$1 $2",["[346-9]"],"","",0],[,"(180[02])(\\d{4})","$1 $2",["1"],"","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],AE:[,[,,"[2-79]\\d{7,8}|800\\d{2,9}","\\d{5,12}"],[,,"[2-4679][2-8]\\d{6}","\\d{7,8}",,,"22345678"],[,,"5[0256]\\d{7}","\\d{9}",,,"501234567"],[,,"400\\d{6}|800\\d{2,9}","\\d{5,12}",,,"800123456"],[,,"900[02]\\d{5}","\\d{9}",,,"900234567"],[,,"700[05]\\d{5}",
|
|
|
"\\d{9}",,,"700012345"],[,,"NA","NA"],[,,"NA","NA"],"AE",971,"00","0",,,"0",,,,[[,"([2-4679])(\\d{3})(\\d{4})","$1 $2 $3",["[2-4679][2-8]"],"0$1","",0],[,"(5[0256])(\\d{3})(\\d{4})","$1 $2 $3",["5"],"0$1","",0],[,"([479]00)(\\d)(\\d{5})","$1 $2 $3",["[479]0"],"$1","",0],[,"([68]00)(\\d{2,9})","$1 $2",["60|8"],"$1","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"600[25]\\d{5}","\\d{9}",,,"600212345"],,,[,,"NA","NA"]],AF:[,[,,"[2-7]\\d{8}","\\d{7,9}"],[,,"(?:[25][0-8]|[34][0-4]|6[0-5])[2-9]\\d{6}","\\d{7,9}",
|
|
|
,,"234567890"],[,,"7(?:[05-9]\\d{7}|29\\d{6})","\\d{9}",,,"701234567"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"AF",93,"00","0",,,"0",,,,[[,"([2-7]\\d)(\\d{3})(\\d{4})","$1 $2 $3",["[2-6]|7[013-9]"],"0$1","",0],[,"(729)(\\d{3})(\\d{3})","$1 $2 $3",["729"],"0$1","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],AG:[,[,,"[2589]\\d{9}","\\d{7}(?:\\d{3})?"],[,,"268(?:4(?:6[0-38]|84)|56[0-2])\\d{4}","\\d{7}(?:\\d{3})?",,,"2684601234"],[,,"268(?:464|7(?:2[0-9]|64|7[0-689]|8[02-68]))\\d{4}",
|
|
|
"\\d{10}",,,"2684641234"],[,,"8(?:00|44|55|66|77|88)[2-9]\\d{6}","\\d{10}",,,"8002123456"],[,,"900[2-9]\\d{6}","\\d{10}",,,"9002123456"],[,,"NA","NA"],[,,"5(?:00|33|44|66|77)[2-9]\\d{6}","\\d{10}",,,"5002345678"],[,,"26848[01]\\d{4}","\\d{10}",,,"2684801234"],"AG",1,"011","1",,,"1",,,,,,[,,"26840[69]\\d{4}","\\d{10}",,,"2684061234"],,"268",[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],AI:[,[,,"[2589]\\d{9}","\\d{7}(?:\\d{3})?"],[,,"2644(?:6[12]|9[78])\\d{4}","\\d{7}(?:\\d{3})?",,,"2644612345"],[,,
|
|
|
"264(?:235|476|5(?:3[6-9]|8[1-4])|7(?:29|72))\\d{4}","\\d{10}",,,"2642351234"],[,,"8(?:00|44|55|66|77|88)[2-9]\\d{6}","\\d{10}",,,"8002123456"],[,,"900[2-9]\\d{6}","\\d{10}",,,"9002123456"],[,,"NA","NA"],[,,"5(?:00|33|44|66|77)[2-9]\\d{6}","\\d{10}",,,"5002345678"],[,,"NA","NA"],"AI",1,"011","1",,,"1",,,,,,[,,"NA","NA"],,"264",[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],AL:[,[,,"[2-57]\\d{7}|6\\d{8}|8\\d{5,7}|9\\d{5}","\\d{5,9}"],[,,"(?:2(?:[168][1-9]|[247]\\d|9[1-7])|3(?:1[1-3]|[2-6]\\d|[79][1-8]|8[1-9])|4\\d{2}|5(?:1[1-4]|[2-578]\\d|6[1-5]|9[1-7])|8(?:[19][1-5]|[2-6]\\d|[78][1-7]))\\d{5}",
|
|
|
"\\d{5,8}",,,"22345678"],[,,"6[6-9]\\d{7}","\\d{9}",,,"661234567"],[,,"800\\d{4}","\\d{7}",,,"8001234"],[,,"900\\d{3}","\\d{6}",,,"900123"],[,,"808\\d{3}","\\d{6}",,,"808123"],[,,"700\\d{5}","\\d{8}",,,"70012345"],[,,"NA","NA"],"AL",355,"00","0",,,"0",,,,[[,"(4)(\\d{3})(\\d{4})","$1 $2 $3",["4[0-6]"],"0$1","",0],[,"(6[6-9])(\\d{3})(\\d{4})","$1 $2 $3",["6"],"0$1","",0],[,"(\\d{2})(\\d{3})(\\d{3})","$1 $2 $3",["[2358][2-5]|4[7-9]"],"0$1","",0],[,"(\\d{3})(\\d{3,5})","$1 $2",["[235][16-9]|8[016-9]|[79]"],
|
|
|
"0$1","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],AM:[,[,,"[1-9]\\d{7}","\\d{5,8}"],[,,"(?:1[01]\\d|2(?:2[2-46]|3[1-8]|4[2-69]|5[2-7]|6[1-9]|8[1-7])|3[12]2|47\\d)\\d{5}","\\d{5,8}",,,"10123456"],[,,"(?:4[139]|55|77|9[1-9])\\d{6}","\\d{8}",,,"77123456"],[,,"800\\d{5}","\\d{8}",,,"80012345"],[,,"90[016]\\d{5}","\\d{8}",,,"90012345"],[,,"80[1-4]\\d{5}","\\d{8}",,,"80112345"],[,,"NA","NA"],[,,"60[2-6]\\d{5}","\\d{8}",,,"60271234"],"AM",374,"00","0",,,"0",,,,[[,"(\\d{2})(\\d{6})",
|
|
|
"$1 $2",["1|47"],"(0$1)","",0],[,"(\\d{2})(\\d{6})","$1 $2",["4[139]|[5-7]|9[1-9]"],"0$1","",0],[,"(\\d{3})(\\d{5})","$1 $2",["[23]"],"(0$1)","",0],[,"(\\d{3})(\\d{2})(\\d{3})","$1 $2 $3",["8|90"],"0 $1","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],AO:[,[,,"[29]\\d{8}","\\d{9}"],[,,"2\\d(?:[26-9]\\d|\\d[26-9])\\d{5}","\\d{9}",,,"222123456"],[,,"9[1-49]\\d{7}","\\d{9}",,,"923123456"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"AO",244,"00",,,,,,,,
|
|
|
[[,"(\\d{3})(\\d{3})(\\d{3})","$1 $2 $3",,"","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],AR:[,[,,"11\\d{8}|[2368]\\d{9}|9\\d{10}","\\d{6,11}"],[,,"11\\d{8}|(?:2(?:2(?:[013]\\d|2[13-79]|4[1-6]|5[2457]|6[124-8]|7[1-4]|8[13-6]|9[1267])|3(?:1[467]|2[03-6]|3[13-8]|[49][2-6]|5[2-8]|[067]\\d)|4(?:7[3-8]|9\\d)|6(?:[01346]\\d|2[24-6]|5[15-8])|80\\d|9(?:[0124789]\\d|3[1-6]|5[234]|6[2-46]))|3(?:3(?:2[79]|6\\d|8[2578])|4(?:[78]\\d|0[0124-9]|[1-35]\\d|4[24-7]|6[02-9]|9[123678])|5(?:[138]\\d|2[1245]|4[1-9]|6[2-4]|7[1-6])|6[24]\\d|7(?:[0469]\\d|1[1568]|2[013-9]|3[145]|5[14-8]|7[2-57]|8[0-24-9])|8(?:[013578]\\d|2[15-7]|4[13-6]|6[1-357-9]|9[124]))|670\\d)\\d{6}",
|
|
|
"\\d{6,10}",,,"1123456789"],[,,"675\\d{7}|9(?:11[2-9]\\d{7}|(?:2(?:2[013]|3[067]|49|6[01346]|80|9[147-9])|3(?:36|4[12358]|5[138]|6[24]|7[069]|8[013578]))[2-9]\\d{6}|\\d{4}[2-9]\\d{5})","\\d{6,11}",,,"91123456789"],[,,"800\\d{7}","\\d{10}",,,"8001234567"],[,,"60[04579]\\d{7}","\\d{10}",,,"6001234567"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"AR",54,"00","0",,,"0?(?:(11|2(?:2(?:02?|[13]|2[13-79]|4[1-6]|5[2457]|6[124-8]|7[1-4]|8[13-6]|9[1267])|3(?:02?|1[467]|2[03-6]|3[13-8]|[49][2-6]|5[2-8]|[67])|4(?:7[3-578]|9)|6(?:[0136]|2[24-6]|4[6-8]?|5[15-8])|80|9(?:0[1-3]|[19]|2\\d|3[1-6]|4[02568]?|5[2-4]|6[2-46]|72?|8[23]?))|3(?:3(?:2[79]|6|8[2578])|4(?:0[124-9]|[12]|3[5-8]?|4[24-7]|5[4-68]?|6[02-9]|7[126]|8[2379]?|9[1-36-8])|5(?:1|2[1245]|3[237]?|4[1-46-9]|6[2-4]|7[1-6]|8[2-5]?)|6[24]|7(?:1[1568]|2[15]|3[145]|4[13]|5[14-8]|[069]|7[2-57]|8[126])|8(?:[01]|2[15-7]|3[2578]?|4[13-6]|5[4-8]?|6[1-357-9]|7[36-8]?|8[5-8]?|9[124])))?15)?",
|
|
|
"9$1",,,[[,"([68]\\d{2})(\\d{3})(\\d{4})","$1-$2-$3",["[68]"],"0$1","",0],[,"(\\d{2})(\\d{4})","$1-$2",["[2-9]"],"$1","",0],[,"(\\d{3})(\\d{4})","$1-$2",["[2-9]"],"$1","",0],[,"(\\d{4})(\\d{4})","$1-$2",["[2-9]"],"$1","",0],[,"(9)(11)(\\d{4})(\\d{4})","$2 15-$3-$4",["911"],"0$1","",0],[,"(9)(\\d{3})(\\d{3})(\\d{4})","$2 15-$3-$4",["9(?:2[234689]|3[3-8])","9(?:2(?:2[013]|3[067]|49|6[01346]|80|9[147-9])|3(?:36|4[12358]|5[138]|6[24]|7[069]|8[013578]))","9(?:2(?:2[013]|3[067]|49|6[01346]|80|9(?:[179]|4[13479]|8[014-9]))|3(?:36|4[12358]|5(?:[18]|3[014-689])|6[24]|7[069]|8(?:[01]|3[013469]|5[0-39]|7[0-2459]|8[0-49])))"],
|
|
|
"0$1","",0],[,"(9)(\\d{4})(\\d{2})(\\d{4})","$2 15-$3-$4",["9[23]"],"0$1","",0],[,"(11)(\\d{4})(\\d{4})","$1 $2-$3",["1"],"0$1","",1],[,"(\\d{3})(\\d{3})(\\d{4})","$1 $2-$3",["2(?:2[013]|3[067]|49|6[01346]|80|9[147-9])|3(?:36|4[12358]|5[138]|6[24]|7[069]|8[013578])","2(?:2[013]|3[067]|49|6[01346]|80|9(?:[179]|4[13479]|8[014-9]))|3(?:36|4[12358]|5(?:[18]|3[0-689])|6[24]|7[069]|8(?:[01]|3[013469]|5[0-39]|7[0-2459]|8[0-49]))"],"0$1","",1],[,"(\\d{4})(\\d{2})(\\d{4})","$1 $2-$3",["[23]"],"0$1","",1],
|
|
|
[,"(\\d{3})","$1",["1[012]|911"],"$1","",0]],[[,"([68]\\d{2})(\\d{3})(\\d{4})","$1-$2-$3",["[68]"],"0$1","",0],[,"(9)(11)(\\d{4})(\\d{4})","$1 $2 $3-$4",["911"]],[,"(9)(\\d{3})(\\d{3})(\\d{4})","$1 $2 $3-$4",["9(?:2[234689]|3[3-8])","9(?:2(?:2[013]|3[067]|49|6[01346]|80|9[147-9])|3(?:36|4[12358]|5[138]|6[24]|7[069]|8[013578]))","9(?:2(?:2[013]|3[067]|49|6[01346]|80|9(?:[179]|4[13479]|8[014-9]))|3(?:36|4[12358]|5(?:[18]|3[014-689])|6[24]|7[069]|8(?:[01]|3[013469]|5[0-39]|7[0-2459]|8[0-49])))"]],[,
|
|
|
"(9)(\\d{4})(\\d{2})(\\d{4})","$1 $2 $3-$4",["9[23]"]],[,"(11)(\\d{4})(\\d{4})","$1 $2-$3",["1"],"0$1","",1],[,"(\\d{3})(\\d{3})(\\d{4})","$1 $2-$3",["2(?:2[013]|3[067]|49|6[01346]|80|9[147-9])|3(?:36|4[12358]|5[138]|6[24]|7[069]|8[013578])","2(?:2[013]|3[067]|49|6[01346]|80|9(?:[179]|4[13479]|8[014-9]))|3(?:36|4[12358]|5(?:[18]|3[0-689])|6[24]|7[069]|8(?:[01]|3[013469]|5[0-39]|7[0-2459]|8[0-49]))"],"0$1","",1],[,"(\\d{4})(\\d{2})(\\d{4})","$1 $2-$3",["[23]"],"0$1","",1]],[,,"NA","NA"],,,[,,"810\\d{7}",
|
|
|
"\\d{10}",,,"8101234567"],[,,"810\\d{7}","\\d{10}",,,"8101234567"],,,[,,"NA","NA"]],AS:[,[,,"[5689]\\d{9}","\\d{7}(?:\\d{3})?"],[,,"6846(?:22|33|44|55|77|88|9[19])\\d{4}","\\d{7}(?:\\d{3})?",,,"6846221234"],[,,"684(?:25[2468]|7(?:3[13]|70))\\d{4}","\\d{10}",,,"6847331234"],[,,"8(?:00|44|55|66|77|88)[2-9]\\d{6}","\\d{10}",,,"8002123456"],[,,"900[2-9]\\d{6}","\\d{10}",,,"9002123456"],[,,"NA","NA"],[,,"5(?:00|33|44|66|77)[2-9]\\d{6}","\\d{10}",,,"5002345678"],[,,"NA","NA"],"AS",1,"011","1",,,"1",,,,
|
|
|
,,[,,"NA","NA"],,"684",[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],AT:[,[,,"[1-9]\\d{3,12}","\\d{3,13}"],[,,"1\\d{3,12}|(?:2(?:1[467]|2[13-8]|5[2357]|6[1-46-8]|7[1-8]|8[124-7]|9[1458])|3(?:1[1-8]|3[23568]|4[5-7]|5[1378]|6[1-38]|8[3-68])|4(?:2[1-8]|35|63|7[1368]|8[2457])|5(?:12|2[1-8]|3[357]|4[147]|5[12578]|6[37])|6(?:13|2[1-47]|4[1-35-8]|5[468]|62)|7(?:2[1-8]|3[25]|4[13478]|5[68]|6[16-8]|7[1-6]|9[45]))\\d{3,10}","\\d{3,13}",,,"1234567890"],[,,"6(?:44|5[0-3579]|6[013-9]|[7-9]\\d)\\d{4,10}","\\d{7,13}",
|
|
|
,,"644123456"],[,,"80[02]\\d{6,10}","\\d{9,13}",,,"800123456"],[,,"(?:711|9(?:0[01]|3[019]))\\d{6,10}","\\d{9,13}",,,"900123456"],[,,"8(?:10|2[018])\\d{6,10}","\\d{9,13}",,,"810123456"],[,,"NA","NA"],[,,"780\\d{6,10}","\\d{9,13}",,,"780123456"],"AT",43,"00","0",,,"0",,,,[[,"(1)(\\d{3,12})","$1 $2",["1"],"0$1","",0],[,"(5\\d)(\\d{3,5})","$1 $2",["5[079]"],"0$1","",0],[,"(5\\d)(\\d{3})(\\d{3,4})","$1 $2 $3",["5[079]"],"0$1","",0],[,"(5\\d)(\\d{4})(\\d{4,7})","$1 $2 $3",["5[079]"],"0$1","",0],[,"(\\d{3})(\\d{3,10})",
|
|
|
"$1 $2",["316|46|51|732|6(?:44|5[0-3579]|[6-9])|7(?:1|[28]0)|[89]"],"0$1","",0],[,"(\\d{4})(\\d{3,9})","$1 $2",["2|3(?:1[1-578]|[3-8])|4[2378]|5[2-6]|6(?:[12]|4[1-35-9]|5[468])|7(?:2[1-8]|35|4[1-8]|[5-79])"],"0$1","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"5(?:(?:0[1-9]|17)\\d{2,10}|[79]\\d{3,11})|720\\d{6,10}","\\d{5,13}",,,"50123"],,,[,,"NA","NA"]],AU:[,[,,"[1-578]\\d{5,9}","\\d{6,10}"],[,,"[237]\\d{8}|8(?:[68]\\d{3}|7[0-69]\\d{2}|9(?:[02-9]\\d{2}|1(?:[0-57-9]\\d|6[0135-9])))\\d{4}","\\d{8,9}",,
|
|
|
,"212345678"],[,,"14(?:5\\d|71)\\d{5}|4(?:[0-2]\\d|3[0-57-9]|4[47-9]|5[0-25-9]|6[6-9]|7[03-9]|8[17-9]|9[017-9])\\d{6}","\\d{9}",,,"412345678"],[,,"180(?:0\\d{3}|2)\\d{3}","\\d{7,10}",,,"1800123456"],[,,"190[0126]\\d{6}","\\d{10}",,,"1900123456"],[,,"13(?:00\\d{2})?\\d{4}","\\d{6,10}",,,"1300123456"],[,,"500\\d{6}","\\d{9}",,,"500123456"],[,,"550\\d{6}","\\d{9}",,,"550123456"],"AU",61,"(?:14(?:1[14]|34|4[17]|[56]6|7[47]|88))?001[14-689]","0",,,"0",,"0011",,[[,"([2378])(\\d{4})(\\d{4})","$1 $2 $3",
|
|
|
["[2378]"],"(0$1)","",0],[,"(\\d{3})(\\d{3})(\\d{3})","$1 $2 $3",["[45]|14"],"0$1","",0],[,"(16)(\\d{3})(\\d{2,4})","$1 $2 $3",["16"],"0$1","",0],[,"(1[389]\\d{2})(\\d{3})(\\d{3})","$1 $2 $3",["1(?:[38]0|90)","1(?:[38]00|90)"],"$1","",0],[,"(180)(2\\d{3})","$1 $2",["180","1802"],"$1","",0],[,"(19\\d)(\\d{3})","$1 $2",["19[13]"],"$1","",0],[,"(19\\d{2})(\\d{4})","$1 $2",["19[67]"],"$1","",0],[,"(13)(\\d{2})(\\d{2})","$1 $2 $3",["13[1-9]"],"$1","",0]],,[,,"16\\d{3,7}","\\d{5,9}",,,"1612345"],1,,[,,
|
|
|
"1(?:3(?:\\d{4}|00\\d{6})|80(?:0\\d{6}|2\\d{3}))","\\d{6,10}",,,"1300123456"],[,,"NA","NA"],,,[,,"NA","NA"]],AW:[,[,,"[25-9]\\d{6}","\\d{7}"],[,,"5(?:2\\d|8[1-9])\\d{4}","\\d{7}",,,"5212345"],[,,"(?:5(?:6\\d|9[2-478])|6(?:[039]0|22|4[01]|6[0-2])|7[34]\\d|9(?:6[45]|9[4-8]))\\d{4}","\\d{7}",,,"5601234"],[,,"800\\d{4}","\\d{7}",,,"8001234"],[,,"900\\d{4}","\\d{7}",,,"9001234"],[,,"NA","NA"],[,,"NA","NA"],[,,"28\\d{5}|501\\d{4}","\\d{7}",,,"5011234"],"AW",297,"00",,,,,,,,[[,"(\\d{3})(\\d{4})","$1 $2",
|
|
|
,"","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],AX:[,[,,"[135]\\d{5,9}|[27]\\d{4,9}|4\\d{5,10}|6\\d{7,8}|8\\d{6,9}","\\d{5,12}"],[,,"18[1-8]\\d{3,9}","\\d{6,12}",,,"1812345678"],[,,"4\\d{5,10}|50\\d{4,8}","\\d{6,11}",,,"412345678"],[,,"800\\d{4,7}","\\d{7,10}",,,"8001234567"],[,,"[67]00\\d{5,6}","\\d{8,9}",,,"600123456"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"AX",358,"00|99[049]","0",,,"0",,,,,,[,,"NA","NA"],,,[,,"[13]00\\d{3,7}|2(?:0(?:0\\d{3,7}|2[023]\\d{1,6}|9[89]\\d{1,6}))|60(?:[12]\\d{5,6}|6\\d{7})|7(?:1\\d{7}|3\\d{8}|5[03-9]\\d{2,7})",
|
|
|
"\\d{5,10}",,,"100123"],[,,"[13]0\\d{4,8}|2(?:0(?:[016-8]\\d{3,7}|[2-59]\\d{2,7})|9\\d{4,8})|60(?:[12]\\d{5,6}|6\\d{7})|7(?:1\\d{7}|3\\d{8}|5[03-9]\\d{2,7})","\\d{5,10}",,,"10112345"],,,[,,"NA","NA"]],AZ:[,[,,"[1-9]\\d{8}","\\d{7,9}"],[,,"(?:1[28]\\d|2(?:02|1[24]|2[2-4]|33|[45]2|6[23])|365)\\d{6}","\\d{7,9}",,,"123123456"],[,,"(?:4[04]|5[015]|60|7[07])\\d{7}","\\d{9}",,,"401234567"],[,,"88\\d{7}","\\d{9}",,,"881234567"],[,,"900200\\d{3}","\\d{9}",,,"900200123"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA",
|
|
|
"NA"],"AZ",994,"00","0",,,"0",,,,[[,"(\\d{2})(\\d{3})(\\d{2})(\\d{2})","$1 $2 $3 $4",["(?:1[28]|2(?:[45]2|[0-36])|365)"],"(0$1)","",0],[,"(\\d{2})(\\d{3})(\\d{2})(\\d{2})","$1 $2 $3 $4",["[4-8]"],"0$1","",0],[,"(\\d{3})(\\d{2})(\\d{2})(\\d{2})","$1 $2 $3 $4",["9"],"0$1","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],BA:[,[,,"[3-9]\\d{7,8}","\\d{6,9}"],[,,"(?:[35]\\d|49)\\d{6}","\\d{6,8}",,,"30123456"],[,,"6(?:03|44|71|[1-356])\\d{6}","\\d{8,9}",,,"61123456"],[,,"8[08]\\d{6}",
|
|
|
"\\d{8}",,,"80123456"],[,,"9[0246]\\d{6}","\\d{8}",,,"90123456"],[,,"8[12]\\d{6}","\\d{8}",,,"82123456"],[,,"NA","NA"],[,,"NA","NA"],"BA",387,"00","0",,,"0",,,,[[,"(\\d{2})(\\d{3})(\\d{3})","$1 $2-$3",["[3-5]"],"0$1","",0],[,"(\\d{2})(\\d{3})(\\d{3})","$1 $2 $3",["6[1-356]|[7-9]"],"0$1","",0],[,"(\\d{2})(\\d{2})(\\d{2})(\\d{3})","$1 $2 $3 $4",["6[047]"],"0$1","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"70[23]\\d{5}","\\d{8}",,,"70223456"],,,[,,"NA","NA"]],BB:[,[,,"[2589]\\d{9}","\\d{7}(?:\\d{3})?"],
|
|
|
[,,"246[2-9]\\d{6}","\\d{7}(?:\\d{3})?",,,"2462345678"],[,,"246(?:(?:2[346]|45|82)\\d|25[0-4])\\d{4}","\\d{10}",,,"2462501234"],[,,"8(?:00|44|55|66|77|88)[2-9]\\d{6}","\\d{10}",,,"8002123456"],[,,"900[2-9]\\d{6}","\\d{10}",,,"9002123456"],[,,"NA","NA"],[,,"5(?:00|33|44|66|77)[2-9]\\d{6}","\\d{10}",,,"5002345678"],[,,"NA","NA"],"BB",1,"011","1",,,"1",,,,,,[,,"NA","NA"],,"246",[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],BD:[,[,,"[2-79]\\d{5,9}|1\\d{9}|8[0-7]\\d{4,8}","\\d{6,10}"],[,,"2(?:7(?:1[0-267]|2[0-289]|3[0-29]|[46][01]|5[1-3]|7[017]|91)|8(?:0[125]|[139][1-6]|2[0157-9]|6[1-35]|7[1-5]|8[1-8])|9(?:0[0-2]|1[1-4]|2[568]|3[3-6]|5[5-7]|6[0167]|7[15]|8[016-8]))\\d{4}|3(?:12?[5-7]\\d{2}|0(?:2(?:[025-79]\\d|[348]\\d{1,2})|3(?:[2-4]\\d|[56]\\d?))|2(?:1\\d{2}|2(?:[12]\\d|[35]\\d{1,2}|4\\d?))|3(?:1\\d{2}|2(?:[2356]\\d|4\\d{1,2}))|4(?:1\\d{2}|2(?:2\\d{1,2}|[47]|5\\d{2}))|5(?:1\\d{2}|29)|[67]1\\d{2}|8(?:1\\d{2}|2(?:2\\d{2}|3|4\\d)))\\d{3}|4(?:0(?:2(?:[09]\\d|7)|33\\d{2})|1\\d{3}|2(?:1\\d{2}|2(?:[25]\\d?|[348]\\d|[67]\\d{1,2}))|3(?:1\\d{2}(?:\\d{2})?|2(?:[045]\\d|[236-9]\\d{1,2})|32\\d{2})|4(?:[18]\\d{2}|2(?:[2-46]\\d{2}|3)|5[25]\\d{2})|5(?:1\\d{2}|2(?:3\\d|5))|6(?:[18]\\d{2}|2(?:3(?:\\d{2})?|[46]\\d{1,2}|5\\d{2}|7\\d)|5(?:3\\d?|4\\d|[57]\\d{1,2}|6\\d{2}|8))|71\\d{2}|8(?:[18]\\d{2}|23\\d{2}|54\\d{2})|9(?:[18]\\d{2}|2[2-5]\\d{2}|53\\d{1,2}))\\d{3}|5(?:02[03489]\\d{2}|1\\d{2}|2(?:1\\d{2}|2(?:2(?:\\d{2})?|[457]\\d{2}))|3(?:1\\d{2}|2(?:[37](?:\\d{2})?|[569]\\d{2}))|4(?:1\\d{2}|2[46]\\d{2})|5(?:1\\d{2}|26\\d{1,2})|6(?:[18]\\d{2}|2|53\\d{2})|7(?:1|24)\\d{2}|8(?:1|26)\\d{2}|91\\d{2})\\d{3}|6(?:0(?:1\\d{2}|2(?:3\\d{2}|4\\d{1,2}))|2(?:2[2-5]\\d{2}|5(?:[3-5]\\d{2}|7)|8\\d{2})|3(?:1|2[3478])\\d{2}|4(?:1|2[34])\\d{2}|5(?:1|2[47])\\d{2}|6(?:[18]\\d{2}|6(?:2(?:2\\d|[34]\\d{2})|5(?:[24]\\d{2}|3\\d|5\\d{1,2})))|72[2-5]\\d{2}|8(?:1\\d{2}|2[2-5]\\d{2})|9(?:1\\d{2}|2[2-6]\\d{2}))\\d{3}|7(?:(?:02|[3-589]1|6[12]|72[24])\\d{2}|21\\d{3}|32)\\d{3}|8(?:(?:4[12]|[5-7]2|1\\d?)|(?:0|3[12]|[5-7]1|217)\\d)\\d{4}|9(?:[35]1|(?:[024]2|81)\\d|(?:1|[24]1)\\d{2})\\d{3}",
|
|
|
"\\d{6,9}",,,"27111234"],[,,"(?:1[13-9]\\d|(?:3[78]|44)[02-9]|6(?:44|6[02-9]))\\d{7}","\\d{10}",,,"1812345678"],[,,"80[03]\\d{7}","\\d{10}",,,"8001234567"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"96(?:0[49]|1[0-4]|6[69])\\d{6}","\\d{10}",,,"9604123456"],"BD",880,"00[12]?","0",,,"0",,"00",,[[,"(2)(\\d{7})","$1-$2",["2"],"0$1","",0],[,"(\\d{2})(\\d{4,6})","$1-$2",["[3-79]1"],"0$1","",0],[,"(\\d{4})(\\d{3,6})","$1-$2",["1|3(?:0|[2-58]2)|4(?:0|[25]2|3[23]|[4689][25])|5(?:[02-578]2|6[25])|6(?:[0347-9]2|[26][25])|7[02-9]2|8(?:[023][23]|[4-7]2)|9(?:[02][23]|[458]2|6[016])"],
|
|
|
"0$1","",0],[,"(\\d{3})(\\d{3,7})","$1-$2",["[3-79][2-9]|8"],"0$1","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],BE:[,[,,"[1-9]\\d{7,8}","\\d{8,9}"],[,,"(?:1[0-69]|[49][23]|5\\d|6[013-57-9]|71|8[0-79])[1-9]\\d{5}|[23][2-8]\\d{6}","\\d{8}",,,"12345678"],[,,"4(?:[679]\\d|8[03-9])\\d{6}","\\d{9}",,,"470123456"],[,,"800\\d{5}","\\d{8}",,,"80012345"],[,,"(?:70[2-7]|90\\d)\\d{5}","\\d{8}",,,"90123456"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"BE",32,"00","0",,,"0",,,,[[,"(4[6-9]\\d)(\\d{2})(\\d{2})(\\d{2})",
|
|
|
"$1 $2 $3 $4",["4[6-9]"],"0$1","",0],[,"([2-49])(\\d{3})(\\d{2})(\\d{2})","$1 $2 $3 $4",["[23]|[49][23]"],"0$1","",0],[,"([15-8]\\d)(\\d{2})(\\d{2})(\\d{2})","$1 $2 $3 $4",["[156]|7[018]|8(?:0[1-9]|[1-79])"],"0$1","",0],[,"([89]\\d{2})(\\d{2})(\\d{3})","$1 $2 $3",["(?:80|9)0"],"0$1","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"78\\d{6}","\\d{8}",,,"78123456"],,,[,,"NA","NA"]],BF:[,[,,"[24-7]\\d{7}","\\d{8}"],[,,"(?:20(?:49|5[23]|9[016-9])|40(?:4[569]|5[4-6]|7[0179])|50(?:[34]\\d|50))\\d{4}","\\d{8}",
|
|
|
,,"20491234"],[,,"6(?:[0-689]\\d|7[0-5])\\d{5}|7\\d{7}","\\d{8}",,,"70123456"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"BF",226,"00",,,,,,,,[[,"(\\d{2})(\\d{2})(\\d{2})(\\d{2})","$1 $2 $3 $4",,"","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],BG:[,[,,"[23567]\\d{5,7}|[489]\\d{6,8}","\\d{5,9}"],[,,"2(?:[0-8]\\d{5,6}|9\\d{4,6})|(?:[36]\\d|5[1-9]|8[1-6]|9[1-7])\\d{5,6}|(?:4(?:[124-7]\\d|3[1-6])|7(?:0[1-9]|[1-9]\\d))\\d{4,5}","\\d{5,8}",,,"2123456"],
|
|
|
[,,"(?:8[7-9]|98)\\d{7}|4(?:3[0789]|8\\d)\\d{5}","\\d{8,9}",,,"48123456"],[,,"800\\d{5}","\\d{8}",,,"80012345"],[,,"90\\d{6}","\\d{8}",,,"90123456"],[,,"NA","NA"],[,,"700\\d{5}","\\d{5,9}",,,"70012345"],[,,"NA","NA"],"BG",359,"00","0",,,"0",,,,[[,"(2)(\\d{5})","$1 $2",["29"],"0$1","",0],[,"(2)(\\d{3})(\\d{3,4})","$1 $2 $3",["2"],"0$1","",0],[,"(\\d{3})(\\d{4})","$1 $2",["43[124-7]|70[1-9]"],"0$1","",0],[,"(\\d{3})(\\d{3})(\\d{2})","$1 $2 $3",["43[124-7]|70[1-9]"],"0$1","",0],[,"(\\d{3})(\\d{2})(\\d{3})",
|
|
|
"$1 $2 $3",["[78]00"],"0$1","",0],[,"(\\d{2})(\\d{3})(\\d{2,3})","$1 $2 $3",["[356]|4[124-7]|7[1-9]|8[1-6]|9[1-7]"],"0$1","",0],[,"(\\d{2})(\\d{3})(\\d{3,4})","$1 $2 $3",["48|8[7-9]|9[08]"],"0$1","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],BH:[,[,,"[136-9]\\d{7}","\\d{8}"],[,,"(?:1(?:3[13-6]|6[0156]|7\\d)\\d|6(?:1[16]\\d|500|6(?:0\\d|3[12]|44|88)|9[69][69])|7(?:7\\d{2}|178))\\d{4}","\\d{8}",,,"17001234"],[,,"(?:3(?:[1-4679]\\d|5[0135]|8[0-48])\\d|6(?:3(?:00|33|6[16])|6(?:[69]\\d|3[03-9])))\\d{4}",
|
|
|
"\\d{8}",,,"36001234"],[,,"80\\d{6}","\\d{8}",,,"80123456"],[,,"(?:87|9[014578])\\d{6}","\\d{8}",,,"90123456"],[,,"84\\d{6}","\\d{8}",,,"84123456"],[,,"NA","NA"],[,,"NA","NA"],"BH",973,"00",,,,,,,,[[,"(\\d{4})(\\d{4})","$1 $2",,"","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],BI:[,[,,"[27]\\d{7}","\\d{8}"],[,,"22(?:2[0-7]|[3-5]0)\\d{4}","\\d{8}",,,"22201234"],[,,"(?:29|7[14-9])\\d{6}","\\d{8}",,,"79561234"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],
|
|
|
"BI",257,"00",,,,,,,,[[,"([27]\\d)(\\d{2})(\\d{2})(\\d{2})","$1 $2 $3 $4",,"","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],BJ:[,[,,"[2689]\\d{7}|7\\d{3}","\\d{4,8}"],[,,"2(?:02|1[037]|2[45]|3[68])\\d{5}","\\d{8}",,,"20211234"],[,,"(?:6[146-8]|9[03-9])\\d{6}","\\d{8}",,,"90011234"],[,,"7[3-5]\\d{2}","\\d{4}",,,"7312"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"857[58]\\d{4}","\\d{8}",,,"85751234"],"BJ",229,"00",,,,,,,,[[,"(\\d{2})(\\d{2})(\\d{2})(\\d{2})","$1 $2 $3 $4",,
|
|
|
"","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"81\\d{6}","\\d{8}",,,"81123456"],,,[,,"NA","NA"]],BL:[,[,,"[56]\\d{8}","\\d{9}"],[,,"590(?:2[7-9]|5[12]|87)\\d{4}","\\d{9}",,,"590271234"],[,,"690(?:0[0-7]|[1-9]\\d)\\d{4}","\\d{9}",,,"690301234"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"BL",590,"00","0",,,"0",,,,,,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],BM:[,[,,"[4589]\\d{9}","\\d{7}(?:\\d{3})?"],[,,"441(?:2(?:02|23|61|[3479]\\d)|[46]\\d{2}|5(?:4\\d|60|89)|824)\\d{4}",
|
|
|
"\\d{7}(?:\\d{3})?",,,"4412345678"],[,,"441(?:[37]\\d|5[0-39])\\d{5}","\\d{10}",,,"4413701234"],[,,"8(?:00|44|55|66|77|88)[2-9]\\d{6}","\\d{10}",,,"8002123456"],[,,"900[2-9]\\d{6}","\\d{10}",,,"9002123456"],[,,"NA","NA"],[,,"5(?:00|33|44|66|77)[2-9]\\d{6}","\\d{10}",,,"5002345678"],[,,"NA","NA"],"BM",1,"011","1",,,"1",,,,,,[,,"NA","NA"],,"441",[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],BN:[,[,,"[2-578]\\d{6}","\\d{7}"],[,,"2(?:[013-9]\\d|2[0-7])\\d{4}|[3-5]\\d{6}","\\d{7}",,,"2345678"],[,,"22[89]\\d{4}|[78]\\d{6}",
|
|
|
"\\d{7}",,,"7123456"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"BN",673,"00",,,,,,,,[[,"([2-578]\\d{2})(\\d{4})","$1 $2",,"","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],BO:[,[,,"[23467]\\d{7}","\\d{7,8}"],[,,"(?:2(?:2\\d{2}|5(?:11|[258]\\d|9[67])|6(?:12|2\\d|9[34])|8(?:2[34]|39|62))|3(?:3\\d{2}|4(?:6\\d|8[24])|8(?:25|42|5[257]|86|9[25])|9(?:2\\d|3[234]|4[248]|5[24]|6[2-6]|7\\d))|4(?:4\\d{2}|6(?:11|[24689]\\d|72)))\\d{4}","\\d{7,8}",,,"22123456"],
|
|
|
[,,"[67]\\d{7}","\\d{8}",,,"71234567"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"BO",591,"00(1\\d)?","0",,,"0(1\\d)?",,,,[[,"([234])(\\d{7})","$1 $2",["[234]"],"","0$CC $1",0],[,"([67]\\d{7})","$1",["[67]"],"","0$CC $1",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],BQ:[,[,,"[347]\\d{6}","\\d{7}"],[,,"(?:318[023]|416[023]|7(?:1[578]|50)\\d)\\d{3}","\\d{7}",,,"7151234"],[,,"(?:318[14-68]|416[15-9]|7(?:0[01]|7[07]|[89]\\d)\\d)\\d{3}","\\d{7}",,,"3181234"],
|
|
|
[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"BQ",599,"00",,,,,,,,,,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],BR:[,[,,"[1-46-9]\\d{7,10}|5\\d{8,9}","\\d{8,11}"],[,,"1[1-9][2-5]\\d{7}|(?:[4689][1-9]|2[12478]|3[1-578]|5[13-5]|7[13-579])[2-5]\\d{7}","\\d{8,11}",,,"1123456789"],[,,"1[1-9](?:7|9\\d)\\d{7}|(?:2[12478]|9[1-9])9?[6-9]\\d{7}|(?:3[1-578]|[468][1-9]|5[13-5]|7[13-579])[6-9]\\d{7}","\\d{10,11}",,,"11961234567"],[,,"800\\d{6,7}","\\d{8,11}",,,"800123456"],
|
|
|
[,,"[359]00\\d{6,7}","\\d{8,11}",,,"300123456"],[,,"[34]00\\d{5}","\\d{8}",,,"40041234"],[,,"NA","NA"],[,,"NA","NA"],"BR",55,"00(?:1[45]|2[135]|31|4[13])","0",,,"0(?:(1[245]|2[135]|31|4[13])(\\d{10,11}))?","$2",,,[[,"(\\d{4})(\\d{4})","$1-$2",["[2-9](?:[1-9]|0[1-9])"],"$1","",0],[,"(\\d{5})(\\d{4})","$1-$2",["9(?:[1-9]|0[1-9])"],"$1","",0],[,"(\\d{3,5})","$1",["1[125689]"],"$1","",0],[,"(\\d{2})(\\d{5})(\\d{4})","$1 $2-$3",["(?:1[1-9]|2[12478]|9[1-9])9"],"($1)","0 $CC ($1)",0],[,"(\\d{2})(\\d{4})(\\d{4})",
|
|
|
"$1 $2-$3",["[1-9][1-9]"],"($1)","0 $CC ($1)",0],[,"([34]00\\d)(\\d{4})","$1-$2",["[34]00"],"","",0],[,"([3589]00)(\\d{2,3})(\\d{4})","$1 $2 $3",["[3589]00"],"0$1","",0]],[[,"(\\d{2})(\\d{5})(\\d{4})","$1 $2-$3",["(?:1[1-9]|2[12478]|9[1-9])9"],"($1)","0 $CC ($1)",0],[,"(\\d{2})(\\d{4})(\\d{4})","$1 $2-$3",["[1-9][1-9]"],"($1)","0 $CC ($1)",0],[,"([34]00\\d)(\\d{4})","$1-$2",["[34]00"],"","",0],[,"([3589]00)(\\d{2,3})(\\d{4})","$1 $2 $3",["[3589]00"],"0$1","",0]],[,,"NA","NA"],,,[,,"[34]00\\d{5}",
|
|
|
"\\d{8}",,,"40041234"],[,,"NA","NA"],,,[,,"NA","NA"]],BS:[,[,,"[2589]\\d{9}","\\d{7}(?:\\d{3})?"],[,,"242(?:3(?:02|[236][1-9]|4[0-24-9]|5[0-68]|7[3467]|8[0-4]|9[2-467])|461|502|6(?:0[12]|12|7[67]|8[78]|9[89])|702)\\d{4}","\\d{7}(?:\\d{3})?",,,"2423456789"],[,,"242(?:3(?:5[79]|[79]5)|4(?:[2-4][1-9]|5[1-8]|6[2-8]|7\\d|81)|5(?:2[45]|3[35]|44|5[1-9]|65|77)|6[34]6|727)\\d{4}","\\d{10}",,,"2423591234"],[,,"242300\\d{4}|8(?:00|44|55|66|77|88)[2-9]\\d{6}","\\d{10}",,,"8002123456"],[,,"900[2-9]\\d{6}","\\d{10}",
|
|
|
,,"9002123456"],[,,"NA","NA"],[,,"5(?:00|33|44|66|77)[2-9]\\d{6}","\\d{10}",,,"5002345678"],[,,"NA","NA"],"BS",1,"011","1",,,"1",,,,,,[,,"NA","NA"],,"242",[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],BT:[,[,,"[1-8]\\d{6,7}","\\d{6,8}"],[,,"(?:2[3-6]|[34][5-7]|5[236]|6[2-46]|7[246]|8[2-4])\\d{5}","\\d{6,7}",,,"2345678"],[,,"[17]7\\d{6}","\\d{8}",,,"17123456"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"BT",975,"00",,,,,,,,[[,"([17]7)(\\d{2})(\\d{2})(\\d{2})","$1 $2 $3 $4",
|
|
|
["1|77"],"","",0],[,"([2-8])(\\d{3})(\\d{3})","$1 $2 $3",["[2-68]|7[246]"],"","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],BW:[,[,,"[2-79]\\d{6,7}","\\d{7,8}"],[,,"(?:2(?:4[0-48]|6[0-24]|9[0578])|3(?:1[0235-9]|55|6\\d|7[01]|9[0-57])|4(?:6[03]|7[1267]|9[0-5])|5(?:3[0389]|4[0489]|7[1-47]|88|9[0-49])|6(?:2[1-35]|5[149]|8[067]))\\d{4}","\\d{7}",,,"2401234"],[,,"7(?:[1-356]\\d|4[0-7]|7[014-7])\\d{5}","\\d{8}",,,"71123456"],[,,"NA","NA"],[,,"90\\d{5}","\\d{7}",,,"9012345"],[,,"NA",
|
|
|
"NA"],[,,"NA","NA"],[,,"79[12][01]\\d{4}","\\d{8}",,,"79101234"],"BW",267,"00",,,,,,,,[[,"(\\d{3})(\\d{4})","$1 $2",["[2-6]"],"","",0],[,"(7\\d)(\\d{3})(\\d{3})","$1 $2 $3",["7"],"","",0],[,"(90)(\\d{5})","$1 $2",["9"],"","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],BY:[,[,,"[1-4]\\d{8}|[89]\\d{9,10}","\\d{7,11}"],[,,"(?:1(?:5(?:1[1-5]|[24]\\d|6[2-4]|9[1-7])|6(?:[235]\\d|4[1-7])|7\\d{2})|2(?:1(?:[246]\\d|3[0-35-9]|5[1-9])|2(?:[235]\\d|4[0-8])|3(?:[26]\\d|3[02-79]|4[024-7]|5[03-7])))\\d{5}",
|
|
|
"\\d{7,9}",,,"152450911"],[,,"(?:2(?:5[5679]|9[1-9])|33\\d|44\\d)\\d{6}","\\d{9}",,,"294911911"],[,,"8(?:0[13]|20\\d)\\d{7}","\\d{10,11}",,,"8011234567"],[,,"(?:810|902)\\d{7}","\\d{10}",,,"9021234567"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"BY",375,"810","8",,,"8?0?",,"8~10",,[[,"(\\d{2})(\\d{3})(\\d{2})(\\d{2})","$1 $2-$3-$4",["17[0-3589]|2[4-9]|[34]","17(?:[02358]|1[0-2]|9[0189])|2[4-9]|[34]"],"8 0$1","",0],[,"(\\d{3})(\\d{2})(\\d{2})(\\d{2})","$1 $2-$3-$4",["1(?:5[24]|6[235]|7[467])|2(?:1[246]|2[25]|3[26])",
|
|
|
"1(?:5[24]|6(?:2|3[04-9]|5[0346-9])|7(?:[46]|7[37-9]))|2(?:1[246]|2[25]|3[26])"],"8 0$1","",0],[,"(\\d{4})(\\d{2})(\\d{3})","$1 $2-$3",["1(?:5[169]|6[3-5]|7[179])|2(?:1[35]|2[34]|3[3-5])","1(?:5[169]|6(?:3[1-3]|4|5[125])|7(?:1[3-9]|7[0-24-6]|9[2-7]))|2(?:1[35]|2[34]|3[3-5])"],"8 0$1","",0],[,"([89]\\d{2})(\\d{3})(\\d{4})","$1 $2 $3",["8[01]|9"],"8 $1","",0],[,"(8\\d{2})(\\d{4})(\\d{4})","$1 $2 $3",["82"],"8 $1","",0]],,[,,"NA","NA"],,,[,,"8(?:[013]|[12]0)\\d{8}|902\\d{7}","\\d{10,11}",,,"82012345678"],
|
|
|
[,,"NA","NA"],,,[,,"NA","NA"]],BZ:[,[,,"[2-8]\\d{6}|0\\d{10}","\\d{7}(?:\\d{4})?"],[,,"[234578][02]\\d{5}","\\d{7}",,,"2221234"],[,,"6[0-367]\\d{5}","\\d{7}",,,"6221234"],[,,"0800\\d{7}","\\d{11}",,,"08001234123"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"BZ",501,"00",,,,,,,,[[,"(\\d{3})(\\d{4})","$1-$2",["[2-8]"],"","",0],[,"(0)(800)(\\d{4})(\\d{3})","$1-$2-$3-$4",["0"],"","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],1,,[,,"NA","NA"]],CA:[,[,,"[2-9]\\d{9}|3\\d{6}","\\d{7}(?:\\d{3})?"],
|
|
|
[,,"(?:2(?:04|[23]6|[48]9|50)|3(?:06|43|65)|4(?:03|1[68]|3[178]|50)|5(?:06|1[49]|79|8[17])|6(?:0[04]|13|39|47)|7(?:0[59]|78|8[02])|8(?:[06]7|19|73)|90[25])[2-9]\\d{6}|310\\d{4}","\\d{7}(?:\\d{3})?",,,"2042345678"],[,,"(?:2(?:04|[23]6|[48]9|50)|3(?:06|43|65)|4(?:03|1[68]|3[178]|50)|5(?:06|1[49]|79|8[17])|6(?:0[04]|13|39|47)|7(?:0[59]|78|8[02])|8(?:[06]7|19|73)|90[25])[2-9]\\d{6}","\\d{7}(?:\\d{3})?",,,"2042345678"],[,,"8(?:00|44|55|66|77|88)[2-9]\\d{6}|310\\d{4}","\\d{7}(?:\\d{3})?",,,"8002123456"],
|
|
|
[,,"900[2-9]\\d{6}","\\d{10}",,,"9002123456"],[,,"NA","NA"],[,,"5(?:00|33|44|66|77)[2-9]\\d{6}","\\d{10}",,,"5002345678"],[,,"NA","NA"],"CA",1,"011","1",,,"1",,,,,,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],CC:[,[,,"[1458]\\d{5,9}","\\d{6,10}"],[,,"89162\\d{4}","\\d{8,9}",,,"891621234"],[,,"14(?:5\\d|71)\\d{5}|4(?:[0-2]\\d|3[0-57-9]|4[47-9]|5[0-25-9]|6[6-9]|7[03-9]|8[17-9]|9[017-9])\\d{6}","\\d{9}",,,"412345678"],[,,"1(?:80(?:0\\d{2})?|3(?:00\\d{2})?)\\d{4}","\\d{6,10}",,,"1800123456"],
|
|
|
[,,"190[0126]\\d{6}","\\d{10}",,,"1900123456"],[,,"NA","NA"],[,,"500\\d{6}","\\d{9}",,,"500123456"],[,,"550\\d{6}","\\d{9}",,,"550123456"],"CC",61,"(?:14(?:1[14]|34|4[17]|[56]6|7[47]|88))?001[14-689]","0",,,"0",,"0011",,,,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],CD:[,[,,"[2-6]\\d{6}|[18]\\d{6,8}|9\\d{8}","\\d{7,9}"],[,,"1(?:2\\d{7}|\\d{6})|[2-6]\\d{6}","\\d{7,9}",,,"1234567"],[,,"8(?:[0-2459]\\d{2}|8)\\d{5}|9[7-9]\\d{7}","\\d{7,9}",,,"991234567"],[,,"NA","NA"],[,,"NA","NA"],[,
|
|
|
,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"CD",243,"00","0",,,"0",,,,[[,"(\\d{2})(\\d{3})(\\d{4})","$1 $2 $3",["12"],"0$1","",0],[,"([89]\\d{2})(\\d{3})(\\d{3})","$1 $2 $3",["8[0-2459]|9"],"0$1","",0],[,"(\\d{2})(\\d{2})(\\d{3})","$1 $2 $3",["88"],"0$1","",0],[,"(\\d{2})(\\d{5})","$1 $2",["[1-6]"],"0$1","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],CF:[,[,,"[278]\\d{7}","\\d{8}"],[,,"2[12]\\d{6}","\\d{8}",,,"21612345"],[,,"7[0257]\\d{6}","\\d{8}",,,"70012345"],[,,"NA","NA"],[,
|
|
|
,"8776\\d{4}","\\d{8}",,,"87761234"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"CF",236,"00",,,,,,,,[[,"(\\d{2})(\\d{2})(\\d{2})(\\d{2})","$1 $2 $3 $4",,"","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],CG:[,[,,"[028]\\d{8}","\\d{9}"],[,,"222[1-589]\\d{5}","\\d{9}",,,"222123456"],[,,"0[14-6]\\d{7}","\\d{9}",,,"061234567"],[,,"800\\d{6}","\\d{9}",,,"800123456"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"CG",242,"00",,,,,,,,[[,"(\\d{2})(\\d{3})(\\d{4})","$1 $2 $3",
|
|
|
["[02]"],"","",0],[,"(\\d)(\\d{4})(\\d{4})","$1 $2 $3",["8"],"","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],1,,[,,"NA","NA"]],CH:[,[,,"[2-9]\\d{8}|860\\d{9}","\\d{9}(?:\\d{3})?"],[,,"(?:2[12467]|3[1-4]|4[134]|5[256]|6[12]|[7-9]1)\\d{7}","\\d{9}",,,"212345678"],[,,"7[5-9]\\d{7}","\\d{9}",,,"781234567"],[,,"800\\d{6}","\\d{9}",,,"800123456"],[,,"90[016]\\d{6}","\\d{9}",,,"900123456"],[,,"84[0248]\\d{6}","\\d{9}",,,"840123456"],[,,"878\\d{6}","\\d{9}",,,"878123456"],[,,"NA","NA"],"CH",41,"00",
|
|
|
"0",,,"0",,,,[[,"([2-9]\\d)(\\d{3})(\\d{2})(\\d{2})","$1 $2 $3 $4",["[2-7]|[89]1"],"0$1","",0],[,"([89]\\d{2})(\\d{3})(\\d{3})","$1 $2 $3",["8[047]|90"],"0$1","",0],[,"(\\d{3})(\\d{2})(\\d{3})(\\d{2})(\\d{2})","$1 $2 $3 $4 $5",["860"],"0$1","",0]],,[,,"74[0248]\\d{6}","\\d{9}",,,"740123456"],,,[,,"NA","NA"],[,,"5[18]\\d{7}","\\d{9}",,,"581234567"],,,[,,"860\\d{9}","\\d{12}",,,"860123456789"]],CI:[,[,,"[02-7]\\d{7}","\\d{8}"],[,,"(?:2(?:0[023]|1[02357]|[23][045]|4[03-5])|3(?:0[06]|1[069]|[2-4][07]|5[09]|6[08]))\\d{5}",
|
|
|
"\\d{8}",,,"21234567"],[,,"(?:0[1-9]|4[0-24-9]|5[4-9]|6[015-79]|7[57])\\d{6}","\\d{8}",,,"01234567"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"CI",225,"00",,,,,,,,[[,"(\\d{2})(\\d{2})(\\d{2})(\\d{2})","$1 $2 $3 $4",,"","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],1,,[,,"NA","NA"]],CK:[,[,,"[2-57]\\d{4}","\\d{5}"],[,,"(?:2\\d|3[13-7]|4[1-5])\\d{3}","\\d{5}",,,"21234"],[,,"(?:5[0-68]|7\\d)\\d{3}","\\d{5}",,,"71234"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA",
|
|
|
"NA"],[,,"NA","NA"],"CK",682,"00",,,,,,,,[[,"(\\d{2})(\\d{3})","$1 $2",,"","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],CL:[,[,,"(?:[2-9]|600|123)\\d{7,8}","\\d{7,11}"],[,,"2(?:2\\d{7}|1962\\d{4})|(?:3[2-5]|[47][1-35]|5[1-3578]|6[13-57])\\d{7}","\\d{7,9}",,,"221234567"],[,,"9[4-9]\\d{7}","\\d{8,9}",,,"961234567"],[,,"800\\d{6}|1230\\d{7}","\\d{9,11}",,,"800123456"],[,,"NA","NA"],[,,"600\\d{7,8}","\\d{10,11}",,,"6001234567"],[,,"NA","NA"],[,,"44\\d{7}","\\d{9}",,,"441234567"],
|
|
|
"CL",56,"(?:0|1(?:1[0-69]|2[0-57]|5[13-58]|69|7[0167]|8[018]))0","0",,,"0|(1(?:1[0-69]|2[0-57]|5[13-58]|69|7[0167]|8[018]))",,,,[[,"(\\d)(\\d{4})(\\d{4})","$1 $2 $3",["22"],"($1)","$CC ($1)",0],[,"(\\d{2})(\\d{3})(\\d{4})","$1 $2 $3",["[357]|4[1-35]|6[13-57]"],"($1)","$CC ($1)",0],[,"(9)(\\d{4})(\\d{4})","$1 $2 $3",["9"],"0$1","",0],[,"(44)(\\d{3})(\\d{4})","$1 $2 $3",["44"],"0$1","",0],[,"([68]00)(\\d{3})(\\d{3,4})","$1 $2 $3",["60|8"],"$1","",0],[,"(600)(\\d{3})(\\d{2})(\\d{3})","$1 $2 $3 $4",["60"],
|
|
|
"$1","",0],[,"(1230)(\\d{3})(\\d{4})","$1 $2 $3",["1"],"$1","",0],[,"(\\d{5})(\\d{4})","$1 $2",["219"],"($1)","$CC ($1)",0],[,"(\\d{4,5})","$1",["[1-9]"],"$1","",0]],[[,"(\\d)(\\d{4})(\\d{4})","$1 $2 $3",["22"],"($1)","$CC ($1)",0],[,"(\\d{2})(\\d{3})(\\d{4})","$1 $2 $3",["[357]|4[1-35]|6[13-57]"],"($1)","$CC ($1)",0],[,"(9)(\\d{4})(\\d{4})","$1 $2 $3",["9"],"0$1","",0],[,"(44)(\\d{3})(\\d{4})","$1 $2 $3",["44"],"0$1","",0],[,"([68]00)(\\d{3})(\\d{3,4})","$1 $2 $3",["60|8"],"$1","",0],[,"(600)(\\d{3})(\\d{2})(\\d{3})",
|
|
|
"$1 $2 $3 $4",["60"],"$1","",0],[,"(1230)(\\d{3})(\\d{4})","$1 $2 $3",["1"],"$1","",0],[,"(\\d{5})(\\d{4})","$1 $2",["219"],"($1)","$CC ($1)",0]],[,,"NA","NA"],,,[,,"600\\d{7,8}","\\d{10,11}",,,"6001234567"],[,,"NA","NA"],,,[,,"NA","NA"]],CM:[,[,,"[2357-9]\\d{7}","\\d{8}"],[,,"(?:22|33)\\d{6}","\\d{8}",,,"22123456"],[,,"[579]\\d{7}","\\d{8}",,,"71234567"],[,,"800\\d{5}","\\d{8}",,,"80012345"],[,,"88\\d{6}","\\d{8}",,,"88012345"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"CM",237,"00",,,,,,,,[[,"([2357-9]\\d)(\\d{2})(\\d{2})(\\d{2})",
|
|
|
"$1 $2 $3 $4",["[23579]|88"],"","",0],[,"(800)(\\d{2})(\\d{3})","$1 $2 $3",["80"],"","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],CN:[,[,,"[1-7]\\d{6,11}|8[0-357-9]\\d{6,9}|9\\d{7,9}","\\d{4,12}"],[,,"21(?:100\\d{2}|95\\d{3,4}|\\d{8,10})|(?:10|2[02-57-9]|3(?:11|7[179])|4(?:[15]1|3[12])|5(?:1\\d|2[37]|3[12]|51|7[13-79]|9[15])|7(?:31|5[457]|6[09]|91)|8(?:71|98))(?:100\\d{2}|95\\d{3,4}|\\d{8})|(?:3(?:1[02-9]|35|49|5\\d|7[02-68]|9[1-68])|4(?:1[02-9]|2[179]|3[3-9]|5[2-9]|6[4789]|7\\d|8[23])|5(?:3[03-9]|4[36]|5[02-9]|6[1-46]|7[028]|80|9[2-46-9])|6(?:3[1-5]|6[0238]|9[12])|7(?:01|[17]\\d|2[248]|3[04-9]|4[3-6]|5[0-3689]|6[2368]|9[02-9])|8(?:1[236-8]|2[5-7]|3\\d|5[1-9]|7[02-9]|8[3678]|9[1-7])|9(?:0[1-3689]|1[1-79]|[379]\\d|4[13]|5[1-5]))(?:100\\d{2}|95\\d{3,4}|\\d{7})|80(?:29|6[03578]|7[018]|81)\\d{4}",
|
|
|
"\\d{4,12}",,,"1012345678"],[,,"1(?:[38]\\d|4[57]|5[0-35-9]|7[06-8])\\d{8}","\\d{11}",,,"13123456789"],[,,"(?:10)?800\\d{7}","\\d{10,12}",,,"8001234567"],[,,"16[08]\\d{5}","\\d{8}",,,"16812345"],[,,"400\\d{7}|(?:10|2[0-57-9]|3(?:[157]\\d|35|49|9[1-68])|4(?:[17]\\d|2[179]|[35][1-9]|6[4789]|8[23])|5(?:[1357]\\d|2[37]|4[36]|6[1-46]|80|9[1-9])|6(?:3[1-5]|6[0238]|9[12])|7(?:01|[1579]\\d|2[248]|3[014-9]|4[3-6]|6[023689])|8(?:1[236-8]|2[5-7]|[37]\\d|5[1-9]|8[3678]|9[1-8])|9(?:0[1-3689]|1[1-79]|[379]\\d|4[13]|5[1-5]))96\\d{3,4}",
|
|
|
"\\d{7,10}",,,"4001234567"],[,,"NA","NA"],[,,"NA","NA"],"CN",86,"(1[1279]\\d{3})?00","0",,,"(1[1279]\\d{3})|0",,"00",,[[,"(80\\d{2})(\\d{4})","$1 $2",["80[2678]"],"0$1","$CC $1",1],[,"([48]00)(\\d{3})(\\d{4})","$1 $2 $3",["[48]00"],"","",0],[,"(\\d{5,6})","$1",["100|95"],"","",0],[,"(\\d{2})(\\d{5,6})","$1 $2",["(?:10|2\\d)[19]","(?:10|2\\d)(?:10|9[56])","(?:10|2\\d)(?:100|9[56])"],"0$1","$CC $1",0],[,"(\\d{3})(\\d{5,6})","$1 $2",["[3-9]","[3-9]\\d{2}[19]","[3-9]\\d{2}(?:10|9[56])"],"0$1","$CC $1",
|
|
|
0],[,"(\\d{3,4})(\\d{4})","$1 $2",["[2-9]"],"","",0],[,"(21)(\\d{4})(\\d{4,6})","$1 $2 $3",["21"],"0$1","$CC $1",1],[,"([12]\\d)(\\d{4})(\\d{4})","$1 $2 $3",["10[1-9]|2[02-9]","10[1-9]|2[02-9]","10(?:[1-79]|8(?:[1-9]|0[1-9]))|2[02-9]"],"0$1","$CC $1",1],[,"(\\d{3})(\\d{4})(\\d{4})","$1 $2 $3",["3(?:11|7[179])|4(?:[15]1|3[12])|5(?:1|2[37]|3[12]|51|7[13-79]|9[15])|7(?:31|5[457]|6[09]|91)|8(?:71|98)"],"0$1","$CC $1",1],[,"(\\d{3})(\\d{3})(\\d{4})","$1 $2 $3",["3(?:1[02-9]|35|49|5|7[02-68]|9[1-68])|4(?:1[02-9]|2[179]|[35][2-9]|6[4789]|7\\d|8[23])|5(?:3[03-9]|4[36]|5[02-9]|6[1-46]|7[028]|80|9[2-46-9])|6(?:3[1-5]|6[0238]|9[12])|7(?:01|[1579]|2[248]|3[04-9]|4[3-6]|6[2368])|8(?:1[236-8]|2[5-7]|3|5[1-9]|7[02-9]|8[3678]|9[1-7])|9(?:0[1-3689]|1[1-79]|[379]|4[13]|5[1-5])"],
|
|
|
"0$1","$CC $1",1],[,"(\\d{3})(\\d{4})(\\d{4})","$1 $2 $3",["1[3-578]"],"","$CC $1",0],[,"(10800)(\\d{3})(\\d{4})","$1 $2 $3",["108","1080","10800"],"","",0]],[[,"(80\\d{2})(\\d{4})","$1 $2",["80[2678]"],"0$1","$CC $1",1],[,"([48]00)(\\d{3})(\\d{4})","$1 $2 $3",["[48]00"],"","",0],[,"(\\d{2})(\\d{5,6})","$1 $2",["(?:10|2\\d)[19]","(?:10|2\\d)(?:10|9[56])","(?:10|2\\d)(?:100|9[56])"],"0$1","$CC $1",0],[,"(\\d{3})(\\d{5,6})","$1 $2",["[3-9]","[3-9]\\d{2}[19]","[3-9]\\d{2}(?:10|9[56])"],"0$1","$CC $1",
|
|
|
0],[,"(21)(\\d{4})(\\d{4,6})","$1 $2 $3",["21"],"0$1","$CC $1",1],[,"([12]\\d)(\\d{4})(\\d{4})","$1 $2 $3",["10[1-9]|2[02-9]","10[1-9]|2[02-9]","10(?:[1-79]|8(?:[1-9]|0[1-9]))|2[02-9]"],"0$1","$CC $1",1],[,"(\\d{3})(\\d{4})(\\d{4})","$1 $2 $3",["3(?:11|7[179])|4(?:[15]1|3[12])|5(?:1|2[37]|3[12]|51|7[13-79]|9[15])|7(?:31|5[457]|6[09]|91)|8(?:71|98)"],"0$1","$CC $1",1],[,"(\\d{3})(\\d{3})(\\d{4})","$1 $2 $3",["3(?:1[02-9]|35|49|5|7[02-68]|9[1-68])|4(?:1[02-9]|2[179]|[35][2-9]|6[4789]|7\\d|8[23])|5(?:3[03-9]|4[36]|5[02-9]|6[1-46]|7[028]|80|9[2-46-9])|6(?:3[1-5]|6[0238]|9[12])|7(?:01|[1579]|2[248]|3[04-9]|4[3-6]|6[2368])|8(?:1[236-8]|2[5-7]|3|5[1-9]|7[02-9]|8[3678]|9[1-7])|9(?:0[1-3689]|1[1-79]|[379]|4[13]|5[1-5])"],
|
|
|
"0$1","$CC $1",1],[,"(\\d{3})(\\d{4})(\\d{4})","$1 $2 $3",["1[3-578]"],"","$CC $1",0],[,"(10800)(\\d{3})(\\d{4})","$1 $2 $3",["108","1080","10800"],"","",0]],[,,"NA","NA"],,,[,,"(?:4|(?:10)?8)00\\d{7}","\\d{10,12}",,,"4001234567"],[,,"NA","NA"],,,[,,"NA","NA"]],CO:[,[,,"(?:[13]\\d{0,3}|[24-8])\\d{7}","\\d{7,11}"],[,,"[124-8][2-9]\\d{6}","\\d{8}",,,"12345678"],[,,"3(?:0[0-5]|1\\d|2[0-2]|5[01])\\d{7}","\\d{10}",,,"3211234567"],[,,"1800\\d{7}","\\d{11}",,,"18001234567"],[,,"19(?:0[01]|4[78])\\d{7}",
|
|
|
"\\d{11}",,,"19001234567"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"CO",57,"00(?:4(?:[14]4|56)|[579])","0",,,"0([3579]|4(?:44|56))?",,,,[[,"(\\d)(\\d{7})","$1 $2",["1(?:8[2-9]|9[0-3]|[2-7])|[24-8]","1(?:8[2-9]|9(?:09|[1-3])|[2-7])|[24-8]"],"($1)","0$CC $1",0],[,"(\\d{3})(\\d{7})","$1 $2",["3"],"","0$CC $1",0],[,"(1)(\\d{3})(\\d{7})","$1-$2-$3",["1(?:80|9[04])","1(?:800|9(?:0[01]|4[78]))"],"0$1","",0]],[[,"(\\d)(\\d{7})","$1 $2",["1(?:8[2-9]|9[0-3]|[2-7])|[24-8]","1(?:8[2-9]|9(?:09|[1-3])|[2-7])|[24-8]"],
|
|
|
"($1)","0$CC $1",0],[,"(\\d{3})(\\d{7})","$1 $2",["3"],"","0$CC $1",0],[,"(1)(\\d{3})(\\d{7})","$1 $2 $3",["1(?:80|9[04])","1(?:800|9(?:0[01]|4[78]))"]]],[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],CR:[,[,,"[24-9]\\d{7,9}","\\d{8,10}"],[,,"2[24-7]\\d{6}","\\d{8}",,,"22123456"],[,,"5(?:0[01]|7[0-3])\\d{5}|6(?:[0-2]\\d|30)\\d{5}|7[0-3]\\d{6}|8[3-9]\\d{6}","\\d{8}",,,"83123456"],[,,"800\\d{7}","\\d{10}",,,"8001234567"],[,,"90[059]\\d{7}","\\d{10}",,,"9001234567"],[,,"NA","NA"],[,,"NA",
|
|
|
"NA"],[,,"210[0-6]\\d{4}|4(?:0(?:0[01]\\d{4}|10[0-3]\\d{3}|2(?:00\\d{3}|900\\d{2})|3[01]\\d{4}|40\\d{4}|5\\d{5}|60\\d{4}|70[01]\\d{3}|8[0-2]\\d{4})|1[01]\\d{5}|20[0-3]\\d{4}|400\\d{4}|70[0-2]\\d{4})|5100\\d{4}","\\d{8}",,,"40001234"],"CR",506,"00",,,,"(19(?:0[012468]|1[09]|20|66|77|99))",,,,[[,"(\\d{4})(\\d{4})","$1 $2",["[24-7]|8[3-9]"],"","$CC $1",0],[,"(\\d{3})(\\d{3})(\\d{4})","$1-$2-$3",["[89]0"],"","$CC $1",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],CU:[,[,,"[2-57]\\d{5,7}",
|
|
|
"\\d{4,8}"],[,,"2[1-4]\\d{5,6}|3(?:1\\d{6}|[23]\\d{4,6})|4(?:[125]\\d{5,6}|[36]\\d{6}|[78]\\d{4,6})|7\\d{6,7}","\\d{4,8}",,,"71234567"],[,,"5\\d{7}","\\d{8}",,,"51234567"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"CU",53,"119","0",,,"0",,,,[[,"(\\d)(\\d{6,7})","$1 $2",["7"],"(0$1)","",0],[,"(\\d{2})(\\d{4,6})","$1 $2",["[2-4]"],"(0$1)","",0],[,"(\\d)(\\d{7})","$1 $2",["5"],"0$1","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],CV:[,[,,"[259]\\d{6}",
|
|
|
"\\d{7}"],[,,"2(?:2[1-7]|3[0-8]|4[12]|5[1256]|6\\d|7[1-3]|8[1-5])\\d{4}","\\d{7}",,,"2211234"],[,,"(?:9\\d|59)\\d{5}","\\d{7}",,,"9911234"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"CV",238,"0",,,,,,,,[[,"(\\d{3})(\\d{2})(\\d{2})","$1 $2 $3",,"","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],CW:[,[,,"[169]\\d{6,7}","\\d{7,8}"],[,,"9(?:[48]\\d{2}|50\\d|7(?:2[0-24]|[34]\\d|6[35-7]|77|8[7-9]))\\d{4}","\\d{7,8}",,,"94151234"],[,,"9(?:5(?:[1246]\\d|3[01])|6(?:[16-9]\\d|3[01]))\\d{4}",
|
|
|
"\\d{7,8}",,,"95181234"],[,,"NA","NA"],[,,"NA","NA"],[,,"(?:10|69)\\d{5}","\\d{7}",,,"1011234"],[,,"NA","NA"],[,,"NA","NA"],"CW",599,"00",,,,,,,,[[,"(\\d{3})(\\d{4})","$1 $2",["[13-7]"],"","",0],[,"(9)(\\d{3})(\\d{4})","$1 $2 $3",["9"],"","",0]],,[,,"955\\d{5}","\\d{7,8}",,,"95581234"],1,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],CX:[,[,,"[1458]\\d{5,9}","\\d{6,10}"],[,,"89164\\d{4}","\\d{8,9}",,,"891641234"],[,,"14(?:5\\d|71)\\d{5}|4(?:[0-2]\\d|3[0-57-9]|4[47-9]|5[0-25-9]|6[6-9]|7[03-9]|8[17-9]|9[017-9])\\d{6}",
|
|
|
"\\d{9}",,,"412345678"],[,,"1(?:80(?:0\\d{2})?|3(?:00\\d{2})?)\\d{4}","\\d{6,10}",,,"1800123456"],[,,"190[0126]\\d{6}","\\d{10}",,,"1900123456"],[,,"NA","NA"],[,,"500\\d{6}","\\d{9}",,,"500123456"],[,,"550\\d{6}","\\d{9}",,,"550123456"],"CX",61,"(?:14(?:1[14]|34|4[17]|[56]6|7[47]|88))?001[14-689]","0",,,"0",,"0011",,,,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],CY:[,[,,"[257-9]\\d{7}","\\d{8}"],[,,"2[2-6]\\d{6}","\\d{8}",,,"22345678"],[,,"9[5-79]\\d{6}","\\d{8}",,,"96123456"],[,,
|
|
|
"800\\d{5}","\\d{8}",,,"80001234"],[,,"90[09]\\d{5}","\\d{8}",,,"90012345"],[,,"80[1-9]\\d{5}","\\d{8}",,,"80112345"],[,,"700\\d{5}","\\d{8}",,,"70012345"],[,,"NA","NA"],"CY",357,"00",,,,,,,,[[,"(\\d{2})(\\d{6})","$1 $2",,"","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"(?:50|77)\\d{6}","\\d{8}",,,"77123456"],,,[,,"NA","NA"]],CZ:[,[,,"[2-8]\\d{8}|9\\d{8,11}","\\d{9,12}"],[,,"2\\d{8}|(?:3[1257-9]|4[16-9]|5[13-9])\\d{7}","\\d{9,12}",,,"212345678"],[,,"(?:60[1-8]|7(?:0[2-5]|[2379]\\d))\\d{6}","\\d{9,12}",
|
|
|
,,"601123456"],[,,"800\\d{6}","\\d{9,12}",,,"800123456"],[,,"9(?:0[05689]|76)\\d{6}","\\d{9,12}",,,"900123456"],[,,"8[134]\\d{7}","\\d{9,12}",,,"811234567"],[,,"70[01]\\d{6}","\\d{9,12}",,,"700123456"],[,,"9[17]0\\d{6}","\\d{9,12}",,,"910123456"],"CZ",420,"00",,,,,,,,[[,"([2-9]\\d{2})(\\d{3})(\\d{3})","$1 $2 $3",["[2-8]|9[015-7]"],"","",0],[,"(96\\d)(\\d{3})(\\d{3})(\\d{3})","$1 $2 $3 $4",["96"],"","",0],[,"(9\\d)(\\d{3})(\\d{3})(\\d{3})","$1 $2 $3 $4",["9[36]"],"","",0]],,[,,"NA","NA"],,,[,,"NA",
|
|
|
"NA"],[,,"9(?:5\\d|7[234])\\d{6}","\\d{9,12}",,,"972123456"],,,[,,"9(?:3\\d{9}|6\\d{7,10})","\\d{9,12}",,,"93123456789"]],DE:[,[,,"[1-35-9]\\d{3,14}|4(?:[0-8]\\d{4,12}|9(?:[0-37]\\d|4(?:[1-35-8]|4\\d?)|5\\d{1,2}|6[1-8]\\d?)\\d{2,8})","\\d{2,15}"],[,,"[246]\\d{5,13}|3(?:0\\d{3,13}|2\\d{9}|[3-9]\\d{4,13})|5(?:0[2-8]|[1256]\\d|[38][0-8]|4\\d{0,2}|[79][0-7])\\d{3,11}|7(?:0[2-8]|[1-9]\\d)\\d{3,10}|8(?:0[2-9]|[1-9]\\d)\\d{3,10}|9(?:0[6-9]\\d{3,10}|1\\d{4,12}|[2-9]\\d{4,11})","\\d{2,15}",,,"30123456"],[,
|
|
|
,"1(?:5[0-2579]\\d{8}|6[023]\\d{7,8}|7(?:[0-57-9]\\d?|6\\d)\\d{7})","\\d{10,11}",,,"15123456789"],[,,"800\\d{7,12}","\\d{10,15}",,,"8001234567890"],[,,"137[7-9]\\d{6}|900(?:[135]\\d{6}|9\\d{7})","\\d{10,11}",,,"9001234567"],[,,"1(?:3(?:7[1-6]\\d{6}|8\\d{4})|80\\d{5,11})","\\d{7,14}",,,"18012345"],[,,"700\\d{8}","\\d{11}",,,"70012345678"],[,,"NA","NA"],"DE",49,"00","0",,,"0",,,,[[,"(1\\d{2})(\\d{7,8})","$1 $2",["1[67]"],"0$1","",0],[,"(1\\d{3})(\\d{7})","$1 $2",["15"],"0$1","",0],[,"(\\d{2})(\\d{3,11})",
|
|
|
"$1 $2",["3[02]|40|[68]9"],"0$1","",0],[,"(\\d{3})(\\d{3,11})","$1 $2",["2(?:\\d1|0[2389]|1[24]|28|34)|3(?:[3-9][15]|40)|[4-8][1-9]1|9(?:06|[1-9]1)"],"0$1","",0],[,"(\\d{4})(\\d{2,11})","$1 $2",["[24-6]|[7-9](?:\\d[1-9]|[1-9]\\d)|3(?:[3569][02-46-9]|4[2-4679]|7[2-467]|8[2-46-8])","[24-6]|[7-9](?:\\d[1-9]|[1-9]\\d)|3(?:3(?:0[1-467]|2[127-9]|3[124578]|[46][1246]|7[1257-9]|8[1256]|9[145])|4(?:2[135]|3[1357]|4[13578]|6[1246]|7[1356]|9[1346])|5(?:0[14]|2[1-3589]|3[1357]|4[1246]|6[1-4]|7[1346]|8[13568]|9[1246])|6(?:0[356]|2[1-489]|3[124-6]|4[1347]|6[13]|7[12579]|8[1-356]|9[135])|7(?:2[1-7]|3[1357]|4[145]|6[1-5]|7[1-4])|8(?:21|3[1468]|4[1347]|6[0135-9]|7[1467]|8[136])|9(?:0[12479]|2[1358]|3[1357]|4[134679]|6[1-9]|7[136]|8[147]|9[1468]))"],
|
|
|
"0$1","",0],[,"(3\\d{4})(\\d{1,10})","$1 $2",["3"],"0$1","",0],[,"(800)(\\d{7,12})","$1 $2",["800"],"0$1","",0],[,"(177)(99)(\\d{7,8})","$1 $2 $3",["177","1779","17799"],"0$1","",0],[,"(\\d{3})(\\d)(\\d{4,10})","$1 $2 $3",["(?:18|90)0|137","1(?:37|80)|900[1359]"],"0$1","",0],[,"(1\\d{2})(\\d{5,11})","$1 $2",["181"],"0$1","",0],[,"(18\\d{3})(\\d{6})","$1 $2",["185","1850","18500"],"0$1","",0],[,"(18\\d{2})(\\d{7})","$1 $2",["18[68]"],"0$1","",0],[,"(18\\d)(\\d{8})","$1 $2",["18[2-579]"],"0$1","",0],
|
|
|
[,"(700)(\\d{4})(\\d{4})","$1 $2 $3",["700"],"0$1","",0],[,"(138)(\\d{4})","$1 $2",["138"],"0$1","",0]],,[,,"16(?:4\\d{1,10}|[89]\\d{1,11})","\\d{4,14}",,,"16412345"],,,[,,"NA","NA"],[,,"18(?:1\\d{5,11}|[2-9]\\d{8})","\\d{8,14}",,,"18500123456"],,,[,,"17799\\d{7,8}","\\d{12,13}",,,"177991234567"]],DJ:[,[,,"[27]\\d{7}","\\d{8}"],[,,"2(?:1[2-5]|7[45])\\d{5}","\\d{8}",,,"21360003"],[,,"77[6-8]\\d{5}","\\d{8}",,,"77831001"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"DJ",253,
|
|
|
"00",,,,,,,,[[,"(\\d{2})(\\d{2})(\\d{2})(\\d{2})","$1 $2 $3 $4",,"","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],DK:[,[,,"[2-9]\\d{7}","\\d{8}"],[,,"(?:[2-7]\\d|8[126-9]|9[1-36-9])\\d{6}","\\d{8}",,,"32123456"],[,,"(?:[2-7]\\d|8[126-9]|9[1-36-9])\\d{6}","\\d{8}",,,"20123456"],[,,"80\\d{6}","\\d{8}",,,"80123456"],[,,"90\\d{6}","\\d{8}",,,"90123456"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"DK",45,"00",,,,,,,1,[[,"(\\d{2})(\\d{2})(\\d{2})(\\d{2})","$1 $2 $3 $4",,"","",0]],
|
|
|
,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],DM:[,[,,"[57-9]\\d{9}","\\d{7}(?:\\d{3})?"],[,,"767(?:2(?:55|66)|4(?:2[01]|4[0-25-9])|50[0-4]|70[1-3])\\d{4}","\\d{7}(?:\\d{3})?",,,"7674201234"],[,,"767(?:2(?:[234689]5|7[5-7])|31[5-7]|61[2-7])\\d{4}","\\d{10}",,,"7672251234"],[,,"8(?:00|44|55|66|77|88)[2-9]\\d{6}","\\d{10}",,,"8002123456"],[,,"900[2-9]\\d{6}","\\d{10}",,,"9002123456"],[,,"NA","NA"],[,,"5(?:00|33|44|66|77)[2-9]\\d{6}","\\d{10}",,,"5002345678"],[,,"NA","NA"],"DM",1,"011",
|
|
|
"1",,,"1",,,,,,[,,"NA","NA"],,"767",[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],DO:[,[,,"[589]\\d{9}","\\d{7}(?:\\d{3})?"],[,,"8(?:[04]9[2-9]\\d{6}|29(?:2(?:[0-59]\\d|6[04-9]|7[0-27]|8[0237-9])|3(?:[0-35-9]\\d|4[7-9])|[45]\\d{2}|6(?:[0-27-9]\\d|[3-5][1-9]|6[0135-8])|7(?:0[013-9]|[1-37]\\d|4[1-35689]|5[1-4689]|6[1-57-9]|8[1-79]|9[1-8])|8(?:0[146-9]|1[0-48]|[248]\\d|3[1-79]|5[01589]|6[013-68]|7[124-8]|9[0-8])|9(?:[0-24]\\d|3[02-46-9]|5[0-79]|60|7[0169]|8[57-9]|9[02-9]))\\d{4})","\\d{7}(?:\\d{3})?",
|
|
|
,,"8092345678"],[,,"8[024]9[2-9]\\d{6}","\\d{7}(?:\\d{3})?",,,"8092345678"],[,,"8(?:00|44|55|66|77|88)[2-9]\\d{6}","\\d{10}",,,"8002123456"],[,,"900[2-9]\\d{6}","\\d{10}",,,"9002123456"],[,,"NA","NA"],[,,"5(?:00|33|44|66|77)[2-9]\\d{6}","\\d{10}",,,"5002345678"],[,,"NA","NA"],"DO",1,"011","1",,,"1",,,,,,[,,"NA","NA"],,"8[024]9",[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],DZ:[,[,,"(?:[1-4]|[5-9]\\d)\\d{7}","\\d{8,9}"],[,,"(?:1\\d|2[014-79]|3[0-8]|4[0135689])\\d{6}|9619\\d{5}","\\d{8,9}",,,"12345678"],
|
|
|
[,,"(?:5[4-6]|7[7-9])\\d{7}|6(?:[569]\\d|7[0-4])\\d{6}","\\d{9}",,,"551234567"],[,,"800\\d{6}","\\d{9}",,,"800123456"],[,,"80[3-689]1\\d{5}","\\d{9}",,,"808123456"],[,,"80[12]1\\d{5}","\\d{9}",,,"801123456"],[,,"NA","NA"],[,,"98[23]\\d{6}","\\d{9}",,,"983123456"],"DZ",213,"00","0",,,"0",,,,[[,"([1-4]\\d)(\\d{2})(\\d{2})(\\d{2})","$1 $2 $3 $4",["[1-4]"],"0$1","",0],[,"([5-8]\\d{2})(\\d{2})(\\d{2})(\\d{2})","$1 $2 $3 $4",["[5-8]"],"0$1","",0],[,"(9\\d)(\\d{3})(\\d{2})(\\d{2})","$1 $2 $3 $4",["9"],"0$1",
|
|
|
"",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],EC:[,[,,"1\\d{9,10}|[2-8]\\d{7}|9\\d{8}","\\d{7,11}"],[,,"[2-7][2-7]\\d{6}","\\d{7,8}",,,"22123456"],[,,"9(?:39|[45][89]|[67][7-9]|[89]\\d)\\d{6}","\\d{9}",,,"991234567"],[,,"1800\\d{6,7}","\\d{10,11}",,,"18001234567"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"[2-7]890\\d{4}","\\d{8}",,,"28901234"],"EC",593,"00","0",,,"0",,,,[[,"(\\d)(\\d{3})(\\d{4})","$1 $2-$3",["[247]|[356][2-8]"],"(0$1)","",0],[,"(\\d{2})(\\d{3})(\\d{4})",
|
|
|
"$1 $2 $3",["9"],"0$1","",0],[,"(1800)(\\d{3})(\\d{3,4})","$1 $2 $3",["1"],"$1","",0]],[[,"(\\d)(\\d{3})(\\d{4})","$1-$2-$3",["[247]|[356][2-8]"]],[,"(\\d{2})(\\d{3})(\\d{4})","$1 $2 $3",["9"],"0$1","",0],[,"(1800)(\\d{3})(\\d{3,4})","$1 $2 $3",["1"],"$1","",0]],[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],EE:[,[,,"1\\d{3,4}|[3-9]\\d{6,7}|800\\d{6,7}","\\d{4,10}"],[,,"(?:3[23589]|4[3-8]|6\\d|7[1-9]|88)\\d{5}","\\d{7}",,,"3212345"],[,,"(?:5\\d|8[1-5])\\d{6}|5(?:[02]\\d{2}|1(?:[0-8]\\d|95)|5[0-478]\\d|64[0-4]|65[1-589])\\d{3}",
|
|
|
"\\d{7,8}",,,"51234567"],[,,"800(?:0\\d{3}|1\\d|[2-9])\\d{3}","\\d{7,10}",,,"80012345"],[,,"(?:40\\d{2}|900)\\d{4}","\\d{7,8}",,,"9001234"],[,,"NA","NA"],[,,"70[0-2]\\d{5}","\\d{8}",,,"70012345"],[,,"NA","NA"],"EE",372,"00",,,,,,,,[[,"([3-79]\\d{2})(\\d{4})","$1 $2",["[369]|4[3-8]|5(?:[0-2]|5[0-478]|6[45])|7[1-9]","[369]|4[3-8]|5(?:[02]|1(?:[0-8]|95)|5[0-478]|6(?:4[0-4]|5[1-589]))|7[1-9]"],"","",0],[,"(70)(\\d{2})(\\d{4})","$1 $2 $3",["70"],"","",0],[,"(8000)(\\d{3})(\\d{3})","$1 $2 $3",["800","8000"],
|
|
|
"","",0],[,"([458]\\d{3})(\\d{3,4})","$1 $2",["40|5|8(?:00|[1-5])","40|5|8(?:00[1-9]|[1-5])"],"","",0]],,[,,"NA","NA"],,,[,,"1\\d{3,4}|800[2-9]\\d{3}","\\d{4,7}",,,"8002123"],[,,"1(?:2[01245]|3[0-6]|4[1-489]|5[0-59]|6[1-46-9]|7[0-27-9]|8[189]|9[012])\\d{1,2}","\\d{4,5}",,,"12123"],,,[,,"NA","NA"]],EG:[,[,,"1\\d{4,9}|[2456]\\d{8}|3\\d{7}|[89]\\d{8,9}","\\d{5,10}"],[,,"(?:1(?:3[23]\\d|5(?:[23]|9\\d))|2[2-4]\\d{2}|3\\d{2}|4(?:0[2-5]|[578][23]|64)\\d|5(?:0[2-7]|[57][23])\\d|6[24-689]3\\d|8(?:2[2-57]|4[26]|6[237]|8[2-4])\\d|9(?:2[27]|3[24]|52|6[2356]|7[2-4])\\d)\\d{5}|1[69]\\d{3}",
|
|
|
"\\d{5,9}",,,"234567890"],[,,"1(?:0[0-269]|1[0-245]|2[0-278])\\d{7}","\\d{10}",,,"1001234567"],[,,"800\\d{7}","\\d{10}",,,"8001234567"],[,,"900\\d{7}","\\d{10}",,,"9001234567"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"EG",20,"00","0",,,"0",,,,[[,"(\\d)(\\d{7,8})","$1 $2",["[23]"],"0$1","",0],[,"(\\d{3})(\\d{3})(\\d{4})","$1 $2 $3",["1[012]|[89]00"],"0$1","",0],[,"(\\d{2})(\\d{6,7})","$1 $2",["1[35]|[4-6]|[89][2-9]"],"0$1","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],EH:[,
|
|
|
[,,"[5689]\\d{8}","\\d{9}"],[,,"528[89]\\d{5}","\\d{9}",,,"528812345"],[,,"6(?:0[0-8]|[12-7]\\d|8[01]|9[2457-9])\\d{6}","\\d{9}",,,"650123456"],[,,"80\\d{7}","\\d{9}",,,"801234567"],[,,"89\\d{7}","\\d{9}",,,"891234567"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"EH",212,"00","0",,,"0",,,,,,[,,"NA","NA"],,"528[89]",[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],ER:[,[,,"[178]\\d{6}","\\d{6,7}"],[,,"1(?:1[12568]|20|40|55|6[146])\\d{4}|8\\d{6}","\\d{6,7}",,,"8370362"],[,,"17[1-3]\\d{4}|7\\d{6}","\\d{7}",
|
|
|
,,"7123456"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"ER",291,"00","0",,,"0",,,,[[,"(\\d)(\\d{3})(\\d{3})","$1 $2 $3",,"0$1","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],ES:[,[,,"[5-9]\\d{8}","\\d{9}"],[,,"8(?:[13]0|[28][0-8]|[47][1-9]|5[01346-9]|6[0457-9])\\d{6}|9(?:[1238][0-8]\\d{6}|4[1-9]\\d{6}|5\\d{7}|6(?:[0-8]\\d{6}|9(?:0(?:[0-57-9]\\d{4}|6(?:0[0-8]|1[1-9]|[2-9]\\d)\\d{2})|[1-9]\\d{5}))|7(?:[124-9]\\d{2}|3(?:[0-8]\\d|9[1-9]))\\d{4})","\\d{9}",
|
|
|
,,"810123456"],[,,"(?:6\\d{6}|7[1-4]\\d{5}|9(?:6906(?:09|10)|7390\\d{2}))\\d{2}","\\d{9}",,,"612345678"],[,,"[89]00\\d{6}","\\d{9}",,,"800123456"],[,,"80[367]\\d{6}","\\d{9}",,,"803123456"],[,,"90[12]\\d{6}","\\d{9}",,,"901123456"],[,,"70\\d{7}","\\d{9}",,,"701234567"],[,,"NA","NA"],"ES",34,"00",,,,,,,,[[,"([5-9]\\d{2})(\\d{2})(\\d{2})(\\d{2})","$1 $2 $3 $4",["[568]|[79][0-8]"],"","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"51\\d{7}","\\d{9}",,,"511234567"],,,[,,"NA","NA"]],ET:[,[,,"[1-59]\\d{8}","\\d{7,9}"],
|
|
|
[,,"(?:11(?:1(?:1[124]|2[2-57]|3[1-5]|5[5-8]|8[6-8])|2(?:13|3[6-8]|5[89]|7[05-9]|8[2-6])|3(?:2[01]|3[0-289]|4[1289]|7[1-4]|87)|4(?:1[69]|3[2-49]|4[0-3]|6[5-8])|5(?:1[57]|44|5[0-4])|6(?:18|2[69]|4[5-7]|5[1-5]|6[0-59]|8[015-8]))|2(?:2(?:11[1-9]|22[0-7]|33\\d|44[1467]|66[1-68])|5(?:11[124-6]|33[2-8]|44[1467]|55[14]|66[1-3679]|77[124-79]|880))|3(?:3(?:11[0-46-8]|22[0-6]|33[0134689]|44[04]|55[0-6]|66[01467])|4(?:44[0-8]|55[0-69]|66[0-3]|77[1-5]))|4(?:6(?:22[0-24-7]|33[1-5]|44[13-69]|55[14-689]|660|88[1-4])|7(?:11[1-9]|22[1-9]|33[13-7]|44[13-6]|55[1-689]))|5(?:7(?:227|55[05]|(?:66|77)[14-8])|8(?:11[149]|22[013-79]|33[0-68]|44[013-8]|550|66[1-5]|77\\d)))\\d{4}",
|
|
|
"\\d{7,9}",,,"111112345"],[,,"9(?:[1-3]\\d|5[89])\\d{6}","\\d{9}",,,"911234567"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"ET",251,"00","0",,,"0",,,,[[,"([1-59]\\d)(\\d{3})(\\d{4})","$1 $2 $3",,"0$1","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],FI:[,[,,"1\\d{4,11}|[2-9]\\d{4,10}","\\d{5,12}"],[,,"1(?:[3569][1-8]\\d{3,9}|[47]\\d{5,10})|2[1-8]\\d{3,9}|3(?:[1-8]\\d{3,9}|9\\d{4,8})|[5689][1-8]\\d{3,9}","\\d{5,12}",,,"1312345678"],[,,"4\\d{5,10}|50\\d{4,8}",
|
|
|
"\\d{6,11}",,,"412345678"],[,,"800\\d{4,7}","\\d{7,10}",,,"8001234567"],[,,"[67]00\\d{5,6}","\\d{8,9}",,,"600123456"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"FI",358,"00|99[049]","0",,,"0",,,,[[,"(\\d{3})(\\d{3,7})","$1 $2",["(?:[1-3]00|[6-8]0)"],"0$1","",0],[,"(\\d{2})(\\d{4,10})","$1 $2",["[14]|2[09]|50|7[135]"],"0$1","",0],[,"(\\d)(\\d{4,11})","$1 $2",["[25689][1-8]|3"],"0$1","",0]],,[,,"NA","NA"],1,,[,,"[13]00\\d{3,7}|2(?:0(?:0\\d{3,7}|2[023]\\d{1,6}|9[89]\\d{1,6}))|60(?:[12]\\d{5,6}|6\\d{7})|7(?:1\\d{7}|3\\d{8}|5[03-9]\\d{2,7})",
|
|
|
"\\d{5,10}",,,"100123"],[,,"[13]0\\d{4,8}|2(?:0(?:[016-8]\\d{3,7}|[2-59]\\d{2,7})|9\\d{4,8})|60(?:[12]\\d{5,6}|6\\d{7})|7(?:1\\d{7}|3\\d{8}|5[03-9]\\d{2,7})","\\d{5,10}",,,"10112345"],,,[,,"NA","NA"]],FJ:[,[,,"[36-9]\\d{6}|0\\d{10}","\\d{7}(?:\\d{4})?"],[,,"(?:3[0-5]|6[25-7]|8[58])\\d{5}","\\d{7}",,,"3212345"],[,,"(?:7[0-8]|8[034679]|9\\d)\\d{5}","\\d{7}",,,"7012345"],[,,"0800\\d{7}","\\d{11}",,,"08001234567"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"FJ",679,"0(?:0|52)",,,,,,"00",
|
|
|
,[[,"(\\d{3})(\\d{4})","$1 $2",["[36-9]"],"","",0],[,"(\\d{4})(\\d{3})(\\d{4})","$1 $2 $3",["0"],"","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],1,,[,,"NA","NA"]],FK:[,[,,"[2-7]\\d{4}","\\d{5}"],[,,"[2-47]\\d{4}","\\d{5}",,,"31234"],[,,"[56]\\d{4}","\\d{5}",,,"51234"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"FK",500,"00",,,,,,,,,,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],FM:[,[,,"[39]\\d{6}","\\d{7}"],[,,"3[2357]0[1-9]\\d{3}|9[2-6]\\d{5}","\\d{7}",
|
|
|
,,"3201234"],[,,"3[2357]0[1-9]\\d{3}|9[2-7]\\d{5}","\\d{7}",,,"3501234"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"FM",691,"00",,,,,,,,[[,"(\\d{3})(\\d{4})","$1 $2",,"","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],FO:[,[,,"[2-9]\\d{5}","\\d{6}"],[,,"(?:20|[3-4]\\d|8[19])\\d{4}","\\d{6}",,,"201234"],[,,"(?:2[1-9]|5\\d|7[1-79])\\d{4}","\\d{6}",,,"211234"],[,,"80[257-9]\\d{3}","\\d{6}",,,"802123"],[,,"90(?:[1345][15-7]|2[125-7]|99)\\d{2}","\\d{6}",
|
|
|
,,"901123"],[,,"NA","NA"],[,,"NA","NA"],[,,"(?:6[0-36]|88)\\d{4}","\\d{6}",,,"601234"],"FO",298,"00",,,,"(10(?:01|[12]0|88))",,,,[[,"(\\d{6})","$1",,"","$CC $1",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],FR:[,[,,"[1-9]\\d{8}","\\d{9}"],[,,"[1-5]\\d{8}","\\d{9}",,,"123456789"],[,,"6\\d{8}|7[5-9]\\d{7}","\\d{9}",,,"612345678"],[,,"80\\d{7}","\\d{9}",,,"801234567"],[,,"89[1-37-9]\\d{6}","\\d{9}",,,"891123456"],[,,"8(?:1[019]|2[0156]|84|90)\\d{6}","\\d{9}",,,"810123456"],[,,"NA",
|
|
|
"NA"],[,,"9\\d{8}","\\d{9}",,,"912345678"],"FR",33,"00","0",,,"0",,,,[[,"([1-79])(\\d{2})(\\d{2})(\\d{2})(\\d{2})","$1 $2 $3 $4 $5",["[1-79]"],"0$1","",0],[,"(1\\d{2})(\\d{3})","$1 $2",["11"],"$1","",0],[,"(8\\d{2})(\\d{2})(\\d{2})(\\d{2})","$1 $2 $3 $4",["8"],"0 $1","",0]],[[,"([1-79])(\\d{2})(\\d{2})(\\d{2})(\\d{2})","$1 $2 $3 $4 $5",["[1-79]"],"0$1","",0],[,"(8\\d{2})(\\d{2})(\\d{2})(\\d{2})","$1 $2 $3 $4",["8"],"0 $1","",0]],[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],GA:[,[,
|
|
|
,"0?\\d{7}","\\d{7,8}"],[,,"01\\d{6}","\\d{8}",,,"01441234"],[,,"0?[2-7]\\d{6}","\\d{7,8}",,,"06031234"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"GA",241,"00",,,,,,,,[[,"(\\d)(\\d{2})(\\d{2})(\\d{2})","$1 $2 $3 $4",["[2-7]"],"0$1","",0],[,"(\\d{2})(\\d{2})(\\d{2})(\\d{2})","$1 $2 $3 $4",["0"],"","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],1,,[,,"NA","NA"]],GB:[,[,,"\\d{7,10}","\\d{4,10}"],[,,"2(?:0[01378]|3[0189]|4[017]|8[0-46-9]|9[012])\\d{7}|1(?:(?:1(?:3[0-48]|[46][0-4]|5[012789]|7[0-49]|8[01349])|21[0-7]|31[0-8]|[459]1\\d|61[0-46-9]))\\d{6}|1(?:2(?:0[024-9]|2[3-9]|3[3-79]|4[1-689]|[58][02-9]|6[0-4789]|7[013-9]|9\\d)|3(?:0\\d|[25][02-9]|3[02-579]|[468][0-46-9]|7[1235679]|9[24578])|4(?:0[03-9]|[28][02-5789]|[37]\\d|4[02-69]|5[0-8]|[69][0-79])|5(?:0[1235-9]|2[024-9]|3[015689]|4[02-9]|5[03-9]|6\\d|7[0-35-9]|8[0-468]|9[0-5789])|6(?:0[034689]|2[0-35689]|[38][013-9]|4[1-467]|5[0-69]|6[13-9]|7[0-8]|9[0124578])|7(?:0[0246-9]|2\\d|3[023678]|4[03-9]|5[0-46-9]|6[013-9]|7[0-35-9]|8[024-9]|9[02-9])|8(?:0[35-9]|2[1-5789]|3[02-578]|4[0-578]|5[124-9]|6[2-69]|7\\d|8[02-9]|9[02569])|9(?:0[02-589]|2[02-689]|3[1-5789]|4[2-9]|5[0-579]|6[234789]|7[0124578]|8\\d|9[2-57]))\\d{6}|1(?:2(?:0(?:46[1-4]|87[2-9])|545[1-79]|76(?:2\\d|3[1-8]|6[1-6])|9(?:7(?:2[0-4]|3[2-5])|8(?:2[2-8]|7[0-4789]|8[345])))|3(?:638[2-5]|647[23]|8(?:47[04-9]|64[015789]))|4(?:044[1-7]|20(?:2[23]|8\\d)|6(?:0(?:30|5[2-57]|6[1-8]|7[2-8])|140)|8(?:052|87[123]))|5(?:24(?:3[2-79]|6\\d)|276\\d|6(?:26[06-9]|686))|6(?:06(?:4\\d|7[4-79])|295[567]|35[34]\\d|47(?:24|61)|59(?:5[08]|6[67]|74)|955[0-4])|7(?:26(?:6[13-9]|7[0-7])|442\\d|50(?:2[0-3]|[3-68]2|76))|8(?:27[56]\\d|37(?:5[2-5]|8[239])|84(?:3[2-58]))|9(?:0(?:0(?:6[1-8]|85)|52\\d)|3583|4(?:66[1-8]|9(?:2[01]|81))|63(?:23|3[1-4])|9561))\\d{3}|176888[234678]\\d{2}|16977[23]\\d{3}",
|
|
|
"\\d{4,10}",,,"1212345678"],[,,"7(?:[1-4]\\d\\d|5(?:0[0-8]|[13-9]\\d|2[0-35-9])|7(?:0[1-9]|[1-7]\\d|8[02-9]|9[0-689])|8(?:[014-9]\\d|[23][0-8])|9(?:[04-9]\\d|1[02-9]|2[0-35-9]|3[0-689]))\\d{6}","\\d{10}",,,"7400123456"],[,,"80(?:0(?:1111|\\d{6,7})|8\\d{7})|500\\d{6}","\\d{7}(?:\\d{2,3})?",,,"8001234567"],[,,"(?:87[123]|9(?:[01]\\d|8[2349]))\\d{7}","\\d{10}",,,"9012345678"],[,,"8(?:4(?:5464\\d|[2-5]\\d{7})|70\\d{7})","\\d{7}(?:\\d{3})?",,,"8431234567"],[,,"70\\d{8}","\\d{10}",,,"7012345678"],[,,"56\\d{8}",
|
|
|
"\\d{10}",,,"5612345678"],"GB",44,"00","0"," x",,"0",,,,[[,"(\\d{2})(\\d{4})(\\d{4})","$1 $2 $3",["2|5[56]|7(?:0|6[013-9])","2|5[56]|7(?:0|6(?:[013-9]|2[0-35-9]))"],"0$1","",0],[,"(\\d{3})(\\d{3})(\\d{4})","$1 $2 $3",["1(?:1|\\d1)|3|9[018]"],"0$1","",0],[,"(\\d{5})(\\d{4,5})","$1 $2",["1(?:38|5[23]|69|76|94)","1(?:387|5(?:24|39)|697|768|946)","1(?:3873|5(?:242|39[456])|697[347]|768[347]|9467)"],"0$1","",0],[,"(1\\d{3})(\\d{5,6})","$1 $2",["1"],"0$1","",0],[,"(7\\d{3})(\\d{6})","$1 $2",["7(?:[1-5789]|62)",
|
|
|
"7(?:[1-5789]|624)"],"0$1","",0],[,"(800)(\\d{4})","$1 $2",["800","8001","80011","800111","8001111"],"0$1","",0],[,"(845)(46)(4\\d)","$1 $2 $3",["845","8454","84546","845464"],"0$1","",0],[,"(8\\d{2})(\\d{3})(\\d{4})","$1 $2 $3",["8(?:4[2-5]|7[0-3])"],"0$1","",0],[,"(80\\d)(\\d{3})(\\d{4})","$1 $2 $3",["80"],"0$1","",0],[,"([58]00)(\\d{6})","$1 $2",["[58]00"],"0$1","",0]],,[,,"76(?:0[012]|2[356]|4[0134]|5[49]|6[0-369]|77|81|9[39])\\d{6}","\\d{10}",,,"7640123456"],1,,[,,"NA","NA"],[,,"(?:3[0347]|55)\\d{8}",
|
|
|
"\\d{10}",,,"5512345678"],,,[,,"NA","NA"]],GD:[,[,,"[4589]\\d{9}","\\d{7}(?:\\d{3})?"],[,,"473(?:2(?:3[0-2]|69)|3(?:2[89]|86)|4(?:[06]8|3[5-9]|4[0-49]|5[5-79]|68|73|90)|63[68]|7(?:58|84)|800|938)\\d{4}","\\d{7}(?:\\d{3})?",,,"4732691234"],[,,"473(?:4(?:0[2-79]|1[04-9]|20|58)|5(?:2[01]|3[3-8])|901)\\d{4}","\\d{10}",,,"4734031234"],[,,"8(?:00|44|55|66|77|88)[2-9]\\d{6}","\\d{10}",,,"8002123456"],[,,"900[2-9]\\d{6}","\\d{10}",,,"9002123456"],[,,"NA","NA"],[,,"5(?:00|33|44|66|77)[2-9]\\d{6}","\\d{10}",
|
|
|
,,"5002345678"],[,,"NA","NA"],"GD",1,"011","1",,,"1",,,,,,[,,"NA","NA"],,"473",[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],GE:[,[,,"[34578]\\d{8}","\\d{6,9}"],[,,"(?:3(?:[256]\\d|4[124-9]|7[0-4])|4(?:1\\d|2[2-7]|3[1-79]|4[2-8]|7[239]|9[1-7]))\\d{6}","\\d{6,9}",,,"322123456"],[,,"5(?:14|5[01578]|68|7[0147-9]|9[0-35-9])\\d{6}","\\d{9}",,,"555123456"],[,,"800\\d{6}","\\d{9}",,,"800123456"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"706\\d{6}","\\d{9}",,,"706123456"],"GE",995,"00","0",,,"0",,,,[[,
|
|
|
"(\\d{3})(\\d{2})(\\d{2})(\\d{2})","$1 $2 $3 $4",["[348]"],"0$1","",0],[,"(\\d{3})(\\d{3})(\\d{3})","$1 $2 $3",["7"],"0$1","",0],[,"(\\d{3})(\\d{2})(\\d{2})(\\d{2})","$1 $2 $3 $4",["5"],"$1","",0]],,[,,"NA","NA"],,,[,,"706\\d{6}","\\d{9}",,,"706123456"],[,,"NA","NA"],,,[,,"NA","NA"]],GF:[,[,,"[56]\\d{8}","\\d{9}"],[,,"594(?:10|2[012457-9]|3[0-57-9]|4[3-9]|5[7-9]|6[0-3]|9[014])\\d{4}","\\d{9}",,,"594101234"],[,,"694(?:[04][0-7]|1[0-5]|3[018]|[29]\\d)\\d{4}","\\d{9}",,,"694201234"],[,,"NA","NA"],[,
|
|
|
,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"GF",594,"00","0",,,"0",,,,[[,"(\\d{3})(\\d{2})(\\d{2})(\\d{2})","$1 $2 $3 $4",,"0$1","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],GG:[,[,,"[135789]\\d{6,9}","\\d{6,10}"],[,,"1481\\d{6}","\\d{6,10}",,,"1481456789"],[,,"7(?:781|839|911)\\d{6}","\\d{10}",,,"7781123456"],[,,"80(?:0(?:1111|\\d{6,7})|8\\d{7})|500\\d{6}","\\d{7}(?:\\d{2,3})?",,,"8001234567"],[,,"(?:87[123]|9(?:[01]\\d|8[0-3]))\\d{7}","\\d{10}",,,"9012345678"],
|
|
|
[,,"8(?:4(?:5464\\d|[2-5]\\d{7})|70\\d{7})","\\d{7}(?:\\d{3})?",,,"8431234567"],[,,"70\\d{8}","\\d{10}",,,"7012345678"],[,,"56\\d{8}","\\d{10}",,,"5612345678"],"GG",44,"00","0"," x",,"0",,,,,,[,,"76(?:0[012]|2[356]|4[0134]|5[49]|6[0-369]|77|81|9[39])\\d{6}","\\d{10}",,,"7640123456"],,,[,,"NA","NA"],[,,"(?:3[0347]|55)\\d{8}","\\d{10}",,,"5512345678"],,,[,,"NA","NA"]],GH:[,[,,"[235]\\d{8}|8\\d{7}","\\d{7,9}"],[,,"3(?:0[237]\\d|[167](?:2[0-6]|7\\d)|2(?:2[0-5]|7\\d)|3(?:2[0-3]|7\\d)|4(?:2[013-9]|3[01]|7\\d)|5(?:2[0-7]|7\\d)|8(?:2[0-2]|7\\d)|9(?:20|7\\d))\\d{5}",
|
|
|
"\\d{7,9}",,,"302345678"],[,,"(?:2[034678]\\d|5(?:[047]\\d|54))\\d{6}","\\d{9}",,,"231234567"],[,,"800\\d{5}","\\d{8}",,,"80012345"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"GH",233,"00","0",,,"0",,,,[[,"(\\d{2})(\\d{3})(\\d{4})","$1 $2 $3",["[235]"],"0$1","",0],[,"(\\d{3})(\\d{5})","$1 $2",["8"],"0$1","",0]],,[,,"NA","NA"],,,[,,"800\\d{5}","\\d{8}",,,"80012345"],[,,"NA","NA"],,,[,,"NA","NA"]],GI:[,[,,"[2568]\\d{7}","\\d{8}"],[,,"2(?:00\\d|1(?:6[24-7]|9\\d)|2(?:00|2[2457]))\\d{4}",
|
|
|
"\\d{8}",,,"20012345"],[,,"(?:5[46-8]|62)\\d{6}","\\d{8}",,,"57123456"],[,,"80\\d{6}","\\d{8}",,,"80123456"],[,,"8[1-689]\\d{6}","\\d{8}",,,"88123456"],[,,"87\\d{6}","\\d{8}",,,"87123456"],[,,"NA","NA"],[,,"NA","NA"],"GI",350,"00",,,,,,,,[[,"(\\d{3})(\\d{5})","$1 $2",["2"],"","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],GL:[,[,,"[1-689]\\d{5}","\\d{6}"],[,,"(?:19|3[1-6]|6[14689]|8[14-79]|9\\d)\\d{4}","\\d{6}",,,"321000"],[,,"[245][2-9]\\d{4}","\\d{6}",,,"221234"],[,,"80\\d{4}",
|
|
|
"\\d{6}",,,"801234"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"3[89]\\d{4}","\\d{6}",,,"381234"],"GL",299,"00",,,,,,,,[[,"(\\d{2})(\\d{2})(\\d{2})","$1 $2 $3",,"","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],GM:[,[,,"[2-9]\\d{6}","\\d{7}"],[,,"(?:4(?:[23]\\d{2}|4(?:1[024679]|[6-9]\\d))|5(?:54[0-7]|6(?:[67]\\d)|7(?:1[04]|2[035]|3[58]|48))|8\\d{3})\\d{3}","\\d{7}",,,"5661234"],[,,"(?:2[0-6]|[3679]\\d)\\d{5}","\\d{7}",,,"3012345"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],
|
|
|
[,,"NA","NA"],[,,"NA","NA"],"GM",220,"00",,,,,,,,[[,"(\\d{3})(\\d{4})","$1 $2",,"","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],GN:[,[,,"[367]\\d{7,8}","\\d{8,9}"],[,,"30(?:24|3[12]|4[1-35-7]|5[13]|6[189]|[78]1|9[1478])\\d{4}","\\d{8}",,,"30241234"],[,,"6[02356]\\d{7}","\\d{9}",,,"601123456"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"722\\d{6}","\\d{9}",,,"722123456"],"GN",224,"00",,,,,,,,[[,"(\\d{2})(\\d{2})(\\d{2})(\\d{2})","$1 $2 $3 $4",["3"],"","",0],
|
|
|
[,"(\\d{3})(\\d{2})(\\d{2})(\\d{2})","$1 $2 $3 $4",["[67]"],"","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],GP:[,[,,"[56]\\d{8}","\\d{9}"],[,,"590(?:0[13468]|1[012]|2[0-68]|3[28]|4[0-8]|5[579]|6[0189]|70|8[0-689]|9\\d)\\d{4}","\\d{9}",,,"590201234"],[,,"690(?:0[0-7]|[1-9]\\d)\\d{4}","\\d{9}",,,"690301234"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"GP",590,"00","0",,,"0",,,,[[,"([56]90)(\\d{2})(\\d{4})","$1 $2-$3",,"0$1","",0]],,[,,"NA","NA"],1,
|
|
|
,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],GQ:[,[,,"[23589]\\d{8}","\\d{9}"],[,,"3(?:3(?:3\\d[7-9]|[0-24-9]\\d[46])|5\\d{2}[7-9])\\d{4}","\\d{9}",,,"333091234"],[,,"(?:222|551)\\d{6}","\\d{9}",,,"222123456"],[,,"80\\d[1-9]\\d{5}","\\d{9}",,,"800123456"],[,,"90\\d[1-9]\\d{5}","\\d{9}",,,"900123456"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"GQ",240,"00",,,,,,,,[[,"(\\d{3})(\\d{3})(\\d{3})","$1 $2 $3",["[235]"],"","",0],[,"(\\d{3})(\\d{6})","$1 $2",["[89]"],"","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],
|
|
|
[,,"NA","NA"],,,[,,"NA","NA"]],GR:[,[,,"[26-9]\\d{9}","\\d{10}"],[,,"2(?:1\\d{2}|2(?:2[1-46-9]|3[1-8]|4[1-7]|5[1-4]|6[1-8]|7[1-5]|[89][1-9])|3(?:1\\d|2[1-57]|[35][1-3]|4[13]|7[1-7]|8[124-6]|9[1-79])|4(?:1\\d|2[1-8]|3[1-4]|4[13-5]|6[1-578]|9[1-5])|5(?:1\\d|[29][1-4]|3[1-5]|4[124]|5[1-6])|6(?:1\\d|3[1245]|4[1-7]|5[13-9]|[269][1-6]|7[14]|8[1-5])|7(?:1\\d|2[1-5]|3[1-6]|4[1-7]|5[1-57]|6[135]|9[125-7])|8(?:1\\d|2[1-5]|[34][1-4]|9[1-57]))\\d{6}","\\d{10}",,,"2123456789"],[,,"69\\d{8}","\\d{10}",,,"6912345678"],
|
|
|
[,,"800\\d{7}","\\d{10}",,,"8001234567"],[,,"90[19]\\d{7}","\\d{10}",,,"9091234567"],[,,"8(?:0[16]|12|25)\\d{7}","\\d{10}",,,"8011234567"],[,,"70\\d{8}","\\d{10}",,,"7012345678"],[,,"NA","NA"],"GR",30,"00",,,,,,,,[[,"([27]\\d)(\\d{4})(\\d{4})","$1 $2 $3",["21|7"],"","",0],[,"(\\d{3})(\\d{3})(\\d{4})","$1 $2 $3",["2[2-9]1|[689]"],"","",0],[,"(2\\d{3})(\\d{6})","$1 $2",["2[2-9][02-9]"],"","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],GT:[,[,,"[2-7]\\d{7}|1[89]\\d{9}","\\d{8}(?:\\d{3})?"],
|
|
|
[,,"[267][2-9]\\d{6}","\\d{8}",,,"22456789"],[,,"[345]\\d{7}","\\d{8}",,,"51234567"],[,,"18[01]\\d{8}","\\d{11}",,,"18001112222"],[,,"19\\d{9}","\\d{11}",,,"19001112222"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"GT",502,"00",,,,,,,,[[,"(\\d{4})(\\d{4})","$1 $2",["[2-7]"],"","",0],[,"(\\d{4})(\\d{3})(\\d{4})","$1 $2 $3",["1"],"","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],GU:[,[,,"[5689]\\d{9}","\\d{7}(?:\\d{3})?"],[,,"671(?:3(?:00|3[39]|4[349]|55|6[26])|4(?:56|7[1-9]|8[236-9])|5(?:55|6[2-5]|88)|6(?:3[2-578]|4[24-9]|5[34]|78|8[5-9])|7(?:[079]7|2[0167]|3[45]|8[789])|8(?:[2-5789]8|6[48])|9(?:2[29]|6[79]|7[179]|8[789]|9[78]))\\d{4}",
|
|
|
"\\d{7}(?:\\d{3})?",,,"6713001234"],[,,"671(?:3(?:00|3[39]|4[349]|55|6[26])|4(?:56|7[1-9]|8[236-9])|5(?:55|6[2-5]|88)|6(?:3[2-578]|4[24-9]|5[34]|78|8[5-9])|7(?:[079]7|2[0167]|3[45]|8[789])|8(?:[2-5789]8|6[48])|9(?:2[29]|6[79]|7[179]|8[789]|9[78]))\\d{4}","\\d{7}(?:\\d{3})?",,,"6713001234"],[,,"8(?:00|44|55|66|77|88)[2-9]\\d{6}","\\d{10}",,,"8002123456"],[,,"900[2-9]\\d{6}","\\d{10}",,,"9002123456"],[,,"NA","NA"],[,,"5(?:00|33|44|66|77)[2-9]\\d{6}","\\d{10}",,,"5002345678"],[,,"NA","NA"],"GU",1,"011",
|
|
|
"1",,,"1",,,1,,,[,,"NA","NA"],,"671",[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],GW:[,[,,"[3-79]\\d{6}","\\d{7}"],[,,"3(?:2[0125]|3[1245]|4[12]|5[1-4]|70|9[1-467])\\d{4}","\\d{7}",,,"3201234"],[,,"(?:[5-7]\\d|9[012])\\d{5}","\\d{7}",,,"5012345"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"40\\d{5}","\\d{7}",,,"4012345"],"GW",245,"00",,,,,,,,[[,"(\\d{3})(\\d{4})","$1 $2",,"","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],GY:[,[,,"[2-4679]\\d{6}","\\d{7}"],[,,
|
|
|
"(?:2(?:1[6-9]|2[0-35-9]|3[1-4]|5[3-9]|6\\d|7[0-24-79])|3(?:2[25-9]|3\\d)|4(?:4[0-24]|5[56])|77[1-57])\\d{4}","\\d{7}",,,"2201234"],[,,"6\\d{6}","\\d{7}",,,"6091234"],[,,"(?:289|862)\\d{4}","\\d{7}",,,"2891234"],[,,"9008\\d{3}","\\d{7}",,,"9008123"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"GY",592,"001",,,,,,,,[[,"(\\d{3})(\\d{4})","$1 $2",,"","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],HK:[,[,,"[235-7]\\d{7}|8\\d{7,8}|9\\d{4,10}","\\d{5,11}"],[,,"(?:[23]\\d|5[78])\\d{6}",
|
|
|
"\\d{8}",,,"21234567"],[,,"(?:5[1-69]\\d|6\\d{2}|9(?:0[1-9]|[1-8]\\d))\\d{5}","\\d{8}",,,"51234567"],[,,"800\\d{6}","\\d{9}",,,"800123456"],[,,"900(?:[0-24-9]\\d{7}|3\\d{1,4})","\\d{5,11}",,,"90012345678"],[,,"NA","NA"],[,,"8[1-3]\\d{6}","\\d{8}",,,"81123456"],[,,"NA","NA"],"HK",852,"00",,,,,,,,[[,"(\\d{4})(\\d{4})","$1 $2",["[235-7]|[89](?:0[1-9]|[1-9])"],"","",0],[,"(800)(\\d{3})(\\d{3})","$1 $2 $3",["800"],"","",0],[,"(900)(\\d{2})(\\d{3})(\\d{3})","$1 $2 $3 $4",["900"],"","",0],[,"(900)(\\d{2,5})",
|
|
|
"$1 $2",["900"],"","",0]],,[,,"7\\d{7}","\\d{8}",,,"71234567"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],HN:[,[,,"[237-9]\\d{7}","\\d{8}"],[,,"2(?:2(?:0[019]|1[1-36]|[23]\\d|4[056]|5[57]|7[01389]|8[0146-9]|9[012])|4(?:2[3-59]|3[13-689]|4[0-68]|5[1-35])|5(?:4[3-5]|5\\d|6[56]|74)|6(?:[056]\\d|4[0-378]|[78][0-8]|9[01])|7(?:6[46-9]|7[02-9]|8[34])|8(?:79|8[0-35789]|9[1-57-9]))\\d{4}","\\d{8}",,,"22123456"],[,,"[37-9]\\d{7}","\\d{8}",,,"91234567"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA",
|
|
|
"NA"],[,,"NA","NA"],"HN",504,"00",,,,,,,,[[,"(\\d{4})(\\d{4})","$1-$2",,"","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],HR:[,[,,"[1-7]\\d{5,8}|[89]\\d{6,11}","\\d{6,12}"],[,,"1\\d{7}|(?:2[0-3]|3[1-5]|4[02-47-9]|5[1-3])\\d{6}","\\d{6,8}",,,"12345678"],[,,"9[1257-9]\\d{6,10}","\\d{8,12}",,,"912345678"],[,,"80[01]\\d{4,7}","\\d{7,10}",,,"8001234567"],[,,"6(?:[09]\\d{7}|[145]\\d{4,7})","\\d{6,9}",,,"611234"],[,,"NA","NA"],[,,"7[45]\\d{4,7}","\\d{6,9}",,,"741234567"],[,,"NA","NA"],
|
|
|
"HR",385,"00","0",,,"0",,,,[[,"(1)(\\d{4})(\\d{3})","$1 $2 $3",["1"],"0$1","",0],[,"(6[09])(\\d{4})(\\d{3})","$1 $2 $3",["6[09]"],"0$1","",0],[,"(62)(\\d{3})(\\d{3,4})","$1 $2 $3",["62"],"0$1","",0],[,"([2-5]\\d)(\\d{3})(\\d{3})","$1 $2 $3",["[2-5]"],"0$1","",0],[,"(9\\d)(\\d{3})(\\d{3,4})","$1 $2 $3",["9"],"0$1","",0],[,"(9\\d)(\\d{4})(\\d{4})","$1 $2 $3",["9"],"0$1","",0],[,"(9\\d)(\\d{3,4})(\\d{3})(\\d{3})","$1 $2 $3 $4",["9"],"0$1","",0],[,"(\\d{2})(\\d{2})(\\d{2,3})","$1 $2 $3",["6[145]|7"],
|
|
|
"0$1","",0],[,"(\\d{2})(\\d{3,4})(\\d{3})","$1 $2 $3",["6[145]|7"],"0$1","",0],[,"(80[01])(\\d{2})(\\d{2,3})","$1 $2 $3",["8"],"0$1","",0],[,"(80[01])(\\d{3,4})(\\d{3})","$1 $2 $3",["8"],"0$1","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"62\\d{6,7}","\\d{8,9}",,,"62123456"],,,[,,"NA","NA"]],HT:[,[,,"[2-489]\\d{7}","\\d{8}"],[,,"2(?:[24]\\d|5[1-5]|94)\\d{5}","\\d{8}",,,"22453300"],[,,"(?:3[1-9]|4\\d)\\d{6}","\\d{8}",,,"34101234"],[,,"8\\d{7}","\\d{8}",,,"80012345"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA",
|
|
|
"NA"],[,,"98[89]\\d{5}","\\d{8}",,,"98901234"],"HT",509,"00",,,,,,,,[[,"(\\d{2})(\\d{2})(\\d{4})","$1 $2 $3",,"","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],HU:[,[,,"[1-9]\\d{7,8}","\\d{6,9}"],[,,"(?:1\\d|2(?:1\\d|[2-9])|3[2-7]|4[24-9]|5[2-79]|6[23689]|7(?:1\\d|[2-9])|8[2-57-9]|9[2-69])\\d{6}","\\d{6,9}",,,"12345678"],[,,"(?:[27]0|3[01])\\d{7}","\\d{9}",,,"201234567"],[,,"80\\d{6}","\\d{8}",,,"80123456"],[,,"9[01]\\d{6}","\\d{8}",,,"90123456"],[,,"40\\d{6}","\\d{8}",,,"40123456"],
|
|
|
[,,"NA","NA"],[,,"NA","NA"],"HU",36,"00","06",,,"06",,,,[[,"(1)(\\d{3})(\\d{4})","$1 $2 $3",["1"],"($1)","",0],[,"(\\d{2})(\\d{3})(\\d{3,4})","$1 $2 $3",["[2-9]"],"($1)","",0]],,[,,"NA","NA"],,,[,,"[48]0\\d{6}","\\d{8}",,,"80123456"],[,,"NA","NA"],,,[,,"NA","NA"]],ID:[,[,,"[1-9]\\d{6,10}","\\d{5,11}"],[,,"2(?:1(?:14\\d{3}|[0-8]\\d{6,7}|500\\d{3}|9\\d{6})|[24]\\d{7,8})|(?:2(?:[35][1-4]|6[0-8]|7[1-6]|8\\d|9[1-8])|3(?:1|2[1-578]|3[1-68]|4[1-3]|5[1-8]|6[1-3568]|7[0-46]|8\\d)|4(?:0[1-589]|1[01347-9]|2[0-36-8]|3[0-24-68]|5[1-378]|6[1-5]|7[134]|8[1245])|5(?:1[1-35-9]|2[25-8]|3[1246-9]|4[1-3589]|5[1-46]|6[1-8])|6(?:19?|[25]\\d|3[1-469]|4[1-6])|7(?:1[1-46-9]|2[14-9]|[36]\\d|4[1-8]|5[1-9]|7[0-36-9])|9(?:0[12]|1[013-8]|2[0-479]|5[125-8]|6[23679]|7[159]|8[01346]))\\d{5,8}",
|
|
|
"\\d{5,10}",,,"612345678"],[,,"(?:2(?:1(?:3[145]|4[01]|5[1-469]|60|8[0359]|9\\d)|2(?:88|9[1256])|3[1-4]9|4(?:36|91)|5(?:1[349]|[2-4]9)|6[0-7]9|7(?:[1-36]9|4[39])|8[1-5]9|9[1-48]9)|3(?:19[1-3]|2[12]9|3[13]9|4(?:1[69]|39)|5[14]9|6(?:1[69]|2[89])|709)|4[13]19|5(?:1(?:19|8[39])|4[129]9|6[12]9)|6(?:19[12]|2(?:[23]9|77))|7(?:1[13]9|2[15]9|419|5(?:1[89]|29)|6[15]9|7[178]9))\\d{5,6}|8[1-35-9]\\d{7,9}","\\d{9,11}",,,"812345678"],[,,"177\\d{6,8}|800\\d{5,7}","\\d{8,11}",,,"8001234567"],[,,"809\\d{7}","\\d{10}",
|
|
|
,,"8091234567"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"ID",62,"0(?:0[1789]|10(?:00|1[67]))","0",,,"0",,,,[[,"(\\d{2})(\\d{5,8})","$1 $2",["2[124]|[36]1"],"(0$1)","",0],[,"(\\d{3})(\\d{5,7})","$1 $2",["[4579]|2[035-9]|[36][02-9]"],"(0$1)","",0],[,"(8\\d{2})(\\d{3,4})(\\d{3,4})","$1-$2-$3",["8[1-35-9]"],"0$1","",0],[,"(177)(\\d{6,8})","$1 $2",["1"],"0$1","",0],[,"(800)(\\d{5,7})","$1 $2",["800"],"0$1","",0],[,"(80\\d)(\\d)(\\d{3})(\\d{3})","$1 $2 $3 $4",["80[79]"],"0$1","",0]],,[,,"NA","NA"],,,
|
|
|
[,,"8071\\d{6}","\\d{10}",,,"8071123456"],[,,"8071\\d{6}","\\d{10}",,,"8071123456"],,,[,,"NA","NA"]],IE:[,[,,"[124-9]\\d{6,9}","\\d{5,10}"],[,,"1\\d{7,8}|2(?:1\\d{6,7}|3\\d{7}|[24-9]\\d{5})|4(?:0[24]\\d{5}|[1-469]\\d{7}|5\\d{6}|7\\d{5}|8[0-46-9]\\d{7})|5(?:0[45]\\d{5}|1\\d{6}|[23679]\\d{7}|8\\d{5})|6(?:1\\d{6}|[237-9]\\d{5}|[4-6]\\d{7})|7[14]\\d{7}|9(?:1\\d{6}|[04]\\d{7}|[35-9]\\d{5})","\\d{5,10}",,,"2212345"],[,,"8(?:22\\d{6}|[35-9]\\d{7})","\\d{9}",,,"850123456"],[,,"1800\\d{6}","\\d{10}",,,"1800123456"],
|
|
|
[,,"15(?:1[2-8]|[2-8]0|9[089])\\d{6}","\\d{10}",,,"1520123456"],[,,"18[59]0\\d{6}","\\d{10}",,,"1850123456"],[,,"700\\d{6}","\\d{9}",,,"700123456"],[,,"76\\d{7}","\\d{9}",,,"761234567"],"IE",353,"00","0",,,"0",,,,[[,"(1)(\\d{3,4})(\\d{4})","$1 $2 $3",["1"],"(0$1)","",0],[,"(\\d{2})(\\d{5})","$1 $2",["2[24-9]|47|58|6[237-9]|9[35-9]"],"(0$1)","",0],[,"(\\d{3})(\\d{5})","$1 $2",["40[24]|50[45]"],"(0$1)","",0],[,"(48)(\\d{4})(\\d{4})","$1 $2 $3",["48"],"(0$1)","",0],[,"(818)(\\d{3})(\\d{3})","$1 $2 $3",
|
|
|
["81"],"(0$1)","",0],[,"(\\d{2})(\\d{3})(\\d{3,4})","$1 $2 $3",["[24-69]|7[14]"],"(0$1)","",0],[,"([78]\\d)(\\d{3,4})(\\d{4})","$1 $2 $3",["76|8[35-9]"],"0$1","",0],[,"(700)(\\d{3})(\\d{3})","$1 $2 $3",["70"],"0$1","",0],[,"(\\d{4})(\\d{3})(\\d{3})","$1 $2 $3",["1(?:8[059]|5)","1(?:8[059]0|5)"],"$1","",0]],,[,,"NA","NA"],,,[,,"18[59]0\\d{6}","\\d{10}",,,"1850123456"],[,,"818\\d{6}","\\d{9}",,,"818123456"],,,[,,"8[35-9]\\d{8}","\\d{10}",,,"8501234567"]],IL:[,[,,"[17]\\d{6,9}|[2-589]\\d{3}(?:\\d{3,6})?|6\\d{3}",
|
|
|
"\\d{4,10}"],[,,"[2-489]\\d{7}","\\d{7,8}",,,"21234567"],[,,"5(?:[02347-9]\\d{2}|5(?:01|2[23]|3[34]|4[45]|5[5689]|6[67]|7[78]|8[89]|9[7-9])|6[2-9]\\d)\\d{5}","\\d{9}",,,"501234567"],[,,"1(?:80[019]\\d{3}|255)\\d{3}","\\d{7,10}",,,"1800123456"],[,,"1(?:212|(?:9(?:0[01]|19)|200)\\d{2})\\d{4}","\\d{8,10}",,,"1919123456"],[,,"1700\\d{6}","\\d{10}",,,"1700123456"],[,,"NA","NA"],[,,"7(?:2[23]\\d|3[237]\\d|47\\d|6(?:5\\d|8[068])|7\\d{2}|8(?:33|55|77|81))\\d{5}","\\d{9}",,,"771234567"],"IL",972,"0(?:0|1[2-9])",
|
|
|
"0",,,"0",,,,[[,"([2-489])(\\d{3})(\\d{4})","$1-$2-$3",["[2-489]"],"0$1","",0],[,"([57]\\d)(\\d{3})(\\d{4})","$1-$2-$3",["[57]"],"0$1","",0],[,"(1)([7-9]\\d{2})(\\d{3})(\\d{3})","$1-$2-$3-$4",["1[7-9]"],"$1","",0],[,"(1255)(\\d{3})","$1-$2",["125"],"$1","",0],[,"(1200)(\\d{3})(\\d{3})","$1-$2-$3",["120"],"$1","",0],[,"(1212)(\\d{2})(\\d{2})","$1-$2-$3",["121"],"$1","",0],[,"(1599)(\\d{6})","$1-$2",["15"],"$1","",0],[,"(\\d{4})","*$1",["[2-689]"],"$1","",0]],,[,,"NA","NA"],,,[,,"1700\\d{6}|[2-689]\\d{3}",
|
|
|
"\\d{4,10}",,,"1700123456"],[,,"[2-689]\\d{3}|1599\\d{6}","\\d{4}(?:\\d{6})?",,,"1599123456"],,,[,,"NA","NA"]],IM:[,[,,"[135789]\\d{6,9}","\\d{6,10}"],[,,"1624\\d{6}","\\d{6,10}",,,"1624456789"],[,,"7[569]24\\d{6}","\\d{10}",,,"7924123456"],[,,"808162\\d{4}","\\d{10}",,,"8081624567"],[,,"(?:872299|90[0167]624)\\d{4}","\\d{10}",,,"9016247890"],[,,"8(?:4(?:40[49]06|5624\\d)|70624\\d)\\d{3}","\\d{10}",,,"8456247890"],[,,"70\\d{8}","\\d{10}",,,"7012345678"],[,,"56\\d{8}","\\d{10}",,,"5612345678"],"IM",
|
|
|
44,"00","0"," x",,"0",,,,,,[,,"NA","NA"],,,[,,"NA","NA"],[,,"3(?:08162\\d|3\\d{5}|4(?:40[49]06|5624\\d)|7(?:0624\\d|2299\\d))\\d{3}|55\\d{8}","\\d{10}",,,"5512345678"],,,[,,"NA","NA"]],IN:[,[,,"1\\d{7,12}|[2-9]\\d{9,10}","\\d{6,13}"],[,,"(?:11|2[02]|33|4[04]|79)[2-7]\\d{7}|80[2-467]\\d{7}|(?:1(?:2[0-249]|3[0-25]|4[145]|[59][14]|6[014]|7[1257]|8[01346])|2(?:1[257]|3[013]|4[01]|5[0137]|6[0158]|78|8[1568]|9[14])|3(?:26|4[1-3]|5[34]|6[01489]|7[02-46]|8[159])|4(?:1[36]|2[1-47]|3[15]|5[12]|6[126-9]|7[0-24-9]|8[013-57]|9[014-7])|5(?:[136][25]|22|4[28]|5[12]|[78]1|9[15])|6(?:12|[2345]1|57|6[13]|7[14]|80)|7(?:12|2[14]|3[134]|4[47]|5[15]|[67]1|88)|8(?:16|2[014]|3[126]|6[136]|7[078]|8[34]|91))[2-7]\\d{6}|(?:(?:1(?:2[35-8]|3[346-9]|4[236-9]|[59][0235-9]|6[235-9]|7[34689]|8[257-9])|2(?:1[134689]|3[24-8]|4[2-8]|5[25689]|6[2-4679]|7[13-79]|8[2-479]|9[235-9])|3(?:01|1[79]|2[1-5]|4[25-8]|5[125689]|6[235-7]|7[157-9]|8[2-467])|4(?:1[14578]|2[5689]|3[2-467]|5[4-7]|6[35]|73|8[2689]|9[2389])|5(?:[16][146-9]|2[14-8]|3[1346]|4[14-69]|5[46]|7[2-4]|8[2-8]|9[246])|6(?:1[1358]|2[2457]|3[2-4]|4[235-7]|[57][2-689]|6[24-578]|8[1-6])|8(?:1[1357-9]|2[235-8]|3[03-57-9]|4[0-24-9]|5\\d|6[2457-9]|7[1-6]|8[1256]|9[2-4]))\\d|7(?:(?:1[013-9]|2[0235-9]|3[2679]|4[1-35689]|5[2-46-9]|[67][02-9]|9\\d)\\d|8(?:2[0-6]|[013-8]\\d)))[2-7]\\d{5}",
|
|
|
"\\d{6,10}",,,"1123456789"],[,,"(?:7(?:0(?:2[2-9]|[3-8]\\d|9[0-4])|2(?:0[04-9]|5[09]|7[5-8]|9[389])|3(?:0[1-9]|[58]\\d|7[3679]|9[689])|4(?:0[1-9]|1[15-9]|[29][89]|39|8[389])|5(?:[034678]\\d|2[03-9]|5[017-9]|9[7-9])|6(?:0[0127]|1[0-257-9]|2[0-4]|3[19]|5[4589]|[6-9]\\d)|7(?:0[2-9]|[1-79]\\d|8[1-9])|8(?:[0-7]\\d|9[013-9]))|8(?:0(?:[01589]\\d|6[67])|1(?:[02-589]\\d|1[0135-9]|7[0-79])|2(?:[236-9]\\d|5[1-9])|3(?:[0357-9]\\d|4[1-9])|[45]\\d{2}|6[02457-9]\\d|7[1-69]\\d|8(?:[0-26-9]\\d|44|5[2-9])|9(?:[035-9]\\d|2[2-9]|4[0-8]))|9\\d{3})\\d{6}",
|
|
|
"\\d{10}",,,"9123456789"],[,,"1(?:600\\d{6}|80(?:0\\d{4,8}|3\\d{9}))","\\d{8,13}",,,"1800123456"],[,,"186[12]\\d{9}","\\d{13}",,,"1861123456789"],[,,"1860\\d{7}","\\d{11}",,,"18603451234"],[,,"NA","NA"],[,,"NA","NA"],"IN",91,"00","0",,,"0",,,,[[,"(\\d{5})(\\d{5})","$1 $2",["7(?:0[2-9]|2[0579]|3[057-9]|4[0-389]|6[0-35-9]|[57]|8[0-79])|8(?:0[015689]|1[0-57-9]|2[2356-9]|3[0-57-9]|[45]|6[02457-9]|7[1-69]|8[0124-9]|9[02-9])|9","7(?:0(?:2[2-9]|[3-8]|9[0-4])|2(?:0[04-9]|5[09]|7[5-8]|9[389])|3(?:0[1-9]|[58]|7[3679]|9[689])|4(?:0[1-9]|1[15-9]|[29][89]|39|8[389])|5(?:[034678]|2[03-9]|5[017-9]|9[7-9])|6(?:0[0-27]|1[0-257-9]|2[0-4]|3[19]|5[4589]|[6-9])|7(?:0[2-9]|[1-79]|8[1-9])|8(?:[0-7]|9[013-9]))|8(?:0(?:[01589]|6[67])|1(?:[02-589]|1[0135-9]|7[0-79])|2(?:[236-9]|5[1-9])|3(?:[0357-9]|4[1-9])|[45]|6[02457-9]|7[1-69]|8(?:[0-26-9]|44|5[2-9])|9(?:[035-9]|2[2-9]|4[0-8]))|9"],
|
|
|
"0$1","",1],[,"(\\d{2})(\\d{4})(\\d{4})","$1 $2 $3",["11|2[02]|33|4[04]|79|80[2-46]"],"0$1","",1],[,"(\\d{3})(\\d{3})(\\d{4})","$1 $2 $3",["1(?:2[0-249]|3[0-25]|4[145]|[569][14]|7[1257]|8[1346]|[68][1-9])|2(?:1[257]|3[013]|4[01]|5[0137]|6[0158]|78|8[1568]|9[14])|3(?:26|4[1-3]|5[34]|6[01489]|7[02-46]|8[159])|4(?:1[36]|2[1-47]|3[15]|5[12]|6[126-9]|7[0-24-9]|8[013-57]|9[014-7])|5(?:[136][25]|22|4[28]|5[12]|[78]1|9[15])|6(?:12|[2345]1|57|6[13]|7[14]|80)"],"0$1","",1],[,"(\\d{3})(\\d{3})(\\d{4})","$1 $2 $3",
|
|
|
["7(?:12|2[14]|3[134]|4[47]|5[15]|[67]1|88)","7(?:12|2[14]|3[134]|4[47]|5(?:1|5[2-6])|[67]1|88)"],"0$1","",1],[,"(\\d{3})(\\d{3})(\\d{4})","$1 $2 $3",["8(?:16|2[014]|3[126]|6[136]|7[078]|8[34]|91)"],"0$1","",1],[,"(\\d{4})(\\d{3})(\\d{3})","$1 $2 $3",["1(?:[23579]|[468][1-9])|[2-8]"],"0$1","",1],[,"(1600)(\\d{2})(\\d{4})","$1 $2 $3",["160","1600"],"$1","",1],[,"(1800)(\\d{4,5})","$1 $2",["180","1800"],"$1","",1],[,"(18[06]0)(\\d{2,4})(\\d{4})","$1 $2 $3",["18[06]","18[06]0"],"$1","",1],[,"(140)(\\d{3})(\\d{4})",
|
|
|
"$1 $2 $3",["140"],"$1","",1],[,"(\\d{4})(\\d{3})(\\d{4})(\\d{2})","$1 $2 $3 $4",["18[06]","18(?:03|6[12])"],"$1","",1]],,[,,"NA","NA"],,,[,,"1(?:600\\d{6}|8(?:0(?:0\\d{4,8}|3\\d{9})|6(?:0\\d{7}|[12]\\d{9})))","\\d{8,13}",,,"1800123456"],[,,"140\\d{7}","\\d{10}",,,"1409305260"],,,[,,"NA","NA"]],IO:[,[,,"3\\d{6}","\\d{7}"],[,,"37\\d{5}","\\d{7}",,,"3709100"],[,,"38\\d{5}","\\d{7}",,,"3801234"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"IO",246,"00",,,,,,,,[[,"(\\d{3})(\\d{4})",
|
|
|
"$1 $2",,"","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],IQ:[,[,,"[1-7]\\d{7,9}","\\d{6,10}"],[,,"1\\d{7}|(?:2[13-5]|3[02367]|4[023]|5[03]|6[026])\\d{6,7}","\\d{6,9}",,,"12345678"],[,,"7[3-9]\\d{8}","\\d{10}",,,"7912345678"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"IQ",964,"00","0",,,"0",,,,[[,"(1)(\\d{3})(\\d{4})","$1 $2 $3",["1"],"0$1","",0],[,"([2-6]\\d)(\\d{3})(\\d{3,4})","$1 $2 $3",["[2-6]"],"0$1","",0],[,"(7\\d{2})(\\d{3})(\\d{4})","$1 $2 $3",
|
|
|
["7"],"0$1","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],IR:[,[,,"[14-8]\\d{6,9}|[23]\\d{4,9}|9(?:[0-4]\\d{8}|9\\d{2,8})","\\d{4,10}"],[,,"1(?:[13-589][12]|[27][1-4])\\d{7}|2(?:1\\d{3,8}|3[12]\\d{7}|4(?:1\\d{4,7}|2\\d{7})|53\\d{7}|6\\d{8}|7[34]\\d{7}|[89][12]\\d{7})|3(?:1[2-5]\\d{7}|2[1-4]\\d{7}|3(?:[125]\\d{7}|4\\d{6,7})|4(?:1\\d{6,7}[24-9]\\d{7})|5(?:1\\d{4,7}|[23]\\d{7})|[6-9][12]\\d{7})|4(?:[135-9][12]\\d{7}|2[1-467]\\d{7}|4(?:1\\d{4,7}|[2-4]\\d{7}))|5(?:1[2-5]\\d{7}|2[89]\\d{7}|3[1-5]\\d{7}|4(?:1\\d{4,7}|[2-8]\\d{7})|[5-7][12]\\d{7}|8[1245]\\d{7})|6(?:1(?:1\\d{6,7}|2\\d{7})|[347-9][12]\\d{7}|5(?:1\\d{7}|2\\d{6,7})|6[1-6]\\d{7})|7(?:1[2-5]|2[1289]|[3589][12]|4[1-4]|6[1-6]|7[1-3])\\d{7}|8(?:[145][12]|3[124578]|6[2-6]|7[1245])\\d{7}",
|
|
|
"\\d{5,10}",,,"2123456789"],[,,"9(?:0[12]|[1-3]\\d)\\d{7}","\\d{10}",,,"9123456789"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"(?:[2-6]0\\d|993)\\d{7}","\\d{10}",,,"9932123456"],"IR",98,"00","0",,,"0",,,,[[,"(21)(\\d{3,5})","$1 $2",["21"],"0$1","",0],[,"(2[15])(\\d{3})(\\d{3,4})","$1 $2 $3",["2(?:1|5[0-47-9])"],"0$1","",0],[,"(\\d{2})(\\d{4})(\\d{4})","$1 $2 $3",["2[156]|31|51|71|86"],"0$1","",0],[,"(\\d{3})(\\d{3})(\\d{3,4})","$1 $2 $3",["[13-9]|2[02-47-9]"],"0$1","",0],[,"(\\d{3})(\\d{2})(\\d{2,3})",
|
|
|
"$1 $2 $3",["[13-9]|2[02-47-9]"],"0$1","",0],[,"(\\d{3})(\\d{3})","$1 $2",["[13-9]|2[02-47-9]"],"0$1","",0]],,[,,"943\\d{7}","\\d{10}",,,"9432123456"],,,[,,"NA","NA"],[,,"9990\\d{0,6}","\\d{4,10}",,,"9990123456"],,,[,,"NA","NA"]],IS:[,[,,"[4-9]\\d{6}|38\\d{7}","\\d{7,9}"],[,,"(?:4(?:1[0-24-6]|2[0-7]|[37][0-8]|4[0-245]|5[0-3568]|6\\d|8[0-36-8])|5(?:05|[156]\\d|2[02578]|3[013-7]|4[03-7]|7[0-2578]|8[0-35-9]|9[013-689])|87[23])\\d{4}","\\d{7}",,,"4101234"],[,,"38[589]\\d{6}|(?:6(?:1[1-8]|3[089]|4[0167]|5[019]|[67][0-69]|9\\d)|7(?:5[057]|7\\d|8[0-36-8])|8(?:2[0-5]|3[0-4]|[469]\\d|5[1-9]))\\d{4}",
|
|
|
"\\d{7,9}",,,"6111234"],[,,"800\\d{4}","\\d{7}",,,"8001234"],[,,"90\\d{5}","\\d{7}",,,"9011234"],[,,"NA","NA"],[,,"NA","NA"],[,,"49\\d{5}","\\d{7}",,,"4921234"],"IS",354,"00",,,,,,,,[[,"(\\d{3})(\\d{4})","$1 $2",["[4-9]"],"","",0],[,"(3\\d{2})(\\d{3})(\\d{3})","$1 $2 $3",["3"],"","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"(?:6(?:2[0-8]|49|8\\d)|87[0189]|95[48])\\d{4}","\\d{7}",,,"6201234"]],IT:[,[,,"[01589]\\d{5,10}|3(?:[12457-9]\\d{8}|[36]\\d{7,9})","\\d{6,11}"],[,,"0(?:[26]\\d{4,9}|(?:1(?:[0159]\\d|[27][1-5]|31|4[1-4]|6[1356]|8[2-57])|3(?:[0159]\\d|2[1-4]|3[12]|[48][1-6]|6[2-59]|7[1-7])|4(?:[0159]\\d|[23][1-9]|4[245]|6[1-5]|7[1-4]|81)|5(?:[0159]\\d|2[1-5]|3[2-6]|4[1-79]|6[4-6]|7[1-578]|8[3-8])|7(?:[0159]\\d|2[12]|3[1-7]|4[2346]|6[13569]|7[13-6]|8[1-59])|8(?:[0159]\\d|2[34578]|3[1-356]|[6-8][1-5])|9(?:[0159]\\d|[238][1-5]|4[12]|6[1-8]|7[1-6]))\\d{2,7})",
|
|
|
"\\d{6,11}",,,"0212345678"],[,,"3(?:[12457-9]\\d{8}|6\\d{7,8}|3\\d{7,9})","\\d{9,11}",,,"3123456789"],[,,"80(?:0\\d{6}|3\\d{3})","\\d{6,9}",,,"800123456"],[,,"0878\\d{5}|1(?:44|6[346])\\d{6}|89(?:2\\d{3}|4(?:[0-4]\\d{2}|[5-9]\\d{4})|5(?:[0-4]\\d{2}|[5-9]\\d{6})|9\\d{6})","\\d{6,10}",,,"899123456"],[,,"84(?:[08]\\d{6}|[17]\\d{3})","\\d{6,9}",,,"848123456"],[,,"1(?:78\\d|99)\\d{6}","\\d{9,10}",,,"1781234567"],[,,"55\\d{8}","\\d{10}",,,"5512345678"],"IT",39,"00",,,,,,,,[[,"(\\d{2})(\\d{3,4})(\\d{4})",
|
|
|
"$1 $2 $3",["0[26]|55"],"","",0],[,"(0[26])(\\d{4})(\\d{5})","$1 $2 $3",["0[26]"],"","",0],[,"(0[26])(\\d{4,6})","$1 $2",["0[26]"],"","",0],[,"(0\\d{2})(\\d{3,4})(\\d{4})","$1 $2 $3",["0[13-57-9][0159]"],"","",0],[,"(\\d{3})(\\d{3,6})","$1 $2",["0[13-57-9][0159]|8(?:03|4[17]|9[245])","0[13-57-9][0159]|8(?:03|4[17]|9(?:2|[45][0-4]))"],"","",0],[,"(0\\d{3})(\\d{3})(\\d{4})","$1 $2 $3",["0[13-57-9][2-46-8]"],"","",0],[,"(0\\d{3})(\\d{2,6})","$1 $2",["0[13-57-9][2-46-8]"],"","",0],[,"(\\d{3})(\\d{3})(\\d{3,4})",
|
|
|
"$1 $2 $3",["[13]|8(?:00|4[08]|9[59])","[13]|8(?:00|4[08]|9(?:5[5-9]|9))"],"","",0],[,"(\\d{4})(\\d{4})","$1 $2",["894","894[5-9]"],"","",0],[,"(\\d{3})(\\d{4})(\\d{4})","$1 $2 $3",["3"],"","",0]],,[,,"NA","NA"],,,[,,"848\\d{6}","\\d{9}",,,"848123456"],[,,"NA","NA"],1,,[,,"NA","NA"]],JE:[,[,,"[135789]\\d{6,9}","\\d{6,10}"],[,,"1534\\d{6}","\\d{6,10}",,,"1534456789"],[,,"7(?:509|7(?:00|97)|829|937)\\d{6}","\\d{10}",,,"7797123456"],[,,"80(?:07(?:35|81)|8901)\\d{4}","\\d{10}",,,"8007354567"],[,,"(?:871206|90(?:066[59]|1810|71(?:07|55)))\\d{4}",
|
|
|
"\\d{10}",,,"9018105678"],[,,"8(?:4(?:4(?:4(?:05|42|69)|703)|5(?:041|800))|70002)\\d{4}","\\d{10}",,,"8447034567"],[,,"701511\\d{4}","\\d{10}",,,"7015115678"],[,,"56\\d{8}","\\d{10}",,,"5612345678"],"JE",44,"00","0"," x",,"0",,,,,,[,,"76(?:0[012]|2[356]|4[0134]|5[49]|6[0-369]|77|81|9[39])\\d{6}","\\d{10}",,,"7640123456"],,,[,,"NA","NA"],[,,"3(?:0(?:07(?:35|81)|8901)|3\\d{4}|4(?:4(?:4(?:05|42|69)|703)|5(?:041|800))|7(?:0002|1206))\\d{4}|55\\d{8}","\\d{10}",,,"5512345678"],,,[,,"NA","NA"]],JM:[,[,,
|
|
|
"[589]\\d{9}","\\d{7}(?:\\d{3})?"],[,,"876(?:5(?:0[12]|1[0-468]|2[35]|63)|6(?:0[1-3579]|1[027-9]|[23]\\d|40|5[06]|6[2-589]|7[05]|8[04]|9[4-9])|7(?:0[2-689]|[1-6]\\d|8[056]|9[45])|9(?:0[1-8]|1[02378]|[2-8]\\d|9[2-468]))\\d{4}","\\d{7}(?:\\d{3})?",,,"8765123456"],[,,"876(?:2[1789]\\d|[348]\\d{2}|5(?:08|27|6[0-24-9]|[3-578]\\d)|7(?:0[07]|7\\d|8[1-47-9]|9[0-36-9])|9(?:[01]9|9[0579]))\\d{4}","\\d{10}",,,"8762101234"],[,,"8(?:00|44|55|66|77|88)[2-9]\\d{6}","\\d{10}",,,"8002123456"],[,,"900[2-9]\\d{6}",
|
|
|
"\\d{10}",,,"9002123456"],[,,"NA","NA"],[,,"5(?:00|33|44|66|77)[2-9]\\d{6}","\\d{10}",,,"5002345678"],[,,"NA","NA"],"JM",1,"011","1",,,"1",,,,,,[,,"NA","NA"],,"876",[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],JO:[,[,,"[235-9]\\d{7,8}","\\d{7,9}"],[,,"(?:2(?:6(?:2[0-35-9]|3[0-57-8]|4[24-7]|5[0-24-8]|[6-8][02]|9[0-2])|7(?:0[1-79]|10|2[014-7]|3[0-689]|4[019]|5[0-3578]))|32(?:0[1-69]|1[1-35-7]|2[024-7]|3\\d|4[0-2]|[57][02]|60)|53(?:0[0-2]|[13][02]|2[0-59]|49|5[0-35-9]|6[15]|7[45]|8[1-6]|9[0-36-9])|6(?:2[50]0|300|4(?:0[0125]|1[2-7]|2[0569]|[38][07-9]|4[025689]|6[0-589]|7\\d|9[0-2])|5(?:[01][056]|2[034]|3[0-57-9]|4[17-8]|5[0-69]|6[0-35-9]|7[1-379]|8[0-68]|9[02-39]))|87(?:[02]0|7[08]|9[09]))\\d{4}",
|
|
|
"\\d{7,8}",,,"62001234"],[,,"7(?:55|7[25-9]|8[05-9]|9[015-9])\\d{6}","\\d{9}",,,"790123456"],[,,"80\\d{6}","\\d{8}",,,"80012345"],[,,"900\\d{5}","\\d{8}",,,"90012345"],[,,"85\\d{6}","\\d{8}",,,"85012345"],[,,"70\\d{7}","\\d{9}",,,"700123456"],[,,"NA","NA"],"JO",962,"00","0",,,"0",,,,[[,"(\\d)(\\d{3})(\\d{4})","$1 $2 $3",["[2356]|87"],"(0$1)","",0],[,"(7)(\\d{4})(\\d{4})","$1 $2 $3",["7[457-9]"],"0$1","",0],[,"(\\d{3})(\\d{5,6})","$1 $2",["70|8[0158]|9"],"0$1","",0]],,[,,"74(?:66|77)\\d{5}","\\d{9}",
|
|
|
,,"746612345"],,,[,,"NA","NA"],[,,"8(?:10|8\\d)\\d{5}","\\d{8}",,,"88101234"],,,[,,"NA","NA"]],JP:[,[,,"[1-9]\\d{8,9}|00(?:[36]\\d{7,14}|7\\d{5,7}|8\\d{7})","\\d{8,17}"],[,,"(?:1(?:1[235-8]|2[3-6]|3[3-9]|4[2-6]|[58][2-8]|6[2-7]|7[2-9]|9[1-9])|2[2-9]\\d|[36][1-9]\\d|4(?:6[02-8]|[2-578]\\d|9[2-59])|5(?:6[1-9]|7[2-8]|[2-589]\\d)|7(?:3[4-9]|4[02-9]|[25-9]\\d)|8(?:3[2-9]|4[5-9]|5[1-9]|8[03-9]|[2679]\\d)|9(?:[679][1-9]|[2-58]\\d))\\d{6}","\\d{9}",,,"312345678"],[,,"[7-9]0[1-9]\\d{7}","\\d{10}",,,"7012345678"],
|
|
|
[,,"120\\d{6}|800\\d{7}|00(?:37\\d{6,13}|66\\d{6,13}|777(?:[01]\\d{2}|5\\d{3}|8\\d{4})|882[1245]\\d{4})","\\d{8,17}",,,"120123456"],[,,"990\\d{6}","\\d{9}",,,"990123456"],[,,"NA","NA"],[,,"60\\d{7}","\\d{9}",,,"601234567"],[,,"50[1-9]\\d{7}","\\d{10}",,,"5012345678"],"JP",81,"010","0",,,"0",,,,[[,"(\\d{3})(\\d{3})(\\d{3})","$1-$2-$3",["(?:12|57|99)0"],"0$1","",0],[,"(\\d{3})(\\d{3})(\\d{4})","$1-$2-$3",["800"],"0$1","",0],[,"(\\d{4})(\\d{4})","$1-$2",["0077"],"$1","",0],[,"(\\d{4})(\\d{2})(\\d{3,4})",
|
|
|
"$1-$2-$3",["0077"],"$1","",0],[,"(\\d{4})(\\d{2})(\\d{4})","$1-$2-$3",["0088"],"$1","",0],[,"(\\d{4})(\\d{3})(\\d{3,4})","$1-$2-$3",["00(?:37|66)"],"$1","",0],[,"(\\d{4})(\\d{4})(\\d{4,5})","$1-$2-$3",["00(?:37|66)"],"$1","",0],[,"(\\d{4})(\\d{5})(\\d{5,6})","$1-$2-$3",["00(?:37|66)"],"$1","",0],[,"(\\d{4})(\\d{6})(\\d{6,7})","$1-$2-$3",["00(?:37|66)"],"$1","",0],[,"(\\d{2})(\\d{4})(\\d{4})","$1-$2-$3",["[2579]0|80[1-9]"],"0$1","",0],[,"(\\d{4})(\\d)(\\d{4})","$1-$2-$3",["1(?:26|3[79]|4[56]|5[4-68]|6[3-5])|5(?:76|97)|499|746|8(?:3[89]|63|47|51)|9(?:49|80|9[16])",
|
|
|
"1(?:267|3(?:7[247]|9[278])|4(?:5[67]|66)|5(?:47|58|64|8[67])|6(?:3[245]|48|5[4-68]))|5(?:76|97)9|499[2468]|7468|8(?:3(?:8[78]|96)|636|477|51[24])|9(?:496|802|9(?:1[23]|69))","1(?:267|3(?:7[247]|9[278])|4(?:5[67]|66)|5(?:47|58|64|8[67])|6(?:3[245]|48|5[4-68]))|5(?:769|979[2-69])|499[2468]|7468|8(?:3(?:8[78]|96[2457-9])|636[2-57-9]|477|51[24])|9(?:496|802|9(?:1[23]|69))"],"0$1","",0],[,"(\\d{3})(\\d{2})(\\d{4})","$1-$2-$3",["1(?:2[3-6]|3[3-9]|4[2-6]|5[2-8]|[68][2-7]|7[2-689]|9[1-578])|2(?:2[03-689]|3[3-58]|4[0-468]|5[04-8]|6[013-8]|7[06-9]|8[02-57-9]|9[13])|4(?:2[28]|3[689]|6[035-7]|7[05689]|80|9[3-5])|5(?:3[1-36-9]|4[4578]|5[013-8]|6[1-9]|7[2-8]|8[14-7]|9[4-9])|7(?:2[15]|3[5-9]|4[02-9]|6[135-8]|7[0-4689]|9[014-9])|8(?:2[49]|3[3-8]|4[5-8]|5[2-9]|6[35-9]|7[579]|8[03-579]|9[2-8])|9(?:[23]0|4[02-46-9]|5[024-79]|6[4-9]|7[2-47-9]|8[02-7]|9[3-7])",
|
|
|
"1(?:2[3-6]|3[3-9]|4[2-6]|5(?:[236-8]|[45][2-69])|[68][2-7]|7[2-689]|9[1-578])|2(?:2(?:[04-689]|3[23])|3[3-58]|4[0-468]|5(?:5[78]|7[2-4]|[0468][2-9])|6(?:[0135-8]|4[2-5])|7(?:[0679]|8[2-7])|8(?:[024578]|3[25-9]|9[6-9])|9(?:11|3[2-4]))|4(?:2(?:2[2-9]|8[237-9])|3[689]|6[035-7]|7(?:[059][2-8]|[68])|80|9[3-5])|5(?:3[1-36-9]|4[4578]|5[013-8]|6[1-9]|7[2-8]|8[14-7]|9(?:[89][2-8]|[4-7]))|7(?:2[15]|3[5-9]|4[02-9]|6[135-8]|7[0-4689]|9(?:[017-9]|4[6-8]|5[2-478]|6[2-589]))|8(?:2(?:4[4-8]|9[2-8])|3(?:7[2-6]|[3-6][2-9]|8[2-5])|4[5-8]|5[2-9]|6(?:[37]|5[4-7]|6[2-9]|8[2-8]|9[236-9])|7[579]|8[03-579]|9[2-8])|9(?:[23]0|4[02-46-9]|5[024-79]|6[4-9]|7[2-47-9]|8[02-7]|9(?:3[34]|[4-7]))",
|
|
|
"1(?:2[3-6]|3[3-9]|4[2-6]|5(?:[236-8]|[45][2-69])|[68][2-7]|7[2-689]|9[1-578])|2(?:2(?:[04-689]|3[23])|3[3-58]|4[0-468]|5(?:5[78]|7[2-4]|[0468][2-9])|6(?:[0135-8]|4[2-5])|7(?:[0679]|8[2-7])|8(?:[024578]|3[25-9]|9[6-9])|9(?:11|3[2-4]))|4(?:2(?:2[2-9]|8[237-9])|3[689]|6[035-7]|7(?:[059][2-8]|[68])|80|9[3-5])|5(?:3[1-36-9]|4[4578]|5[013-8]|6[1-9]|7[2-8]|8[14-7]|9(?:[89][2-8]|[4-7]))|7(?:2[15]|3[5-9]|4[02-9]|6[135-8]|7[0-4689]|9(?:[017-9]|4[6-8]|5[2-478]|6[2-589]))|8(?:2(?:4[4-8]|9(?:[3578]|20|4[04-9]|6[56]))|3(?:7(?:[2-5]|6[0-59])|[3-6][2-9]|8[2-5])|4[5-8]|5[2-9]|6(?:[37]|5(?:[467]|5[014-9])|6(?:[2-8]|9[02-69])|8[2-8]|9(?:[236-8]|9[23]))|7[579]|8[03-579]|9[2-8])|9(?:[23]0|4[02-46-9]|5[024-79]|6[4-9]|7[2-47-9]|8[02-7]|9(?:3(?:3[02-9]|4[0-24689])|4[2-69]|[5-7]))",
|
|
|
"1(?:2[3-6]|3[3-9]|4[2-6]|5(?:[236-8]|[45][2-69])|[68][2-7]|7[2-689]|9[1-578])|2(?:2(?:[04-689]|3[23])|3[3-58]|4[0-468]|5(?:5[78]|7[2-4]|[0468][2-9])|6(?:[0135-8]|4[2-5])|7(?:[0679]|8[2-7])|8(?:[024578]|3[25-9]|9[6-9])|9(?:11|3[2-4]))|4(?:2(?:2[2-9]|8[237-9])|3[689]|6[035-7]|7(?:[059][2-8]|[68])|80|9[3-5])|5(?:3[1-36-9]|4[4578]|5[013-8]|6[1-9]|7[2-8]|8[14-7]|9(?:[89][2-8]|[4-7]))|7(?:2[15]|3[5-9]|4[02-9]|6[135-8]|7[0-4689]|9(?:[017-9]|4[6-8]|5[2-478]|6[2-589]))|8(?:2(?:4[4-8]|9(?:[3578]|20|4[04-9]|6(?:5[25]|60)))|3(?:7(?:[2-5]|6[0-59])|[3-6][2-9]|8[2-5])|4[5-8]|5[2-9]|6(?:[37]|5(?:[467]|5[014-9])|6(?:[2-8]|9[02-69])|8[2-8]|9(?:[236-8]|9[23]))|7[579]|8[03-579]|9[2-8])|9(?:[23]0|4[02-46-9]|5[024-79]|6[4-9]|7[2-47-9]|8[02-7]|9(?:3(?:3[02-9]|4[0-24689])|4[2-69]|[5-7]))"],
|
|
|
"0$1","",0],[,"(\\d{2})(\\d{3})(\\d{4})","$1-$2-$3",["1|2(?:2[37]|5[5-9]|64|78|8[39]|91)|4(?:2[2689]|64|7[347])|5(?:[2-589]|39)|60|8(?:[46-9]|3[279]|2[124589])|9(?:[235-8]|93)","1|2(?:2[37]|5(?:[57]|[68]0|9[19])|64|78|8[39]|917)|4(?:2(?:[68]|20|9[178])|64|7[347])|5(?:[2-589]|39[67])|60|8(?:[46-9]|3[279]|2[124589])|9(?:[235-8]|93[34])","1|2(?:2[37]|5(?:[57]|[68]0|9(?:17|99))|64|78|8[39]|917)|4(?:2(?:[68]|20|9[178])|64|7[347])|5(?:[2-589]|39[67])|60|8(?:[46-9]|3[279]|2[124589])|9(?:[235-8]|93(?:31|4))"],
|
|
|
"0$1","",0],[,"(\\d{3})(\\d{2})(\\d{4})","$1-$2-$3",["2(?:9[14-79]|74|[34]7|[56]9)|82|993"],"0$1","",0],[,"(\\d)(\\d{4})(\\d{4})","$1-$2-$3",["3|4(?:2[09]|7[01])|6[1-9]"],"0$1","",0],[,"(\\d{2})(\\d{3})(\\d{4})","$1-$2-$3",["[2479][1-9]"],"0$1","",0]],[[,"(\\d{3})(\\d{3})(\\d{3})","$1-$2-$3",["(?:12|57|99)0"],"0$1","",0],[,"(\\d{3})(\\d{3})(\\d{4})","$1-$2-$3",["800"],"0$1","",0],[,"(\\d{2})(\\d{4})(\\d{4})","$1-$2-$3",["[2579]0|80[1-9]"],"0$1","",0],[,"(\\d{4})(\\d)(\\d{4})","$1-$2-$3",["1(?:26|3[79]|4[56]|5[4-68]|6[3-5])|5(?:76|97)|499|746|8(?:3[89]|63|47|51)|9(?:49|80|9[16])",
|
|
|
"1(?:267|3(?:7[247]|9[278])|4(?:5[67]|66)|5(?:47|58|64|8[67])|6(?:3[245]|48|5[4-68]))|5(?:76|97)9|499[2468]|7468|8(?:3(?:8[78]|96)|636|477|51[24])|9(?:496|802|9(?:1[23]|69))","1(?:267|3(?:7[247]|9[278])|4(?:5[67]|66)|5(?:47|58|64|8[67])|6(?:3[245]|48|5[4-68]))|5(?:769|979[2-69])|499[2468]|7468|8(?:3(?:8[78]|96[2457-9])|636[2-57-9]|477|51[24])|9(?:496|802|9(?:1[23]|69))"],"0$1","",0],[,"(\\d{3})(\\d{2})(\\d{4})","$1-$2-$3",["1(?:2[3-6]|3[3-9]|4[2-6]|5[2-8]|[68][2-7]|7[2-689]|9[1-578])|2(?:2[03-689]|3[3-58]|4[0-468]|5[04-8]|6[013-8]|7[06-9]|8[02-57-9]|9[13])|4(?:2[28]|3[689]|6[035-7]|7[05689]|80|9[3-5])|5(?:3[1-36-9]|4[4578]|5[013-8]|6[1-9]|7[2-8]|8[14-7]|9[4-9])|7(?:2[15]|3[5-9]|4[02-9]|6[135-8]|7[0-4689]|9[014-9])|8(?:2[49]|3[3-8]|4[5-8]|5[2-9]|6[35-9]|7[579]|8[03-579]|9[2-8])|9(?:[23]0|4[02-46-9]|5[024-79]|6[4-9]|7[2-47-9]|8[02-7]|9[3-7])",
|
|
|
"1(?:2[3-6]|3[3-9]|4[2-6]|5(?:[236-8]|[45][2-69])|[68][2-7]|7[2-689]|9[1-578])|2(?:2(?:[04-689]|3[23])|3[3-58]|4[0-468]|5(?:5[78]|7[2-4]|[0468][2-9])|6(?:[0135-8]|4[2-5])|7(?:[0679]|8[2-7])|8(?:[024578]|3[25-9]|9[6-9])|9(?:11|3[2-4]))|4(?:2(?:2[2-9]|8[237-9])|3[689]|6[035-7]|7(?:[059][2-8]|[68])|80|9[3-5])|5(?:3[1-36-9]|4[4578]|5[013-8]|6[1-9]|7[2-8]|8[14-7]|9(?:[89][2-8]|[4-7]))|7(?:2[15]|3[5-9]|4[02-9]|6[135-8]|7[0-4689]|9(?:[017-9]|4[6-8]|5[2-478]|6[2-589]))|8(?:2(?:4[4-8]|9[2-8])|3(?:7[2-6]|[3-6][2-9]|8[2-5])|4[5-8]|5[2-9]|6(?:[37]|5[4-7]|6[2-9]|8[2-8]|9[236-9])|7[579]|8[03-579]|9[2-8])|9(?:[23]0|4[02-46-9]|5[024-79]|6[4-9]|7[2-47-9]|8[02-7]|9(?:3[34]|[4-7]))",
|
|
|
"1(?:2[3-6]|3[3-9]|4[2-6]|5(?:[236-8]|[45][2-69])|[68][2-7]|7[2-689]|9[1-578])|2(?:2(?:[04-689]|3[23])|3[3-58]|4[0-468]|5(?:5[78]|7[2-4]|[0468][2-9])|6(?:[0135-8]|4[2-5])|7(?:[0679]|8[2-7])|8(?:[024578]|3[25-9]|9[6-9])|9(?:11|3[2-4]))|4(?:2(?:2[2-9]|8[237-9])|3[689]|6[035-7]|7(?:[059][2-8]|[68])|80|9[3-5])|5(?:3[1-36-9]|4[4578]|5[013-8]|6[1-9]|7[2-8]|8[14-7]|9(?:[89][2-8]|[4-7]))|7(?:2[15]|3[5-9]|4[02-9]|6[135-8]|7[0-4689]|9(?:[017-9]|4[6-8]|5[2-478]|6[2-589]))|8(?:2(?:4[4-8]|9(?:[3578]|20|4[04-9]|6[56]))|3(?:7(?:[2-5]|6[0-59])|[3-6][2-9]|8[2-5])|4[5-8]|5[2-9]|6(?:[37]|5(?:[467]|5[014-9])|6(?:[2-8]|9[02-69])|8[2-8]|9(?:[236-8]|9[23]))|7[579]|8[03-579]|9[2-8])|9(?:[23]0|4[02-46-9]|5[024-79]|6[4-9]|7[2-47-9]|8[02-7]|9(?:3(?:3[02-9]|4[0-24689])|4[2-69]|[5-7]))",
|
|
|
"1(?:2[3-6]|3[3-9]|4[2-6]|5(?:[236-8]|[45][2-69])|[68][2-7]|7[2-689]|9[1-578])|2(?:2(?:[04-689]|3[23])|3[3-58]|4[0-468]|5(?:5[78]|7[2-4]|[0468][2-9])|6(?:[0135-8]|4[2-5])|7(?:[0679]|8[2-7])|8(?:[024578]|3[25-9]|9[6-9])|9(?:11|3[2-4]))|4(?:2(?:2[2-9]|8[237-9])|3[689]|6[035-7]|7(?:[059][2-8]|[68])|80|9[3-5])|5(?:3[1-36-9]|4[4578]|5[013-8]|6[1-9]|7[2-8]|8[14-7]|9(?:[89][2-8]|[4-7]))|7(?:2[15]|3[5-9]|4[02-9]|6[135-8]|7[0-4689]|9(?:[017-9]|4[6-8]|5[2-478]|6[2-589]))|8(?:2(?:4[4-8]|9(?:[3578]|20|4[04-9]|6(?:5[25]|60)))|3(?:7(?:[2-5]|6[0-59])|[3-6][2-9]|8[2-5])|4[5-8]|5[2-9]|6(?:[37]|5(?:[467]|5[014-9])|6(?:[2-8]|9[02-69])|8[2-8]|9(?:[236-8]|9[23]))|7[579]|8[03-579]|9[2-8])|9(?:[23]0|4[02-46-9]|5[024-79]|6[4-9]|7[2-47-9]|8[02-7]|9(?:3(?:3[02-9]|4[0-24689])|4[2-69]|[5-7]))"],
|
|
|
"0$1","",0],[,"(\\d{2})(\\d{3})(\\d{4})","$1-$2-$3",["1|2(?:2[37]|5[5-9]|64|78|8[39]|91)|4(?:2[2689]|64|7[347])|5(?:[2-589]|39)|60|8(?:[46-9]|3[279]|2[124589])|9(?:[235-8]|93)","1|2(?:2[37]|5(?:[57]|[68]0|9[19])|64|78|8[39]|917)|4(?:2(?:[68]|20|9[178])|64|7[347])|5(?:[2-589]|39[67])|60|8(?:[46-9]|3[279]|2[124589])|9(?:[235-8]|93[34])","1|2(?:2[37]|5(?:[57]|[68]0|9(?:17|99))|64|78|8[39]|917)|4(?:2(?:[68]|20|9[178])|64|7[347])|5(?:[2-589]|39[67])|60|8(?:[46-9]|3[279]|2[124589])|9(?:[235-8]|93(?:31|4))"],
|
|
|
"0$1","",0],[,"(\\d{3})(\\d{2})(\\d{4})","$1-$2-$3",["2(?:9[14-79]|74|[34]7|[56]9)|82|993"],"0$1","",0],[,"(\\d)(\\d{4})(\\d{4})","$1-$2-$3",["3|4(?:2[09]|7[01])|6[1-9]"],"0$1","",0],[,"(\\d{2})(\\d{3})(\\d{4})","$1-$2-$3",["[2479][1-9]"],"0$1","",0]],[,,"20\\d{8}","\\d{10}",,,"2012345678"],,,[,,"00(?:37\\d{6,13}|66\\d{6,13}|777(?:[01]\\d{2}|5\\d{3}|8\\d{4})|882[1245]\\d{4})","\\d{8,17}",,,"00777012"],[,,"570\\d{6}","\\d{9}",,,"570123456"],1,,[,,"NA","NA"]],KE:[,[,,"20\\d{6,7}|[4-9]\\d{6,9}","\\d{7,10}"],
|
|
|
[,,"20\\d{6,7}|4(?:[0136]\\d{7}|[245]\\d{5,7})|5(?:[08]\\d{7}|[1-79]\\d{5,7})|6(?:[01457-9]\\d{5,7}|[26]\\d{7})","\\d{7,9}",,,"202012345"],[,,"7(?:[0-36]\\d|5[0-6]|7[0-5]|8[0-25-9])\\d{6}","\\d{9}",,,"712123456"],[,,"800[24-8]\\d{5,6}","\\d{9,10}",,,"800223456"],[,,"900[02-9]\\d{5}","\\d{9}",,,"900223456"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"KE",254,"000","0",,,"0",,,,[[,"(\\d{2})(\\d{5,7})","$1 $2",["[24-6]"],"0$1","",0],[,"(\\d{3})(\\d{6,7})","$1 $2",["7"],"0$1","",0],[,"(\\d{3})(\\d{3})(\\d{3,4})",
|
|
|
"$1 $2 $3",["[89]"],"0$1","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],KG:[,[,,"[235-8]\\d{8,9}","\\d{5,10}"],[,,"(?:3(?:1(?:[256]\\d|3[1-9]|47)|2(?:22|3[0-479]|6[0-7])|4(?:22|5[6-9]|6\\d)|5(?:22|3[4-7]|59|6\\d)|6(?:22|5[35-7]|6\\d)|7(?:22|3[468]|4[1-9]|59|[67]\\d)|9(?:22|4[1-8]|6\\d))|6(?:09|12|2[2-4])\\d)\\d{5}","\\d{5,10}",,,"312123456"],[,,"(?:20[0-35]|5[124-7]\\d|7[07]\\d)\\d{6}","\\d{9}",,,"700123456"],[,,"800\\d{6,7}","\\d{9,10}",,,"800123456"],[,,"NA","NA"],[,,"NA",
|
|
|
"NA"],[,,"NA","NA"],[,,"NA","NA"],"KG",996,"00","0",,,"0",,,,[[,"(\\d{3})(\\d{3})(\\d{3})","$1 $2 $3",["[25-7]|31[25]"],"0$1","",0],[,"(\\d{4})(\\d{5})","$1 $2",["3(?:1[36]|[2-9])"],"0$1","",0],[,"(\\d{3})(\\d{3})(\\d)(\\d{3})","$1 $2 $3 $4",["8"],"0$1","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],KH:[,[,,"[1-9]\\d{7,9}","\\d{6,10}"],[,,"(?:2[3-6]|3[2-6]|4[2-4]|[5-7][2-5])(?:[237-9]|4[56]|5\\d|6\\d?)\\d{5}|23(?:4[234]|8\\d{2})\\d{4}","\\d{6,9}",,,"23756789"],[,,"(?:1(?:[013-9]|2\\d?)|3[18]\\d|6[016-9]|7(?:[07-9]|6\\d)|8(?:[013-79]|8\\d)|9(?:6\\d|7\\d?|[0-589]))\\d{6}",
|
|
|
"\\d{8,9}",,,"91234567"],[,,"1800(?:1\\d|2[019])\\d{4}","\\d{10}",,,"1800123456"],[,,"1900(?:1\\d|2[09])\\d{4}","\\d{10}",,,"1900123456"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"KH",855,"00[14-9]","0",,,"0",,,,[[,"(\\d{2})(\\d{3})(\\d{3,4})","$1 $2 $3",["1\\d[1-9]|[2-9]"],"0$1","",0],[,"(1[89]00)(\\d{3})(\\d{3})","$1 $2 $3",["1[89]0"],"","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],KI:[,[,,"[2458]\\d{4}|3\\d{4,7}|7\\d{7}","\\d{5,8}"],[,,"(?:[24]\\d|3[1-9]|50|8[0-5])\\d{3}",
|
|
|
"\\d{5}",,,"31234"],[,,"7(?:[24]\\d|3[1-9]|8[0-5])\\d{5}","\\d{8}",,,"72012345"],[,,"NA","NA"],[,,"3001\\d{4}","\\d{5,8}",,,"30010000"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"KI",686,"00",,,,"0",,,,,,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],KM:[,[,,"[379]\\d{6}","\\d{7}"],[,,"7(?:6[0-37-9]|7[0-57-9])\\d{4}","\\d{7}",,,"7712345"],[,,"3[234]\\d{5}","\\d{7}",,,"3212345"],[,,"NA","NA"],[,,"(?:39[01]|9[01]0)\\d{4}","\\d{7}",,,"9001234"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],
|
|
|
"KM",269,"00",,,,,,,,[[,"(\\d{3})(\\d{2})(\\d{2})","$1 $2 $3",,"","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],KN:[,[,,"[589]\\d{9}","\\d{7}(?:\\d{3})?"],[,,"869(?:2(?:29|36)|302|4(?:6[015-9]|70))\\d{4}","\\d{7}(?:\\d{3})?",,,"8692361234"],[,,"869(?:5(?:5[6-8]|6[5-7])|66\\d|76[02-6])\\d{4}","\\d{10}",,,"8697652917"],[,,"8(?:00|44|55|66|77|88)[2-9]\\d{6}","\\d{10}",,,"8002123456"],[,,"900[2-9]\\d{6}","\\d{10}",,,"9002123456"],[,,"NA","NA"],[,,"5(?:00|33|44|66|77)[2-9]\\d{6}",
|
|
|
"\\d{10}",,,"5002345678"],[,,"NA","NA"],"KN",1,"011","1",,,"1",,,,,,[,,"NA","NA"],,"869",[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],KP:[,[,,"1\\d{9}|[28]\\d{7}","\\d{6,8}|\\d{10}"],[,,"2\\d{7}|85\\d{6}","\\d{6,8}",,,"21234567"],[,,"19[123]\\d{7}","\\d{10}",,,"1921234567"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"KP",850,"00|99","0",,,"0",,,,[[,"(\\d{3})(\\d{3})(\\d{4})","$1 $2 $3",["1"],"0$1","",0],[,"(\\d)(\\d{3})(\\d{4})","$1 $2 $3",["2"],"0$1","",0],[,"(\\d{2})(\\d{3})(\\d{3})",
|
|
|
"$1 $2 $3",["8"],"0$1","",0]],,[,,"NA","NA"],,,[,,"2(?:[0-24-9]\\d{2}|3(?:[0-79]\\d|8[02-9]))\\d{4}","\\d{8}",,,"23821234"],[,,"NA","NA"],,,[,,"NA","NA"]],KR:[,[,,"[1-7]\\d{3,9}|8\\d{8}","\\d{4,10}"],[,,"(?:2|3[1-3]|[46][1-4]|5[1-5])(?:1\\d{2,3}|[1-9]\\d{6,7})","\\d{4,10}",,,"22123456"],[,,"1[0-26-9]\\d{7,8}","\\d{9,10}",,,"1023456789"],[,,"80\\d{7}","\\d{9}",,,"801234567"],[,,"60[2-9]\\d{6}","\\d{9}",,,"602345678"],[,,"NA","NA"],[,,"50\\d{8}","\\d{10}",,,"5012345678"],[,,"70\\d{8}","\\d{10}",,,"7012345678"],
|
|
|
"KR",82,"00(?:[124-68]|[37]\\d{2})","0",,,"0(8[1-46-8]|85\\d{2})?",,,,[[,"(\\d{2})(\\d{4})(\\d{4})","$1-$2-$3",["1(?:0|1[19]|[69]9|5[458])|[57]0","1(?:0|1[19]|[69]9|5(?:44|59|8))|[57]0"],"0$1","0$CC-$1",0],[,"(\\d{2})(\\d{3,4})(\\d{4})","$1-$2-$3",["1(?:[169][2-8]|[78]|5[1-4])|[68]0|[3-6][1-9][1-9]","1(?:[169][2-8]|[78]|5(?:[1-3]|4[56]))|[68]0|[3-6][1-9][1-9]"],"0$1","0$CC-$1",0],[,"(\\d{3})(\\d)(\\d{4})","$1-$2-$3",["131","1312"],"0$1","0$CC-$1",0],[,"(\\d{3})(\\d{2})(\\d{4})","$1-$2-$3",["131",
|
|
|
"131[13-9]"],"0$1","0$CC-$1",0],[,"(\\d{3})(\\d{3})(\\d{4})","$1-$2-$3",["13[2-9]"],"0$1","0$CC-$1",0],[,"(\\d{2})(\\d{2})(\\d{3})(\\d{4})","$1-$2-$3-$4",["30"],"0$1","0$CC-$1",0],[,"(\\d)(\\d{3,4})(\\d{4})","$1-$2-$3",["2[1-9]"],"0$1","0$CC-$1",0],[,"(\\d)(\\d{3,4})","$1-$2",["21[0-46-9]"],"0$1","0$CC-$1",0],[,"(\\d{2})(\\d{3,4})","$1-$2",["[3-6][1-9]1","[3-6][1-9]1(?:[0-46-9])"],"0$1","0$CC-$1",0],[,"(\\d{4})(\\d{4})","$1-$2",["1(?:5[46-9]|6[04678])","1(?:5(?:44|66|77|88|99)|6(?:00|44|6[16]|70|88))"],
|
|
|
"$1","0$CC-$1",0]],,[,,"15\\d{7,8}","\\d{9,10}",,,"1523456789"],,,[,,"NA","NA"],[,,"1(?:5(?:44|66|77|88|99)|6(?:00|44|6[16]|70|88))\\d{4}","\\d{8}",,,"15441234"],,,[,,"NA","NA"]],KW:[,[,,"[12569]\\d{6,7}","\\d{7,8}"],[,,"(?:18\\d|2(?:[23]\\d{2}|4(?:[1-35-9]\\d|44)|5(?:0[034]|[2-46]\\d|5[1-3]|7[1-7])))\\d{4}","\\d{7,8}",,,"22345678"],[,,"(?:5(?:[05]\\d|1[0-6])|6(?:0[034679]|5[015-9]|6\\d|7[067]|9[0369])|9(?:0[09]|4[049]|55|6[069]|[79]\\d|8[089]))\\d{5}","\\d{8}",,,"50012345"],[,,"NA","NA"],[,,"NA",
|
|
|
"NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"KW",965,"00",,,,,,,,[[,"(\\d{4})(\\d{3,4})","$1 $2",["[1269]"],"","",0],[,"(5[015]\\d)(\\d{5})","$1 $2",["5"],"","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],KY:[,[,,"[3589]\\d{9}","\\d{7}(?:\\d{3})?"],[,,"345(?:2(?:22|44)|444|6(?:23|38|40)|7(?:4[35-79]|6[6-9]|77)|8(?:00|1[45]|25|[48]8)|9(?:14|4[035-9]))\\d{4}","\\d{7}(?:\\d{3})?",,,"3452221234"],[,,"345(?:32[1-9]|5(?:1[67]|2[5-7]|4[6-8]|76)|9(?:1[67]|2[3-9]|3[689]))\\d{4}",
|
|
|
"\\d{10}",,,"3453231234"],[,,"8(?:00|44|55|66|77|88)[2-9]\\d{6}","\\d{10}",,,"8002345678"],[,,"900[2-9]\\d{6}|345976\\d{4}","\\d{10}",,,"9002345678"],[,,"NA","NA"],[,,"5(?:00|33|44|66|77)[2-9]\\d{6}","\\d{10}",,,"5002345678"],[,,"NA","NA"],"KY",1,"011","1",,,"1",,,,,,[,,"345849\\d{4}","\\d{10}",,,"3458491234"],,"345",[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],KZ:[,[,,"(?:33\\d|7\\d{2}|80[09])\\d{7}","\\d{10}"],[,,"33622\\d{5}|7(?:1(?:0(?:[23]\\d|4[023]|59|63)|1(?:[23]\\d|4[0-79]|59)|2(?:[23]\\d|59)|3(?:2\\d|3[1-79]|4[0-35-9]|59)|4(?:2\\d|3[013-79]|4[0-8]|5[1-79])|5(?:2\\d|3[1-8]|4[1-7]|59)|6(?:[234]\\d|5[19]|61)|72\\d|8(?:[27]\\d|3[1-46-9]|4[0-5]))|2(?:1(?:[23]\\d|4[46-9]|5[3469])|2(?:2\\d|3[0679]|46|5[12679])|3(?:[234]\\d|5[139])|4(?:2\\d|3[1235-9]|59)|5(?:[23]\\d|4[01246-8]|59|61)|6(?:2\\d|3[1-9]|4[0-4]|59)|7(?:[237]\\d|40|5[279])|8(?:[23]\\d|4[0-3]|59)|9(?:2\\d|3[124578]|59)))\\d{5}",
|
|
|
"\\d{10}",,,"7123456789"],[,,"7(?:0[012578]|47|6[02-4]|7[15-8]|85)\\d{7}","\\d{10}",,,"7710009998"],[,,"800\\d{7}","\\d{10}",,,"8001234567"],[,,"809\\d{7}","\\d{10}",,,"8091234567"],[,,"NA","NA"],[,,"NA","NA"],[,,"751\\d{7}","\\d{10}",,,"7511234567"],"KZ",7,"810","8",,,"8",,"8~10",,,,[,,"NA","NA"],,,[,,"751\\d{7}","\\d{10}",,,"7511234567"],[,,"NA","NA"],,,[,,"NA","NA"]],LA:[,[,,"[2-8]\\d{7,9}","\\d{6,10}"],[,,"(?:2[13]|3(?:0\\d|[14])|[5-7][14]|41|8[1468])\\d{6}","\\d{6,9}",,,"21212862"],[,,"20(?:2[2389]|5[4-689]|7[6-8]|9[57-9])\\d{6}",
|
|
|
"\\d{10}",,,"2023123456"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"LA",856,"00","0",,,"0",,,,[[,"(20)(\\d{2})(\\d{3})(\\d{3})","$1 $2 $3 $4",["20"],"0$1","",0],[,"([2-8]\\d)(\\d{3})(\\d{3})","$1 $2 $3",["2[13]|3[14]|[4-8]"],"0$1","",0],[,"(30)(\\d{2})(\\d{2})(\\d{3})","$1 $2 $3 $4",["30"],"0$1","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],LB:[,[,,"[13-9]\\d{6,7}","\\d{7,8}"],[,,"(?:[14-6]\\d{2}|7(?:[2-579]\\d|62|8[0-7])|[89][2-9]\\d)\\d{4}","\\d{7}",
|
|
|
,,"1123456"],[,,"(?:3\\d|7(?:[019]\\d|6[013-9]|8[89]))\\d{5}","\\d{7,8}",,,"71123456"],[,,"NA","NA"],[,,"9[01]\\d{6}","\\d{8}",,,"90123456"],[,,"8[01]\\d{6}","\\d{8}",,,"80123456"],[,,"NA","NA"],[,,"NA","NA"],"LB",961,"00","0",,,"0",,,,[[,"(\\d)(\\d{3})(\\d{3})","$1 $2 $3",["[13-6]|7(?:[2-579]|62|8[0-7])|[89][2-9]"],"0$1","",0],[,"([7-9]\\d)(\\d{3})(\\d{3})","$1 $2 $3",["[89][01]|7(?:[019]|6[013-9]|8[89])"],"","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],LC:[,[,,"[5789]\\d{9}",
|
|
|
"\\d{7}(?:\\d{3})?"],[,,"758(?:4(?:30|5[0-9]|6[2-9]|8[0-2])|57[0-2]|638)\\d{4}","\\d{7}(?:\\d{3})?",,,"7584305678"],[,,"758(?:28[4-7]|384|4(?:6[01]|8[4-9])|5(?:1[89]|20|84)|7(?:1[2-9]|2[0-8]))\\d{4}","\\d{10}",,,"7582845678"],[,,"8(?:00|44|55|66|77|88)[2-9]\\d{6}","\\d{10}",,,"8002123456"],[,,"900[2-9]\\d{6}","\\d{10}",,,"9002123456"],[,,"NA","NA"],[,,"5(?:00|33|44|66|77)[2-9]\\d{6}","\\d{10}",,,"5002345678"],[,,"NA","NA"],"LC",1,"011","1",,,"1",,,,,,[,,"NA","NA"],,"758",[,,"NA","NA"],[,,"NA","NA"],
|
|
|
,,[,,"NA","NA"]],LI:[,[,,"6\\d{8}|[23789]\\d{6}","\\d{7,9}"],[,,"(?:2(?:01|1[27]|3\\d|6[02-578]|96)|3(?:7[0135-7]|8[048]|9[0269]))\\d{4}","\\d{7}",,,"2345678"],[,,"6(?:51[01]|6(?:[01][0-4]|2[016-9]|88)|710)\\d{5}|7(?:36|4[25]|56|[7-9]\\d)\\d{4}","\\d{7,9}",,,"661234567"],[,,"80(?:0(?:2[238]|79)|9\\d{2})\\d{2}","\\d{7}",,,"8002222"],[,,"90(?:0(?:2[278]|79)|1(?:23|3[012])|6(?:4\\d|6[0126]))\\d{2}","\\d{7}",,,"9002222"],[,,"NA","NA"],[,,"701\\d{4}","\\d{7}",,,"7011234"],[,,"NA","NA"],"LI",423,"00","0",
|
|
|
,,"0",,,,[[,"(\\d{3})(\\d{2})(\\d{2})","$1 $2 $3",["[23]|7[3-57-9]|87"],"","",0],[,"(6\\d)(\\d{3})(\\d{3})","$1 $2 $3",["6"],"","",0],[,"(6[567]\\d)(\\d{3})(\\d{3})","$1 $2 $3",["6[567]"],"","",0],[,"(69)(7\\d{2})(\\d{4})","$1 $2 $3",["697"],"","",0],[,"([7-9]0\\d)(\\d{2})(\\d{2})","$1 $2 $3",["[7-9]0"],"","",0],[,"([89]0\\d)(\\d{2})(\\d{2})(\\d{2})","$1 $2 $3 $4",["[89]0"],"0$1","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"87(?:0[128]|7[0-4])\\d{3}","\\d{7}",,,"8770123"],,,[,,"697(?:[35]6|4[25]|[7-9]\\d)\\d{4}",
|
|
|
"\\d{9}",,,"697361234"]],LK:[,[,,"[1-9]\\d{8}","\\d{7,9}"],[,,"(?:[189]1|2[13-7]|3[1-8]|4[157]|5[12457]|6[35-7])[2-57]\\d{6}","\\d{7,9}",,,"112345678"],[,,"7[125-8]\\d{7}","\\d{9}",,,"712345678"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"LK",94,"00","0",,,"0",,,,[[,"(\\d{2})(\\d{1})(\\d{6})","$1 $2 $3",["[1-689]"],"0$1","",0],[,"(\\d{2})(\\d{3})(\\d{4})","$1 $2 $3",["7"],"0$1","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],LR:[,[,,"2\\d{7}|[37-9]\\d{8}|[45]\\d{6}",
|
|
|
"\\d{7,9}"],[,,"2\\d{7}","\\d{8}",,,"21234567"],[,,"(?:330\\d|4[67]|5\\d|77\\d{2}|88\\d{2}|994\\d)\\d{5}","\\d{7,9}",,,"770123456"],[,,"NA","NA"],[,,"90[03]\\d{6}","\\d{9}",,,"900123456"],[,,"NA","NA"],[,,"NA","NA"],[,,"332(?:0[02]|5\\d)\\d{4}","\\d{9}",,,"332001234"],"LR",231,"00","0",,,"0",,,,[[,"(2\\d)(\\d{3})(\\d{3})","$1 $2 $3",["2"],"0$1","",0],[,"([79]\\d{2})(\\d{3})(\\d{3})","$1 $2 $3",["[79]"],"0$1","",0],[,"([4-6])(\\d{3})(\\d{3})","$1 $2 $3",["[4-6]"],"0$1","",0],[,"(\\d{2})(\\d{3})(\\d{4})",
|
|
|
"$1 $2 $3",["[38]"],"0$1","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],LS:[,[,,"[2568]\\d{7}","\\d{8}"],[,,"2\\d{7}","\\d{8}",,,"22123456"],[,,"[56]\\d{7}","\\d{8}",,,"50123456"],[,,"800[256]\\d{4}","\\d{8}",,,"80021234"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"LS",266,"00",,,,,,,,[[,"(\\d{4})(\\d{4})","$1 $2",,"","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],LT:[,[,,"[3-9]\\d{7}","\\d{8}"],[,,"(?:3[1478]|4[124-6]|52)\\d{6}","\\d{8}",
|
|
|
,,"31234567"],[,,"6\\d{7}","\\d{8}",,,"61234567"],[,,"800\\d{5}","\\d{8}",,,"80012345"],[,,"9(?:0[0239]|10)\\d{5}","\\d{8}",,,"90012345"],[,,"808\\d{5}","\\d{8}",,,"80812345"],[,,"700\\d{5}","\\d{8}",,,"70012345"],[,,"NA","NA"],"LT",370,"00","8",,,"[08]",,,,[[,"([34]\\d)(\\d{6})","$1 $2",["37|4(?:1|5[45]|6[2-4])"],"(8-$1)","",1],[,"([3-6]\\d{2})(\\d{5})","$1 $2",["3[148]|4(?:[24]|6[09])|528|6"],"(8-$1)","",1],[,"([7-9]\\d{2})(\\d{2})(\\d{3})","$1 $2 $3",["[7-9]"],"8 $1","",1],[,"(5)(2\\d{2})(\\d{4})",
|
|
|
"$1 $2 $3",["52[0-79]"],"(8-$1)","",1]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"70[67]\\d{5}","\\d{8}",,,"70712345"],,,[,,"NA","NA"]],LU:[,[,,"[24-9]\\d{3,10}|3(?:[0-46-9]\\d{2,9}|5[013-9]\\d{1,8})","\\d{4,11}"],[,,"(?:2(?:2\\d{1,2}|3[2-9]|[67]\\d|4[1-8]\\d?|5[1-5]\\d?|9[0-24-9]\\d?)|3(?:[059][05-9]|[13]\\d|[26][015-9]|4[0-26-9]|7[0-389]|8[08])\\d?|4\\d{2,3}|5(?:[01458]\\d|[27][0-69]|3[0-3]|[69][0-7])\\d?|7(?:1[019]|2[05-9]|3[05]|[45][07-9]|[679][089]|8[06-9])\\d?|8(?:0[2-9]|1[0-36-9]|3[3-9]|[469]9|[58][7-9]|7[89])\\d?|9(?:0[89]|2[0-49]|37|49|5[0-27-9]|7[7-9]|9[0-478])\\d?)\\d{1,7}",
|
|
|
"\\d{4,11}",,,"27123456"],[,,"6(?:[269][18]|71)\\d{6}","\\d{9}",,,"628123456"],[,,"800\\d{5}","\\d{8}",,,"80012345"],[,,"90[01]\\d{5}","\\d{8}",,,"90012345"],[,,"801\\d{5}","\\d{8}",,,"80112345"],[,,"70\\d{6}","\\d{8}",,,"70123456"],[,,"20(?:1\\d{5}|[2-689]\\d{1,7})","\\d{4,10}",,,"20201234"],"LU",352,"00",,,,"(15(?:0[06]|1[12]|35|4[04]|55|6[26]|77|88|99)\\d)",,,,[[,"(\\d{2})(\\d{3})","$1 $2",["[2-5]|7[1-9]|[89](?:[1-9]|0[2-9])"],"","$CC $1",0],[,"(\\d{2})(\\d{2})(\\d{2})","$1 $2 $3",["[2-5]|7[1-9]|[89](?:[1-9]|0[2-9])"],
|
|
|
"","$CC $1",0],[,"(\\d{2})(\\d{2})(\\d{3})","$1 $2 $3",["20"],"","$CC $1",0],[,"(\\d{2})(\\d{2})(\\d{2})(\\d{1,2})","$1 $2 $3 $4",["2(?:[0367]|4[3-8])"],"","$CC $1",0],[,"(\\d{2})(\\d{2})(\\d{2})(\\d{3})","$1 $2 $3 $4",["20"],"","$CC $1",0],[,"(\\d{2})(\\d{2})(\\d{2})(\\d{2})(\\d{1,2})","$1 $2 $3 $4 $5",["2(?:[0367]|4[3-8])"],"","$CC $1",0],[,"(\\d{2})(\\d{2})(\\d{2})(\\d{1,4})","$1 $2 $3 $4",["2(?:[12589]|4[12])|[3-5]|7[1-9]|[89](?:[1-9]|0[2-9])"],"","$CC $1",0],[,"(\\d{3})(\\d{2})(\\d{3})","$1 $2 $3",
|
|
|
["[89]0[01]|70"],"","$CC $1",0],[,"(\\d{3})(\\d{3})(\\d{3})","$1 $2 $3",["6"],"","$CC $1",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],LV:[,[,,"[2689]\\d{7}","\\d{8}"],[,,"6[3-8]\\d{6}","\\d{8}",,,"63123456"],[,,"2\\d{7}","\\d{8}",,,"21234567"],[,,"80\\d{6}","\\d{8}",,,"80123456"],[,,"90\\d{6}","\\d{8}",,,"90123456"],[,,"81\\d{6}","\\d{8}",,,"81123456"],[,,"NA","NA"],[,,"NA","NA"],"LV",371,"00",,,,,,,,[[,"([2689]\\d)(\\d{3})(\\d{3})","$1 $2 $3",,"","",0]],,[,,"NA","NA"],,,[,,
|
|
|
"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],LY:[,[,,"[25679]\\d{8}","\\d{7,9}"],[,,"(?:2[1345]|5[1347]|6[123479]|71)\\d{7}","\\d{7,9}",,,"212345678"],[,,"9[1-6]\\d{7}","\\d{9}",,,"912345678"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"LY",218,"00","0",,,"0",,,,[[,"([25679]\\d)(\\d{7})","$1-$2",,"0$1","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],MA:[,[,,"[5689]\\d{8}","\\d{9}"],[,,"5(?:2(?:(?:[015-7]\\d|2[2-9]|3[2-57]|4[2-8]|8[235-7])\\d|9(?:0\\d|[89]0))|3(?:(?:[0-4]\\d|[57][2-9]|6[235-8]|9[3-9])\\d|8(?:0\\d|[89]0)))\\d{4}",
|
|
|
"\\d{9}",,,"520123456"],[,,"6(?:0[0-8]|[12-7]\\d|8[01]|9[2457-9])\\d{6}","\\d{9}",,,"650123456"],[,,"80\\d{7}","\\d{9}",,,"801234567"],[,,"89\\d{7}","\\d{9}",,,"891234567"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"MA",212,"00","0",,,"0",,,,[[,"([56]\\d{2})(\\d{6})","$1-$2",["5(?:2[015-7]|3[0-4])|6"],"0$1","",0],[,"([58]\\d{3})(\\d{5})","$1-$2",["5(?:2[2-489]|3[5-9])|892","5(?:2(?:[2-48]|90)|3(?:[5-79]|80))|892"],"0$1","",0],[,"(5\\d{4})(\\d{4})","$1-$2",["5(?:29|38)","5(?:29|38)[89]"],"0$1","",
|
|
|
0],[,"(8[09])(\\d{7})","$1-$2",["8(?:0|9[013-9])"],"0$1","",0]],,[,,"NA","NA"],1,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],MC:[,[,,"[4689]\\d{7,8}","\\d{8,9}"],[,,"870\\d{5}|9[2-47-9]\\d{6}","\\d{8}",,,"99123456"],[,,"6\\d{8}|4(?:4\\d|5[2-9])\\d{5}","\\d{8,9}",,,"612345678"],[,,"90\\d{6}","\\d{8}",,,"90123456"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"MC",377,"00","0",,,"0",,,,[[,"(\\d{2})(\\d{2})(\\d{2})(\\d{2})","$1 $2 $3 $4",["9"],"$1","",0],[,"(\\d{2})(\\d{3})(\\d{3})","$1 $2 $3",
|
|
|
["4"],"0$1","",0],[,"(6)(\\d{2})(\\d{2})(\\d{2})(\\d{2})","$1 $2 $3 $4 $5",["6"],"0$1","",0],[,"(\\d{3})(\\d{3})(\\d{2})","$1 $2 $3",["8"],"$1","",0]],,[,,"NA","NA"],,,[,,"8\\d{7}","\\d{8}"],[,,"NA","NA"],,,[,,"NA","NA"]],MD:[,[,,"[235-9]\\d{7}","\\d{8}"],[,,"(?:2(?:1[0569]|2\\d|3[015-7]|4[1-46-9]|5[0-24689]|6[2-589]|7[1-37]|9[1347-9])|5(?:33|5[257]))\\d{5}","\\d{8}",,,"22212345"],[,,"(?:562\\d|6(?:[089]\\d{2}|1[01]\\d|21\\d|50\\d|7(?:[1-6]\\d|7[0-4]))|7(?:6[07]|7[457-9]|[89]\\d)\\d)\\d{4}","\\d{8}",
|
|
|
,,"65012345"],[,,"800\\d{5}","\\d{8}",,,"80012345"],[,,"90[056]\\d{5}","\\d{8}",,,"90012345"],[,,"808\\d{5}","\\d{8}",,,"80812345"],[,,"NA","NA"],[,,"3[08]\\d{6}","\\d{8}",,,"30123456"],"MD",373,"00","0",,,"0",,,,[[,"(\\d{2})(\\d{3})(\\d{3})","$1 $2 $3",["22|3"],"0$1","",0],[,"([25-7]\\d{2})(\\d{2})(\\d{3})","$1 $2 $3",["2[13-79]|[5-7]"],"0$1","",0],[,"([89]\\d{2})(\\d{5})","$1 $2",["[89]"],"0$1","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"8(?:03|14)\\d{5}","\\d{8}",,,"80312345"],,,[,,"NA","NA"]],ME:[,
|
|
|
[,,"[2-9]\\d{7,8}","\\d{6,9}"],[,,"(?:20[2-8]|3(?:0[2-7]|1[35-7]|2[3567]|3[4-7])|4(?:0[237]|1[27])|5(?:0[47]|1[27]|2[378]))\\d{5}","\\d{6,8}",,,"30234567"],[,,"6(?:32\\d|[89]\\d{2}|7(?:[0-8]\\d|9(?:[3-9]|[0-2]\\d)))\\d{4}","\\d{8,9}",,,"67622901"],[,,"800[28]\\d{4}","\\d{8}",,,"80080002"],[,,"(?:88\\d|9(?:4[13-8]|5[16-8]))\\d{5}","\\d{8}",,,"94515151"],[,,"NA","NA"],[,,"NA","NA"],[,,"78[1-9]\\d{5}","\\d{8}",,,"78108780"],"ME",382,"00","0",,,"0",,,,[[,"(\\d{2})(\\d{3})(\\d{3})","$1 $2 $3",["[2-57-9]|6[3789]",
|
|
|
"[2-57-9]|6(?:[389]|7(?:[0-8]|9[3-9]))"],"0$1","",0],[,"(67)(9)(\\d{3})(\\d{3})","$1 $2 $3 $4",["679","679[0-2]"],"0$1","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"77\\d{6}","\\d{8}",,,"77273012"],,,[,,"NA","NA"]],MF:[,[,,"[56]\\d{8}","\\d{9}"],[,,"590(?:[02][79]|13|5[0-268]|[78]7)\\d{4}","\\d{9}",,,"590271234"],[,,"690(?:0[0-7]|[1-9]\\d)\\d{4}","\\d{9}",,,"690301234"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"MF",590,"00","0",,,"0",,,,,,[,,"NA","NA"],,,[,,"NA","NA"],[,
|
|
|
,"NA","NA"],,,[,,"NA","NA"]],MG:[,[,,"[23]\\d{8}","\\d{7,9}"],[,,"20(?:2\\d{2}|4[47]\\d|5[3467]\\d|6[279]\\d|7(?:2[29]|[35]\\d)|8[268]\\d|9[245]\\d)\\d{4}","\\d{7,9}",,,"202123456"],[,,"3[2-49]\\d{7}","\\d{9}",,,"321234567"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"22\\d{7}","\\d{9}",,,"221234567"],"MG",261,"00","0",,,"0",,,,[[,"([23]\\d)(\\d{2})(\\d{3})(\\d{2})","$1 $2 $3 $4",,"0$1","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],MH:[,[,,"[2-6]\\d{6}","\\d{7}"],
|
|
|
[,,"(?:247|528|625)\\d{4}","\\d{7}",,,"2471234"],[,,"(?:235|329|45[56]|545)\\d{4}","\\d{7}",,,"2351234"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"635\\d{4}","\\d{7}",,,"6351234"],"MH",692,"011","1",,,"1",,,,[[,"(\\d{3})(\\d{4})","$1-$2",,"","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],MK:[,[,,"[2-578]\\d{7}","\\d{8}"],[,,"(?:2(?:[23]\\d|5[124578]|6[01])|3(?:1[3-6]|[23][2-6]|4[2356])|4(?:[23][2-6]|4[3-6]|5[256]|6[25-8]|7[24-6]|8[4-6]))\\d{5}","\\d{6,8}",,
|
|
|
,"22212345"],[,,"7(?:[0-25-8]\\d{2}|32\\d|421)\\d{4}","\\d{8}",,,"72345678"],[,,"800\\d{5}","\\d{8}",,,"80012345"],[,,"5[02-9]\\d{6}","\\d{8}",,,"50012345"],[,,"8(?:0[1-9]|[1-9]\\d)\\d{5}","\\d{8}",,,"80123456"],[,,"NA","NA"],[,,"NA","NA"],"MK",389,"00","0",,,"0",,,,[[,"(2)(\\d{3})(\\d{4})","$1 $2 $3",["2"],"0$1","",0],[,"([347]\\d)(\\d{3})(\\d{3})","$1 $2 $3",["[347]"],"0$1","",0],[,"([58]\\d{2})(\\d)(\\d{2})(\\d{2})","$1 $2 $3 $4",["[58]"],"0$1","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],
|
|
|
,,[,,"NA","NA"]],ML:[,[,,"[246-9]\\d{7}","\\d{8}"],[,,"(?:2(?:0(?:2[0-589]|7\\d)|1(?:2[5-7]|[3-689]\\d|7[2-4689]))|44[239]\\d)\\d{4}","\\d{8}",,,"20212345"],[,,"[67]\\d{7}|9[0-25-9]\\d{6}","\\d{8}",,,"65012345"],[,,"800\\d{5}","\\d{8}",,,"80012345"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"ML",223,"00",,,,,,,,[[,"(\\d{2})(\\d{2})(\\d{2})(\\d{2})","$1 $2 $3 $4",["[246-9]"],"","",0],[,"(\\d{4})","$1",["67|74"],"","",0]],[[,"(\\d{2})(\\d{2})(\\d{2})(\\d{2})","$1 $2 $3 $4",["[246-9]"],
|
|
|
"","",0]],[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],MM:[,[,,"[14578]\\d{5,7}|[26]\\d{5,8}|9(?:2\\d{0,2}|[58]|3\\d|4\\d{1,2}|6\\d?|[79]\\d{0,2})\\d{6}","\\d{5,10}"],[,,"1(?:2\\d{1,2}|[3-5]\\d|6\\d?|[89][0-6]\\d)\\d{4}|2(?:[236-9]\\d{4}|4(?:0\\d{5}|\\d{4})|5(?:1\\d{3,6}|[02-9]\\d{3,5}))|4(?:2[245-8]|[346][2-6]|5[3-5])\\d{4}|5(?:2(?:20?|[3-8])|3[2-68]|4(?:21?|[4-8])|5[23]|6[2-4]|7[2-8]|8[24-7]|9[2-7])\\d{4}|6(?:0[23]|1[2356]|[24][2-6]|3[24-6]|5[2-4]|6[2-8]|7(?:[2367]|4\\d|5\\d?|8[145]\\d)|8[245]|9[24])\\d{4}|7(?:[04][24-8]|[15][2-7]|22|3[2-4])\\d{4}|8(?:1(?:2\\d?|[3-689])|2[2-8]|3[24]|4[24-7]|5[245]|6[23])\\d{4}",
|
|
|
"\\d{5,9}",,,"1234567"],[,,"17[01]\\d{4}|9(?:2(?:[0-4]|5\\d{2})|3[136]\\d|4(?:0[0-4]\\d|[1379]\\d|[24][0-589]\\d|5\\d{2}|88)|5[0-6]|61?\\d|7(?:3\\d|9\\d{2})|8\\d|9(?:1\\d|7\\d{2}|[089]))\\d{5}","\\d{7,10}",,,"92123456"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"1333\\d{4}","\\d{8}",,,"13331234"],"MM",95,"00","0",,,"0",,,,[[,"(\\d)(\\d{3})(\\d{3,4})","$1 $2 $3",["1|2[45]"],"0$1","",0],[,"(2)(\\d{4})(\\d{4})","$1 $2 $3",["251"],"0$1","",0],[,"(\\d)(\\d{2})(\\d{3})","$1 $2 $3",["16|2"],
|
|
|
"0$1","",0],[,"(\\d{2})(\\d{3})(\\d{3,4})","$1 $2 $3",["67|81"],"0$1","",0],[,"(\\d{2})(\\d{2})(\\d{3,4})","$1 $2 $3",["[4-8]"],"0$1","",0],[,"(9)(\\d{3})(\\d{4,6})","$1 $2 $3",["9(?:2[0-4]|[35-9]|4[13789])"],"0$1","",0],[,"(9)(4\\d{4})(\\d{4})","$1 $2 $3",["94[0245]"],"0$1","",0],[,"(9)(\\d{3})(\\d{3})(\\d{3})","$1 $2 $3 $4",["925"],"0$1","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],MN:[,[,,"[12]\\d{7,9}|[57-9]\\d{7}","\\d{6,10}"],[,,"[12](?:1\\d|2(?:[1-3]\\d?|7\\d)|3[2-8]\\d{1,2}|4[2-68]\\d{1,2}|5[1-4689]\\d{1,2})\\d{5}|5[0568]\\d{6}",
|
|
|
"\\d{6,10}",,,"50123456"],[,,"(?:8[689]|9[013-9])\\d{6}","\\d{8}",,,"88123456"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"7[05-8]\\d{6}","\\d{8}",,,"75123456"],"MN",976,"001","0",,,"0",,,,[[,"([12]\\d)(\\d{2})(\\d{4})","$1 $2 $3",["[12]1"],"0$1","",0],[,"([12]2\\d)(\\d{5,6})","$1 $2",["[12]2[1-3]"],"0$1","",0],[,"([12]\\d{3})(\\d{5})","$1 $2",["[12](?:27|[3-5])","[12](?:27|[3-5]\\d)2"],"0$1","",0],[,"(\\d{4})(\\d{4})","$1 $2",["[57-9]"],"$1","",0],[,"([12]\\d{4})(\\d{4,5})","$1 $2",
|
|
|
["[12](?:27|[3-5])","[12](?:27|[3-5]\\d)[4-9]"],"0$1","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],MO:[,[,,"[268]\\d{7}","\\d{8}"],[,,"(?:28[2-57-9]|8[2-57-9]\\d)\\d{5}","\\d{8}",,,"28212345"],[,,"6[236]\\d{6}","\\d{8}",,,"66123456"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"MO",853,"00",,,,,,,,[[,"([268]\\d{3})(\\d{4})","$1 $2",,"","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],MP:[,[,,"[5689]\\d{9}","\\d{7}(?:\\d{3})?"],
|
|
|
[,,"670(?:2(?:3[3-7]|56|8[5-8])|32[1238]|4(?:33|8[348])|5(?:32|55|88)|6(?:64|70|82)|78[589]|8[3-9]8|989)\\d{4}","\\d{7}(?:\\d{3})?",,,"6702345678"],[,,"670(?:2(?:3[3-7]|56|8[5-8])|32[1238]|4(?:33|8[348])|5(?:32|55|88)|6(?:64|70|82)|78[589]|8[3-9]8|989)\\d{4}","\\d{7}(?:\\d{3})?",,,"6702345678"],[,,"8(?:00|44|55|66|77|88)[2-9]\\d{6}","\\d{10}",,,"8002123456"],[,,"900[2-9]\\d{6}","\\d{10}",,,"9002123456"],[,,"NA","NA"],[,,"5(?:00|33|44|66|77)[2-9]\\d{6}","\\d{10}",,,"5002345678"],[,,"NA","NA"],"MP",
|
|
|
1,"011","1",,,"1",,,1,,,[,,"NA","NA"],,"670",[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],MQ:[,[,,"[56]\\d{8}","\\d{9}"],[,,"596(?:0[2-5]|[12]0|3[05-9]|4[024-8]|[5-7]\\d|89|9[4-8])\\d{4}","\\d{9}",,,"596301234"],[,,"696(?:[0-479]\\d|5[01]|8[0-689])\\d{4}","\\d{9}",,,"696201234"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"MQ",596,"00","0",,,"0",,,,[[,"(\\d{3})(\\d{2})(\\d{2})(\\d{2})","$1 $2 $3 $4",,"0$1","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],
|
|
|
MR:[,[,,"[2-48]\\d{7}","\\d{8}"],[,,"25[08]\\d{5}|35\\d{6}|45[1-7]\\d{5}","\\d{8}",,,"35123456"],[,,"(?:2(?:2\\d|70)|3(?:3\\d|6[1-36]|7[1-3])|4(?:[49]\\d|6[0457-9]|7[4-9]|8[01346-8]))\\d{5}","\\d{8}",,,"22123456"],[,,"800\\d{5}","\\d{8}",,,"80012345"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"MR",222,"00",,,,,,,,[[,"([2-48]\\d)(\\d{2})(\\d{2})(\\d{2})","$1 $2 $3 $4",,"","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],MS:[,[,,"[5689]\\d{9}","\\d{7}(?:\\d{3})?"],
|
|
|
[,,"664491\\d{4}","\\d{7}(?:\\d{3})?",,,"6644912345"],[,,"66449[2-6]\\d{4}","\\d{10}",,,"6644923456"],[,,"8(?:00|44|55|66|77|88)[2-9]\\d{6}","\\d{10}",,,"8002123456"],[,,"900[2-9]\\d{6}","\\d{10}",,,"9002123456"],[,,"NA","NA"],[,,"5(?:00|33|44|66|77)[2-9]\\d{6}","\\d{10}",,,"5002345678"],[,,"NA","NA"],"MS",1,"011","1",,,"1",,,,,,[,,"NA","NA"],,"664",[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],MT:[,[,,"[2357-9]\\d{7}","\\d{8}"],[,,"2(?:0(?:1[0-6]|3[1-4]|[69]\\d)|[1-357]\\d{2})\\d{4}","\\d{8}",,,"21001234"],
|
|
|
[,,"(?:7(?:210|[79]\\d{2})|9(?:2(?:1[01]|31)|696|8(?:1[1-3]|89|97)|9\\d{2}))\\d{4}","\\d{8}",,,"96961234"],[,,"800[3467]\\d{4}","\\d{8}",,,"80071234"],[,,"5(?:0(?:0(?:37|43)|6\\d{2}|70\\d|9[0168])|[12]\\d0[1-5])\\d{3}","\\d{8}",,,"50037123"],[,,"NA","NA"],[,,"NA","NA"],[,,"3550\\d{4}","\\d{8}",,,"35501234"],"MT",356,"00",,,,,,,,[[,"(\\d{4})(\\d{4})","$1 $2",,"","",0]],,[,,"7117\\d{4}","\\d{8}",,,"71171234"],,,[,,"NA","NA"],[,,"501\\d{5}","\\d{8}",,,"50112345"],,,[,,"NA","NA"]],MU:[,[,,"[2-9]\\d{6,7}",
|
|
|
"\\d{7,8}"],[,,"(?:2(?:[03478]\\d|1[0-7]|6[1-69])|4(?:[013568]\\d|2[4-7])|5(?:44\\d|471)|6\\d{2}|8(?:14|3[129]))\\d{4}","\\d{7,8}",,,"2012345"],[,,"5(?:2[59]\\d|4(?:2[1-389]|4\\d|7[1-9]|9\\d)|7\\d{2}|8(?:[256]\\d|7[15-8])|9[0-8]\\d)\\d{4}","\\d{8}",,,"52512345"],[,,"80[012]\\d{4}","\\d{7}",,,"8001234"],[,,"30\\d{5}","\\d{7}",,,"3012345"],[,,"NA","NA"],[,,"NA","NA"],[,,"3(?:20|9\\d)\\d{4}","\\d{7}",,,"3201234"],"MU",230,"0(?:0|[2-7]0|33)",,,,,,"020",,[[,"([2-46-9]\\d{2})(\\d{4})","$1 $2",["[2-46-9]"],
|
|
|
"","",0],[,"(5\\d{3})(\\d{4})","$1 $2",["5"],"","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],MV:[,[,,"[3467]\\d{6}|9(?:00\\d{7}|\\d{6})","\\d{7,10}"],[,,"(?:3(?:0[01]|3[0-59])|6(?:[567][02468]|8[024689]|90))\\d{4}","\\d{7}",,,"6701234"],[,,"(?:46[46]|7[3-9]\\d|9[16-9]\\d)\\d{4}","\\d{7}",,,"7712345"],[,,"NA","NA"],[,,"900\\d{7}","\\d{10}",,,"9001234567"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"MV",960,"0(?:0|19)",,,,,,"00",,[[,"(\\d{3})(\\d{4})","$1-$2",["[3467]|9(?:[1-9]|0[1-9])"],
|
|
|
"","",0],[,"(\\d{3})(\\d{3})(\\d{4})","$1 $2 $3",["900"],"","",0]],,[,,"781\\d{4}","\\d{7}",,,"7812345"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],MW:[,[,,"(?:1(?:\\d{2})?|[2789]\\d{2})\\d{6}","\\d{7,9}"],[,,"(?:1[2-9]|21\\d{2})\\d{5}","\\d{7,9}",,,"1234567"],[,,"(?:111|77\\d|88\\d|99\\d)\\d{6}","\\d{9}",,,"991234567"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"MW",265,"00","0",,,"0",,,,[[,"(\\d)(\\d{3})(\\d{3})","$1 $2 $3",["1"],"0$1","",0],[,"(2\\d{2})(\\d{3})(\\d{3})",
|
|
|
"$1 $2 $3",["2"],"0$1","",0],[,"(\\d{3})(\\d{2})(\\d{2})(\\d{2})","$1 $2 $3 $4",["[1789]"],"0$1","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],MX:[,[,,"[1-9]\\d{9,10}","\\d{7,11}"],[,,"(?:33|55|81)\\d{8}|(?:2(?:2[2-9]|3[1-35-8]|4[13-9]|7[1-689]|8[1-578]|9[467])|3(?:1[1-79]|[2458][1-9]|7[1-8]|9[1-5])|4(?:1[1-57-9]|[24-6][1-9]|[37][1-8]|8[1-35-9]|9[2-689])|5(?:88|9[1-79])|6(?:1[2-68]|[234][1-9]|5[1-3689]|6[12457-9]|7[1-7]|8[67]|9[4-8])|7(?:[13467][1-9]|2[1-8]|5[13-9]|8[1-69]|9[17])|8(?:2[13-689]|3[1-6]|4[124-6]|6[1246-9]|7[1-378]|9[12479])|9(?:1[346-9]|2[1-4]|3[2-46-8]|5[1348]|[69][1-9]|7[12]|8[1-8]))\\d{7}",
|
|
|
"\\d{7,10}",,,"2221234567"],[,,"1(?:(?:33|55|81)\\d{8}|(?:2(?:2[2-9]|3[1-35-8]|4[13-9]|7[1-689]|8[1-578]|9[467])|3(?:1[1-79]|[2458][1-9]|7[1-8]|9[1-5])|4(?:1[1-57-9]|[24-6][1-9]|[37][1-8]|8[1-35-9]|9[2-689])|5(?:88|9[1-79])|6(?:1[2-68]|[2-4][1-9]|5[1-3689]|6[12457-9]|7[1-7]|8[67]|9[4-8])|7(?:[13467][1-9]|2[1-8]|5[13-9]|8[1-69]|9[17])|8(?:2[13-689]|3[1-6]|4[124-6]|6[1246-9]|7[1-378]|9[12479])|9(?:1[346-9]|2[1-4]|3[2-46-8]|5[1348]|[69][1-9]|7[12]|8[1-8]))\\d{7})","\\d{11}",,,"12221234567"],[,,"800\\d{7}",
|
|
|
"\\d{10}",,,"8001234567"],[,,"900\\d{7}","\\d{10}",,,"9001234567"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"MX",52,"0[09]","01",,,"0[12]|04[45](\\d{10})","1$1",,,[[,"([358]\\d)(\\d{4})(\\d{4})","$1 $2 $3",["33|55|81"],"01 $1","",1],[,"(\\d{3})(\\d{3})(\\d{4})","$1 $2 $3",["[2467]|3[12457-9]|5[89]|8[02-9]|9[0-35-9]"],"01 $1","",1],[,"(1)([358]\\d)(\\d{4})(\\d{4})","044 $2 $3 $4",["1(?:33|55|81)"],"$1","",1],[,"(1)(\\d{3})(\\d{3})(\\d{4})","044 $2 $3 $4",["1(?:[2467]|3[12457-9]|5[89]|8[2-9]|9[1-35-9])"],
|
|
|
"$1","",1]],[[,"([358]\\d)(\\d{4})(\\d{4})","$1 $2 $3",["33|55|81"],"01 $1","",1],[,"(\\d{3})(\\d{3})(\\d{4})","$1 $2 $3",["[2467]|3[12457-9]|5[89]|8[02-9]|9[0-35-9]"],"01 $1","",1],[,"(1)([358]\\d)(\\d{4})(\\d{4})","$1 $2 $3 $4",["1(?:33|55|81)"]],[,"(1)(\\d{3})(\\d{3})(\\d{4})","$1 $2 $3 $4",["1(?:[2467]|3[12457-9]|5[89]|8[2-9]|9[1-35-9])"]]],[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],1,,[,,"NA","NA"]],MY:[,[,,"[13-9]\\d{7,9}","\\d{6,10}"],[,,"(?:3[2-9]\\d|[4-9][2-9])\\d{6}","\\d{6,9}",,,"323456789"],
|
|
|
[,,"1(?:1[1-3]\\d{2}|[02-4679][2-9]\\d|59\\d{2}|8(?:1[23]|[2-9]\\d))\\d{5}","\\d{9,10}",,,"123456789"],[,,"1[378]00\\d{6}","\\d{10}",,,"1300123456"],[,,"1600\\d{6}","\\d{10}",,,"1600123456"],[,,"NA","NA"],[,,"NA","NA"],[,,"154\\d{7}","\\d{10}",,,"1541234567"],"MY",60,"00","0",,,"0",,,,[[,"([4-79])(\\d{3})(\\d{4})","$1-$2 $3",["[4-79]"],"0$1","",0],[,"(3)(\\d{4})(\\d{4})","$1-$2 $3",["3"],"0$1","",0],[,"([18]\\d)(\\d{3})(\\d{3,4})","$1-$2 $3",["1[02-46-9][1-9]|8"],"0$1","",0],[,"(1)([36-8]00)(\\d{2})(\\d{4})",
|
|
|
"$1-$2-$3-$4",["1[36-8]0"],"","",0],[,"(11)(\\d{4})(\\d{4})","$1-$2 $3",["11"],"0$1","",0],[,"(15[49])(\\d{3})(\\d{4})","$1-$2 $3",["15"],"0$1","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],MZ:[,[,,"[28]\\d{7,8}","\\d{8,9}"],[,,"2(?:[1346]\\d|5[0-2]|[78][12]|93)\\d{5}","\\d{8}",,,"21123456"],[,,"8[23467]\\d{7}","\\d{9}",,,"821234567"],[,,"800\\d{6}","\\d{9}",,,"800123456"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"MZ",258,"00",,,,,,,,[[,"([28]\\d)(\\d{3})(\\d{3,4})",
|
|
|
"$1 $2 $3",["2|8[2-7]"],"","",0],[,"(80\\d)(\\d{3})(\\d{3})","$1 $2 $3",["80"],"","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],NA:[,[,,"[68]\\d{7,8}","\\d{8,9}"],[,,"6(?:1(?:17|2(?:[0189]\\d|[2-6]|7\\d?)|3(?:[01378]|2\\d)|4[01]|69|7[014])|2(?:17|5(?:[0-36-8]|4\\d?)|69|70)|3(?:17|2(?:[0237]\\d?|[14-689])|34|6[29]|7[01]|81)|4(?:17|2(?:[012]|7?)|4(?:[06]|1\\d)|5(?:[01357]|[25]\\d?)|69|7[01])|5(?:17|2(?:[0459]|[23678]\\d?)|69|7[01])|6(?:17|2(?:5|6\\d?)|38|42|69|7[01])|7(?:17|2(?:[569]|[234]\\d?)|3(?:0\\d?|[13])|69|7[01]))\\d{4}",
|
|
|
"\\d{8,9}",,,"61221234"],[,,"(?:60|8[125])\\d{7}","\\d{9}",,,"811234567"],[,,"NA","NA"],[,,"8701\\d{5}","\\d{9}",,,"870123456"],[,,"NA","NA"],[,,"NA","NA"],[,,"8(?:3\\d{2}|86)\\d{5}","\\d{8,9}",,,"88612345"],"NA",264,"00","0",,,"0",,,,[[,"(8\\d)(\\d{3})(\\d{4})","$1 $2 $3",["8[1235]"],"0$1","",0],[,"(6\\d)(\\d{2,3})(\\d{4})","$1 $2 $3",["6"],"0$1","",0],[,"(88)(\\d{3})(\\d{3})","$1 $2 $3",["88"],"0$1","",0],[,"(870)(\\d{3})(\\d{3})","$1 $2 $3",["870"],"0$1","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,
|
|
|
,"NA","NA"],,,[,,"NA","NA"]],NC:[,[,,"[2-57-9]\\d{5}","\\d{6}"],[,,"(?:2[03-9]|3[0-5]|4[1-7]|88)\\d{4}","\\d{6}",,,"201234"],[,,"(?:5[0-4]|[79]\\d|8[0-79])\\d{4}","\\d{6}",,,"751234"],[,,"NA","NA"],[,,"36\\d{4}","\\d{6}",,,"366711"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"NC",687,"00",,,,,,,,[[,"(\\d{2})(\\d{2})(\\d{2})","$1.$2.$3",["[2-46-9]|5[0-4]"],"","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],NE:[,[,,"[0289]\\d{7}","\\d{8}"],[,,"2(?:0(?:20|3[1-7]|4[134]|5[14]|6[14578]|7[1-578])|1(?:4[145]|5[14]|6[14-68]|7[169]|88))\\d{4}",
|
|
|
"\\d{8}",,,"20201234"],[,,"(?:89|9\\d)\\d{6}","\\d{8}",,,"93123456"],[,,"08\\d{6}","\\d{8}",,,"08123456"],[,,"09\\d{6}","\\d{8}",,,"09123456"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"NE",227,"00",,,,,,,,[[,"(\\d{2})(\\d{2})(\\d{2})(\\d{2})","$1 $2 $3 $4",["[289]|09"],"","",0],[,"(08)(\\d{3})(\\d{3})","$1 $2 $3",["08"],"","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],1,,[,,"NA","NA"]],NF:[,[,,"[13]\\d{5}","\\d{5,6}"],[,,"(?:1(?:06|17|28|39)|3[012]\\d)\\d{3}","\\d{5,6}",,,"106609"],[,,"38\\d{4}",
|
|
|
"\\d{5,6}",,,"381234"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"NF",672,"00",,,,,,,,[[,"(\\d{2})(\\d{4})","$1 $2",["1"],"","",0],[,"(\\d)(\\d{5})","$1 $2",["3"],"","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],NG:[,[,,"[1-6]\\d{5,8}|9\\d{5,9}|[78]\\d{5,13}","\\d{5,14}"],[,,"[12]\\d{6,7}|9(?:0[3-9]|[1-9]\\d)\\d{5}|(?:3\\d|4[023568]|5[02368]|6[02-469]|7[4-69]|8[2-9])\\d{6}|(?:4[47]|5[14579]|6[1578]|7[0-357])\\d{5,6}|(?:78|41)\\d{5}","\\d{5,9}",,,
|
|
|
"12345678"],[,,"(?:1(?:7[34]\\d|8(?:04|[124579]\\d|8[0-3])|95\\d)|287[0-7]|3(?:18[1-8]|88[0-7]|9(?:8[5-9]|6[1-5]))|4(?:28[0-2]|6(?:7[1-9]|8[02-47])|88[0-2])|5(?:2(?:7[7-9]|8\\d)|38[1-79]|48[0-7]|68[4-7])|6(?:2(?:7[7-9]|8\\d)|4(?:3[7-9]|[68][129]|7[04-69]|9[1-8])|58[0-2]|98[7-9])|7(?:38[0-7]|69[1-8]|78[2-4])|8(?:28[3-9]|38[0-2]|4(?:2[12]|3[147-9]|5[346]|7[4-9]|8[014-689]|90)|58[1-8]|78[2-9]|88[5-7])|98[07]\\d)\\d{4}|(?:70(?:[13-9]\\d|2[1-9])|8(?:0[2-9]|1\\d)\\d|90[239]\\d)\\d{6}","\\d{8,10}",,,"8021234567"],
|
|
|
[,,"800\\d{7,11}","\\d{10,14}",,,"80017591759"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"NG",234,"009","0",,,"0",,,,[[,"([129])(\\d{3})(\\d{3,4})","$1 $2 $3",["[129]"],"0$1","",0],[,"(\\d{2})(\\d{3})(\\d{2,3})","$1 $2 $3",["[3-6]|7(?:[1-79]|0[1-9])|8[2-9]"],"0$1","",0],[,"(\\d{3})(\\d{3})(\\d{3,4})","$1 $2 $3",["70|8[01]|90[239]"],"0$1","",0],[,"([78]00)(\\d{4})(\\d{4,5})","$1 $2 $3",["[78]00"],"0$1","",0],[,"([78]00)(\\d{5})(\\d{5,6})","$1 $2 $3",["[78]00"],"0$1","",0],[,"(78)(\\d{2})(\\d{3})",
|
|
|
"$1 $2 $3",["78"],"0$1","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"700\\d{7,11}","\\d{10,14}",,,"7001234567"],,,[,,"NA","NA"]],NI:[,[,,"[12578]\\d{7}","\\d{8}"],[,,"2\\d{7}","\\d{8}",,,"21234567"],[,,"5(?:5[0-7]\\d{5}|[78]\\d{6})|7[5-8]\\d{6}|8\\d{7}","\\d{8}",,,"81234567"],[,,"1800\\d{4}","\\d{8}",,,"18001234"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"NI",505,"00",,,,,,,,[[,"(\\d{4})(\\d{4})","$1 $2",,"","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],NL:[,[,,
|
|
|
"1\\d{4,8}|[2-7]\\d{8}|[89]\\d{6,9}","\\d{5,10}"],[,,"(?:1[0135-8]|2[02-69]|3[0-68]|4[0135-9]|[57]\\d|8[478])\\d{7}","\\d{9}",,,"101234567"],[,,"6[1-58]\\d{7}","\\d{9}",,,"612345678"],[,,"800\\d{4,7}","\\d{7,10}",,,"8001234"],[,,"90[069]\\d{4,7}","\\d{7,10}",,,"9061234"],[,,"NA","NA"],[,,"NA","NA"],[,,"85\\d{7}","\\d{9}",,,"851234567"],"NL",31,"00","0",,,"0",,,,[[,"([1-578]\\d)(\\d{3})(\\d{4})","$1 $2 $3",["1[035]|2[0346]|3[03568]|4[0356]|5[0358]|7|8[4578]"],"0$1","",0],[,"([1-5]\\d{2})(\\d{3})(\\d{3})",
|
|
|
"$1 $2 $3",["1[16-8]|2[259]|3[124]|4[17-9]|5[124679]"],"0$1","",0],[,"(6)(\\d{8})","$1 $2",["6[0-57-9]"],"0$1","",0],[,"(66)(\\d{7})","$1 $2",["66"],"0$1","",0],[,"(14)(\\d{3,4})","$1 $2",["14"],"$1","",0],[,"([89]0\\d)(\\d{4,7})","$1 $2",["80|9"],"0$1","",0]],,[,,"66\\d{7}","\\d{9}",,,"662345678"],,,[,,"14\\d{3,4}","\\d{5,6}"],[,,"140(?:1(?:[035]|[16-8]\\d)|2(?:[0346]|[259]\\d)|3(?:[03568]|[124]\\d)|4(?:[0356]|[17-9]\\d)|5(?:[0358]|[124679]\\d)|7\\d|8[458])","\\d{5,6}",,,"14020"],,,[,,"NA","NA"]],
|
|
|
NO:[,[,,"0\\d{4}|[2-9]\\d{7}","\\d{5}(?:\\d{3})?"],[,,"(?:2[1-4]|3[1-3578]|5[1-35-7]|6[1-4679]|7[0-8])\\d{6}","\\d{8}",,,"21234567"],[,,"(?:4[015-8]|5[89]|9\\d)\\d{6}","\\d{8}",,,"40612345"],[,,"80[01]\\d{5}","\\d{8}",,,"80012345"],[,,"82[09]\\d{5}","\\d{8}",,,"82012345"],[,,"810(?:0[0-6]|[2-8]\\d)\\d{3}","\\d{8}",,,"81021234"],[,,"880\\d{5}","\\d{8}",,,"88012345"],[,,"85[0-5]\\d{5}","\\d{8}",,,"85012345"],"NO",47,"00",,,,,,,,[[,"([489]\\d{2})(\\d{2})(\\d{3})","$1 $2 $3",["[489]"],"","",0],[,"([235-7]\\d)(\\d{2})(\\d{2})(\\d{2})",
|
|
|
"$1 $2 $3 $4",["[235-7]"],"","",0]],,[,,"NA","NA"],1,,[,,"NA","NA"],[,,"0\\d{4}|81(?:0(?:0[7-9]|1\\d)|5\\d{2})\\d{3}","\\d{5}(?:\\d{3})?",,,"01234"],1,,[,,"81[23]\\d{5}","\\d{8}",,,"81212345"]],NP:[,[,,"[1-8]\\d{7}|9(?:[1-69]\\d{6}|7[2-6]\\d{5,7}|8\\d{8})","\\d{6,10}"],[,,"(?:1[0124-6]|2[13-79]|3[135-8]|4[146-9]|5[135-7]|6[13-9]|7[15-9]|8[1-46-9]|9[1-79])\\d{6}","\\d{6,8}",,,"14567890"],[,,"9(?:7[45]|8[01456])\\d{7}","\\d{10}",,,"9841234567"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],
|
|
|
[,,"NA","NA"],"NP",977,"00","0",,,"0",,,,[[,"(1)(\\d{7})","$1-$2",["1[2-6]"],"0$1","",0],[,"(\\d{2})(\\d{6})","$1-$2",["1[01]|[2-8]|9(?:[1-69]|7[15-9])"],"0$1","",0],[,"(9\\d{2})(\\d{7})","$1-$2",["9(?:7[45]|8)"],"0$1","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],NR:[,[,,"[458]\\d{6}","\\d{7}"],[,,"(?:444|888)\\d{4}","\\d{7}",,,"4441234"],[,,"55[5-9]\\d{4}","\\d{7}",,,"5551234"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"NR",674,"00",,,,,,,,[[,"(\\d{3})(\\d{4})",
|
|
|
"$1 $2",,"","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],NU:[,[,,"[1-5]\\d{3}","\\d{4}"],[,,"[34]\\d{3}","\\d{4}",,,"4002"],[,,"[125]\\d{3}","\\d{4}",,,"1234"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"NU",683,"00",,,,,,,,,,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],NZ:[,[,,"6[235-9]\\d{6}|[2-57-9]\\d{7,10}","\\d{7,11}"],[,,"(?:3[2-79]|[49][2-689]|6[235-9]|7[2-5789])\\d{6}|24099\\d{3}","\\d{7,8}",,,"32345678"],[,,"2(?:[028]\\d{7,8}|1(?:[03]\\d{5,7}|[12457]\\d{5,6}|[689]\\d{5})|[79]\\d{7})",
|
|
|
"\\d{8,10}",,,"211234567"],[,,"508\\d{6,7}|80\\d{6,8}","\\d{8,10}",,,"800123456"],[,,"90\\d{7,9}","\\d{9,11}",,,"900123456"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"NZ",64,"0(?:0|161)","0",,,"0",,"00",,[[,"([34679])(\\d{3})(\\d{4})","$1-$2 $3",["[3467]|9[1-9]"],"0$1","",0],[,"(24099)(\\d{3})","$1 $2",["240","2409","24099"],"0$1","",0],[,"(\\d{2})(\\d{3})(\\d{3})","$1 $2 $3",["21"],"0$1","",0],[,"(\\d{2})(\\d{3})(\\d{3,5})","$1 $2 $3",["2(?:1[1-9]|[69]|7[0-35-9])|86"],"0$1","",0],[,"(2\\d)(\\d{3,4})(\\d{4})",
|
|
|
"$1 $2 $3",["2[028]"],"0$1","",0],[,"(\\d{3})(\\d{3})(\\d{3,4})","$1 $2 $3",["2(?:10|74)|5|[89]0"],"0$1","",0]],,[,,"[28]6\\d{6,7}","\\d{8,9}",,,"26123456"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],OM:[,[,,"(?:2[2-6]|5|9[1-9])\\d{6}|800\\d{5,6}","\\d{7,9}"],[,,"2[2-6]\\d{6}","\\d{8}",,,"23123456"],[,,"9[1-9]\\d{6}","\\d{8}",,,"92123456"],[,,"8007\\d{4,5}|500\\d{4}","\\d{7,9}",,,"80071234"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"OM",968,"00",,,,,,,,[[,"(2\\d)(\\d{6})","$1 $2",
|
|
|
["2"],"","",0],[,"(9\\d{3})(\\d{4})","$1 $2",["9"],"","",0],[,"([58]00)(\\d{4,6})","$1 $2",["[58]"],"","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],PA:[,[,,"[1-9]\\d{6,7}","\\d{7,8}"],[,,"(?:1(?:0[02-579]|19|2[37]|3[03]|4[479]|57|65|7[016-8]|8[58]|9[1349])|2(?:[0235679]\\d|1[0-7]|4[04-9]|8[028])|3(?:[09]\\d|1[14-7]|2[0-3]|3[03]|4[0457]|5[56]|6[068]|7[06-8]|8[089])|4(?:3[013-69]|4\\d|7[0-689])|5(?:[01]\\d|2[0-7]|[56]0|79)|7(?:0[09]|2[0-267]|3[06]|[49]0|5[06-9]|7[0-24-7]|8[89])|8(?:[34]\\d|5[0-4]|8[02])|9(?:0[6-8]|1[016-8]|2[036-8]|3[3679]|40|5[0489]|6[06-9]|7[046-9]|8[36-8]|9[1-9]))\\d{4}",
|
|
|
"\\d{7}",,,"2001234"],[,,"(?:1[16]1|21[89]|8(?:1[01]|7[23]))\\d{4}|6(?:[024-9]\\d|1[0-5]|3[0-24-9])\\d{5}","\\d{7,8}",,,"60012345"],[,,"80[09]\\d{4}","\\d{7}",,,"8001234"],[,,"(?:779|8(?:2[235]|55|60|7[578]|86|95)|9(?:0[0-2]|81))\\d{4}","\\d{7}",,,"8601234"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"PA",507,"00",,,,,,,,[[,"(\\d{3})(\\d{4})","$1-$2",["[1-57-9]"],"","",0],[,"(\\d{4})(\\d{4})","$1-$2",["6"],"","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],PE:[,[,,"[14-9]\\d{7,8}",
|
|
|
"\\d{6,9}"],[,,"(?:1\\d|4[1-4]|5[1-46]|6[1-7]|7[2-46]|8[2-4])\\d{6}","\\d{6,8}",,,"11234567"],[,,"9\\d{8}","\\d{9}",,,"912345678"],[,,"800\\d{5}","\\d{8}",,,"80012345"],[,,"805\\d{5}","\\d{8}",,,"80512345"],[,,"801\\d{5}","\\d{8}",,,"80112345"],[,,"80[24]\\d{5}","\\d{8}",,,"80212345"],[,,"NA","NA"],"PE",51,"19(?:1[124]|77|90)00","0"," Anexo ",,"0",,,,[[,"(1)(\\d{7})","$1 $2",["1"],"(0$1)","",0],[,"([4-8]\\d)(\\d{6})","$1 $2",["[4-7]|8[2-4]"],"(0$1)","",0],[,"(\\d{3})(\\d{5})","$1 $2",["80"],"(0$1)",
|
|
|
"",0],[,"(9\\d{2})(\\d{3})(\\d{3})","$1 $2 $3",["9"],"$1","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],PF:[,[,,"4\\d{5,7}|8\\d{7}","\\d{6}(?:\\d{2})?"],[,,"4(?:[09][45689]\\d|4)\\d{4}","\\d{6}(?:\\d{2})?",,,"40412345"],[,,"8[79]\\d{6}","\\d{8}",,,"87123456"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"PF",689,"00",,,,,,,,[[,"(\\d{2})(\\d{2})(\\d{2})(\\d{2})","$1 $2 $3 $4",["4[09]|8[79]"],"","",0],[,"(\\d{2})(\\d{2})(\\d{2})","$1 $2 $3",["44"],"",
|
|
|
"",0]],,[,,"NA","NA"],,,[,,"44\\d{4}","\\d{6}",,,"441234"],[,,"NA","NA"],,,[,,"NA","NA"]],PG:[,[,,"[1-9]\\d{6,7}","\\d{7,8}"],[,,"(?:3[0-2]\\d|4[25]\\d|5[34]\\d|64[1-9]|77(?:[0-24]\\d|30)|85[02-46-9]|9[78]\\d)\\d{4}","\\d{7}",,,"3123456"],[,,"(?:20150|68\\d{2}|7(?:[0-369]\\d|75)\\d{2})\\d{3}","\\d{7,8}",,,"6812345"],[,,"180\\d{4}","\\d{7}",,,"1801234"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"275\\d{4}","\\d{7}",,,"2751234"],"PG",675,"00",,,,,,,,[[,"(\\d{3})(\\d{4})","$1 $2",["[13-689]|27"],
|
|
|
"","",0],[,"(\\d{4})(\\d{4})","$1 $2",["20|7"],"","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],PH:[,[,,"2\\d{5,7}|[3-9]\\d{7,9}|1800\\d{7,9}","\\d{5,13}"],[,,"2\\d{5}(?:\\d{2})?|(?:3[2-68]|4[2-9]|5[2-6]|6[2-58]|7[24578]|8[2-8])\\d{7}|88(?:22\\d{6}|42\\d{4})","\\d{5,10}",,,"21234567"],[,,"(?:81[37]|9(?:0[5-9]|1[024-9]|2[0-35-9]|3[02-9]|4[236-9]|7[34-79]|89|9[4-9]))\\d{7}","\\d{10}",,,"9051234567"],[,,"1800\\d{7,9}","\\d{11,13}",,,"180012345678"],[,,"NA","NA"],[,,"NA","NA"],
|
|
|
[,,"NA","NA"],[,,"NA","NA"],"PH",63,"00","0",,,"0",,,,[[,"(2)(\\d{3})(\\d{4})","$1 $2 $3",["2"],"(0$1)","",0],[,"(2)(\\d{5})","$1 $2",["2"],"(0$1)","",0],[,"(\\d{4})(\\d{4,6})","$1 $2",["3(?:23|39|46)|4(?:2[3-6]|[35]9|4[26]|76)|5(?:22|44)|642|8(?:62|8[245])","3(?:230|397|461)|4(?:2(?:35|[46]4|51)|396|4(?:22|63)|59[347]|76[15])|5(?:221|446)|642[23]|8(?:622|8(?:[24]2|5[13]))"],"(0$1)","",0],[,"(\\d{5})(\\d{4})","$1 $2",["346|4(?:27|9[35])|883","3469|4(?:279|9(?:30|56))|8834"],"(0$1)","",0],[,"([3-8]\\d)(\\d{3})(\\d{4})",
|
|
|
"$1 $2 $3",["[3-8]"],"(0$1)","",0],[,"(\\d{3})(\\d{3})(\\d{4})","$1 $2 $3",["81|9"],"0$1","",0],[,"(1800)(\\d{3})(\\d{4})","$1 $2 $3",["1"],"","",0],[,"(1800)(\\d{1,2})(\\d{3})(\\d{4})","$1 $2 $3 $4",["1"],"","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],PK:[,[,,"1\\d{8}|[2-8]\\d{5,11}|9(?:[013-9]\\d{4,9}|2\\d(?:111\\d{6}|\\d{3,7}))","\\d{6,12}"],[,,"(?:21|42)[2-9]\\d{7}|(?:2[25]|4[0146-9]|5[1-35-7]|6[1-8]|7[14]|8[16]|91)[2-9]\\d{6}|(?:2(?:3[2358]|4[2-4]|9[2-8])|45[3479]|54[2-467]|60[468]|72[236]|8(?:2[2-689]|3[23578]|4[3478]|5[2356])|9(?:1|2[2-8]|3[27-9]|4[2-6]|6[3569]|9[25-8]))[2-9]\\d{5,6}|58[126]\\d{7}",
|
|
|
"\\d{6,10}",,,"2123456789"],[,,"3(?:0\\d|[12][0-5]|3[1-7]|4[0-7]|55|64)\\d{7}","\\d{10}",,,"3012345678"],[,,"800\\d{5}","\\d{8}",,,"80012345"],[,,"900\\d{5}","\\d{8}",,,"90012345"],[,,"NA","NA"],[,,"122\\d{6}","\\d{9}",,,"122044444"],[,,"NA","NA"],"PK",92,"00","0",,,"0",,,,[[,"(\\d{2})(111)(\\d{3})(\\d{3})","$1 $2 $3 $4",["(?:2[125]|4[0-246-9]|5[1-35-7]|6[1-8]|7[14]|8[16]|91)1","(?:2[125]|4[0-246-9]|5[1-35-7]|6[1-8]|7[14]|8[16]|91)11","(?:2[125]|4[0-246-9]|5[1-35-7]|6[1-8]|7[14]|8[16]|91)111"],"(0$1)",
|
|
|
"",0],[,"(\\d{3})(111)(\\d{3})(\\d{3})","$1 $2 $3 $4",["2[349]|45|54|60|72|8[2-5]|9[2-9]","(?:2[349]|45|54|60|72|8[2-5]|9[2-9])\\d1","(?:2[349]|45|54|60|72|8[2-5]|9[2-9])\\d11","(?:2[349]|45|54|60|72|8[2-5]|9[2-9])\\d111"],"(0$1)","",0],[,"(\\d{2})(\\d{7,8})","$1 $2",["(?:2[125]|4[0-246-9]|5[1-35-7]|6[1-8]|7[14]|8[16]|91)[2-9]"],"(0$1)","",0],[,"(\\d{3})(\\d{6,7})","$1 $2",["2[349]|45|54|60|72|8[2-5]|9[2-9]","(?:2[349]|45|54|60|72|8[2-5]|9[2-9])\\d[2-9]"],"(0$1)","",0],[,"(3\\d{2})(\\d{7})","$1 $2",
|
|
|
["3"],"0$1","",0],[,"([15]\\d{3})(\\d{5,6})","$1 $2",["58[12]|1"],"(0$1)","",0],[,"(586\\d{2})(\\d{5})","$1 $2",["586"],"(0$1)","",0],[,"([89]00)(\\d{3})(\\d{2})","$1 $2 $3",["[89]00"],"0$1","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"(?:2(?:[125]|3[2358]|4[2-4]|9[2-8])|4(?:[0-246-9]|5[3479])|5(?:[1-35-7]|4[2-467])|6(?:[1-8]|0[468])|7(?:[14]|2[236])|8(?:[16]|2[2-689]|3[23578]|4[3478]|5[2356])|9(?:1|22|3[27-9]|4[2-6]|6[3569]|9[2-7]))111\\d{6}","\\d{11,12}",,,"21111825888"],,,[,,"NA","NA"]],PL:[,[,,"[12]\\d{6,8}|[3-57-9]\\d{8}|6\\d{5,8}",
|
|
|
"\\d{6,9}"],[,,"(?:1[2-8]|2[2-59]|3[2-4]|4[1-468]|5[24-689]|6[1-3578]|7[14-7]|8[1-79]|9[145])\\d{7}|[12]2\\d{5}","\\d{6,9}",,,"123456789"],[,,"(?:5[0137]|6[069]|7[2389]|88)\\d{7}","\\d{9}",,,"512345678"],[,,"800\\d{6}","\\d{9}",,,"800123456"],[,,"70\\d{7}","\\d{9}",,,"701234567"],[,,"801\\d{6}","\\d{9}",,,"801234567"],[,,"NA","NA"],[,,"39\\d{7}","\\d{9}",,,"391234567"],"PL",48,"00",,,,,,,,[[,"(\\d{2})(\\d{3})(\\d{2})(\\d{2})","$1 $2 $3 $4",["[124]|3[2-4]|5[24-689]|6[1-3578]|7[14-7]|8[1-79]|9[145]"],
|
|
|
"","",0],[,"(\\d{2})(\\d{1})(\\d{4})","$1 $2 $3",["[12]2"],"","",0],[,"(\\d{3})(\\d{3})(\\d{3})","$1 $2 $3",["39|5[0137]|6[0469]|7[02389]|8[08]"],"","",0],[,"(\\d{3})(\\d{2})(\\d{2,3})","$1 $2 $3",["64"],"","",0],[,"(\\d{3})(\\d{3})","$1 $2",["64"],"","",0]],,[,,"64\\d{4,7}","\\d{6,9}",,,"641234567"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],PM:[,[,,"[45]\\d{5}","\\d{6}"],[,,"41\\d{4}","\\d{6}",,,"411234"],[,,"55\\d{4}","\\d{6}",,,"551234"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],
|
|
|
[,,"NA","NA"],"PM",508,"00","0",,,"0",,,,[[,"([45]\\d)(\\d{2})(\\d{2})","$1 $2 $3",,"0$1","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],PR:[,[,,"[5789]\\d{9}","\\d{7}(?:\\d{3})?"],[,,"(?:787|939)[2-9]\\d{6}","\\d{7}(?:\\d{3})?",,,"7872345678"],[,,"(?:787|939)[2-9]\\d{6}","\\d{7}(?:\\d{3})?",,,"7872345678"],[,,"8(?:00|44|55|66|77|88)[2-9]\\d{6}","\\d{10}",,,"8002345678"],[,,"900[2-9]\\d{6}","\\d{10}",,,"9002345678"],[,,"NA","NA"],[,,"5(?:00|33|44|66|77)[2-9]\\d{6}","\\d{10}",
|
|
|
,,"5002345678"],[,,"NA","NA"],"PR",1,"011","1",,,"1",,,1,,,[,,"NA","NA"],,"787|939",[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],PS:[,[,,"[24589]\\d{7,8}|1(?:[78]\\d{8}|[49]\\d{2,3})","\\d{4,10}"],[,,"(?:22[234789]|42[45]|82[01458]|92[369])\\d{5}","\\d{7,8}",,,"22234567"],[,,"5[69]\\d{7}","\\d{9}",,,"599123456"],[,,"1800\\d{6}","\\d{10}",,,"1800123456"],[,,"1(?:4|9\\d)\\d{2}","\\d{4,5}",,,"19123"],[,,"1700\\d{6}","\\d{10}",,,"1700123456"],[,,"NA","NA"],[,,"NA","NA"],"PS",970,"00","0",,,"0",,,,[[,
|
|
|
"([2489])(2\\d{2})(\\d{4})","$1 $2 $3",["[2489]"],"0$1","",0],[,"(5[69]\\d)(\\d{3})(\\d{3})","$1 $2 $3",["5"],"0$1","",0],[,"(1[78]00)(\\d{3})(\\d{3})","$1 $2 $3",["1[78]"],"$1","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],PT:[,[,,"[2-46-9]\\d{8}","\\d{9}"],[,,"2(?:[12]\\d|[35][1-689]|4[1-59]|6[1-35689]|7[1-9]|8[1-69]|9[1256])\\d{6}","\\d{9}",,,"212345678"],[,,"9(?:[136]\\d{2}|2[0-79]\\d|480)\\d{5}","\\d{9}",,,"912345678"],[,,"80[02]\\d{6}","\\d{9}",,,"800123456"],[,,"76(?:0[1-57]|1[2-47]|2[237])\\d{5}",
|
|
|
"\\d{9}",,,"760123456"],[,,"80(?:8\\d|9[1579])\\d{5}","\\d{9}",,,"808123456"],[,,"884[128]\\d{5}","\\d{9}",,,"884123456"],[,,"30\\d{7}","\\d{9}",,,"301234567"],"PT",351,"00",,,,,,,,[[,"(2\\d)(\\d{3})(\\d{4})","$1 $2 $3",["2[12]"],"","",0],[,"([2-46-9]\\d{2})(\\d{3})(\\d{3})","$1 $2 $3",["2[3-9]|[346-9]"],"","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"70(?:7\\d|8[17])\\d{5}","\\d{9}",,,"707123456"],,,[,,"NA","NA"]],PW:[,[,,"[2-8]\\d{6}","\\d{7}"],[,,"2552255|(?:277|345|488|5(?:35|44|87)|6(?:22|54|79)|7(?:33|47)|8(?:24|55|76))\\d{4}",
|
|
|
"\\d{7}",,,"2771234"],[,,"(?:6[234689]0|77[45789])\\d{4}","\\d{7}",,,"6201234"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"PW",680,"01[12]",,,,,,,,[[,"(\\d{3})(\\d{4})","$1 $2",,"","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],PY:[,[,,"5[0-5]\\d{4,7}|[2-46-9]\\d{5,8}","\\d{5,9}"],[,,"(?:[26]1|3[289]|4[124678]|7[123]|8[1236])\\d{5,7}|(?:2(?:2[4568]|7[15]|9[1-5])|3(?:18|3[167]|4[2357]|51)|4(?:18|2[45]|3[12]|5[13]|64|71|9[1-47])|5(?:[1-4]\\d|5[0234])|6(?:3[1-3]|44|7[1-4678])|7(?:17|4[0-4]|6[1-578]|75|8[0-8])|858)\\d{5,6}",
|
|
|
"\\d{5,9}",,,"212345678"],[,,"9(?:6[12]|[78][1-6]|9[1-5])\\d{6}","\\d{9}",,,"961456789"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"8700[0-4]\\d{4}","\\d{9}",,,"870012345"],"PY",595,"00","0",,,"0",,,,[[,"(\\d{2})(\\d{5,7})","$1 $2",["(?:[26]1|3[289]|4[124678]|7[123]|8[1236])"],"($1)","",0],[,"(\\d{3})(\\d{3,6})","$1 $2",["[2-9]0"],"0$1","",0],[,"(\\d{3})(\\d{6})","$1 $2",["9[1-9]"],"0$1","",0],[,"(\\d{2})(\\d{3})(\\d{4})","$1 $2 $3",["8700"],"","",0],[,"(\\d{3})(\\d{4,6})","$1 $2",
|
|
|
["[2-8][1-9]"],"($1)","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"[2-9]0\\d{4,7}","\\d{6,9}",,,"201234567"],,,[,,"NA","NA"]],QA:[,[,,"[2-8]\\d{6,7}","\\d{7,8}"],[,,"4[04]\\d{6}","\\d{7,8}",,,"44123456"],[,,"[3567]\\d{7}","\\d{7,8}",,,"33123456"],[,,"800\\d{4}","\\d{7,8}",,,"8001234"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"QA",974,"00",,,,,,,,[[,"([28]\\d{2})(\\d{4})","$1 $2",["[28]"],"","",0],[,"([3-7]\\d{3})(\\d{4})","$1 $2",["[3-7]"],"","",0]],,[,,"2(?:[12]\\d|61)\\d{4}","\\d{7}",
|
|
|
,,"2123456"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],RE:[,[,,"[268]\\d{8}","\\d{9}"],[,,"262\\d{6}","\\d{9}",,,"262161234"],[,,"6(?:9[23]|47)\\d{6}","\\d{9}",,,"692123456"],[,,"80\\d{7}","\\d{9}",,,"801234567"],[,,"89[1-37-9]\\d{6}","\\d{9}",,,"891123456"],[,,"8(?:1[019]|2[0156]|84|90)\\d{6}","\\d{9}",,,"810123456"],[,,"NA","NA"],[,,"NA","NA"],"RE",262,"00","0",,,"0",,,,[[,"([268]\\d{2})(\\d{2})(\\d{2})(\\d{2})","$1 $2 $3 $4",,"0$1","",0]],,[,,"NA","NA"],1,"262|6[49]|8",[,,"NA","NA"],[,,"NA",
|
|
|
"NA"],,,[,,"NA","NA"]],RO:[,[,,"2\\d{5,8}|[37-9]\\d{8}","\\d{6,9}"],[,,"2(?:1(?:\\d{7}|9\\d{3})|[3-6](?:\\d{7}|\\d9\\d{2}))|3[13-6]\\d{7}","\\d{6,9}",,,"211234567"],[,,"7(?:000|[1-8]\\d{2}|99\\d)\\d{5}","\\d{9}",,,"712345678"],[,,"800\\d{6}","\\d{9}",,,"800123456"],[,,"90[036]\\d{6}","\\d{9}",,,"900123456"],[,,"801\\d{6}","\\d{9}",,,"801123456"],[,,"802\\d{6}","\\d{9}",,,"802123456"],[,,"NA","NA"],"RO",40,"00","0"," int ",,"0",,,,[[,"([237]\\d)(\\d{3})(\\d{4})","$1 $2 $3",["[23]1"],"0$1","",0],[,
|
|
|
"(21)(\\d{4})","$1 $2",["21"],"0$1","",0],[,"(\\d{3})(\\d{3})(\\d{3})","$1 $2 $3",["[23][3-7]|[7-9]"],"0$1","",0],[,"(2\\d{2})(\\d{3})","$1 $2",["2[3-6]"],"0$1","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"37\\d{7}","\\d{9}",,,"372123456"],,,[,,"NA","NA"]],RS:[,[,,"[126-9]\\d{4,11}|3(?:[0-79]\\d{3,10}|8[2-9]\\d{2,9})","\\d{5,12}"],[,,"(?:1(?:[02-9][2-9]|1[1-9])\\d|2(?:[0-24-7][2-9]\\d|[389](?:0[2-9]|[2-9]\\d))|3(?:[0-8][2-9]\\d|9(?:[2-9]\\d|0[2-9])))\\d{3,8}","\\d{5,12}",,,"10234567"],[,,"6(?:[0-689]|7\\d)\\d{6,7}",
|
|
|
"\\d{8,10}",,,"601234567"],[,,"800\\d{3,9}","\\d{6,12}",,,"80012345"],[,,"(?:90[0169]|78\\d)\\d{3,7}","\\d{6,12}",,,"90012345"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"RS",381,"00","0",,,"0",,,,[[,"([23]\\d{2})(\\d{4,9})","$1 $2",["(?:2[389]|39)0"],"0$1","",0],[,"([1-3]\\d)(\\d{5,10})","$1 $2",["1|2(?:[0-24-7]|[389][1-9])|3(?:[0-8]|9[1-9])"],"0$1","",0],[,"(6\\d)(\\d{6,8})","$1 $2",["6"],"0$1","",0],[,"([89]\\d{2})(\\d{3,9})","$1 $2",["[89]"],"0$1","",0],[,"(7[26])(\\d{4,9})","$1 $2",["7[26]"],
|
|
|
"0$1","",0],[,"(7[08]\\d)(\\d{4,9})","$1 $2",["7[08]"],"0$1","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"7[06]\\d{4,10}","\\d{6,12}",,,"700123456"],,,[,,"NA","NA"]],RU:[,[,,"[3489]\\d{9}","\\d{10}"],[,,"(?:3(?:0[12]|4[1-35-79]|5[1-3]|8[1-58]|9[0145])|4(?:01|1[1356]|2[13467]|7[1-5]|8[1-7]|9[1-689])|8(?:1[1-8]|2[01]|3[13-6]|4[0-8]|5[15]|6[1-35-7]|7[1-37-9]))\\d{7}","\\d{10}",,,"3011234567"],[,,"9\\d{9}","\\d{10}",,,"9123456789"],[,,"80[04]\\d{7}","\\d{10}",,,"8001234567"],[,,"80[39]\\d{7}","\\d{10}",
|
|
|
,,"8091234567"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"RU",7,"810","8",,,"8",,"8~10",,[[,"(\\d{3})(\\d{2})(\\d{2})","$1-$2-$3",["[1-79]"],"$1","",1],[,"([3489]\\d{2})(\\d{3})(\\d{2})(\\d{2})","$1 $2-$3-$4",["[34689]"],"8 ($1)","",1],[,"(7\\d{2})(\\d{3})(\\d{4})","$1 $2 $3",["7"],"8 ($1)","",1]],[[,"([3489]\\d{2})(\\d{3})(\\d{2})(\\d{2})","$1 $2-$3-$4",["[34689]"],"8 ($1)","",1],[,"(7\\d{2})(\\d{3})(\\d{4})","$1 $2 $3",["7"],"8 ($1)","",1]],[,,"NA","NA"],1,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA",
|
|
|
"NA"]],RW:[,[,,"[027-9]\\d{7,8}","\\d{8,9}"],[,,"2[258]\\d{7}|06\\d{6}","\\d{8,9}",,,"250123456"],[,,"7[238]\\d{7}","\\d{9}",,,"720123456"],[,,"800\\d{6}","\\d{9}",,,"800123456"],[,,"900\\d{6}","\\d{9}",,,"900123456"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"RW",250,"00","0",,,"0",,,,[[,"(2\\d{2})(\\d{3})(\\d{3})","$1 $2 $3",["2"],"$1","",0],[,"([7-9]\\d{2})(\\d{3})(\\d{3})","$1 $2 $3",["[7-9]"],"0$1","",0],[,"(0\\d)(\\d{2})(\\d{2})(\\d{2})","$1 $2 $3 $4",["0"],"","",0]],,[,,"NA","NA"],,,[,,"NA",
|
|
|
"NA"],[,,"NA","NA"],1,,[,,"NA","NA"]],SA:[,[,,"1\\d{7,8}|(?:[2-467]|92)\\d{7}|5\\d{8}|8\\d{9}","\\d{7,10}"],[,,"11\\d{7}|1?(?:2[24-8]|3[35-8]|4[3-68]|6[2-5]|7[235-7])\\d{6}","\\d{7,9}",,,"112345678"],[,,"(?:5(?:[013-689]\\d|7[0-26-8])|811\\d)\\d{6}","\\d{9,10}",,,"512345678"],[,,"800\\d{7}","\\d{10}",,,"8001234567"],[,,"NA","NA"],[,,"92[05]\\d{6}","\\d{9}",,,"920012345"],[,,"NA","NA"],[,,"NA","NA"],"SA",966,"00","0",,,"0",,,,[[,"([1-467])(\\d{3})(\\d{4})","$1 $2 $3",["[1-467]"],"0$1","",0],[,"(1\\d)(\\d{3})(\\d{4})",
|
|
|
"$1 $2 $3",["1[1-467]"],"0$1","",0],[,"(5\\d)(\\d{3})(\\d{4})","$1 $2 $3",["5"],"0$1","",0],[,"(92\\d{2})(\\d{5})","$1 $2",["92"],"$1","",0],[,"(800)(\\d{3})(\\d{4})","$1 $2 $3",["80"],"$1","",0],[,"(811)(\\d{3})(\\d{3,4})","$1 $2 $3",["81"],"0$1","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],SB:[,[,,"[1-9]\\d{4,6}","\\d{5,7}"],[,,"(?:1[4-79]|[23]\\d|4[01]|5[03]|6[0-37])\\d{3}","\\d{5}",,,"40123"],[,,"48\\d{3}|7(?:[0146-8]\\d|5[025-9]|9[0124])\\d{4}|8[4-8]\\d{5}|9(?:[46]\\d|5[0-46-9]|7[0-689]|8[0-79]|9[0-8])\\d{4}",
|
|
|
"\\d{5,7}",,,"7421234"],[,,"1[38]\\d{3}","\\d{5}",,,"18123"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"5[12]\\d{3}","\\d{5}",,,"51123"],"SB",677,"0[01]",,,,,,,,[[,"(\\d{2})(\\d{5})","$1 $2",["[7-9]"],"","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],SC:[,[,,"[24689]\\d{5,6}","\\d{6,7}"],[,,"4[2-46]\\d{5}","\\d{7}",,,"4217123"],[,,"2[5-8]\\d{5}","\\d{7}",,,"2510123"],[,,"8000\\d{2}","\\d{6}",,,"800000"],[,,"98\\d{4}","\\d{6}",,,"981234"],[,,"NA","NA"],[,,"NA","NA"],[,,"64\\d{5}",
|
|
|
"\\d{7}",,,"6412345"],"SC",248,"0[0-2]",,,,,,"00",,[[,"(\\d{3})(\\d{3})","$1 $2",["[89]"],"","",0],[,"(\\d)(\\d{3})(\\d{3})","$1 $2 $3",["[246]"],"","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],SD:[,[,,"[19]\\d{8}","\\d{9}"],[,,"1(?:[125]\\d|8[3567])\\d{6}","\\d{9}",,,"121231234"],[,,"9[012569]\\d{7}","\\d{9}",,,"911231234"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"SD",249,"00","0",,,"0",,,,[[,"(\\d{2})(\\d{3})(\\d{4})","$1 $2 $3",,"0$1","",0]],
|
|
|
,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],SE:[,[,,"[1-9]\\d{5,9}","\\d{5,10}"],[,,"1(?:0[1-8]\\d{6}|[136]\\d{5,7}|(?:2[0-35]|4[0-4]|5[0-25-9]|7[13-6]|[89]\\d)\\d{5,6})|2(?:[136]\\d{5,7}|(?:2[0-7]|4[0136-8]|5[0138]|7[018]|8[01]|9[0-57])\\d{5,6})|3(?:[356]\\d{5,7}|(?:0[0-4]|1\\d|2[0-25]|4[056]|7[0-2]|8[0-3]|9[023])\\d{5,6})|4(?:0[1-9]\\d{4,6}|[246]\\d{5,7}|(?:1[013-8]|3[0135]|5[14-79]|7[0-246-9]|8[0156]|9[0-689])\\d{5,6})|5(?:0[0-6]|[15][0-5]|2[0-68]|3[0-4]|4\\d|6[03-5]|7[013]|8[0-79]|9[01])\\d{5,6}|6(?:0[1-9]\\d{4,6}|3\\d{5,7}|(?:1[1-3]|2[0-4]|4[02-57]|5[0-37]|6[0-3]|7[0-2]|8[0247]|9[0-356])\\d{5,6})|8[1-9]\\d{5,7}|9(?:0[1-9]\\d{4,6}|(?:1[0-68]|2\\d|3[02-5]|4[0-3]|5[0-4]|[68][01]|7[0135-8])\\d{5,6})",
|
|
|
"\\d{5,9}",,,"8123456"],[,,"7[0236]\\d{7}","\\d{9}",,,"701234567"],[,,"20(?:0(?:0\\d{2}|[1-9](?:0\\d{1,4}|[1-9]\\d{4}))|1(?:0\\d{4}|[1-9]\\d{4,5})|[2-9]\\d{5})","\\d{6,9}",,,"20123456"],[,,"9(?:00|39|44)(?:1(?:[0-26]\\d{5}|[3-57-9]\\d{2})|2(?:[0-2]\\d{5}|[3-9]\\d{2})|3(?:[0139]\\d{5}|[24-8]\\d{2})|4(?:[045]\\d{5}|[1-36-9]\\d{2})|5(?:5\\d{5}|[0-46-9]\\d{2})|6(?:[679]\\d{5}|[0-58]\\d{2})|7(?:[078]\\d{5}|[1-69]\\d{2})|8(?:[578]\\d{5}|[0-469]\\d{2}))","\\d{7}(?:\\d{3})?",,,"9001234567"],[,,"77(?:0(?:0\\d{2}|[1-9](?:0\\d|[1-9]\\d{4}))|[1-6][1-9]\\d{5})",
|
|
|
"\\d{6}(?:\\d{3})?",,,"771234567"],[,,"75[1-8]\\d{6}","\\d{9}",,,"751234567"],[,,"NA","NA"],"SE",46,"00","0",,,"0",,,,[[,"(8)(\\d{2,3})(\\d{2,3})(\\d{2})","$1-$2 $3 $4",["8"],"0$1","",0],[,"([1-69]\\d)(\\d{2,3})(\\d{2})(\\d{2})","$1-$2 $3 $4",["1[013689]|2[0136]|3[1356]|4[0246]|54|6[03]|90"],"0$1","",0],[,"([1-69]\\d)(\\d{3})(\\d{2})","$1-$2 $3",["1[13689]|2[136]|3[1356]|4[0246]|54|6[03]|90"],"0$1","",0],[,"(\\d{3})(\\d{2})(\\d{2})(\\d{2})","$1-$2 $3 $4",["1[2457]|2[2457-9]|3[0247-9]|4[1357-9]|5[0-35-9]|6[124-9]|9(?:[125-8]|3[0-5]|4[0-3])"],
|
|
|
"0$1","",0],[,"(\\d{3})(\\d{2,3})(\\d{2})","$1-$2 $3",["1[2457]|2[2457-9]|3[0247-9]|4[1357-9]|5[0-35-9]|6[124-9]|9(?:[125-8]|3[0-5]|4[0-3])"],"0$1","",0],[,"(7\\d)(\\d{3})(\\d{2})(\\d{2})","$1-$2 $3 $4",["7"],"0$1","",0],[,"(77)(\\d{2})(\\d{2})","$1-$2$3",["7"],"0$1","",0],[,"(20)(\\d{2,3})(\\d{2})","$1-$2 $3",["20"],"0$1","",0],[,"(9[034]\\d)(\\d{2})(\\d{2})(\\d{3})","$1-$2 $3 $4",["9[034]"],"0$1","",0],[,"(9[034]\\d)(\\d{4})","$1-$2",["9[034]"],"0$1","",0]],[[,"(8)(\\d{2,3})(\\d{2,3})(\\d{2})",
|
|
|
"$1 $2 $3 $4",["8"]],[,"([1-69]\\d)(\\d{2,3})(\\d{2})(\\d{2})","$1 $2 $3 $4",["1[013689]|2[0136]|3[1356]|4[0246]|54|6[03]|90"]],[,"([1-69]\\d)(\\d{3})(\\d{2})","$1 $2 $3",["1[13689]|2[136]|3[1356]|4[0246]|54|6[03]|90"]],[,"(\\d{3})(\\d{2})(\\d{2})(\\d{2})","$1 $2 $3 $4",["1[2457]|2[2457-9]|3[0247-9]|4[1357-9]|5[0-35-9]|6[124-9]|9(?:[125-8]|3[0-5]|4[0-3])"]],[,"(\\d{3})(\\d{2,3})(\\d{2})","$1 $2 $3",["1[2457]|2[2457-9]|3[0247-9]|4[1357-9]|5[0-35-9]|6[124-9]|9(?:[125-8]|3[0-5]|4[0-3])"]],[,"(7\\d)(\\d{3})(\\d{2})(\\d{2})",
|
|
|
"$1 $2 $3 $4",["7"]],[,"(77)(\\d{2})(\\d{2})","$1 $2 $3",["7"]],[,"(20)(\\d{2,3})(\\d{2})","$1 $2 $3",["20"]],[,"(9[034]\\d)(\\d{2})(\\d{2})(\\d{3})","$1 $2 $3 $4",["9[034]"]],[,"(9[034]\\d)(\\d{4})","$1 $2",["9[034]"]]],[,,"74[02-9]\\d{6}","\\d{9}",,,"740123456"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],SG:[,[,,"[36]\\d{7}|[17-9]\\d{7,10}","\\d{8,11}"],[,,"6[1-9]\\d{6}","\\d{8}",,,"61234567"],[,,"(?:8[1-7]|9[0-8])\\d{6}","\\d{8}",,,"81234567"],[,,"1?800\\d{7}","\\d{10,11}",,,"18001234567"],
|
|
|
[,,"1900\\d{7}","\\d{11}",,,"19001234567"],[,,"NA","NA"],[,,"NA","NA"],[,,"3[12]\\d{6}","\\d{8}",,,"31234567"],"SG",65,"0[0-3]\\d",,,,,,,,[[,"([3689]\\d{3})(\\d{4})","$1 $2",["[369]|8[1-9]"],"","",0],[,"(1[89]00)(\\d{3})(\\d{4})","$1 $2 $3",["1[89]"],"","",0],[,"(7000)(\\d{4})(\\d{3})","$1 $2 $3",["70"],"","",0],[,"(800)(\\d{3})(\\d{4})","$1 $2 $3",["80"],"","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"7000\\d{7}","\\d{11}",,,"70001234567"],,,[,,"NA","NA"]],SH:[,[,,"[2-79]\\d{3,4}","\\d{4,5}"],[,,"2(?:[0-57-9]\\d|6[4-9])\\d{2}|(?:[2-46]\\d|7[01])\\d{2}",
|
|
|
"\\d{4,5}",,,"2158"],[,,"NA","NA"],[,,"NA","NA"],[,,"(?:[59]\\d|7[2-9])\\d{2}","\\d{4,5}",,,"5012"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"SH",290,"00",,,,,,,,,,[,,"NA","NA"],1,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],SI:[,[,,"[1-7]\\d{6,7}|[89]\\d{4,7}","\\d{5,8}"],[,,"(?:1\\d|[25][2-8]|3[4-8]|4[24-8]|7[3-8])\\d{6}","\\d{7,8}",,,"11234567"],[,,"(?:[37][01]|4[0139]|51|6[48])\\d{6}","\\d{8}",,,"31234567"],[,,"80\\d{4,6}","\\d{6,8}",,,"80123456"],[,,"90\\d{4,6}|89[1-3]\\d{2,5}","\\d{5,8}",
|
|
|
,,"90123456"],[,,"NA","NA"],[,,"NA","NA"],[,,"(?:59|8[1-3])\\d{6}","\\d{8}",,,"59012345"],"SI",386,"00","0",,,"0",,,,[[,"(\\d)(\\d{3})(\\d{2})(\\d{2})","$1 $2 $3 $4",["[12]|3[4-8]|4[24-8]|5[2-8]|7[3-8]"],"(0$1)","",0],[,"([3-7]\\d)(\\d{3})(\\d{3})","$1 $2 $3",["[37][01]|4[0139]|51|6"],"0$1","",0],[,"([89][09])(\\d{3,6})","$1 $2",["[89][09]"],"0$1","",0],[,"([58]\\d{2})(\\d{5})","$1 $2",["59|8[1-3]"],"0$1","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],SJ:[,[,,"0\\d{4}|[4789]\\d{7}",
|
|
|
"\\d{5}(?:\\d{3})?"],[,,"79\\d{6}","\\d{8}",,,"79123456"],[,,"(?:4[015-8]|5[89]|9\\d)\\d{6}","\\d{8}",,,"41234567"],[,,"80[01]\\d{5}","\\d{8}",,,"80012345"],[,,"82[09]\\d{5}","\\d{8}",,,"82012345"],[,,"810(?:0[0-6]|[2-8]\\d)\\d{3}","\\d{8}",,,"81021234"],[,,"880\\d{5}","\\d{8}",,,"88012345"],[,,"85[0-5]\\d{5}","\\d{8}",,,"85012345"],"SJ",47,"00",,,,,,,,,,[,,"NA","NA"],,,[,,"NA","NA"],[,,"0\\d{4}|81(?:0(?:0[7-9]|1\\d)|5\\d{2})\\d{3}","\\d{5}(?:\\d{3})?",,,"01234"],1,,[,,"81[23]\\d{5}","\\d{8}",,,"81212345"]],
|
|
|
SK:[,[,,"[2-689]\\d{8}","\\d{9}"],[,,"[2-5]\\d{8}","\\d{9}",,,"212345678"],[,,"9(?:0[1-8]|1[0-24-9]|4[0489])\\d{6}","\\d{9}",,,"912123456"],[,,"800\\d{6}","\\d{9}",,,"800123456"],[,,"9(?:[78]\\d{7}|00\\d{6})","\\d{9}",,,"900123456"],[,,"8[5-9]\\d{7}","\\d{9}",,,"850123456"],[,,"NA","NA"],[,,"6(?:5[0-4]|9[0-6])\\d{6}","\\d{9}",,,"690123456"],"SK",421,"00","0",,,"0",,,,[[,"(2)(\\d{3})(\\d{3})(\\d{2})","$1/$2 $3 $4",["2"],"0$1","",0],[,"([3-5]\\d)(\\d{3})(\\d{2})(\\d{2})","$1/$2 $3 $4",["[3-5]"],"0$1",
|
|
|
"",0],[,"([689]\\d{2})(\\d{3})(\\d{3})","$1 $2 $3",["[689]"],"0$1","",0]],,[,,"NA","NA"],,,[,,"(?:8(?:00|[5-9]\\d)|9(?:00|[78]\\d))\\d{6}","\\d{9}",,,"800123456"],[,,"96\\d{7}","\\d{9}",,,"961234567"],,,[,,"NA","NA"]],SL:[,[,,"[2-578]\\d{7}","\\d{6,8}"],[,,"[235]2[2-4][2-9]\\d{4}","\\d{6,8}",,,"22221234"],[,,"(?:2[15]|3[034]|4[04]|5[05]|7[6-9]|88)\\d{6}","\\d{6,8}",,,"25123456"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"SL",232,"00","0",,,"0",,,,[[,"(\\d{2})(\\d{6})",
|
|
|
"$1 $2",,"(0$1)","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],SM:[,[,,"[05-7]\\d{7,9}","\\d{6,10}"],[,,"0549(?:8[0157-9]|9\\d)\\d{4}","\\d{6,10}",,,"0549886377"],[,,"6[16]\\d{6}","\\d{8}",,,"66661212"],[,,"NA","NA"],[,,"7[178]\\d{6}","\\d{8}",,,"71123456"],[,,"NA","NA"],[,,"NA","NA"],[,,"5[158]\\d{6}","\\d{8}",,,"58001110"],"SM",378,"00",,,,"(?:0549)?([89]\\d{5})","0549$1",,,[[,"(\\d{2})(\\d{2})(\\d{2})(\\d{2})","$1 $2 $3 $4",["[5-7]"],"","",0],[,"(0549)(\\d{6})","$1 $2",["0"],
|
|
|
"","",0],[,"(\\d{6})","0549 $1",["[89]"],"","",0]],[[,"(\\d{2})(\\d{2})(\\d{2})(\\d{2})","$1 $2 $3 $4",["[5-7]"],"","",0],[,"(0549)(\\d{6})","($1) $2",["0"]],[,"(\\d{6})","(0549) $1",["[89]"]]],[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],1,,[,,"NA","NA"]],SN:[,[,,"[3789]\\d{8}","\\d{9}"],[,,"3(?:0(?:1[0-2]|80)|282|3(?:8[1-9]|9[3-9])|611|90[1-5])\\d{5}","\\d{9}",,,"301012345"],[,,"7(?:[067]\\d|21|8[0-26]|90)\\d{6}","\\d{9}",,,"701234567"],[,,"800\\d{6}","\\d{9}",,,"800123456"],[,,"88[4689]\\d{6}",
|
|
|
"\\d{9}",,,"884123456"],[,,"81[02468]\\d{6}","\\d{9}",,,"810123456"],[,,"NA","NA"],[,,"3392\\d{5}|93330\\d{4}","\\d{9}",,,"933301234"],"SN",221,"00",,,,,,,,[[,"(\\d{2})(\\d{3})(\\d{2})(\\d{2})","$1 $2 $3 $4",["[379]"],"","",0],[,"(\\d{3})(\\d{2})(\\d{2})(\\d{2})","$1 $2 $3 $4",["8"],"","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],SO:[,[,,"[1-79]\\d{6,8}","\\d{7,9}"],[,,"(?:1\\d|2[0-79]|3[0-46-8]|4[0-7]|59)\\d{5}","\\d{7}",,,"4012345"],[,,"(?:15\\d|2(?:4\\d|8)|6[137-9]?\\d{2}|7[1-9]\\d|907\\d)\\d{5}",
|
|
|
"\\d{7,9}",,,"71123456"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"SO",252,"00","0",,,"0",,,,[[,"(\\d)(\\d{6})","$1 $2",["2[0-79]|[13-5]"],"","",0],[,"(\\d)(\\d{7})","$1 $2",["24|[67]"],"","",0],[,"(\\d{2})(\\d{5,7})","$1 $2",["15|28|6[1378]"],"","",0],[,"(69\\d)(\\d{6})","$1 $2",["69"],"","",0],[,"(90\\d)(\\d{3})(\\d{3})","$1 $2 $3",["90"],"","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],SR:[,[,,"[2-8]\\d{5,6}","\\d{6,7}"],[,,"(?:2[1-3]|3[0-7]|4\\d|5[2-58]|68\\d)\\d{4}",
|
|
|
"\\d{6,7}",,,"211234"],[,,"(?:7[124-7]|8[1-9])\\d{5}","\\d{7}",,,"7412345"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"5(?:6\\d{4}|90[0-4]\\d{3})","\\d{6,7}",,,"561234"],"SR",597,"00",,,,,,,,[[,"(\\d{3})(\\d{3})","$1-$2",["[2-4]|5[2-58]"],"","",0],[,"(\\d{2})(\\d{2})(\\d{2})","$1-$2-$3",["56"],"","",0],[,"(\\d{3})(\\d{4})","$1-$2",["59|[6-8]"],"","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],SS:[,[,,"[19]\\d{8}","\\d{9}"],[,,"18\\d{7}","\\d{9}",,,"181234567"],
|
|
|
[,,"(?:12|9[1257])\\d{7}","\\d{9}",,,"977123456"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"SS",211,"00","0",,,"0",,,,[[,"(\\d{3})(\\d{3})(\\d{3})","$1 $2 $3",,"0$1","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],ST:[,[,,"[29]\\d{6}","\\d{7}"],[,,"22\\d{5}","\\d{7}",,,"2221234"],[,,"9[89]\\d{5}","\\d{7}",,,"9812345"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"ST",239,"00",,,,,,,,[[,"(\\d{3})(\\d{4})","$1 $2",,"","",0]],
|
|
|
,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],SV:[,[,,"[267]\\d{7}|[89]\\d{6}(?:\\d{4})?","\\d{7,8}|\\d{11}"],[,,"2[1-6]\\d{6}","\\d{8}",,,"21234567"],[,,"[67]\\d{7}","\\d{8}",,,"70123456"],[,,"800\\d{4}(?:\\d{4})?","\\d{7}(?:\\d{4})?",,,"8001234"],[,,"900\\d{4}(?:\\d{4})?","\\d{7}(?:\\d{4})?",,,"9001234"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"SV",503,"00",,,,,,,,[[,"(\\d{4})(\\d{4})","$1 $2",["[267]"],"","",0],[,"(\\d{3})(\\d{4})","$1 $2",["[89]"],"","",0],[,"(\\d{3})(\\d{4})(\\d{4})",
|
|
|
"$1 $2 $3",["[89]"],"","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],SX:[,[,,"[5789]\\d{9}","\\d{7}(?:\\d{3})?"],[,,"7215(?:4[2-8]|8[239]|9[056])\\d{4}","\\d{7}(?:\\d{3})?",,,"7215425678"],[,,"7215(?:1[02]|2\\d|5[034679]|8[014-8])\\d{4}","\\d{10}",,,"7215205678"],[,,"8(?:00|44|55|66|77|88)[2-9]\\d{6}","\\d{10}",,,"8002123456"],[,,"900[2-9]\\d{6}","\\d{10}",,,"9002123456"],[,,"NA","NA"],[,,"5(?:00|33|44|66|77)[2-9]\\d{6}","\\d{10}",,,"5002345678"],[,,"NA","NA"],"SX",1,"011",
|
|
|
"1",,,"1",,,,,,[,,"NA","NA"],,"721",[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],SY:[,[,,"[1-59]\\d{7,8}","\\d{6,9}"],[,,"(?:1(?:1\\d?|4\\d|[2356])|2(?:1\\d?|[235])|3(?:[13]\\d|4)|4[13]|5[1-3])\\d{6}","\\d{6,9}",,,"112345678"],[,,"9(?:22|[35][0-8]|4\\d|6[024-9]|88|9[0-489])\\d{6}","\\d{9}",,,"944567890"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"SY",963,"00","0",,,"0",,,,[[,"(\\d{2})(\\d{3})(\\d{3,4})","$1 $2 $3",["[1-5]"],"0$1","",1],[,"(9\\d{2})(\\d{3})(\\d{3})","$1 $2 $3",
|
|
|
["9"],"0$1","",1]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],SZ:[,[,,"[027]\\d{7}","\\d{8}"],[,,"2(?:2(?:0[07]|[13]7|2[57])|3(?:0[34]|[1278]3|3[23]|[46][34])|(?:40[4-69]|67)|5(?:0[5-7]|1[6-9]|[23][78]|48|5[01]))\\d{4}","\\d{8}",,,"22171234"],[,,"7[6-8]\\d{6}","\\d{8}",,,"76123456"],[,,"0800\\d{4}","\\d{8}",,,"08001234"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"SZ",268,"00",,,,,,,,[[,"(\\d{4})(\\d{4})","$1 $2",["[027]"],"","",0]],,[,,"NA","NA"],,,[,,"0800\\d{4}",
|
|
|
"\\d{8}",,,"08001234"],[,,"NA","NA"],1,,[,,"NA","NA"]],TA:[,[,,"8\\d{3}","\\d{4}"],[,,"8\\d{3}","\\d{4}",,,"8999"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"TA",290,"00",,,,,,,,,,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],TC:[,[,,"[5689]\\d{9}","\\d{7}(?:\\d{3})?"],[,,"649(?:712|9(?:4\\d|50))\\d{4}","\\d{7}(?:\\d{3})?",,,"6497121234"],[,,"649(?:2(?:3[129]|4[1-7])|3(?:3[1-389]|4[1-7])|4[34][1-3])\\d{4}","\\d{10}",,,"6492311234"],[,,"8(?:00|44|55|66|77|88)[2-9]\\d{6}",
|
|
|
"\\d{10}",,,"8002345678"],[,,"900[2-9]\\d{6}","\\d{10}",,,"9002345678"],[,,"NA","NA"],[,,"5(?:00|33|44|66|77)[2-9]\\d{6}","\\d{10}",,,"5002345678"],[,,"64971[01]\\d{4}","\\d{10}",,,"6497101234"],"TC",1,"011","1",,,"1",,,,,,[,,"NA","NA"],,"649",[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],TD:[,[,,"[2679]\\d{7}","\\d{8}"],[,,"22(?:[3789]0|5[0-5]|6[89])\\d{4}","\\d{8}",,,"22501234"],[,,"(?:6[02368]\\d|77\\d|9(?:5[0-4]|9\\d))\\d{5}","\\d{8}",,,"63012345"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,
|
|
|
"NA","NA"],[,,"NA","NA"],"TD",235,"00|16",,,,,,"00",,[[,"(\\d{2})(\\d{2})(\\d{2})(\\d{2})","$1 $2 $3 $4",,"","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],TG:[,[,,"[29]\\d{7}","\\d{8}"],[,,"2(?:2[2-7]|3[23]|44|55|66|77)\\d{5}","\\d{8}",,,"22212345"],[,,"9[0-289]\\d{6}","\\d{8}",,,"90112345"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"TG",228,"00",,,,,,,,[[,"(\\d{2})(\\d{2})(\\d{2})(\\d{2})","$1 $2 $3 $4",,"","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],
|
|
|
[,,"NA","NA"],,,[,,"NA","NA"]],TH:[,[,,"[2-9]\\d{7,8}|1\\d{3}(?:\\d{6})?","\\d{4}|\\d{8,10}"],[,,"(?:2\\d|3[2-9]|4[2-5]|5[2-6]|7[3-7])\\d{6}","\\d{8}",,,"21234567"],[,,"(?:61|[89]\\d)\\d{7}","\\d{9}",,,"812345678"],[,,"1800\\d{6}","\\d{10}",,,"1800123456"],[,,"1900\\d{6}","\\d{10}",,,"1900123456"],[,,"NA","NA"],[,,"NA","NA"],[,,"6[08]\\d{7}","\\d{9}",,,"601234567"],"TH",66,"00","0",,,"0",,,,[[,"(2)(\\d{3})(\\d{4})","$1 $2 $3",["2"],"0$1","",0],[,"([3-9]\\d)(\\d{3})(\\d{3,4})","$1 $2 $3",["[3-9]"],
|
|
|
"0$1","",0],[,"(1[89]00)(\\d{3})(\\d{3})","$1 $2 $3",["1"],"$1","",0]],,[,,"NA","NA"],,,[,,"1\\d{3}","\\d{4}",,,"1100"],[,,"1\\d{3}","\\d{4}",,,"1100"],,,[,,"NA","NA"]],TJ:[,[,,"[3-59]\\d{8}","\\d{3,9}"],[,,"(?:3(?:1[3-5]|2[245]|3[12]|4[24-7]|5[25]|72)|4(?:46|74|87))\\d{6}","\\d{3,9}",,,"372123456"],[,,"(?:50[125]|9[0-35-9]\\d)\\d{6}","\\d{9}",,,"917123456"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"TJ",992,"810","8",,,"8",,"8~10",,[[,"([349]\\d{2})(\\d{2})(\\d{4})","$1 $2 $3",
|
|
|
["[34]7|91[78]"],"(8) $1","",1],[,"([459]\\d)(\\d{3})(\\d{4})","$1 $2 $3",["4[48]|5|9(?:1[59]|[0235-9])"],"(8) $1","",1],[,"(331700)(\\d)(\\d{2})","$1 $2 $3",["331","3317","33170","331700"],"(8) $1","",1],[,"(\\d{4})(\\d)(\\d{4})","$1 $2 $3",["3[1-5]","3(?:[1245]|3(?:[02-9]|1[0-589]))"],"(8) $1","",1]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],TK:[,[,,"[2-9]\\d{3}","\\d{4}"],[,,"[2-4]\\d{3}","\\d{4}",,,"3010"],[,,"[5-9]\\d{3}","\\d{4}",,,"5190"],[,,"NA","NA"],[,,"NA","NA"],[,,
|
|
|
"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"TK",690,"00",,,,,,,,,,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],TL:[,[,,"[2-489]\\d{6}|7\\d{6,7}","\\d{7,8}"],[,,"(?:2[1-5]|3[1-9]|4[1-4])\\d{5}","\\d{7}",,,"2112345"],[,,"7[3-8]\\d{6}","\\d{8}",,,"77212345"],[,,"80\\d{5}","\\d{7}",,,"8012345"],[,,"90\\d{5}","\\d{7}",,,"9012345"],[,,"NA","NA"],[,,"70\\d{5}","\\d{7}",,,"7012345"],[,,"NA","NA"],"TL",670,"00",,,,,,,,[[,"(\\d{3})(\\d{4})","$1 $2",["[2-489]"],"","",0],[,"(\\d{4})(\\d{4})","$1 $2",
|
|
|
["7"],"","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],TM:[,[,,"[1-6]\\d{7}","\\d{8}"],[,,"(?:1(?:2\\d|3[1-9])|2(?:22|4[0-35-8])|3(?:22|4[03-9])|4(?:22|3[128]|4\\d|6[15])|5(?:22|5[7-9]|6[014-689]))\\d{5}","\\d{8}",,,"12345678"],[,,"6[2-8]\\d{6}","\\d{8}",,,"66123456"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"TM",993,"810","8",,,"8",,"8~10",,[[,"(\\d{2})(\\d{2})(\\d{2})(\\d{2})","$1 $2-$3-$4",["12"],"(8 $1)","",0],[,"(\\d{2})(\\d{6})","$1 $2",["6"],
|
|
|
"8 $1","",0],[,"(\\d{3})(\\d)(\\d{2})(\\d{2})","$1 $2-$3-$4",["13|[2-5]"],"(8 $1)","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],TN:[,[,,"[2-57-9]\\d{7}","\\d{8}"],[,,"3[012]\\d{6}|7\\d{7}|81200\\d{3}","\\d{8}",,,"71234567"],[,,"(?:[259]\\d|4[0-24])\\d{6}","\\d{8}",,,"20123456"],[,,"8010\\d{4}","\\d{8}",,,"80101234"],[,,"88\\d{6}","\\d{8}",,,"88123456"],[,,"8[12]10\\d{4}","\\d{8}",,,"81101234"],[,,"NA","NA"],[,,"NA","NA"],"TN",216,"00",,,,,,,,[[,"(\\d{2})(\\d{3})(\\d{3})","$1 $2 $3",
|
|
|
,"","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],TO:[,[,,"[02-8]\\d{4,6}","\\d{5,7}"],[,,"(?:2\\d|3[1-8]|4[1-4]|[56]0|7[0149]|8[05])\\d{3}","\\d{5}",,,"20123"],[,,"(?:7[578]|8[7-9])\\d{5}","\\d{7}",,,"7715123"],[,,"0800\\d{3}","\\d{7}",,,"0800222"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"TO",676,"00",,,,,,,,[[,"(\\d{2})(\\d{3})","$1-$2",["[1-6]|7[0-4]|8[05]"],"","",0],[,"(\\d{3})(\\d{4})","$1 $2",["7[5-9]|8[7-9]"],"","",0],[,"(\\d{4})(\\d{3})","$1 $2",["0"],
|
|
|
"","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],1,,[,,"NA","NA"]],TR:[,[,,"[2-589]\\d{9}|444\\d{4}","\\d{7,10}"],[,,"(?:2(?:[13][26]|[28][2468]|[45][268]|[67][246])|3(?:[13][28]|[24-6][2468]|[78][02468]|92)|4(?:[16][246]|[23578][2468]|4[26]))\\d{7}","\\d{10}",,,"2123456789"],[,,"5(?:0[1-7]|22|[34]\\d|5[1-59]|9[246])\\d{7}","\\d{10}",,,"5012345678"],[,,"800\\d{7}","\\d{10}",,,"8001234567"],[,,"900\\d{7}","\\d{10}",,,"9001234567"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"TR",90,"00","0",,,
|
|
|
"0",,,,[[,"(\\d{3})(\\d{3})(\\d{4})","$1 $2 $3",["[23]|4(?:[0-35-9]|4[0-35-9])"],"(0$1)","",1],[,"(\\d{3})(\\d{3})(\\d{4})","$1 $2 $3",["[589]"],"0$1","",1],[,"(444)(\\d{1})(\\d{3})","$1 $2 $3",["444"],"","",0]],,[,,"512\\d{7}","\\d{10}",,,"5123456789"],,,[,,"444\\d{4}","\\d{7}",,,"4441444"],[,,"444\\d{4}|850\\d{7}","\\d{7,10}",,,"4441444"],,,[,,"NA","NA"]],TT:[,[,,"[589]\\d{9}","\\d{7}(?:\\d{3})?"],[,,"868(?:2(?:01|2[1-5])|6(?:0[79]|1[02-9]|2[1-9]|[3-69]\\d|7[0-79])|82[124])\\d{4}","\\d{7}(?:\\d{3})?",
|
|
|
,,"8682211234"],[,,"868(?:2(?:[89]\\d)|3(?:0[1-9]|1[02-9]|[2-9]\\d)|4[6-9]\\d|6(?:20|78|8\\d)|7(?:0[1-9]|1[02-9]|[2-9]\\d))\\d{4}","\\d{10}",,,"8682911234"],[,,"8(?:00|44|55|66|77|88)[2-9]\\d{6}","\\d{10}",,,"8002345678"],[,,"900[2-9]\\d{6}","\\d{10}",,,"9002345678"],[,,"NA","NA"],[,,"5(?:00|33|44|66|77)[2-9]\\d{6}","\\d{10}",,,"5002345678"],[,,"NA","NA"],"TT",1,"011","1",,,"1",,,,,,[,,"NA","NA"],,"868",[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],TV:[,[,,"[29]\\d{4,5}","\\d{5,6}"],[,,"2[02-9]\\d{3}",
|
|
|
"\\d{5}",,,"20123"],[,,"90\\d{4}","\\d{6}",,,"901234"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"TV",688,"00",,,,,,,,,,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],TW:[,[,,"[2-689]\\d{7,8}|7\\d{7,9}","\\d{8,10}"],[,,"[2-8]\\d{7,8}","\\d{8,9}",,,"21234567"],[,,"9\\d{8}","\\d{9}",,,"912345678"],[,,"800\\d{6}","\\d{9}",,,"800123456"],[,,"900\\d{6}","\\d{9}",,,"900123456"],[,,"NA","NA"],[,,"NA","NA"],[,,"70\\d{8}","\\d{10}",,,"7012345678"],"TW",886,"0(?:0[25679]|19)",
|
|
|
"0","#",,"0",,,,[[,"([2-8])(\\d{3,4})(\\d{4})","$1 $2 $3",["[2-6]|[78][1-9]"],"0$1","",0],[,"([89]\\d{2})(\\d{3})(\\d{3})","$1 $2 $3",["80|9"],"0$1","",0],[,"(70)(\\d{4})(\\d{4})","$1 $2 $3",["70"],"0$1","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],TZ:[,[,,"\\d{9}","\\d{7,9}"],[,,"2[2-8]\\d{7}","\\d{7,9}",,,"222345678"],[,,"(?:6[158]|7[1-9])\\d{7}","\\d{9}",,,"612345678"],[,,"80[08]\\d{6}","\\d{9}",,,"800123456"],[,,"90\\d{7}","\\d{9}",,,"900123456"],[,,"8(?:40|6[01])\\d{6}",
|
|
|
"\\d{9}",,,"840123456"],[,,"NA","NA"],[,,"41\\d{7}","\\d{9}",,,"412345678"],"TZ",255,"00[056]","0",,,"0",,,,[[,"([24]\\d)(\\d{3})(\\d{4})","$1 $2 $3",["[24]"],"0$1","",0],[,"([67]\\d{2})(\\d{3})(\\d{3})","$1 $2 $3",["[67]"],"0$1","",0],[,"([89]\\d{2})(\\d{2})(\\d{4})","$1 $2 $3",["[89]"],"0$1","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],UA:[,[,,"[3-689]\\d{8}","\\d{5,9}"],[,,"(?:3[1-8]|4[13-8]|5[1-7]|6[12459])\\d{7}","\\d{5,9}",,,"311234567"],[,,"(?:39|50|6[36-8]|9[1-9])\\d{7}",
|
|
|
"\\d{9}",,,"391234567"],[,,"800\\d{6}","\\d{9}",,,"800123456"],[,,"900\\d{6}","\\d{9}",,,"900123456"],[,,"NA","NA"],[,,"NA","NA"],[,,"89\\d{7}","\\d{9}",,,"891234567"],"UA",380,"00","0",,,"0",,"0~0",,[[,"([3-689]\\d)(\\d{3})(\\d{4})","$1 $2 $3",["[38]9|4(?:[45][0-5]|87)|5(?:0|6[37]|7[37])|6[36-8]|9[1-9]","[38]9|4(?:[45][0-5]|87)|5(?:0|6(?:3[14-7]|7)|7[37])|6[36-8]|9[1-9]"],"0$1","",0],[,"([3-689]\\d{2})(\\d{3})(\\d{3})","$1 $2 $3",["3[1-8]2|4[13678]2|5(?:[12457]2|6[24])|6(?:[49]2|[12][29]|5[24])|8[0-8]|90",
|
|
|
"3(?:[1-46-8]2[013-9]|52)|4(?:[1378]2|62[013-9])|5(?:[12457]2|6[24])|6(?:[49]2|[12][29]|5[24])|8[0-8]|90"],"0$1","",0],[,"([3-6]\\d{3})(\\d{5})","$1 $2",["3(?:5[013-9]|[1-46-8])|4(?:[137][013-9]|6|[45][6-9]|8[4-6])|5(?:[1245][013-9]|6[0135-9]|3|7[4-6])|6(?:[49][013-9]|5[0135-9]|[12][13-8])","3(?:5[013-9]|[1-46-8](?:22|[013-9]))|4(?:[137][013-9]|6(?:[013-9]|22)|[45][6-9]|8[4-6])|5(?:[1245][013-9]|6(?:3[02389]|[015689])|3|7[4-6])|6(?:[49][013-9]|5[0135-9]|[12][13-8])"],"0$1","",0]],,[,,"NA","NA"],,
|
|
|
,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],UG:[,[,,"\\d{9}","\\d{5,9}"],[,,"20(?:[0147]\\d{2}|2(?:40|[5-9]\\d)|3[23]\\d|5[0-4]\\d|6[03]\\d|8[0-2]\\d)\\d{4}|[34]\\d{8}","\\d{5,9}",,,"312345678"],[,,"2030\\d{5}|7(?:0[0-7]|[15789]\\d|2[03]|30|[46][0-4])\\d{6}","\\d{9}",,,"712345678"],[,,"800[123]\\d{5}","\\d{9}",,,"800123456"],[,,"90[123]\\d{6}","\\d{9}",,,"901123456"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"UG",256,"00[057]","0",,,"0",,,,[[,"(\\d{3})(\\d{6})","$1 $2",["[7-9]|20(?:[013-8]|2[5-9])|4(?:6[45]|[7-9])"],
|
|
|
"0$1","",0],[,"(\\d{2})(\\d{7})","$1 $2",["3|4(?:[1-5]|6[0-36-9])"],"0$1","",0],[,"(2024)(\\d{5})","$1 $2",["2024"],"0$1","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],US:[,[,,"[2-9]\\d{9}","\\d{7}(?:\\d{3})?"],[,,"(?:2(?:0[1-35-9]|1[02-9]|2[4589]|3[149]|4[08]|5[1-46]|6[0279]|7[026]|8[13])|3(?:0[1-57-9]|1[02-9]|2[0135]|3[014679]|4[67]|5[12]|6[014]|8[56])|4(?:0[124-9]|1[02-579]|2[3-5]|3[0245]|4[0235]|58|69|7[0589]|8[04])|5(?:0[1-57-9]|1[0235-8]|20|3[0149]|4[01]|5[19]|6[1-37]|7[013-5]|8[056])|6(?:0[1-35-9]|1[024-9]|2[036]|3[016]|4[16]|5[017]|6[0-279]|78|8[12])|7(?:0[1-46-8]|1[02-9]|2[0457]|3[1247]|4[07]|5[47]|6[02359]|7[02-59]|8[156])|8(?:0[1-68]|1[02-8]|28|3[0-25]|4[3578]|5[06-9]|6[02-5]|7[028])|9(?:0[1346-9]|1[02-9]|2[0589]|3[01678]|4[0179]|5[12469]|7[0-3589]|8[0459]))[2-9]\\d{6}",
|
|
|
"\\d{7}(?:\\d{3})?",,,"2015555555"],[,,"(?:2(?:0[1-35-9]|1[02-9]|2[4589]|3[149]|4[08]|5[1-46]|6[0279]|7[026]|8[13])|3(?:0[1-57-9]|1[02-9]|2[0135]|3[014679]|4[67]|5[12]|6[014]|8[56])|4(?:0[124-9]|1[02-579]|2[3-5]|3[0245]|4[0235]|58|69|7[0589]|8[04])|5(?:0[1-57-9]|1[0235-8]|20|3[0149]|4[01]|5[19]|6[1-37]|7[013-5]|8[056])|6(?:0[1-35-9]|1[024-9]|2[036]|3[016]|4[16]|5[017]|6[0-279]|78|8[12])|7(?:0[1-46-8]|1[02-9]|2[0457]|3[1247]|4[07]|5[47]|6[02359]|7[02-59]|8[156])|8(?:0[1-68]|1[02-8]|28|3[0-25]|4[3578]|5[06-9]|6[02-5]|7[028])|9(?:0[1346-9]|1[02-9]|2[0589]|3[01678]|4[0179]|5[12469]|7[0-3589]|8[0459]))[2-9]\\d{6}",
|
|
|
"\\d{7}(?:\\d{3})?",,,"2015555555"],[,,"8(?:00|44|55|66|77|88)[2-9]\\d{6}","\\d{10}",,,"8002345678"],[,,"900[2-9]\\d{6}","\\d{10}",,,"9002345678"],[,,"NA","NA"],[,,"5(?:00|33|44|66|77)[2-9]\\d{6}","\\d{10}",,,"5002345678"],[,,"NA","NA"],"US",1,"011","1",,,"1",,,1,[[,"(\\d{3})(\\d{4})","$1-$2",,"","",1],[,"(\\d{3})(\\d{3})(\\d{4})","($1) $2-$3",,"","",1]],[[,"(\\d{3})(\\d{3})(\\d{4})","$1-$2-$3"]],[,,"NA","NA"],1,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],UY:[,[,,"[2489]\\d{6,7}","\\d{7,8}"],[,
|
|
|
,"2\\d{7}|4[2-7]\\d{6}","\\d{7,8}",,,"21231234"],[,,"9[1-9]\\d{6}","\\d{8}",,,"94231234"],[,,"80[05]\\d{4}","\\d{7}",,,"8001234"],[,,"90[0-8]\\d{4}","\\d{7}",,,"9001234"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"UY",598,"0(?:1[3-9]\\d|0)","0"," int. ",,"0",,"00",,[[,"(\\d{4})(\\d{4})","$1 $2",["[24]"],"","",0],[,"(\\d{2})(\\d{3})(\\d{3})","$1 $2 $3",["9[1-9]"],"0$1","",0],[,"(\\d{3})(\\d{4})","$1 $2",["[89]0"],"0$1","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],UZ:[,[,,"[679]\\d{8}",
|
|
|
"\\d{7,9}"],[,,"(?:6(?:1(?:22|3[124]|4[1-4]|5[123578]|64)|2(?:22|3[0-57-9]|41)|5(?:22|3[3-7]|5[024-8])|6\\d{2}|7(?:[23]\\d|7[69])|9(?:22|4[1-8]|6[135]))|7(?:0(?:5[4-9]|6[0146]|7[12456]|9[135-8])|1[12]\\d|2(?:22|3[1345789]|4[123579]|5[14])|3(?:2\\d|3[1578]|4[1-35-7]|5[1-57]|61)|4(?:2\\d|3[1-579]|7[1-79])|5(?:22|5[1-9]|6[1457])|6(?:22|3[12457]|4[13-8])|9(?:22|5[1-9])))\\d{5}","\\d{7,9}",,,"662345678"],[,,"6(?:1(?:2(?:98|2[01])|35[0-4]|50\\d|61[23]|7(?:[01][017]|4\\d|55|9[5-9]))|2(?:11\\d|2(?:[12]1|9[01379])|5(?:[126]\\d|3[0-4])|7\\d{2})|5(?:19[01]|2(?:27|9[26])|30\\d|59\\d|7\\d{2})|6(?:2(?:1[5-9]|2[0367]|38|41|52|60)|3[79]\\d|4(?:56|83)|7(?:[07]\\d|1[017]|3[07]|4[047]|5[057]|67|8[0178]|9[79])|9[0-3]\\d)|7(?:2(?:24|3[237]|4[5-9]|7[15-8])|5(?:7[12]|8[0589])|7(?:0\\d|[39][07])|9(?:0\\d|7[079]))|9(?:2(?:1[1267]|5\\d|3[01]|7[0-4])|5[67]\\d|6(?:2[0-26]|8\\d)|7\\d{2}))\\d{4}|7(?:0\\d{3}|1(?:13[01]|6(?:0[47]|1[67]|66)|71[3-69]|98\\d)|2(?:2(?:2[79]|95)|3(?:2[5-9]|6[0-6])|57\\d|7(?:0\\d|1[17]|2[27]|3[37]|44|5[057]|66|88))|3(?:2(?:1[0-6]|21|3[469]|7[159])|33\\d|5(?:0[0-4]|5[579]|9\\d)|7(?:[0-3579]\\d|4[0467]|6[67]|8[078])|9[4-6]\\d)|4(?:2(?:29|5[0257]|6[0-7]|7[1-57])|5(?:1[0-4]|8\\d|9[5-9])|7(?:0\\d|1[024589]|2[0127]|3[0137]|[46][07]|5[01]|7[5-9]|9[079])|9(?:7[015-9]|[89]\\d))|5(?:112|2(?:0\\d|2[29]|[49]4)|3[1568]\\d|52[6-9]|7(?:0[01578]|1[017]|[23]7|4[047]|[5-7]\\d|8[78]|9[079]))|6(?:2(?:2[1245]|4[2-4])|39\\d|41[179]|5(?:[349]\\d|5[0-2])|7(?:0[017]|[13]\\d|22|44|55|67|88))|9(?:22[128]|3(?:2[0-4]|7\\d)|57[05629]|7(?:2[05-9]|3[37]|4\\d|60|7[2579]|87|9[07])))\\d{4}|9[0-57-9]\\d{7}",
|
|
|
"\\d{7,9}",,,"912345678"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"UZ",998,"810","8",,,"8",,"8~10",,[[,"([679]\\d)(\\d{3})(\\d{2})(\\d{2})","$1 $2 $3 $4",,"8 $1","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],VA:[,[,,"06\\d{8}","\\d{10}"],[,,"06698\\d{5}","\\d{10}",,,"0669812345"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"VA",379,"00",,,,,,,,[[,"(06)(\\d{4})(\\d{4})","$1 $2 $3",,"","",0]],,[,,"NA","NA"],
|
|
|
,,[,,"NA","NA"],[,,"NA","NA"],1,,[,,"NA","NA"]],VC:[,[,,"[5789]\\d{9}","\\d{7}(?:\\d{3})?"],[,,"784(?:266|3(?:6[6-9]|7\\d|8[0-24-6])|4(?:38|5[0-36-8]|8[0-8])|5(?:55|7[0-2]|93)|638|784)\\d{4}","\\d{7}(?:\\d{3})?",,,"7842661234"],[,,"784(?:4(?:3[0-4]|5[45]|89|9[0-5])|5(?:2[6-9]|3[0-4]))\\d{4}","\\d{10}",,,"7844301234"],[,,"8(?:00|44|55|66|77|88)[2-9]\\d{6}","\\d{10}",,,"8002345678"],[,,"900[2-9]\\d{6}","\\d{10}",,,"9002345678"],[,,"NA","NA"],[,,"5(?:00|33|44|66|77)[2-9]\\d{6}","\\d{10}",,,"5002345678"],
|
|
|
[,,"NA","NA"],"VC",1,"011","1",,,"1",,,,,,[,,"NA","NA"],,"784",[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],VE:[,[,,"[24589]\\d{9}","\\d{7,10}"],[,,"(?:2(?:12|3[457-9]|[58][1-9]|[467]\\d|9[1-6])|50[01])\\d{7}","\\d{7,10}",,,"2121234567"],[,,"4(?:1[24-8]|2[46])\\d{7}","\\d{10}",,,"4121234567"],[,,"800\\d{7}","\\d{10}",,,"8001234567"],[,,"900\\d{7}","\\d{10}",,,"9001234567"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"VE",58,"00","0",,,"0",,,,[[,"(\\d{3})(\\d{7})","$1-$2",,"0$1","$CC $1",0]],,[,,"NA",
|
|
|
"NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],VG:[,[,,"[2589]\\d{9}","\\d{7}(?:\\d{3})?"],[,,"284(?:(?:229|4(?:22|9[45])|774|8(?:52|6[459]))\\d{4}|496[0-5]\\d{3})","\\d{7}(?:\\d{3})?",,,"2842291234"],[,,"284(?:(?:3(?:0[0-3]|4[0-367])|4(?:4[0-6]|68|99)|54[0-57])\\d{4}|496[6-9]\\d{3})","\\d{10}",,,"2843001234"],[,,"8(?:00|44|55|66|77|88)[2-9]\\d{6}","\\d{10}",,,"8002345678"],[,,"900[2-9]\\d{6}","\\d{10}",,,"9002345678"],[,,"NA","NA"],[,,"5(?:00|33|44|66|77)[2-9]\\d{6}","\\d{10}",,,"5002345678"],
|
|
|
[,,"NA","NA"],"VG",1,"011","1",,,"1",,,,,,[,,"NA","NA"],,"284",[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],VI:[,[,,"[3589]\\d{9}","\\d{7}(?:\\d{3})?"],[,,"340(?:2(?:01|2[0678]|44|77)|3(?:32|44)|4(?:22|7[34])|5(?:1[34]|55)|6(?:26|4[23]|77|9[023])|7(?:1[2-589]|27|7\\d)|884|998)\\d{4}","\\d{7}(?:\\d{3})?",,,"3406421234"],[,,"340(?:2(?:01|2[0678]|44|77)|3(?:32|44)|4(?:22|7[34])|5(?:1[34]|55)|6(?:26|4[23]|77|9[023])|7(?:1[2-589]|27|7\\d)|884|998)\\d{4}","\\d{7}(?:\\d{3})?",,,"3406421234"],[,,"8(?:00|44|55|66|77|88)[2-9]\\d{6}",
|
|
|
"\\d{10}",,,"8002345678"],[,,"900[2-9]\\d{6}","\\d{10}",,,"9002345678"],[,,"NA","NA"],[,,"5(?:00|33|44|66|77)[2-9]\\d{6}","\\d{10}",,,"5002345678"],[,,"NA","NA"],"VI",1,"011","1",,,"1",,,1,,,[,,"NA","NA"],,"340",[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],VN:[,[,,"[17]\\d{6,9}|[2-69]\\d{7,9}|8\\d{6,8}","\\d{7,10}"],[,,"(?:2(?:[025-79]|1[0189]|[348][01])|3(?:[0136-9]|[25][01])|4\\d|5(?:[01][01]|[2-9])|6(?:[0-46-8]|5[01])|7(?:[02-79]|[18][01])|8[1-9])\\d{7}","\\d{9,10}",,,"2101234567"],[,,"(?:9\\d|1(?:2\\d|6[2-9]|8[68]|99))\\d{7}",
|
|
|
"\\d{9,10}",,,"912345678"],[,,"1800\\d{4,6}","\\d{8,10}",,,"1800123456"],[,,"1900\\d{4,6}","\\d{8,10}",,,"1900123456"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"VN",84,"00","0",,,"0",,,,[[,"([17]99)(\\d{4})","$1 $2",["[17]99"],"0$1","",1],[,"([48])(\\d{4})(\\d{4})","$1 $2 $3",["[48]"],"0$1","",1],[,"([235-7]\\d)(\\d{4})(\\d{3})","$1 $2 $3",["2[025-79]|3[0136-9]|5[2-9]|6[0-46-8]|7[02-79]"],"0$1","",1],[,"(80)(\\d{5})","$1 $2",["80"],"0$1","",1],[,"(69\\d)(\\d{4,5})","$1 $2",["69"],"0$1","",1],[,"([235-7]\\d{2})(\\d{4})(\\d{3})",
|
|
|
"$1 $2 $3",["2[1348]|3[25]|5[01]|65|7[18]"],"0$1","",1],[,"(9\\d)(\\d{3})(\\d{2})(\\d{2})","$1 $2 $3 $4",["9"],"0$1","",1],[,"(1[2689]\\d)(\\d{3})(\\d{4})","$1 $2 $3",["1(?:[26]|8[68]|99)"],"0$1","",1],[,"(1[89]00)(\\d{4,6})","$1 $2",["1[89]0"],"$1","",1]],,[,,"NA","NA"],,,[,,"[17]99\\d{4}|69\\d{5,6}","\\d{7,8}",,,"1992000"],[,,"[17]99\\d{4}|69\\d{5,6}|80\\d{5}","\\d{7,8}",,,"1992000"],,,[,,"NA","NA"]],VU:[,[,,"[2-57-9]\\d{4,6}","\\d{5,7}"],[,,"(?:2[02-9]\\d|3(?:[5-7]\\d|8[0-8])|48[4-9]|88\\d)\\d{2}",
|
|
|
"\\d{5}",,,"22123"],[,,"(?:5(?:7[2-5]|[3-69]\\d)|7[013-7]\\d)\\d{4}","\\d{7}",,,"5912345"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"VU",678,"00",,,,,,,,[[,"(\\d{3})(\\d{4})","$1 $2",["[579]"],"","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"3[03]\\d{3}|900\\d{4}","\\d{5,7}",,,"30123"],,,[,,"NA","NA"]],WF:[,[,,"[5-7]\\d{5}","\\d{6}"],[,,"(?:50|68|72)\\d{4}","\\d{6}",,,"501234"],[,,"(?:50|68|72)\\d{4}","\\d{6}",,,"501234"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA",
|
|
|
"NA"],[,,"NA","NA"],"WF",681,"00",,,,,,,1,[[,"(\\d{2})(\\d{2})(\\d{2})","$1 $2 $3",,"","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],WS:[,[,,"[2-8]\\d{4,6}","\\d{5,7}"],[,,"(?:[2-5]\\d|6[1-9]|84\\d{2})\\d{3}","\\d{5,7}",,,"22123"],[,,"(?:60|7[25-7]\\d)\\d{4}","\\d{6,7}",,,"601234"],[,,"800\\d{3}","\\d{6}",,,"800123"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"WS",685,"0",,,,,,,,[[,"(8\\d{2})(\\d{3,4})","$1 $2",["8"],"","",0],[,"(7\\d)(\\d{5})","$1 $2",["7"],"",
|
|
|
"",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],YE:[,[,,"[1-7]\\d{6,8}","\\d{6,9}"],[,,"(?:1(?:7\\d|[2-68])|2[2-68]|3[2358]|4[2-58]|5[2-6]|6[3-58]|7[24-68])\\d{5}","\\d{6,8}",,,"1234567"],[,,"7[0137]\\d{7}","\\d{9}",,,"712345678"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"YE",967,"00","0",,,"0",,,,[[,"([1-7])(\\d{3})(\\d{3,4})","$1 $2 $3",["[1-6]|7[24-68]"],"0$1","",0],[,"(7\\d{2})(\\d{3})(\\d{3})","$1 $2 $3",["7[0137]"],"0$1","",0]],,[,,"NA","NA"],
|
|
|
,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],YT:[,[,,"[268]\\d{8}","\\d{9}"],[,,"2696[0-4]\\d{4}","\\d{9}",,,"269601234"],[,,"639\\d{6}","\\d{9}",,,"639123456"],[,,"80\\d{7}","\\d{9}",,,"801234567"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"YT",262,"00","0",,,"0",,,,,,[,,"NA","NA"],,"269|63",[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],ZA:[,[,,"[1-79]\\d{8}|8(?:[067]\\d{7}|[1-4]\\d{3,7})","\\d{5,9}"],[,,"(?:1[0-8]|2[0-378]|3[1-69]|4\\d|5[1346-8])\\d{7}","\\d{9}",,,"101234567"],[,
|
|
|
,"(?:6[0-5]|7[0-46-9])\\d{7}|8[1-4]\\d{3,7}","\\d{5,9}",,,"711234567"],[,,"80\\d{7}","\\d{9}",,,"801234567"],[,,"86[2-9]\\d{6}|90\\d{7}","\\d{9}",,,"862345678"],[,,"860\\d{6}","\\d{9}",,,"860123456"],[,,"NA","NA"],[,,"87\\d{7}","\\d{9}",,,"871234567"],"ZA",27,"00","0",,,"0",,,,[[,"(860)(\\d{3})(\\d{3})","$1 $2 $3",["860"],"0$1","",0],[,"(\\d{2})(\\d{3})(\\d{4})","$1 $2 $3",["[1-79]|8(?:[0-47]|6[1-9])"],"0$1","",0],[,"(\\d{2})(\\d{3,4})","$1 $2",["8[1-4]"],"0$1","",0],[,"(\\d{2})(\\d{3})(\\d{2,3})",
|
|
|
"$1 $2 $3",["8[1-4]"],"0$1","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"861\\d{6}","\\d{9}",,,"861123456"],,,[,,"NA","NA"]],ZM:[,[,,"[289]\\d{8}","\\d{9}"],[,,"21[1-8]\\d{6}","\\d{9}",,,"211234567"],[,,"9(?:5[05]|6\\d|7[1-9])\\d{6}","\\d{9}",,,"955123456"],[,,"800\\d{6}","\\d{9}",,,"800123456"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"ZM",260,"00","0",,,"0",,,,[[,"([29]\\d)(\\d{7})","$1 $2",["[29]"],"0$1","",0],[,"(800)(\\d{3})(\\d{3})","$1 $2 $3",["8"],"0$1","",0]],,[,,"NA","NA"],,
|
|
|
,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],ZW:[,[,,"2(?:[012457-9]\\d{3,8}|6\\d{3,6})|[13-79]\\d{4,8}|8[06]\\d{8}","\\d{3,10}"],[,,"(?:1[3-9]|2(?:0[45]|[16]|2[28]|[49]8?|58[23]|7[246]|8[1346-9])|3(?:08?|17?|3[78]|[2456]|7[1569]|8[379])|5(?:[07-9]|1[78]|483|5(?:7?|8))|6(?:0|28|37?|[45][68][78]|98?)|848)\\d{3,6}|(?:2(?:27|5|7[135789]|8[25])|3[39]|5[1-46]|6[126-8])\\d{4,6}|2(?:(?:0|70)\\d{5,6}|2[05]\\d{7})|(?:4\\d|9[2-8])\\d{4,7}","\\d{3,10}",,,"1312345"],[,,"7[1378]\\d{7}|86(?:22|44)\\d{6}","\\d{9,10}",
|
|
|
,,"711234567"],[,,"800\\d{7}","\\d{10}",,,"8001234567"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"86(?:1[12]|30|55|77|8[367]|99)\\d{6}","\\d{10}",,,"8686123456"],"ZW",263,"00","0",,,"0",,,,[[,"([49])(\\d{3})(\\d{2,5})","$1 $2 $3",["4|9[2-9]"],"0$1","",0],[,"([179]\\d)(\\d{3})(\\d{3,4})","$1 $2 $3",["[19]1|7"],"0$1","",0],[,"(86\\d{2})(\\d{3})(\\d{3})","$1 $2 $3",["86[24]"],"0$1","",0],[,"([2356]\\d{2})(\\d{3,5})","$1 $2",["2(?:[278]|0[45]|[49]8)|3(?:08|17|3[78]|[78])|5[15][78]|6(?:[29]8|37|[68][78])"],
|
|
|
"0$1","",0],[,"(\\d{3})(\\d{3})(\\d{3,4})","$1 $2 $3",["2(?:[278]|0[45]|48)|3(?:08|17|3[78]|[78])|5[15][78]|6(?:[29]8|37|[68][78])|80"],"0$1","",0],[,"([1-356]\\d)(\\d{3,5})","$1 $2",["1[3-9]|2(?:[1-469]|0[0-35-9]|[45][0-79])|3(?:0[0-79]|1[0-689]|[24-69]|3[0-69])|5(?:[02-46-9]|[15][0-69])|6(?:[0145]|[29][0-79]|3[0-689]|[68][0-69])"],"0$1","",0],[,"([1-356]\\d)(\\d{3})(\\d{3})","$1 $2 $3",["1[3-9]|2(?:[1-469]|0[0-35-9]|[45][0-79])|3(?:0[0-79]|1[0-689]|[24-69]|3[0-69])|5(?:[02-46-9]|[15][0-69])|6(?:[0145]|[29][0-79]|3[0-689]|[68][0-69])"],
|
|
|
"0$1","",0],[,"([25]\\d{3})(\\d{3,5})","$1 $2",["(?:25|54)8","258[23]|5483"],"0$1","",0],[,"([25]\\d{3})(\\d{3})(\\d{3})","$1 $2 $3",["(?:25|54)8","258[23]|5483"],"0$1","",0],[,"(8\\d{3})(\\d{6})","$1 $2",["86"],"0$1","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],800:[,[,,"\\d{8}","\\d{8}",,,"12345678"],[,,"NA","NA",,,"12345678"],[,,"NA","NA",,,"12345678"],[,,"\\d{8}","\\d{8}",,,"12345678"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"001",800,"",,,,,,,1,[[,"(\\d{4})(\\d{4})",
|
|
|
"$1 $2",,"","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],1,,[,,"NA","NA"]],808:[,[,,"\\d{8}","\\d{8}",,,"12345678"],[,,"NA","NA",,,"12345678"],[,,"NA","NA",,,"12345678"],[,,"NA","NA"],[,,"NA","NA"],[,,"\\d{8}","\\d{8}",,,"12345678"],[,,"NA","NA"],[,,"NA","NA"],"001",808,"",,,,,,,1,[[,"(\\d{4})(\\d{4})","$1 $2",,"","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],1,,[,,"NA","NA"]],870:[,[,,"[35-7]\\d{8}","\\d{9}",,,"301234567"],[,,"NA","NA",,,"301234567"],[,,"(?:[356]\\d|7[6-8])\\d{7}","\\d{9}",
|
|
|
,,"301234567"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"001",870,"",,,,,,,,[[,"(\\d{3})(\\d{3})(\\d{3})","$1 $2 $3",,"","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],878:[,[,,"1\\d{11}","\\d{12}",,,"101234567890"],[,,"NA","NA",,,"101234567890"],[,,"NA","NA",,,"101234567890"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"10\\d{10}","\\d{12}",,,"101234567890"],"001",878,"",,,,,,,1,[[,"(\\d{2})(\\d{5})(\\d{5})","$1 $2 $3",,"","",0]],
|
|
|
,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],881:[,[,,"[67]\\d{8}","\\d{9}",,,"612345678"],[,,"NA","NA",,,"612345678"],[,,"[67]\\d{8}","\\d{9}",,,"612345678"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"001",881,"",,,,,,,,[[,"(\\d)(\\d{3})(\\d{5})","$1 $2 $3",["[67]"],"","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],882:[,[,,"[13]\\d{6,11}","\\d{7,12}",,,"3451234567"],[,,"NA","NA",,,"3451234567"],[,,"3(?:2\\d{3}|37\\d{2}|4(?:2|7\\d{3}))\\d{4}",
|
|
|
"\\d{7,10}",,,"3451234567"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"1(?:3(?:0[0347]|[13][0139]|2[035]|4[013568]|6[0459]|7[06]|8[15678]|9[0689])\\d{4}|6\\d{5,10})|345\\d{7}","\\d{7,12}",,,"3451234567"],"001",882,"",,,,,,,,[[,"(\\d{2})(\\d{4})(\\d{3})","$1 $2 $3",["3[23]"],"","",0],[,"(\\d{2})(\\d{5})","$1 $2",["16|342"],"","",0],[,"(\\d{2})(\\d{4})(\\d{4})","$1 $2 $3",["34[57]"],"","",0],[,"(\\d{3})(\\d{4})(\\d{4})","$1 $2 $3",["348"],"","",0],[,"(\\d{2})(\\d{2})(\\d{4})","$1 $2 $3",
|
|
|
["1"],"","",0],[,"(\\d{2})(\\d{3,4})(\\d{4})","$1 $2 $3",["16"],"","",0],[,"(\\d{2})(\\d{4,5})(\\d{5})","$1 $2 $3",["16"],"","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"348[57]\\d{7}","\\d{11}",,,"3451234567"]],883:[,[,,"51\\d{7}(?:\\d{3})?","\\d{9}(?:\\d{3})?",,,"510012345"],[,,"NA","NA",,,"510012345"],[,,"NA","NA",,,"510012345"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"51(?:00\\d{5}(?:\\d{3})?|[13]0\\d{8})","\\d{9}(?:\\d{3})?",,,"510012345"],"001",883,"",,,,,,,1,
|
|
|
[[,"(\\d{3})(\\d{3})(\\d{3})","$1 $2 $3",["510"],"","",0],[,"(\\d{3})(\\d{3})(\\d{3})(\\d{3})","$1 $2 $3 $4",["510"],"","",0],[,"(\\d{4})(\\d{4})(\\d{4})","$1 $2 $3",["51[13]"],"","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],,,[,,"NA","NA"]],888:[,[,,"\\d{11}","\\d{11}",,,"12345678901"],[,,"NA","NA",,,"12345678901"],[,,"NA","NA",,,"12345678901"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"001",888,"",,,,,,,1,[[,"(\\d{3})(\\d{3})(\\d{5})","$1 $2 $3",,"","",0]],,[,
|
|
|
,"NA","NA"],,,[,,"NA","NA"],[,,"\\d{11}","\\d{11}",,,"12345678901"],1,,[,,"NA","NA"]],979:[,[,,"\\d{9}","\\d{9}",,,"123456789"],[,,"NA","NA",,,"123456789"],[,,"NA","NA",,,"123456789"],[,,"NA","NA"],[,,"\\d{9}","\\d{9}",,,"123456789"],[,,"NA","NA"],[,,"NA","NA"],[,,"NA","NA"],"001",979,"",,,,,,,1,[[,"(\\d)(\\d{4})(\\d{4})","$1 $2 $3",,"","",0]],,[,,"NA","NA"],,,[,,"NA","NA"],[,,"NA","NA"],1,,[,,"NA","NA"]]};/*
|
|
|
|
|
|
Copyright (C) 2010 The Libphonenumber Authors.
|
|
|
|
|
|
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.
|
|
|
*/
|
|
|
i18n.phonenumbers.PhoneNumberUtil=function(){this.regionToMetadataMap={}};goog.addSingletonGetter(i18n.phonenumbers.PhoneNumberUtil);i18n.phonenumbers.Error={INVALID_COUNTRY_CODE:"Invalid country calling code",NOT_A_NUMBER:"The string supplied did not seem to be a phone number",TOO_SHORT_AFTER_IDD:"Phone number too short after IDD",TOO_SHORT_NSN:"The string supplied is too short to be a phone number",TOO_LONG:"The string supplied is too long to be a phone number"};
|
|
|
i18n.phonenumbers.PhoneNumberUtil.NANPA_COUNTRY_CODE_=1;i18n.phonenumbers.PhoneNumberUtil.MIN_LENGTH_FOR_NSN_=2;i18n.phonenumbers.PhoneNumberUtil.MAX_LENGTH_FOR_NSN_=17;i18n.phonenumbers.PhoneNumberUtil.MAX_LENGTH_COUNTRY_CODE_=3;i18n.phonenumbers.PhoneNumberUtil.MAX_INPUT_STRING_LENGTH_=250;i18n.phonenumbers.PhoneNumberUtil.UNKNOWN_REGION_="ZZ";i18n.phonenumbers.PhoneNumberUtil.COLOMBIA_MOBILE_TO_FIXED_LINE_PREFIX_="3";i18n.phonenumbers.PhoneNumberUtil.MOBILE_TOKEN_MAPPINGS_={52:"1",54:"9"};
|
|
|
i18n.phonenumbers.PhoneNumberUtil.PLUS_SIGN="+";i18n.phonenumbers.PhoneNumberUtil.STAR_SIGN_="*";i18n.phonenumbers.PhoneNumberUtil.RFC3966_EXTN_PREFIX_=";ext=";i18n.phonenumbers.PhoneNumberUtil.RFC3966_PREFIX_="tel:";i18n.phonenumbers.PhoneNumberUtil.RFC3966_PHONE_CONTEXT_=";phone-context=";i18n.phonenumbers.PhoneNumberUtil.RFC3966_ISDN_SUBADDRESS_=";isub=";
|
|
|
i18n.phonenumbers.PhoneNumberUtil.DIGIT_MAPPINGS={0:"0",1:"1",2:"2",3:"3",4:"4",5:"5",6:"6",7:"7",8:"8",9:"9","\uff10":"0","\uff11":"1","\uff12":"2","\uff13":"3","\uff14":"4","\uff15":"5","\uff16":"6","\uff17":"7","\uff18":"8","\uff19":"9","\u0660":"0","\u0661":"1","\u0662":"2","\u0663":"3","\u0664":"4","\u0665":"5","\u0666":"6","\u0667":"7","\u0668":"8","\u0669":"9","\u06f0":"0","\u06f1":"1","\u06f2":"2","\u06f3":"3","\u06f4":"4","\u06f5":"5","\u06f6":"6","\u06f7":"7","\u06f8":"8","\u06f9":"9"};
|
|
|
i18n.phonenumbers.PhoneNumberUtil.DIALLABLE_CHAR_MAPPINGS_={0:"0",1:"1",2:"2",3:"3",4:"4",5:"5",6:"6",7:"7",8:"8",9:"9","+":i18n.phonenumbers.PhoneNumberUtil.PLUS_SIGN,"*":"*"};i18n.phonenumbers.PhoneNumberUtil.ALPHA_MAPPINGS_={A:"2",B:"2",C:"2",D:"3",E:"3",F:"3",G:"4",H:"4",I:"4",J:"5",K:"5",L:"5",M:"6",N:"6",O:"6",P:"7",Q:"7",R:"7",S:"7",T:"8",U:"8",V:"8",W:"9",X:"9",Y:"9",Z:"9"};
|
|
|
i18n.phonenumbers.PhoneNumberUtil.ALL_NORMALIZATION_MAPPINGS_={0:"0",1:"1",2:"2",3:"3",4:"4",5:"5",6:"6",7:"7",8:"8",9:"9","\uff10":"0","\uff11":"1","\uff12":"2","\uff13":"3","\uff14":"4","\uff15":"5","\uff16":"6","\uff17":"7","\uff18":"8","\uff19":"9","\u0660":"0","\u0661":"1","\u0662":"2","\u0663":"3","\u0664":"4","\u0665":"5","\u0666":"6","\u0667":"7","\u0668":"8","\u0669":"9","\u06f0":"0","\u06f1":"1","\u06f2":"2","\u06f3":"3","\u06f4":"4","\u06f5":"5","\u06f6":"6","\u06f7":"7","\u06f8":"8","\u06f9":"9",
|
|
|
A:"2",B:"2",C:"2",D:"3",E:"3",F:"3",G:"4",H:"4",I:"4",J:"5",K:"5",L:"5",M:"6",N:"6",O:"6",P:"7",Q:"7",R:"7",S:"7",T:"8",U:"8",V:"8",W:"9",X:"9",Y:"9",Z:"9"};
|
|
|
i18n.phonenumbers.PhoneNumberUtil.ALL_PLUS_NUMBER_GROUPING_SYMBOLS_={0:"0",1:"1",2:"2",3:"3",4:"4",5:"5",6:"6",7:"7",8:"8",9:"9",A:"A",B:"B",C:"C",D:"D",E:"E",F:"F",G:"G",H:"H",I:"I",J:"J",K:"K",L:"L",M:"M",N:"N",O:"O",P:"P",Q:"Q",R:"R",S:"S",T:"T",U:"U",V:"V",W:"W",X:"X",Y:"Y",Z:"Z",a:"A",b:"B",c:"C",d:"D",e:"E",f:"F",g:"G",h:"H",i:"I",j:"J",k:"K",l:"L",m:"M",n:"N",o:"O",p:"P",q:"Q",r:"R",s:"S",t:"T",u:"U",v:"V",w:"W",x:"X",y:"Y",z:"Z","-":"-","\uff0d":"-","\u2010":"-","\u2011":"-","\u2012":"-",
|
|
|
"\u2013":"-","\u2014":"-","\u2015":"-","\u2212":"-","/":"/","\uff0f":"/"," ":" ","\u3000":" ","\u2060":" ",".":".","\uff0e":"."};i18n.phonenumbers.PhoneNumberUtil.UNIQUE_INTERNATIONAL_PREFIX_=/[\d]+(?:[~\u2053\u223C\uFF5E][\d]+)?/;i18n.phonenumbers.PhoneNumberUtil.VALID_PUNCTUATION="-x\u2010-\u2015\u2212\u30fc\uff0d-\uff0f \u00a0\u00ad\u200b\u2060\u3000()\uff08\uff09\uff3b\uff3d.\\[\\]/~\u2053\u223c\uff5e";i18n.phonenumbers.PhoneNumberUtil.VALID_DIGITS_="0-9\uff10-\uff19\u0660-\u0669\u06f0-\u06f9";
|
|
|
i18n.phonenumbers.PhoneNumberUtil.VALID_ALPHA_="A-Za-z";i18n.phonenumbers.PhoneNumberUtil.PLUS_CHARS_="+\uff0b";i18n.phonenumbers.PhoneNumberUtil.PLUS_CHARS_PATTERN=new RegExp("["+i18n.phonenumbers.PhoneNumberUtil.PLUS_CHARS_+"]+");i18n.phonenumbers.PhoneNumberUtil.LEADING_PLUS_CHARS_PATTERN_=new RegExp("^["+i18n.phonenumbers.PhoneNumberUtil.PLUS_CHARS_+"]+");i18n.phonenumbers.PhoneNumberUtil.SEPARATOR_PATTERN_="["+i18n.phonenumbers.PhoneNumberUtil.VALID_PUNCTUATION+"]+";
|
|
|
i18n.phonenumbers.PhoneNumberUtil.CAPTURING_DIGIT_PATTERN=new RegExp("(["+i18n.phonenumbers.PhoneNumberUtil.VALID_DIGITS_+"])");i18n.phonenumbers.PhoneNumberUtil.VALID_START_CHAR_PATTERN_=new RegExp("["+i18n.phonenumbers.PhoneNumberUtil.PLUS_CHARS_+i18n.phonenumbers.PhoneNumberUtil.VALID_DIGITS_+"]");i18n.phonenumbers.PhoneNumberUtil.SECOND_NUMBER_START_PATTERN_=/[\\\/] *x/;
|
|
|
i18n.phonenumbers.PhoneNumberUtil.UNWANTED_END_CHAR_PATTERN_=new RegExp("[^"+i18n.phonenumbers.PhoneNumberUtil.VALID_DIGITS_+i18n.phonenumbers.PhoneNumberUtil.VALID_ALPHA_+"#]+$");i18n.phonenumbers.PhoneNumberUtil.VALID_ALPHA_PHONE_PATTERN_=/(?:.*?[A-Za-z]){3}.*/;i18n.phonenumbers.PhoneNumberUtil.MIN_LENGTH_PHONE_NUMBER_PATTERN_="["+i18n.phonenumbers.PhoneNumberUtil.VALID_DIGITS_+"]{"+i18n.phonenumbers.PhoneNumberUtil.MIN_LENGTH_FOR_NSN_+"}";
|
|
|
i18n.phonenumbers.PhoneNumberUtil.VALID_PHONE_NUMBER_="["+i18n.phonenumbers.PhoneNumberUtil.PLUS_CHARS_+"]*(?:["+i18n.phonenumbers.PhoneNumberUtil.VALID_PUNCTUATION+i18n.phonenumbers.PhoneNumberUtil.STAR_SIGN_+"]*["+i18n.phonenumbers.PhoneNumberUtil.VALID_DIGITS_+"]){3,}["+i18n.phonenumbers.PhoneNumberUtil.VALID_PUNCTUATION+i18n.phonenumbers.PhoneNumberUtil.STAR_SIGN_+i18n.phonenumbers.PhoneNumberUtil.VALID_ALPHA_+i18n.phonenumbers.PhoneNumberUtil.VALID_DIGITS_+"]*";
|
|
|
i18n.phonenumbers.PhoneNumberUtil.DEFAULT_EXTN_PREFIX_=" ext. ";i18n.phonenumbers.PhoneNumberUtil.CAPTURING_EXTN_DIGITS_="(["+i18n.phonenumbers.PhoneNumberUtil.VALID_DIGITS_+"]{1,7})";
|
|
|
i18n.phonenumbers.PhoneNumberUtil.EXTN_PATTERNS_FOR_PARSING_=i18n.phonenumbers.PhoneNumberUtil.RFC3966_EXTN_PREFIX_+i18n.phonenumbers.PhoneNumberUtil.CAPTURING_EXTN_DIGITS_+"|[ \u00a0\\t,]*(?:e?xt(?:ensi(?:o\u0301?|\u00f3))?n?|\uff45?\uff58\uff54\uff4e?|[,x\uff58#\uff03~\uff5e]|int|anexo|\uff49\uff4e\uff54)[:\\.\uff0e]?[ \u00a0\\t,-]*"+i18n.phonenumbers.PhoneNumberUtil.CAPTURING_EXTN_DIGITS_+"#?|[- ]+(["+i18n.phonenumbers.PhoneNumberUtil.VALID_DIGITS_+"]{1,5})#";
|
|
|
i18n.phonenumbers.PhoneNumberUtil.EXTN_PATTERN_=new RegExp("(?:"+i18n.phonenumbers.PhoneNumberUtil.EXTN_PATTERNS_FOR_PARSING_+")$","i");i18n.phonenumbers.PhoneNumberUtil.VALID_PHONE_NUMBER_PATTERN_=new RegExp("^"+i18n.phonenumbers.PhoneNumberUtil.MIN_LENGTH_PHONE_NUMBER_PATTERN_+"$|^"+i18n.phonenumbers.PhoneNumberUtil.VALID_PHONE_NUMBER_+"(?:"+i18n.phonenumbers.PhoneNumberUtil.EXTN_PATTERNS_FOR_PARSING_+")?$","i");i18n.phonenumbers.PhoneNumberUtil.NON_DIGITS_PATTERN_=/\D+/;
|
|
|
i18n.phonenumbers.PhoneNumberUtil.FIRST_GROUP_PATTERN_=/(\$\d)/;i18n.phonenumbers.PhoneNumberUtil.NP_PATTERN_=/\$NP/;i18n.phonenumbers.PhoneNumberUtil.FG_PATTERN_=/\$FG/;i18n.phonenumbers.PhoneNumberUtil.CC_PATTERN_=/\$CC/;i18n.phonenumbers.PhoneNumberUtil.FIRST_GROUP_ONLY_PREFIX_PATTERN_=/^\(?\$1\)?$/;i18n.phonenumbers.PhoneNumberUtil.REGION_CODE_FOR_NON_GEO_ENTITY="001";i18n.phonenumbers.PhoneNumberFormat={E164:0,INTERNATIONAL:1,NATIONAL:2,RFC3966:3};
|
|
|
i18n.phonenumbers.PhoneNumberType={FIXED_LINE:0,MOBILE:1,FIXED_LINE_OR_MOBILE:2,TOLL_FREE:3,PREMIUM_RATE:4,SHARED_COST:5,VOIP:6,PERSONAL_NUMBER:7,PAGER:8,UAN:9,VOICEMAIL:10,UNKNOWN:-1};i18n.phonenumbers.PhoneNumberUtil.MatchType={NOT_A_NUMBER:0,NO_MATCH:1,SHORT_NSN_MATCH:2,NSN_MATCH:3,EXACT_MATCH:4};i18n.phonenumbers.PhoneNumberUtil.ValidationResult={IS_POSSIBLE:0,INVALID_COUNTRY_CODE:1,TOO_SHORT:2,TOO_LONG:3};
|
|
|
i18n.phonenumbers.PhoneNumberUtil.extractPossibleNumber=function(a){var b=a.search(i18n.phonenumbers.PhoneNumberUtil.VALID_START_CHAR_PATTERN_);0<=b?(a=a.substring(b),a=a.replace(i18n.phonenumbers.PhoneNumberUtil.UNWANTED_END_CHAR_PATTERN_,""),b=a.search(i18n.phonenumbers.PhoneNumberUtil.SECOND_NUMBER_START_PATTERN_),0<=b&&(a=a.substring(0,b))):a="";return a};
|
|
|
i18n.phonenumbers.PhoneNumberUtil.isViablePhoneNumber=function(a){return a.length<i18n.phonenumbers.PhoneNumberUtil.MIN_LENGTH_FOR_NSN_?!1:i18n.phonenumbers.PhoneNumberUtil.matchesEntirely_(i18n.phonenumbers.PhoneNumberUtil.VALID_PHONE_NUMBER_PATTERN_,a)};
|
|
|
i18n.phonenumbers.PhoneNumberUtil.normalize=function(a){return i18n.phonenumbers.PhoneNumberUtil.matchesEntirely_(i18n.phonenumbers.PhoneNumberUtil.VALID_ALPHA_PHONE_PATTERN_,a)?i18n.phonenumbers.PhoneNumberUtil.normalizeHelper_(a,i18n.phonenumbers.PhoneNumberUtil.ALL_NORMALIZATION_MAPPINGS_,!0):i18n.phonenumbers.PhoneNumberUtil.normalizeDigitsOnly(a)};i18n.phonenumbers.PhoneNumberUtil.normalizeSB_=function(a){var b=i18n.phonenumbers.PhoneNumberUtil.normalize(a.toString());a.clear();a.append(b)};
|
|
|
i18n.phonenumbers.PhoneNumberUtil.normalizeDigitsOnly=function(a){return i18n.phonenumbers.PhoneNumberUtil.normalizeHelper_(a,i18n.phonenumbers.PhoneNumberUtil.DIGIT_MAPPINGS,!0)};i18n.phonenumbers.PhoneNumberUtil.convertAlphaCharactersInNumber=function(a){return i18n.phonenumbers.PhoneNumberUtil.normalizeHelper_(a,i18n.phonenumbers.PhoneNumberUtil.ALL_NORMALIZATION_MAPPINGS_,!1)};
|
|
|
i18n.phonenumbers.PhoneNumberUtil.prototype.getLengthOfGeographicalAreaCode=function(a){var b=this.getMetadataForRegion(this.getRegionCodeForNumber(a));return null!=b&&(b.hasNationalPrefix()||a.hasItalianLeadingZero())&&this.isNumberGeographical(a)?this.getLengthOfNationalDestinationCode(a):0};
|
|
|
i18n.phonenumbers.PhoneNumberUtil.prototype.getLengthOfNationalDestinationCode=function(a){var b;a.hasExtension()?(b=a.clone(),b.clearExtension()):b=a;b=this.format(b,i18n.phonenumbers.PhoneNumberFormat.INTERNATIONAL).split(i18n.phonenumbers.PhoneNumberUtil.NON_DIGITS_PATTERN_);0==b[0].length&&b.shift();return 2>=b.length?0:this.getNumberType(a)==i18n.phonenumbers.PhoneNumberType.MOBILE&&(a=i18n.phonenumbers.PhoneNumberUtil.getCountryMobileToken(a.getCountryCodeOrDefault()),""!=a)?b[2].length+a.length:
|
|
|
b[1].length};i18n.phonenumbers.PhoneNumberUtil.getCountryMobileToken=function(a){return i18n.phonenumbers.PhoneNumberUtil.MOBILE_TOKEN_MAPPINGS_[a]||""};i18n.phonenumbers.PhoneNumberUtil.prototype.getSupportedRegions=function(){return goog.array.filter(Object.keys(i18n.phonenumbers.metadata.countryToMetadata),function(a){return isNaN(a)})};
|
|
|
i18n.phonenumbers.PhoneNumberUtil.prototype.getSupportedGlobalNetworkCallingCodes=function(){var a=goog.array.filter(Object.keys(i18n.phonenumbers.metadata.countryToMetadata),function(a){return!isNaN(a)});return goog.array.map(a,function(a){return parseInt(a,10)})};i18n.phonenumbers.PhoneNumberUtil.normalizeHelper_=function(a,b,c){for(var d=new goog.string.StringBuffer,e,f,g=a.length,h=0;h<g;++h)e=a.charAt(h),f=b[e.toUpperCase()],null!=f?d.append(f):c||d.append(e);return d.toString()};
|
|
|
i18n.phonenumbers.PhoneNumberUtil.prototype.formattingRuleHasFirstGroupOnly=function(a){return 0==a.length||i18n.phonenumbers.PhoneNumberUtil.FIRST_GROUP_ONLY_PREFIX_PATTERN_.test(a)};i18n.phonenumbers.PhoneNumberUtil.prototype.isNumberGeographical=function(a){a=this.getNumberType(a);return a==i18n.phonenumbers.PhoneNumberType.FIXED_LINE||a==i18n.phonenumbers.PhoneNumberType.FIXED_LINE_OR_MOBILE};
|
|
|
i18n.phonenumbers.PhoneNumberUtil.prototype.isValidRegionCode_=function(a){return null!=a&&isNaN(a)&&a.toUpperCase()in i18n.phonenumbers.metadata.countryToMetadata};i18n.phonenumbers.PhoneNumberUtil.prototype.hasValidCountryCallingCode_=function(a){return a in i18n.phonenumbers.metadata.countryCodeToRegionCodeMap};
|
|
|
i18n.phonenumbers.PhoneNumberUtil.prototype.format=function(a,b){if(0==a.getNationalNumber()&&a.hasRawInput()){var c=a.getRawInputOrDefault();if(0<c.length)return c}var c=a.getCountryCodeOrDefault(),d=this.getNationalSignificantNumber(a);if(b==i18n.phonenumbers.PhoneNumberFormat.E164)return this.prefixNumberWithCountryCallingCode_(c,i18n.phonenumbers.PhoneNumberFormat.E164,d,"");if(!this.hasValidCountryCallingCode_(c))return d;var e=this.getRegionCodeForCountryCode(c),f=this.getMetadataForRegionOrCallingCode_(c,
|
|
|
e),e=this.maybeGetFormattedExtension_(a,f,b),d=this.formatNsn_(d,f,b);return this.prefixNumberWithCountryCallingCode_(c,b,d,e)};
|
|
|
i18n.phonenumbers.PhoneNumberUtil.prototype.formatByPattern=function(a,b,c){var d=a.getCountryCodeOrDefault(),e=this.getNationalSignificantNumber(a);if(!this.hasValidCountryCallingCode_(d))return e;var f=this.getRegionCodeForCountryCode(d),f=this.getMetadataForRegionOrCallingCode_(d,f),g="",g=this.chooseFormattingPatternForNumber_(c,e);if(null==g)g=e;else{c=g.clone();g=g.getNationalPrefixFormattingRuleOrDefault();if(0<g.length){var h=f.getNationalPrefixOrDefault();0<h.length?(g=g.replace(i18n.phonenumbers.PhoneNumberUtil.NP_PATTERN_,
|
|
|
h).replace(i18n.phonenumbers.PhoneNumberUtil.FG_PATTERN_,"$1"),c.setNationalPrefixFormattingRule(g)):c.clearNationalPrefixFormattingRule()}g=this.formatNsnUsingPattern_(e,c,b)}a=this.maybeGetFormattedExtension_(a,f,b);return this.prefixNumberWithCountryCallingCode_(d,b,g,a)};
|
|
|
i18n.phonenumbers.PhoneNumberUtil.prototype.formatNationalNumberWithCarrierCode=function(a,b){var c=a.getCountryCodeOrDefault(),d=this.getNationalSignificantNumber(a);if(!this.hasValidCountryCallingCode_(c))return d;var e=this.getRegionCodeForCountryCode(c),f=this.getMetadataForRegionOrCallingCode_(c,e),e=this.maybeGetFormattedExtension_(a,f,i18n.phonenumbers.PhoneNumberFormat.NATIONAL),d=this.formatNsn_(d,f,i18n.phonenumbers.PhoneNumberFormat.NATIONAL,b);return this.prefixNumberWithCountryCallingCode_(c,
|
|
|
i18n.phonenumbers.PhoneNumberFormat.NATIONAL,d,e)};i18n.phonenumbers.PhoneNumberUtil.prototype.getMetadataForRegionOrCallingCode_=function(a,b){return i18n.phonenumbers.PhoneNumberUtil.REGION_CODE_FOR_NON_GEO_ENTITY==b?this.getMetadataForNonGeographicalRegion(a):this.getMetadataForRegion(b)};
|
|
|
i18n.phonenumbers.PhoneNumberUtil.prototype.formatNationalNumberWithPreferredCarrierCode=function(a,b){return this.formatNationalNumberWithCarrierCode(a,a.hasPreferredDomesticCarrierCode()?a.getPreferredDomesticCarrierCodeOrDefault():b)};
|
|
|
i18n.phonenumbers.PhoneNumberUtil.prototype.formatNumberForMobileDialing=function(a,b,c){var d=a.getCountryCodeOrDefault();if(!this.hasValidCountryCallingCode_(d))return a.hasRawInput()?a.getRawInputOrDefault():"";var e="";a=a.clone();a.clearExtension();var f=this.getRegionCodeForCountryCode(d),g=this.getNumberType(a),h=g!=i18n.phonenumbers.PhoneNumberType.UNKNOWN;if(b==f)e=g==i18n.phonenumbers.PhoneNumberType.FIXED_LINE||g==i18n.phonenumbers.PhoneNumberType.MOBILE||g==i18n.phonenumbers.PhoneNumberType.FIXED_LINE_OR_MOBILE,
|
|
|
"CO"==f&&g==i18n.phonenumbers.PhoneNumberType.FIXED_LINE?e=this.formatNationalNumberWithCarrierCode(a,i18n.phonenumbers.PhoneNumberUtil.COLOMBIA_MOBILE_TO_FIXED_LINE_PREFIX_):"BR"==f&&e?e=a.hasPreferredDomesticCarrierCode()?this.formatNationalNumberWithPreferredCarrierCode(a,""):"":h&&"HU"==f?e=this.getNddPrefixForRegion(f,!0)+" "+this.format(a,i18n.phonenumbers.PhoneNumberFormat.NATIONAL):d==i18n.phonenumbers.PhoneNumberUtil.NANPA_COUNTRY_CODE_?(b=this.getMetadataForRegion(b),e=this.canBeInternationallyDialled(a)&&
|
|
|
!this.isShorterThanPossibleNormalNumber_(b,this.getNationalSignificantNumber(a))?this.format(a,i18n.phonenumbers.PhoneNumberFormat.INTERNATIONAL):this.format(a,i18n.phonenumbers.PhoneNumberFormat.NATIONAL)):e=(f==i18n.phonenumbers.PhoneNumberUtil.REGION_CODE_FOR_NON_GEO_ENTITY||("MX"==f||"CL"==f)&&e)&&this.canBeInternationallyDialled(a)?this.format(a,i18n.phonenumbers.PhoneNumberFormat.INTERNATIONAL):this.format(a,i18n.phonenumbers.PhoneNumberFormat.NATIONAL);else if(h&&this.canBeInternationallyDialled(a))return c?
|
|
|
this.format(a,i18n.phonenumbers.PhoneNumberFormat.INTERNATIONAL):this.format(a,i18n.phonenumbers.PhoneNumberFormat.E164);return c?e:i18n.phonenumbers.PhoneNumberUtil.normalizeHelper_(e,i18n.phonenumbers.PhoneNumberUtil.DIALLABLE_CHAR_MAPPINGS_,!0)};
|
|
|
i18n.phonenumbers.PhoneNumberUtil.prototype.formatOutOfCountryCallingNumber=function(a,b){if(!this.isValidRegionCode_(b))return this.format(a,i18n.phonenumbers.PhoneNumberFormat.INTERNATIONAL);var c=a.getCountryCodeOrDefault(),d=this.getNationalSignificantNumber(a);if(!this.hasValidCountryCallingCode_(c))return d;if(c==i18n.phonenumbers.PhoneNumberUtil.NANPA_COUNTRY_CODE_){if(this.isNANPACountry(b))return c+" "+this.format(a,i18n.phonenumbers.PhoneNumberFormat.NATIONAL)}else if(c==this.getCountryCodeForValidRegion_(b))return this.format(a,
|
|
|
i18n.phonenumbers.PhoneNumberFormat.NATIONAL);var e=this.getMetadataForRegion(b),f=e.getInternationalPrefixOrDefault(),g="";i18n.phonenumbers.PhoneNumberUtil.matchesEntirely_(i18n.phonenumbers.PhoneNumberUtil.UNIQUE_INTERNATIONAL_PREFIX_,f)?g=f:e.hasPreferredInternationalPrefix()&&(g=e.getPreferredInternationalPrefixOrDefault());e=this.getRegionCodeForCountryCode(c);e=this.getMetadataForRegionOrCallingCode_(c,e);d=this.formatNsn_(d,e,i18n.phonenumbers.PhoneNumberFormat.INTERNATIONAL);e=this.maybeGetFormattedExtension_(a,
|
|
|
e,i18n.phonenumbers.PhoneNumberFormat.INTERNATIONAL);return 0<g.length?g+" "+c+" "+d+e:this.prefixNumberWithCountryCallingCode_(c,i18n.phonenumbers.PhoneNumberFormat.INTERNATIONAL,d,e)};
|
|
|
i18n.phonenumbers.PhoneNumberUtil.prototype.formatInOriginalFormat=function(a,b){if(a.hasRawInput()&&(this.hasUnexpectedItalianLeadingZero_(a)||!this.hasFormattingPatternForNumber_(a)))return a.getRawInputOrDefault();if(!a.hasCountryCodeSource())return this.format(a,i18n.phonenumbers.PhoneNumberFormat.NATIONAL);var c;switch(a.getCountryCodeSource()){case i18n.phonenumbers.PhoneNumber.CountryCodeSource.FROM_NUMBER_WITH_PLUS_SIGN:c=this.format(a,i18n.phonenumbers.PhoneNumberFormat.INTERNATIONAL);break;
|
|
|
case i18n.phonenumbers.PhoneNumber.CountryCodeSource.FROM_NUMBER_WITH_IDD:c=this.formatOutOfCountryCallingNumber(a,b);break;case i18n.phonenumbers.PhoneNumber.CountryCodeSource.FROM_NUMBER_WITHOUT_PLUS_SIGN:c=this.format(a,i18n.phonenumbers.PhoneNumberFormat.INTERNATIONAL).substring(1);break;default:var d=this.getRegionCodeForCountryCode(a.getCountryCodeOrDefault()),e=this.getNddPrefixForRegion(d,!0);c=this.format(a,i18n.phonenumbers.PhoneNumberFormat.NATIONAL);if(null==e||0==e.length)break;if(this.rawInputContainsNationalPrefix_(a.getRawInputOrDefault(),
|
|
|
e,d))break;d=this.getMetadataForRegion(d);e=this.getNationalSignificantNumber(a);d=this.chooseFormattingPatternForNumber_(d.numberFormatArray(),e);if(null==d)break;var e=d.getNationalPrefixFormattingRuleOrDefault(),f=e.indexOf("$1");if(0>=f)break;e=e.substring(0,f);e=i18n.phonenumbers.PhoneNumberUtil.normalizeDigitsOnly(e);if(0==e.length)break;c=d.clone();c.clearNationalPrefixFormattingRule();c=this.formatByPattern(a,i18n.phonenumbers.PhoneNumberFormat.NATIONAL,[c])}d=a.getRawInputOrDefault();null!=
|
|
|
c&&0<d.length&&(e=i18n.phonenumbers.PhoneNumberUtil.normalizeHelper_(c,i18n.phonenumbers.PhoneNumberUtil.DIALLABLE_CHAR_MAPPINGS_,!0),f=i18n.phonenumbers.PhoneNumberUtil.normalizeHelper_(d,i18n.phonenumbers.PhoneNumberUtil.DIALLABLE_CHAR_MAPPINGS_,!0),e!=f&&(c=d));return c};
|
|
|
i18n.phonenumbers.PhoneNumberUtil.prototype.rawInputContainsNationalPrefix_=function(a,b,c){a=i18n.phonenumbers.PhoneNumberUtil.normalizeDigitsOnly(a);if(goog.string.startsWith(a,b))try{return this.isValidNumber(this.parse(a.substring(b.length),c))}catch(d){}return!1};i18n.phonenumbers.PhoneNumberUtil.prototype.hasUnexpectedItalianLeadingZero_=function(a){return a.hasItalianLeadingZero()&&!this.isLeadingZeroPossible(a.getCountryCodeOrDefault())};
|
|
|
i18n.phonenumbers.PhoneNumberUtil.prototype.hasFormattingPatternForNumber_=function(a){var b=a.getCountryCodeOrDefault(),c=this.getRegionCodeForCountryCode(b),b=this.getMetadataForRegionOrCallingCode_(b,c);if(null==b)return!1;a=this.getNationalSignificantNumber(a);return null!=this.chooseFormattingPatternForNumber_(b.numberFormatArray(),a)};
|
|
|
i18n.phonenumbers.PhoneNumberUtil.prototype.formatOutOfCountryKeepingAlphaChars=function(a,b){var c=a.getRawInputOrDefault();if(0==c.length)return this.formatOutOfCountryCallingNumber(a,b);var d=a.getCountryCodeOrDefault();if(!this.hasValidCountryCallingCode_(d))return c;var c=i18n.phonenumbers.PhoneNumberUtil.normalizeHelper_(c,i18n.phonenumbers.PhoneNumberUtil.ALL_PLUS_NUMBER_GROUPING_SYMBOLS_,!0),e=this.getNationalSignificantNumber(a);if(3<e.length){var f=c.indexOf(e.substring(0,3));-1!=f&&(c=
|
|
|
c.substring(f))}f=this.getMetadataForRegion(b);if(d==i18n.phonenumbers.PhoneNumberUtil.NANPA_COUNTRY_CODE_){if(this.isNANPACountry(b))return d+" "+c}else if(null!=f&&d==this.getCountryCodeForValidRegion_(b)){d=this.chooseFormattingPatternForNumber_(f.numberFormatArray(),e);if(null==d)return c;d=d.clone();d.setPattern("(\\d+)(.*)");d.setFormat("$1$2");return this.formatNsnUsingPattern_(c,d,i18n.phonenumbers.PhoneNumberFormat.NATIONAL)}e="";null!=f&&(e=f.getInternationalPrefixOrDefault(),e=i18n.phonenumbers.PhoneNumberUtil.matchesEntirely_(i18n.phonenumbers.PhoneNumberUtil.UNIQUE_INTERNATIONAL_PREFIX_,
|
|
|
e)?e:f.getPreferredInternationalPrefixOrDefault());f=this.getRegionCodeForCountryCode(d);f=this.getMetadataForRegionOrCallingCode_(d,f);f=this.maybeGetFormattedExtension_(a,f,i18n.phonenumbers.PhoneNumberFormat.INTERNATIONAL);return 0<e.length?e+" "+d+" "+c+f:this.prefixNumberWithCountryCallingCode_(d,i18n.phonenumbers.PhoneNumberFormat.INTERNATIONAL,c,f)};
|
|
|
i18n.phonenumbers.PhoneNumberUtil.prototype.getNationalSignificantNumber=function(a){var b=""+a.getNationalNumber();return a.hasItalianLeadingZero()&&a.getItalianLeadingZero()?Array(a.getNumberOfLeadingZerosOrDefault()+1).join("0")+b:b};
|
|
|
i18n.phonenumbers.PhoneNumberUtil.prototype.prefixNumberWithCountryCallingCode_=function(a,b,c,d){switch(b){case i18n.phonenumbers.PhoneNumberFormat.E164:return i18n.phonenumbers.PhoneNumberUtil.PLUS_SIGN+a+c+d;case i18n.phonenumbers.PhoneNumberFormat.INTERNATIONAL:return i18n.phonenumbers.PhoneNumberUtil.PLUS_SIGN+a+" "+c+d;case i18n.phonenumbers.PhoneNumberFormat.RFC3966:return i18n.phonenumbers.PhoneNumberUtil.RFC3966_PREFIX_+i18n.phonenumbers.PhoneNumberUtil.PLUS_SIGN+a+"-"+c+d;default:return c+
|
|
|
d}};i18n.phonenumbers.PhoneNumberUtil.prototype.formatNsn_=function(a,b,c,d){b=0==b.intlNumberFormatArray().length||c==i18n.phonenumbers.PhoneNumberFormat.NATIONAL?b.numberFormatArray():b.intlNumberFormatArray();b=this.chooseFormattingPatternForNumber_(b,a);return null==b?a:this.formatNsnUsingPattern_(a,b,c,d)};
|
|
|
i18n.phonenumbers.PhoneNumberUtil.prototype.chooseFormattingPatternForNumber_=function(a,b){for(var c,d=a.length,e=0;e<d;++e){c=a[e];var f=c.leadingDigitsPatternCount();if(0==f||0==b.search(c.getLeadingDigitsPattern(f-1)))if(f=new RegExp(c.getPattern()),i18n.phonenumbers.PhoneNumberUtil.matchesEntirely_(f,b))return c}return null};
|
|
|
i18n.phonenumbers.PhoneNumberUtil.prototype.formatNsnUsingPattern_=function(a,b,c,d){var e=b.getFormatOrDefault(),f=new RegExp(b.getPattern()),g=b.getDomesticCarrierCodeFormattingRuleOrDefault(),h="";c==i18n.phonenumbers.PhoneNumberFormat.NATIONAL&&null!=d&&0<d.length&&0<g.length?(b=g.replace(i18n.phonenumbers.PhoneNumberUtil.CC_PATTERN_,d),e=e.replace(i18n.phonenumbers.PhoneNumberUtil.FIRST_GROUP_PATTERN_,b),h=a.replace(f,e)):(b=b.getNationalPrefixFormattingRuleOrDefault(),h=c==i18n.phonenumbers.PhoneNumberFormat.NATIONAL&&
|
|
|
null!=b&&0<b.length?a.replace(f,e.replace(i18n.phonenumbers.PhoneNumberUtil.FIRST_GROUP_PATTERN_,b)):a.replace(f,e));c==i18n.phonenumbers.PhoneNumberFormat.RFC3966&&(h=h.replace(new RegExp("^"+i18n.phonenumbers.PhoneNumberUtil.SEPARATOR_PATTERN_),""),h=h.replace(new RegExp(i18n.phonenumbers.PhoneNumberUtil.SEPARATOR_PATTERN_,"g"),"-"));return h};i18n.phonenumbers.PhoneNumberUtil.prototype.getExampleNumber=function(a){return this.getExampleNumberForType(a,i18n.phonenumbers.PhoneNumberType.FIXED_LINE)};
|
|
|
i18n.phonenumbers.PhoneNumberUtil.prototype.getExampleNumberForType=function(a,b){if(!this.isValidRegionCode_(a))return null;var c=this.getNumberDescByType_(this.getMetadataForRegion(a),b);try{if(c.hasExampleNumber())return this.parse(c.getExampleNumberOrDefault(),a)}catch(d){}return null};
|
|
|
i18n.phonenumbers.PhoneNumberUtil.prototype.getExampleNumberForNonGeoEntity=function(a){var b=this.getMetadataForNonGeographicalRegion(a);if(null!=b){b=b.getGeneralDesc();try{if(b.hasExampleNumber())return this.parse("+"+a+b.getExampleNumber(),"ZZ")}catch(c){}}return null};
|
|
|
i18n.phonenumbers.PhoneNumberUtil.prototype.maybeGetFormattedExtension_=function(a,b,c){return a.hasExtension()&&0!=a.getExtension().length?c==i18n.phonenumbers.PhoneNumberFormat.RFC3966?i18n.phonenumbers.PhoneNumberUtil.RFC3966_EXTN_PREFIX_+a.getExtension():b.hasPreferredExtnPrefix()?b.getPreferredExtnPrefix()+a.getExtensionOrDefault():i18n.phonenumbers.PhoneNumberUtil.DEFAULT_EXTN_PREFIX_+a.getExtensionOrDefault():""};
|
|
|
i18n.phonenumbers.PhoneNumberUtil.prototype.getNumberDescByType_=function(a,b){switch(b){case i18n.phonenumbers.PhoneNumberType.PREMIUM_RATE:return a.getPremiumRate();case i18n.phonenumbers.PhoneNumberType.TOLL_FREE:return a.getTollFree();case i18n.phonenumbers.PhoneNumberType.MOBILE:return a.getMobile();case i18n.phonenumbers.PhoneNumberType.FIXED_LINE:case i18n.phonenumbers.PhoneNumberType.FIXED_LINE_OR_MOBILE:return a.getFixedLine();case i18n.phonenumbers.PhoneNumberType.SHARED_COST:return a.getSharedCost();
|
|
|
case i18n.phonenumbers.PhoneNumberType.VOIP:return a.getVoip();case i18n.phonenumbers.PhoneNumberType.PERSONAL_NUMBER:return a.getPersonalNumber();case i18n.phonenumbers.PhoneNumberType.PAGER:return a.getPager();case i18n.phonenumbers.PhoneNumberType.UAN:return a.getUan();case i18n.phonenumbers.PhoneNumberType.VOICEMAIL:return a.getVoicemail();default:return a.getGeneralDesc()}};
|
|
|
i18n.phonenumbers.PhoneNumberUtil.prototype.getNumberType=function(a){var b=this.getRegionCodeForNumber(a),b=this.getMetadataForRegionOrCallingCode_(a.getCountryCodeOrDefault(),b);if(null==b)return i18n.phonenumbers.PhoneNumberType.UNKNOWN;a=this.getNationalSignificantNumber(a);return this.getNumberTypeHelper_(a,b)};
|
|
|
i18n.phonenumbers.PhoneNumberUtil.prototype.getNumberTypeHelper_=function(a,b){return this.isNumberMatchingDesc_(a,b.getGeneralDesc())?this.isNumberMatchingDesc_(a,b.getPremiumRate())?i18n.phonenumbers.PhoneNumberType.PREMIUM_RATE:this.isNumberMatchingDesc_(a,b.getTollFree())?i18n.phonenumbers.PhoneNumberType.TOLL_FREE:this.isNumberMatchingDesc_(a,b.getSharedCost())?i18n.phonenumbers.PhoneNumberType.SHARED_COST:this.isNumberMatchingDesc_(a,b.getVoip())?i18n.phonenumbers.PhoneNumberType.VOIP:this.isNumberMatchingDesc_(a,
|
|
|
b.getPersonalNumber())?i18n.phonenumbers.PhoneNumberType.PERSONAL_NUMBER:this.isNumberMatchingDesc_(a,b.getPager())?i18n.phonenumbers.PhoneNumberType.PAGER:this.isNumberMatchingDesc_(a,b.getUan())?i18n.phonenumbers.PhoneNumberType.UAN:this.isNumberMatchingDesc_(a,b.getVoicemail())?i18n.phonenumbers.PhoneNumberType.VOICEMAIL:this.isNumberMatchingDesc_(a,b.getFixedLine())?b.getSameMobileAndFixedLinePattern()||this.isNumberMatchingDesc_(a,b.getMobile())?i18n.phonenumbers.PhoneNumberType.FIXED_LINE_OR_MOBILE:
|
|
|
i18n.phonenumbers.PhoneNumberType.FIXED_LINE:!b.getSameMobileAndFixedLinePattern()&&this.isNumberMatchingDesc_(a,b.getMobile())?i18n.phonenumbers.PhoneNumberType.MOBILE:i18n.phonenumbers.PhoneNumberType.UNKNOWN:i18n.phonenumbers.PhoneNumberType.UNKNOWN};
|
|
|
i18n.phonenumbers.PhoneNumberUtil.prototype.getMetadataForRegion=function(a){if(null==a)return null;a=a.toUpperCase();var b=this.regionToMetadataMap[a];if(null==b){var b=new goog.proto2.PbLiteSerializer,c=i18n.phonenumbers.metadata.countryToMetadata[a];if(null==c)return null;b=b.deserialize(i18n.phonenumbers.PhoneMetadata.getDescriptor(),c);this.regionToMetadataMap[a]=b}return b};
|
|
|
i18n.phonenumbers.PhoneNumberUtil.prototype.getMetadataForNonGeographicalRegion=function(a){return this.getMetadataForRegion(""+a)};i18n.phonenumbers.PhoneNumberUtil.prototype.isNumberMatchingDesc_=function(a,b){return i18n.phonenumbers.PhoneNumberUtil.matchesEntirely_(b.getPossibleNumberPatternOrDefault(),a)&&i18n.phonenumbers.PhoneNumberUtil.matchesEntirely_(b.getNationalNumberPatternOrDefault(),a)};
|
|
|
i18n.phonenumbers.PhoneNumberUtil.prototype.isValidNumber=function(a){var b=this.getRegionCodeForNumber(a);return this.isValidNumberForRegion(a,b)};
|
|
|
i18n.phonenumbers.PhoneNumberUtil.prototype.isValidNumberForRegion=function(a,b){var c=a.getCountryCodeOrDefault(),d=this.getMetadataForRegionOrCallingCode_(c,b);if(null==d||i18n.phonenumbers.PhoneNumberUtil.REGION_CODE_FOR_NON_GEO_ENTITY!=b&&c!=this.getCountryCodeForValidRegion_(b))return!1;c=this.getNationalSignificantNumber(a);return this.getNumberTypeHelper_(c,d)!=i18n.phonenumbers.PhoneNumberType.UNKNOWN};
|
|
|
i18n.phonenumbers.PhoneNumberUtil.prototype.getRegionCodeForNumber=function(a){if(null==a)return null;var b=a.getCountryCodeOrDefault(),b=i18n.phonenumbers.metadata.countryCodeToRegionCodeMap[b];return null==b?null:1==b.length?b[0]:this.getRegionCodeForNumberFromRegionList_(a,b)};
|
|
|
i18n.phonenumbers.PhoneNumberUtil.prototype.getRegionCodeForNumberFromRegionList_=function(a,b){for(var c=this.getNationalSignificantNumber(a),d,e=b.length,f=0;f<e;f++){d=b[f];var g=this.getMetadataForRegion(d);if(g.hasLeadingDigits()){if(0==c.search(g.getLeadingDigits()))return d}else if(this.getNumberTypeHelper_(c,g)!=i18n.phonenumbers.PhoneNumberType.UNKNOWN)return d}return null};
|
|
|
i18n.phonenumbers.PhoneNumberUtil.prototype.getRegionCodeForCountryCode=function(a){a=i18n.phonenumbers.metadata.countryCodeToRegionCodeMap[a];return null==a?i18n.phonenumbers.PhoneNumberUtil.UNKNOWN_REGION_:a[0]};i18n.phonenumbers.PhoneNumberUtil.prototype.getRegionCodesForCountryCode=function(a){a=i18n.phonenumbers.metadata.countryCodeToRegionCodeMap[a];return null==a?[]:a};
|
|
|
i18n.phonenumbers.PhoneNumberUtil.prototype.getCountryCodeForRegion=function(a){return this.isValidRegionCode_(a)?this.getCountryCodeForValidRegion_(a):0};i18n.phonenumbers.PhoneNumberUtil.prototype.getCountryCodeForValidRegion_=function(a){var b=this.getMetadataForRegion(a);if(null==b)throw"Invalid region code: "+a;return b.getCountryCodeOrDefault()};
|
|
|
i18n.phonenumbers.PhoneNumberUtil.prototype.getNddPrefixForRegion=function(a,b){var c=this.getMetadataForRegion(a);if(null==c)return null;c=c.getNationalPrefixOrDefault();if(0==c.length)return null;b&&(c=c.replace("~",""));return c};i18n.phonenumbers.PhoneNumberUtil.prototype.isNANPACountry=function(a){return null!=a&&goog.array.contains(i18n.phonenumbers.metadata.countryCodeToRegionCodeMap[i18n.phonenumbers.PhoneNumberUtil.NANPA_COUNTRY_CODE_],a.toUpperCase())};
|
|
|
i18n.phonenumbers.PhoneNumberUtil.prototype.isLeadingZeroPossible=function(a){a=this.getMetadataForRegionOrCallingCode_(a,this.getRegionCodeForCountryCode(a));return null!=a&&a.getLeadingZeroPossibleOrDefault()};
|
|
|
i18n.phonenumbers.PhoneNumberUtil.prototype.isAlphaNumber=function(a){if(!i18n.phonenumbers.PhoneNumberUtil.isViablePhoneNumber(a))return!1;a=new goog.string.StringBuffer(a);this.maybeStripExtension(a);return i18n.phonenumbers.PhoneNumberUtil.matchesEntirely_(i18n.phonenumbers.PhoneNumberUtil.VALID_ALPHA_PHONE_PATTERN_,a.toString())};i18n.phonenumbers.PhoneNumberUtil.prototype.isPossibleNumber=function(a){return this.isPossibleNumberWithReason(a)==i18n.phonenumbers.PhoneNumberUtil.ValidationResult.IS_POSSIBLE};
|
|
|
i18n.phonenumbers.PhoneNumberUtil.prototype.testNumberLengthAgainstPattern_=function(a,b){return i18n.phonenumbers.PhoneNumberUtil.matchesEntirely_(a,b)?i18n.phonenumbers.PhoneNumberUtil.ValidationResult.IS_POSSIBLE:0==b.search(a)?i18n.phonenumbers.PhoneNumberUtil.ValidationResult.TOO_LONG:i18n.phonenumbers.PhoneNumberUtil.ValidationResult.TOO_SHORT};
|
|
|
i18n.phonenumbers.PhoneNumberUtil.prototype.isShorterThanPossibleNormalNumber_=function(a,b){var c=a.getGeneralDesc().getPossibleNumberPatternOrDefault();return this.testNumberLengthAgainstPattern_(c,b)==i18n.phonenumbers.PhoneNumberUtil.ValidationResult.TOO_SHORT};
|
|
|
i18n.phonenumbers.PhoneNumberUtil.prototype.isPossibleNumberWithReason=function(a){var b=this.getNationalSignificantNumber(a);a=a.getCountryCodeOrDefault();if(!this.hasValidCountryCallingCode_(a))return i18n.phonenumbers.PhoneNumberUtil.ValidationResult.INVALID_COUNTRY_CODE;var c=this.getRegionCodeForCountryCode(a);a=this.getMetadataForRegionOrCallingCode_(a,c).getGeneralDesc().getPossibleNumberPatternOrDefault();return this.testNumberLengthAgainstPattern_(a,b)};
|
|
|
i18n.phonenumbers.PhoneNumberUtil.prototype.isPossibleNumberString=function(a,b){try{return this.isPossibleNumber(this.parse(a,b))}catch(c){return!1}};
|
|
|
i18n.phonenumbers.PhoneNumberUtil.prototype.truncateTooLongNumber=function(a){if(this.isValidNumber(a))return!0;var b=a.clone(),c=a.getNationalNumberOrDefault();do if(c=Math.floor(c/10),b.setNationalNumber(c),0==c||this.isPossibleNumberWithReason(b)==i18n.phonenumbers.PhoneNumberUtil.ValidationResult.TOO_SHORT)return!1;while(!this.isValidNumber(b));a.setNationalNumber(c);return!0};
|
|
|
i18n.phonenumbers.PhoneNumberUtil.prototype.extractCountryCode=function(a,b){var c=a.toString();if(0==c.length||"0"==c.charAt(0))return 0;for(var d,e=c.length,f=1;f<=i18n.phonenumbers.PhoneNumberUtil.MAX_LENGTH_COUNTRY_CODE_&&f<=e;++f)if(d=parseInt(c.substring(0,f),10),d in i18n.phonenumbers.metadata.countryCodeToRegionCodeMap)return b.append(c.substring(f)),d;return 0};
|
|
|
i18n.phonenumbers.PhoneNumberUtil.prototype.maybeExtractCountryCode=function(a,b,c,d,e){if(0==a.length)return 0;a=new goog.string.StringBuffer(a);var f;null!=b&&(f=b.getInternationalPrefix());null==f&&(f="NonMatch");f=this.maybeStripInternationalPrefixAndNormalize(a,f);d&&e.setCountryCodeSource(f);if(f!=i18n.phonenumbers.PhoneNumber.CountryCodeSource.FROM_DEFAULT_COUNTRY){if(a.getLength()<=i18n.phonenumbers.PhoneNumberUtil.MIN_LENGTH_FOR_NSN_)throw i18n.phonenumbers.Error.TOO_SHORT_AFTER_IDD;c=this.extractCountryCode(a,
|
|
|
c);if(0!=c)return e.setCountryCode(c),c;throw i18n.phonenumbers.Error.INVALID_COUNTRY_CODE;}if(null!=b){f=b.getCountryCodeOrDefault();var g=""+f,h=a.toString();if(goog.string.startsWith(h,g)){var k=new goog.string.StringBuffer(h.substring(g.length)),h=b.getGeneralDesc(),g=new RegExp(h.getNationalNumberPatternOrDefault());this.maybeStripNationalPrefixAndCarrierCode(k,b,null);b=k.toString();h=h.getPossibleNumberPatternOrDefault();if(!i18n.phonenumbers.PhoneNumberUtil.matchesEntirely_(g,a.toString())&&
|
|
|
i18n.phonenumbers.PhoneNumberUtil.matchesEntirely_(g,b)||this.testNumberLengthAgainstPattern_(h,a.toString())==i18n.phonenumbers.PhoneNumberUtil.ValidationResult.TOO_LONG)return c.append(b),d&&e.setCountryCodeSource(i18n.phonenumbers.PhoneNumber.CountryCodeSource.FROM_NUMBER_WITHOUT_PLUS_SIGN),e.setCountryCode(f),f}}e.setCountryCode(0);return 0};
|
|
|
i18n.phonenumbers.PhoneNumberUtil.prototype.parsePrefixAsIdd_=function(a,b){var c=b.toString();if(0==c.search(a)){var d=c.match(a)[0].length,e=c.substring(d).match(i18n.phonenumbers.PhoneNumberUtil.CAPTURING_DIGIT_PATTERN);if(e&&null!=e[1]&&0<e[1].length&&"0"==i18n.phonenumbers.PhoneNumberUtil.normalizeDigitsOnly(e[1]))return!1;b.clear();b.append(c.substring(d));return!0}return!1};
|
|
|
i18n.phonenumbers.PhoneNumberUtil.prototype.maybeStripInternationalPrefixAndNormalize=function(a,b){var c=a.toString();if(0==c.length)return i18n.phonenumbers.PhoneNumber.CountryCodeSource.FROM_DEFAULT_COUNTRY;if(i18n.phonenumbers.PhoneNumberUtil.LEADING_PLUS_CHARS_PATTERN_.test(c))return c=c.replace(i18n.phonenumbers.PhoneNumberUtil.LEADING_PLUS_CHARS_PATTERN_,""),a.clear(),a.append(i18n.phonenumbers.PhoneNumberUtil.normalize(c)),i18n.phonenumbers.PhoneNumber.CountryCodeSource.FROM_NUMBER_WITH_PLUS_SIGN;
|
|
|
c=new RegExp(b);i18n.phonenumbers.PhoneNumberUtil.normalizeSB_(a);return this.parsePrefixAsIdd_(c,a)?i18n.phonenumbers.PhoneNumber.CountryCodeSource.FROM_NUMBER_WITH_IDD:i18n.phonenumbers.PhoneNumber.CountryCodeSource.FROM_DEFAULT_COUNTRY};
|
|
|
i18n.phonenumbers.PhoneNumberUtil.prototype.maybeStripNationalPrefixAndCarrierCode=function(a,b,c){var d=a.toString(),e=d.length,f=b.getNationalPrefixForParsing();if(0==e||null==f||0==f.length)return!1;var g=new RegExp("^(?:"+f+")");if(e=g.exec(d)){var f=new RegExp(b.getGeneralDesc().getNationalNumberPatternOrDefault()),h=i18n.phonenumbers.PhoneNumberUtil.matchesEntirely_(f,d),k=e.length-1;b=b.getNationalPrefixTransformRule();if(null==b||0==b.length||null==e[k]||0==e[k].length){if(h&&!i18n.phonenumbers.PhoneNumberUtil.matchesEntirely_(f,
|
|
|
d.substring(e[0].length)))return!1;null!=c&&0<k&&null!=e[k]&&c.append(e[1]);a.set(d.substring(e[0].length))}else{d=d.replace(g,b);if(h&&!i18n.phonenumbers.PhoneNumberUtil.matchesEntirely_(f,d))return!1;null!=c&&0<k&&c.append(e[1]);a.set(d)}return!0}return!1};
|
|
|
i18n.phonenumbers.PhoneNumberUtil.prototype.maybeStripExtension=function(a){var b=a.toString(),c=b.search(i18n.phonenumbers.PhoneNumberUtil.EXTN_PATTERN_);if(0<=c&&i18n.phonenumbers.PhoneNumberUtil.isViablePhoneNumber(b.substring(0,c)))for(var d=b.match(i18n.phonenumbers.PhoneNumberUtil.EXTN_PATTERN_),e=d.length,f=1;f<e;++f)if(null!=d[f]&&0<d[f].length)return a.clear(),a.append(b.substring(0,c)),d[f];return""};
|
|
|
i18n.phonenumbers.PhoneNumberUtil.prototype.checkRegionForParsing_=function(a,b){return this.isValidRegionCode_(b)||null!=a&&0<a.length&&i18n.phonenumbers.PhoneNumberUtil.LEADING_PLUS_CHARS_PATTERN_.test(a)};i18n.phonenumbers.PhoneNumberUtil.prototype.parse=function(a,b){return this.parseHelper_(a,b,!1,!0)};
|
|
|
i18n.phonenumbers.PhoneNumberUtil.prototype.parseAndKeepRawInput=function(a,b){if(!this.isValidRegionCode_(b)&&0<a.length&&a.charAt(0)!=i18n.phonenumbers.PhoneNumberUtil.PLUS_SIGN)throw i18n.phonenumbers.Error.INVALID_COUNTRY_CODE;return this.parseHelper_(a,b,!0,!0)};i18n.phonenumbers.PhoneNumberUtil.prototype.setItalianLeadingZerosForPhoneNumber_=function(a,b){if(1<a.length&&"0"==a.charAt(0)){b.setItalianLeadingZero(!0);for(var c=1;c<a.length-1&&"0"==a.charAt(c);)c++;1!=c&&b.setNumberOfLeadingZeros(c)}};
|
|
|
i18n.phonenumbers.PhoneNumberUtil.prototype.parseHelper_=function(a,b,c,d){if(null==a)throw i18n.phonenumbers.Error.NOT_A_NUMBER;if(a.length>i18n.phonenumbers.PhoneNumberUtil.MAX_INPUT_STRING_LENGTH_)throw i18n.phonenumbers.Error.TOO_LONG;var e=new goog.string.StringBuffer;this.buildNationalNumberForParsing_(a,e);if(!i18n.phonenumbers.PhoneNumberUtil.isViablePhoneNumber(e.toString()))throw i18n.phonenumbers.Error.NOT_A_NUMBER;if(d&&!this.checkRegionForParsing_(e.toString(),b))throw i18n.phonenumbers.Error.INVALID_COUNTRY_CODE;
|
|
|
d=new i18n.phonenumbers.PhoneNumber;c&&d.setRawInput(a);a=this.maybeStripExtension(e);0<a.length&&d.setExtension(a);a=this.getMetadataForRegion(b);var f=new goog.string.StringBuffer,g=0,h=e.toString();try{g=this.maybeExtractCountryCode(h,a,f,c,d)}catch(k){if(k==i18n.phonenumbers.Error.INVALID_COUNTRY_CODE&&i18n.phonenumbers.PhoneNumberUtil.LEADING_PLUS_CHARS_PATTERN_.test(h)){if(h=h.replace(i18n.phonenumbers.PhoneNumberUtil.LEADING_PLUS_CHARS_PATTERN_,""),g=this.maybeExtractCountryCode(h,a,f,c,d),
|
|
|
0==g)throw k;}else throw k;}0!=g?(e=this.getRegionCodeForCountryCode(g),e!=b&&(a=this.getMetadataForRegionOrCallingCode_(g,e))):(i18n.phonenumbers.PhoneNumberUtil.normalizeSB_(e),f.append(e.toString()),null!=b?(g=a.getCountryCodeOrDefault(),d.setCountryCode(g)):c&&d.clearCountryCodeSource());if(f.getLength()<i18n.phonenumbers.PhoneNumberUtil.MIN_LENGTH_FOR_NSN_)throw i18n.phonenumbers.Error.TOO_SHORT_NSN;null!=a&&(b=new goog.string.StringBuffer,e=new goog.string.StringBuffer(f.toString()),this.maybeStripNationalPrefixAndCarrierCode(e,
|
|
|
a,b),this.isShorterThanPossibleNormalNumber_(a,e.toString())||(f=e,c&&d.setPreferredDomesticCarrierCode(b.toString())));c=f.toString();b=c.length;if(b<i18n.phonenumbers.PhoneNumberUtil.MIN_LENGTH_FOR_NSN_)throw i18n.phonenumbers.Error.TOO_SHORT_NSN;if(b>i18n.phonenumbers.PhoneNumberUtil.MAX_LENGTH_FOR_NSN_)throw i18n.phonenumbers.Error.TOO_LONG;this.setItalianLeadingZerosForPhoneNumber_(c,d);d.setNationalNumber(parseInt(c,10));return d};
|
|
|
i18n.phonenumbers.PhoneNumberUtil.prototype.buildNationalNumberForParsing_=function(a,b){var c=a.indexOf(i18n.phonenumbers.PhoneNumberUtil.RFC3966_PHONE_CONTEXT_);if(0<c){var d=c+i18n.phonenumbers.PhoneNumberUtil.RFC3966_PHONE_CONTEXT_.length;if(a.charAt(d)==i18n.phonenumbers.PhoneNumberUtil.PLUS_SIGN){var e=a.indexOf(";",d);0<e?b.append(a.substring(d,e)):b.append(a.substring(d))}d=a.indexOf(i18n.phonenumbers.PhoneNumberUtil.RFC3966_PREFIX_);b.append(a.substring(0<=d?d+i18n.phonenumbers.PhoneNumberUtil.RFC3966_PREFIX_.length:
|
|
|
0,c))}else b.append(i18n.phonenumbers.PhoneNumberUtil.extractPossibleNumber(a));c=b.toString();d=c.indexOf(i18n.phonenumbers.PhoneNumberUtil.RFC3966_ISDN_SUBADDRESS_);0<d&&(b.clear(),b.append(c.substring(0,d)))};
|
|
|
i18n.phonenumbers.PhoneNumberUtil.prototype.isNumberMatch=function(a,b){var c,d;if("string"==typeof a)try{c=this.parse(a,i18n.phonenumbers.PhoneNumberUtil.UNKNOWN_REGION_)}catch(e){if(e!=i18n.phonenumbers.Error.INVALID_COUNTRY_CODE)return i18n.phonenumbers.PhoneNumberUtil.MatchType.NOT_A_NUMBER;if("string"!=typeof b){var f=this.getRegionCodeForCountryCode(b.getCountryCodeOrDefault());if(f!=i18n.phonenumbers.PhoneNumberUtil.UNKNOWN_REGION_){try{c=this.parse(a,f)}catch(g){return i18n.phonenumbers.PhoneNumberUtil.MatchType.NOT_A_NUMBER}c=
|
|
|
this.isNumberMatch(c,b);return c==i18n.phonenumbers.PhoneNumberUtil.MatchType.EXACT_MATCH?i18n.phonenumbers.PhoneNumberUtil.MatchType.NSN_MATCH:c}}try{c=this.parseHelper_(a,null,!1,!1)}catch(h){return i18n.phonenumbers.PhoneNumberUtil.MatchType.NOT_A_NUMBER}}else c=a.clone();if("string"==typeof b)try{return d=this.parse(b,i18n.phonenumbers.PhoneNumberUtil.UNKNOWN_REGION_),this.isNumberMatch(a,d)}catch(k){return k!=i18n.phonenumbers.Error.INVALID_COUNTRY_CODE?i18n.phonenumbers.PhoneNumberUtil.MatchType.NOT_A_NUMBER:
|
|
|
this.isNumberMatch(b,c)}else d=b.clone();c.clearRawInput();c.clearCountryCodeSource();c.clearPreferredDomesticCarrierCode();d.clearRawInput();d.clearCountryCodeSource();d.clearPreferredDomesticCarrierCode();c.hasExtension()&&0==c.getExtension().length&&c.clearExtension();d.hasExtension()&&0==d.getExtension().length&&d.clearExtension();if(c.hasExtension()&&d.hasExtension()&&c.getExtension()!=d.getExtension())return i18n.phonenumbers.PhoneNumberUtil.MatchType.NO_MATCH;var f=c.getCountryCodeOrDefault(),
|
|
|
l=d.getCountryCodeOrDefault();if(0!=f&&0!=l)return c.equals(d)?i18n.phonenumbers.PhoneNumberUtil.MatchType.EXACT_MATCH:f==l&&this.isNationalNumberSuffixOfTheOther_(c,d)?i18n.phonenumbers.PhoneNumberUtil.MatchType.SHORT_NSN_MATCH:i18n.phonenumbers.PhoneNumberUtil.MatchType.NO_MATCH;c.setCountryCode(0);d.setCountryCode(0);return c.equals(d)?i18n.phonenumbers.PhoneNumberUtil.MatchType.NSN_MATCH:this.isNationalNumberSuffixOfTheOther_(c,d)?i18n.phonenumbers.PhoneNumberUtil.MatchType.SHORT_NSN_MATCH:i18n.phonenumbers.PhoneNumberUtil.MatchType.NO_MATCH};
|
|
|
i18n.phonenumbers.PhoneNumberUtil.prototype.isNationalNumberSuffixOfTheOther_=function(a,b){var c=""+a.getNationalNumber(),d=""+b.getNationalNumber();return goog.string.endsWith(c,d)||goog.string.endsWith(d,c)};i18n.phonenumbers.PhoneNumberUtil.prototype.canBeInternationallyDialled=function(a){var b=this.getMetadataForRegion(this.getRegionCodeForNumber(a));if(null==b)return!0;a=this.getNationalSignificantNumber(a);return!this.isNumberMatchingDesc_(a,b.getNoInternationalDialling())};
|
|
|
i18n.phonenumbers.PhoneNumberUtil.matchesEntirely_=function(a,b){var c="string"==typeof a?b.match("^(?:"+a+")$"):b.match(a);return c&&c[0].length==b.length?!0:!1};/*
|
|
|
|
|
|
Copyright (C) 2014 codedust.
|
|
|
|
|
|
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.
|
|
|
*/
|
|
|
var libphonenumber=function(){var a={},b=i18n.phonenumbers.PhoneNumberUtil.getInstance();a.parse=function(a,d){try{return b.parse(a,d)}catch(e){throw e;}};a.getCountryCodeForRegion=function(a){return b.getCountryCodeForRegion(a)};a.getRegionCodeForCountryCode=function(a){return b.getRegionCodeForCountryCode(a)};a.getRegionCodeForNumber=function(a){return b.getRegionCodeForNumber(a)};a.isValidNumber=function(a){return b.isValidNumber(a)};a.isValidNumberForRegion=function(a,d){return b.isValidNumberForRegion(a,
|
|
|
d)};a.PhoneNumberFormat=i18n.phonenumbers.PhoneNumberFormat;a.format=function(a,d){return b.format(a,d)};return a}();goog.exportSymbol("libphonenumber.parse",libphonenumber.parse);goog.exportSymbol("libphonenumber.getCountryCodeForRegion",libphonenumber.getCountryCodeForRegion);goog.exportSymbol("libphonenumber.getRegionCodeForCountryCode",libphonenumber.getRegionCodeForCountryCode);goog.exportSymbol("libphonenumber.getRegionCodeForNumber",libphonenumber.getRegionCodeForNumber);
|
|
|
goog.exportSymbol("libphonenumber.isValidNumber",libphonenumber.isValidNumber);goog.exportSymbol("libphonenumber.isValidNumberForRegion",libphonenumber.isValidNumberForRegion);goog.exportSymbol("libphonenumber.PhoneNumberFormat",libphonenumber.PhoneNumberFormat);goog.exportSymbol("libphonenumber.PhoneNumberFormat.E164",libphonenumber.PhoneNumberFormat.E164);goog.exportSymbol("libphonenumber.PhoneNumberFormat.INTERNATIONAL",libphonenumber.PhoneNumberFormat.INTERNATIONAL);
|
|
|
goog.exportSymbol("libphonenumber.PhoneNumberFormat.NATIONAL",libphonenumber.PhoneNumberFormat.NATIONAL);goog.exportSymbol("libphonenumber.PhoneNumberFormat.RFC3966",libphonenumber.PhoneNumberFormat.RFC3966);goog.exportSymbol("libphonenumber.format",libphonenumber.format);})();
|
|
|
|
|
|
//! moment.js
|
|
|
//! version : 2.8.3
|
|
|
//! authors : Tim Wood, Iskren Chernev, Moment.js contributors
|
|
|
//! license : MIT
|
|
|
//! momentjs.com
|
|
|
|
|
|
(function (undefined) {
|
|
|
/************************************
|
|
|
Constants
|
|
|
************************************/
|
|
|
|
|
|
var moment,
|
|
|
VERSION = '2.8.3',
|
|
|
// the global-scope this is NOT the global object in Node.js
|
|
|
globalScope = typeof global !== 'undefined' ? global : this,
|
|
|
oldGlobalMoment,
|
|
|
round = Math.round,
|
|
|
hasOwnProperty = Object.prototype.hasOwnProperty,
|
|
|
i,
|
|
|
|
|
|
YEAR = 0,
|
|
|
MONTH = 1,
|
|
|
DATE = 2,
|
|
|
HOUR = 3,
|
|
|
MINUTE = 4,
|
|
|
SECOND = 5,
|
|
|
MILLISECOND = 6,
|
|
|
|
|
|
// internal storage for locale config files
|
|
|
locales = {},
|
|
|
|
|
|
// extra moment internal properties (plugins register props here)
|
|
|
momentProperties = [],
|
|
|
|
|
|
// check for nodeJS
|
|
|
hasModule = (typeof module !== 'undefined' && module.exports),
|
|
|
|
|
|
// ASP.NET json date format regex
|
|
|
aspNetJsonRegex = /^\/?Date\((\-?\d+)/i,
|
|
|
aspNetTimeSpanJsonRegex = /(\-)?(?:(\d*)\.)?(\d+)\:(\d+)(?:\:(\d+)\.?(\d{3})?)?/,
|
|
|
|
|
|
// 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
|
|
|
isoDurationRegex = /^(-)?P(?:(?:([0-9,.]*)Y)?(?:([0-9,.]*)M)?(?:([0-9,.]*)D)?(?:T(?:([0-9,.]*)H)?(?:([0-9,.]*)M)?(?:([0-9,.]*)S)?)?|([0-9,.]*)W)$/,
|
|
|
|
|
|
// format tokens
|
|
|
formattingTokens = /(\[[^\[]*\])|(\\)?(Mo|MM?M?M?|Do|DDDo|DD?D?D?|ddd?d?|do?|w[o|w]?|W[o|W]?|Q|YYYYYY|YYYYY|YYYY|YY|gg(ggg?)?|GG(GGG?)?|e|E|a|A|hh?|HH?|mm?|ss?|S{1,4}|X|zz?|ZZ?|.)/g,
|
|
|
localFormattingTokens = /(\[[^\[]*\])|(\\)?(LT|LL?L?L?|l{1,4})/g,
|
|
|
|
|
|
// parsing token regexes
|
|
|
parseTokenOneOrTwoDigits = /\d\d?/, // 0 - 99
|
|
|
parseTokenOneToThreeDigits = /\d{1,3}/, // 0 - 999
|
|
|
parseTokenOneToFourDigits = /\d{1,4}/, // 0 - 9999
|
|
|
parseTokenOneToSixDigits = /[+\-]?\d{1,6}/, // -999,999 - 999,999
|
|
|
parseTokenDigits = /\d+/, // nonzero number of digits
|
|
|
parseTokenWord = /[0-9]*['a-z\u00A0-\u05FF\u0700-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF]+|[\u0600-\u06FF\/]+(\s*?[\u0600-\u06FF]+){1,2}/i, // any word (or two) characters or numbers including two/three word month in arabic.
|
|
|
parseTokenTimezone = /Z|[\+\-]\d\d:?\d\d/gi, // +00:00 -00:00 +0000 -0000 or Z
|
|
|
parseTokenT = /T/i, // T (ISO separator)
|
|
|
parseTokenTimestampMs = /[\+\-]?\d+(\.\d{1,3})?/, // 123456789 123456789.123
|
|
|
parseTokenOrdinal = /\d{1,2}/,
|
|
|
|
|
|
//strict parsing regexes
|
|
|
parseTokenOneDigit = /\d/, // 0 - 9
|
|
|
parseTokenTwoDigits = /\d\d/, // 00 - 99
|
|
|
parseTokenThreeDigits = /\d{3}/, // 000 - 999
|
|
|
parseTokenFourDigits = /\d{4}/, // 0000 - 9999
|
|
|
parseTokenSixDigits = /[+-]?\d{6}/, // -999,999 - 999,999
|
|
|
parseTokenSignedNumber = /[+-]?\d+/, // -inf - inf
|
|
|
|
|
|
// 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)
|
|
|
isoRegex = /^\s*(?:[+-]\d{6}|\d{4})-(?:(\d\d-\d\d)|(W\d\d$)|(W\d\d-\d)|(\d\d\d))((T| )(\d\d(:\d\d(:\d\d(\.\d+)?)?)?)?([\+\-]\d\d(?::?\d\d)?|\s*Z)?)?$/,
|
|
|
|
|
|
isoFormat = 'YYYY-MM-DDTHH:mm:ssZ',
|
|
|
|
|
|
isoDates = [
|
|
|
['YYYYYY-MM-DD', /[+-]\d{6}-\d{2}-\d{2}/],
|
|
|
['YYYY-MM-DD', /\d{4}-\d{2}-\d{2}/],
|
|
|
['GGGG-[W]WW-E', /\d{4}-W\d{2}-\d/],
|
|
|
['GGGG-[W]WW', /\d{4}-W\d{2}/],
|
|
|
['YYYY-DDD', /\d{4}-\d{3}/]
|
|
|
],
|
|
|
|
|
|
// iso time formats and regexes
|
|
|
isoTimes = [
|
|
|
['HH:mm:ss.SSSS', /(T| )\d\d:\d\d:\d\d\.\d+/],
|
|
|
['HH:mm:ss', /(T| )\d\d:\d\d:\d\d/],
|
|
|
['HH:mm', /(T| )\d\d:\d\d/],
|
|
|
['HH', /(T| )\d\d/]
|
|
|
],
|
|
|
|
|
|
// timezone chunker '+10:00' > ['10', '00'] or '-1530' > ['-15', '30']
|
|
|
parseTimezoneChunker = /([\+\-]|\d\d)/gi,
|
|
|
|
|
|
// getter and setter names
|
|
|
proxyGettersAndSetters = 'Date|Hours|Minutes|Seconds|Milliseconds'.split('|'),
|
|
|
unitMillisecondFactors = {
|
|
|
'Milliseconds' : 1,
|
|
|
'Seconds' : 1e3,
|
|
|
'Minutes' : 6e4,
|
|
|
'Hours' : 36e5,
|
|
|
'Days' : 864e5,
|
|
|
'Months' : 2592e6,
|
|
|
'Years' : 31536e6
|
|
|
},
|
|
|
|
|
|
unitAliases = {
|
|
|
ms : 'millisecond',
|
|
|
s : 'second',
|
|
|
m : 'minute',
|
|
|
h : 'hour',
|
|
|
d : 'day',
|
|
|
D : 'date',
|
|
|
w : 'week',
|
|
|
W : 'isoWeek',
|
|
|
M : 'month',
|
|
|
Q : 'quarter',
|
|
|
y : 'year',
|
|
|
DDD : 'dayOfYear',
|
|
|
e : 'weekday',
|
|
|
E : 'isoWeekday',
|
|
|
gg: 'weekYear',
|
|
|
GG: 'isoWeekYear'
|
|
|
},
|
|
|
|
|
|
camelFunctions = {
|
|
|
dayofyear : 'dayOfYear',
|
|
|
isoweekday : 'isoWeekday',
|
|
|
isoweek : 'isoWeek',
|
|
|
weekyear : 'weekYear',
|
|
|
isoweekyear : 'isoWeekYear'
|
|
|
},
|
|
|
|
|
|
// format function strings
|
|
|
formatFunctions = {},
|
|
|
|
|
|
// default relative time thresholds
|
|
|
relativeTimeThresholds = {
|
|
|
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
|
|
|
},
|
|
|
|
|
|
// tokens to ordinalize and pad
|
|
|
ordinalizeTokens = 'DDD w W M D d'.split(' '),
|
|
|
paddedTokens = 'M D H h m s w W'.split(' '),
|
|
|
|
|
|
formatTokenFunctions = {
|
|
|
M : function () {
|
|
|
return this.month() + 1;
|
|
|
},
|
|
|
MMM : function (format) {
|
|
|
return this.localeData().monthsShort(this, format);
|
|
|
},
|
|
|
MMMM : function (format) {
|
|
|
return this.localeData().months(this, format);
|
|
|
},
|
|
|
D : function () {
|
|
|
return this.date();
|
|
|
},
|
|
|
DDD : function () {
|
|
|
return this.dayOfYear();
|
|
|
},
|
|
|
d : function () {
|
|
|
return this.day();
|
|
|
},
|
|
|
dd : function (format) {
|
|
|
return this.localeData().weekdaysMin(this, format);
|
|
|
},
|
|
|
ddd : function (format) {
|
|
|
return this.localeData().weekdaysShort(this, format);
|
|
|
},
|
|
|
dddd : function (format) {
|
|
|
return this.localeData().weekdays(this, format);
|
|
|
},
|
|
|
w : function () {
|
|
|
return this.week();
|
|
|
},
|
|
|
W : function () {
|
|
|
return this.isoWeek();
|
|
|
},
|
|
|
YY : function () {
|
|
|
return leftZeroFill(this.year() % 100, 2);
|
|
|
},
|
|
|
YYYY : function () {
|
|
|
return leftZeroFill(this.year(), 4);
|
|
|
},
|
|
|
YYYYY : function () {
|
|
|
return leftZeroFill(this.year(), 5);
|
|
|
},
|
|
|
YYYYYY : function () {
|
|
|
var y = this.year(), sign = y >= 0 ? '+' : '-';
|
|
|
return sign + leftZeroFill(Math.abs(y), 6);
|
|
|
},
|
|
|
gg : function () {
|
|
|
return leftZeroFill(this.weekYear() % 100, 2);
|
|
|
},
|
|
|
gggg : function () {
|
|
|
return leftZeroFill(this.weekYear(), 4);
|
|
|
},
|
|
|
ggggg : function () {
|
|
|
return leftZeroFill(this.weekYear(), 5);
|
|
|
},
|
|
|
GG : function () {
|
|
|
return leftZeroFill(this.isoWeekYear() % 100, 2);
|
|
|
},
|
|
|
GGGG : function () {
|
|
|
return leftZeroFill(this.isoWeekYear(), 4);
|
|
|
},
|
|
|
GGGGG : function () {
|
|
|
return leftZeroFill(this.isoWeekYear(), 5);
|
|
|
},
|
|
|
e : function () {
|
|
|
return this.weekday();
|
|
|
},
|
|
|
E : function () {
|
|
|
return this.isoWeekday();
|
|
|
},
|
|
|
a : function () {
|
|
|
return this.localeData().meridiem(this.hours(), this.minutes(), true);
|
|
|
},
|
|
|
A : function () {
|
|
|
return this.localeData().meridiem(this.hours(), this.minutes(), false);
|
|
|
},
|
|
|
H : function () {
|
|
|
return this.hours();
|
|
|
},
|
|
|
h : function () {
|
|
|
return this.hours() % 12 || 12;
|
|
|
},
|
|
|
m : function () {
|
|
|
return this.minutes();
|
|
|
},
|
|
|
s : function () {
|
|
|
return this.seconds();
|
|
|
},
|
|
|
S : function () {
|
|
|
return toInt(this.milliseconds() / 100);
|
|
|
},
|
|
|
SS : function () {
|
|
|
return leftZeroFill(toInt(this.milliseconds() / 10), 2);
|
|
|
},
|
|
|
SSS : function () {
|
|
|
return leftZeroFill(this.milliseconds(), 3);
|
|
|
},
|
|
|
SSSS : function () {
|
|
|
return leftZeroFill(this.milliseconds(), 3);
|
|
|
},
|
|
|
Z : function () {
|
|
|
var a = -this.zone(),
|
|
|
b = '+';
|
|
|
if (a < 0) {
|
|
|
a = -a;
|
|
|
b = '-';
|
|
|
}
|
|
|
return b + leftZeroFill(toInt(a / 60), 2) + ':' + leftZeroFill(toInt(a) % 60, 2);
|
|
|
},
|
|
|
ZZ : function () {
|
|
|
var a = -this.zone(),
|
|
|
b = '+';
|
|
|
if (a < 0) {
|
|
|
a = -a;
|
|
|
b = '-';
|
|
|
}
|
|
|
return b + leftZeroFill(toInt(a / 60), 2) + leftZeroFill(toInt(a) % 60, 2);
|
|
|
},
|
|
|
z : function () {
|
|
|
return this.zoneAbbr();
|
|
|
},
|
|
|
zz : function () {
|
|
|
return this.zoneName();
|
|
|
},
|
|
|
X : function () {
|
|
|
return this.unix();
|
|
|
},
|
|
|
Q : function () {
|
|
|
return this.quarter();
|
|
|
}
|
|
|
},
|
|
|
|
|
|
deprecations = {},
|
|
|
|
|
|
lists = ['months', 'monthsShort', 'weekdays', 'weekdaysShort', 'weekdaysMin'];
|
|
|
|
|
|
// Pick the first defined of two or three arguments. dfl comes from
|
|
|
// default.
|
|
|
function dfl(a, b, c) {
|
|
|
switch (arguments.length) {
|
|
|
case 2: return a != null ? a : b;
|
|
|
case 3: return a != null ? a : b != null ? b : c;
|
|
|
default: throw new Error('Implement me');
|
|
|
}
|
|
|
}
|
|
|
|
|
|
function hasOwnProp(a, b) {
|
|
|
return hasOwnProperty.call(a, b);
|
|
|
}
|
|
|
|
|
|
function defaultParsingFlags() {
|
|
|
// We need to deep clone this object, and es5 standard is not very
|
|
|
// helpful.
|
|
|
return {
|
|
|
empty : false,
|
|
|
unusedTokens : [],
|
|
|
unusedInput : [],
|
|
|
overflow : -2,
|
|
|
charsLeftOver : 0,
|
|
|
nullInput : false,
|
|
|
invalidMonth : null,
|
|
|
invalidFormat : false,
|
|
|
userInvalidated : false,
|
|
|
iso: false
|
|
|
};
|
|
|
}
|
|
|
|
|
|
function printMsg(msg) {
|
|
|
if (moment.suppressDeprecationWarnings === false &&
|
|
|
typeof console !== 'undefined' && console.warn) {
|
|
|
console.warn('Deprecation warning: ' + msg);
|
|
|
}
|
|
|
}
|
|
|
|
|
|
function deprecate(msg, fn) {
|
|
|
var firstTime = true;
|
|
|
return extend(function () {
|
|
|
if (firstTime) {
|
|
|
printMsg(msg);
|
|
|
firstTime = false;
|
|
|
}
|
|
|
return fn.apply(this, arguments);
|
|
|
}, fn);
|
|
|
}
|
|
|
|
|
|
function deprecateSimple(name, msg) {
|
|
|
if (!deprecations[name]) {
|
|
|
printMsg(msg);
|
|
|
deprecations[name] = true;
|
|
|
}
|
|
|
}
|
|
|
|
|
|
function padToken(func, count) {
|
|
|
return function (a) {
|
|
|
return leftZeroFill(func.call(this, a), count);
|
|
|
};
|
|
|
}
|
|
|
function ordinalizeToken(func, period) {
|
|
|
return function (a) {
|
|
|
return this.localeData().ordinal(func.call(this, a), period);
|
|
|
};
|
|
|
}
|
|
|
|
|
|
while (ordinalizeTokens.length) {
|
|
|
i = ordinalizeTokens.pop();
|
|
|
formatTokenFunctions[i + 'o'] = ordinalizeToken(formatTokenFunctions[i], i);
|
|
|
}
|
|
|
while (paddedTokens.length) {
|
|
|
i = paddedTokens.pop();
|
|
|
formatTokenFunctions[i + i] = padToken(formatTokenFunctions[i], 2);
|
|
|
}
|
|
|
formatTokenFunctions.DDDD = padToken(formatTokenFunctions.DDD, 3);
|
|
|
|
|
|
|
|
|
/************************************
|
|
|
Constructors
|
|
|
************************************/
|
|
|
|
|
|
function Locale() {
|
|
|
}
|
|
|
|
|
|
// Moment prototype object
|
|
|
function Moment(config, skipOverflow) {
|
|
|
if (skipOverflow !== false) {
|
|
|
checkOverflow(config);
|
|
|
}
|
|
|
copyConfig(this, config);
|
|
|
this._d = new Date(+config._d);
|
|
|
}
|
|
|
|
|
|
// Duration Constructor
|
|
|
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 = moment.localeData();
|
|
|
|
|
|
this._bubble();
|
|
|
}
|
|
|
|
|
|
/************************************
|
|
|
Helpers
|
|
|
************************************/
|
|
|
|
|
|
|
|
|
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 copyConfig(to, from) {
|
|
|
var i, prop, val;
|
|
|
|
|
|
if (typeof from._isAMomentObject !== 'undefined') {
|
|
|
to._isAMomentObject = from._isAMomentObject;
|
|
|
}
|
|
|
if (typeof from._i !== 'undefined') {
|
|
|
to._i = from._i;
|
|
|
}
|
|
|
if (typeof from._f !== 'undefined') {
|
|
|
to._f = from._f;
|
|
|
}
|
|
|
if (typeof from._l !== 'undefined') {
|
|
|
to._l = from._l;
|
|
|
}
|
|
|
if (typeof from._strict !== 'undefined') {
|
|
|
to._strict = from._strict;
|
|
|
}
|
|
|
if (typeof from._tzm !== 'undefined') {
|
|
|
to._tzm = from._tzm;
|
|
|
}
|
|
|
if (typeof from._isUTC !== 'undefined') {
|
|
|
to._isUTC = from._isUTC;
|
|
|
}
|
|
|
if (typeof from._offset !== 'undefined') {
|
|
|
to._offset = from._offset;
|
|
|
}
|
|
|
if (typeof from._pf !== 'undefined') {
|
|
|
to._pf = from._pf;
|
|
|
}
|
|
|
if (typeof from._locale !== 'undefined') {
|
|
|
to._locale = from._locale;
|
|
|
}
|
|
|
|
|
|
if (momentProperties.length > 0) {
|
|
|
for (i in momentProperties) {
|
|
|
prop = momentProperties[i];
|
|
|
val = from[prop];
|
|
|
if (typeof val !== 'undefined') {
|
|
|
to[prop] = val;
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
|
|
|
return to;
|
|
|
}
|
|
|
|
|
|
function absRound(number) {
|
|
|
if (number < 0) {
|
|
|
return Math.ceil(number);
|
|
|
} else {
|
|
|
return Math.floor(number);
|
|
|
}
|
|
|
}
|
|
|
|
|
|
// left zero fill a number
|
|
|
// see http://jsperf.com/left-zero-filling for performance comparison
|
|
|
function leftZeroFill(number, targetLength, forceSign) {
|
|
|
var output = '' + Math.abs(number),
|
|
|
sign = number >= 0;
|
|
|
|
|
|
while (output.length < targetLength) {
|
|
|
output = '0' + output;
|
|
|
}
|
|
|
return (sign ? (forceSign ? '+' : '') : '-') + output;
|
|
|
}
|
|
|
|
|
|
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;
|
|
|
other = makeAs(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;
|
|
|
}
|
|
|
|
|
|
// 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 = moment.duration(val, period);
|
|
|
addOrSubtractDurationFromMoment(this, dur, direction);
|
|
|
return this;
|
|
|
};
|
|
|
}
|
|
|
|
|
|
function addOrSubtractDurationFromMoment(mom, duration, isAdding, updateOffset) {
|
|
|
var milliseconds = duration._milliseconds,
|
|
|
days = duration._days,
|
|
|
months = duration._months;
|
|
|
updateOffset = updateOffset == null ? true : updateOffset;
|
|
|
|
|
|
if (milliseconds) {
|
|
|
mom._d.setTime(+mom._d + milliseconds * isAdding);
|
|
|
}
|
|
|
if (days) {
|
|
|
rawSetter(mom, 'Date', rawGetter(mom, 'Date') + days * isAdding);
|
|
|
}
|
|
|
if (months) {
|
|
|
rawMonthSetter(mom, rawGetter(mom, 'Month') + months * isAdding);
|
|
|
}
|
|
|
if (updateOffset) {
|
|
|
moment.updateOffset(mom, days || months);
|
|
|
}
|
|
|
}
|
|
|
|
|
|
// check if is an array
|
|
|
function isArray(input) {
|
|
|
return Object.prototype.toString.call(input) === '[object Array]';
|
|
|
}
|
|
|
|
|
|
function isDate(input) {
|
|
|
return Object.prototype.toString.call(input) === '[object Date]' ||
|
|
|
input instanceof Date;
|
|
|
}
|
|
|
|
|
|
// 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 normalizeUnits(units) {
|
|
|
if (units) {
|
|
|
var lowered = units.toLowerCase().replace(/(.)s$/, '$1');
|
|
|
units = unitAliases[units] || camelFunctions[lowered] || lowered;
|
|
|
}
|
|
|
return units;
|
|
|
}
|
|
|
|
|
|
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 makeList(field) {
|
|
|
var count, setter;
|
|
|
|
|
|
if (field.indexOf('week') === 0) {
|
|
|
count = 7;
|
|
|
setter = 'day';
|
|
|
}
|
|
|
else if (field.indexOf('month') === 0) {
|
|
|
count = 12;
|
|
|
setter = 'month';
|
|
|
}
|
|
|
else {
|
|
|
return;
|
|
|
}
|
|
|
|
|
|
moment[field] = function (format, index) {
|
|
|
var i, getter,
|
|
|
method = moment._locale[field],
|
|
|
results = [];
|
|
|
|
|
|
if (typeof format === 'number') {
|
|
|
index = format;
|
|
|
format = undefined;
|
|
|
}
|
|
|
|
|
|
getter = function (i) {
|
|
|
var m = moment().utc().set(setter, i);
|
|
|
return method.call(moment._locale, m, format || '');
|
|
|
};
|
|
|
|
|
|
if (index != null) {
|
|
|
return getter(index);
|
|
|
}
|
|
|
else {
|
|
|
for (i = 0; i < count; i++) {
|
|
|
results.push(getter(i));
|
|
|
}
|
|
|
return results;
|
|
|
}
|
|
|
};
|
|
|
}
|
|
|
|
|
|
function toInt(argumentForCoercion) {
|
|
|
var coercedNumber = +argumentForCoercion,
|
|
|
value = 0;
|
|
|
|
|
|
if (coercedNumber !== 0 && isFinite(coercedNumber)) {
|
|
|
if (coercedNumber >= 0) {
|
|
|
value = Math.floor(coercedNumber);
|
|
|
} else {
|
|
|
value = Math.ceil(coercedNumber);
|
|
|
}
|
|
|
}
|
|
|
|
|
|
return value;
|
|
|
}
|
|
|
|
|
|
function daysInMonth(year, month) {
|
|
|
return new Date(Date.UTC(year, month + 1, 0)).getUTCDate();
|
|
|
}
|
|
|
|
|
|
function weeksInYear(year, dow, doy) {
|
|
|
return weekOfYear(moment([year, 11, 31 + dow - doy]), dow, doy).week;
|
|
|
}
|
|
|
|
|
|
function daysInYear(year) {
|
|
|
return isLeapYear(year) ? 366 : 365;
|
|
|
}
|
|
|
|
|
|
function isLeapYear(year) {
|
|
|
return (year % 4 === 0 && year % 100 !== 0) || year % 400 === 0;
|
|
|
}
|
|
|
|
|
|
function checkOverflow(m) {
|
|
|
var overflow;
|
|
|
if (m._a && m._pf.overflow === -2) {
|
|
|
overflow =
|
|
|
m._a[MONTH] < 0 || m._a[MONTH] > 11 ? MONTH :
|
|
|
m._a[DATE] < 1 || m._a[DATE] > daysInMonth(m._a[YEAR], m._a[MONTH]) ? DATE :
|
|
|
m._a[HOUR] < 0 || m._a[HOUR] > 23 ? HOUR :
|
|
|
m._a[MINUTE] < 0 || m._a[MINUTE] > 59 ? MINUTE :
|
|
|
m._a[SECOND] < 0 || m._a[SECOND] > 59 ? SECOND :
|
|
|
m._a[MILLISECOND] < 0 || m._a[MILLISECOND] > 999 ? MILLISECOND :
|
|
|
-1;
|
|
|
|
|
|
if (m._pf._overflowDayOfYear && (overflow < YEAR || overflow > DATE)) {
|
|
|
overflow = DATE;
|
|
|
}
|
|
|
|
|
|
m._pf.overflow = overflow;
|
|
|
}
|
|
|
}
|
|
|
|
|
|
function isValid(m) {
|
|
|
if (m._isValid == null) {
|
|
|
m._isValid = !isNaN(m._d.getTime()) &&
|
|
|
m._pf.overflow < 0 &&
|
|
|
!m._pf.empty &&
|
|
|
!m._pf.invalidMonth &&
|
|
|
!m._pf.nullInput &&
|
|
|
!m._pf.invalidFormat &&
|
|
|
!m._pf.userInvalidated;
|
|
|
|
|
|
if (m._strict) {
|
|
|
m._isValid = m._isValid &&
|
|
|
m._pf.charsLeftOver === 0 &&
|
|
|
m._pf.unusedTokens.length === 0;
|
|
|
}
|
|
|
}
|
|
|
return m._isValid;
|
|
|
}
|
|
|
|
|
|
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;
|
|
|
if (!locales[name] && hasModule) {
|
|
|
try {
|
|
|
oldLocale = moment.locale();
|
|
|
require('./locale/' + name);
|
|
|
// because defineLocale currently also sets the global locale, we want to undo that for lazy loaded locales
|
|
|
moment.locale(oldLocale);
|
|
|
} catch (e) { }
|
|
|
}
|
|
|
return locales[name];
|
|
|
}
|
|
|
|
|
|
// Return a moment from input, that is local/utc/zone equivalent to model.
|
|
|
function makeAs(input, model) {
|
|
|
return model._isUTC ? moment(input).zone(model._offset || 0) :
|
|
|
moment(input).local();
|
|
|
}
|
|
|
|
|
|
/************************************
|
|
|
Locale
|
|
|
************************************/
|
|
|
|
|
|
|
|
|
extend(Locale.prototype, {
|
|
|
|
|
|
set : function (config) {
|
|
|
var prop, i;
|
|
|
for (i in config) {
|
|
|
prop = config[i];
|
|
|
if (typeof prop === 'function') {
|
|
|
this[i] = prop;
|
|
|
} else {
|
|
|
this['_' + i] = prop;
|
|
|
}
|
|
|
}
|
|
|
},
|
|
|
|
|
|
_months : 'January_February_March_April_May_June_July_August_September_October_November_December'.split('_'),
|
|
|
months : function (m) {
|
|
|
return this._months[m.month()];
|
|
|
},
|
|
|
|
|
|
_monthsShort : 'Jan_Feb_Mar_Apr_May_Jun_Jul_Aug_Sep_Oct_Nov_Dec'.split('_'),
|
|
|
monthsShort : function (m) {
|
|
|
return this._monthsShort[m.month()];
|
|
|
},
|
|
|
|
|
|
monthsParse : function (monthName) {
|
|
|
var i, mom, regex;
|
|
|
|
|
|
if (!this._monthsParse) {
|
|
|
this._monthsParse = [];
|
|
|
}
|
|
|
|
|
|
for (i = 0; i < 12; i++) {
|
|
|
// make the regex if we don't have it already
|
|
|
if (!this._monthsParse[i]) {
|
|
|
mom = moment.utc([2000, i]);
|
|
|
regex = '^' + this.months(mom, '') + '|^' + this.monthsShort(mom, '');
|
|
|
this._monthsParse[i] = new RegExp(regex.replace('.', ''), 'i');
|
|
|
}
|
|
|
// test the regex
|
|
|
if (this._monthsParse[i].test(monthName)) {
|
|
|
return i;
|
|
|
}
|
|
|
}
|
|
|
},
|
|
|
|
|
|
_weekdays : 'Sunday_Monday_Tuesday_Wednesday_Thursday_Friday_Saturday'.split('_'),
|
|
|
weekdays : function (m) {
|
|
|
return this._weekdays[m.day()];
|
|
|
},
|
|
|
|
|
|
_weekdaysShort : 'Sun_Mon_Tue_Wed_Thu_Fri_Sat'.split('_'),
|
|
|
weekdaysShort : function (m) {
|
|
|
return this._weekdaysShort[m.day()];
|
|
|
},
|
|
|
|
|
|
_weekdaysMin : 'Su_Mo_Tu_We_Th_Fr_Sa'.split('_'),
|
|
|
weekdaysMin : function (m) {
|
|
|
return this._weekdaysMin[m.day()];
|
|
|
},
|
|
|
|
|
|
weekdaysParse : function (weekdayName) {
|
|
|
var i, mom, regex;
|
|
|
|
|
|
if (!this._weekdaysParse) {
|
|
|
this._weekdaysParse = [];
|
|
|
}
|
|
|
|
|
|
for (i = 0; i < 7; i++) {
|
|
|
// make the regex if we don't have it already
|
|
|
if (!this._weekdaysParse[i]) {
|
|
|
mom = moment([2000, 1]).day(i);
|
|
|
regex = '^' + this.weekdays(mom, '') + '|^' + this.weekdaysShort(mom, '') + '|^' + this.weekdaysMin(mom, '');
|
|
|
this._weekdaysParse[i] = new RegExp(regex.replace('.', ''), 'i');
|
|
|
}
|
|
|
// test the regex
|
|
|
if (this._weekdaysParse[i].test(weekdayName)) {
|
|
|
return i;
|
|
|
}
|
|
|
}
|
|
|
},
|
|
|
|
|
|
_longDateFormat : {
|
|
|
LT : 'h:mm A',
|
|
|
L : 'MM/DD/YYYY',
|
|
|
LL : 'MMMM D, YYYY',
|
|
|
LLL : 'MMMM D, YYYY LT',
|
|
|
LLLL : 'dddd, MMMM D, YYYY LT'
|
|
|
},
|
|
|
longDateFormat : function (key) {
|
|
|
var output = this._longDateFormat[key];
|
|
|
if (!output && this._longDateFormat[key.toUpperCase()]) {
|
|
|
output = this._longDateFormat[key.toUpperCase()].replace(/MMMM|MM|DD|dddd/g, function (val) {
|
|
|
return val.slice(1);
|
|
|
});
|
|
|
this._longDateFormat[key] = output;
|
|
|
}
|
|
|
return output;
|
|
|
},
|
|
|
|
|
|
isPM : function (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');
|
|
|
},
|
|
|
|
|
|
_meridiemParse : /[ap]\.?m?\.?/i,
|
|
|
meridiem : function (hours, minutes, isLower) {
|
|
|
if (hours > 11) {
|
|
|
return isLower ? 'pm' : 'PM';
|
|
|
} else {
|
|
|
return isLower ? 'am' : 'AM';
|
|
|
}
|
|
|
},
|
|
|
|
|
|
_calendar : {
|
|
|
sameDay : '[Today at] LT',
|
|
|
nextDay : '[Tomorrow at] LT',
|
|
|
nextWeek : 'dddd [at] LT',
|
|
|
lastDay : '[Yesterday at] LT',
|
|
|
lastWeek : '[Last] dddd [at] LT',
|
|
|
sameElse : 'L'
|
|
|
},
|
|
|
calendar : function (key, mom) {
|
|
|
var output = this._calendar[key];
|
|
|
return typeof output === 'function' ? output.apply(mom) : output;
|
|
|
},
|
|
|
|
|
|
_relativeTime : {
|
|
|
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'
|
|
|
},
|
|
|
|
|
|
relativeTime : function (number, withoutSuffix, string, isFuture) {
|
|
|
var output = this._relativeTime[string];
|
|
|
return (typeof output === 'function') ?
|
|
|
output(number, withoutSuffix, string, isFuture) :
|
|
|
output.replace(/%d/i, number);
|
|
|
},
|
|
|
|
|
|
pastFuture : function (diff, output) {
|
|
|
var format = this._relativeTime[diff > 0 ? 'future' : 'past'];
|
|
|
return typeof format === 'function' ? format(output) : format.replace(/%s/i, output);
|
|
|
},
|
|
|
|
|
|
ordinal : function (number) {
|
|
|
return this._ordinal.replace('%d', number);
|
|
|
},
|
|
|
_ordinal : '%d',
|
|
|
|
|
|
preparse : function (string) {
|
|
|
return string;
|
|
|
},
|
|
|
|
|
|
postformat : function (string) {
|
|
|
return string;
|
|
|
},
|
|
|
|
|
|
week : function (mom) {
|
|
|
return weekOfYear(mom, this._week.dow, this._week.doy).week;
|
|
|
},
|
|
|
|
|
|
_week : {
|
|
|
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.
|
|
|
},
|
|
|
|
|
|
_invalidDate: 'Invalid date',
|
|
|
invalidDate: function () {
|
|
|
return this._invalidDate;
|
|
|
}
|
|
|
});
|
|
|
|
|
|
/************************************
|
|
|
Formatting
|
|
|
************************************/
|
|
|
|
|
|
|
|
|
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());
|
|
|
|
|
|
if (!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;
|
|
|
}
|
|
|
|
|
|
|
|
|
/************************************
|
|
|
Parsing
|
|
|
************************************/
|
|
|
|
|
|
|
|
|
// get the regex to find the next token
|
|
|
function getParseRegexForToken(token, config) {
|
|
|
var a, strict = config._strict;
|
|
|
switch (token) {
|
|
|
case 'Q':
|
|
|
return parseTokenOneDigit;
|
|
|
case 'DDDD':
|
|
|
return parseTokenThreeDigits;
|
|
|
case 'YYYY':
|
|
|
case 'GGGG':
|
|
|
case 'gggg':
|
|
|
return strict ? parseTokenFourDigits : parseTokenOneToFourDigits;
|
|
|
case 'Y':
|
|
|
case 'G':
|
|
|
case 'g':
|
|
|
return parseTokenSignedNumber;
|
|
|
case 'YYYYYY':
|
|
|
case 'YYYYY':
|
|
|
case 'GGGGG':
|
|
|
case 'ggggg':
|
|
|
return strict ? parseTokenSixDigits : parseTokenOneToSixDigits;
|
|
|
case 'S':
|
|
|
if (strict) {
|
|
|
return parseTokenOneDigit;
|
|
|
}
|
|
|
/* falls through */
|
|
|
case 'SS':
|
|
|
if (strict) {
|
|
|
return parseTokenTwoDigits;
|
|
|
}
|
|
|
/* falls through */
|
|
|
case 'SSS':
|
|
|
if (strict) {
|
|
|
return parseTokenThreeDigits;
|
|
|
}
|
|
|
/* falls through */
|
|
|
case 'DDD':
|
|
|
return parseTokenOneToThreeDigits;
|
|
|
case 'MMM':
|
|
|
case 'MMMM':
|
|
|
case 'dd':
|
|
|
case 'ddd':
|
|
|
case 'dddd':
|
|
|
return parseTokenWord;
|
|
|
case 'a':
|
|
|
case 'A':
|
|
|
return config._locale._meridiemParse;
|
|
|
case 'X':
|
|
|
return parseTokenTimestampMs;
|
|
|
case 'Z':
|
|
|
case 'ZZ':
|
|
|
return parseTokenTimezone;
|
|
|
case 'T':
|
|
|
return parseTokenT;
|
|
|
case 'SSSS':
|
|
|
return parseTokenDigits;
|
|
|
case 'MM':
|
|
|
case 'DD':
|
|
|
case 'YY':
|
|
|
case 'GG':
|
|
|
case 'gg':
|
|
|
case 'HH':
|
|
|
case 'hh':
|
|
|
case 'mm':
|
|
|
case 'ss':
|
|
|
case 'ww':
|
|
|
case 'WW':
|
|
|
return strict ? parseTokenTwoDigits : parseTokenOneOrTwoDigits;
|
|
|
case 'M':
|
|
|
case 'D':
|
|
|
case 'd':
|
|
|
case 'H':
|
|
|
case 'h':
|
|
|
case 'm':
|
|
|
case 's':
|
|
|
case 'w':
|
|
|
case 'W':
|
|
|
case 'e':
|
|
|
case 'E':
|
|
|
return parseTokenOneOrTwoDigits;
|
|
|
case 'Do':
|
|
|
return parseTokenOrdinal;
|
|
|
default :
|
|
|
a = new RegExp(regexpEscape(unescapeFormat(token.replace('\\', '')), 'i'));
|
|
|
return a;
|
|
|
}
|
|
|
}
|
|
|
|
|
|
function timezoneMinutesFromString(string) {
|
|
|
string = string || '';
|
|
|
var possibleTzMatches = (string.match(parseTokenTimezone) || []),
|
|
|
tzChunk = possibleTzMatches[possibleTzMatches.length - 1] || [],
|
|
|
parts = (tzChunk + '').match(parseTimezoneChunker) || ['-', 0, 0],
|
|
|
minutes = +(parts[1] * 60) + toInt(parts[2]);
|
|
|
|
|
|
return parts[0] === '+' ? -minutes : minutes;
|
|
|
}
|
|
|
|
|
|
// function to convert string input to date
|
|
|
function addTimeToArrayFromToken(token, input, config) {
|
|
|
var a, datePartArray = config._a;
|
|
|
|
|
|
switch (token) {
|
|
|
// QUARTER
|
|
|
case 'Q':
|
|
|
if (input != null) {
|
|
|
datePartArray[MONTH] = (toInt(input) - 1) * 3;
|
|
|
}
|
|
|
break;
|
|
|
// MONTH
|
|
|
case 'M' : // fall through to MM
|
|
|
case 'MM' :
|
|
|
if (input != null) {
|
|
|
datePartArray[MONTH] = toInt(input) - 1;
|
|
|
}
|
|
|
break;
|
|
|
case 'MMM' : // fall through to MMMM
|
|
|
case 'MMMM' :
|
|
|
a = config._locale.monthsParse(input);
|
|
|
// if we didn't find a month name, mark the date as invalid.
|
|
|
if (a != null) {
|
|
|
datePartArray[MONTH] = a;
|
|
|
} else {
|
|
|
config._pf.invalidMonth = input;
|
|
|
}
|
|
|
break;
|
|
|
// DAY OF MONTH
|
|
|
case 'D' : // fall through to DD
|
|
|
case 'DD' :
|
|
|
if (input != null) {
|
|
|
datePartArray[DATE] = toInt(input);
|
|
|
}
|
|
|
break;
|
|
|
case 'Do' :
|
|
|
if (input != null) {
|
|
|
datePartArray[DATE] = toInt(parseInt(input, 10));
|
|
|
}
|
|
|
break;
|
|
|
// DAY OF YEAR
|
|
|
case 'DDD' : // fall through to DDDD
|
|
|
case 'DDDD' :
|
|
|
if (input != null) {
|
|
|
config._dayOfYear = toInt(input);
|
|
|
}
|
|
|
|
|
|
break;
|
|
|
// YEAR
|
|
|
case 'YY' :
|
|
|
datePartArray[YEAR] = moment.parseTwoDigitYear(input);
|
|
|
break;
|
|
|
case 'YYYY' :
|
|
|
case 'YYYYY' :
|
|
|
case 'YYYYYY' :
|
|
|
datePartArray[YEAR] = toInt(input);
|
|
|
break;
|
|
|
// AM / PM
|
|
|
case 'a' : // fall through to A
|
|
|
case 'A' :
|
|
|
config._isPm = config._locale.isPM(input);
|
|
|
break;
|
|
|
// 24 HOUR
|
|
|
case 'H' : // fall through to hh
|
|
|
case 'HH' : // fall through to hh
|
|
|
case 'h' : // fall through to hh
|
|
|
case 'hh' :
|
|
|
datePartArray[HOUR] = toInt(input);
|
|
|
break;
|
|
|
// MINUTE
|
|
|
case 'm' : // fall through to mm
|
|
|
case 'mm' :
|
|
|
datePartArray[MINUTE] = toInt(input);
|
|
|
break;
|
|
|
// SECOND
|
|
|
case 's' : // fall through to ss
|
|
|
case 'ss' :
|
|
|
datePartArray[SECOND] = toInt(input);
|
|
|
break;
|
|
|
// MILLISECOND
|
|
|
case 'S' :
|
|
|
case 'SS' :
|
|
|
case 'SSS' :
|
|
|
case 'SSSS' :
|
|
|
datePartArray[MILLISECOND] = toInt(('0.' + input) * 1000);
|
|
|
break;
|
|
|
// UNIX TIMESTAMP WITH MS
|
|
|
case 'X':
|
|
|
config._d = new Date(parseFloat(input) * 1000);
|
|
|
break;
|
|
|
// TIMEZONE
|
|
|
case 'Z' : // fall through to ZZ
|
|
|
case 'ZZ' :
|
|
|
config._useUTC = true;
|
|
|
config._tzm = timezoneMinutesFromString(input);
|
|
|
break;
|
|
|
// WEEKDAY - human
|
|
|
case 'dd':
|
|
|
case 'ddd':
|
|
|
case 'dddd':
|
|
|
a = config._locale.weekdaysParse(input);
|
|
|
// if we didn't get a weekday name, mark the date as invalid
|
|
|
if (a != null) {
|
|
|
config._w = config._w || {};
|
|
|
config._w['d'] = a;
|
|
|
} else {
|
|
|
config._pf.invalidWeekday = input;
|
|
|
}
|
|
|
break;
|
|
|
// WEEK, WEEK DAY - numeric
|
|
|
case 'w':
|
|
|
case 'ww':
|
|
|
case 'W':
|
|
|
case 'WW':
|
|
|
case 'd':
|
|
|
case 'e':
|
|
|
case 'E':
|
|
|
token = token.substr(0, 1);
|
|
|
/* falls through */
|
|
|
case 'gggg':
|
|
|
case 'GGGG':
|
|
|
case 'GGGGG':
|
|
|
token = token.substr(0, 2);
|
|
|
if (input) {
|
|
|
config._w = config._w || {};
|
|
|
config._w[token] = toInt(input);
|
|
|
}
|
|
|
break;
|
|
|
case 'gg':
|
|
|
case 'GG':
|
|
|
config._w = config._w || {};
|
|
|
config._w[token] = moment.parseTwoDigitYear(input);
|
|
|
}
|
|
|
}
|
|
|
|
|
|
function dayOfYearFromWeekInfo(config) {
|
|
|
var w, weekYear, week, weekday, dow, doy, temp;
|
|
|
|
|
|
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 = dfl(w.GG, config._a[YEAR], weekOfYear(moment(), 1, 4).year);
|
|
|
week = dfl(w.W, 1);
|
|
|
weekday = dfl(w.E, 1);
|
|
|
} else {
|
|
|
dow = config._locale._week.dow;
|
|
|
doy = config._locale._week.doy;
|
|
|
|
|
|
weekYear = dfl(w.gg, config._a[YEAR], weekOfYear(moment(), dow, doy).year);
|
|
|
week = dfl(w.w, 1);
|
|
|
|
|
|
if (w.d != null) {
|
|
|
// weekday -- low day numbers are considered next week
|
|
|
weekday = w.d;
|
|
|
if (weekday < dow) {
|
|
|
++week;
|
|
|
}
|
|
|
} else if (w.e != null) {
|
|
|
// local weekday -- counting starts from begining of week
|
|
|
weekday = w.e + dow;
|
|
|
} else {
|
|
|
// default to begining of week
|
|
|
weekday = dow;
|
|
|
}
|
|
|
}
|
|
|
temp = dayOfYearFromWeeks(weekYear, week, weekday, doy, dow);
|
|
|
|
|
|
config._a[YEAR] = temp.year;
|
|
|
config._dayOfYear = temp.dayOfYear;
|
|
|
}
|
|
|
|
|
|
// 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 dateFromConfig(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 = dfl(config._a[YEAR], currentDate[YEAR]);
|
|
|
|
|
|
if (config._dayOfYear > daysInYear(yearToUse)) {
|
|
|
config._pf._overflowDayOfYear = true;
|
|
|
}
|
|
|
|
|
|
date = makeUTCDate(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];
|
|
|
}
|
|
|
|
|
|
config._d = (config._useUTC ? makeUTCDate : makeDate).apply(null, input);
|
|
|
// Apply timezone offset from input. The actual zone can be changed
|
|
|
// with parseZone.
|
|
|
if (config._tzm != null) {
|
|
|
config._d.setUTCMinutes(config._d.getUTCMinutes() + config._tzm);
|
|
|
}
|
|
|
}
|
|
|
|
|
|
function dateFromObject(config) {
|
|
|
var normalizedInput;
|
|
|
|
|
|
if (config._d) {
|
|
|
return;
|
|
|
}
|
|
|
|
|
|
normalizedInput = normalizeObjectUnits(config._i);
|
|
|
config._a = [
|
|
|
normalizedInput.year,
|
|
|
normalizedInput.month,
|
|
|
normalizedInput.day,
|
|
|
normalizedInput.hour,
|
|
|
normalizedInput.minute,
|
|
|
normalizedInput.second,
|
|
|
normalizedInput.millisecond
|
|
|
];
|
|
|
|
|
|
dateFromConfig(config);
|
|
|
}
|
|
|
|
|
|
function currentDateArray(config) {
|
|
|
var now = new Date();
|
|
|
if (config._useUTC) {
|
|
|
return [
|
|
|
now.getUTCFullYear(),
|
|
|
now.getUTCMonth(),
|
|
|
now.getUTCDate()
|
|
|
];
|
|
|
} else {
|
|
|
return [now.getFullYear(), now.getMonth(), now.getDate()];
|
|
|
}
|
|
|
}
|
|
|
|
|
|
// date from string and format string
|
|
|
function makeDateFromStringAndFormat(config) {
|
|
|
if (config._f === moment.ISO_8601) {
|
|
|
parseISO(config);
|
|
|
return;
|
|
|
}
|
|
|
|
|
|
config._a = [];
|
|
|
config._pf.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];
|
|
|
if (parsedInput) {
|
|
|
skipped = string.substr(0, string.indexOf(parsedInput));
|
|
|
if (skipped.length > 0) {
|
|
|
config._pf.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) {
|
|
|
config._pf.empty = false;
|
|
|
}
|
|
|
else {
|
|
|
config._pf.unusedTokens.push(token);
|
|
|
}
|
|
|
addTimeToArrayFromToken(token, parsedInput, config);
|
|
|
}
|
|
|
else if (config._strict && !parsedInput) {
|
|
|
config._pf.unusedTokens.push(token);
|
|
|
}
|
|
|
}
|
|
|
|
|
|
// add remaining unparsed input length to the string
|
|
|
config._pf.charsLeftOver = stringLength - totalParsedInputLength;
|
|
|
if (string.length > 0) {
|
|
|
config._pf.unusedInput.push(string);
|
|
|
}
|
|
|
|
|
|
// handle am pm
|
|
|
if (config._isPm && config._a[HOUR] < 12) {
|
|
|
config._a[HOUR] += 12;
|
|
|
}
|
|
|
// if is 12 am, change hours to 0
|
|
|
if (config._isPm === false && config._a[HOUR] === 12) {
|
|
|
config._a[HOUR] = 0;
|
|
|
}
|
|
|
|
|
|
dateFromConfig(config);
|
|
|
checkOverflow(config);
|
|
|
}
|
|
|
|
|
|
function unescapeFormat(s) {
|
|
|
return s.replace(/\\(\[)|\\(\])|\[([^\]\[]*)\]|\\(.)/g, function (matched, p1, p2, p3, p4) {
|
|
|
return p1 || p2 || p3 || p4;
|
|
|
});
|
|
|
}
|
|
|
|
|
|
// Code from http://stackoverflow.com/questions/3561493/is-there-a-regexp-escape-function-in-javascript
|
|
|
function regexpEscape(s) {
|
|
|
return s.replace(/[-\/\\^$*+?.()|[\]{}]/g, '\\$&');
|
|
|
}
|
|
|
|
|
|
// date from string and array of format strings
|
|
|
function makeDateFromStringAndArray(config) {
|
|
|
var tempConfig,
|
|
|
bestMoment,
|
|
|
|
|
|
scoreToBeat,
|
|
|
i,
|
|
|
currentScore;
|
|
|
|
|
|
if (config._f.length === 0) {
|
|
|
config._pf.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._pf = defaultParsingFlags();
|
|
|
tempConfig._f = config._f[i];
|
|
|
makeDateFromStringAndFormat(tempConfig);
|
|
|
|
|
|
if (!isValid(tempConfig)) {
|
|
|
continue;
|
|
|
}
|
|
|
|
|
|
// if there is any input that was not parsed add a penalty for that format
|
|
|
currentScore += tempConfig._pf.charsLeftOver;
|
|
|
|
|
|
//or tokens
|
|
|
currentScore += tempConfig._pf.unusedTokens.length * 10;
|
|
|
|
|
|
tempConfig._pf.score = currentScore;
|
|
|
|
|
|
if (scoreToBeat == null || currentScore < scoreToBeat) {
|
|
|
scoreToBeat = currentScore;
|
|
|
bestMoment = tempConfig;
|
|
|
}
|
|
|
}
|
|
|
|
|
|
extend(config, bestMoment || tempConfig);
|
|
|
}
|
|
|
|
|
|
// date from iso format
|
|
|
function parseISO(config) {
|
|
|
var i, l,
|
|
|
string = config._i,
|
|
|
match = isoRegex.exec(string);
|
|
|
|
|
|
if (match) {
|
|
|
config._pf.iso = true;
|
|
|
for (i = 0, l = isoDates.length; i < l; i++) {
|
|
|
if (isoDates[i][1].exec(string)) {
|
|
|
// match[5] should be 'T' or undefined
|
|
|
config._f = isoDates[i][0] + (match[6] || ' ');
|
|
|
break;
|
|
|
}
|
|
|
}
|
|
|
for (i = 0, l = isoTimes.length; i < l; i++) {
|
|
|
if (isoTimes[i][1].exec(string)) {
|
|
|
config._f += isoTimes[i][0];
|
|
|
break;
|
|
|
}
|
|
|
}
|
|
|
if (string.match(parseTokenTimezone)) {
|
|
|
config._f += 'Z';
|
|
|
}
|
|
|
makeDateFromStringAndFormat(config);
|
|
|
} else {
|
|
|
config._isValid = false;
|
|
|
}
|
|
|
}
|
|
|
|
|
|
// date from iso format or fallback
|
|
|
function makeDateFromString(config) {
|
|
|
parseISO(config);
|
|
|
if (config._isValid === false) {
|
|
|
delete config._isValid;
|
|
|
moment.createFromInputFallback(config);
|
|
|
}
|
|
|
}
|
|
|
|
|
|
function map(arr, fn) {
|
|
|
var res = [], i;
|
|
|
for (i = 0; i < arr.length; ++i) {
|
|
|
res.push(fn(arr[i], i));
|
|
|
}
|
|
|
return res;
|
|
|
}
|
|
|
|
|
|
function makeDateFromInput(config) {
|
|
|
var input = config._i, matched;
|
|
|
if (input === undefined) {
|
|
|
config._d = new Date();
|
|
|
} else if (isDate(input)) {
|
|
|
config._d = new Date(+input);
|
|
|
} else if ((matched = aspNetJsonRegex.exec(input)) !== null) {
|
|
|
config._d = new Date(+matched[1]);
|
|
|
} else if (typeof input === 'string') {
|
|
|
makeDateFromString(config);
|
|
|
} else if (isArray(input)) {
|
|
|
config._a = map(input.slice(0), function (obj) {
|
|
|
return parseInt(obj, 10);
|
|
|
});
|
|
|
dateFromConfig(config);
|
|
|
} else if (typeof(input) === 'object') {
|
|
|
dateFromObject(config);
|
|
|
} else if (typeof(input) === 'number') {
|
|
|
// from milliseconds
|
|
|
config._d = new Date(input);
|
|
|
} else {
|
|
|
moment.createFromInputFallback(config);
|
|
|
}
|
|
|
}
|
|
|
|
|
|
function makeDate(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 doesn't accept years < 1970
|
|
|
if (y < 1970) {
|
|
|
date.setFullYear(y);
|
|
|
}
|
|
|
return date;
|
|
|
}
|
|
|
|
|
|
function makeUTCDate(y) {
|
|
|
var date = new Date(Date.UTC.apply(null, arguments));
|
|
|
if (y < 1970) {
|
|
|
date.setUTCFullYear(y);
|
|
|
}
|
|
|
return date;
|
|
|
}
|
|
|
|
|
|
function parseWeekday(input, locale) {
|
|
|
if (typeof input === 'string') {
|
|
|
if (!isNaN(input)) {
|
|
|
input = parseInt(input, 10);
|
|
|
}
|
|
|
else {
|
|
|
input = locale.weekdaysParse(input);
|
|
|
if (typeof input !== 'number') {
|
|
|
return null;
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
return input;
|
|
|
}
|
|
|
|
|
|
/************************************
|
|
|
Relative Time
|
|
|
************************************/
|
|
|
|
|
|
|
|
|
// 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 relativeTime(posNegDuration, withoutSuffix, locale) {
|
|
|
var duration = moment.duration(posNegDuration).abs(),
|
|
|
seconds = round(duration.as('s')),
|
|
|
minutes = round(duration.as('m')),
|
|
|
hours = round(duration.as('h')),
|
|
|
days = round(duration.as('d')),
|
|
|
months = round(duration.as('M')),
|
|
|
years = round(duration.as('y')),
|
|
|
|
|
|
args = seconds < relativeTimeThresholds.s && ['s', seconds] ||
|
|
|
minutes === 1 && ['m'] ||
|
|
|
minutes < relativeTimeThresholds.m && ['mm', minutes] ||
|
|
|
hours === 1 && ['h'] ||
|
|
|
hours < relativeTimeThresholds.h && ['hh', hours] ||
|
|
|
days === 1 && ['d'] ||
|
|
|
days < relativeTimeThresholds.d && ['dd', days] ||
|
|
|
months === 1 && ['M'] ||
|
|
|
months < relativeTimeThresholds.M && ['MM', months] ||
|
|
|
years === 1 && ['y'] || ['yy', years];
|
|
|
|
|
|
args[2] = withoutSuffix;
|
|
|
args[3] = +posNegDuration > 0;
|
|
|
args[4] = locale;
|
|
|
return substituteTimeAgo.apply({}, args);
|
|
|
}
|
|
|
|
|
|
|
|
|
/************************************
|
|
|
Week of Year
|
|
|
************************************/
|
|
|
|
|
|
|
|
|
// firstDayOfWeek 0 = sun, 6 = sat
|
|
|
// the day of the week that starts the week
|
|
|
// (usually sunday or monday)
|
|
|
// firstDayOfWeekOfYear 0 = sun, 6 = sat
|
|
|
// the first week is the week that contains the first
|
|
|
// of this day of the week
|
|
|
// (eg. ISO weeks use thursday (4))
|
|
|
function weekOfYear(mom, firstDayOfWeek, firstDayOfWeekOfYear) {
|
|
|
var end = firstDayOfWeekOfYear - firstDayOfWeek,
|
|
|
daysToDayOfWeek = firstDayOfWeekOfYear - mom.day(),
|
|
|
adjustedMoment;
|
|
|
|
|
|
|
|
|
if (daysToDayOfWeek > end) {
|
|
|
daysToDayOfWeek -= 7;
|
|
|
}
|
|
|
|
|
|
if (daysToDayOfWeek < end - 7) {
|
|
|
daysToDayOfWeek += 7;
|
|
|
}
|
|
|
|
|
|
adjustedMoment = moment(mom).add(daysToDayOfWeek, 'd');
|
|
|
return {
|
|
|
week: Math.ceil(adjustedMoment.dayOfYear() / 7),
|
|
|
year: adjustedMoment.year()
|
|
|
};
|
|
|
}
|
|
|
|
|
|
//http://en.wikipedia.org/wiki/ISO_week_date#Calculating_a_date_given_the_year.2C_week_number_and_weekday
|
|
|
function dayOfYearFromWeeks(year, week, weekday, firstDayOfWeekOfYear, firstDayOfWeek) {
|
|
|
var d = makeUTCDate(year, 0, 1).getUTCDay(), daysToAdd, dayOfYear;
|
|
|
|
|
|
d = d === 0 ? 7 : d;
|
|
|
weekday = weekday != null ? weekday : firstDayOfWeek;
|
|
|
daysToAdd = firstDayOfWeek - d + (d > firstDayOfWeekOfYear ? 7 : 0) - (d < firstDayOfWeek ? 7 : 0);
|
|
|
dayOfYear = 7 * (week - 1) + (weekday - firstDayOfWeek) + daysToAdd + 1;
|
|
|
|
|
|
return {
|
|
|
year: dayOfYear > 0 ? year : year - 1,
|
|
|
dayOfYear: dayOfYear > 0 ? dayOfYear : daysInYear(year - 1) + dayOfYear
|
|
|
};
|
|
|
}
|
|
|
|
|
|
/************************************
|
|
|
Top Level Functions
|
|
|
************************************/
|
|
|
|
|
|
function makeMoment(config) {
|
|
|
var input = config._i,
|
|
|
format = config._f;
|
|
|
|
|
|
config._locale = config._locale || moment.localeData(config._l);
|
|
|
|
|
|
if (input === null || (format === undefined && input === '')) {
|
|
|
return moment.invalid({nullInput: true});
|
|
|
}
|
|
|
|
|
|
if (typeof input === 'string') {
|
|
|
config._i = input = config._locale.preparse(input);
|
|
|
}
|
|
|
|
|
|
if (moment.isMoment(input)) {
|
|
|
return new Moment(input, true);
|
|
|
} else if (format) {
|
|
|
if (isArray(format)) {
|
|
|
makeDateFromStringAndArray(config);
|
|
|
} else {
|
|
|
makeDateFromStringAndFormat(config);
|
|
|
}
|
|
|
} else {
|
|
|
makeDateFromInput(config);
|
|
|
}
|
|
|
|
|
|
return new Moment(config);
|
|
|
}
|
|
|
|
|
|
moment = function (input, format, locale, strict) {
|
|
|
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 = {};
|
|
|
c._isAMomentObject = true;
|
|
|
c._i = input;
|
|
|
c._f = format;
|
|
|
c._l = locale;
|
|
|
c._strict = strict;
|
|
|
c._isUTC = false;
|
|
|
c._pf = defaultParsingFlags();
|
|
|
|
|
|
return makeMoment(c);
|
|
|
};
|
|
|
|
|
|
moment.suppressDeprecationWarnings = false;
|
|
|
|
|
|
moment.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);
|
|
|
}
|
|
|
);
|
|
|
|
|
|
// 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 moment();
|
|
|
}
|
|
|
res = moments[0];
|
|
|
for (i = 1; i < moments.length; ++i) {
|
|
|
if (moments[i][fn](res)) {
|
|
|
res = moments[i];
|
|
|
}
|
|
|
}
|
|
|
return res;
|
|
|
}
|
|
|
|
|
|
moment.min = function () {
|
|
|
var args = [].slice.call(arguments, 0);
|
|
|
|
|
|
return pickBy('isBefore', args);
|
|
|
};
|
|
|
|
|
|
moment.max = function () {
|
|
|
var args = [].slice.call(arguments, 0);
|
|
|
|
|
|
return pickBy('isAfter', args);
|
|
|
};
|
|
|
|
|
|
// creating with utc
|
|
|
moment.utc = function (input, format, locale, strict) {
|
|
|
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 = {};
|
|
|
c._isAMomentObject = true;
|
|
|
c._useUTC = true;
|
|
|
c._isUTC = true;
|
|
|
c._l = locale;
|
|
|
c._i = input;
|
|
|
c._f = format;
|
|
|
c._strict = strict;
|
|
|
c._pf = defaultParsingFlags();
|
|
|
|
|
|
return makeMoment(c).utc();
|
|
|
};
|
|
|
|
|
|
// creating with unix timestamp (in seconds)
|
|
|
moment.unix = function (input) {
|
|
|
return moment(input * 1000);
|
|
|
};
|
|
|
|
|
|
// duration
|
|
|
moment.duration = function (input, key) {
|
|
|
var duration = input,
|
|
|
// matching against regexp is expensive, do it on demand
|
|
|
match = null,
|
|
|
sign,
|
|
|
ret,
|
|
|
parseIso,
|
|
|
diffRes;
|
|
|
|
|
|
if (moment.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 = aspNetTimeSpanJsonRegex.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 = isoDurationRegex.exec(input))) {
|
|
|
sign = (match[1] === '-') ? -1 : 1;
|
|
|
parseIso = function (inp) {
|
|
|
// 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;
|
|
|
};
|
|
|
duration = {
|
|
|
y: parseIso(match[2]),
|
|
|
M: parseIso(match[3]),
|
|
|
d: parseIso(match[4]),
|
|
|
h: parseIso(match[5]),
|
|
|
m: parseIso(match[6]),
|
|
|
s: parseIso(match[7]),
|
|
|
w: parseIso(match[8])
|
|
|
};
|
|
|
} else if (typeof duration === 'object' &&
|
|
|
('from' in duration || 'to' in duration)) {
|
|
|
diffRes = momentsDifference(moment(duration.from), moment(duration.to));
|
|
|
|
|
|
duration = {};
|
|
|
duration.ms = diffRes.milliseconds;
|
|
|
duration.M = diffRes.months;
|
|
|
}
|
|
|
|
|
|
ret = new Duration(duration);
|
|
|
|
|
|
if (moment.isDuration(input) && hasOwnProp(input, '_locale')) {
|
|
|
ret._locale = input._locale;
|
|
|
}
|
|
|
|
|
|
return ret;
|
|
|
};
|
|
|
|
|
|
// version number
|
|
|
moment.version = VERSION;
|
|
|
|
|
|
// default format
|
|
|
moment.defaultFormat = isoFormat;
|
|
|
|
|
|
// constant that refers to the ISO standard
|
|
|
moment.ISO_8601 = function () {};
|
|
|
|
|
|
// Plugins that add properties should also add the key here (null value),
|
|
|
// so we can properly clone ourselves.
|
|
|
moment.momentProperties = momentProperties;
|
|
|
|
|
|
// This function will be called whenever a moment is mutated.
|
|
|
// It is intended to keep the offset in sync with the timezone.
|
|
|
moment.updateOffset = function () {};
|
|
|
|
|
|
// This function allows you to set a threshold for relative time strings
|
|
|
moment.relativeTimeThreshold = function (threshold, limit) {
|
|
|
if (relativeTimeThresholds[threshold] === undefined) {
|
|
|
return false;
|
|
|
}
|
|
|
if (limit === undefined) {
|
|
|
return relativeTimeThresholds[threshold];
|
|
|
}
|
|
|
relativeTimeThresholds[threshold] = limit;
|
|
|
return true;
|
|
|
};
|
|
|
|
|
|
moment.lang = deprecate(
|
|
|
'moment.lang is deprecated. Use moment.locale instead.',
|
|
|
function (key, value) {
|
|
|
return moment.locale(key, value);
|
|
|
}
|
|
|
);
|
|
|
|
|
|
// 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.
|
|
|
moment.locale = function (key, values) {
|
|
|
var data;
|
|
|
if (key) {
|
|
|
if (typeof(values) !== 'undefined') {
|
|
|
data = moment.defineLocale(key, values);
|
|
|
}
|
|
|
else {
|
|
|
data = moment.localeData(key);
|
|
|
}
|
|
|
|
|
|
if (data) {
|
|
|
moment.duration._locale = moment._locale = data;
|
|
|
}
|
|
|
}
|
|
|
|
|
|
return moment._locale._abbr;
|
|
|
};
|
|
|
|
|
|
moment.defineLocale = function (name, values) {
|
|
|
if (values !== null) {
|
|
|
values.abbr = name;
|
|
|
if (!locales[name]) {
|
|
|
locales[name] = new Locale();
|
|
|
}
|
|
|
locales[name].set(values);
|
|
|
|
|
|
// backwards compat for now: also set the locale
|
|
|
moment.locale(name);
|
|
|
|
|
|
return locales[name];
|
|
|
} else {
|
|
|
// useful for testing
|
|
|
delete locales[name];
|
|
|
return null;
|
|
|
}
|
|
|
};
|
|
|
|
|
|
moment.langData = deprecate(
|
|
|
'moment.langData is deprecated. Use moment.localeData instead.',
|
|
|
function (key) {
|
|
|
return moment.localeData(key);
|
|
|
}
|
|
|
);
|
|
|
|
|
|
// returns locale data
|
|
|
moment.localeData = function (key) {
|
|
|
var locale;
|
|
|
|
|
|
if (key && key._locale && key._locale._abbr) {
|
|
|
key = key._locale._abbr;
|
|
|
}
|
|
|
|
|
|
if (!key) {
|
|
|
return moment._locale;
|
|
|
}
|
|
|
|
|
|
if (!isArray(key)) {
|
|
|
//short-circuit everything else
|
|
|
locale = loadLocale(key);
|
|
|
if (locale) {
|
|
|
return locale;
|
|
|
}
|
|
|
key = [key];
|
|
|
}
|
|
|
|
|
|
return chooseLocale(key);
|
|
|
};
|
|
|
|
|
|
// compare moment object
|
|
|
moment.isMoment = function (obj) {
|
|
|
return obj instanceof Moment ||
|
|
|
(obj != null && hasOwnProp(obj, '_isAMomentObject'));
|
|
|
};
|
|
|
|
|
|
// for typechecking Duration objects
|
|
|
moment.isDuration = function (obj) {
|
|
|
return obj instanceof Duration;
|
|
|
};
|
|
|
|
|
|
for (i = lists.length - 1; i >= 0; --i) {
|
|
|
makeList(lists[i]);
|
|
|
}
|
|
|
|
|
|
moment.normalizeUnits = function (units) {
|
|
|
return normalizeUnits(units);
|
|
|
};
|
|
|
|
|
|
moment.invalid = function (flags) {
|
|
|
var m = moment.utc(NaN);
|
|
|
if (flags != null) {
|
|
|
extend(m._pf, flags);
|
|
|
}
|
|
|
else {
|
|
|
m._pf.userInvalidated = true;
|
|
|
}
|
|
|
|
|
|
return m;
|
|
|
};
|
|
|
|
|
|
moment.parseZone = function () {
|
|
|
return moment.apply(null, arguments).parseZone();
|
|
|
};
|
|
|
|
|
|
moment.parseTwoDigitYear = function (input) {
|
|
|
return toInt(input) + (toInt(input) > 68 ? 1900 : 2000);
|
|
|
};
|
|
|
|
|
|
/************************************
|
|
|
Moment Prototype
|
|
|
************************************/
|
|
|
|
|
|
|
|
|
extend(moment.fn = Moment.prototype, {
|
|
|
|
|
|
clone : function () {
|
|
|
return moment(this);
|
|
|
},
|
|
|
|
|
|
valueOf : function () {
|
|
|
return +this._d + ((this._offset || 0) * 60000);
|
|
|
},
|
|
|
|
|
|
unix : function () {
|
|
|
return Math.floor(+this / 1000);
|
|
|
},
|
|
|
|
|
|
toString : function () {
|
|
|
return this.clone().locale('en').format('ddd MMM DD YYYY HH:mm:ss [GMT]ZZ');
|
|
|
},
|
|
|
|
|
|
toDate : function () {
|
|
|
return this._offset ? new Date(+this) : this._d;
|
|
|
},
|
|
|
|
|
|
toISOString : function () {
|
|
|
var m = moment(this).utc();
|
|
|
if (0 < m.year() && m.year() <= 9999) {
|
|
|
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]');
|
|
|
}
|
|
|
},
|
|
|
|
|
|
toArray : function () {
|
|
|
var m = this;
|
|
|
return [
|
|
|
m.year(),
|
|
|
m.month(),
|
|
|
m.date(),
|
|
|
m.hours(),
|
|
|
m.minutes(),
|
|
|
m.seconds(),
|
|
|
m.milliseconds()
|
|
|
];
|
|
|
},
|
|
|
|
|
|
isValid : function () {
|
|
|
return isValid(this);
|
|
|
},
|
|
|
|
|
|
isDSTShifted : function () {
|
|
|
if (this._a) {
|
|
|
return this.isValid() && compareArrays(this._a, (this._isUTC ? moment.utc(this._a) : moment(this._a)).toArray()) > 0;
|
|
|
}
|
|
|
|
|
|
return false;
|
|
|
},
|
|
|
|
|
|
parsingFlags : function () {
|
|
|
return extend({}, this._pf);
|
|
|
},
|
|
|
|
|
|
invalidAt: function () {
|
|
|
return this._pf.overflow;
|
|
|
},
|
|
|
|
|
|
utc : function (keepLocalTime) {
|
|
|
return this.zone(0, keepLocalTime);
|
|
|
},
|
|
|
|
|
|
local : function (keepLocalTime) {
|
|
|
if (this._isUTC) {
|
|
|
this.zone(0, keepLocalTime);
|
|
|
this._isUTC = false;
|
|
|
|
|
|
if (keepLocalTime) {
|
|
|
this.add(this._dateTzOffset(), 'm');
|
|
|
}
|
|
|
}
|
|
|
return this;
|
|
|
},
|
|
|
|
|
|
format : function (inputString) {
|
|
|
var output = formatMoment(this, inputString || moment.defaultFormat);
|
|
|
return this.localeData().postformat(output);
|
|
|
},
|
|
|
|
|
|
add : createAdder(1, 'add'),
|
|
|
|
|
|
subtract : createAdder(-1, 'subtract'),
|
|
|
|
|
|
diff : function (input, units, asFloat) {
|
|
|
var that = makeAs(input, this),
|
|
|
zoneDiff = (this.zone() - that.zone()) * 6e4,
|
|
|
diff, output, daysAdjust;
|
|
|
|
|
|
units = normalizeUnits(units);
|
|
|
|
|
|
if (units === 'year' || units === 'month') {
|
|
|
// average number of days in the months in the given dates
|
|
|
diff = (this.daysInMonth() + that.daysInMonth()) * 432e5; // 24 * 60 * 60 * 1000 / 2
|
|
|
// difference in months
|
|
|
output = ((this.year() - that.year()) * 12) + (this.month() - that.month());
|
|
|
// adjust by taking difference in days, average number of days
|
|
|
// and dst in the given months.
|
|
|
daysAdjust = (this - moment(this).startOf('month')) -
|
|
|
(that - moment(that).startOf('month'));
|
|
|
// same as above but with zones, to negate all dst
|
|
|
daysAdjust -= ((this.zone() - moment(this).startOf('month').zone()) -
|
|
|
(that.zone() - moment(that).startOf('month').zone())) * 6e4;
|
|
|
output += daysAdjust / diff;
|
|
|
if (units === 'year') {
|
|
|
output = output / 12;
|
|
|
}
|
|
|
} else {
|
|
|
diff = (this - that);
|
|
|
output = units === 'second' ? diff / 1e3 : // 1000
|
|
|
units === 'minute' ? diff / 6e4 : // 1000 * 60
|
|
|
units === 'hour' ? diff / 36e5 : // 1000 * 60 * 60
|
|
|
units === 'day' ? (diff - zoneDiff) / 864e5 : // 1000 * 60 * 60 * 24, negate dst
|
|
|
units === 'week' ? (diff - zoneDiff) / 6048e5 : // 1000 * 60 * 60 * 24 * 7, negate dst
|
|
|
diff;
|
|
|
}
|
|
|
return asFloat ? output : absRound(output);
|
|
|
},
|
|
|
|
|
|
from : function (time, withoutSuffix) {
|
|
|
return moment.duration({to: this, from: time}).locale(this.locale()).humanize(!withoutSuffix);
|
|
|
},
|
|
|
|
|
|
fromNow : function (withoutSuffix) {
|
|
|
return this.from(moment(), withoutSuffix);
|
|
|
},
|
|
|
|
|
|
calendar : function (time) {
|
|
|
// We want to compare the start of today, vs this.
|
|
|
// Getting start-of-today depends on whether we're zone'd or not.
|
|
|
var now = time || moment(),
|
|
|
sod = makeAs(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';
|
|
|
return this.format(this.localeData().calendar(format, this));
|
|
|
},
|
|
|
|
|
|
isLeapYear : function () {
|
|
|
return isLeapYear(this.year());
|
|
|
},
|
|
|
|
|
|
isDST : function () {
|
|
|
return (this.zone() < this.clone().month(0).zone() ||
|
|
|
this.zone() < this.clone().month(5).zone());
|
|
|
},
|
|
|
|
|
|
day : function (input) {
|
|
|
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;
|
|
|
}
|
|
|
},
|
|
|
|
|
|
month : makeAccessor('Month', true),
|
|
|
|
|
|
startOf : function (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);
|
|
|
/* falls through */
|
|
|
}
|
|
|
|
|
|
// weeks are a special case
|
|
|
if (units === 'week') {
|
|
|
this.weekday(0);
|
|
|
} else if (units === 'isoWeek') {
|
|
|
this.isoWeekday(1);
|
|
|
}
|
|
|
|
|
|
// quarters are also special
|
|
|
if (units === 'quarter') {
|
|
|
this.month(Math.floor(this.month() / 3) * 3);
|
|
|
}
|
|
|
|
|
|
return this;
|
|
|
},
|
|
|
|
|
|
endOf: function (units) {
|
|
|
units = normalizeUnits(units);
|
|
|
return this.startOf(units).add(1, (units === 'isoWeek' ? 'week' : units)).subtract(1, 'ms');
|
|
|
},
|
|
|
|
|
|
isAfter: function (input, units) {
|
|
|
units = normalizeUnits(typeof units !== 'undefined' ? units : 'millisecond');
|
|
|
if (units === 'millisecond') {
|
|
|
input = moment.isMoment(input) ? input : moment(input);
|
|
|
return +this > +input;
|
|
|
} else {
|
|
|
return +this.clone().startOf(units) > +moment(input).startOf(units);
|
|
|
}
|
|
|
},
|
|
|
|
|
|
isBefore: function (input, units) {
|
|
|
units = normalizeUnits(typeof units !== 'undefined' ? units : 'millisecond');
|
|
|
if (units === 'millisecond') {
|
|
|
input = moment.isMoment(input) ? input : moment(input);
|
|
|
return +this < +input;
|
|
|
} else {
|
|
|
return +this.clone().startOf(units) < +moment(input).startOf(units);
|
|
|
}
|
|
|
},
|
|
|
|
|
|
isSame: function (input, units) {
|
|
|
units = normalizeUnits(units || 'millisecond');
|
|
|
if (units === 'millisecond') {
|
|
|
input = moment.isMoment(input) ? input : moment(input);
|
|
|
return +this === +input;
|
|
|
} else {
|
|
|
return +this.clone().startOf(units) === +makeAs(input, this).startOf(units);
|
|
|
}
|
|
|
},
|
|
|
|
|
|
min: deprecate(
|
|
|
'moment().min is deprecated, use moment.min instead. https://github.com/moment/moment/issues/1548',
|
|
|
function (other) {
|
|
|
other = moment.apply(null, arguments);
|
|
|
return other < this ? this : other;
|
|
|
}
|
|
|
),
|
|
|
|
|
|
max: deprecate(
|
|
|
'moment().max is deprecated, use moment.max instead. https://github.com/moment/moment/issues/1548',
|
|
|
function (other) {
|
|
|
other = moment.apply(null, arguments);
|
|
|
return other > this ? this : other;
|
|
|
}
|
|
|
),
|
|
|
|
|
|
// keepLocalTime = true means only change the timezone, without
|
|
|
// affecting the local hour. So 5:31:26 +0300 --[zone(2, true)]-->
|
|
|
// 5:31:26 +0200 It is possible that 5:31:26 doesn't exist int zone
|
|
|
// +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.
|
|
|
zone : function (input, keepLocalTime) {
|
|
|
var offset = this._offset || 0,
|
|
|
localAdjust;
|
|
|
if (input != null) {
|
|
|
if (typeof input === 'string') {
|
|
|
input = timezoneMinutesFromString(input);
|
|
|
}
|
|
|
if (Math.abs(input) < 16) {
|
|
|
input = input * 60;
|
|
|
}
|
|
|
if (!this._isUTC && keepLocalTime) {
|
|
|
localAdjust = this._dateTzOffset();
|
|
|
}
|
|
|
this._offset = input;
|
|
|
this._isUTC = true;
|
|
|
if (localAdjust != null) {
|
|
|
this.subtract(localAdjust, 'm');
|
|
|
}
|
|
|
if (offset !== input) {
|
|
|
if (!keepLocalTime || this._changeInProgress) {
|
|
|
addOrSubtractDurationFromMoment(this,
|
|
|
moment.duration(offset - input, 'm'), 1, false);
|
|
|
} else if (!this._changeInProgress) {
|
|
|
this._changeInProgress = true;
|
|
|
moment.updateOffset(this, true);
|
|
|
this._changeInProgress = null;
|
|
|
}
|
|
|
}
|
|
|
} else {
|
|
|
return this._isUTC ? offset : this._dateTzOffset();
|
|
|
}
|
|
|
return this;
|
|
|
},
|
|
|
|
|
|
zoneAbbr : function () {
|
|
|
return this._isUTC ? 'UTC' : '';
|
|
|
},
|
|
|
|
|
|
zoneName : function () {
|
|
|
return this._isUTC ? 'Coordinated Universal Time' : '';
|
|
|
},
|
|
|
|
|
|
parseZone : function () {
|
|
|
if (this._tzm) {
|
|
|
this.zone(this._tzm);
|
|
|
} else if (typeof this._i === 'string') {
|
|
|
this.zone(this._i);
|
|
|
}
|
|
|
return this;
|
|
|
},
|
|
|
|
|
|
hasAlignedHourOffset : function (input) {
|
|
|
if (!input) {
|
|
|
input = 0;
|
|
|
}
|
|
|
else {
|
|
|
input = moment(input).zone();
|
|
|
}
|
|
|
|
|
|
return (this.zone() - input) % 60 === 0;
|
|
|
},
|
|
|
|
|
|
daysInMonth : function () {
|
|
|
return daysInMonth(this.year(), this.month());
|
|
|
},
|
|
|
|
|
|
dayOfYear : function (input) {
|
|
|
var dayOfYear = round((moment(this).startOf('day') - moment(this).startOf('year')) / 864e5) + 1;
|
|
|
return input == null ? dayOfYear : this.add((input - dayOfYear), 'd');
|
|
|
},
|
|
|
|
|
|
quarter : function (input) {
|
|
|
return input == null ? Math.ceil((this.month() + 1) / 3) : this.month((input - 1) * 3 + this.month() % 3);
|
|
|
},
|
|
|
|
|
|
weekYear : function (input) {
|
|
|
var year = weekOfYear(this, this.localeData()._week.dow, this.localeData()._week.doy).year;
|
|
|
return input == null ? year : this.add((input - year), 'y');
|
|
|
},
|
|
|
|
|
|
isoWeekYear : function (input) {
|
|
|
var year = weekOfYear(this, 1, 4).year;
|
|
|
return input == null ? year : this.add((input - year), 'y');
|
|
|
},
|
|
|
|
|
|
week : function (input) {
|
|
|
var week = this.localeData().week(this);
|
|
|
return input == null ? week : this.add((input - week) * 7, 'd');
|
|
|
},
|
|
|
|
|
|
isoWeek : function (input) {
|
|
|
var week = weekOfYear(this, 1, 4).week;
|
|
|
return input == null ? week : this.add((input - week) * 7, 'd');
|
|
|
},
|
|
|
|
|
|
weekday : function (input) {
|
|
|
var weekday = (this.day() + 7 - this.localeData()._week.dow) % 7;
|
|
|
return input == null ? weekday : this.add(input - weekday, 'd');
|
|
|
},
|
|
|
|
|
|
isoWeekday : function (input) {
|
|
|
// 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);
|
|
|
},
|
|
|
|
|
|
isoWeeksInYear : function () {
|
|
|
return weeksInYear(this.year(), 1, 4);
|
|
|
},
|
|
|
|
|
|
weeksInYear : function () {
|
|
|
var weekInfo = this.localeData()._week;
|
|
|
return weeksInYear(this.year(), weekInfo.dow, weekInfo.doy);
|
|
|
},
|
|
|
|
|
|
get : function (units) {
|
|
|
units = normalizeUnits(units);
|
|
|
return this[units]();
|
|
|
},
|
|
|
|
|
|
set : function (units, value) {
|
|
|
units = normalizeUnits(units);
|
|
|
if (typeof this[units] === 'function') {
|
|
|
this[units](value);
|
|
|
}
|
|
|
return this;
|
|
|
},
|
|
|
|
|
|
// If passed a locale key, it will set the locale for this
|
|
|
// instance. Otherwise, it will return the locale configuration
|
|
|
// variables for this instance.
|
|
|
locale : function (key) {
|
|
|
var newLocaleData;
|
|
|
|
|
|
if (key === undefined) {
|
|
|
return this._locale._abbr;
|
|
|
} else {
|
|
|
newLocaleData = moment.localeData(key);
|
|
|
if (newLocaleData != null) {
|
|
|
this._locale = newLocaleData;
|
|
|
}
|
|
|
return this;
|
|
|
}
|
|
|
},
|
|
|
|
|
|
lang : deprecate(
|
|
|
'moment().lang() is deprecated. Use moment().localeData() instead.',
|
|
|
function (key) {
|
|
|
if (key === undefined) {
|
|
|
return this.localeData();
|
|
|
} else {
|
|
|
return this.locale(key);
|
|
|
}
|
|
|
}
|
|
|
),
|
|
|
|
|
|
localeData : function () {
|
|
|
return this._locale;
|
|
|
},
|
|
|
|
|
|
_dateTzOffset : function () {
|
|
|
// On Firefox.24 Date#getTimezoneOffset returns a floating point.
|
|
|
// https://github.com/moment/moment/pull/1871
|
|
|
return Math.round(this._d.getTimezoneOffset() / 15) * 15;
|
|
|
}
|
|
|
});
|
|
|
|
|
|
function rawMonthSetter(mom, value) {
|
|
|
var dayOfMonth;
|
|
|
|
|
|
// TODO: Move this out of here!
|
|
|
if (typeof value === 'string') {
|
|
|
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 rawGetter(mom, unit) {
|
|
|
return mom._d['get' + (mom._isUTC ? 'UTC' : '') + unit]();
|
|
|
}
|
|
|
|
|
|
function rawSetter(mom, unit, value) {
|
|
|
if (unit === 'Month') {
|
|
|
return rawMonthSetter(mom, value);
|
|
|
} else {
|
|
|
return mom._d['set' + (mom._isUTC ? 'UTC' : '') + unit](value);
|
|
|
}
|
|
|
}
|
|
|
|
|
|
function makeAccessor(unit, keepTime) {
|
|
|
return function (value) {
|
|
|
if (value != null) {
|
|
|
rawSetter(this, unit, value);
|
|
|
moment.updateOffset(this, keepTime);
|
|
|
return this;
|
|
|
} else {
|
|
|
return rawGetter(this, unit);
|
|
|
}
|
|
|
};
|
|
|
}
|
|
|
|
|
|
moment.fn.millisecond = moment.fn.milliseconds = makeAccessor('Milliseconds', false);
|
|
|
moment.fn.second = moment.fn.seconds = makeAccessor('Seconds', false);
|
|
|
moment.fn.minute = moment.fn.minutes = makeAccessor('Minutes', false);
|
|
|
// 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.
|
|
|
moment.fn.hour = moment.fn.hours = makeAccessor('Hours', true);
|
|
|
// moment.fn.month is defined separately
|
|
|
moment.fn.date = makeAccessor('Date', true);
|
|
|
moment.fn.dates = deprecate('dates accessor is deprecated. Use date instead.', makeAccessor('Date', true));
|
|
|
moment.fn.year = makeAccessor('FullYear', true);
|
|
|
moment.fn.years = deprecate('years accessor is deprecated. Use year instead.', makeAccessor('FullYear', true));
|
|
|
|
|
|
// add plural methods
|
|
|
moment.fn.days = moment.fn.day;
|
|
|
moment.fn.months = moment.fn.month;
|
|
|
moment.fn.weeks = moment.fn.week;
|
|
|
moment.fn.isoWeeks = moment.fn.isoWeek;
|
|
|
moment.fn.quarters = moment.fn.quarter;
|
|
|
|
|
|
// add aliased format methods
|
|
|
moment.fn.toJSON = moment.fn.toISOString;
|
|
|
|
|
|
/************************************
|
|
|
Duration Prototype
|
|
|
************************************/
|
|
|
|
|
|
|
|
|
function daysToYears (days) {
|
|
|
// 400 years have 146097 days (taking into account leap year rules)
|
|
|
return days * 400 / 146097;
|
|
|
}
|
|
|
|
|
|
function yearsToDays (years) {
|
|
|
// years * 365 + absRound(years / 4) -
|
|
|
// absRound(years / 100) + absRound(years / 400);
|
|
|
return years * 146097 / 400;
|
|
|
}
|
|
|
|
|
|
extend(moment.duration.fn = Duration.prototype, {
|
|
|
|
|
|
_bubble : function () {
|
|
|
var milliseconds = this._milliseconds,
|
|
|
days = this._days,
|
|
|
months = this._months,
|
|
|
data = this._data,
|
|
|
seconds, minutes, hours, years = 0;
|
|
|
|
|
|
// The following code bubbles up values, see the tests for
|
|
|
// examples of what that means.
|
|
|
data.milliseconds = milliseconds % 1000;
|
|
|
|
|
|
seconds = absRound(milliseconds / 1000);
|
|
|
data.seconds = seconds % 60;
|
|
|
|
|
|
minutes = absRound(seconds / 60);
|
|
|
data.minutes = minutes % 60;
|
|
|
|
|
|
hours = absRound(minutes / 60);
|
|
|
data.hours = hours % 24;
|
|
|
|
|
|
days += absRound(hours / 24);
|
|
|
|
|
|
// Accurately convert days to years, assume start from year 0.
|
|
|
years = absRound(daysToYears(days));
|
|
|
days -= absRound(yearsToDays(years));
|
|
|
|
|
|
// 30 days to a month
|
|
|
// TODO (iskren): Use anchor date (like 1st Jan) to compute this.
|
|
|
months += absRound(days / 30);
|
|
|
days %= 30;
|
|
|
|
|
|
// 12 months -> 1 year
|
|
|
years += absRound(months / 12);
|
|
|
months %= 12;
|
|
|
|
|
|
data.days = days;
|
|
|
data.months = months;
|
|
|
data.years = years;
|
|
|
},
|
|
|
|
|
|
abs : function () {
|
|
|
this._milliseconds = Math.abs(this._milliseconds);
|
|
|
this._days = Math.abs(this._days);
|
|
|
this._months = Math.abs(this._months);
|
|
|
|
|
|
this._data.milliseconds = Math.abs(this._data.milliseconds);
|
|
|
this._data.seconds = Math.abs(this._data.seconds);
|
|
|
this._data.minutes = Math.abs(this._data.minutes);
|
|
|
this._data.hours = Math.abs(this._data.hours);
|
|
|
this._data.months = Math.abs(this._data.months);
|
|
|
this._data.years = Math.abs(this._data.years);
|
|
|
|
|
|
return this;
|
|
|
},
|
|
|
|
|
|
weeks : function () {
|
|
|
return absRound(this.days() / 7);
|
|
|
},
|
|
|
|
|
|
valueOf : function () {
|
|
|
return this._milliseconds +
|
|
|
this._days * 864e5 +
|
|
|
(this._months % 12) * 2592e6 +
|
|
|
toInt(this._months / 12) * 31536e6;
|
|
|
},
|
|
|
|
|
|
humanize : function (withSuffix) {
|
|
|
var output = relativeTime(this, !withSuffix, this.localeData());
|
|
|
|
|
|
if (withSuffix) {
|
|
|
output = this.localeData().pastFuture(+this, output);
|
|
|
}
|
|
|
|
|
|
return this.localeData().postformat(output);
|
|
|
},
|
|
|
|
|
|
add : function (input, val) {
|
|
|
// supports only 2.0-style add(1, 's') or add(moment)
|
|
|
var dur = moment.duration(input, val);
|
|
|
|
|
|
this._milliseconds += dur._milliseconds;
|
|
|
this._days += dur._days;
|
|
|
this._months += dur._months;
|
|
|
|
|
|
this._bubble();
|
|
|
|
|
|
return this;
|
|
|
},
|
|
|
|
|
|
subtract : function (input, val) {
|
|
|
var dur = moment.duration(input, val);
|
|
|
|
|
|
this._milliseconds -= dur._milliseconds;
|
|
|
this._days -= dur._days;
|
|
|
this._months -= dur._months;
|
|
|
|
|
|
this._bubble();
|
|
|
|
|
|
return this;
|
|
|
},
|
|
|
|
|
|
get : function (units) {
|
|
|
units = normalizeUnits(units);
|
|
|
return this[units.toLowerCase() + 's']();
|
|
|
},
|
|
|
|
|
|
as : function (units) {
|
|
|
var days, months;
|
|
|
units = normalizeUnits(units);
|
|
|
|
|
|
if (units === 'month' || units === 'year') {
|
|
|
days = this._days + this._milliseconds / 864e5;
|
|
|
months = this._months + daysToYears(days) * 12;
|
|
|
return units === 'month' ? months : months / 12;
|
|
|
} else {
|
|
|
// handle milliseconds separately because of floating point math errors (issue #1867)
|
|
|
days = this._days + yearsToDays(this._months / 12);
|
|
|
switch (units) {
|
|
|
case 'week': return days / 7 + this._milliseconds / 6048e5;
|
|
|
case 'day': return days + this._milliseconds / 864e5;
|
|
|
case 'hour': return days * 24 + this._milliseconds / 36e5;
|
|
|
case 'minute': return days * 24 * 60 + this._milliseconds / 6e4;
|
|
|
case 'second': return days * 24 * 60 * 60 + this._milliseconds / 1000;
|
|
|
// Math.floor prevents floating point math errors here
|
|
|
case 'millisecond': return Math.floor(days * 24 * 60 * 60 * 1000) + this._milliseconds;
|
|
|
default: throw new Error('Unknown unit ' + units);
|
|
|
}
|
|
|
}
|
|
|
},
|
|
|
|
|
|
lang : moment.fn.lang,
|
|
|
locale : moment.fn.locale,
|
|
|
|
|
|
toIsoString : deprecate(
|
|
|
'toIsoString() is deprecated. Please use toISOString() instead ' +
|
|
|
'(notice the capitals)',
|
|
|
function () {
|
|
|
return this.toISOString();
|
|
|
}
|
|
|
),
|
|
|
|
|
|
toISOString : function () {
|
|
|
// inspired by https://github.com/dordille/moment-isoduration/blob/master/moment.isoduration.js
|
|
|
var years = Math.abs(this.years()),
|
|
|
months = Math.abs(this.months()),
|
|
|
days = Math.abs(this.days()),
|
|
|
hours = Math.abs(this.hours()),
|
|
|
minutes = Math.abs(this.minutes()),
|
|
|
seconds = Math.abs(this.seconds() + this.milliseconds() / 1000);
|
|
|
|
|
|
if (!this.asSeconds()) {
|
|
|
// this is the same as C#'s (Noda) and python (isodate)...
|
|
|
// but not other JS (goog.date)
|
|
|
return 'P0D';
|
|
|
}
|
|
|
|
|
|
return (this.asSeconds() < 0 ? '-' : '') +
|
|
|
'P' +
|
|
|
(years ? years + 'Y' : '') +
|
|
|
(months ? months + 'M' : '') +
|
|
|
(days ? days + 'D' : '') +
|
|
|
((hours || minutes || seconds) ? 'T' : '') +
|
|
|
(hours ? hours + 'H' : '') +
|
|
|
(minutes ? minutes + 'M' : '') +
|
|
|
(seconds ? seconds + 'S' : '');
|
|
|
},
|
|
|
|
|
|
localeData : function () {
|
|
|
return this._locale;
|
|
|
}
|
|
|
});
|
|
|
|
|
|
moment.duration.fn.toString = moment.duration.fn.toISOString;
|
|
|
|
|
|
function makeDurationGetter(name) {
|
|
|
moment.duration.fn[name] = function () {
|
|
|
return this._data[name];
|
|
|
};
|
|
|
}
|
|
|
|
|
|
for (i in unitMillisecondFactors) {
|
|
|
if (hasOwnProp(unitMillisecondFactors, i)) {
|
|
|
makeDurationGetter(i.toLowerCase());
|
|
|
}
|
|
|
}
|
|
|
|
|
|
moment.duration.fn.asMilliseconds = function () {
|
|
|
return this.as('ms');
|
|
|
};
|
|
|
moment.duration.fn.asSeconds = function () {
|
|
|
return this.as('s');
|
|
|
};
|
|
|
moment.duration.fn.asMinutes = function () {
|
|
|
return this.as('m');
|
|
|
};
|
|
|
moment.duration.fn.asHours = function () {
|
|
|
return this.as('h');
|
|
|
};
|
|
|
moment.duration.fn.asDays = function () {
|
|
|
return this.as('d');
|
|
|
};
|
|
|
moment.duration.fn.asWeeks = function () {
|
|
|
return this.as('weeks');
|
|
|
};
|
|
|
moment.duration.fn.asMonths = function () {
|
|
|
return this.as('M');
|
|
|
};
|
|
|
moment.duration.fn.asYears = function () {
|
|
|
return this.as('y');
|
|
|
};
|
|
|
|
|
|
/************************************
|
|
|
Default Locale
|
|
|
************************************/
|
|
|
|
|
|
|
|
|
// Set default locale, other locale will inherit from English.
|
|
|
moment.locale('en', {
|
|
|
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;
|
|
|
}
|
|
|
});
|
|
|
|
|
|
/* EMBED_LOCALES */
|
|
|
|
|
|
/************************************
|
|
|
Exposing Moment
|
|
|
************************************/
|
|
|
|
|
|
function makeGlobal(shouldDeprecate) {
|
|
|
/*global ender:false */
|
|
|
if (typeof ender !== 'undefined') {
|
|
|
return;
|
|
|
}
|
|
|
oldGlobalMoment = globalScope.moment;
|
|
|
if (shouldDeprecate) {
|
|
|
globalScope.moment = deprecate(
|
|
|
'Accessing Moment through the global scope is ' +
|
|
|
'deprecated, and will be removed in an upcoming ' +
|
|
|
'release.',
|
|
|
moment);
|
|
|
} else {
|
|
|
globalScope.moment = moment;
|
|
|
}
|
|
|
}
|
|
|
|
|
|
// CommonJS module is defined
|
|
|
if (hasModule) {
|
|
|
module.exports = moment;
|
|
|
} else if (typeof define === 'function' && define.amd) {
|
|
|
define('moment', function (require, exports, module) {
|
|
|
if (module.config && module.config() && module.config().noGlobal === true) {
|
|
|
// release the global variable
|
|
|
globalScope.moment = oldGlobalMoment;
|
|
|
}
|
|
|
|
|
|
return moment;
|
|
|
});
|
|
|
makeGlobal(true);
|
|
|
} else {
|
|
|
makeGlobal();
|
|
|
}
|
|
|
}).call(this);
|
|
|
|
|
|
// Copyright (c) 2012 The Chromium Authors. All rights reserved.
|
|
|
// Use of this source code is governed by a BSD-style license that can be
|
|
|
// found in the LICENSE file.
|
|
|
|
|
|
// Set to true when the Document is loaded IFF "test=true" is in the query
|
|
|
// string.
|
|
|
var isTest = false;
|
|
|
|
|
|
// Set to true when loading a "Release" NaCl module, false when loading a
|
|
|
// "Debug" NaCl module.
|
|
|
var isRelease = false;
|
|
|
|
|
|
// Javascript module pattern:
|
|
|
// see http://en.wikipedia.org/wiki/Unobtrusive_JavaScript#Namespaces
|
|
|
// In essence, we define an anonymous function which is immediately called and
|
|
|
// returns a new object. The new object contains only the exported definitions;
|
|
|
// all other definitions in the anonymous function are inaccessible to external
|
|
|
// code.
|
|
|
var common = (function() {
|
|
|
|
|
|
function isHostToolchain(tool) {
|
|
|
return tool == 'win' || tool == 'linux' || tool == 'mac';
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* Return the mime type for NaCl plugin.
|
|
|
*
|
|
|
* @param {string} tool The name of the toolchain, e.g. "glibc", "newlib" etc.
|
|
|
* @return {string} The mime-type for the kind of NaCl plugin matching
|
|
|
* the given toolchain.
|
|
|
*/
|
|
|
function mimeTypeForTool(tool) {
|
|
|
// For NaCl modules use application/x-nacl.
|
|
|
var mimetype = 'application/x-nacl';
|
|
|
if (isHostToolchain(tool)) {
|
|
|
// For non-NaCl PPAPI plugins use the x-ppapi-debug/release
|
|
|
// mime type.
|
|
|
if (isRelease)
|
|
|
mimetype = 'application/x-ppapi-release';
|
|
|
else
|
|
|
mimetype = 'application/x-ppapi-debug';
|
|
|
} else if (tool == 'pnacl' && isRelease) {
|
|
|
mimetype = 'application/x-pnacl';
|
|
|
}
|
|
|
return mimetype;
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* Check if the browser supports NaCl plugins.
|
|
|
*
|
|
|
* @param {string} tool The name of the toolchain, e.g. "glibc", "newlib" etc.
|
|
|
* @return {bool} True if the browser supports the type of NaCl plugin
|
|
|
* produced by the given toolchain.
|
|
|
*/
|
|
|
function browserSupportsNaCl(tool) {
|
|
|
// Assume host toolchains always work with the given browser.
|
|
|
// The below mime-type checking might not work with
|
|
|
// --register-pepper-plugins.
|
|
|
if (isHostToolchain(tool)) {
|
|
|
return true;
|
|
|
}
|
|
|
var mimetype = mimeTypeForTool(tool);
|
|
|
return navigator.mimeTypes[mimetype] !== undefined;
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* Inject a script into the DOM, and call a callback when it is loaded.
|
|
|
*
|
|
|
* @param {string} url The url of the script to load.
|
|
|
* @param {Function} onload The callback to call when the script is loaded.
|
|
|
* @param {Function} onerror The callback to call if the script fails to load.
|
|
|
*/
|
|
|
function injectScript(url, onload, onerror) {
|
|
|
var scriptEl = document.createElement('script');
|
|
|
scriptEl.type = 'text/javascript';
|
|
|
scriptEl.src = url;
|
|
|
scriptEl.onload = onload;
|
|
|
if (onerror) {
|
|
|
scriptEl.addEventListener('error', onerror, false);
|
|
|
}
|
|
|
document.head.appendChild(scriptEl);
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* Run all tests for this example.
|
|
|
*
|
|
|
* @param {Object} moduleEl The module DOM element.
|
|
|
*/
|
|
|
function runTests(moduleEl) {
|
|
|
console.log('runTests()');
|
|
|
common.tester = new Tester();
|
|
|
|
|
|
// All NaCl SDK examples are OK if the example exits cleanly; (i.e. the
|
|
|
// NaCl module returns 0 or calls exit(0)).
|
|
|
//
|
|
|
// Without this exception, the browser_tester thinks that the module
|
|
|
// has crashed.
|
|
|
common.tester.exitCleanlyIsOK();
|
|
|
|
|
|
common.tester.addAsyncTest('loaded', function(test) {
|
|
|
test.pass();
|
|
|
});
|
|
|
|
|
|
if (typeof window.addTests !== 'undefined') {
|
|
|
window.addTests();
|
|
|
}
|
|
|
|
|
|
common.tester.waitFor(moduleEl);
|
|
|
common.tester.run();
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* Create the Native Client <embed> element as a child of the DOM element
|
|
|
* named "listener".
|
|
|
*
|
|
|
* @param {string} name The name of the example.
|
|
|
* @param {string} tool The name of the toolchain, e.g. "glibc", "newlib" etc.
|
|
|
* @param {string} path Directory name where .nmf file can be found.
|
|
|
* @param {number} width The width to create the plugin.
|
|
|
* @param {number} height The height to create the plugin.
|
|
|
* @param {Object} attrs Dictionary of attributes to set on the module.
|
|
|
*/
|
|
|
function createNaClModule(name, tool, path, width, height, attrs) {
|
|
|
var moduleEl = document.createElement('embed');
|
|
|
moduleEl.setAttribute('name', 'nacl_module');
|
|
|
moduleEl.setAttribute('id', 'nacl_module');
|
|
|
moduleEl.setAttribute('width', width);
|
|
|
moduleEl.setAttribute('height', height);
|
|
|
moduleEl.setAttribute('path', path);
|
|
|
moduleEl.setAttribute('src', path + '/' + name + '.nmf');
|
|
|
|
|
|
// Add any optional arguments
|
|
|
if (attrs) {
|
|
|
for (var key in attrs) {
|
|
|
moduleEl.setAttribute(key, attrs[key]);
|
|
|
}
|
|
|
}
|
|
|
|
|
|
var mimetype = mimeTypeForTool(tool);
|
|
|
moduleEl.setAttribute('type', mimetype);
|
|
|
|
|
|
// The <EMBED> element is wrapped inside a <DIV>, which has both a 'load'
|
|
|
// and a 'message' event listener attached. This wrapping method is used
|
|
|
// instead of attaching the event listeners directly to the <EMBED> element
|
|
|
// to ensure that the listeners are active before the NaCl module 'load'
|
|
|
// event fires.
|
|
|
var listenerDiv = document.getElementById('listener');
|
|
|
listenerDiv.appendChild(moduleEl);
|
|
|
|
|
|
// Host plugins don't send a moduleDidLoad message. We'll fake it here.
|
|
|
var isHost = isHostToolchain(tool);
|
|
|
if (isHost) {
|
|
|
window.setTimeout(function() {
|
|
|
moduleEl.readyState = 1;
|
|
|
moduleEl.dispatchEvent(new CustomEvent('loadstart'));
|
|
|
moduleEl.readyState = 4;
|
|
|
moduleEl.dispatchEvent(new CustomEvent('load'));
|
|
|
moduleEl.dispatchEvent(new CustomEvent('loadend'));
|
|
|
}, 100); // 100 ms
|
|
|
}
|
|
|
|
|
|
// This is code that is only used to test the SDK.
|
|
|
if (isTest) {
|
|
|
var loadNaClTest = function() {
|
|
|
injectScript('nacltest.js', function() {
|
|
|
runTests(moduleEl);
|
|
|
});
|
|
|
};
|
|
|
|
|
|
// Try to load test.js for the example. Whether or not it exists, load
|
|
|
// nacltest.js.
|
|
|
injectScript('test.js', loadNaClTest, loadNaClTest);
|
|
|
}
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* Add the default "load" and "message" event listeners to the element with
|
|
|
* id "listener".
|
|
|
*
|
|
|
* The "load" event is sent when the module is successfully loaded. The
|
|
|
* "message" event is sent when the naclModule posts a message using
|
|
|
* PPB_Messaging.PostMessage() (in C) or pp::Instance().PostMessage() (in
|
|
|
* C++).
|
|
|
*/
|
|
|
function attachDefaultListeners() {
|
|
|
var listenerDiv = document.getElementById('listener');
|
|
|
listenerDiv.addEventListener('load', moduleDidLoad, true);
|
|
|
listenerDiv.addEventListener('message', handleMessage, true);
|
|
|
listenerDiv.addEventListener('crash', handleCrash, true);
|
|
|
if (typeof window.attachListeners !== 'undefined') {
|
|
|
window.attachListeners();
|
|
|
}
|
|
|
}
|
|
|
|
|
|
|
|
|
/**
|
|
|
* Called when the Browser can not communicate with the Module
|
|
|
*
|
|
|
* This event listener is registered in attachDefaultListeners above.
|
|
|
*/
|
|
|
function handleCrash(event) {
|
|
|
if (common.naclModule.exitStatus == -1) {
|
|
|
updateStatus('CRASHED');
|
|
|
} else {
|
|
|
updateStatus('EXITED [' + common.naclModule.exitStatus + ']');
|
|
|
}
|
|
|
if (typeof window.handleCrash !== 'undefined') {
|
|
|
window.handleCrash(common.naclModule.lastError);
|
|
|
}
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* Called when the NaCl module is loaded.
|
|
|
*
|
|
|
* This event listener is registered in attachDefaultListeners above.
|
|
|
*/
|
|
|
function moduleDidLoad() {
|
|
|
common.naclModule = document.getElementById('nacl_module');
|
|
|
updateStatus('RUNNING');
|
|
|
|
|
|
if (typeof window.moduleDidLoad !== 'undefined') {
|
|
|
window.moduleDidLoad();
|
|
|
}
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* Hide the NaCl module's embed element.
|
|
|
*
|
|
|
* We don't want to hide by default; if we do, it is harder to determine that
|
|
|
* a plugin failed to load. Instead, call this function inside the example's
|
|
|
* "moduleDidLoad" function.
|
|
|
*
|
|
|
*/
|
|
|
function hideModule() {
|
|
|
// Setting common.naclModule.style.display = "None" doesn't work; the
|
|
|
// module will no longer be able to receive postMessages.
|
|
|
common.naclModule.style.height = '0';
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* Remove the NaCl module from the page.
|
|
|
*/
|
|
|
function removeModule() {
|
|
|
common.naclModule.parentNode.removeChild(common.naclModule);
|
|
|
common.naclModule = null;
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* Return true when |s| starts with the string |prefix|.
|
|
|
*
|
|
|
* @param {string} s The string to search.
|
|
|
* @param {string} prefix The prefix to search for in |s|.
|
|
|
*/
|
|
|
function startsWith(s, prefix) {
|
|
|
// indexOf would search the entire string, lastIndexOf(p, 0) only checks at
|
|
|
// the first index. See: http://stackoverflow.com/a/4579228
|
|
|
return s.lastIndexOf(prefix, 0) === 0;
|
|
|
}
|
|
|
|
|
|
/** Maximum length of logMessageArray. */
|
|
|
var kMaxLogMessageLength = 20;
|
|
|
|
|
|
/** An array of messages to display in the element with id "log". */
|
|
|
var logMessageArray = [];
|
|
|
|
|
|
/**
|
|
|
* Add a message to an element with id "log".
|
|
|
*
|
|
|
* This function is used by the default "log:" message handler.
|
|
|
*
|
|
|
* @param {string} message The message to log.
|
|
|
*/
|
|
|
function logMessage(message) {
|
|
|
logMessageArray.push(message);
|
|
|
if (logMessageArray.length > kMaxLogMessageLength)
|
|
|
logMessageArray.shift();
|
|
|
|
|
|
document.getElementById('log').textContent = logMessageArray.join('\n');
|
|
|
console.log(message);
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
*/
|
|
|
var defaultMessageTypes = {
|
|
|
'alert': alert,
|
|
|
'log': logMessage
|
|
|
};
|
|
|
|
|
|
/**
|
|
|
* Called when the NaCl module sends a message to JavaScript (via
|
|
|
* PPB_Messaging.PostMessage())
|
|
|
*
|
|
|
* This event listener is registered in createNaClModule above.
|
|
|
*
|
|
|
* @param {Event} message_event A message event. message_event.data contains
|
|
|
* the data sent from the NaCl module.
|
|
|
*/
|
|
|
function handleMessage(message_event) {
|
|
|
if (typeof message_event.data === 'string') {
|
|
|
for (var type in defaultMessageTypes) {
|
|
|
if (defaultMessageTypes.hasOwnProperty(type)) {
|
|
|
if (startsWith(message_event.data, type + ':')) {
|
|
|
func = defaultMessageTypes[type];
|
|
|
func(message_event.data.slice(type.length + 1));
|
|
|
return;
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
|
|
|
if (typeof window.handleMessage !== 'undefined') {
|
|
|
window.handleMessage(message_event);
|
|
|
return;
|
|
|
}
|
|
|
|
|
|
logMessage('Unhandled message: ' + message_event.data);
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* Called when the DOM content has loaded; i.e. the page's document is fully
|
|
|
* parsed. At this point, we can safely query any elements in the document via
|
|
|
* document.querySelector, document.getElementById, etc.
|
|
|
*
|
|
|
* @param {string} name The name of the example.
|
|
|
* @param {string} tool The name of the toolchain, e.g. "glibc", "newlib" etc.
|
|
|
* @param {string} path Directory name where .nmf file can be found.
|
|
|
* @param {number} width The width to create the plugin.
|
|
|
* @param {number} height The height to create the plugin.
|
|
|
* @param {Object} attrs Optional dictionary of additional attributes.
|
|
|
*/
|
|
|
function domContentLoaded(name, tool, path, width, height, attrs) {
|
|
|
// If the page loads before the Native Client module loads, then set the
|
|
|
// status message indicating that the module is still loading. Otherwise,
|
|
|
// do not change the status message.
|
|
|
updateStatus('Page loaded.');
|
|
|
if (!browserSupportsNaCl(tool)) {
|
|
|
updateStatus(
|
|
|
'Browser does not support NaCl (' + tool + '), or NaCl is disabled');
|
|
|
} else if (common.naclModule == null) {
|
|
|
updateStatus('Creating embed: ' + tool);
|
|
|
|
|
|
// We use a non-zero sized embed to give Chrome space to place the bad
|
|
|
// plug-in graphic, if there is a problem.
|
|
|
width = typeof width !== 'undefined' ? width : 200;
|
|
|
height = typeof height !== 'undefined' ? height : 200;
|
|
|
attachDefaultListeners();
|
|
|
createNaClModule(name, tool, path, width, height, attrs);
|
|
|
} else {
|
|
|
// It's possible that the Native Client module onload event fired
|
|
|
// before the page's onload event. In this case, the status message
|
|
|
// will reflect 'SUCCESS', but won't be displayed. This call will
|
|
|
// display the current message.
|
|
|
updateStatus('Waiting.');
|
|
|
}
|
|
|
}
|
|
|
|
|
|
/** Saved text to display in the element with id 'statusField'. */
|
|
|
var statusText = 'NO-STATUSES';
|
|
|
|
|
|
/**
|
|
|
* Set the global status message. If the element with id 'statusField'
|
|
|
* exists, then set its HTML to the status message as well.
|
|
|
*
|
|
|
* @param {string} opt_message The message to set. If null or undefined, then
|
|
|
* set element 'statusField' to the message from the last call to
|
|
|
* updateStatus.
|
|
|
*/
|
|
|
function updateStatus(opt_message) {
|
|
|
if (opt_message) {
|
|
|
statusText = opt_message;
|
|
|
}
|
|
|
var statusField = document.getElementById('statusField');
|
|
|
if (statusField) {
|
|
|
statusField.innerHTML = statusText;
|
|
|
}
|
|
|
}
|
|
|
|
|
|
// The symbols to export.
|
|
|
return {
|
|
|
/** A reference to the NaCl module, once it is loaded. */
|
|
|
naclModule: null,
|
|
|
|
|
|
attachDefaultListeners: attachDefaultListeners,
|
|
|
domContentLoaded: domContentLoaded,
|
|
|
createNaClModule: createNaClModule,
|
|
|
hideModule: hideModule,
|
|
|
removeModule: removeModule,
|
|
|
logMessage: logMessage,
|
|
|
updateStatus: updateStatus
|
|
|
};
|
|
|
|
|
|
}());
|
|
|
|
|
|
// Listen for the DOM content to be loaded. This event is fired when parsing of
|
|
|
// the page's document has finished.
|
|
|
document.addEventListener('DOMContentLoaded', function() {
|
|
|
var body = document.body;
|
|
|
|
|
|
// The data-* attributes on the body can be referenced via body.dataset.
|
|
|
if (body.dataset) {
|
|
|
var loadFunction;
|
|
|
if (!body.dataset.customLoad) {
|
|
|
loadFunction = common.domContentLoaded;
|
|
|
} else if (typeof window.domContentLoaded !== 'undefined') {
|
|
|
loadFunction = window.domContentLoaded;
|
|
|
}
|
|
|
|
|
|
// From https://developer.mozilla.org/en-US/docs/DOM/window.location
|
|
|
var searchVars = {};
|
|
|
if (window.location.search.length > 1) {
|
|
|
var pairs = window.location.search.substr(1).split('&');
|
|
|
for (var key_ix = 0; key_ix < pairs.length; key_ix++) {
|
|
|
var keyValue = pairs[key_ix].split('=');
|
|
|
searchVars[unescape(keyValue[0])] =
|
|
|
keyValue.length > 1 ? unescape(keyValue[1]) : '';
|
|
|
}
|
|
|
}
|
|
|
|
|
|
if (loadFunction) {
|
|
|
var toolchains = body.dataset.tools.split(' ');
|
|
|
var configs = body.dataset.configs.split(' ');
|
|
|
|
|
|
var attrs = {};
|
|
|
if (body.dataset.attrs) {
|
|
|
var attr_list = body.dataset.attrs.split(' ');
|
|
|
for (var key in attr_list) {
|
|
|
var attr = attr_list[key].split('=');
|
|
|
var key = attr[0];
|
|
|
var value = attr[1];
|
|
|
attrs[key] = value;
|
|
|
}
|
|
|
}
|
|
|
|
|
|
var tc = toolchains.indexOf(searchVars.tc) !== -1 ?
|
|
|
searchVars.tc : toolchains[0];
|
|
|
var config = configs.indexOf(searchVars.config) !== -1 ?
|
|
|
searchVars.config : configs[0];
|
|
|
var pathFormat = body.dataset.path;
|
|
|
var path = pathFormat.replace('{tc}', tc).replace('{config}', config);
|
|
|
|
|
|
isTest = searchVars.test === 'true';
|
|
|
isRelease = path.toLowerCase().indexOf('release') != -1;
|
|
|
|
|
|
loadFunction(body.dataset.name, tc, path, body.dataset.width,
|
|
|
body.dataset.height, attrs);
|
|
|
}
|
|
|
}
|
|
|
});
|