diff --git a/.gitignore b/.gitignore
index 543ecab..5b048cd 100644
--- a/.gitignore
+++ b/.gitignore
@@ -3,8 +3,4 @@
/.idea
/coverage/
/.nyc_output/
-/.idea/
-
-
-/src/example.ts
-*.xlsx
\ No newline at end of file
+/.idea/
\ No newline at end of file
diff --git a/README.md b/README.md
index d263d30..c45a579 100644
--- a/README.md
+++ b/README.md
@@ -3,6 +3,12 @@
# WIP current version of lib is under developing
+# Introduction
+
+This library makes generating xlsx files (Excel) easly.
+
+It consumes template which is common Excel file, then add yours data (called ViewModel). Blend it and done, as result you'll get pretty Excel.
+
# Getting Started:
1. install package
@@ -11,50 +17,58 @@
npm i xlsx-renderer --save
```
-`npm install`
+2. TODO add some example here (https://github.com/Siemienik/xlsx-renderer/issues/6)
-## In Cell Commands:
-
-1. `## varToDisplayInThisCell`
-2. `#! FINISH` - finish processing current worksheet
-2. `#! FINISH booleanVar` - if true it finishes processing current template worksheet, if false it add again this worksheet to output
-3. `#! END_ROW`
-4. `#! DELETE varName`
-5. `#! HYPERLINK labelVar urlVar`
-6. `#! WS_NAME nameVar` set worksheet name
-7. `#! FOR_EACH item collection` (to write item property `## item.property`),
-8. `#! CONTINUE item` item is set to the next collection item.
-9. `#! END_LOOP item`
-10. `#! AVERAGE item` write average formula of all items from previous for-each, it has to be placed after the for-each was finished.
-11. `#! SUM item` similar to average
-12. `#! DUMP_COLS arrayVar` write to next columns all array items (1 item = 1 column)
-
-
-## Sample code: (possible to be outdated yet - todo)
+## Sample code:
```javascript
import Renderer from './xls-renderer/Renderer'
-import {Workbook} from 'exceljs'
-
-//*
-import DebugCellTemplatePool from "./xls-renderer-debug/CellTemplateDebugPool";
-const renderer = new Renderer(new DebugCellTemplatePool());
-/*/
-import CellTemplatePool from "./xls-renderer/CellTemplatePool";
-const renderer = new Renderer(new CellTemplatePool());
-//*/
-
const viewModel = new MyAwesomeReportVm(); //or something else
(async () => {
- const result = await renderer.render(async () => {
- const template = new Workbook();
- return await template.xlsx.readFile('./my-awesome-raport-template.xlsx');
- }, viewModel);
+ const result = await renderer.renderFromFile('./my-awesome-raport-template.xlsx', viewModel);
await result.xlsx.writeFile('./my-awesome-raport.xlsx');
})();
```
+
+# Documentation:
+
+## Cells:
+
+| Category | Name | Order | match rule | Description | More info |
+|----------|-----:|-------|--------|-------------|:---------|
+| - | [BaseCell](./src/cell/BaseCell.ts) | n/o | n/o | All Cell\`s definition classes extend it. | **abstract** |
+| Content | [NormalCell](./src/cell/NormalCell.ts) | 1 | not started by `##` or `#!` | This one copy all styles, width, properties and value form template. | **default** |
+| Content | [VariableCell](./src/cell/VariableCell.ts) | 3 | `## pathToVariable ` | Write variable from `ViewModel`.
Paths to object's property or array item are allowed. | **Examples:**
`simplePath`
`someObject.property`
`array.0.field`
`items.1.path.to.object.prop`|
+| Content | **TODO: describe it!** [HyperlinkCell](./src/cell/HyperlinkCell.ts) | | | | |
+| Content | **TODO: describe it!** [FormulaCell](./src/cell/FormulaCell.ts) | | | | |
+| Navigation | **TODO: describe it!** [EndRowCell](./src/cell/EndRowCell.ts) | | | | |
+| Worksheet
Navigation
Loop | [FinishCell](./src/cell/FinishCell.ts) | 7 | `#! FINISH conditionPath` | Finish rendering for current worksheet and:
1) go to next worksheet if `conditionPath===true`
2) repeat this template worksheet again (`conditionPath === false`) - looping through worksheets
3) finished whole rendering when this worksheet is the last one. | **Examples:**
`#! FINISHED ` or `#! FINISHED itemFromLoop.__iterated` |
+| Worksheet | **TODO: describe it!** [WsNameCell](./src/cell/WsNameCell.ts) | | | | |
+| View Model | **TODO: describe it!** [DeleteCell](./src/cell/DeleteCell.ts) | | | | |
+| Loop | **TODO: describe it!** [DumpColsCell](./src/cell/DumpColsCell.ts) | | | | |
+| Loop | **TODO: describe it!** [ForEachCell](./src/cell/ForEachCell.ts) | | | | |
+| Loop | **TODO: describe it!** [ContinueCell](./src/cell/ContinueCell.ts) | | | | |
+| Loop | **TODO: describe it!** [EndLoopCell](./src/cell/EndLoopCell.ts) | | | | |
+| Aggregation | **TODO: describe it!** [AverageCell](./src/cell/AverageCell.ts) | | | | |
+| Aggregation| **TODO: describe it!** [SumCell](./src/cell/SumCell.ts) | | | | |
+
+
+## Commands [PREVIOUS VERSION]:
+
+1. `#! END_ROW`
+4. `#! DELETE varName`
+5. `#! HYPERLINK labelVar urlVar`
+6. `#! WS_NAME nameVar` set worksheet name
+7. `#! FOR_EACH item collection` (to write item property `## item.property`),
+8. `#! CONTINUE item` item is set to the next collection item.
+9. `#! END_LOOP item`
+10. `#! AVERAGE item` write average formula of all items from previous for-each, it has to be placed after the for-each was finished.
+11. `#! SUM item` similar to average
+12. `#! DUMP_COLS arrayVar` write to next columns all array items (1 item = 1 column)
+
+
[LICENSE](LICENSE)
diff --git a/package.json b/package.json
index e8d66c3..12ab1e5 100644
--- a/package.json
+++ b/package.json
@@ -33,6 +33,7 @@
"postversion": "git push && git push --tags"
},
"devDependencies": {
+ "@types/node": "^12.12.6",
"@types/chai": "^4.1.7",
"@types/mocha": "^5.2.6",
"chai": "^4.2.0",
diff --git a/src/Renderer.ts b/src/Renderer.ts
index fa36b30..203a56f 100644
--- a/src/Renderer.ts
+++ b/src/Renderer.ts
@@ -19,4 +19,13 @@ export class Renderer {
return output;
}
+
+ public async renderFromFile(templatePath: string, viewModel: any): Promise {
+ const result = await this.render(async () => {
+ const template = new Workbook();
+ return await template.xlsx.readFile(templatePath);
+ }, viewModel);
+
+ return await result;
+ }
}
diff --git a/src/cell/BaseCell.ts b/src/cell/BaseCell.ts
index 93d1c9b..082c023 100644
--- a/src/cell/BaseCell.ts
+++ b/src/cell/BaseCell.ts
@@ -4,7 +4,7 @@ import { Scope } from '../Scope';
export declare type CellType = typeof BaseCell;
-export class BaseCell {
+export /* abstract */ class BaseCell { // can't be abstract :(
/**
* check if this commend can parse `value`
*/
@@ -12,7 +12,7 @@ export class BaseCell {
return false;
}
- public BaseCell() {
+ public BaseCell() { // can't be marked by abstract keyword, so it throw type error.
throw new TypeError(`Cannot construct ${BaseCell.name} instances directly. It's abstract.`);
}
diff --git a/src/example.ts b/src/example.ts
deleted file mode 100644
index 64e9431..0000000
--- a/src/example.ts
+++ /dev/null
@@ -1,27 +0,0 @@
-import { Renderer } from './Renderer';
-import { CellTemplateDebugPool } from './CellTemplateDebugPool';
-import { CellTemplatePool } from './CellTemplatePool';
-import { Workbook } from 'exceljs';
-
-// tslint:disable:comment-format
-//*
-const debug = true;
-/*/
-const debug = false;
-//*/
-
-let renderer;
-if (debug) {
- renderer = new Renderer(new CellTemplateDebugPool());
-} else {
- renderer = new Renderer(new CellTemplatePool());
-}
-
-(async () => {
- const result = await renderer.render(async () => {
- const template = new Workbook();
- return await template.xlsx.readFile('./my-awesome-raport-template.xlsx');
- }, {});
-
- await result.xlsx.writeFile('./my-awesome-raport.xlsx');
-})();
diff --git a/tests/integration/Renderer.ts b/tests/integration/Renderer.ts
new file mode 100644
index 0000000..c47a12c
--- /dev/null
+++ b/tests/integration/Renderer.ts
@@ -0,0 +1,92 @@
+import {Renderer} from "../../src/Renderer";
+import * as fs from "fs";
+import * as path from "path";
+import {Workbook} from "exceljs";
+import * as chai from 'chai'
+
+
+function assertCells(expected: Workbook, result: Workbook, factor: number = 10) {
+ chai.expect(expected.worksheets.length).eql(result.worksheets.length);
+ chai.expect(expected.worksheets.map(x => x.name)).eql(result.worksheets.map(x => x.name));
+
+ for (let wi = 0; wi < expected.worksheets.length; wi++) {
+ const ws = {e: expected.worksheets[wi], r: result.worksheets[wi]};
+ for (let i = 0; i < factor * factor; i++) {
+ const r = Math.floor(i / factor) + 1;
+ const c = i % factor + 1;
+ const cell = {
+ e: ws.e.getCell(r, c),
+ r: ws.r.getCell(r, c)
+ };
+
+
+ if (r === 1) {
+ chai.expect(ws.e.getColumn(c).width).eql(ws.r.getColumn(c).width);
+ }
+ if (c === 1) {
+ chai.expect(ws.e.getRow(r).height).eql(ws.r.getRow(r).height);
+ }
+
+ chai.expect(cell.e.style).eql(cell.r.style);
+ chai.expect(cell.e.text).eql(cell.r.text);
+ chai.expect(cell.e.value).eql(cell.r.value);
+ }
+ }
+}
+
+describe('INTEGRATION:: Test xlsx renderer ', function () {
+
+ describe('Checking if assertCells works ok.', function () {
+ it('Same - should pass ok', async function () {
+ const expected = await new Workbook().xlsx.readFile(path.join(__dirname, 'data', 'assertCells', 'main.xlsx'));
+ const correct = await new Workbook().xlsx.readFile(path.join(__dirname, 'data', 'assertCells', 'correct.xlsx'));
+
+ assertCells(expected, correct, 20);
+ });
+
+ it('Different - attempt to broke assertions', async function () {
+ const expected = await new Workbook().xlsx.readFile(path.join(__dirname, 'data', 'assertCells', 'main.xlsx'));
+ const failedWorksheetAmount = await new Workbook().xlsx.readFile(path.join(__dirname, 'data', 'assertCells', 'f-ws-amount.xlsx'));
+ const failedWorksheetNames = await new Workbook().xlsx.readFile(path.join(__dirname, 'data', 'assertCells', 'f-ws-names.xlsx'));
+ const failedWidth = await new Workbook().xlsx.readFile(path.join(__dirname, 'data', 'assertCells', 'f-width.xlsx'));
+ const failedHeight = await new Workbook().xlsx.readFile(path.join(__dirname, 'data', 'assertCells', 'f-height.xlsx'));
+ const failedStyle = await new Workbook().xlsx.readFile(path.join(__dirname, 'data', 'assertCells', 'f-style.xlsx'));
+ const failedText = await new Workbook().xlsx.readFile(path.join(__dirname, 'data', 'assertCells', 'f-text.xlsx'));
+ const failedValue = await new Workbook().xlsx.readFile(path.join(__dirname, 'data', 'assertCells', 'f-value.xlsx'));
+ const failedTable = await new Workbook().xlsx.readFile(path.join(__dirname, 'data', 'assertCells', 'f-table.xlsx'));
+
+ chai.expect(() => assertCells(expected, failedWorksheetAmount, 20)).throw("expected 2 to deeply equal 3");
+ chai.expect(() => assertCells(expected, failedWorksheetNames, 20)).throw('expected [ \'Sheet1\', \'Sheet2\' ] to deeply equal [ \'Sheet1\', \'Sheet3\' ]');
+ chai.expect(() => assertCells(expected, failedWidth, 20)).throw("expected 13 to deeply equal 7.90625");
+ chai.expect(() => assertCells(expected, failedHeight, 20)).throw("expected 15 to deeply equal 34.5");
+ chai.expect(() => assertCells(expected, failedStyle, 20)).throw("expected { Object (font, border, ...) } to deeply equal { Object (font, border, ...) }");
+ chai.expect(() => assertCells(expected, failedText, 20)).throw('expected \'sadasd\' to deeply equal \'sadas\'');
+ chai.expect(() => assertCells(expected, failedValue, 20)).throw('expected { Object (formula, result) } to deeply equal \'asdasda\'');
+ chai.expect(() => assertCells(expected, failedTable, 20)).throw('expected { Object (font, border, ...) } to deeply equal { Object (font, border, ...) }');
+ });
+ });
+ describe('Load examples, render and compare with expected result', function () {
+ const dataPath = path.normalize(path.join(__dirname, 'data/'));
+ const sets = fs.readdirSync(path.normalize(dataPath), {withFileTypes: true})
+ .filter(i => i.isDirectory())
+ .filter(d => /^Renderer[0-9]*-/.test(d.name));
+
+
+ const renderer = new Renderer();
+ sets.forEach(s => {
+ it(`Test for ${s.name}`, async function () {
+ const result = await renderer.renderFromFile(
+ path.join(dataPath, s.name, "template.xlsx"),
+ require(path.join(dataPath, s.name, 'viewModel.json'))
+ );
+
+ const expected = await new Workbook().xlsx.readFile(path.join(dataPath, s.name, "expected.xlsx"));
+
+ assertCells(expected, result);
+ });
+
+
+ })
+
+ });
+});
\ No newline at end of file
diff --git a/tests/integration/data/Renderer000-FinishCell/expected.xlsx b/tests/integration/data/Renderer000-FinishCell/expected.xlsx
new file mode 100644
index 0000000..b4eb627
Binary files /dev/null and b/tests/integration/data/Renderer000-FinishCell/expected.xlsx differ
diff --git a/tests/integration/data/Renderer000-FinishCell/template.xlsx b/tests/integration/data/Renderer000-FinishCell/template.xlsx
new file mode 100644
index 0000000..decb5f3
Binary files /dev/null and b/tests/integration/data/Renderer000-FinishCell/template.xlsx differ
diff --git a/tests/integration/data/Renderer000-FinishCell/viewModel.json b/tests/integration/data/Renderer000-FinishCell/viewModel.json
new file mode 100644
index 0000000..9e26dfe
--- /dev/null
+++ b/tests/integration/data/Renderer000-FinishCell/viewModel.json
@@ -0,0 +1 @@
+{}
\ No newline at end of file
diff --git a/tests/integration/data/assertCells/correct.xlsx b/tests/integration/data/assertCells/correct.xlsx
new file mode 100644
index 0000000..aaf5eff
Binary files /dev/null and b/tests/integration/data/assertCells/correct.xlsx differ
diff --git a/tests/integration/data/assertCells/f-height.xlsx b/tests/integration/data/assertCells/f-height.xlsx
new file mode 100644
index 0000000..25e9730
Binary files /dev/null and b/tests/integration/data/assertCells/f-height.xlsx differ
diff --git a/tests/integration/data/assertCells/f-style.xlsx b/tests/integration/data/assertCells/f-style.xlsx
new file mode 100644
index 0000000..87a096e
Binary files /dev/null and b/tests/integration/data/assertCells/f-style.xlsx differ
diff --git a/tests/integration/data/assertCells/f-table.xlsx b/tests/integration/data/assertCells/f-table.xlsx
new file mode 100644
index 0000000..2796514
Binary files /dev/null and b/tests/integration/data/assertCells/f-table.xlsx differ
diff --git a/tests/integration/data/assertCells/f-text.xlsx b/tests/integration/data/assertCells/f-text.xlsx
new file mode 100644
index 0000000..c57cb10
Binary files /dev/null and b/tests/integration/data/assertCells/f-text.xlsx differ
diff --git a/tests/integration/data/assertCells/f-value.xlsx b/tests/integration/data/assertCells/f-value.xlsx
new file mode 100644
index 0000000..60bf56e
Binary files /dev/null and b/tests/integration/data/assertCells/f-value.xlsx differ
diff --git a/tests/integration/data/assertCells/f-width.xlsx b/tests/integration/data/assertCells/f-width.xlsx
new file mode 100644
index 0000000..0d7bf6e
Binary files /dev/null and b/tests/integration/data/assertCells/f-width.xlsx differ
diff --git a/tests/integration/data/assertCells/f-ws-amount.xlsx b/tests/integration/data/assertCells/f-ws-amount.xlsx
new file mode 100644
index 0000000..a9b218c
Binary files /dev/null and b/tests/integration/data/assertCells/f-ws-amount.xlsx differ
diff --git a/tests/integration/data/assertCells/f-ws-names.xlsx b/tests/integration/data/assertCells/f-ws-names.xlsx
new file mode 100644
index 0000000..9d41240
Binary files /dev/null and b/tests/integration/data/assertCells/f-ws-names.xlsx differ
diff --git a/tests/integration/data/assertCells/main.xlsx b/tests/integration/data/assertCells/main.xlsx
new file mode 100644
index 0000000..d4ce6eb
Binary files /dev/null and b/tests/integration/data/assertCells/main.xlsx differ