@atomictech/xlsx-write-stream
Version:
Stream huge amount of data into an XLSX generated file stream with minimum memory footprint.
159 lines (140 loc) • 15.1 kB
JavaScript
"use strict";
Object.defineProperty(exports, "__esModule", {
value: true
});
exports.default = exports.defaultStyleDefinitions = exports.getBestNumberTypeStyleKey = exports.TypeStyleKey = void 0;
const generateNumFmt = styleDefinition => ` <numFmt numFmtId="${styleDefinition.numFmtId}" formatCode="${styleDefinition.formatCode}" />`;
const generateNumFmts = styleDefinitions => {
const customFormats = Object.entries(styleDefinitions).reduce((result, [key, value]) => {
if (value.formatCode) result[key] = value;
return result;
}, {});
return ` <numFmts count="${Object.keys(customFormats).length}">\n${Object.values(customFormats).map(generateNumFmt).join('\n')}\n </numFmts>`;
};
const generateXf = styleDefinition => ` <xf numFmtId="${styleDefinition.numFmtId}" fontId="0" fillId="0" borderId="0" xfId="0"/>`;
const generateCellXfs = styleDefinitions => ` <cellXfs count="${Object.keys(styleDefinitions).length}">\n${Object.values(styleDefinitions).map(generateXf).join('\n')}\n </cellXfs>`;
const TypeStyleKey = {
NUMBER: 'default',
INT: 'int',
FLOAT: 'float',
BIG_INT: 'bigInt',
BIG_FLOAT: 'bigFloat',
EXP_NUMBER: 'expNumber',
TEXT: 'text',
DATE: 'date',
DATETIME: 'datetime'
};
exports.TypeStyleKey = TypeStyleKey;
const getBestNumberTypeStyleKey = number => {
const positiveNumber = Math.abs(number);
if (positiveNumber.toString().length >= 11) {
return TypeStyleKey.EXP_NUMBER;
}
if (positiveNumber % 1 === 0) {
return positiveNumber >= 1000 ? TypeStyleKey.BIG_INT : TypeStyleKey.INT;
}
return positiveNumber >= 1000 ? TypeStyleKey.BIG_FLOAT : TypeStyleKey.FLOAT;
};
exports.getBestNumberTypeStyleKey = getBestNumberTypeStyleKey;
const defaultStyleDefinitions = {
default: {
numFmtId: 0
},
int: {
numFmtId: 1
},
float: {
numFmtId: 2
},
bigInt: {
numFmtId: 3
},
bigFloat: {
numFmtId: 4
},
expNumber: {
numFmtId: 11
},
text: {
numFmtId: 49
},
date: {
formatCode: 'yyyy-mm-dd'
},
datetime: {
formatCode: 'yyyy-mm-dd hh:mm:ss'
}
};
exports.defaultStyleDefinitions = defaultStyleDefinitions;
const minNumFmtId = 200;
const patchDefinitions = styleDefinitions => {
let styleIndex = 0;
let nextNumFmtId = minNumFmtId;
for (const [, styleDefinition] of Object.entries(styleDefinitions)) {
styleDefinition.index = styleIndex++;
if (!styleDefinition.numFmtId) {
if (!styleDefinition.formatCode) continue;
styleDefinition.numFmtId = nextNumFmtId++;
}
}
};
class Styles {
constructor(styleDefinitions = {}) {
this.typeStyleDefinitions = Object.assign({}, defaultStyleDefinitions, styleDefinitions);
patchDefinitions(this.typeStyleDefinitions);
}
getStyleId(typeStyle) {
return this.typeStyleDefinitions[typeStyle].index;
}
render() {
return `<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
<styleSheet xmlns="http://schemas.openxmlformats.org/spreadsheetml/2006/main"
xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006" mc:Ignorable="x14ac"
xmlns:x14ac="http://schemas.microsoft.com/office/spreadsheetml/2009/9/ac">
${generateNumFmts(this.typeStyleDefinitions)}
<fonts count="1" x14ac:knownFonts="1">
<font>
<sz val="11"/>
<color theme="1"/>
<name val="Calibri"/>
<family val="2"/>
<scheme val="minor"/>
</font>
</fonts>
<fills count="2">
<fill>
<patternFill patternType="none"/>
</fill>
<fill>
<patternFill patternType="gray125"/>
</fill>
</fills>
<borders count="1">
<border>
<left/>
<right/>
<top/>
<bottom/>
<diagonal/>
</border>
</borders>
<cellStyleXfs count="1">
<xf numFmtId="0" fontId="0" fillId="0" borderId="0"/>
</cellStyleXfs>
${generateCellXfs(this.typeStyleDefinitions)}
<cellStyles count="1">
<cellStyle name="Normal" xfId="0" builtinId="0"/>
</cellStyles>
<dxfs count="0"/>
<tableStyles count="0" defaultTableStyle="TableStyleMedium2" defaultPivotStyle="PivotStyleLight16"/>
<extLst>
<ext uri="{EB79DEF2-80B8-43e5-95BD-54CBDDF9020C}"
xmlns:x14="http://schemas.microsoft.com/office/spreadsheetml/2009/9/main">
<x14:slicerStyles defaultSlicerStyle="SlicerStyleLight1"/>
</ext>
</extLst>
</styleSheet>`;
}
}
exports.default = Styles;
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/templates/styles.js"],"names":["generateNumFmt","styleDefinition","numFmtId","formatCode","generateNumFmts","styleDefinitions","customFormats","Object","entries","reduce","result","key","value","keys","length","values","map","join","generateXf","generateCellXfs","TypeStyleKey","NUMBER","INT","FLOAT","BIG_INT","BIG_FLOAT","EXP_NUMBER","TEXT","DATE","DATETIME","getBestNumberTypeStyleKey","number","positiveNumber","Math","abs","toString","defaultStyleDefinitions","default","int","float","bigInt","bigFloat","expNumber","text","date","datetime","minNumFmtId","patchDefinitions","styleIndex","nextNumFmtId","index","Styles","constructor","typeStyleDefinitions","assign","getStyleId","typeStyle","render"],"mappings":"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","sourcesContent":["const generateNumFmt = styleDefinition => `    <numFmt numFmtId=\"${styleDefinition.numFmtId}\" formatCode=\"${styleDefinition.formatCode}\" />`;\nconst generateNumFmts = styleDefinitions => {\n  const customFormats = Object.entries(styleDefinitions).reduce((result, [key, value]) => {\n    if (value.formatCode) result[key] = value;\n    return result;\n  }, {});\n  return `  <numFmts count=\"${Object.keys(customFormats).length}\">\\n${Object.values(customFormats)\n    .map(generateNumFmt)\n    .join('\\n')}\\n  </numFmts>`;\n};\n\nconst generateXf = styleDefinition => `    <xf numFmtId=\"${styleDefinition.numFmtId}\" fontId=\"0\" fillId=\"0\" borderId=\"0\" xfId=\"0\"/>`;\nconst generateCellXfs = styleDefinitions =>\n  `  <cellXfs count=\"${Object.keys(styleDefinitions).length}\">\\n${Object.values(styleDefinitions)\n    .map(generateXf)\n    .join('\\n')}\\n  </cellXfs>`;\n\nexport const TypeStyleKey = {\n  NUMBER: 'default',\n  INT: 'int',\n  FLOAT: 'float',\n  BIG_INT: 'bigInt',\n  BIG_FLOAT: 'bigFloat',\n  EXP_NUMBER: 'expNumber',\n  TEXT: 'text',\n  DATE: 'date',\n  DATETIME: 'datetime'\n};\n\nexport const getBestNumberTypeStyleKey = number => {\n  const positiveNumber = Math.abs(number);\n  if (positiveNumber.toString().length >= 11) {\n    return TypeStyleKey.EXP_NUMBER;\n  }\n  if (positiveNumber % 1 === 0) {\n    return positiveNumber >= 1000 ? TypeStyleKey.BIG_INT : TypeStyleKey.INT;\n  }\n  return positiveNumber >= 1000 ? TypeStyleKey.BIG_FLOAT : TypeStyleKey.FLOAT;\n};\n\nexport const defaultStyleDefinitions = {\n  default: { numFmtId: 0 },\n  int: { numFmtId: 1 },\n  float: { numFmtId: 2 },\n  bigInt: { numFmtId: 3 },\n  bigFloat: { numFmtId: 4 },\n  expNumber: { numFmtId: 11 },\n  text: { numFmtId: 49 },\n  date: { formatCode: 'yyyy-mm-dd' },\n  datetime: { formatCode: 'yyyy-mm-dd hh:mm:ss' }\n};\n\nconst minNumFmtId = 200;\nconst patchDefinitions = styleDefinitions => {\n  let styleIndex = 0;\n  let nextNumFmtId = minNumFmtId;\n\n  for (const [, styleDefinition] of Object.entries(styleDefinitions)) {\n    styleDefinition.index = styleIndex++;\n\n    if (!styleDefinition.numFmtId) {\n      if (!styleDefinition.formatCode) continue;\n\n      styleDefinition.numFmtId = nextNumFmtId++;\n    }\n  }\n};\n\nexport default class Styles {\n  constructor(styleDefinitions = {}) {\n    this.typeStyleDefinitions = Object.assign({}, defaultStyleDefinitions, styleDefinitions);\n    patchDefinitions(this.typeStyleDefinitions);\n  }\n\n  getStyleId(typeStyle) {\n    return this.typeStyleDefinitions[typeStyle].index;\n  }\n\n  render() {\n    return `<?xml version=\"1.0\" encoding=\"UTF-8\" standalone=\"yes\"?>\n<styleSheet xmlns=\"http://schemas.openxmlformats.org/spreadsheetml/2006/main\"\n  xmlns:mc=\"http://schemas.openxmlformats.org/markup-compatibility/2006\" mc:Ignorable=\"x14ac\"\n  xmlns:x14ac=\"http://schemas.microsoft.com/office/spreadsheetml/2009/9/ac\">\n${generateNumFmts(this.typeStyleDefinitions)}\n  <fonts count=\"1\" x14ac:knownFonts=\"1\">\n    <font>\n      <sz val=\"11\"/>\n      <color theme=\"1\"/>\n      <name val=\"Calibri\"/>\n      <family val=\"2\"/>\n      <scheme val=\"minor\"/>\n    </font>\n  </fonts>\n  <fills count=\"2\">\n    <fill>\n      <patternFill patternType=\"none\"/>\n    </fill>\n    <fill>\n      <patternFill patternType=\"gray125\"/>\n    </fill>\n  </fills>\n  <borders count=\"1\">\n    <border>\n      <left/>\n      <right/>\n      <top/>\n      <bottom/>\n      <diagonal/>\n    </border>\n  </borders>\n  <cellStyleXfs count=\"1\">\n    <xf numFmtId=\"0\" fontId=\"0\" fillId=\"0\" borderId=\"0\"/>\n  </cellStyleXfs>\n${generateCellXfs(this.typeStyleDefinitions)}\n  <cellStyles count=\"1\">\n    <cellStyle name=\"Normal\" xfId=\"0\" builtinId=\"0\"/>\n  </cellStyles>\n  <dxfs count=\"0\"/>\n  <tableStyles count=\"0\" defaultTableStyle=\"TableStyleMedium2\" defaultPivotStyle=\"PivotStyleLight16\"/>\n  <extLst>\n    <ext uri=\"{EB79DEF2-80B8-43e5-95BD-54CBDDF9020C}\"\n      xmlns:x14=\"http://schemas.microsoft.com/office/spreadsheetml/2009/9/main\">\n      <x14:slicerStyles defaultSlicerStyle=\"SlicerStyleLight1\"/>\n    </ext>\n  </extLst>\n</styleSheet>`;\n  }\n}\n"]}