@borgar/fx 1.0.0 → 2.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/README.md CHANGED
@@ -22,6 +22,7 @@ The library is also provided as an ES6 module in an NPM package:
22
22
  |- | - | -
23
23
  | `emitRanges` | `false` | Adds offset ranges on the tokens: `{ range: [ start, end ] }`
24
24
  | `mergeRanges` | `true` | Should ranges be returned as whole references (`Sheet1!A1:B2`) or as separate tokens for each part: (`Sheet1`,`!`,`A1`,`:`,`B2`).
25
+ | `negativeNumbers` | `false` | Merges all unary minuses with their immediately following number tokens (`-`,`1`) => `-1`
25
26
  | `r1c1` | `false` | Ranges are expected to be in the R1C1 style format rather than the more popular A1 style.
26
27
 
27
28
  The returned output will be an array of objects representing the tokens:
@@ -106,13 +107,13 @@ The returned output will be the same array of tokens but the following propertie
106
107
 
107
108
  Matching parens will be tagged with `.groupId` string identifier as well as a `.depth` number value (indicating the level of nesting).
108
109
 
109
- Parens without a counterpart will be tagged with `.error` (boolean true).
110
+ Closing parens without a counterpart will be tagged with `.error` (boolean true).
110
111
 
111
112
  #### Curly brackets { }
112
113
 
113
114
  Matching curly brackets will be tagged with `.groupId` string identifier. These may not be nested in Excel.
114
115
 
115
- Curly brackets without a counterpart will be tagged with `.error` (boolean `true`).
116
+ Closing curly brackets without a counterpart will be tagged with `.error` (boolean `true`).
116
117
 
117
118
  #### Ranges (`RANGE` or `RANGE_BEAM` type tokens)
118
119
 
package/dist/fx.js ADDED
@@ -0,0 +1,2 @@
1
+ "use strict";function t(t){return t.split(/(\d+|[a-zA-Z]+)/).every((t=>{const e=parseInt(t,36);return!!isNaN(e)||(isFinite(t)?e<=2183880786:e<=43321)}))}Object.defineProperty(exports,"__esModule",{value:!0});const e={R:1048577,"R[":1048576,C:16385,"C[":16384};function r(t){const r=/([RC]\[?)(-?\d+)/gi;let n;for(;null!==(n=r.exec(t));){const t=e[n[1]],r=parseInt(n[2],10);if(r>=t||r<=-t)return!1}return!0}const n="range",o="range-beam",a=/^#(NAME\?|FIELD!|CALC!|VALUE!|REF!|DIV\/0!|NULL!|NUM!|N\/A|GETTING_DATA\b|SPILL!|UNKNOWN!|FIELD\b|CALC\b|SYNTAX\?|ERROR!)/i,i=/^(<=|>=|<>|[-+/*^%&<>=]|[{},;]|[()]|@|:|!|#)/,l=/^(TRUE|FALSE)\b/i,c=/^[A-Z_]+[A-Z\d_.]+(?=\s*\()/i,u=/^\n+/,p=/^[ \f\r\t\v\u00a0\u1680\u2000-\u200a\u2028\u2029\u202f\u205f\u3000\ufeff]+/,s=/^"(?:""|[^"])*("|$)/,f=/^'(?:''|[^'])*('|$)/,$=/^\[(?:[^\]])+(\]|$)/,g=/^([^ \t\n$!"`'#%&(){}<>,;:^@|~=*+-]+)(?=!)/,h=/^\$?[A-Z]{1,3}:\$?[A-Z]{1,3}/i,m=/^\$?[1-9][0-9]{0,6}:\$?[1-9][0-9]{0,6}/i,b=/^\$?[A-Z]{1,3}\$?[1-9][0-9]{0,6}/i,d="(?:R(?:\\[[+-]?\\d+\\]|[1-9][0-9]{0,6})?)",v="(?:C(?:\\[[+-]?\\d+\\]|[1-9][0-9]{0,4})?)",R=new RegExp("^".concat(v,"(:").concat(v,")?(?=\\W|$)"),"i"),x=new RegExp("^".concat(d,"(:").concat(d,")?(?=\\W|$)"),"i"),N=new RegExp("^(?:(?=[RC])".concat(d).concat(v,")"),"i"),A=/^(?:\d+(\.\d+)?(?:[eE][+-]?\d+)?|\d+)/,E=/^[A-Z\d\\_.?]+/i,y=[["error",a],["operator",i],["bool",l],["function",c],["newline",u],["whitespace",p],["string",s],["path-quote",f],["path-brace",$],["path-prefix",g],[n,b,t],[o,h,t],[o,m,t],["number",A],["range-named",E]],C=[["error",a],["operator",i],["bool",l],["function",c],["newline",u],["whitespace",p],["string",s],["path-quote",f],["path-brace",$],["path-prefix",g],[n,N,r],[o,x,r],[o,R,r],["number",A],["range-named",E]],I=[["operator",/^[!:]/],["path-quote",f],["path-brace",$],["path-prefix",g],[n,b,t],[o,h,t],[o,m,t],["range-named",E]],w=[["operator",/^!/],["path-quote",f],["path-brace",$],["path-prefix",g],[n,N,r],[o,x,r],[o,R,r],["range-named",E]],k=(t,e)=>t&&t.type===e,T=t=>t&&":"===t.value,O=t=>t&&"!"===t.value,L={emitRanges:!1,mergeRanges:!0,negativeNumbers:!1,r1c1:!1};function M(t,e){let r=arguments.length>2&&void 0!==arguments[2]?arguments[2]:{};const{emitRanges:a,mergeRanges:i,negativeNumbers:l}=Object.assign({},L,r),c=[];let u=0;const p=t=>c[c.length-t],s=t=>{const e=c.filter((t=>!k(t,"whitespace")&&!k(t,"newline")));return e[e.length-t]};if(/^=/.test(t)){const t={type:"fx-prefix",value:"=",...a?{range:[0,1]}:{}};u++,c.push(t)}for(;u<t.length;){const r=u,f=t.slice(u);let $="",g="";for(let t=0;t<e.length;t++){const[r,n,o]=e[t],a=n.exec(f);if(a&&(!o||o(a[0]))){$=r,g=a[0],u+=a[0].length;break}}$||($="unknown",g=t[u],u++);let h={type:$,value:g,...a?{range:[r,u]}:{}};if("string"!==$||g.endsWith('"')||(h.unterminated=!0),l&&"number"===$){const t=p(1);if(t&&k(t,"operator")&&"-"===t.value){const t=s(2);(!t||k(t,"fx-prefix")||k(t,"operator")&&!["%","}",")","#"].includes(t.value))&&(c.pop(),h.value="-"+g)}}if(i&&($===n||"range-named"===$||$===o)){const t=[];if(T(p(1))&&k(p(2),n)&&$===n&&t.unshift(...c.splice(-2,2)),O(p(1)))if(k(p(2),"path-quote")||k(p(2),"path-brace"))t.unshift(...c.splice(-2,2));else if(k(p(2),"path-prefix")){const e=k(p(3),"path-brace")?3:2;t.unshift(...c.splice(-e,e))}t.length&&(h={type:$,value:t.map((t=>t.value)).join("")+g,...a?{range:[t[0].range[0],u]}:{}})}c.push(h)}return c}function _(t){let e=arguments.length>1&&void 0!==arguments[1]?arguments[1]:{};const r=e.r1c1?C:y;return M(t,r,e)}const Z=t=>t.slice(1,-1).replace(/''/g,"'"),S=t=>t&&":"===t.value&&{},U=t=>t&&t.type===n&&{r0:t.value},F=t=>t&&t.type===n&&{r1:t.value},P=t=>t&&"!"===t.value&&{},q=t=>t&&t.type===o&&{r0:t.value},W=t=>t&&"path-prefix"===t.type&&/^[^:\\/?*[\]]{0,31}$/.test(t.value)&&{sheetName:t.value},j=t=>t&&"path-brace"===t.type&&{workbookName:t.value.slice(1,-1)},G=t=>t&&"range-named"===t.type&&{name:t.value},X=t=>{if(t&&"path-quote"===t.type){const e=/(?:\[(.+?)\])?([^[\]]+?)$/.exec(Z(t.value));if(e){const[,t,r]=e;if(!r||/^[^:\\/?*[\]]{0,31}$/.test(r))return{workbookName:t||"",sheetName:r||""}}}},D=[[U],[U,S,F],[q],[X,P,U],[X,P,U,S,F],[X,P,q],[W,P,U],[W,P,U,S,F],[W,P,q],[j,W,P,U],[j,W,P,U,S,F],[j,W,P,q]],B=D.concat([[G],[t=>t&&"path-prefix"===t.type&&{workbookName:t.value},P,G],[t=>t&&"path-quote"===t.type&&{workbookName:Z(t.value)},P,G]]);function H(t){let e=!(arguments.length>1&&void 0!==arguments[1])||arguments[1],r=arguments.length>2&&void 0!==arguments[2]?arguments[2]:[];const n=M(t,r,{emitRanges:!1,mergeRanges:!1}),o={sheetName:"",workbookName:"",r0:"",r1:"",name:""};n.length&&"fx-prefix"===n[0].type&&n.shift();const a=e?B:D;for(let t=0;t<a.length;t++){const e={...o};if(a[t].length===n.length){if(a[t].every(((t,r)=>{const o=t(n[r]);return Object.assign(e,o),o})))return e}}return null}function z(t){const e=(t||"").toUpperCase();let r=0,n=0;for(;n!==e.length;++n){const t=e.charCodeAt(n);t>=65&&t<=90&&(r=26*r+t-64)}return r-1}function K(t){let e=t,r="";for(;e>=0;)r=String.fromCharCode(e%26+65)+r,e=Math.floor(e/26)-1;return r}function V(t){return+t-1}function Q(t){return String(t+1)}function Y(t){const{top:e,left:r,bottom:n,right:o}=t;return{top:e,left:r,bottom:n,right:o,$left:!1,$right:!1,$top:!1,$bottom:!1}}function J(t){const e=t=>t?"$":"",{top:r,left:n,bottom:o,right:a,$left:i,$right:l,$top:c,$bottom:u}=t;return 0===r&&1048575===o?e(i)+K(n)+":"+e(l)+K(a):0===n&&16383===a?e(c)+Q(r)+":"+e(u)+Q(o):null==a||null==o||a===n&&o===r?e(i)+K(n)+e(c)+Q(r):e(i)+K(n)+e(c)+Q(r)+":"+e(l)+K(a)+e(u)+Q(o)}function tt(t){let e,r=0,n=0,o=1048575,a=16383,i=!1,l=!1,c=!1,u=!1;if(e=/^(\$?)([A-Z]{1,3}):(\$?)([A-Z]{1,3})$/.exec(t)){const t=z(e[2]),p=z(e[4]);return n=Math.min(t,p),a=Math.max(t,p),l=!!e[t<=p?1:3],u=!!e[t<=p?3:1],i=!0,c=!0,{top:r,left:n,bottom:o,right:a,$top:i,$left:l,$bottom:c,$right:u}}if(e=/^(\$?)([1-9]\d{0,6}):(\$?)([1-9]\d{0,6})$/.exec(t)){const t=V(e[2]),p=V(e[4]);return r=Math.min(t,p),o=Math.max(t,p),i=!!e[t<=p?1:3],c=!!e[t<=p?3:1],l=!0,u=!0,{top:r,left:n,bottom:o,right:a,$top:i,$left:l,$bottom:c,$right:u}}{const[p,s]=t.split(":");if(e=/^(\$?)([A-Z]{1,3})(\$?)([1-9]\d{0,6})$/i.exec(p))return n=z(e[2]),r=V(e[4]),l=!!e[1],i=!!e[3],s&&(e=/^(\$?)([A-Z]{1,3})(\$?)([1-9]\d{0,6})$/i.exec(s))?(a=z(e[2]),o=V(e[4]),u=!!e[1],c=!!e[3],o<r&&([r,o,i,c]=[o,r,c,i]),a<n&&([n,a,l,u]=[a,n,u,l])):(o=r,a=n,c=i,u=l),{top:r,left:n,bottom:o,right:a,$top:i,$left:l,$bottom:c,$right:u}}return null}function et(t){let e=!(arguments.length>1&&void 0!==arguments[1])||arguments[1];const r=H(t,e,I);if(r&&(r.r0||r.name)){let t=null;return r.r0&&(t=r.r1?tt(r.r0+":"+r.r1):tt(r.r0)),r.name||t?(r.range=t,delete r.r0,delete r.r1,r):null}return null}var rt={fromCol:z,toCol:K,toRelative:Y,toAbsolute:function(t){const{top:e,left:r,bottom:n,right:o}=t;return{top:e,left:r,bottom:n,right:o,$left:!0,$right:!0,$top:!0,$bottom:!0}},to:J,from:tt,parse:et};function nt(){let t=1;return()=>"fxg"+t++}function ot(t,e){return e?String(t+1):t?"["+t+"]":""}function at(t){const{r0:e,c0:r,r1:n,c1:o,$c0:a,$c1:i,$r0:l,$r1:c}=t;if(0===e&&1048575===n){const t=ot(r,a),e=ot(o,i);return"C"+(t===e?t:t+":C"+e)}if(0===r&&16383===o){const t=ot(e,l),r=ot(n,c);return"R"+(t===r?t:t+":R"+r)}const u=ot(e,l),p=ot(n,c),s=ot(r,a),f=ot(o,i);return u!==p||s!==f?"R"+u+"C"+s+":R"+p+"C"+f:"R"+u+"C"+s}function it(t){let e=0,r=0,n=1048575,o=16383,a=!1,i=!1,l=!1,c=!1;const u=/^R(?:\[([+-]?\d+)\]|(\d+))?/.exec(t);u?(u[1]?e=parseInt(u[1],10):u[2]&&(e=parseInt(u[2],10)-1,a=!0),n=e,l=a,t=t.slice(u[0].length)):(a=!0,l=!0);const p=/^C(?:\[([+-]?\d+)\]|(\d+))?/.exec(t);return p?(p[1]?r=parseInt(p[1],10):p[2]&&(r=parseInt(p[2],10)-1,i=!0),o=r,c=i,t=t.slice(p[0].length)):(i=!0,c=!0),!u&&!p||t.length?null:{r0:e,c0:r,r1:n,c1:o,$r0:a,$c0:i,$r1:l,$c1:c}}function lt(t){const[e,r]=t.split(":",2),n=it(e);if(!n)return null;if(n&&r){const t=it(r);if(!t)return null;n.r1=t.r1,n.c1=t.c1,n.$r1=t.$r1,n.$c1=t.$c1}return n}var ct={to:at,from:lt,parse:function(t){let e=!(arguments.length>1&&void 0!==arguments[1])||arguments[1];const r=H(t,e,w);if(r&&(r.r0||r.name)){const t=r.r1?lt(r.r0+":"+r.r1):lt(r.r0);return r.name||t?(r.range=t,delete r.r0,delete r.r1,r):null}return null}};function ut(t,e,r,n){let o=t;return e||(o=r+t,o<0&&(o=n+o+1),o>n&&(o-=n+1)),o}const pt={OPERATOR:"operator",BOOLEAN:"bool",ERROR:"error",NUMBER:"number",FUNCTION:"function",NEWLINE:"newline",WHITESPACE:"whitespace",STRING:"string",PATH_QUOTE:"path-quote",PATH_BRACE:"path-brace",PATH_PREFIX:"path-prefix",RANGE:n,RANGE_BEAM:o,RANGE_NAMED:"range-named",FX_PREFIX:"fx-prefix",UNKNOWN:"unknown"};exports.MAX_COLS=16383,exports.MAX_ROWS=1048575,exports.a1=rt,exports.addMeta=function(t){let{sheetName:e="",workbookName:r=""}=arguments.length>1&&void 0!==arguments[1]?arguments[1]:{};const a=[];let i=null;const l={},c=nt();return t.forEach(((t,u)=>{if(t.index=u,t.depth=a.length,"("===t.value)t.depth=a.length+1,a.push(t);else if(")"===t.value){const e=a.pop();if(e){const r=c();t.groupId=r,t.depth=e.depth,e.groupId=r}else t.error=!0}else if("{"===t.value)i?t.error=!0:i=t;else if("}"===t.value){if(i){const e=c();t.groupId=e,i.groupId=e}else t.error=!0;i=null}else if(t.type===n||t.type===o){const n=et(t.value,!1),o=n&&"[".concat(n.workbookName||r,"]").concat(n.sheetName||e,"!").concat(n.range?J(Y(n.range)):n.name).toLowerCase();o&&(o in l?t.groupId=l[o]:(t.groupId=c(),l[o]=t.groupId))}else"unknown"===t.type&&(t.error=!0)})),t},exports.rc=ct,exports.tokenTypes=pt,exports.tokenize=_,exports.translateToA1=function(t,e){const r=tt(e),a="string"==typeof t,i=a?_(t,{emitRanges:!1,mergeRanges:!1,r1c1:!0}):t;return i.forEach((t=>{if(t.type===n||t.type===o){const e={},n=lt(t.value),o=ut(n.r0,n.$r0,r.top,1048575),a=ut(n.r1,n.$r1,r.top,1048575);o>a?(e.top=a,e.$top=n.$r1,e.bottom=o,e.$bottom=n.$r0):(e.top=o,e.$top=n.$r0,e.bottom=a,e.$bottom=n.$r1);const i=ut(n.c0,n.$c0,r.left,16383),l=ut(n.c1,n.$c1,r.left,16383);i>l?(e.left=l,e.$left=n.$c1,e.right=i,e.$right=n.$c0):(e.left=i,e.$left=n.$c0,e.right=l,e.$right=n.$c1),t.value=J(e)}})),a?i.map((t=>t.value)).join(""):i},exports.translateToRC=function(t,e){const{top:r,left:a}=tt(e),i="string"==typeof t,l=i?_(t,{emitRanges:!1,mergeRanges:!1,r1c2:!1}):t;return l.forEach((t=>{if(t.type===n){const e={},n=tt(t.value);e.r0=n.$top?n.top:n.top-r,e.$r0=n.$top,e.r1=n.$bottom?n.bottom:n.bottom-r,e.$r1=n.$bottom,e.c0=n.$left?n.left:n.left-a,e.$c0=n.$left,e.c1=n.$right?n.right:n.right-a,e.$c1=n.$right,t.value=at(e)}else if(t.type===o){const e={},n=tt(t.value);e.r0=n.$top?n.top:n.top-r,e.$r0=n.$top,e.r1=n.$bottom?n.bottom:n.bottom-r,e.$r1=n.$bottom,e.c0=n.$left?n.left:n.left-a,e.$c0=n.$left,e.c1=n.$right?n.right:n.right-a,e.$c1=n.$right,t.value=at(e)}})),i?l.map((t=>t.value)).join(""):l};
2
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"fx.js","sources":["../lib/quickVerify.js","../lib/constants.js","../lib/lexer.js","../lib/parseRef.js","../lib/a1.js","../lib/addMeta.js","../lib/rc.js","../lib/translate.js","../lib/index.js"],"sourcesContent":["export function quickVerifyRangeA1 (tokenValue) {\n  // Quickly determine if range valus are out of bounds: Split the string into\n  // numeric, alphabetical, and \"other\" chunks and parse them as base 36 numbers.\n  // Both rows and cols can be represented in base 36 and we can decide which we\n  // hold by checking if the chunk was already a finite number. So we can simply\n  // compare against the base 36 parsed versions of XFD and 1048576.\n  return tokenValue.split(/(\\d+|[a-zA-Z]+)/).every(d => {\n    const n = parseInt(d, 36);\n    if (!isNaN(n)) {\n      return isFinite(d)\n        ? n <= 2183880786 // max rows\n        : n <= 43321;     // max cols\n    }\n    return true;\n  });\n}\n\nconst bounds = {\n  'R': 1048577,\n  'R[': 1048576,\n  'C': 16385,\n  'C[': 16384\n};\nexport function quickVerifyRangeRC (tokenValue) {\n  const reNums = /([RC]\\[?)(-?\\d+)/gi;\n  let m;\n  while ((m = reNums.exec(tokenValue)) !== null) {\n    const x = bounds[m[1]];\n    const val = parseInt(m[2], 10);\n    if (val >= x || val <= -x) {\n      return false;\n    }\n  }\n  return true;\n}\n","import { quickVerifyRangeA1, quickVerifyRangeRC } from './quickVerify.js';\n\nexport const OPERATOR = 'operator';\nexport const BOOLEAN = 'bool';\nexport const ERROR = 'error';\nexport const NUMBER = 'number';\nexport const FUNCTION = 'function';\nexport const NEWLINE = 'newline';\nexport const WHITESPACE = 'whitespace';\nexport const STRING = 'string';\nexport const PATH_QUOTE = 'path-quote';\nexport const PATH_BRACE = 'path-brace';\nexport const PATH_PREFIX = 'path-prefix';\nexport const RANGE = 'range';\nexport const RANGE_BEAM = 'range-beam';\nexport const RANGE_NAMED = 'range-named';\nexport const FX_PREFIX = 'fx-prefix';\nexport const UNKNOWN = 'unknown';\n\nexport const MAX_COLS = 2 ** 14 - 1; // 16383\nexport const MAX_ROWS = 2 ** 20 - 1; // 1048575\n\nconst re_ERROR = /^#(NAME\\?|FIELD!|CALC!|VALUE!|REF!|DIV\\/0!|NULL!|NUM!|N\\/A|GETTING_DATA\\b|SPILL!|UNKNOWN!|FIELD\\b|CALC\\b|SYNTAX\\?|ERROR!)/i;\nconst re_OPERATOR = /^(<=|>=|<>|[-+/*^%&<>=]|[{},;]|[()]|@|:|!|#)/;\nconst re_BOOLEAN = /^(TRUE|FALSE)\\b/i;\nconst re_FUNCTION = /^[A-Z_]+[A-Z\\d_.]+(?=\\s*\\()/i;\nconst re_NEWLINE = /^\\n+/;\nconst re_WHITESPACE = /^[ \\f\\r\\t\\v\\u00a0\\u1680\\u2000-\\u200a\\u2028\\u2029\\u202f\\u205f\\u3000\\ufeff]+/;\nconst re_STRING = /^\"(?:\"\"|[^\"])*(\"|$)/;\nconst re_PATH_QUOTE = /^'(?:''|[^'])*('|$)/;\nconst re_PATH_BRACE = /^\\[(?:[^\\]])+(\\]|$)/;\nconst re_PATH_PREFIX = /^([^ \\t\\n$!\"`'#%&(){}<>,;:^@|~=*+-]+)(?=!)/; // Sheets: [^:\\\\/?*[\\]]{0,31} (but WB names?)\nconst re_A1COL = /^\\$?[A-Z]{1,3}:\\$?[A-Z]{1,3}/i;\nconst re_A1ROW = /^\\$?[1-9][0-9]{0,6}:\\$?[1-9][0-9]{0,6}/i;\nconst re_A1RANGE = /^\\$?[A-Z]{1,3}\\$?[1-9][0-9]{0,6}/i;\nconst rPart = '(?:R(?:\\\\[[+-]?\\\\d+\\\\]|[1-9][0-9]{0,6})?)';\nconst cPart = '(?:C(?:\\\\[[+-]?\\\\d+\\\\]|[1-9][0-9]{0,4})?)';\nconst re_RCCOL = new RegExp(`^${cPart}(:${cPart})?(?=\\\\W|$)`, 'i');\nconst re_RCROW = new RegExp(`^${rPart}(:${rPart})?(?=\\\\W|$)`, 'i');\nconst re_RCRANGE = new RegExp(`^(?:(?=[RC])${rPart}${cPart})`, 'i');\nconst re_NUMBER = /^(?:\\d+(\\.\\d+)?(?:[eE][+-]?\\d+)?|\\d+)/;\nconst re_NAMED = /^[A-Z\\d\\\\_.?]+/i; // FIXME there are stricter rules for this!\n// const re_NAMED = /^(?![RC]$)[A-ZÀ-ȳ_\\\\][\\\\?\\wÀ-ȳ.]{0,255}$/i;\n\nexport const tokenHandlersA1 = [\n  [ ERROR,       re_ERROR ],\n  [ OPERATOR,    re_OPERATOR ],\n  [ BOOLEAN,     re_BOOLEAN ],\n  [ FUNCTION,    re_FUNCTION ],\n  [ NEWLINE,     re_NEWLINE ],\n  [ WHITESPACE,  re_WHITESPACE ],\n  [ STRING,      re_STRING ],\n  [ PATH_QUOTE,  re_PATH_QUOTE ],\n  [ PATH_BRACE,  re_PATH_BRACE ],\n  [ PATH_PREFIX, re_PATH_PREFIX ],\n  [ RANGE,       re_A1RANGE, quickVerifyRangeA1 ],\n  [ RANGE_BEAM,  re_A1COL, quickVerifyRangeA1 ],\n  [ RANGE_BEAM,  re_A1ROW, quickVerifyRangeA1 ],\n  [ NUMBER,      re_NUMBER ],\n  [ RANGE_NAMED, re_NAMED ]\n];\n\nexport const tokenHandlersRC = [\n  [ ERROR,       re_ERROR ],\n  [ OPERATOR,    re_OPERATOR ],\n  [ BOOLEAN,     re_BOOLEAN ],\n  [ FUNCTION,    re_FUNCTION ],\n  [ NEWLINE,     re_NEWLINE ],\n  [ WHITESPACE,  re_WHITESPACE ],\n  [ STRING,      re_STRING ],\n  [ PATH_QUOTE,  re_PATH_QUOTE ],\n  [ PATH_BRACE,  re_PATH_BRACE ],\n  [ PATH_PREFIX, re_PATH_PREFIX ],\n  [ RANGE,       re_RCRANGE, quickVerifyRangeRC ],\n  [ RANGE_BEAM,  re_RCROW, quickVerifyRangeRC ],\n  [ RANGE_BEAM,  re_RCCOL, quickVerifyRangeRC ],\n  [ NUMBER,      re_NUMBER ],\n  [ RANGE_NAMED, re_NAMED ]\n];\n\nexport const tokenHandlersRefsA1 = [\n  [ OPERATOR,    /^[!:]/ ],\n  [ PATH_QUOTE,  re_PATH_QUOTE ],\n  [ PATH_BRACE,  re_PATH_BRACE ],\n  [ PATH_PREFIX, re_PATH_PREFIX ],\n  [ RANGE,       re_A1RANGE, quickVerifyRangeA1 ],\n  [ RANGE_BEAM,  re_A1COL, quickVerifyRangeA1 ],\n  [ RANGE_BEAM,  re_A1ROW, quickVerifyRangeA1 ],\n  [ RANGE_NAMED, re_NAMED ]\n];\n\nexport const tokenHandlersRefsRC = [\n  [ OPERATOR,    /^!/ ],\n  [ PATH_QUOTE,  re_PATH_QUOTE ],\n  [ PATH_BRACE,  re_PATH_BRACE ],\n  [ PATH_PREFIX, re_PATH_PREFIX ],\n  [ RANGE,       re_RCRANGE, quickVerifyRangeRC ],\n  [ RANGE_BEAM,  re_RCROW, quickVerifyRangeRC ],\n  [ RANGE_BEAM,  re_RCCOL, quickVerifyRangeRC ],\n  [ RANGE_NAMED, re_NAMED ]\n];\n","import {\n  FX_PREFIX,\n  NEWLINE,\n  NUMBER,\n  OPERATOR,\n  PATH_BRACE,\n  PATH_PREFIX,\n  PATH_QUOTE,\n  RANGE,\n  RANGE_BEAM,\n  RANGE_NAMED,\n  STRING,\n  UNKNOWN,\n  WHITESPACE,\n  tokenHandlersA1,\n  tokenHandlersRC\n} from './constants.js';\n\nconst isType = (t, type) => t && t.type === type;\nconst isRangeOp = t => t && t.value === ':';\nconst isBangOp = t => t && t.value === '!';\n\nconst defaultOptions = {\n  emitRanges: false,\n  mergeRanges: true,\n  negativeNumbers: false,\n  r1c1: false\n};\n\nexport function getTokens (fx, tokenHandlers, options = {}) {\n  const { emitRanges, mergeRanges, negativeNumbers } = Object.assign({}, defaultOptions, options);\n  const tokens = [];\n  let pos = 0;\n  const lookBehind = n => tokens[tokens.length - n];\n  const lookBehindIgnoreWS = n => {\n    const noWs = tokens.filter(t => !isType(t, WHITESPACE) && !isType(t, NEWLINE));\n    return noWs[noWs.length - n];\n  };\n\n  if (/^=/.test(fx)) {\n    const token = {\n      type: FX_PREFIX,\n      value: '=',\n      ...(emitRanges ? { range: [ 0, 1 ] } : {})\n    };\n    pos++;\n    tokens.push(token);\n  }\n\n  while (pos < fx.length) {\n    const startPos = pos;\n    const s = fx.slice(pos);\n    let tokenType = '';\n    let tokenValue = '';\n    for (let i = 0; i < tokenHandlers.length; i++) {\n      const [ type, reTest, fnTest ] = tokenHandlers[i];\n      const m = reTest.exec(s);\n      if (m && (!fnTest || fnTest(m[0]))) {\n        tokenType = type;\n        tokenValue = m[0];\n        pos += m[0].length;\n        break;\n      }\n    }\n\n    if (!tokenType) {\n      tokenType = UNKNOWN;\n      tokenValue = fx[pos];\n      pos++;\n    }\n\n    let token = {\n      type: tokenType,\n      value: tokenValue,\n      ...(emitRanges ? { range: [ startPos, pos ] } : {})\n    };\n\n    if (tokenType === STRING && !tokenValue.endsWith('\"')) {\n      token.unterminated = true;\n    }\n\n    if (negativeNumbers && tokenType === NUMBER) {\n      const last1 = lookBehind(1);\n      if (last1 && isType(last1, OPERATOR) && last1.value === '-') {\n        // we have a number preceded by a minus\n        const last2 = lookBehindIgnoreWS(2);\n        // missing last2 means we are at the start of the stream\n        if (!last2 || isType(last2, FX_PREFIX) || (isType(last2, OPERATOR) && ![ '%', '}', ')', '#' ].includes(last2.value))) {\n          tokens.pop();\n          token.value = '-' + tokenValue;\n        }\n      }\n    }\n\n    if (mergeRanges) {\n      if (tokenType === RANGE || tokenType === RANGE_NAMED || tokenType === RANGE_BEAM) {\n        const merge = [];\n        // join A1:A1 or RC:RC-\n        if (isRangeOp(lookBehind(1)) && isType(lookBehind(2), RANGE) && tokenType === RANGE) {\n          // should not be done this if current or last ranges are A:A or 1:1, or R or C)\n          merge.unshift(...tokens.splice(-2, 2));\n        }\n        // join prefixes\n        if (isBangOp(lookBehind(1))) {\n          if (isType(lookBehind(2), PATH_QUOTE) || isType(lookBehind(2), PATH_BRACE)) {\n            merge.unshift(...tokens.splice(-2, 2));\n          }\n          else if (isType(lookBehind(2), PATH_PREFIX)) {\n            const n = isType(lookBehind(3), PATH_BRACE) ? 3 : 2;\n            merge.unshift(...tokens.splice(-n, n));\n          }\n        }\n        // want to merge?\n        if (merge.length) {\n          token = {\n            type: tokenType,\n            value: merge.map(d => d.value).join('') + tokenValue,\n            ...(emitRanges ? { range: [ merge[0].range[0], pos ] } : {})\n          };\n        }\n      }\n    }\n    tokens.push(token);\n  }\n\n  return tokens;\n}\n\n// Formulas can either have RC or A1 style refs, not both: because C1 and R1 are both!\n// Refmode: A1 | RC | Anostic emit ranges with RANGE\nexport function tokenize (fx, options = {}) {\n  const tokenHandlers = options.r1c1 ? tokenHandlersRC : tokenHandlersA1;\n  return getTokens(fx, tokenHandlers, options);\n}\n","import { FX_PREFIX, PATH_BRACE, PATH_PREFIX, PATH_QUOTE, RANGE, RANGE_NAMED, RANGE_BEAM } from './constants.js';\nimport { getTokens } from './lexer.js';\n\nconst unquote = d => d.slice(1, -1).replace(/''/g, \"'\");\n\nconst pRangeOp = t => t && t.value === ':' && {};\nconst pRange = t => t && t.type === RANGE && { r0: t.value };\nconst pRange2 = t => t && t.type === RANGE && { r1: t.value };\nconst pBang = t => t && t.value === '!' && {};\nconst pBeam = t => t && t.type === RANGE_BEAM && { r0: t.value };\nconst pSheet = t => t && t.type === PATH_PREFIX && /^[^:\\\\/?*[\\]]{0,31}$/.test(t.value) && { sheetName: t.value };\nconst pFile = t => t && t.type === PATH_BRACE && { workbookName: t.value.slice(1, -1) };\nconst pFile2 = t => t && t.type === PATH_PREFIX && { workbookName: t.value };\nconst pNamed = t => t && t.type === RANGE_NAMED && { name: t.value };\n\nconst pQuoted = t => {\n  if (t && t.type === PATH_QUOTE) {\n    const m = /(?:\\[(.+?)\\])?([^[\\]]+?)$/.exec(unquote(t.value));\n    if (m) {\n      const [ , file, sheet ] = m;\n      if (!sheet || /^[^:\\\\/?*[\\]]{0,31}$/.test(sheet)) {\n        return {\n          workbookName: file || '',\n          sheetName: sheet || ''\n        };\n      }\n    }\n  }\n};\nconst pQuoted2 = t => t && t.type === PATH_QUOTE && { workbookName: unquote(t.value) };\n\nconst validRuns = [\n  [ pRange ],\n  [ pRange, pRangeOp, pRange2 ],\n  [ pBeam ],\n  [ pQuoted, pBang, pRange ],\n  [ pQuoted, pBang, pRange, pRangeOp, pRange2 ],\n  [ pQuoted, pBang, pBeam ],\n  [ pSheet, pBang, pRange ],\n  [ pSheet, pBang, pRange, pRangeOp, pRange2 ],\n  [ pSheet, pBang, pBeam ],\n  [ pFile, pSheet, pBang, pRange ],\n  [ pFile, pSheet, pBang, pRange, pRangeOp, pRange2 ],\n  [ pFile, pSheet, pBang, pBeam ]\n];\n\nconst validRunsNamed = validRuns.concat([\n  [ pNamed ],\n  [ pFile2, pBang, pNamed ],\n  [ pQuoted2, pBang, pNamed ]\n]);\n\nexport function parseRef (ref, allow_named = true, tokenHandlers = []) {\n  const tokens = getTokens(ref, tokenHandlers, {\n    emitRanges: false,\n    mergeRanges: false\n  });\n  const refData = {\n    sheetName: '',\n    workbookName: '',\n    r0: '',\n    r1: '',\n    name: ''\n  };\n  // discard the \"=\"-prefix if it is there\n  if (tokens.length && tokens[0].type === FX_PREFIX) {\n    tokens.shift();\n  }\n  const runs = allow_named ? validRunsNamed : validRuns;\n  for (let i = 0; i < runs.length; i++) {\n    const data = { ...refData };\n    if (runs[i].length === tokens.length) {\n      const valid = runs[i].every((parse, j) => {\n        const d = parse(tokens[j]);\n        Object.assign(data, d);\n        return d;\n      });\n      if (valid) {\n        return data;\n      }\n    }\n  }\n  return null;\n}\n","import { MAX_ROWS, MAX_COLS, tokenHandlersRefsA1 } from './constants.js';\nimport { parseRef } from './parseRef.js';\n\nexport function fromCol (colStr) {\n  const c = (colStr || '').toUpperCase();\n  let d = 0;\n  let i = 0;\n  for (; i !== c.length; ++i) {\n    const chr = c.charCodeAt(i);\n    if (chr >= 65 && chr <= 90) { // omits any non A-Z character\n      d = 26 * d + chr - 64;\n    }\n  }\n  return d - 1;\n}\n\nexport function toCol (left) {\n  let n = left;\n  let c = '';\n  while (n >= 0) {\n    c = String.fromCharCode(n % 26 + 65) + c;\n    n = Math.floor(n / 26) - 1;\n  }\n  return c;\n}\n\nexport function fromRow (rowStr) {\n  return +rowStr - 1;\n}\n\nexport function toRow (top) {\n  return String(top + 1);\n}\n\nexport function toRelative (range) {\n  const { top, left, bottom, right } = range;\n  return { top, left, bottom, right, $left: false, $right: false, $top: false, $bottom: false };\n}\n\nexport function toAbsolute (range) {\n  const { top, left, bottom, right } = range;\n  return { top, left, bottom, right, $left: true, $right: true, $top: true, $bottom: true };\n}\n\nexport function toA1 (range) {\n  const toAbs = d => (d ? '$' : '');\n  const { top, left, bottom, right, $left, $right, $top, $bottom } = range;\n  // A:A\n  if (top === 0 && bottom === MAX_ROWS) {\n    return toAbs($left) + toCol(left) + ':' + toAbs($right) + toCol(right);\n  }\n  // 1:1\n  if (left === 0 && right === MAX_COLS) {\n    return toAbs($top) + toRow(top) + ':' + toAbs($bottom) + toRow(bottom);\n  }\n  // A1:A1\n  if (right != null && bottom != null && (right !== left || bottom !== top)) {\n    return toAbs($left) + toCol(left) + toAbs($top) + toRow(top) + ':' + toAbs($right) + toCol(right) + toAbs($bottom) + toRow(bottom);\n  }\n  // A1\n  return toAbs($left) + toCol(left) + toAbs($top) + toRow(top);\n}\n\nexport function fromA1 (rangeStr) {\n  let m;\n  let top = 0;\n  let left = 0;\n  let bottom = MAX_ROWS;\n  let right = MAX_COLS;\n  let $top = false;\n  let $left = false;\n  let $bottom = false;\n  let $right = false;\n  // A:A\n  if ((m = /^(\\$?)([A-Z]{1,3}):(\\$?)([A-Z]{1,3})$/.exec(rangeStr))) {\n    const a = fromCol(m[2]);\n    const b = fromCol(m[4]);\n    left = Math.min(a, b);\n    right = Math.max(a, b);\n    $left = !!m[a <= b ? 1 : 3];\n    $right = !!m[a <= b ? 3 : 1];\n    $top = true;\n    $bottom = true;\n    return { top, left, bottom, right, $top, $left, $bottom, $right };\n  }\n  // 1:1\n  else if ((m = /^(\\$?)([1-9]\\d{0,6}):(\\$?)([1-9]\\d{0,6})$/.exec(rangeStr))) {\n    const a = fromRow(m[2]);\n    const b = fromRow(m[4]);\n    top = Math.min(a, b);\n    bottom = Math.max(a, b);\n    $top = !!m[a <= b ? 1 : 3];\n    $bottom = !!m[a <= b ? 3 : 1];\n    $left = true;\n    $right = true;\n    return { top, left, bottom, right, $top, $left, $bottom, $right };\n  }\n  // A1 | A1:B2\n  else {\n    const [ part1, part2 ] = rangeStr.split(':');\n    if ((m = /^(\\$?)([A-Z]{1,3})(\\$?)([1-9]\\d{0,6})$/i.exec(part1))) {\n      left = fromCol(m[2]);\n      top = fromRow(m[4]);\n      $left = !!m[1];\n      $top = !!m[3];\n      if (part2 && (m = /^(\\$?)([A-Z]{1,3})(\\$?)([1-9]\\d{0,6})$/i.exec(part2))) {\n        right = fromCol(m[2]);\n        bottom = fromRow(m[4]);\n        $right = !!m[1];\n        $bottom = !!m[3];\n        // need to flip?\n        if (bottom < top) {\n          [ top, bottom, $top, $bottom ] = [ bottom, top, $bottom, $top ];\n        }\n        if (right < left) {\n          [ left, right, $left, $right ] = [ right, left, $right, $left ];\n        }\n      }\n      else {\n        bottom = top;\n        right = left;\n        $bottom = $top;\n        $right = $left;\n      }\n      return { top, left, bottom, right, $top, $left, $bottom, $right };\n    }\n  }\n  return null;\n}\n\nexport function parseA1Ref (ref, allow_named = true) {\n  const d = parseRef(ref, allow_named, tokenHandlersRefsA1);\n  if (d && (d.r0 || d.name)) {\n    let range = null;\n    if (d.r0) {\n      range = d.r1 ? fromA1(d.r0 + ':' + d.r1) : fromA1(d.r0);\n    }\n    if (d.name || range) {\n      d.range = range;\n      delete d.r0;\n      delete d.r1;\n      return d;\n    }\n    else {\n      return null;\n    }\n  }\n  return null;\n}\n\nexport default {\n  fromCol,\n  toCol,\n  toRelative,\n  toAbsolute,\n  to: toA1,\n  from: fromA1,\n  parse: parseA1Ref\n};\n","import { RANGE, RANGE_BEAM, UNKNOWN } from './constants.js';\nimport { parseA1Ref, toRelative, toA1 } from './a1.js';\n\nfunction getIDer () {\n  let i = 1;\n  return () => 'fxg' + (i++);\n}\n\n// when context is Sheet1, we should consider Sheet!A1 == A1\nexport function addMeta (tokens, { sheetName = '', workbookName = '' } = {}) {\n  const parenStack = [];\n  let arrayStart = null;\n  const a1Map = {};\n  const uid = getIDer();\n\n  tokens.forEach((token, i) => {\n    token.index = i;\n    token.depth = parenStack.length;\n    if (token.value === '(') {\n      token.depth = parenStack.length + 1;\n      parenStack.push(token);\n    }\n    else if (token.value === ')') {\n      const counter = parenStack.pop();\n      if (counter) {\n        const pairId = uid();\n        token.groupId = pairId;\n        token.depth = counter.depth;\n        counter.groupId = pairId;\n      }\n      else {\n        token.error = true;\n      }\n    }\n    else if (token.value === '{') {\n      if (!arrayStart) {\n        arrayStart = token;\n      }\n      else {\n        token.error = true;\n      }\n    }\n    else if (token.value === '}') {\n      if (arrayStart) {\n        const pairId = uid();\n        token.groupId = pairId;\n        arrayStart.groupId = pairId;\n      }\n      else {\n        token.error = true;\n      }\n      arrayStart = null;\n    }\n    else if (token.type === RANGE || token.type === RANGE_BEAM) {\n      const ref = parseA1Ref(token.value, false);\n      const a1 = ref && (`[${ref.workbookName || workbookName}]${ref.sheetName || sheetName}!${ref.range ? toA1(toRelative(ref.range)) : ref.name}`).toLowerCase();\n      if (a1) {\n        if (a1 in a1Map) {\n          token.groupId = a1Map[a1];\n        }\n        else {\n          token.groupId = uid();\n          a1Map[a1] = token.groupId;\n        }\n      }\n    }\n    else if (token.type === UNKNOWN) {\n      token.error = true;\n    }\n  });\n  return tokens;\n}\n","/*\n  RC notation works differently from A1 in that we can't merge static references joined by `:`.\n  Merging can only work between references that are relative/absolute on the same axes, so:\n  R1C1:R2C2 will work, R[1]C1:R[2]C2 will also work, but R[1]C[1]:R2C2 cannot be represented\n  simply by a rectangle as without knowing the cell context.\n*/\n\nimport { MAX_ROWS, MAX_COLS, tokenHandlersRefsRC } from './constants.js';\nimport { parseRef } from './parseRef.js';\n\nfunction toCoord (value, isAbs) {\n  if (isAbs) {\n    return String(value + 1);\n  }\n  return value ? '[' + value + ']' : '';\n}\n\nexport function toRC (range) {\n  const { r0, c0, r1, c1, $c0, $c1, $r0, $r1 } = range;\n  // C:C\n  if (r0 === 0 && r1 === MAX_ROWS) {\n    const a = toCoord(c0, $c0);\n    const b = toCoord(c1, $c1);\n    return 'C' + (a === b ? a : a + ':C' + b);\n  }\n  // R:R\n  if (c0 === 0 && c1 === MAX_COLS) {\n    const a = toCoord(r0, $r0);\n    const b = toCoord(r1, $r1);\n    return 'R' + (a === b ? a : a + ':R' + b);\n  }\n  // RC:RC\n  const s_r0 = toCoord(r0, $r0);\n  const s_r1 = toCoord(r1, $r1);\n  const s_c0 = toCoord(c0, $c0);\n  const s_c1 = toCoord(c1, $c1);\n  if (s_r0 !== s_r1 || s_c0 !== s_c1) {\n    return 'R' + s_r0 + 'C' + s_c0 + ':R' + s_r1 + 'C' + s_c1;\n  }\n  // RC\n  return 'R' + s_r0 + 'C' + s_c0;\n}\n\nfunction parseRCPart (ref) {\n  let r0 = 0;\n  let c0 = 0;\n  let r1 = MAX_ROWS;\n  let c1 = MAX_COLS;\n  let $r0 = false;\n  let $c0 = false;\n  let $r1 = false;\n  let $c1 = false;\n  // R part\n  const rm = /^R(?:\\[([+-]?\\d+)\\]|(\\d+))?/.exec(ref);\n  if (rm) {\n    if (rm[1]) {\n      r0 = parseInt(rm[1], 10);\n    }\n    else if (rm[2]) {\n      r0 = parseInt(rm[2], 10) - 1;\n      $r0 = true;\n    }\n    r1 = r0;\n    $r1 = $r0;\n    ref = ref.slice(rm[0].length);\n  }\n  else {\n    // r0 = 0, r1 = MAX_ROWS\n    $r0 = true;\n    $r1 = true;\n  }\n  // C part\n  const cm = /^C(?:\\[([+-]?\\d+)\\]|(\\d+))?/.exec(ref);\n  if (cm) {\n    if (cm[1]) {\n      c0 = parseInt(cm[1], 10);\n    }\n    else if (cm[2]) {\n      c0 = parseInt(cm[2], 10) - 1;\n      $c0 = true;\n    }\n    c1 = c0;\n    $c1 = $c0;\n    ref = ref.slice(cm[0].length);\n  }\n  else {\n    // c0 = 0, c1 = MAX_COLS\n    $c0 = true;\n    $c1 = true;\n  }\n  // must have at least one part (and nothing more)\n  if ((!rm && !cm) || ref.length) {\n    return null;\n  }\n  return { r0, c0, r1, c1, $r0, $c0, $r1, $c1 };\n}\n\nexport function fromRC (ref) {\n  const [ part1, part2 ] = ref.split(':', 2);\n  const range = parseRCPart(part1);\n  if (!range) {\n    return null;\n  }\n  if (range && part2) {\n    const extendTo = parseRCPart(part2);\n    if (extendTo) {\n      // Note: R[-1]C[-1]:R1C1 is only meaningful once we have an anchor\n      range.r1 = extendTo.r1;\n      range.c1 = extendTo.c1;\n      range.$r1 = extendTo.$r1;\n      range.$c1 = extendTo.$c1;\n    }\n    else {\n      return null;\n    }\n  }\n  return range;\n}\n\nexport function parseRCRef (ref, allow_named = true) {\n  const d = parseRef(ref, allow_named, tokenHandlersRefsRC);\n  if (d && (d.r0 || d.name)) {\n    const range = d.r1 ? fromRC(d.r0 + ':' + d.r1) : fromRC(d.r0);\n    if (d.name || range) {\n      d.range = range;\n      delete d.r0;\n      delete d.r1;\n      return d;\n    }\n    else {\n      return null;\n    }\n  }\n  return null;\n}\n\nexport default {\n  to: toRC,\n  from: fromRC,\n  parse: parseRCRef\n};\n","import { RANGE, RANGE_BEAM, MAX_ROWS, MAX_COLS } from './constants.js';\nimport { fromA1, toA1 } from './a1.js';\nimport { fromRC, toRC } from './rc.js';\nimport { tokenize } from './lexer.js';\n\nexport function translateToRC (fx, anchorCell) {\n  const { top, left } = fromA1(anchorCell);\n  const isString = typeof fx === 'string';\n\n  const tokens = isString\n    ? tokenize(fx, { emitRanges: false, mergeRanges: false, r1c2: false })\n    : fx;\n\n  tokens.forEach(token => {\n    if (token.type === RANGE) {\n      const range = {};\n      const d = fromA1(token.value);\n\n      range.r0 = d.$top ? d.top : d.top - top;\n      range.$r0 = d.$top;\n      range.r1 = d.$bottom ? d.bottom : d.bottom - top;\n      range.$r1 = d.$bottom;\n\n      range.c0 = d.$left ? d.left : d.left - left;\n      range.$c0 = d.$left;\n      range.c1 = d.$right ? d.right : d.right - left;\n      range.$c1 = d.$right;\n\n      // console.error(token.value, d, range, [ top, left ]);\n      token.value = toRC(range);\n    }\n    else if (token.type === RANGE_BEAM) {\n      const range = {};\n      const d = fromA1(token.value);\n\n      range.r0 = d.$top ? d.top : d.top - top;\n      range.$r0 = d.$top;\n      range.r1 = d.$bottom ? d.bottom : d.bottom - top;\n      range.$r1 = d.$bottom;\n\n      range.c0 = d.$left ? d.left : d.left - left;\n      range.$c0 = d.$left;\n      range.c1 = d.$right ? d.right : d.right - left;\n      range.$c1 = d.$right;\n\n      token.value = toRC(range);\n    }\n  });\n\n  return isString\n    ? tokens.map(d => d.value).join('')\n    : tokens;\n}\n\nfunction toFixed (val, abs, base, max) {\n  let v = val;\n  if (!abs) {\n    v = base + val;\n    // Excel \"wraps around\" when value goes out of lower bounds.\n    // It's a bit quirky on entry as Excel wants to re-rewite the\n    // references but behaviour is consistent with INDIRECT:\n    // ... In A1: RC[-1] => R1C[16383].\n    if (v < 0) {\n      v = max + v + 1;\n    }\n    // ... In B1: =RC[16383] => =RC[-1]\n    if (v > max) {\n      v -= max + 1;\n    }\n  }\n  return v;\n}\n\nexport function translateToA1 (fx, anchorCell) {\n  const anchor = fromA1(anchorCell);\n  const isString = typeof fx === 'string';\n\n  const tokens = isString\n    ? tokenize(fx, { emitRanges: false, mergeRanges: false, r1c1: true })\n    : fx;\n\n  tokens.forEach(token => {\n    if (token.type === RANGE || token.type === RANGE_BEAM) {\n      const range = {};\n      const d = fromRC(token.value);\n      const r0 = toFixed(d.r0, d.$r0, anchor.top, MAX_ROWS);\n      const r1 = toFixed(d.r1, d.$r1, anchor.top, MAX_ROWS);\n      if (r0 > r1) {\n        range.top = r1;\n        range.$top = d.$r1;\n        range.bottom = r0;\n        range.$bottom = d.$r0;\n      }\n      else {\n        range.top = r0;\n        range.$top = d.$r0;\n        range.bottom = r1;\n        range.$bottom = d.$r1;\n      }\n      const c0 = toFixed(d.c0, d.$c0, anchor.left, MAX_COLS);\n      const c1 = toFixed(d.c1, d.$c1, anchor.left, MAX_COLS);\n      if (c0 > c1) {\n        range.left = c1;\n        range.$left = d.$c1;\n        range.right = c0;\n        range.$right = d.$c0;\n      }\n      else {\n        range.left = c0;\n        range.$left = d.$c0;\n        range.right = c1;\n        range.$right = d.$c1;\n      }\n      token.value = toA1(range);\n    }\n  });\n\n  return isString\n    ? tokens.map(d => d.value).join('')\n    : tokens;\n}\n","export { tokenize } from './lexer.js';\nexport { addMeta } from './addMeta.js';\nexport { translateToRC, translateToA1 } from './translate.js';\nexport { default as a1 } from './a1.js';\nexport { default as rc } from './rc.js';\nexport { MAX_COLS, MAX_ROWS } from './constants.js';\n\nimport {\n  OPERATOR,\n  BOOLEAN,\n  ERROR,\n  NUMBER,\n  FUNCTION,\n  NEWLINE,\n  WHITESPACE,\n  STRING,\n  PATH_QUOTE,\n  PATH_BRACE,\n  PATH_PREFIX,\n  RANGE,\n  RANGE_BEAM,\n  RANGE_NAMED,\n  FX_PREFIX,\n  UNKNOWN\n} from './constants.js';\n\nexport const tokenTypes = {\n  OPERATOR,\n  BOOLEAN,\n  ERROR,\n  NUMBER,\n  FUNCTION,\n  NEWLINE,\n  WHITESPACE,\n  STRING,\n  PATH_QUOTE,\n  PATH_BRACE,\n  PATH_PREFIX,\n  RANGE,\n  RANGE_BEAM,\n  RANGE_NAMED,\n  FX_PREFIX,\n  UNKNOWN\n};\n"],"names":["quickVerifyRangeA1","tokenValue","split","every","d","n","parseInt","isNaN","isFinite","bounds","R","C","quickVerifyRangeRC","reNums","m","exec","x","val","RANGE","RANGE_BEAM","re_ERROR","re_OPERATOR","re_BOOLEAN","re_FUNCTION","re_NEWLINE","re_WHITESPACE","re_STRING","re_PATH_QUOTE","re_PATH_BRACE","re_PATH_PREFIX","re_A1COL","re_A1ROW","re_A1RANGE","rPart","cPart","re_RCCOL","RegExp","concat","re_RCROW","re_RCRANGE","re_NUMBER","re_NAMED","tokenHandlersA1","tokenHandlersRC","tokenHandlersRefsA1","tokenHandlersRefsRC","isType","t","type","isRangeOp","value","isBangOp","defaultOptions","emitRanges","mergeRanges","negativeNumbers","r1c1","getTokens","fx","tokenHandlers","options","Object","assign","tokens","pos","lookBehind","length","lookBehindIgnoreWS","noWs","filter","test","token","range","push","startPos","s","slice","tokenType","i","reTest","fnTest","endsWith","unterminated","last1","last2","includes","pop","merge","unshift","splice","map","join","tokenize","unquote","replace","pRangeOp","pRange","r0","pRange2","r1","pBang","pBeam","pSheet","sheetName","pFile","workbookName","pNamed","name","pQuoted","file","sheet","validRuns","validRunsNamed","parseRef","ref","allow_named","refData","shift","runs","data","parse","j","fromCol","colStr","c","toUpperCase","chr","charCodeAt","toCol","left","String","fromCharCode","Math","floor","fromRow","rowStr","toRow","top","toRelative","bottom","right","$left","$right","$top","$bottom","toA1","toAbs","fromA1","rangeStr","a","b","min","max","part1","part2","parseA1Ref","a1","toAbsolute","to","from","getIDer","toCoord","isAbs","toRC","c0","c1","$c0","$c1","$r0","$r1","s_r0","s_r1","s_c0","s_c1","parseRCPart","rm","cm","fromRC","extendTo","rc","toFixed","abs","base","v","tokenTypes","OPERATOR","BOOLEAN","ERROR","NUMBER","FUNCTION","NEWLINE","WHITESPACE","STRING","PATH_QUOTE","PATH_BRACE","PATH_PREFIX","RANGE_NAMED","FX_PREFIX","UNKNOWN","parenStack","arrayStart","a1Map","uid","forEach","index","depth","counter","pairId","groupId","error","toLowerCase","anchorCell","anchor","isString","r1c2"],"mappings":"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"}
@@ -0,0 +1 @@
1
+ { "type": "commonjs" }
File without changes
File without changes
@@ -0,0 +1,77 @@
1
+ import { test, Test } from 'tape';
2
+ import { FX_PREFIX, OPERATOR, NUMBER, RANGE, RANGE_BEAM } from './constants.js';
3
+ import { addMeta } from './addMeta.js';
4
+ import { tokenize } from './lexer.js';
5
+
6
+ Test.prototype.isMetaTokens = function isTokens (expr, result, opts) {
7
+ this.deepEqual(addMeta(tokenize(expr), opts), result, expr);
8
+ };
9
+
10
+ test('add extra meta to operators', t => {
11
+ // parens should be grouped and tagged with depth
12
+ t.isMetaTokens('=((1)+(1))', [
13
+ { index: 0, depth: 0, type: FX_PREFIX, value: '=' },
14
+ { index: 1, depth: 1, type: OPERATOR, value: '(', groupId: 'fxg3' },
15
+ { index: 2, depth: 2, type: OPERATOR, value: '(', groupId: 'fxg1' },
16
+ { index: 3, depth: 2, type: NUMBER, value: '1' },
17
+ { index: 4, depth: 2, type: OPERATOR, value: ')', groupId: 'fxg1' },
18
+ { index: 5, depth: 1, type: OPERATOR, value: '+' },
19
+ { index: 6, depth: 2, type: OPERATOR, value: '(', groupId: 'fxg2' },
20
+ { index: 7, depth: 2, type: NUMBER, value: '1' },
21
+ { index: 8, depth: 2, type: OPERATOR, value: ')', groupId: 'fxg2' },
22
+ { index: 9, depth: 1, type: OPERATOR, value: ')', groupId: 'fxg3' }
23
+ ]);
24
+
25
+ // don't be fooled by imbalanced parens
26
+ t.isMetaTokens('=)())', [
27
+ { index: 0, depth: 0, type: FX_PREFIX, value: '=' },
28
+ { index: 1, depth: 0, type: OPERATOR, value: ')', error: true },
29
+ { index: 2, depth: 1, type: OPERATOR, value: '(', groupId: 'fxg1' },
30
+ { index: 3, depth: 1, type: OPERATOR, value: ')', groupId: 'fxg1' },
31
+ { index: 4, depth: 0, type: OPERATOR, value: ')', error: true }
32
+ ]);
33
+
34
+ // don't be fooled by nested curlys
35
+ t.isMetaTokens('={{}}', [
36
+ { index: 0, depth: 0, type: FX_PREFIX, value: '=' },
37
+ { index: 1, depth: 0, type: OPERATOR, value: '{', groupId: 'fxg1' },
38
+ { index: 2, depth: 0, type: OPERATOR, value: '{', error: true },
39
+ { index: 3, depth: 0, type: OPERATOR, value: '}', groupId: 'fxg1' },
40
+ { index: 4, depth: 0, type: OPERATOR, value: '}', error: true }
41
+ ]);
42
+
43
+ // group ranges if they are equivalent
44
+ t.isMetaTokens("=B11,B11:B12,'Sheet11'!B11,SHEET1!$B11,sheet1!$b$11,A1:B11,[foo]Sheet1!B11,'[foo]Sheet1'!B11", [
45
+ { index: 0, depth: 0, type: FX_PREFIX, value: '=' },
46
+ { index: 1, depth: 0, type: RANGE, value: 'B11', groupId: 'fxg1' },
47
+ { index: 2, depth: 0, type: OPERATOR, value: ',' },
48
+ { index: 3, depth: 0, type: RANGE, value: 'B11:B12', groupId: 'fxg2' },
49
+ { index: 4, depth: 0, type: OPERATOR, value: ',' },
50
+ { index: 5, depth: 0, type: RANGE, value: "'Sheet11'!B11", groupId: 'fxg3' },
51
+ { index: 6, depth: 0, type: OPERATOR, value: ',' },
52
+ { index: 7, depth: 0, type: RANGE, value: 'SHEET1!$B11', groupId: 'fxg1' },
53
+ { index: 8, depth: 0, type: OPERATOR, value: ',' },
54
+ { index: 9, depth: 0, type: RANGE, value: 'sheet1!$b$11', groupId: 'fxg1' },
55
+ { index: 10, depth: 0, type: OPERATOR, value: ',' },
56
+ { index: 11, depth: 0, type: RANGE, value: 'A1:B11', groupId: 'fxg4' },
57
+ { index: 12, depth: 0, type: OPERATOR, value: ',' },
58
+ { index: 13, depth: 0, type: RANGE, value: '[foo]Sheet1!B11', groupId: 'fxg1' },
59
+ { index: 14, depth: 0, type: OPERATOR, value: ',' },
60
+ { index: 15, depth: 0, type: RANGE, value: "'[foo]Sheet1'!B11", groupId: 'fxg1' }
61
+ ], { sheetName: 'Sheet1', workbookName: 'foo' });
62
+
63
+ t.isMetaTokens('=A:A,1:1,Sheet1!A:A:1:1,[foo]Sheet1!1:1', [
64
+ { index: 0, depth: 0, type: FX_PREFIX, value: '=' },
65
+ { index: 1, depth: 0, type: RANGE_BEAM, value: 'A:A', groupId: 'fxg1' },
66
+ { index: 2, depth: 0, type: OPERATOR, value: ',' },
67
+ { index: 3, depth: 0, type: RANGE_BEAM, value: '1:1', groupId: 'fxg2' },
68
+ { index: 4, depth: 0, type: OPERATOR, value: ',' },
69
+ { index: 5, depth: 0, type: RANGE_BEAM, value: 'Sheet1!A:A', groupId: 'fxg1' },
70
+ { index: 6, depth: 0, type: OPERATOR, value: ':' },
71
+ { index: 7, depth: 0, type: RANGE_BEAM, value: '1:1', groupId: 'fxg2' },
72
+ { index: 8, depth: 0, type: OPERATOR, value: ',' },
73
+ { index: 9, depth: 0, type: RANGE_BEAM, value: '[foo]Sheet1!1:1', groupId: 'fxg2' }
74
+ ], { sheetName: 'Sheet1', workbookName: 'foo' });
75
+
76
+ t.end();
77
+ });
@@ -13,9 +13,11 @@ export function addMeta (tokens, { sheetName = '', workbookName = '' } = {}) {
13
13
  const a1Map = {};
14
14
  const uid = getIDer();
15
15
 
16
- tokens.forEach(token => {
16
+ tokens.forEach((token, i) => {
17
+ token.index = i;
18
+ token.depth = parenStack.length;
17
19
  if (token.value === '(') {
18
- token.depth = parenStack.length;
20
+ token.depth = parenStack.length + 1;
19
21
  parenStack.push(token);
20
22
  }
21
23
  else if (token.value === ')') {
@@ -23,9 +23,9 @@ export const MAX_ROWS = 2 ** 20 - 1; // 1048575
23
23
  const re_ERROR = /^#(NAME\?|FIELD!|CALC!|VALUE!|REF!|DIV\/0!|NULL!|NUM!|N\/A|GETTING_DATA\b|SPILL!|UNKNOWN!|FIELD\b|CALC\b|SYNTAX\?|ERROR!)/i;
24
24
  const re_OPERATOR = /^(<=|>=|<>|[-+/*^%&<>=]|[{},;]|[()]|@|:|!|#)/;
25
25
  const re_BOOLEAN = /^(TRUE|FALSE)\b/i;
26
- const re_FUNCTION = /^[A-Z]+(?=\s*\()/i;
26
+ const re_FUNCTION = /^[A-Z_]+[A-Z\d_.]+(?=\s*\()/i;
27
27
  const re_NEWLINE = /^\n+/;
28
- const re_WHITESPACE = /^\s+/;
28
+ const re_WHITESPACE = /^[ \f\r\t\v\u00a0\u1680\u2000-\u200a\u2028\u2029\u202f\u205f\u3000\ufeff]+/;
29
29
  const re_STRING = /^"(?:""|[^"])*("|$)/;
30
30
  const re_PATH_QUOTE = /^'(?:''|[^'])*('|$)/;
31
31
  const re_PATH_BRACE = /^\[(?:[^\]])+(\]|$)/;
File without changes
@@ -5,6 +5,9 @@ import { tokenize } from './lexer.js';
5
5
  Test.prototype.isTokens = function isTokens (expr, result, opts) {
6
6
  this.deepEqual(tokenize(expr, (opts || {})), result, expr);
7
7
  };
8
+ Test.prototype.isTokensNeg = function isTokensNeg (expr, result, opts) {
9
+ this.deepEqual(tokenize(expr, { ...opts, negativeNumbers: true }), result, expr);
10
+ };
8
11
 
9
12
  test('tokenize operators', t => {
10
13
  t.isTokens('=1>1', [
@@ -269,6 +272,56 @@ test('tokenize functions', t => {
269
272
  { type: BOOLEAN, value: 'TRUE' },
270
273
  { type: OPERATOR, value: ')' }
271
274
  ]);
275
+ t.isTokens('=BINOM.DIST.RANGE(1)', [
276
+ { type: FX_PREFIX, value: '=' },
277
+ { type: FUNCTION, value: 'BINOM.DIST.RANGE' },
278
+ { type: OPERATOR, value: '(' },
279
+ { type: NUMBER, value: '1' },
280
+ { type: OPERATOR, value: ')' }
281
+ ]);
282
+ t.isTokens('=OCT2BIN(1)', [
283
+ { type: FX_PREFIX, value: '=' },
284
+ { type: FUNCTION, value: 'OCT2BIN' },
285
+ { type: OPERATOR, value: '(' },
286
+ { type: NUMBER, value: '1' },
287
+ { type: OPERATOR, value: ')' }
288
+ ]);
289
+ t.isTokens('=TEST_FUNC(1)', [
290
+ { type: FX_PREFIX, value: '=' },
291
+ { type: FUNCTION, value: 'TEST_FUNC' },
292
+ { type: OPERATOR, value: '(' },
293
+ { type: NUMBER, value: '1' },
294
+ { type: OPERATOR, value: ')' }
295
+ ]);
296
+ t.isTokens('=_xlfn.FOO(1)', [
297
+ { type: FX_PREFIX, value: '=' },
298
+ { type: FUNCTION, value: '_xlfn.FOO' },
299
+ { type: OPERATOR, value: '(' },
300
+ { type: NUMBER, value: '1' },
301
+ { type: OPERATOR, value: ')' }
302
+ ]);
303
+ t.isTokens('=_FOO(1)', [
304
+ { type: FX_PREFIX, value: '=' },
305
+ { type: FUNCTION, value: '_FOO' },
306
+ { type: OPERATOR, value: '(' },
307
+ { type: NUMBER, value: '1' },
308
+ { type: OPERATOR, value: ')' }
309
+ ]);
310
+ t.isTokens('=.FOO(1)', [
311
+ { type: FX_PREFIX, value: '=' },
312
+ { type: RANGE_NAMED, value: '.FOO' },
313
+ { type: OPERATOR, value: '(' },
314
+ { type: NUMBER, value: '1' },
315
+ { type: OPERATOR, value: ')' }
316
+ ]);
317
+ t.isTokens('=9FOO(1)', [
318
+ { type: FX_PREFIX, value: '=' },
319
+ { type: NUMBER, value: '9' },
320
+ { type: FUNCTION, value: 'FOO' },
321
+ { type: OPERATOR, value: '(' },
322
+ { type: NUMBER, value: '1' },
323
+ { type: OPERATOR, value: ')' }
324
+ ]);
272
325
  t.end();
273
326
  });
274
327
 
@@ -277,9 +330,9 @@ test('tokenize numbers', t => {
277
330
  { type: FX_PREFIX, value: '=' },
278
331
  { type: NUMBER, value: '0' }
279
332
  ]);
280
- t.isTokens('=-0', [
333
+ t.isTokens('=+0', [
281
334
  { type: FX_PREFIX, value: '=' },
282
- { type: OPERATOR, value: '-' },
335
+ { type: OPERATOR, value: '+' },
283
336
  { type: NUMBER, value: '0' }
284
337
  ]);
285
338
  t.isTokens('=+1', [
@@ -338,6 +391,231 @@ test('tokenize numbers', t => {
338
391
  t.end();
339
392
  });
340
393
 
394
+ test('tokenize negative numbers', t => {
395
+ t.isTokensNeg('=-0', [
396
+ { type: FX_PREFIX, value: '=' },
397
+ { type: NUMBER, value: '-0' }
398
+ ]);
399
+ t.isTokensNeg('=-1123', [
400
+ { type: FX_PREFIX, value: '=' },
401
+ { type: NUMBER, value: '-1123' }
402
+ ]);
403
+ t.isTokensNeg('=-1.5', [
404
+ { type: FX_PREFIX, value: '=' },
405
+ { type: NUMBER, value: '-1.5' }
406
+ ]);
407
+ t.isTokensNeg('=-1234.5678', [
408
+ { type: FX_PREFIX, value: '=' },
409
+ { type: NUMBER, value: '-1234.5678' }
410
+ ]);
411
+ t.isTokensNeg('=1E-1', [
412
+ { type: FX_PREFIX, value: '=' },
413
+ { type: NUMBER, value: '1E-1' }
414
+ ]);
415
+ t.isTokensNeg('=-1E-1', [
416
+ { type: FX_PREFIX, value: '=' },
417
+ { type: NUMBER, value: '-1E-1' }
418
+ ]);
419
+ t.isTokensNeg('=1.5E-10', [
420
+ { type: FX_PREFIX, value: '=' },
421
+ { type: NUMBER, value: '1.5E-10' }
422
+ ]);
423
+ t.isTokensNeg('=-1.5E-10', [
424
+ { type: FX_PREFIX, value: '=' },
425
+ { type: NUMBER, value: '-1.5E-10' }
426
+ ]);
427
+ t.isTokensNeg('-1', [
428
+ { type: 'number', value: '-1' }
429
+ ]);
430
+
431
+ //
432
+ t.isTokensNeg('=1-1', [
433
+ { type: FX_PREFIX, value: '=' },
434
+ { type: NUMBER, value: '1' },
435
+ { type: OPERATOR, value: '-' },
436
+ { type: NUMBER, value: '1' }
437
+ ]);
438
+ t.isTokensNeg('1--1', [
439
+ { type: 'number', value: '1' },
440
+ { type: 'operator', value: '-' },
441
+ { type: 'number', value: '-1' }
442
+ ]);
443
+ t.isTokensNeg('1 - -1', [
444
+ { type: 'number', value: '1' },
445
+ { type: 'whitespace', value: ' ' },
446
+ { type: 'operator', value: '-' },
447
+ { type: 'whitespace', value: ' ' },
448
+ { type: 'number', value: '-1' }
449
+ ]);
450
+ t.isTokensNeg('1 - - 1', [
451
+ { type: 'number', value: '1' },
452
+ { type: 'whitespace', value: ' ' },
453
+ { type: 'operator', value: '-' },
454
+ { type: 'whitespace', value: ' ' },
455
+ { type: 'operator', value: '-' },
456
+ { type: 'whitespace', value: ' ' },
457
+ { type: 'number', value: '1' }
458
+ ]);
459
+ t.isTokensNeg('1 \n - \n -1', [
460
+ { type: 'number', value: '1' },
461
+ { type: 'whitespace', value: ' ' },
462
+ { type: 'newline', value: '\n' },
463
+ { type: 'whitespace', value: ' ' },
464
+ { type: 'operator', value: '-' },
465
+ { type: 'whitespace', value: ' ' },
466
+ { type: 'newline', value: '\n' },
467
+ { type: 'whitespace', value: ' ' },
468
+ { type: 'number', value: '-1' }
469
+ ]);
470
+ t.isTokensNeg('-(-1)', [
471
+ { type: 'operator', value: '-' },
472
+ { type: 'operator', value: '(' },
473
+ { type: 'number', value: '-1' },
474
+ { type: 'operator', value: ')' }
475
+ ]);
476
+ t.isTokensNeg('-( -1 )', [
477
+ { type: 'operator', value: '-' },
478
+ { type: 'operator', value: '(' },
479
+ { type: 'whitespace', value: ' ' },
480
+ { type: 'number', value: '-1' },
481
+ { type: 'whitespace', value: ' ' },
482
+ { type: 'operator', value: ')' }
483
+ ]);
484
+
485
+ t.isTokensNeg('=true-1', [
486
+ { type: 'fx-prefix', value: '=' },
487
+ { type: 'bool', value: 'true' },
488
+ { type: 'operator', value: '-' },
489
+ { type: 'number', value: '1' }
490
+ ]);
491
+ t.isTokensNeg('=true -1', [
492
+ { type: 'fx-prefix', value: '=' },
493
+ { type: 'bool', value: 'true' },
494
+ { type: 'whitespace', value: ' ' },
495
+ { type: 'operator', value: '-' },
496
+ { type: 'number', value: '1' }
497
+ ]);
498
+ t.isTokensNeg('=true - 1', [
499
+ { type: 'fx-prefix', value: '=' },
500
+ { type: 'bool', value: 'true' },
501
+ { type: 'whitespace', value: ' ' },
502
+ { type: 'operator', value: '-' },
503
+ { type: 'whitespace', value: ' ' },
504
+ { type: 'number', value: '1' }
505
+ ]);
506
+ t.isTokensNeg('=#VALUE!-1', [
507
+ { type: 'fx-prefix', value: '=' },
508
+ { type: 'error', value: '#VALUE!' },
509
+ { type: 'operator', value: '-' },
510
+ { type: 'number', value: '1' }
511
+ ]);
512
+ t.isTokensNeg('=#VALUE! -1', [
513
+ { type: 'fx-prefix', value: '=' },
514
+ { type: 'error', value: '#VALUE!' },
515
+ { type: 'whitespace', value: ' ' },
516
+ { type: 'operator', value: '-' },
517
+ { type: 'number', value: '1' }
518
+ ]);
519
+ t.isTokensNeg('=SUM(-1) -1', [
520
+ { type: 'fx-prefix', value: '=' },
521
+ { type: 'function', value: 'SUM' },
522
+ { type: 'operator', value: '(' },
523
+ { type: 'number', value: '-1' },
524
+ { type: 'operator', value: ')' },
525
+ { type: 'whitespace', value: ' ' },
526
+ { type: 'operator', value: '-' },
527
+ { type: 'number', value: '1' }
528
+ ]);
529
+ t.isTokensNeg('=SUM( -1)-1', [
530
+ { type: 'fx-prefix', value: '=' },
531
+ { type: 'function', value: 'SUM' },
532
+ { type: 'operator', value: '(' },
533
+ { type: 'whitespace', value: ' ' },
534
+ { type: 'number', value: '-1' },
535
+ { type: 'operator', value: ')' },
536
+ { type: 'operator', value: '-' },
537
+ { type: 'number', value: '1' }
538
+ ]);
539
+ t.isTokensNeg('=A1-1', [
540
+ { type: 'fx-prefix', value: '=' },
541
+ { type: 'range', value: 'A1' },
542
+ { type: 'operator', value: '-' },
543
+ { type: 'number', value: '1' }
544
+ ]);
545
+ t.isTokensNeg('=A1 -1', [
546
+ { type: 'fx-prefix', value: '=' },
547
+ { type: 'range', value: 'A1' },
548
+ { type: 'whitespace', value: ' ' },
549
+ { type: 'operator', value: '-' },
550
+ { type: 'number', value: '1' }
551
+ ]);
552
+ t.isTokensNeg('=foo-1', [
553
+ { type: 'fx-prefix', value: '=' },
554
+ { type: 'range-named', value: 'foo' },
555
+ { type: 'operator', value: '-' },
556
+ { type: 'number', value: '1' }
557
+ ]);
558
+ t.isTokensNeg('=foo -1', [
559
+ { type: 'fx-prefix', value: '=' },
560
+ { type: 'range-named', value: 'foo' },
561
+ { type: 'whitespace', value: ' ' },
562
+ { type: 'operator', value: '-' },
563
+ { type: 'number', value: '1' }
564
+ ]);
565
+ t.isTokensNeg('="true"-1', [
566
+ { type: 'fx-prefix', value: '=' },
567
+ { type: 'string', value: '"true"' },
568
+ { type: 'operator', value: '-' },
569
+ { type: 'number', value: '1' }
570
+ ]);
571
+ t.isTokensNeg('="true" -1', [
572
+ { type: 'fx-prefix', value: '=' },
573
+ { type: 'string', value: '"true"' },
574
+ { type: 'whitespace', value: ' ' },
575
+ { type: 'operator', value: '-' },
576
+ { type: 'number', value: '1' }
577
+ ]);
578
+
579
+ t.isTokensNeg('=SUM(1)-1', [
580
+ { type: FX_PREFIX, value: '=' },
581
+ { type: FUNCTION, value: 'SUM' },
582
+ { type: OPERATOR, value: '(' },
583
+ { type: NUMBER, value: '1' },
584
+ { type: OPERATOR, value: ')' },
585
+ { type: OPERATOR, value: '-' },
586
+ { type: NUMBER, value: '1' }
587
+ ]);
588
+ t.isTokensNeg('={1, 2, 3}-4', [
589
+ { type: FX_PREFIX, value: '=' },
590
+ { type: OPERATOR, value: '{' },
591
+ { type: NUMBER, value: '1' },
592
+ { type: OPERATOR, value: ',' },
593
+ { type: WHITESPACE, value: ' ' },
594
+ { type: NUMBER, value: '2' },
595
+ { type: OPERATOR, value: ',' },
596
+ { type: WHITESPACE, value: ' ' },
597
+ { type: NUMBER, value: '3' },
598
+ { type: OPERATOR, value: '}' },
599
+ { type: OPERATOR, value: '-' },
600
+ { type: NUMBER, value: '4' }
601
+ ]);
602
+ t.isTokensNeg('=10%-1', [
603
+ { type: FX_PREFIX, value: '=' },
604
+ { type: NUMBER, value: '10' },
605
+ { type: OPERATOR, value: '%' },
606
+ { type: OPERATOR, value: '-' },
607
+ { type: NUMBER, value: '1' }
608
+ ]);
609
+ t.isTokensNeg('=A1#-1', [
610
+ { type: FX_PREFIX, value: '=' },
611
+ { type: RANGE, value: 'A1' },
612
+ { type: OPERATOR, value: '#' },
613
+ { type: OPERATOR, value: '-' },
614
+ { type: NUMBER, value: '1' }
615
+ ]);
616
+ t.end();
617
+ });
618
+
341
619
  test('tokenize simple equations', t => {
342
620
  t.isTokens('=1 + 2', [
343
621
  { type: FX_PREFIX, value: '=' },
@@ -700,6 +978,13 @@ test('tokenize A1 style references', t => {
700
978
  { type: RANGE_BEAM, value: 'AA:JJ' }
701
979
  ]);
702
980
 
981
+ t.isTokens('=XFD:XFF', [
982
+ { type: FX_PREFIX, value: '=' },
983
+ { type: RANGE_NAMED, value: 'XFD' },
984
+ { type: OPERATOR, value: ':' },
985
+ { type: RANGE_NAMED, value: 'XFF' }
986
+ ]);
987
+
703
988
  t.isTokens('=Sheetname!A1', [
704
989
  { type: FX_PREFIX, value: '=' },
705
990
  { type: RANGE, value: 'Sheetname!A1' }
@@ -852,12 +1137,11 @@ test('tokenize A1 style references', t => {
852
1137
  { type: RANGE, value: 'X4' }
853
1138
  ], { mergeRanges: false });
854
1139
 
855
- // largest possible A1 ref
1140
+ // largest possible A1 ref (in Excel)
856
1141
  t.isTokens('=XFD1048576', [
857
1142
  { type: FX_PREFIX, value: '=' },
858
1143
  { type: RANGE, value: 'XFD1048576' }
859
1144
  ]);
860
-
861
1145
  t.isTokens('=XFD1048577', [
862
1146
  { type: FX_PREFIX, value: '=' },
863
1147
  { type: RANGE_NAMED, value: 'XFD1048577' }
@@ -1055,7 +1339,7 @@ test('tokenize strings', t => {
1055
1339
  // so open ended strings at the end of output are tagged.
1056
1340
  t.isTokens('="incomple', [
1057
1341
  { type: FX_PREFIX, value: '=' },
1058
- { type: STRING, value: '"incomple' }
1342
+ { type: STRING, value: '"incomple', unterminated: true }
1059
1343
  ]);
1060
1344
 
1061
1345
  t.end();
@@ -1,12 +1,17 @@
1
1
  import {
2
2
  FX_PREFIX,
3
+ NEWLINE,
4
+ NUMBER,
5
+ OPERATOR,
3
6
  PATH_BRACE,
4
7
  PATH_PREFIX,
5
8
  PATH_QUOTE,
6
9
  RANGE,
7
- RANGE_NAMED,
8
10
  RANGE_BEAM,
11
+ RANGE_NAMED,
12
+ STRING,
9
13
  UNKNOWN,
14
+ WHITESPACE,
10
15
  tokenHandlersA1,
11
16
  tokenHandlersRC
12
17
  } from './constants.js';
@@ -15,10 +20,22 @@ const isType = (t, type) => t && t.type === type;
15
20
  const isRangeOp = t => t && t.value === ':';
16
21
  const isBangOp = t => t && t.value === '!';
17
22
 
18
- export function getTokens (fx, tokenHandlers = [], emitRanges = false, mergeRanges = false) {
23
+ const defaultOptions = {
24
+ emitRanges: false,
25
+ mergeRanges: true,
26
+ negativeNumbers: false,
27
+ r1c1: false
28
+ };
29
+
30
+ export function getTokens (fx, tokenHandlers, options = {}) {
31
+ const { emitRanges, mergeRanges, negativeNumbers } = Object.assign({}, defaultOptions, options);
19
32
  const tokens = [];
20
33
  let pos = 0;
21
34
  const lookBehind = n => tokens[tokens.length - n];
35
+ const lookBehindIgnoreWS = n => {
36
+ const noWs = tokens.filter(t => !isType(t, WHITESPACE) && !isType(t, NEWLINE));
37
+ return noWs[noWs.length - n];
38
+ };
22
39
 
23
40
  if (/^=/.test(fx)) {
24
41
  const token = {
@@ -58,6 +75,23 @@ export function getTokens (fx, tokenHandlers = [], emitRanges = false, mergeRang
58
75
  ...(emitRanges ? { range: [ startPos, pos ] } : {})
59
76
  };
60
77
 
78
+ if (tokenType === STRING && !tokenValue.endsWith('"')) {
79
+ token.unterminated = true;
80
+ }
81
+
82
+ if (negativeNumbers && tokenType === NUMBER) {
83
+ const last1 = lookBehind(1);
84
+ if (last1 && isType(last1, OPERATOR) && last1.value === '-') {
85
+ // we have a number preceded by a minus
86
+ const last2 = lookBehindIgnoreWS(2);
87
+ // missing last2 means we are at the start of the stream
88
+ if (!last2 || isType(last2, FX_PREFIX) || (isType(last2, OPERATOR) && ![ '%', '}', ')', '#' ].includes(last2.value))) {
89
+ tokens.pop();
90
+ token.value = '-' + tokenValue;
91
+ }
92
+ }
93
+ }
94
+
61
95
  if (mergeRanges) {
62
96
  if (tokenType === RANGE || tokenType === RANGE_NAMED || tokenType === RANGE_BEAM) {
63
97
  const merge = [];
@@ -94,7 +128,7 @@ export function getTokens (fx, tokenHandlers = [], emitRanges = false, mergeRang
94
128
 
95
129
  // Formulas can either have RC or A1 style refs, not both: because C1 and R1 are both!
96
130
  // Refmode: A1 | RC | Anostic emit ranges with RANGE
97
- export function tokenize (fx, { emitRanges = false, mergeRanges = true, r1c1 = false } = {}) {
98
- const tokenHandlers = r1c1 ? tokenHandlersRC : tokenHandlersA1;
99
- return getTokens(fx, tokenHandlers, emitRanges, mergeRanges);
131
+ export function tokenize (fx, options = {}) {
132
+ const tokenHandlers = options.r1c1 ? tokenHandlersRC : tokenHandlersA1;
133
+ return getTokens(fx, tokenHandlers, options);
100
134
  }
@@ -0,0 +1 @@
1
+ { "type": "module" }
@@ -51,7 +51,10 @@ const validRunsNamed = validRuns.concat([
51
51
  ]);
52
52
 
53
53
  export function parseRef (ref, allow_named = true, tokenHandlers = []) {
54
- const tokens = getTokens(ref, tokenHandlers, false, false);
54
+ const tokens = getTokens(ref, tokenHandlers, {
55
+ emitRanges: false,
56
+ mergeRanges: false
57
+ });
55
58
  const refData = {
56
59
  sheetName: '',
57
60
  workbookName: '',
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
package/package.json CHANGED
@@ -1,38 +1,53 @@
1
1
  {
2
2
  "name": "@borgar/fx",
3
- "version": "1.0.0",
3
+ "version": "2.0.0",
4
4
  "description": "Utilities for working with Excel formulas",
5
- "main": "src/index.js",
6
- "type": "module",
5
+ "main": "dist/fx.js",
6
+ "module": "lib/index.js",
7
+ "exports": {
8
+ ".": {
9
+ "require": "./dist/fx.js",
10
+ "default": "./lib/index.js"
11
+ }
12
+ },
7
13
  "scripts": {
8
- "lint": "eslint index.js src/*.js",
9
- "test": "tape run src/*-test.js | tap-min"
14
+ "lint": "eslint index.js lib/*.js",
15
+ "test": "tape lib/*-test.js | tap-min",
16
+ "build": "NODE_ENV=production rollup -c"
10
17
  },
11
18
  "repository": {
12
19
  "type": "git",
13
- "url": "git+https://github.com/borgar/xlsx-reader.git"
20
+ "url": "git+https://github.com/borgar/fx.git"
14
21
  },
15
22
  "bugs": {
16
- "url": "https://github.com/borgar/xlsx-reader/issues"
23
+ "url": "https://github.com/borgar/fx/issues"
17
24
  },
25
+ "browserslist": [
26
+ "defaults",
27
+ "not IE 11",
28
+ "maintained node versions"
29
+ ],
18
30
  "keywords": [
19
31
  "excel",
20
- "json",
21
32
  "xlsx",
22
- "csf",
23
- "office",
24
- "spreadsheet",
25
- "workbook"
33
+ "formula",
34
+ "spreadsheet"
26
35
  ],
27
36
  "author": "Borgar Þorsteinsson <borgar@borgar.net> (http://borgar.net/)",
28
37
  "license": "MIT",
29
38
  "devDependencies": {
30
- "@babel/core": "~7.15.5",
31
- "@babel/eslint-parser": "~7.15.4",
39
+ "@babel/core": "~7.18.13",
40
+ "@babel/eslint-parser": "~7.18.9",
41
+ "@babel/preset-env": "~7.18.10",
32
42
  "@borgar/eslint-config": "~2.2.1",
43
+ "@rollup/plugin-babel": "~5.3.1",
33
44
  "babel-eslint": "~10.1.0",
34
- "eslint": "~7.32.0",
45
+ "eslint": "~8.23.0",
46
+ "rollup": "~2.79.0",
35
47
  "tap-min": "~2.0.0",
36
- "tape": "~5.3.1"
48
+ "tape": "~5.6.0"
49
+ },
50
+ "dependencies": {
51
+ "rollup-plugin-terser": "~7.0.2"
37
52
  }
38
53
  }
@@ -0,0 +1,21 @@
1
+ import babel from '@rollup/plugin-babel';
2
+ import { terser } from 'rollup-plugin-terser';
3
+ import pkg from './package.json';
4
+
5
+ const isProd = process.env.NODE_ENV === 'production';
6
+
7
+ export default {
8
+ input: pkg.module,
9
+ plugins: [
10
+ babel({
11
+ babelHelpers: 'bundled',
12
+ presets: [ '@babel/preset-env' ]
13
+ })
14
+ ],
15
+ output: {
16
+ file: pkg.main,
17
+ format: 'cjs',
18
+ sourcemap: 'inline',
19
+ plugins: [ isProd && terser() ]
20
+ }
21
+ };
@@ -1,77 +0,0 @@
1
- import { test, Test } from 'tape';
2
- import { FX_PREFIX, OPERATOR, NUMBER, RANGE, RANGE_BEAM } from './constants.js';
3
- import { addMeta } from './addMeta.js';
4
- import { tokenize } from './lexer.js';
5
-
6
- Test.prototype.isMetaTokens = function isTokens (expr, result, opts) {
7
- this.deepEqual(addMeta(tokenize(expr), opts), result, expr);
8
- };
9
-
10
- test('add extra meta to operators', t => {
11
- // parens should be grouped and tagged with depth
12
- t.isMetaTokens('=((1)+(1))', [
13
- { type: FX_PREFIX, value: '=' },
14
- { type: OPERATOR, value: '(', depth: 0, groupId: 'fxg3' },
15
- { type: OPERATOR, value: '(', depth: 1, groupId: 'fxg1' },
16
- { type: NUMBER, value: '1' },
17
- { type: OPERATOR, value: ')', depth: 1, groupId: 'fxg1' },
18
- { type: OPERATOR, value: '+' },
19
- { type: OPERATOR, value: '(', depth: 1, groupId: 'fxg2' },
20
- { type: NUMBER, value: '1' },
21
- { type: OPERATOR, value: ')', depth: 1, groupId: 'fxg2' },
22
- { type: OPERATOR, value: ')', depth: 0, groupId: 'fxg3' }
23
- ]);
24
-
25
- // don't be fooled by imbalanced parens
26
- t.isMetaTokens('=)())', [
27
- { type: FX_PREFIX, value: '=' },
28
- { type: OPERATOR, value: ')', error: true },
29
- { type: OPERATOR, value: '(', depth: 0, groupId: 'fxg1' },
30
- { type: OPERATOR, value: ')', depth: 0, groupId: 'fxg1' },
31
- { type: OPERATOR, value: ')', error: true }
32
- ]);
33
-
34
- // don't be fooled by nested curlys
35
- t.isMetaTokens('={{}}', [
36
- { type: FX_PREFIX, value: '=' },
37
- { type: OPERATOR, value: '{', groupId: 'fxg1' },
38
- { type: OPERATOR, value: '{', error: true },
39
- { type: OPERATOR, value: '}', groupId: 'fxg1' },
40
- { type: OPERATOR, value: '}', error: true }
41
- ]);
42
-
43
- // group ranges if they are equivalent
44
- t.isMetaTokens("=B11,B11:B12,'Sheet11'!B11,SHEET1!$B11,sheet1!$b$11,A1:B11,[foo]Sheet1!B11,'[foo]Sheet1'!B11", [
45
- { type: FX_PREFIX, value: '=' },
46
- { type: RANGE, value: 'B11', groupId: 'fxg1' },
47
- { type: OPERATOR, value: ',' },
48
- { type: RANGE, value: 'B11:B12', groupId: 'fxg2' },
49
- { type: OPERATOR, value: ',' },
50
- { type: RANGE, value: "'Sheet11'!B11", groupId: 'fxg3' },
51
- { type: OPERATOR, value: ',' },
52
- { type: RANGE, value: 'SHEET1!$B11', groupId: 'fxg1' },
53
- { type: OPERATOR, value: ',' },
54
- { type: RANGE, value: 'sheet1!$b$11', groupId: 'fxg1' },
55
- { type: OPERATOR, value: ',' },
56
- { type: RANGE, value: 'A1:B11', groupId: 'fxg4' },
57
- { type: OPERATOR, value: ',' },
58
- { type: RANGE, value: '[foo]Sheet1!B11', groupId: 'fxg1' },
59
- { type: OPERATOR, value: ',' },
60
- { type: RANGE, value: "'[foo]Sheet1'!B11", groupId: 'fxg1' }
61
- ], { sheetName: 'Sheet1', workbookName: 'foo' });
62
-
63
- t.isMetaTokens('=A:A,1:1,Sheet1!A:A:1:1,[foo]Sheet1!1:1', [
64
- { type: FX_PREFIX, value: '=' },
65
- { type: RANGE_BEAM, value: 'A:A', groupId: 'fxg1' },
66
- { type: OPERATOR, value: ',' },
67
- { type: RANGE_BEAM, value: '1:1', groupId: 'fxg2' },
68
- { type: OPERATOR, value: ',' },
69
- { type: RANGE_BEAM, value: 'Sheet1!A:A', groupId: 'fxg1' },
70
- { type: OPERATOR, value: ':' },
71
- { type: RANGE_BEAM, value: '1:1', groupId: 'fxg2' },
72
- { type: OPERATOR, value: ',' },
73
- { type: RANGE_BEAM, value: '[foo]Sheet1!1:1', groupId: 'fxg2' }
74
- ], { sheetName: 'Sheet1', workbookName: 'foo' });
75
-
76
- t.end();
77
- });