forked from sheetjs/sheetjs
		
	Use charAt for IE6-8 compat (fixes #9)
				
					
				
			This commit is contained in:
		
							parent
							
								
									56b414b496
								
							
						
					
					
						commit
						ecb4515847
					
				@ -17,7 +17,7 @@
 | 
			
		||||
.*/shim.js
 | 
			
		||||
 | 
			
		||||
[include]
 | 
			
		||||
ssf.js
 | 
			
		||||
ssf.flow.js
 | 
			
		||||
.*/bin/.*.njs
 | 
			
		||||
 | 
			
		||||
[libs]
 | 
			
		||||
 | 
			
		||||
							
								
								
									
										2
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
								
								
								
								
								
									
									
								
							
						
						
									
										2
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							@ -3,3 +3,5 @@ misc/coverage.html
 | 
			
		||||
.vocrc
 | 
			
		||||
v8.log
 | 
			
		||||
perf.log
 | 
			
		||||
ctest/test.js
 | 
			
		||||
ctest/sauce*
 | 
			
		||||
 | 
			
		||||
							
								
								
									
										4
									
								
								Makefile
									
									
									
									
									
								
							
							
								
								
								
								
								
									
									
								
							
						
						
									
										4
									
								
								Makefile
									
									
									
									
									
								
							@ -45,6 +45,10 @@ test_min:
 | 
			
		||||
travis: ## Run test suite with minimal output
 | 
			
		||||
	mocha -R dot -t 30000
 | 
			
		||||
 | 
			
		||||
.PHONY: ctest
 | 
			
		||||
ctest:
 | 
			
		||||
	browserify -t brfs test/{exp,fraction,general,implied,oddities,utilities,comma}.js > ctest/test.js
 | 
			
		||||
 | 
			
		||||
## Code Checking
 | 
			
		||||
 | 
			
		||||
.PHONY: lint
 | 
			
		||||
 | 
			
		||||
@ -72,7 +72,7 @@ function write_num_flt(type/*:string*/, fmt/*:string*/, val/*:number*/)/*:string
 | 
			
		||||
	if(fmt.charCodeAt(fmt.length - 1) === 44) return write_num_cm(type, fmt, val);
 | 
			
		||||
	if(fmt.indexOf('%') !== -1) return write_num_pct(type, fmt, val);
 | 
			
		||||
	if(fmt.indexOf('E') !== -1) return write_num_exp(fmt, val);
 | 
			
		||||
	if(fmt.charCodeAt(0) === 36) return "$"+write_num_flt(type,fmt.substr(fmt[1]==' '?2:1),val);
 | 
			
		||||
	if(fmt.charCodeAt(0) === 36) return "$"+write_num_flt(type,fmt.substr(fmt.charAt(1)==' '?2:1),val);
 | 
			
		||||
	var o;
 | 
			
		||||
	var r/*:?Array<string>*/, ri, ff, aval = Math.abs(val), sign = val < 0 ? "-" : "";
 | 
			
		||||
	if(fmt.match(/^00+$/)) return sign + pad0r(aval,fmt.length);
 | 
			
		||||
@ -99,7 +99,7 @@ function write_num_flt(type/*:string*/, fmt/*:string*/, val/*:number*/)/*:string
 | 
			
		||||
	if((r = fmt.match(/^([0#]+)(\\?-([0#]+))+$/))) {
 | 
			
		||||
		o = _strrev(write_num_flt(type, fmt.replace(/[\\-]/g,""), val));
 | 
			
		||||
		ri = 0;
 | 
			
		||||
		return _strrev(_strrev(fmt.replace(/\\/g,"")).replace(/[0#]/g,function(x){return ri<o.length?o[ri++]:x==='0'?'0':"";}));
 | 
			
		||||
		return _strrev(_strrev(fmt.replace(/\\/g,"")).replace(/[0#]/g,function(x){return ri<o.length?o.charAt(ri++):x==='0'?'0':"";}));
 | 
			
		||||
	}
 | 
			
		||||
	if(fmt.match(phone)) {
 | 
			
		||||
		o = write_num_flt(type, "##########", val);
 | 
			
		||||
@ -111,7 +111,7 @@ function write_num_flt(type/*:string*/, fmt/*:string*/, val/*:number*/)/*:string
 | 
			
		||||
		ff = frac(aval, Math.pow(10,ri)-1, false);
 | 
			
		||||
		o = "" + sign;
 | 
			
		||||
		oa = write_num("n", /*::String(*/r[1]/*::)*/, ff[1]);
 | 
			
		||||
		if(oa[oa.length-1] == " ") oa = oa.substr(0,oa.length-1) + "0";
 | 
			
		||||
		if(oa.charAt(oa.length-1) == " ") oa = oa.substr(0,oa.length-1) + "0";
 | 
			
		||||
		o += oa + /*::String(*/r[2]/*::)*/ + "/" + /*::String(*/r[3]/*::)*/;
 | 
			
		||||
		oa = rpad_(ff[2],ri);
 | 
			
		||||
		if(oa.length < r[4].length) oa = hashq(r[4].substr(r[4].length-oa.length)) + oa;
 | 
			
		||||
@ -182,7 +182,7 @@ function write_num_int(type/*:string*/, fmt/*:string*/, val/*:number*/)/*:string
 | 
			
		||||
	if(fmt.charCodeAt(fmt.length - 1) === 44) return write_num_cm2(type, fmt, val);
 | 
			
		||||
	if(fmt.indexOf('%') !== -1) return write_num_pct2(type, fmt, val);
 | 
			
		||||
	if(fmt.indexOf('E') !== -1) return write_num_exp2(fmt, val);
 | 
			
		||||
	if(fmt.charCodeAt(0) === 36) return "$"+write_num_int(type,fmt.substr(fmt[1]==' '?2:1),val);
 | 
			
		||||
	if(fmt.charCodeAt(0) === 36) return "$"+write_num_int(type,fmt.substr(fmt.charAt(1)==' '?2:1),val);
 | 
			
		||||
	var o;
 | 
			
		||||
	var r, ri, ff, aval = Math.abs(val), sign = val < 0 ? "-" : "";
 | 
			
		||||
	if(fmt.match(/^00+$/)) return sign + pad0(aval,fmt.length);
 | 
			
		||||
@ -194,14 +194,10 @@ function write_num_int(type/*:string*/, fmt/*:string*/, val/*:number*/)/*:string
 | 
			
		||||
	if(fmt.match(/^#+0+$/)) return sign + pad0(aval,fmt.length - fmt.indexOf("0"));
 | 
			
		||||
	if((r = fmt.match(dec1))) {
 | 
			
		||||
		/*:: if(!Array.isArray(r)) throw "unreachable"; */
 | 
			
		||||
		// $FlowIgnore
 | 
			
		||||
		o = (""+val).replace(/^([^\.]+)$/,"$1."+r[1]).replace(/\.$/,"."+r[1]);
 | 
			
		||||
		// $FlowIgnore
 | 
			
		||||
		o = o.replace(/\.(\d*)$/,function($$, $1) {
 | 
			
		||||
		/*:: if(!Array.isArray(r)) throw "unreachable"; */
 | 
			
		||||
		// $FlowIgnore
 | 
			
		||||
			return "." + $1 + fill("0", r[1].length-$1.length); });
 | 
			
		||||
		// $FlowIgnore
 | 
			
		||||
		return fmt.indexOf("0.") !== -1 ? o : o.replace(/^0\./,".");
 | 
			
		||||
	}
 | 
			
		||||
	fmt = fmt.replace(/^#+([0.])/, "$1");
 | 
			
		||||
@ -216,7 +212,7 @@ function write_num_int(type/*:string*/, fmt/*:string*/, val/*:number*/)/*:string
 | 
			
		||||
	if((r = fmt.match(/^([0#]+)(\\?-([0#]+))+$/))) {
 | 
			
		||||
		o = _strrev(write_num_int(type, fmt.replace(/[\\-]/g,""), val));
 | 
			
		||||
		ri = 0;
 | 
			
		||||
		return _strrev(_strrev(fmt.replace(/\\/g,"")).replace(/[0#]/g,function(x){return ri<o.length?o[ri++]:x==='0'?'0':"";}));
 | 
			
		||||
		return _strrev(_strrev(fmt.replace(/\\/g,"")).replace(/[0#]/g,function(x){return ri<o.length?o.charAt(ri++):x==='0'?'0':"";}));
 | 
			
		||||
	}
 | 
			
		||||
	if(fmt.match(phone)) {
 | 
			
		||||
		o = write_num_int(type, "##########", val);
 | 
			
		||||
@ -228,7 +224,7 @@ function write_num_int(type/*:string*/, fmt/*:string*/, val/*:number*/)/*:string
 | 
			
		||||
		ff = frac(aval, Math.pow(10,ri)-1, false);
 | 
			
		||||
		o = "" + sign;
 | 
			
		||||
		oa = write_num("n", r[1], ff[1]);
 | 
			
		||||
		if(oa[oa.length-1] == " ") oa = oa.substr(0,oa.length-1) + "0";
 | 
			
		||||
		if(oa.charAt(oa.length-1) == " ") oa = oa.substr(0,oa.length-1) + "0";
 | 
			
		||||
		o += oa + r[2] + "/" + r[3];
 | 
			
		||||
		oa = rpad_(ff[2],ri);
 | 
			
		||||
		if(oa.length < r[4].length) oa = hashq(r[4].substr(r[4].length-oa.length)) + oa;
 | 
			
		||||
 | 
			
		||||
@ -156,7 +156,7 @@ function eval_fmt(fmt/*:string*/, v/*:any*/, opts/*:any*/, flen/*:number*/) {
 | 
			
		||||
				j=out[i].v.indexOf(".")>-1&&i===decpt?out[i].v.indexOf(".")-1:out[i].v.length-1;
 | 
			
		||||
				vv = out[i].v.substr(j+1);
 | 
			
		||||
				for(; j>=0; --j) {
 | 
			
		||||
					if(jj>=0 && (out[i].v[j] === "0" || out[i].v[j] === "#")) vv = ostr[jj--] + vv;
 | 
			
		||||
					if(jj>=0 && (out[i].v.charAt(j) === "0" || out[i].v.charAt(j) === "#")) vv = ostr.charAt(jj--) + vv;
 | 
			
		||||
				}
 | 
			
		||||
				out[i].v = vv;
 | 
			
		||||
				out[i].t = 't';
 | 
			
		||||
@ -169,7 +169,7 @@ function eval_fmt(fmt/*:string*/, v/*:any*/, opts/*:any*/, flen/*:number*/) {
 | 
			
		||||
				j=out[i].v.indexOf(".")>-1&&i===decpt?out[i].v.indexOf(".")+1:0;
 | 
			
		||||
				vv = out[i].v.substr(0,j);
 | 
			
		||||
				for(; j<out[i].v.length; ++j) {
 | 
			
		||||
					if(jj<ostr.length) vv += ostr[jj++];
 | 
			
		||||
					if(jj<ostr.length) vv += ostr.charAt(jj++);
 | 
			
		||||
				}
 | 
			
		||||
				out[i].v = vv;
 | 
			
		||||
				out[i].t = 't';
 | 
			
		||||
 | 
			
		||||
							
								
								
									
										33
									
								
								ctest/index.html
									
									
									
									
									
										Normal file
									
								
							
							
								
								
								
								
								
									
									
								
							
						
						
									
										33
									
								
								ctest/index.html
									
									
									
									
									
										Normal file
									
								
							@ -0,0 +1,33 @@
 | 
			
		||||
<!DOCTYPE html>
 | 
			
		||||
<html>
 | 
			
		||||
  <head>
 | 
			
		||||
    <title>SSF Test Runner</title>
 | 
			
		||||
    <meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
 | 
			
		||||
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
 | 
			
		||||
    <link rel="stylesheet" href="mocha.css" />
 | 
			
		||||
    <link rel="icon" type="image/png" href="//oss.sheetjs.com/assets/img/logo.png" />
 | 
			
		||||
    <style>
 | 
			
		||||
      #t { font: 20px/1.5 "Helvetica Neue", Helvetica, Arial, sans-serif; margin: 0px 60px; font-weight: bold; }
 | 
			
		||||
      #tt{ font: 16px/1.5 "Helvetica Neue", Helvetica, Arial, sans-serif; margin: 0px 60px; }
 | 
			
		||||
      th { font: 16px/1.5 "Helvetica Neue", Helvetica, Arial, sans-serif; margin: 0px 60px; font-weight: bold; text-align: left; }
 | 
			
		||||
    </style>
 | 
			
		||||
  </head>
 | 
			
		||||
  <body>
 | 
			
		||||
    <h1 id="t">SSF Tests</h1>
 | 
			
		||||
    <div id="mocha"></div>
 | 
			
		||||
    <script src="shim.js"></script>
 | 
			
		||||
		<script src="json2.js"></script>
 | 
			
		||||
    <script src="mocha.js"></script>
 | 
			
		||||
    <script>
 | 
			
		||||
      window.initMochaPhantomJS && window.initMochaPhantomJS();
 | 
			
		||||
      mocha.setup('bdd');
 | 
			
		||||
    </script>
 | 
			
		||||
    <script src="test.js"></script>
 | 
			
		||||
    <script>
 | 
			
		||||
      if(typeof mochaSaucePlease !== "undefined") mochaSaucePlease();
 | 
			
		||||
      else if(window.mochaPhantomJS) mochaPhantomJS.run();
 | 
			
		||||
      else mocha.run();
 | 
			
		||||
    </script>
 | 
			
		||||
  </body>
 | 
			
		||||
</html>
 | 
			
		||||
 | 
			
		||||
							
								
								
									
										489
									
								
								ctest/json2.js
									
									
									
									
									
										Normal file
									
								
							
							
								
								
								
								
								
									
									
								
							
						
						
									
										489
									
								
								ctest/json2.js
									
									
									
									
									
										Normal file
									
								
							@ -0,0 +1,489 @@
 | 
			
		||||
/*
 | 
			
		||||
    json2.js
 | 
			
		||||
    2014-02-04
 | 
			
		||||
 | 
			
		||||
    Public Domain.
 | 
			
		||||
 | 
			
		||||
    NO WARRANTY EXPRESSED OR IMPLIED. USE AT YOUR OWN RISK.
 | 
			
		||||
 | 
			
		||||
    See http://www.JSON.org/js.html
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
    This code should be minified before deployment.
 | 
			
		||||
    See http://javascript.crockford.com/jsmin.html
 | 
			
		||||
 | 
			
		||||
    USE YOUR OWN COPY. IT IS EXTREMELY UNWISE TO LOAD CODE FROM SERVERS YOU DO
 | 
			
		||||
    NOT CONTROL.
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
    This file creates a global JSON object containing two methods: stringify
 | 
			
		||||
    and parse.
 | 
			
		||||
 | 
			
		||||
        JSON.stringify(value, replacer, space)
 | 
			
		||||
            value       any JavaScript value, usually an object or array.
 | 
			
		||||
 | 
			
		||||
            replacer    an optional parameter that determines how object
 | 
			
		||||
                        values are stringified for objects. It can be a
 | 
			
		||||
                        function or an array of strings.
 | 
			
		||||
 | 
			
		||||
            space       an optional parameter that specifies the indentation
 | 
			
		||||
                        of nested structures. If it is omitted, the text will
 | 
			
		||||
                        be packed without extra whitespace. If it is a number,
 | 
			
		||||
                        it will specify the number of spaces to indent at each
 | 
			
		||||
                        level. If it is a string (such as '\t' or ' '),
 | 
			
		||||
                        it contains the characters used to indent at each level.
 | 
			
		||||
 | 
			
		||||
            This method produces a JSON text from a JavaScript value.
 | 
			
		||||
 | 
			
		||||
            When an object value is found, if the object contains a toJSON
 | 
			
		||||
            method, its toJSON method will be called and the result will be
 | 
			
		||||
            stringified. A toJSON method does not serialize: it returns the
 | 
			
		||||
            value represented by the name/value pair that should be serialized,
 | 
			
		||||
            or undefined if nothing should be serialized. The toJSON method
 | 
			
		||||
            will be passed the key associated with the value, and this will be
 | 
			
		||||
            bound to the value
 | 
			
		||||
 | 
			
		||||
            For example, this would serialize Dates as ISO strings.
 | 
			
		||||
 | 
			
		||||
                Date.prototype.toJSON = function (key) {
 | 
			
		||||
                    function f(n) {
 | 
			
		||||
                        // Format integers to have at least two digits.
 | 
			
		||||
                        return n < 10 ? '0' + n : n;
 | 
			
		||||
                    }
 | 
			
		||||
 | 
			
		||||
                    return this.getUTCFullYear()   + '-' +
 | 
			
		||||
                         f(this.getUTCMonth() + 1) + '-' +
 | 
			
		||||
                         f(this.getUTCDate())      + 'T' +
 | 
			
		||||
                         f(this.getUTCHours())     + ':' +
 | 
			
		||||
                         f(this.getUTCMinutes())   + ':' +
 | 
			
		||||
                         f(this.getUTCSeconds())   + 'Z';
 | 
			
		||||
                };
 | 
			
		||||
 | 
			
		||||
            You can provide an optional replacer method. It will be passed the
 | 
			
		||||
            key and value of each member, with this bound to the containing
 | 
			
		||||
            object. The value that is returned from your method will be
 | 
			
		||||
            serialized. If your method returns undefined, then the member will
 | 
			
		||||
            be excluded from the serialization.
 | 
			
		||||
 | 
			
		||||
            If the replacer parameter is an array of strings, then it will be
 | 
			
		||||
            used to select the members to be serialized. It filters the results
 | 
			
		||||
            such that only members with keys listed in the replacer array are
 | 
			
		||||
            stringified.
 | 
			
		||||
 | 
			
		||||
            Values that do not have JSON representations, such as undefined or
 | 
			
		||||
            functions, will not be serialized. Such values in objects will be
 | 
			
		||||
            dropped; in arrays they will be replaced with null. You can use
 | 
			
		||||
            a replacer function to replace those with JSON values.
 | 
			
		||||
            JSON.stringify(undefined) returns undefined.
 | 
			
		||||
 | 
			
		||||
            The optional space parameter produces a stringification of the
 | 
			
		||||
            value that is filled with line breaks and indentation to make it
 | 
			
		||||
            easier to read.
 | 
			
		||||
 | 
			
		||||
            If the space parameter is a non-empty string, then that string will
 | 
			
		||||
            be used for indentation. If the space parameter is a number, then
 | 
			
		||||
            the indentation will be that many spaces.
 | 
			
		||||
 | 
			
		||||
            Example:
 | 
			
		||||
 | 
			
		||||
            text = JSON.stringify(['e', {pluribus: 'unum'}]);
 | 
			
		||||
            // text is '["e",{"pluribus":"unum"}]'
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
            text = JSON.stringify(['e', {pluribus: 'unum'}], null, '\t');
 | 
			
		||||
            // text is '[\n\t"e",\n\t{\n\t\t"pluribus": "unum"\n\t}\n]'
 | 
			
		||||
 | 
			
		||||
            text = JSON.stringify([new Date()], function (key, value) {
 | 
			
		||||
                return this[key] instanceof Date ?
 | 
			
		||||
                    'Date(' + this[key] + ')' : value;
 | 
			
		||||
            });
 | 
			
		||||
            // text is '["Date(---current time---)"]'
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
        JSON.parse(text, reviver)
 | 
			
		||||
            This method parses a JSON text to produce an object or array.
 | 
			
		||||
            It can throw a SyntaxError exception.
 | 
			
		||||
 | 
			
		||||
            The optional reviver parameter is a function that can filter and
 | 
			
		||||
            transform the results. It receives each of the keys and values,
 | 
			
		||||
            and its return value is used instead of the original value.
 | 
			
		||||
            If it returns what it received, then the structure is not modified.
 | 
			
		||||
            If it returns undefined then the member is deleted.
 | 
			
		||||
 | 
			
		||||
            Example:
 | 
			
		||||
 | 
			
		||||
            // Parse the text. Values that look like ISO date strings will
 | 
			
		||||
            // be converted to Date objects.
 | 
			
		||||
 | 
			
		||||
            myData = JSON.parse(text, function (key, value) {
 | 
			
		||||
                var a;
 | 
			
		||||
                if (typeof value === 'string') {
 | 
			
		||||
                    a =
 | 
			
		||||
/^(\d{4})-(\d{2})-(\d{2})T(\d{2}):(\d{2}):(\d{2}(?:\.\d*)?)Z$/.exec(value);
 | 
			
		||||
                    if (a) {
 | 
			
		||||
                        return new Date(Date.UTC(+a[1], +a[2] - 1, +a[3], +a[4],
 | 
			
		||||
                            +a[5], +a[6]));
 | 
			
		||||
                    }
 | 
			
		||||
                }
 | 
			
		||||
                return value;
 | 
			
		||||
            });
 | 
			
		||||
 | 
			
		||||
            myData = JSON.parse('["Date(09/09/2001)"]', function (key, value) {
 | 
			
		||||
                var d;
 | 
			
		||||
                if (typeof value === 'string' &&
 | 
			
		||||
                        value.slice(0, 5) === 'Date(' &&
 | 
			
		||||
                        value.slice(-1) === ')') {
 | 
			
		||||
                    d = new Date(value.slice(5, -1));
 | 
			
		||||
                    if (d) {
 | 
			
		||||
                        return d;
 | 
			
		||||
                    }
 | 
			
		||||
                }
 | 
			
		||||
                return value;
 | 
			
		||||
            });
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
    This is a reference implementation. You are free to copy, modify, or
 | 
			
		||||
    redistribute.
 | 
			
		||||
*/
 | 
			
		||||
 | 
			
		||||
/*jslint evil: true, regexp: true */
 | 
			
		||||
 | 
			
		||||
/*members "", "\b", "\t", "\n", "\f", "\r", "\"", JSON, "\\", apply,
 | 
			
		||||
    call, charCodeAt, getUTCDate, getUTCFullYear, getUTCHours,
 | 
			
		||||
    getUTCMinutes, getUTCMonth, getUTCSeconds, hasOwnProperty, join,
 | 
			
		||||
    lastIndex, length, parse, prototype, push, replace, slice, stringify,
 | 
			
		||||
    test, toJSON, toString, valueOf
 | 
			
		||||
*/
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
// Create a JSON object only if one does not already exist. We create the
 | 
			
		||||
// methods in a closure to avoid creating global variables.
 | 
			
		||||
 | 
			
		||||
if (typeof JSON !== 'object') {
 | 
			
		||||
    JSON = {};
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
(function () {
 | 
			
		||||
    'use strict';
 | 
			
		||||
 | 
			
		||||
    function f(n) {
 | 
			
		||||
        // Format integers to have at least two digits.
 | 
			
		||||
        return n < 10 ? '0' + n : n;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    if (typeof Date.prototype.toJSON !== 'function') {
 | 
			
		||||
 | 
			
		||||
        Date.prototype.toJSON = function () {
 | 
			
		||||
 | 
			
		||||
            return isFinite(this.valueOf())
 | 
			
		||||
                ? this.getUTCFullYear()     + '-' +
 | 
			
		||||
                    f(this.getUTCMonth() + 1) + '-' +
 | 
			
		||||
                    f(this.getUTCDate())      + 'T' +
 | 
			
		||||
                    f(this.getUTCHours())     + ':' +
 | 
			
		||||
                    f(this.getUTCMinutes())   + ':' +
 | 
			
		||||
                    f(this.getUTCSeconds())   + 'Z'
 | 
			
		||||
                : null;
 | 
			
		||||
        };
 | 
			
		||||
 | 
			
		||||
        String.prototype.toJSON      =
 | 
			
		||||
            Number.prototype.toJSON  =
 | 
			
		||||
            Boolean.prototype.toJSON = function () {
 | 
			
		||||
                return this.valueOf();
 | 
			
		||||
            };
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    var cx,
 | 
			
		||||
        escapable,
 | 
			
		||||
        gap,
 | 
			
		||||
        indent,
 | 
			
		||||
        meta,
 | 
			
		||||
        rep;
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
    function quote(string) {
 | 
			
		||||
 | 
			
		||||
// If the string contains no control characters, no quote characters, and no
 | 
			
		||||
// backslash characters, then we can safely slap some quotes around it.
 | 
			
		||||
// Otherwise we must also replace the offending characters with safe escape
 | 
			
		||||
// sequences.
 | 
			
		||||
 | 
			
		||||
        escapable.lastIndex = 0;
 | 
			
		||||
        return escapable.test(string) ? '"' + string.replace(escapable, function (a) {
 | 
			
		||||
            var c = meta[a];
 | 
			
		||||
            return typeof c === 'string'
 | 
			
		||||
                ? c
 | 
			
		||||
                : '\\u' + ('0000' + a.charCodeAt(0).toString(16)).slice(-4);
 | 
			
		||||
        }) + '"' : '"' + string + '"';
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
    function str(key, holder) {
 | 
			
		||||
 | 
			
		||||
// Produce a string from holder[key].
 | 
			
		||||
 | 
			
		||||
        var i,          // The loop counter.
 | 
			
		||||
            k,          // The member key.
 | 
			
		||||
            v,          // The member value.
 | 
			
		||||
            length,
 | 
			
		||||
            mind = gap,
 | 
			
		||||
            partial,
 | 
			
		||||
            value = holder[key];
 | 
			
		||||
 | 
			
		||||
// If the value has a toJSON method, call it to obtain a replacement value.
 | 
			
		||||
 | 
			
		||||
        if (value && typeof value === 'object' &&
 | 
			
		||||
                typeof value.toJSON === 'function') {
 | 
			
		||||
            value = value.toJSON(key);
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
// If we were called with a replacer function, then call the replacer to
 | 
			
		||||
// obtain a replacement value.
 | 
			
		||||
 | 
			
		||||
        if (typeof rep === 'function') {
 | 
			
		||||
            value = rep.call(holder, key, value);
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
// What happens next depends on the value's type.
 | 
			
		||||
 | 
			
		||||
        switch (typeof value) {
 | 
			
		||||
        case 'string':
 | 
			
		||||
            return quote(value);
 | 
			
		||||
 | 
			
		||||
        case 'number':
 | 
			
		||||
 | 
			
		||||
// JSON numbers must be finite. Encode non-finite numbers as null.
 | 
			
		||||
 | 
			
		||||
            return isFinite(value) ? String(value) : 'null';
 | 
			
		||||
 | 
			
		||||
        case 'boolean':
 | 
			
		||||
        case 'null':
 | 
			
		||||
 | 
			
		||||
// If the value is a boolean or null, convert it to a string. Note:
 | 
			
		||||
// typeof null does not produce 'null'. The case is included here in
 | 
			
		||||
// the remote chance that this gets fixed someday.
 | 
			
		||||
 | 
			
		||||
            return String(value);
 | 
			
		||||
 | 
			
		||||
// If the type is 'object', we might be dealing with an object or an array or
 | 
			
		||||
// null.
 | 
			
		||||
 | 
			
		||||
        case 'object':
 | 
			
		||||
 | 
			
		||||
// Due to a specification blunder in ECMAScript, typeof null is 'object',
 | 
			
		||||
// so watch out for that case.
 | 
			
		||||
 | 
			
		||||
            if (!value) {
 | 
			
		||||
                return 'null';
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
// Make an array to hold the partial results of stringifying this object value.
 | 
			
		||||
 | 
			
		||||
            gap += indent;
 | 
			
		||||
            partial = [];
 | 
			
		||||
 | 
			
		||||
// Is the value an array?
 | 
			
		||||
 | 
			
		||||
            if (Object.prototype.toString.apply(value) === '[object Array]') {
 | 
			
		||||
 | 
			
		||||
// The value is an array. Stringify every element. Use null as a placeholder
 | 
			
		||||
// for non-JSON values.
 | 
			
		||||
 | 
			
		||||
                length = value.length;
 | 
			
		||||
                for (i = 0; i < length; i += 1) {
 | 
			
		||||
                    partial[i] = str(i, value) || 'null';
 | 
			
		||||
                }
 | 
			
		||||
 | 
			
		||||
// Join all of the elements together, separated with commas, and wrap them in
 | 
			
		||||
// brackets.
 | 
			
		||||
 | 
			
		||||
                v = partial.length === 0
 | 
			
		||||
                    ? '[]'
 | 
			
		||||
                    : gap
 | 
			
		||||
                    ? '[\n' + gap + partial.join(',\n' + gap) + '\n' + mind + ']'
 | 
			
		||||
                    : '[' + partial.join(',') + ']';
 | 
			
		||||
                gap = mind;
 | 
			
		||||
                return v;
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
// If the replacer is an array, use it to select the members to be stringified.
 | 
			
		||||
 | 
			
		||||
            if (rep && typeof rep === 'object') {
 | 
			
		||||
                length = rep.length;
 | 
			
		||||
                for (i = 0; i < length; i += 1) {
 | 
			
		||||
                    if (typeof rep[i] === 'string') {
 | 
			
		||||
                        k = rep[i];
 | 
			
		||||
                        v = str(k, value);
 | 
			
		||||
                        if (v) {
 | 
			
		||||
                            partial.push(quote(k) + (gap ? ': ' : ':') + v);
 | 
			
		||||
                        }
 | 
			
		||||
                    }
 | 
			
		||||
                }
 | 
			
		||||
            } else {
 | 
			
		||||
 | 
			
		||||
// Otherwise, iterate through all of the keys in the object.
 | 
			
		||||
 | 
			
		||||
                for (k in value) {
 | 
			
		||||
                    if (Object.prototype.hasOwnProperty.call(value, k)) {
 | 
			
		||||
                        v = str(k, value);
 | 
			
		||||
                        if (v) {
 | 
			
		||||
                            partial.push(quote(k) + (gap ? ': ' : ':') + v);
 | 
			
		||||
                        }
 | 
			
		||||
                    }
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
// Join all of the member texts together, separated with commas,
 | 
			
		||||
// and wrap them in braces.
 | 
			
		||||
 | 
			
		||||
            v = partial.length === 0
 | 
			
		||||
                ? '{}'
 | 
			
		||||
                : gap
 | 
			
		||||
                ? '{\n' + gap + partial.join(',\n' + gap) + '\n' + mind + '}'
 | 
			
		||||
                : '{' + partial.join(',') + '}';
 | 
			
		||||
            gap = mind;
 | 
			
		||||
            return v;
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
// If the JSON object does not yet have a stringify method, give it one.
 | 
			
		||||
 | 
			
		||||
    if (typeof JSON.stringify !== 'function') {
 | 
			
		||||
        escapable = /[\\\"\x00-\x1f\x7f-\x9f\u00ad\u0600-\u0604\u070f\u17b4\u17b5\u200c-\u200f\u2028-\u202f\u2060-\u206f\ufeff\ufff0-\uffff]/g;
 | 
			
		||||
        meta = {    // table of character substitutions
 | 
			
		||||
            '\b': '\\b',
 | 
			
		||||
            '\t': '\\t',
 | 
			
		||||
            '\n': '\\n',
 | 
			
		||||
            '\f': '\\f',
 | 
			
		||||
            '\r': '\\r',
 | 
			
		||||
            '"' : '\\"',
 | 
			
		||||
            '\\': '\\\\'
 | 
			
		||||
        };
 | 
			
		||||
        JSON.stringify = function (value, replacer, space) {
 | 
			
		||||
 | 
			
		||||
// The stringify method takes a value and an optional replacer, and an optional
 | 
			
		||||
// space parameter, and returns a JSON text. The replacer can be a function
 | 
			
		||||
// that can replace values, or an array of strings that will select the keys.
 | 
			
		||||
// A default replacer method can be provided. Use of the space parameter can
 | 
			
		||||
// produce text that is more easily readable.
 | 
			
		||||
 | 
			
		||||
            var i;
 | 
			
		||||
            gap = '';
 | 
			
		||||
            indent = '';
 | 
			
		||||
 | 
			
		||||
// If the space parameter is a number, make an indent string containing that
 | 
			
		||||
// many spaces.
 | 
			
		||||
 | 
			
		||||
            if (typeof space === 'number') {
 | 
			
		||||
                for (i = 0; i < space; i += 1) {
 | 
			
		||||
                    indent += ' ';
 | 
			
		||||
                }
 | 
			
		||||
 | 
			
		||||
// If the space parameter is a string, it will be used as the indent string.
 | 
			
		||||
 | 
			
		||||
            } else if (typeof space === 'string') {
 | 
			
		||||
                indent = space;
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
// If there is a replacer, it must be a function or an array.
 | 
			
		||||
// Otherwise, throw an error.
 | 
			
		||||
 | 
			
		||||
            rep = replacer;
 | 
			
		||||
            if (replacer && typeof replacer !== 'function' &&
 | 
			
		||||
                    (typeof replacer !== 'object' ||
 | 
			
		||||
                    typeof replacer.length !== 'number')) {
 | 
			
		||||
                throw new Error('JSON.stringify');
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
// Make a fake root object containing our value under the key of ''.
 | 
			
		||||
// Return the result of stringifying the value.
 | 
			
		||||
 | 
			
		||||
            return str('', {'': value});
 | 
			
		||||
        };
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
// If the JSON object does not yet have a parse method, give it one.
 | 
			
		||||
 | 
			
		||||
    if (typeof JSON.parse !== 'function') {
 | 
			
		||||
        cx = /[\u0000\u00ad\u0600-\u0604\u070f\u17b4\u17b5\u200c-\u200f\u2028-\u202f\u2060-\u206f\ufeff\ufff0-\uffff]/g;
 | 
			
		||||
        JSON.parse = function (text, reviver) {
 | 
			
		||||
 | 
			
		||||
// The parse method takes a text and an optional reviver function, and returns
 | 
			
		||||
// a JavaScript value if the text is a valid JSON text.
 | 
			
		||||
 | 
			
		||||
            var j;
 | 
			
		||||
 | 
			
		||||
            function walk(holder, key) {
 | 
			
		||||
 | 
			
		||||
// The walk method is used to recursively walk the resulting structure so
 | 
			
		||||
// that modifications can be made.
 | 
			
		||||
 | 
			
		||||
                var k, v, value = holder[key];
 | 
			
		||||
                if (value && typeof value === 'object') {
 | 
			
		||||
                    for (k in value) {
 | 
			
		||||
                        if (Object.prototype.hasOwnProperty.call(value, k)) {
 | 
			
		||||
                            v = walk(value, k);
 | 
			
		||||
                            if (v !== undefined) {
 | 
			
		||||
                                value[k] = v;
 | 
			
		||||
                            } else {
 | 
			
		||||
                                delete value[k];
 | 
			
		||||
                            }
 | 
			
		||||
                        }
 | 
			
		||||
                    }
 | 
			
		||||
                }
 | 
			
		||||
                return reviver.call(holder, key, value);
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
// Parsing happens in four stages. In the first stage, we replace certain
 | 
			
		||||
// Unicode characters with escape sequences. JavaScript handles many characters
 | 
			
		||||
// incorrectly, either silently deleting them, or treating them as line endings.
 | 
			
		||||
 | 
			
		||||
            text = String(text);
 | 
			
		||||
            cx.lastIndex = 0;
 | 
			
		||||
            if (cx.test(text)) {
 | 
			
		||||
                text = text.replace(cx, function (a) {
 | 
			
		||||
                    return '\\u' +
 | 
			
		||||
                        ('0000' + a.charCodeAt(0).toString(16)).slice(-4);
 | 
			
		||||
                });
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
// In the second stage, we run the text against regular expressions that look
 | 
			
		||||
// for non-JSON patterns. We are especially concerned with '()' and 'new'
 | 
			
		||||
// because they can cause invocation, and '=' because it can cause mutation.
 | 
			
		||||
// But just to be safe, we want to reject all unexpected forms.
 | 
			
		||||
 | 
			
		||||
// We split the second stage into 4 regexp operations in order to work around
 | 
			
		||||
// crippling inefficiencies in IE's and Safari's regexp engines. First we
 | 
			
		||||
// replace the JSON backslash pairs with '@' (a non-JSON character). Second, we
 | 
			
		||||
// replace all simple value tokens with ']' characters. Third, we delete all
 | 
			
		||||
// open brackets that follow a colon or comma or that begin the text. Finally,
 | 
			
		||||
// we look to see that the remaining characters are only whitespace or ']' or
 | 
			
		||||
// ',' or ':' or '{' or '}'. If that is so, then the text is safe for eval.
 | 
			
		||||
 | 
			
		||||
            if (/^[\],:{}\s]*$/
 | 
			
		||||
                    .test(text.replace(/\\(?:["\\\/bfnrt]|u[0-9a-fA-F]{4})/g, '@')
 | 
			
		||||
                        .replace(/"[^"\\\n\r]*"|true|false|null|-?\d+(?:\.\d*)?(?:[eE][+\-]?\d+)?/g, ']')
 | 
			
		||||
                        .replace(/(?:^|:|,)(?:\s*\[)+/g, ''))) {
 | 
			
		||||
 | 
			
		||||
// In the third stage we use the eval function to compile the text into a
 | 
			
		||||
// JavaScript structure. The '{' operator is subject to a syntactic ambiguity
 | 
			
		||||
// in JavaScript: it can begin a block or an object literal. We wrap the text
 | 
			
		||||
// in parens to eliminate the ambiguity.
 | 
			
		||||
 | 
			
		||||
                j = eval('(' + text + ')');
 | 
			
		||||
 | 
			
		||||
// In the optional fourth stage, we recursively walk the new structure, passing
 | 
			
		||||
// each name/value pair to a reviver function for possible transformation.
 | 
			
		||||
 | 
			
		||||
                return typeof reviver === 'function'
 | 
			
		||||
                    ? walk({'': j}, '')
 | 
			
		||||
                    : j;
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
// If the text is not JSON parseable, then a SyntaxError is thrown.
 | 
			
		||||
 | 
			
		||||
            throw new SyntaxError('JSON.parse');
 | 
			
		||||
        };
 | 
			
		||||
    }
 | 
			
		||||
}());
 | 
			
		||||
							
								
								
									
										260
									
								
								ctest/mocha.css
									
									
									
									
									
										Normal file
									
								
							
							
								
								
								
								
								
									
									
								
							
						
						
									
										260
									
								
								ctest/mocha.css
									
									
									
									
									
										Normal file
									
								
							@ -0,0 +1,260 @@
 | 
			
		||||
@charset "utf-8";
 | 
			
		||||
 | 
			
		||||
body {
 | 
			
		||||
  margin:0;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
#mocha {
 | 
			
		||||
  font: 20px/1.5 "Helvetica Neue", Helvetica, Arial, sans-serif;
 | 
			
		||||
  margin: 60px 50px;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
#mocha ul, #mocha li {
 | 
			
		||||
  margin: 0;
 | 
			
		||||
  padding: 0;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
#mocha ul {
 | 
			
		||||
  list-style: none;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
#mocha h1, #mocha h2 {
 | 
			
		||||
  margin: 0;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
#mocha h1 {
 | 
			
		||||
  margin-top: 15px;
 | 
			
		||||
  font-size: 1em;
 | 
			
		||||
  font-weight: 200;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
#mocha h1 a {
 | 
			
		||||
  text-decoration: none;
 | 
			
		||||
  color: inherit;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
#mocha h1 a:hover {
 | 
			
		||||
  text-decoration: underline;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
#mocha .suite .suite h1 {
 | 
			
		||||
  margin-top: 0;
 | 
			
		||||
  font-size: .8em;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
#mocha .hidden {
 | 
			
		||||
  display: none;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
#mocha h2 {
 | 
			
		||||
  font-size: 12px;
 | 
			
		||||
  font-weight: normal;
 | 
			
		||||
  cursor: pointer;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
#mocha .suite {
 | 
			
		||||
  margin-left: 15px;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
#mocha .test {
 | 
			
		||||
  margin-left: 15px;
 | 
			
		||||
  overflow: hidden;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
#mocha .test.pending:hover h2::after {
 | 
			
		||||
  content: '(pending)';
 | 
			
		||||
  font-family: arial, sans-serif;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
#mocha .test.pass.medium .duration {
 | 
			
		||||
  background: #C09853;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
#mocha .test.pass.slow .duration {
 | 
			
		||||
  background: #B94A48;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
#mocha .test.pass::before {
 | 
			
		||||
  content: '✓';
 | 
			
		||||
  font-size: 12px;
 | 
			
		||||
  display: block;
 | 
			
		||||
  float: left;
 | 
			
		||||
  margin-right: 5px;
 | 
			
		||||
  color: #00d6b2;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
#mocha .test.pass .duration {
 | 
			
		||||
  font-size: 9px;
 | 
			
		||||
  margin-left: 5px;
 | 
			
		||||
  padding: 2px 5px;
 | 
			
		||||
  color: white;
 | 
			
		||||
  -webkit-box-shadow: inset 0 1px 1px rgba(0,0,0,.2);
 | 
			
		||||
  -moz-box-shadow: inset 0 1px 1px rgba(0,0,0,.2);
 | 
			
		||||
  box-shadow: inset 0 1px 1px rgba(0,0,0,.2);
 | 
			
		||||
  -webkit-border-radius: 5px;
 | 
			
		||||
  -moz-border-radius: 5px;
 | 
			
		||||
  -ms-border-radius: 5px;
 | 
			
		||||
  -o-border-radius: 5px;
 | 
			
		||||
  border-radius: 5px;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
#mocha .test.pass.fast .duration {
 | 
			
		||||
  display: none;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
#mocha .test.pending {
 | 
			
		||||
  color: #0b97c4;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
#mocha .test.pending::before {
 | 
			
		||||
  content: '◦';
 | 
			
		||||
  color: #0b97c4;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
#mocha .test.fail {
 | 
			
		||||
  color: #c00;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
#mocha .test.fail pre {
 | 
			
		||||
  color: black;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
#mocha .test.fail::before {
 | 
			
		||||
  content: '✖';
 | 
			
		||||
  font-size: 12px;
 | 
			
		||||
  display: block;
 | 
			
		||||
  float: left;
 | 
			
		||||
  margin-right: 5px;
 | 
			
		||||
  color: #c00;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
#mocha .test pre.error {
 | 
			
		||||
  color: #c00;
 | 
			
		||||
  max-height: 300px;
 | 
			
		||||
  overflow: auto;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
#mocha .test pre {
 | 
			
		||||
  display: block;
 | 
			
		||||
  float: left;
 | 
			
		||||
  clear: left;
 | 
			
		||||
  font: 12px/1.5 monaco, monospace;
 | 
			
		||||
  margin: 5px;
 | 
			
		||||
  padding: 15px;
 | 
			
		||||
  border: 1px solid #eee;
 | 
			
		||||
  border-bottom-color: #ddd;
 | 
			
		||||
  -webkit-border-radius: 3px;
 | 
			
		||||
  -webkit-box-shadow: 0 1px 3px #eee;
 | 
			
		||||
  -moz-border-radius: 3px;
 | 
			
		||||
  -moz-box-shadow: 0 1px 3px #eee;
 | 
			
		||||
  border-radius: 3px;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
#mocha .test h2 {
 | 
			
		||||
  position: relative;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
#mocha .test a.replay {
 | 
			
		||||
  position: absolute;
 | 
			
		||||
  top: 3px;
 | 
			
		||||
  right: 0;
 | 
			
		||||
  text-decoration: none;
 | 
			
		||||
  vertical-align: middle;
 | 
			
		||||
  display: block;
 | 
			
		||||
  width: 15px;
 | 
			
		||||
  height: 15px;
 | 
			
		||||
  line-height: 15px;
 | 
			
		||||
  text-align: center;
 | 
			
		||||
  background: #eee;
 | 
			
		||||
  font-size: 15px;
 | 
			
		||||
  -moz-border-radius: 15px;
 | 
			
		||||
  border-radius: 15px;
 | 
			
		||||
  -webkit-transition: opacity 200ms;
 | 
			
		||||
  -moz-transition: opacity 200ms;
 | 
			
		||||
  transition: opacity 200ms;
 | 
			
		||||
  opacity: 0.3;
 | 
			
		||||
  color: #888;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
#mocha .test:hover a.replay {
 | 
			
		||||
  opacity: 1;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
#mocha-report.pass .test.fail {
 | 
			
		||||
  display: none;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
#mocha-report.fail .test.pass {
 | 
			
		||||
  display: none;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
#mocha-report.pending .test.pass,
 | 
			
		||||
#mocha-report.pending .test.fail {
 | 
			
		||||
  display: none;
 | 
			
		||||
}
 | 
			
		||||
#mocha-report.pending .test.pass.pending {
 | 
			
		||||
  display: block;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
#mocha-error {
 | 
			
		||||
  color: #c00;
 | 
			
		||||
  font-size: 1.5em;
 | 
			
		||||
  font-weight: 100;
 | 
			
		||||
  letter-spacing: 1px;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
#mocha-stats {
 | 
			
		||||
  position: fixed;
 | 
			
		||||
  top: 15px;
 | 
			
		||||
  right: 10px;
 | 
			
		||||
  font-size: 12px;
 | 
			
		||||
  margin: 0;
 | 
			
		||||
  color: #888;
 | 
			
		||||
  z-index: 1;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
#mocha-stats .progress {
 | 
			
		||||
  float: right;
 | 
			
		||||
  padding-top: 0;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
#mocha-stats em {
 | 
			
		||||
  color: black;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
#mocha-stats a {
 | 
			
		||||
  text-decoration: none;
 | 
			
		||||
  color: inherit;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
#mocha-stats a:hover {
 | 
			
		||||
  border-bottom: 1px solid #eee;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
#mocha-stats li {
 | 
			
		||||
  display: inline-block;
 | 
			
		||||
  margin: 0 5px;
 | 
			
		||||
  list-style: none;
 | 
			
		||||
  padding-top: 11px;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
#mocha-stats canvas {
 | 
			
		||||
  width: 40px;
 | 
			
		||||
  height: 40px;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
#mocha code .comment { color: #ddd }
 | 
			
		||||
#mocha code .init { color: #2F6FAD }
 | 
			
		||||
#mocha code .string { color: #5890AD }
 | 
			
		||||
#mocha code .keyword { color: #8A6343 }
 | 
			
		||||
#mocha code .number { color: #2F6FAD }
 | 
			
		||||
 | 
			
		||||
@media screen and (max-device-width: 480px) {
 | 
			
		||||
  #mocha  {
 | 
			
		||||
    margin: 60px 0px;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  #mocha #stats {
 | 
			
		||||
    position: absolute;
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										5554
									
								
								ctest/mocha.js
									
									
									
									
									
										Normal file
									
								
							
							
								
								
								
								
								
									
									
								
							
						
						
									
										5554
									
								
								ctest/mocha.js
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										330
									
								
								ctest/shim.js
									
									
									
									
									
										Normal file
									
								
							
							
								
								
								
								
								
									
									
								
							
						
						
									
										330
									
								
								ctest/shim.js
									
									
									
									
									
										Normal file
									
								
							@ -0,0 +1,330 @@
 | 
			
		||||
// From https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Object/keys
 | 
			
		||||
if (!Object.keys) {
 | 
			
		||||
  Object.keys = (function () {
 | 
			
		||||
    var hasOwnProperty = Object.prototype.hasOwnProperty,
 | 
			
		||||
        hasDontEnumBug = !({toString: null}).propertyIsEnumerable('toString'),
 | 
			
		||||
        dontEnums = [
 | 
			
		||||
          'toString',
 | 
			
		||||
          'toLocaleString',
 | 
			
		||||
          'valueOf',
 | 
			
		||||
          'hasOwnProperty',
 | 
			
		||||
          'isPrototypeOf',
 | 
			
		||||
          'propertyIsEnumerable',
 | 
			
		||||
          'constructor'
 | 
			
		||||
        ],
 | 
			
		||||
        dontEnumsLength = dontEnums.length;
 | 
			
		||||
 | 
			
		||||
    return function (obj) {
 | 
			
		||||
      if (typeof obj !== 'object' && typeof obj !== 'function' || obj === null) throw new TypeError('Object.keys called on non-object');
 | 
			
		||||
 | 
			
		||||
      var result = [];
 | 
			
		||||
 | 
			
		||||
      for (var prop in obj) {
 | 
			
		||||
        if (hasOwnProperty.call(obj, prop)) result.push(prop);
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
      if (hasDontEnumBug) {
 | 
			
		||||
        for (var i=0; i < dontEnumsLength; i++) {
 | 
			
		||||
          if (hasOwnProperty.call(obj, dontEnums[i])) result.push(dontEnums[i]);
 | 
			
		||||
        }
 | 
			
		||||
      }
 | 
			
		||||
      return result;
 | 
			
		||||
    };
 | 
			
		||||
  })();
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// From https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/filter
 | 
			
		||||
if (!Array.prototype.filter)
 | 
			
		||||
{
 | 
			
		||||
  Array.prototype.filter = function(fun /*, thisp */)
 | 
			
		||||
  {
 | 
			
		||||
    "use strict";
 | 
			
		||||
 | 
			
		||||
    if (this == null)
 | 
			
		||||
      throw new TypeError();
 | 
			
		||||
 | 
			
		||||
    var t = Object(this);
 | 
			
		||||
    var len = t.length >>> 0;
 | 
			
		||||
    if (typeof fun != "function")
 | 
			
		||||
      throw new TypeError();
 | 
			
		||||
 | 
			
		||||
    var res = [];
 | 
			
		||||
    var thisp = arguments[1];
 | 
			
		||||
    for (var i = 0; i < len; i++)
 | 
			
		||||
    {
 | 
			
		||||
      if (i in t)
 | 
			
		||||
      {
 | 
			
		||||
        var val = t[i]; // in case fun mutates this
 | 
			
		||||
        if (fun.call(thisp, val, i, t))
 | 
			
		||||
          res.push(val);
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    return res;
 | 
			
		||||
  };
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// From https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/String/Trim
 | 
			
		||||
if (!String.prototype.trim) {
 | 
			
		||||
  String.prototype.trim = function () {
 | 
			
		||||
    return this.replace(/^\s+|\s+$/g, '');
 | 
			
		||||
  };
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// From https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/forEach
 | 
			
		||||
if (!Array.prototype.forEach)
 | 
			
		||||
{
 | 
			
		||||
  Array.prototype.forEach = function(fun /*, thisArg */)
 | 
			
		||||
  {
 | 
			
		||||
    "use strict";
 | 
			
		||||
 | 
			
		||||
    if (this === void 0 || this === null)
 | 
			
		||||
      throw new TypeError();
 | 
			
		||||
 | 
			
		||||
    var t = Object(this);
 | 
			
		||||
    var len = t.length >>> 0;
 | 
			
		||||
    if (typeof fun !== "function")
 | 
			
		||||
      throw new TypeError();
 | 
			
		||||
 | 
			
		||||
    var thisArg = arguments.length >= 2 ? arguments[1] : void 0;
 | 
			
		||||
    for (var i = 0; i < len; i++)
 | 
			
		||||
    {
 | 
			
		||||
      if (i in t)
 | 
			
		||||
        fun.call(thisArg, t[i], i, t);
 | 
			
		||||
    }
 | 
			
		||||
  };
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Production steps of ECMA-262, Edition 5, 15.4.4.19
 | 
			
		||||
// Reference: http://es5.github.com/#x15.4.4.19
 | 
			
		||||
if (!Array.prototype.map) {
 | 
			
		||||
  Array.prototype.map = function(callback, thisArg) {
 | 
			
		||||
 | 
			
		||||
    var T, A, k;
 | 
			
		||||
 | 
			
		||||
    if (this == null) {
 | 
			
		||||
      throw new TypeError(" this is null or not defined");
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    // 1. Let O be the result of calling ToObject passing the |this| value as the argument.
 | 
			
		||||
    var O = Object(this);
 | 
			
		||||
 | 
			
		||||
    // 2. Let lenValue be the result of calling the Get internal method of O with the argument "length".
 | 
			
		||||
    // 3. Let len be ToUint32(lenValue).
 | 
			
		||||
    var len = O.length >>> 0;
 | 
			
		||||
 | 
			
		||||
    // 4. If IsCallable(callback) is false, throw a TypeError exception.
 | 
			
		||||
    // See: http://es5.github.com/#x9.11
 | 
			
		||||
    if (typeof callback !== "function") {
 | 
			
		||||
      throw new TypeError(callback + " is not a function");
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    // 5. If thisArg was supplied, let T be thisArg; else let T be undefined.
 | 
			
		||||
    if (thisArg) {
 | 
			
		||||
      T = thisArg;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    // 6. Let A be a new array created as if by the expression new Array(len) where Array is
 | 
			
		||||
    // the standard built-in constructor with that name and len is the value of len.
 | 
			
		||||
    A = new Array(len);
 | 
			
		||||
 | 
			
		||||
    // 7. Let k be 0
 | 
			
		||||
    k = 0;
 | 
			
		||||
 | 
			
		||||
    // 8. Repeat, while k < len
 | 
			
		||||
    while(k < len) {
 | 
			
		||||
 | 
			
		||||
      var kValue, mappedValue;
 | 
			
		||||
 | 
			
		||||
      // a. Let Pk be ToString(k).
 | 
			
		||||
      //   This is implicit for LHS operands of the in operator
 | 
			
		||||
      // b. Let kPresent be the result of calling the HasProperty internal method of O with argument Pk.
 | 
			
		||||
      //   This step can be combined with c
 | 
			
		||||
      // c. If kPresent is true, then
 | 
			
		||||
      if (k in O) {
 | 
			
		||||
 | 
			
		||||
        // i. Let kValue be the result of calling the Get internal method of O with argument Pk.
 | 
			
		||||
        kValue = O[ k ];
 | 
			
		||||
 | 
			
		||||
        // ii. Let mappedValue be the result of calling the Call internal method of callback
 | 
			
		||||
        // with T as the this value and argument list containing kValue, k, and O.
 | 
			
		||||
        mappedValue = callback.call(T, kValue, k, O);
 | 
			
		||||
 | 
			
		||||
        // iii. Call the DefineOwnProperty internal method of A with arguments
 | 
			
		||||
        // Pk, Property Descriptor {Value: mappedValue, : true, Enumerable: true, Configurable: true},
 | 
			
		||||
        // and false.
 | 
			
		||||
 | 
			
		||||
        // In browsers that support Object.defineProperty, use the following:
 | 
			
		||||
        // Object.defineProperty(A, Pk, { value: mappedValue, writable: true, enumerable: true, configurable: true });
 | 
			
		||||
 | 
			
		||||
        // For best browser support, use the following:
 | 
			
		||||
        A[ k ] = mappedValue;
 | 
			
		||||
      }
 | 
			
		||||
      // d. Increase k by 1.
 | 
			
		||||
      k++;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    // 9. return A
 | 
			
		||||
    return A;
 | 
			
		||||
  };
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// From https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/indexOf
 | 
			
		||||
if (!Array.prototype.indexOf) {
 | 
			
		||||
  Array.prototype.indexOf = function (searchElement, fromIndex) {
 | 
			
		||||
    if ( this === undefined || this === null ) {
 | 
			
		||||
      throw new TypeError( '"this" is null or not defined' );
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    var length = this.length >>> 0; // Hack to convert object.length to a UInt32
 | 
			
		||||
 | 
			
		||||
    fromIndex = +fromIndex || 0;
 | 
			
		||||
 | 
			
		||||
    if (Math.abs(fromIndex) === Infinity) {
 | 
			
		||||
      fromIndex = 0;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    if (fromIndex < 0) {
 | 
			
		||||
      fromIndex += length;
 | 
			
		||||
      if (fromIndex < 0) {
 | 
			
		||||
        fromIndex = 0;
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    for (;fromIndex < length; fromIndex++) {
 | 
			
		||||
      if (this[fromIndex] === searchElement) {
 | 
			
		||||
        return fromIndex;
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    return -1;
 | 
			
		||||
  };
 | 
			
		||||
}
 | 
			
		||||
// Based on https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/isArray
 | 
			
		||||
 | 
			
		||||
if (! Array.isArray) {
 | 
			
		||||
    Array.isArray = function(obj) {
 | 
			
		||||
        return Object.prototype.toString.call(obj) === "[object Array]";
 | 
			
		||||
    };
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// https://github.com/ttaubert/node-arraybuffer-slice
 | 
			
		||||
// (c) 2013 Tim Taubert <tim@timtaubert.de>
 | 
			
		||||
// arraybuffer-slice may be freely distributed under the MIT license.
 | 
			
		||||
 | 
			
		||||
"use strict";
 | 
			
		||||
 | 
			
		||||
if (typeof ArrayBuffer !== 'undefined' && !ArrayBuffer.prototype.slice) {
 | 
			
		||||
  ArrayBuffer.prototype.slice = function (begin, end) {
 | 
			
		||||
    begin = (begin|0) || 0;
 | 
			
		||||
    var num = this.byteLength;
 | 
			
		||||
    end = end === (void 0) ? num : (end|0);
 | 
			
		||||
 | 
			
		||||
    // Handle negative values.
 | 
			
		||||
    if (begin < 0) begin += num;
 | 
			
		||||
    if (end < 0) end += num;
 | 
			
		||||
 | 
			
		||||
    if (num === 0 || begin >= num || begin >= end) {
 | 
			
		||||
      return new ArrayBuffer(0);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    var length = Math.min(num - begin, end - begin);
 | 
			
		||||
    var target = new ArrayBuffer(length);
 | 
			
		||||
    var targetArray = new Uint8Array(target);
 | 
			
		||||
    targetArray.set(new Uint8Array(this, begin, length));
 | 
			
		||||
    return target;
 | 
			
		||||
  };
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// https://github.com/davidchambers/Base64.js
 | 
			
		||||
// (C) 2015 David Chambers
 | 
			
		||||
// Base64.js may be freely distributed under the Apache 2.0 License.
 | 
			
		||||
 | 
			
		||||
;(function () {
 | 
			
		||||
 | 
			
		||||
  var object = typeof exports != 'undefined' ? exports : typeof self != 'undefined' ? self : eval("this");
 | 
			
		||||
  var chars = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/=';
 | 
			
		||||
 | 
			
		||||
  function InvalidCharacterError(message) {
 | 
			
		||||
    this.message = message;
 | 
			
		||||
  }
 | 
			
		||||
  InvalidCharacterError.prototype = new Error;
 | 
			
		||||
  InvalidCharacterError.prototype.name = 'InvalidCharacterError';
 | 
			
		||||
 | 
			
		||||
  // encoder
 | 
			
		||||
  // [https://gist.github.com/999166] by [https://github.com/nignag]
 | 
			
		||||
  object.btoa || (
 | 
			
		||||
  object.btoa = function (input) {
 | 
			
		||||
    var str = String(input);
 | 
			
		||||
    for (
 | 
			
		||||
      // initialize result and counter
 | 
			
		||||
      var block, charCode, idx = 0, map = chars, output = '';
 | 
			
		||||
      // if the next str index does not exist:
 | 
			
		||||
      //   change the mapping table to "="
 | 
			
		||||
      //   check if d has no fractional digits
 | 
			
		||||
      str.charAt(idx | 0) || (map = '=', idx % 1);
 | 
			
		||||
      // "8 - idx % 1 * 8" generates the sequence 2, 4, 6, 8
 | 
			
		||||
      output += map.charAt(63 & block >> 8 - idx % 1 * 8)
 | 
			
		||||
    ) {
 | 
			
		||||
      charCode = str.charCodeAt(idx += 3/4);
 | 
			
		||||
      if (charCode > 0xFF) {
 | 
			
		||||
        throw new InvalidCharacterError("'btoa' failed: The string to be encoded contains characters outside of the Latin1 range.");
 | 
			
		||||
      }
 | 
			
		||||
      block = block << 8 | charCode;
 | 
			
		||||
    }
 | 
			
		||||
    return output;
 | 
			
		||||
  });
 | 
			
		||||
 | 
			
		||||
  // decoder
 | 
			
		||||
  // [https://gist.github.com/1020396] by [https://github.com/atk]
 | 
			
		||||
  object.atob || (
 | 
			
		||||
  object.atob = function (input) {
 | 
			
		||||
    var str = String(input).replace(new RegExp("=+$"), '');
 | 
			
		||||
    if (str.length % 4 == 1) {
 | 
			
		||||
      throw new InvalidCharacterError("'atob' failed: The string to be decoded is not correctly encoded.");
 | 
			
		||||
    }
 | 
			
		||||
    for (
 | 
			
		||||
      // initialize result and counters
 | 
			
		||||
      var bc = 0, bs, buffer, idx = 0, output = '';
 | 
			
		||||
      // get next character
 | 
			
		||||
      buffer = str.charAt(idx++);
 | 
			
		||||
      // character found in table? initialize bit storage and add its ascii value;
 | 
			
		||||
      ~buffer && (bs = bc % 4 ? bs * 64 + buffer : buffer,
 | 
			
		||||
        // and if not first of each 4 characters,
 | 
			
		||||
        // convert the first 8 bits to one ascii character
 | 
			
		||||
        bc++ % 4) ? output += String.fromCharCode(255 & bs >> (-2 * bc & 6)) : 0
 | 
			
		||||
    ) {
 | 
			
		||||
      // try to find character in table (0-63, not found => -1)
 | 
			
		||||
      buffer = chars.indexOf(buffer);
 | 
			
		||||
    }
 | 
			
		||||
    return output;
 | 
			
		||||
  });
 | 
			
		||||
 | 
			
		||||
}());
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
// From https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Date/toISOString
 | 
			
		||||
if (!Date.prototype.toISOString) {
 | 
			
		||||
  (function() {
 | 
			
		||||
 | 
			
		||||
    function pad(number) {
 | 
			
		||||
      if (number < 10) {
 | 
			
		||||
        return '0' + number;
 | 
			
		||||
      }
 | 
			
		||||
      return number;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    Date.prototype.toISOString = function() {
 | 
			
		||||
      return this.getUTCFullYear() +
 | 
			
		||||
        '-' + pad(this.getUTCMonth() + 1) +
 | 
			
		||||
        '-' + pad(this.getUTCDate()) +
 | 
			
		||||
        'T' + pad(this.getUTCHours()) +
 | 
			
		||||
        ':' + pad(this.getUTCMinutes()) +
 | 
			
		||||
        ':' + pad(this.getUTCSeconds()) +
 | 
			
		||||
        '.' + (this.getUTCMilliseconds() / 1000).toFixed(3).slice(2, 5) +
 | 
			
		||||
        'Z';
 | 
			
		||||
    };
 | 
			
		||||
 | 
			
		||||
  }());
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
							
								
								
									
										20
									
								
								ssf.flow.js
									
									
									
									
									
								
							
							
								
								
								
								
								
									
									
								
							
						
						
									
										20
									
								
								ssf.flow.js
									
									
									
									
									
								
							@ -332,7 +332,7 @@ function write_num_flt(type/*:string*/, fmt/*:string*/, val/*:number*/)/*:string
 | 
			
		||||
	if(fmt.charCodeAt(fmt.length - 1) === 44) return write_num_cm(type, fmt, val);
 | 
			
		||||
	if(fmt.indexOf('%') !== -1) return write_num_pct(type, fmt, val);
 | 
			
		||||
	if(fmt.indexOf('E') !== -1) return write_num_exp(fmt, val);
 | 
			
		||||
	if(fmt.charCodeAt(0) === 36) return "$"+write_num_flt(type,fmt.substr(fmt[1]==' '?2:1),val);
 | 
			
		||||
	if(fmt.charCodeAt(0) === 36) return "$"+write_num_flt(type,fmt.substr(fmt.charAt(1)==' '?2:1),val);
 | 
			
		||||
	var o;
 | 
			
		||||
	var r/*:?Array<string>*/, ri, ff, aval = Math.abs(val), sign = val < 0 ? "-" : "";
 | 
			
		||||
	if(fmt.match(/^00+$/)) return sign + pad0r(aval,fmt.length);
 | 
			
		||||
@ -359,7 +359,7 @@ function write_num_flt(type/*:string*/, fmt/*:string*/, val/*:number*/)/*:string
 | 
			
		||||
	if((r = fmt.match(/^([0#]+)(\\?-([0#]+))+$/))) {
 | 
			
		||||
		o = _strrev(write_num_flt(type, fmt.replace(/[\\-]/g,""), val));
 | 
			
		||||
		ri = 0;
 | 
			
		||||
		return _strrev(_strrev(fmt.replace(/\\/g,"")).replace(/[0#]/g,function(x){return ri<o.length?o[ri++]:x==='0'?'0':"";}));
 | 
			
		||||
		return _strrev(_strrev(fmt.replace(/\\/g,"")).replace(/[0#]/g,function(x){return ri<o.length?o.charAt(ri++):x==='0'?'0':"";}));
 | 
			
		||||
	}
 | 
			
		||||
	if(fmt.match(phone)) {
 | 
			
		||||
		o = write_num_flt(type, "##########", val);
 | 
			
		||||
@ -371,7 +371,7 @@ function write_num_flt(type/*:string*/, fmt/*:string*/, val/*:number*/)/*:string
 | 
			
		||||
		ff = frac(aval, Math.pow(10,ri)-1, false);
 | 
			
		||||
		o = "" + sign;
 | 
			
		||||
		oa = write_num("n", /*::String(*/r[1]/*::)*/, ff[1]);
 | 
			
		||||
		if(oa[oa.length-1] == " ") oa = oa.substr(0,oa.length-1) + "0";
 | 
			
		||||
		if(oa.charAt(oa.length-1) == " ") oa = oa.substr(0,oa.length-1) + "0";
 | 
			
		||||
		o += oa + /*::String(*/r[2]/*::)*/ + "/" + /*::String(*/r[3]/*::)*/;
 | 
			
		||||
		oa = rpad_(ff[2],ri);
 | 
			
		||||
		if(oa.length < r[4].length) oa = hashq(r[4].substr(r[4].length-oa.length)) + oa;
 | 
			
		||||
@ -442,7 +442,7 @@ function write_num_int(type/*:string*/, fmt/*:string*/, val/*:number*/)/*:string
 | 
			
		||||
	if(fmt.charCodeAt(fmt.length - 1) === 44) return write_num_cm2(type, fmt, val);
 | 
			
		||||
	if(fmt.indexOf('%') !== -1) return write_num_pct2(type, fmt, val);
 | 
			
		||||
	if(fmt.indexOf('E') !== -1) return write_num_exp2(fmt, val);
 | 
			
		||||
	if(fmt.charCodeAt(0) === 36) return "$"+write_num_int(type,fmt.substr(fmt[1]==' '?2:1),val);
 | 
			
		||||
	if(fmt.charCodeAt(0) === 36) return "$"+write_num_int(type,fmt.substr(fmt.charAt(1)==' '?2:1),val);
 | 
			
		||||
	var o;
 | 
			
		||||
	var r, ri, ff, aval = Math.abs(val), sign = val < 0 ? "-" : "";
 | 
			
		||||
	if(fmt.match(/^00+$/)) return sign + pad0(aval,fmt.length);
 | 
			
		||||
@ -454,14 +454,10 @@ function write_num_int(type/*:string*/, fmt/*:string*/, val/*:number*/)/*:string
 | 
			
		||||
	if(fmt.match(/^#+0+$/)) return sign + pad0(aval,fmt.length - fmt.indexOf("0"));
 | 
			
		||||
	if((r = fmt.match(dec1))) {
 | 
			
		||||
		/*:: if(!Array.isArray(r)) throw "unreachable"; */
 | 
			
		||||
		// $FlowIgnore
 | 
			
		||||
		o = (""+val).replace(/^([^\.]+)$/,"$1."+r[1]).replace(/\.$/,"."+r[1]);
 | 
			
		||||
		// $FlowIgnore
 | 
			
		||||
		o = o.replace(/\.(\d*)$/,function($$, $1) {
 | 
			
		||||
		/*:: if(!Array.isArray(r)) throw "unreachable"; */
 | 
			
		||||
		// $FlowIgnore
 | 
			
		||||
			return "." + $1 + fill("0", r[1].length-$1.length); });
 | 
			
		||||
		// $FlowIgnore
 | 
			
		||||
		return fmt.indexOf("0.") !== -1 ? o : o.replace(/^0\./,".");
 | 
			
		||||
	}
 | 
			
		||||
	fmt = fmt.replace(/^#+([0.])/, "$1");
 | 
			
		||||
@ -476,7 +472,7 @@ function write_num_int(type/*:string*/, fmt/*:string*/, val/*:number*/)/*:string
 | 
			
		||||
	if((r = fmt.match(/^([0#]+)(\\?-([0#]+))+$/))) {
 | 
			
		||||
		o = _strrev(write_num_int(type, fmt.replace(/[\\-]/g,""), val));
 | 
			
		||||
		ri = 0;
 | 
			
		||||
		return _strrev(_strrev(fmt.replace(/\\/g,"")).replace(/[0#]/g,function(x){return ri<o.length?o[ri++]:x==='0'?'0':"";}));
 | 
			
		||||
		return _strrev(_strrev(fmt.replace(/\\/g,"")).replace(/[0#]/g,function(x){return ri<o.length?o.charAt(ri++):x==='0'?'0':"";}));
 | 
			
		||||
	}
 | 
			
		||||
	if(fmt.match(phone)) {
 | 
			
		||||
		o = write_num_int(type, "##########", val);
 | 
			
		||||
@ -488,7 +484,7 @@ function write_num_int(type/*:string*/, fmt/*:string*/, val/*:number*/)/*:string
 | 
			
		||||
		ff = frac(aval, Math.pow(10,ri)-1, false);
 | 
			
		||||
		o = "" + sign;
 | 
			
		||||
		oa = write_num("n", r[1], ff[1]);
 | 
			
		||||
		if(oa[oa.length-1] == " ") oa = oa.substr(0,oa.length-1) + "0";
 | 
			
		||||
		if(oa.charAt(oa.length-1) == " ") oa = oa.substr(0,oa.length-1) + "0";
 | 
			
		||||
		o += oa + r[2] + "/" + r[3];
 | 
			
		||||
		oa = rpad_(ff[2],ri);
 | 
			
		||||
		if(oa.length < r[4].length) oa = hashq(r[4].substr(r[4].length-oa.length)) + oa;
 | 
			
		||||
@ -740,7 +736,7 @@ function eval_fmt(fmt/*:string*/, v/*:any*/, opts/*:any*/, flen/*:number*/) {
 | 
			
		||||
				j=out[i].v.indexOf(".")>-1&&i===decpt?out[i].v.indexOf(".")-1:out[i].v.length-1;
 | 
			
		||||
				vv = out[i].v.substr(j+1);
 | 
			
		||||
				for(; j>=0; --j) {
 | 
			
		||||
					if(jj>=0 && (out[i].v[j] === "0" || out[i].v[j] === "#")) vv = ostr[jj--] + vv;
 | 
			
		||||
					if(jj>=0 && (out[i].v.charAt(j) === "0" || out[i].v.charAt(j) === "#")) vv = ostr.charAt(jj--) + vv;
 | 
			
		||||
				}
 | 
			
		||||
				out[i].v = vv;
 | 
			
		||||
				out[i].t = 't';
 | 
			
		||||
@ -753,7 +749,7 @@ function eval_fmt(fmt/*:string*/, v/*:any*/, opts/*:any*/, flen/*:number*/) {
 | 
			
		||||
				j=out[i].v.indexOf(".")>-1&&i===decpt?out[i].v.indexOf(".")+1:0;
 | 
			
		||||
				vv = out[i].v.substr(0,j);
 | 
			
		||||
				for(; j<out[i].v.length; ++j) {
 | 
			
		||||
					if(jj<ostr.length) vv += ostr[jj++];
 | 
			
		||||
					if(jj<ostr.length) vv += ostr.charAt(jj++);
 | 
			
		||||
				}
 | 
			
		||||
				out[i].v = vv;
 | 
			
		||||
				out[i].t = 't';
 | 
			
		||||
 | 
			
		||||
							
								
								
									
										24
									
								
								ssf.js
									
									
									
									
									
								
							
							
								
								
								
								
								
									
									
								
							
						
						
									
										24
									
								
								ssf.js
									
									
									
									
									
								
							@ -331,7 +331,7 @@ function write_num_flt(type, fmt, val) {
 | 
			
		||||
	if(fmt.charCodeAt(fmt.length - 1) === 44) return write_num_cm(type, fmt, val);
 | 
			
		||||
	if(fmt.indexOf('%') !== -1) return write_num_pct(type, fmt, val);
 | 
			
		||||
	if(fmt.indexOf('E') !== -1) return write_num_exp(fmt, val);
 | 
			
		||||
	if(fmt.charCodeAt(0) === 36) return "$"+write_num_flt(type,fmt.substr(fmt[1]==' '?2:1),val);
 | 
			
		||||
	if(fmt.charCodeAt(0) === 36) return "$"+write_num_flt(type,fmt.substr(fmt.charAt(1)==' '?2:1),val);
 | 
			
		||||
	var o;
 | 
			
		||||
	var r, ri, ff, aval = Math.abs(val), sign = val < 0 ? "-" : "";
 | 
			
		||||
	if(fmt.match(/^00+$/)) return sign + pad0r(aval,fmt.length);
 | 
			
		||||
@ -358,7 +358,7 @@ function write_num_flt(type, fmt, val) {
 | 
			
		||||
	if((r = fmt.match(/^([0#]+)(\\?-([0#]+))+$/))) {
 | 
			
		||||
		o = _strrev(write_num_flt(type, fmt.replace(/[\\-]/g,""), val));
 | 
			
		||||
		ri = 0;
 | 
			
		||||
		return _strrev(_strrev(fmt.replace(/\\/g,"")).replace(/[0#]/g,function(x){return ri<o.length?o[ri++]:x==='0'?'0':"";}));
 | 
			
		||||
		return _strrev(_strrev(fmt.replace(/\\/g,"")).replace(/[0#]/g,function(x){return ri<o.length?o.charAt(ri++):x==='0'?'0':"";}));
 | 
			
		||||
	}
 | 
			
		||||
	if(fmt.match(phone)) {
 | 
			
		||||
		o = write_num_flt(type, "##########", val);
 | 
			
		||||
@ -370,7 +370,7 @@ function write_num_flt(type, fmt, val) {
 | 
			
		||||
		ff = frac(aval, Math.pow(10,ri)-1, false);
 | 
			
		||||
		o = "" + sign;
 | 
			
		||||
		oa = write_num("n", r[1], ff[1]);
 | 
			
		||||
		if(oa[oa.length-1] == " ") oa = oa.substr(0,oa.length-1) + "0";
 | 
			
		||||
		if(oa.charAt(oa.length-1) == " ") oa = oa.substr(0,oa.length-1) + "0";
 | 
			
		||||
		o += oa + r[2] + "/" + r[3];
 | 
			
		||||
		oa = rpad_(ff[2],ri);
 | 
			
		||||
		if(oa.length < r[4].length) oa = hashq(r[4].substr(r[4].length-oa.length)) + oa;
 | 
			
		||||
@ -441,7 +441,7 @@ function write_num_int(type, fmt, val) {
 | 
			
		||||
	if(fmt.charCodeAt(fmt.length - 1) === 44) return write_num_cm2(type, fmt, val);
 | 
			
		||||
	if(fmt.indexOf('%') !== -1) return write_num_pct2(type, fmt, val);
 | 
			
		||||
	if(fmt.indexOf('E') !== -1) return write_num_exp2(fmt, val);
 | 
			
		||||
	if(fmt.charCodeAt(0) === 36) return "$"+write_num_int(type,fmt.substr(fmt[1]==' '?2:1),val);
 | 
			
		||||
	if(fmt.charCodeAt(0) === 36) return "$"+write_num_int(type,fmt.substr(fmt.charAt(1)==' '?2:1),val);
 | 
			
		||||
	var o;
 | 
			
		||||
	var r, ri, ff, aval = Math.abs(val), sign = val < 0 ? "-" : "";
 | 
			
		||||
	if(fmt.match(/^00+$/)) return sign + pad0(aval,fmt.length);
 | 
			
		||||
@ -452,13 +452,9 @@ function write_num_int(type, fmt, val) {
 | 
			
		||||
	if((r = fmt.match(frac1))) return write_num_f2(r, aval, sign);
 | 
			
		||||
	if(fmt.match(/^#+0+$/)) return sign + pad0(aval,fmt.length - fmt.indexOf("0"));
 | 
			
		||||
	if((r = fmt.match(dec1))) {
 | 
			
		||||
// $FlowIgnore
 | 
			
		||||
		o = (""+val).replace(/^([^\.]+)$/,"$1."+r[1]).replace(/\.$/,"."+r[1]);
 | 
			
		||||
		// $FlowIgnore
 | 
			
		||||
o = (""+val).replace(/^([^\.]+)$/,"$1."+r[1]).replace(/\.$/,"."+r[1]);
 | 
			
		||||
		o = o.replace(/\.(\d*)$/,function($$, $1) {
 | 
			
		||||
// $FlowIgnore
 | 
			
		||||
			return "." + $1 + fill("0", r[1].length-$1.length); });
 | 
			
		||||
		// $FlowIgnore
 | 
			
		||||
return "." + $1 + fill("0", r[1].length-$1.length); });
 | 
			
		||||
		return fmt.indexOf("0.") !== -1 ? o : o.replace(/^0\./,".");
 | 
			
		||||
	}
 | 
			
		||||
	fmt = fmt.replace(/^#+([0.])/, "$1");
 | 
			
		||||
@ -473,7 +469,7 @@ function write_num_int(type, fmt, val) {
 | 
			
		||||
	if((r = fmt.match(/^([0#]+)(\\?-([0#]+))+$/))) {
 | 
			
		||||
		o = _strrev(write_num_int(type, fmt.replace(/[\\-]/g,""), val));
 | 
			
		||||
		ri = 0;
 | 
			
		||||
		return _strrev(_strrev(fmt.replace(/\\/g,"")).replace(/[0#]/g,function(x){return ri<o.length?o[ri++]:x==='0'?'0':"";}));
 | 
			
		||||
		return _strrev(_strrev(fmt.replace(/\\/g,"")).replace(/[0#]/g,function(x){return ri<o.length?o.charAt(ri++):x==='0'?'0':"";}));
 | 
			
		||||
	}
 | 
			
		||||
	if(fmt.match(phone)) {
 | 
			
		||||
		o = write_num_int(type, "##########", val);
 | 
			
		||||
@ -485,7 +481,7 @@ function write_num_int(type, fmt, val) {
 | 
			
		||||
		ff = frac(aval, Math.pow(10,ri)-1, false);
 | 
			
		||||
		o = "" + sign;
 | 
			
		||||
		oa = write_num("n", r[1], ff[1]);
 | 
			
		||||
		if(oa[oa.length-1] == " ") oa = oa.substr(0,oa.length-1) + "0";
 | 
			
		||||
		if(oa.charAt(oa.length-1) == " ") oa = oa.substr(0,oa.length-1) + "0";
 | 
			
		||||
		o += oa + r[2] + "/" + r[3];
 | 
			
		||||
		oa = rpad_(ff[2],ri);
 | 
			
		||||
		if(oa.length < r[4].length) oa = hashq(r[4].substr(r[4].length-oa.length)) + oa;
 | 
			
		||||
@ -734,7 +730,7 @@ out[i].v = write_date(out[i].t.charCodeAt(0), out[i].v, dt, ss0);
 | 
			
		||||
				j=out[i].v.indexOf(".")>-1&&i===decpt?out[i].v.indexOf(".")-1:out[i].v.length-1;
 | 
			
		||||
				vv = out[i].v.substr(j+1);
 | 
			
		||||
				for(; j>=0; --j) {
 | 
			
		||||
					if(jj>=0 && (out[i].v[j] === "0" || out[i].v[j] === "#")) vv = ostr[jj--] + vv;
 | 
			
		||||
					if(jj>=0 && (out[i].v.charAt(j) === "0" || out[i].v.charAt(j) === "#")) vv = ostr.charAt(jj--) + vv;
 | 
			
		||||
				}
 | 
			
		||||
				out[i].v = vv;
 | 
			
		||||
				out[i].t = 't';
 | 
			
		||||
@ -747,7 +743,7 @@ out[i].v = write_date(out[i].t.charCodeAt(0), out[i].v, dt, ss0);
 | 
			
		||||
				j=out[i].v.indexOf(".")>-1&&i===decpt?out[i].v.indexOf(".")+1:0;
 | 
			
		||||
				vv = out[i].v.substr(0,j);
 | 
			
		||||
				for(; j<out[i].v.length; ++j) {
 | 
			
		||||
					if(jj<ostr.length) vv += ostr[jj++];
 | 
			
		||||
					if(jj<ostr.length) vv += ostr.charAt(jj++);
 | 
			
		||||
				}
 | 
			
		||||
				out[i].v = vv;
 | 
			
		||||
				out[i].t = 't';
 | 
			
		||||
 | 
			
		||||
@ -8,7 +8,7 @@ function doit(d, headers) {
 | 
			
		||||
    for(var w = 1; w < headers.length; ++w) {
 | 
			
		||||
      var expected = d[w].replace("|", ""), actual;
 | 
			
		||||
      try { actual = SSF.format(headers[w], parseFloat(d[0]), {}); } catch(e) { }
 | 
			
		||||
      if(actual != expected && d[w][0] !== "|") throw new Error([actual, expected, w, headers[w],d[0],d].join("|"));
 | 
			
		||||
      if(actual != expected && d[w].charAt(0) !== "|") throw new Error([actual, expected, w, headers[w],d[0],d].join("|"));
 | 
			
		||||
    }
 | 
			
		||||
  });
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
@ -3,7 +3,6 @@
 | 
			
		||||
var SSF = require('../');
 | 
			
		||||
var fs = require('fs'), assert = require('assert');
 | 
			
		||||
var is_date = JSON.parse(fs.readFileSync('./test/is_date.json','utf8'));
 | 
			
		||||
var skip = [];
 | 
			
		||||
describe('utilities', function() {
 | 
			
		||||
  it('correctly determines if formats are dates', function() {
 | 
			
		||||
		is_date.forEach(function(d) { assert.equal(SSF.is_date(d[0]), d[1], d[0]); });
 | 
			
		||||
 | 
			
		||||
		Loading…
	
		Reference in New Issue
	
	Block a user