forked from sheetjs/sheetjs
version bump 0.2.1: more support for implied types
This commit is contained in:
parent
e176abd8de
commit
22f04832e3
6
Makefile
Normal file
6
Makefile
Normal file
@ -0,0 +1,6 @@
|
||||
.PHONY: test ssf
|
||||
ssf: ssf.md
|
||||
voc ssf.md
|
||||
|
||||
test:
|
||||
npm test
|
@ -1,6 +1,6 @@
|
||||
{
|
||||
"name": "ssf",
|
||||
"version": "0.2.0",
|
||||
"version": "0.2.1",
|
||||
"author": "SheetJS",
|
||||
"description": "pure-JS library to format data using ECMA-376 spreadsheet Format Codes",
|
||||
"keywords": [ "format", "sprintf", "spreadsheet" ],
|
||||
|
52
ssf.js
52
ssf.js
@ -1,3 +1,4 @@
|
||||
/* ssf.js (C) 2013 SheetJS -- http://sheetjs.com */
|
||||
var SSF;
|
||||
(function(SSF){
|
||||
String.prototype.reverse=function(){return this.split("").reverse().join("");};
|
||||
@ -65,6 +66,11 @@ var months = [
|
||||
];
|
||||
var general_fmt = function(v) {
|
||||
if(typeof v === 'boolean') return v ? "TRUE" : "FALSE";
|
||||
if(typeof v === 'number') {
|
||||
return v.toString().substr(0,11);
|
||||
}
|
||||
if(typeof v === 'string') return v;
|
||||
throw "unsupport value in General format: " + v;
|
||||
};
|
||||
SSF._general = general_fmt;
|
||||
var parse_date_code = function parse_date_code(v,opts) {
|
||||
@ -128,14 +134,41 @@ var write_date = function(type, fmt, val) {
|
||||
case 's': switch(fmt) { /* seconds */
|
||||
case 's': return val.S;
|
||||
case 'ss': return pad(val.S, 2);
|
||||
case 'ss.0': console.log(val);
|
||||
default: throw 'bad second format: ' + fmt;
|
||||
} break;
|
||||
case 'Z': switch(fmt) {
|
||||
case '[h]': return val.D*24+val.H;
|
||||
default: throw 'bad abstime format: ' + fmt;
|
||||
} break;
|
||||
/* TODO: handle the ECMA spec format ee -> yy */
|
||||
case 'e': { return val.y; } break;
|
||||
case 'A': return (val.h>=12 ? 'P' : 'A') + fmt.substr(1);
|
||||
default: throw 'bad format type ' + type + ' in ' + fmt;
|
||||
}
|
||||
};
|
||||
String.prototype.reverse = function() { return this.split("").reverse().join(""); }
|
||||
var commaify = function(s) { return s.reverse().replace(/.../g,"$&,").reverse(); };
|
||||
var write_num = function(type, fmt, val) {
|
||||
var mul = 0;
|
||||
fmt = fmt.replace(/%/g,function(x) { mul++; return ""; });
|
||||
if(mul !== 0) return write_num(type, fmt, val * Math.pow(10,2*mul)) + fill("%",mul);
|
||||
if(fmt.indexOf("E") > -1) {
|
||||
var o = val.toExponential(fmt.indexOf("E") - fmt.indexOf(".") - 1);
|
||||
if(fmt.match(/E\+00$/) && o.match(/e[+-][0-9]$/)) o = o.substr(0,o.length-1) + "0" + o[o.length-1];
|
||||
if(fmt.match(/E\-/) && o.match(/e\+/)) o = o.replace(/e\+/,"e");
|
||||
return o.replace("e","E");
|
||||
}
|
||||
switch(fmt) {
|
||||
case "0": return Math.round(val);
|
||||
case "0.00": return Math.round(val*100)/100;
|
||||
case "#,##0": return commaify(String(Math.round(val)));
|
||||
case "#,##0.00": return commaify(String(Math.floor(val))) + "." + Math.round((val-Math.floor(val))*100);
|
||||
default:
|
||||
}
|
||||
console.log(type, fmt, val);
|
||||
return "0";
|
||||
};
|
||||
function split_fmt(fmt) {
|
||||
var out = [];
|
||||
var in_str = -1;
|
||||
@ -169,6 +202,7 @@ function eval_fmt(fmt, v, opts) {
|
||||
case 'm': case 'd': case 'y': case 'h': case 's': case 'e':
|
||||
if(!dt) dt = parse_date_code(v, opts);
|
||||
o = fmt[i]; while(fmt[++i] === c) o+=c;
|
||||
if(c === 's' && fmt[i] === '.' && fmt[i+1] === '0') { o+='.'; while(fmt[++i] === '0') o+= '0'; }
|
||||
if(c === 'm' && lst.toLowerCase() === 'h') c = 'M'; /* m = minute */
|
||||
if(c === 'h') c = hr;
|
||||
q={t:c, v:o}; out.push(q); lst = c; break;
|
||||
@ -180,7 +214,18 @@ function eval_fmt(fmt, v, opts) {
|
||||
else q.t = "t";
|
||||
out.push(q); lst = c; break;
|
||||
case '[': /* TODO: Fix this -- ignore all conditionals and formatting */
|
||||
while(fmt[i++] !== ']'); break;
|
||||
o = c;
|
||||
while(fmt[i++] !== ']') o += fmt[i];
|
||||
if(o == "[h]") out.push({t:'Z', v:o});
|
||||
break;
|
||||
/* Numbers */
|
||||
case '0': case '#':
|
||||
var nn = ""; while("0#.,E+-%".indexOf(c=fmt[i++]) > -1) nn += c;
|
||||
out.push({t:'n', v:nn}); break;
|
||||
case '?':
|
||||
o = fmt[i]; while(fmt[++i] === c) o+=c;
|
||||
q={t:c, v:o}; out.push(q); lst = c; break;
|
||||
|
||||
default:
|
||||
if("$-+/():!^&'~{}<>= ".indexOf(c) === -1)
|
||||
throw 'unrecognized character ' + fmt[i] + ' in ' + fmt;
|
||||
@ -200,9 +245,12 @@ function eval_fmt(fmt, v, opts) {
|
||||
for(i=0; i < out.length; ++i) {
|
||||
switch(out[i].t) {
|
||||
case 't': case 'T': break;
|
||||
case 'd': case 'm': case 'y': case 'h': case 'H': case 'M': case 's': case 'A': case 'e':
|
||||
case 'd': case 'm': case 'y': case 'h': case 'H': case 'M': case 's': case 'A': case 'e': case 'Z':
|
||||
out[i].v = write_date(out[i].t, out[i].v, dt);
|
||||
out[i].t = 't'; break;
|
||||
case 'n':
|
||||
out[i].v = write_num(out[i].t, out[i].v, v);
|
||||
out[i].t = 't'; break;
|
||||
default: throw "unrecognized type " + out[i].t;
|
||||
}
|
||||
}
|
||||
|
133
ssf.md
133
ssf.md
@ -134,9 +134,22 @@ Booleans are serialized in upper case:
|
||||
if(typeof v === 'boolean') return v ? "TRUE" : "FALSE";
|
||||
```
|
||||
|
||||
|
||||
For numbers, try to display up to 11 digits of the number:
|
||||
|
||||
```
|
||||
if(typeof v === 'number') {
|
||||
return v.toString().substr(0,11);
|
||||
}
|
||||
```
|
||||
|
||||
For strings, just return the text as-is:
|
||||
|
||||
```
|
||||
if(typeof v === 'string') return v;
|
||||
```
|
||||
|
||||
```
|
||||
throw "unsupport value in General format: " + v;
|
||||
};
|
||||
SSF._general = general_fmt;
|
||||
```
|
||||
@ -322,6 +335,48 @@ Because JS dates cannot represent the bad leap day, this returns an object:
|
||||
SSF.parse_date_code = parse_date_code;
|
||||
```
|
||||
|
||||
## Evaluating Number Formats
|
||||
|
||||
```js>tmp/number.js
|
||||
String.prototype.reverse = function() { return this.split("").reverse().join(""); }
|
||||
var commaify = function(s) { return s.reverse().replace(/.../g,"$&,").reverse(); };
|
||||
var write_num = function(type, fmt, val) {
|
||||
```
|
||||
|
||||
Percentage values should be physically shifted:
|
||||
|
||||
```js>tmp/number.js
|
||||
var mul = 0;
|
||||
fmt = fmt.replace(/%/g,function(x) { mul++; return ""; });
|
||||
if(mul !== 0) return write_num(type, fmt, val * Math.pow(10,2*mul)) + fill("%",mul);
|
||||
```
|
||||
|
||||
For exponents, get the exponent and mantissa and format them separately:
|
||||
|
||||
```
|
||||
if(fmt.indexOf("E") > -1) {
|
||||
var o = val.toExponential(fmt.indexOf("E") - fmt.indexOf(".") - 1);
|
||||
if(fmt.match(/E\+00$/) && o.match(/e[+-][0-9]$/)) o = o.substr(0,o.length-1) + "0" + o[o.length-1];
|
||||
if(fmt.match(/E\-/) && o.match(/e\+/)) o = o.replace(/e\+/,"e");
|
||||
return o.replace("e","E");
|
||||
}
|
||||
```
|
||||
|
||||
The default cases are hard-coded. TODO: actually parse them
|
||||
|
||||
```js>tmp/number.js
|
||||
switch(fmt) {
|
||||
case "0": return Math.round(val);
|
||||
case "0.00": return Math.round(val*100)/100;
|
||||
case "#,##0": return commaify(String(Math.round(val)));
|
||||
case "#,##0.00": return commaify(String(Math.floor(val))) + "." + Math.round((val-Math.floor(val))*100);
|
||||
default:
|
||||
}
|
||||
console.log(type, fmt, val);
|
||||
return "0";
|
||||
};
|
||||
```
|
||||
|
||||
## Evaluating Format Strings
|
||||
|
||||
```js>tmp/main.js
|
||||
@ -362,6 +417,17 @@ The date codes `m,d,y,h,s` are standard. There are some special formats like
|
||||
case 'm': case 'd': case 'y': case 'h': case 's': case 'e':
|
||||
if(!dt) dt = parse_date_code(v, opts);
|
||||
o = fmt[i]; while(fmt[++i] === c) o+=c;
|
||||
```
|
||||
|
||||
For the special case of s.00, the suffix should be swallowed with the s:
|
||||
|
||||
```
|
||||
if(c === 's' && fmt[i] === '.' && fmt[i+1] === '0') { o+='.'; while(fmt[++i] === '0') o+= '0'; }
|
||||
```
|
||||
|
||||
Only the forward corrections are made here. The reverse corrections are made later:
|
||||
|
||||
```
|
||||
if(c === 'm' && lst.toLowerCase() === 'h') c = 'M'; /* m = minute */
|
||||
if(c === 'h') c = hr;
|
||||
q={t:c, v:o}; out.push(q); lst = c; break;
|
||||
@ -383,8 +449,37 @@ the HH/hh jazz. TODO: investigate this further.
|
||||
else if(fmt.substr(i,5) === "AM/PM") { q.v = dt.H >= 12 ? "PM" : "AM"; q.t = 'T'; i+=5; hr='h'; }
|
||||
else q.t = "t";
|
||||
out.push(q); lst = c; break;
|
||||
```
|
||||
|
||||
Conditional and color blocks should be handled at one point (TODO). For now,
|
||||
only the absolute time `[h]` is captured (using the pseudo-type `Z`):
|
||||
|
||||
```
|
||||
case '[': /* TODO: Fix this -- ignore all conditionals and formatting */
|
||||
while(fmt[i++] !== ']'); break;
|
||||
o = c;
|
||||
while(fmt[i++] !== ']') o += fmt[i];
|
||||
if(o == "[h]") out.push({t:'Z', v:o});
|
||||
break;
|
||||
```
|
||||
|
||||
Number blocks (following the general pattern `[0#][0#.,E+-%]*`) are grouped together:
|
||||
|
||||
```
|
||||
/* Numbers */
|
||||
case '0': case '#':
|
||||
var nn = ""; while("0#.,E+-%".indexOf(c=fmt[i++]) > -1) nn += c;
|
||||
out.push({t:'n', v:nn}); break;
|
||||
|
||||
```
|
||||
|
||||
The fraction question mark characters present their own challenges. For example, the
|
||||
number 123.456 under format `|??| / |???| |???| foo` is `|15432| / |125| | | foo`:
|
||||
|
||||
```
|
||||
case '?':
|
||||
o = fmt[i]; while(fmt[++i] === c) o+=c;
|
||||
q={t:c, v:o}; out.push(q); lst = c; break;
|
||||
|
||||
default:
|
||||
if("$-+/():!^&'~{}<>= ".indexOf(c) === -1)
|
||||
throw 'unrecognized character ' + fmt[i] + ' in ' + fmt;
|
||||
@ -404,9 +499,12 @@ the HH/hh jazz. TODO: investigate this further.
|
||||
for(i=0; i < out.length; ++i) {
|
||||
switch(out[i].t) {
|
||||
case 't': case 'T': break;
|
||||
case 'd': case 'm': case 'y': case 'h': case 'H': case 'M': case 's': case 'A': case 'e':
|
||||
case 'd': case 'm': case 'y': case 'h': case 'H': case 'M': case 's': case 'A': case 'e': case 'Z':
|
||||
out[i].v = write_date(out[i].t, out[i].v, dt);
|
||||
out[i].t = 't'; break;
|
||||
case 'n':
|
||||
out[i].v = write_num(out[i].t, out[i].v, v);
|
||||
out[i].t = 't'; break;
|
||||
default: throw "unrecognized type " + out[i].t;
|
||||
}
|
||||
}
|
||||
@ -464,10 +562,20 @@ var write_date = function(type, fmt, val) {
|
||||
case 's': switch(fmt) { /* seconds */
|
||||
case 's': return val.S;
|
||||
case 'ss': return pad(val.S, 2);
|
||||
case 'ss.0': console.log(val);
|
||||
default: throw 'bad second format: ' + fmt;
|
||||
} break;
|
||||
```
|
||||
|
||||
The `Z` type refers to absolute time measures:
|
||||
|
||||
```
|
||||
case 'Z': switch(fmt) {
|
||||
case '[h]': return val.D*24+val.H;
|
||||
default: throw 'bad abstime format: ' + fmt;
|
||||
} break;
|
||||
```
|
||||
|
||||
The `e` format behavior in excel diverges from the spec. It claims that `ee`
|
||||
should be a two-digit year, but `ee` in excel is actually the four-digit year:
|
||||
|
||||
@ -514,6 +622,7 @@ SSF.format = format;
|
||||
## JS Boilerplate
|
||||
|
||||
```js>tmp/00_header.js
|
||||
/* ssf.js (C) 2013 SheetJS -- http://sheetjs.com */
|
||||
var SSF;
|
||||
(function(SSF){
|
||||
String.prototype.reverse=function(){return this.split("").reverse().join("");};
|
||||
@ -535,8 +644,8 @@ function pad(v,d){var t=String(v);return t.length>=d?t:(fill(0,d-t.length)+t);}
|
||||
```bash>tmp/post.sh
|
||||
#!/bin/bash
|
||||
npm install
|
||||
cat tmp/{00_header,opts,consts,general,date,main,zz_footer_n}.js > ssf_node.js
|
||||
cat tmp/{00_header,opts,consts,general,date,main,zz_footer}.js > ssf.js
|
||||
cat tmp/{00_header,opts,consts,general,date,number,main,zz_footer_n}.js > ssf_node.js
|
||||
cat tmp/{00_header,opts,consts,general,date,number,main,zz_footer}.js > ssf.js
|
||||
```
|
||||
|
||||
```json>.vocrc
|
||||
@ -552,10 +661,19 @@ node_modules/
|
||||
.vocrc
|
||||
```
|
||||
|
||||
```make>Makefile
|
||||
.PHONY: test ssf
|
||||
ssf: ssf.md
|
||||
voc ssf.md
|
||||
|
||||
test:
|
||||
npm test
|
||||
```
|
||||
|
||||
```json>package.json
|
||||
{
|
||||
"name": "ssf",
|
||||
"version": "0.1.0",
|
||||
"version": "0.2.1",
|
||||
"author": "SheetJS",
|
||||
"description": "pure-JS library to format data using ECMA-376 spreadsheet Format Codes",
|
||||
"keywords": [ "format", "sprintf", "spreadsheet" ],
|
||||
@ -597,9 +715,10 @@ The mocha test driver tests the implied formats:
|
||||
var SSF = require('../');
|
||||
var fs = require('fs'), assert = require('assert');
|
||||
var data = JSON.parse(fs.readFileSync('./test/implied.json','utf8'));
|
||||
var skip = [12, 13, 47, 48];
|
||||
describe('implied formats', function() {
|
||||
data.forEach(function(d) {
|
||||
it(d[1]+" for "+d[0], (d[1]<14||d[1]>22)?null:function(){
|
||||
it(d[1]+" for "+d[0], skip.indexOf(d[1]) > -1 ? null : function(){
|
||||
assert.equal(SSF.format(d[1], d[0], {}), d[2]);
|
||||
});
|
||||
});
|
||||
|
52
ssf_node.js
52
ssf_node.js
@ -1,3 +1,4 @@
|
||||
/* ssf.js (C) 2013 SheetJS -- http://sheetjs.com */
|
||||
var SSF;
|
||||
(function(SSF){
|
||||
String.prototype.reverse=function(){return this.split("").reverse().join("");};
|
||||
@ -65,6 +66,11 @@ var months = [
|
||||
];
|
||||
var general_fmt = function(v) {
|
||||
if(typeof v === 'boolean') return v ? "TRUE" : "FALSE";
|
||||
if(typeof v === 'number') {
|
||||
return v.toString().substr(0,11);
|
||||
}
|
||||
if(typeof v === 'string') return v;
|
||||
throw "unsupport value in General format: " + v;
|
||||
};
|
||||
SSF._general = general_fmt;
|
||||
var parse_date_code = function parse_date_code(v,opts) {
|
||||
@ -128,14 +134,41 @@ var write_date = function(type, fmt, val) {
|
||||
case 's': switch(fmt) { /* seconds */
|
||||
case 's': return val.S;
|
||||
case 'ss': return pad(val.S, 2);
|
||||
case 'ss.0': console.log(val);
|
||||
default: throw 'bad second format: ' + fmt;
|
||||
} break;
|
||||
case 'Z': switch(fmt) {
|
||||
case '[h]': return val.D*24+val.H;
|
||||
default: throw 'bad abstime format: ' + fmt;
|
||||
} break;
|
||||
/* TODO: handle the ECMA spec format ee -> yy */
|
||||
case 'e': { return val.y; } break;
|
||||
case 'A': return (val.h>=12 ? 'P' : 'A') + fmt.substr(1);
|
||||
default: throw 'bad format type ' + type + ' in ' + fmt;
|
||||
}
|
||||
};
|
||||
String.prototype.reverse = function() { return this.split("").reverse().join(""); }
|
||||
var commaify = function(s) { return s.reverse().replace(/.../g,"$&,").reverse(); };
|
||||
var write_num = function(type, fmt, val) {
|
||||
var mul = 0;
|
||||
fmt = fmt.replace(/%/g,function(x) { mul++; return ""; });
|
||||
if(mul !== 0) return write_num(type, fmt, val * Math.pow(10,2*mul)) + fill("%",mul);
|
||||
if(fmt.indexOf("E") > -1) {
|
||||
var o = val.toExponential(fmt.indexOf("E") - fmt.indexOf(".") - 1);
|
||||
if(fmt.match(/E\+00$/) && o.match(/e[+-][0-9]$/)) o = o.substr(0,o.length-1) + "0" + o[o.length-1];
|
||||
if(fmt.match(/E\-/) && o.match(/e\+/)) o = o.replace(/e\+/,"e");
|
||||
return o.replace("e","E");
|
||||
}
|
||||
switch(fmt) {
|
||||
case "0": return Math.round(val);
|
||||
case "0.00": return Math.round(val*100)/100;
|
||||
case "#,##0": return commaify(String(Math.round(val)));
|
||||
case "#,##0.00": return commaify(String(Math.floor(val))) + "." + Math.round((val-Math.floor(val))*100);
|
||||
default:
|
||||
}
|
||||
console.log(type, fmt, val);
|
||||
return "0";
|
||||
};
|
||||
function split_fmt(fmt) {
|
||||
var out = [];
|
||||
var in_str = -1;
|
||||
@ -169,6 +202,7 @@ function eval_fmt(fmt, v, opts) {
|
||||
case 'm': case 'd': case 'y': case 'h': case 's': case 'e':
|
||||
if(!dt) dt = parse_date_code(v, opts);
|
||||
o = fmt[i]; while(fmt[++i] === c) o+=c;
|
||||
if(c === 's' && fmt[i] === '.' && fmt[i+1] === '0') { o+='.'; while(fmt[++i] === '0') o+= '0'; }
|
||||
if(c === 'm' && lst.toLowerCase() === 'h') c = 'M'; /* m = minute */
|
||||
if(c === 'h') c = hr;
|
||||
q={t:c, v:o}; out.push(q); lst = c; break;
|
||||
@ -180,7 +214,18 @@ function eval_fmt(fmt, v, opts) {
|
||||
else q.t = "t";
|
||||
out.push(q); lst = c; break;
|
||||
case '[': /* TODO: Fix this -- ignore all conditionals and formatting */
|
||||
while(fmt[i++] !== ']'); break;
|
||||
o = c;
|
||||
while(fmt[i++] !== ']') o += fmt[i];
|
||||
if(o == "[h]") out.push({t:'Z', v:o});
|
||||
break;
|
||||
/* Numbers */
|
||||
case '0': case '#':
|
||||
var nn = ""; while("0#.,E+-%".indexOf(c=fmt[i++]) > -1) nn += c;
|
||||
out.push({t:'n', v:nn}); break;
|
||||
case '?':
|
||||
o = fmt[i]; while(fmt[++i] === c) o+=c;
|
||||
q={t:c, v:o}; out.push(q); lst = c; break;
|
||||
|
||||
default:
|
||||
if("$-+/():!^&'~{}<>= ".indexOf(c) === -1)
|
||||
throw 'unrecognized character ' + fmt[i] + ' in ' + fmt;
|
||||
@ -200,9 +245,12 @@ function eval_fmt(fmt, v, opts) {
|
||||
for(i=0; i < out.length; ++i) {
|
||||
switch(out[i].t) {
|
||||
case 't': case 'T': break;
|
||||
case 'd': case 'm': case 'y': case 'h': case 'H': case 'M': case 's': case 'A': case 'e':
|
||||
case 'd': case 'm': case 'y': case 'h': case 'H': case 'M': case 's': case 'A': case 'e': case 'Z':
|
||||
out[i].v = write_date(out[i].t, out[i].v, dt);
|
||||
out[i].t = 't'; break;
|
||||
case 'n':
|
||||
out[i].v = write_num(out[i].t, out[i].v, v);
|
||||
out[i].t = 't'; break;
|
||||
default: throw "unrecognized type " + out[i].t;
|
||||
}
|
||||
}
|
||||
|
@ -2,9 +2,10 @@
|
||||
var SSF = require('../');
|
||||
var fs = require('fs'), assert = require('assert');
|
||||
var data = JSON.parse(fs.readFileSync('./test/implied.json','utf8'));
|
||||
var skip = [12, 13, 47, 48];
|
||||
describe('implied formats', function() {
|
||||
data.forEach(function(d) {
|
||||
it(d[1]+" for "+d[0], (d[1]<14||d[1]>22)?null:function(){
|
||||
it(d[1]+" for "+d[0], skip.indexOf(d[1]) > -1 ? null : function(){
|
||||
assert.equal(SSF.format(d[1], d[0], {}), d[2]);
|
||||
});
|
||||
});
|
||||
|
Loading…
Reference in New Issue
Block a user