// http://officeopenxml.com/WPtableGrid.php import { XmlComponent } from "file/xml-components"; import { AlignmentType } from "../paragraph"; import { TableGrid } from "./grid"; import { TableCell, VerticalMergeType, WidthType } from "./table-cell"; import { ITableBordersOptions, ITableFloatOptions, TableProperties } from "./table-properties"; import { TableLayoutType } from "./table-properties/table-layout"; import { TableRow } from "./table-row"; /* 0-width columns don't get rendered correctly, so we need to give them some value. A reasonable default would be ~6in / numCols, but if we do that it becomes very hard to resize the table using setWidth, unless the layout algorithm is set to 'fixed'. Instead, the approach here means even in 'auto' layout, setting a width on the table will make it look reasonable, as the layout algorithm will expand columns to fit its content */ export interface ITableOptions { readonly rows: TableRow[]; readonly width?: { readonly size: number; readonly type?: WidthType; }; readonly columnWidths?: number[]; readonly margins?: { readonly marginUnitType?: WidthType; readonly top?: number; readonly bottom?: number; readonly right?: number; readonly left?: number; }; readonly float?: ITableFloatOptions; readonly layout?: TableLayoutType; readonly borders?: ITableBordersOptions; readonly alignment?: AlignmentType; } export class Table extends XmlComponent { private readonly properties: TableProperties; constructor({ rows, width, columnWidths = Array(Math.max(...rows.map((row) => row.CellCount))).fill(100), margins: { marginUnitType, top, bottom, right, left } = { marginUnitType: WidthType.AUTO, top: 0, bottom: 0, right: 0, left: 0 }, float, layout, borders, alignment, }: ITableOptions) { super("w:tbl"); this.properties = new TableProperties(); this.root.push(this.properties); if (borders) { this.properties.setBorder(borders); } else { this.properties.setBorder({}); } if (width) { this.properties.setWidth(width.size, width.type); } else { this.properties.setWidth(100); } this.properties.CellMargin.addBottomMargin(bottom || 0, marginUnitType); this.properties.CellMargin.addTopMargin(top || 0, marginUnitType); this.properties.CellMargin.addLeftMargin(left || 0, marginUnitType); this.properties.CellMargin.addRightMargin(right || 0, marginUnitType); this.root.push(new TableGrid(columnWidths)); for (const row of rows) { this.root.push(row); } rows.forEach((row, rowIndex) => { row.cells.forEach((cell, cellIndex) => { // Row Span has to be added in this method and not the constructor because it needs to know information about the column which happens after Table Cell construction // Row Span of 1 will crash word as it will add RESTART and not a corresponding CONTINUE if (cell.options.rowSpan && cell.options.rowSpan > 1) { const columnIndex = row.rootIndexToColumnIndex(cellIndex + 1); const startRowIndex = rowIndex + 1; const endRowIndex = rowIndex + (cell.options.rowSpan - 1); for (let i = startRowIndex; i <= endRowIndex; i++) { rows[i].addCellToColumnIndex( new TableCell({ columnSpan: cell.options.columnSpan, borders: cell.options.borders, children: [], verticalMerge: VerticalMergeType.CONTINUE, }), columnIndex, ); } } }); }); if (float) { this.properties.setTableFloatProperties(float); } if (layout) { this.properties.setLayout(layout); } if (alignment) { this.properties.setAlignment(alignment); } } }