@pdfme/schemas 3.4.3 → 4.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.
Files changed (138) hide show
  1. package/dist/cjs/__tests__/text.test.js +5 -4
  2. package/dist/cjs/__tests__/text.test.js.map +1 -1
  3. package/dist/cjs/__tests__/utils.test.js +3 -0
  4. package/dist/cjs/__tests__/utils.test.js.map +1 -1
  5. package/dist/cjs/src/barcodes/propPanel.js +23 -11
  6. package/dist/cjs/src/barcodes/propPanel.js.map +1 -1
  7. package/dist/cjs/src/barcodes/uiRender.js +1 -1
  8. package/dist/cjs/src/barcodes/uiRender.js.map +1 -1
  9. package/dist/cjs/src/graphics/image.js +8 -11
  10. package/dist/cjs/src/graphics/image.js.map +1 -1
  11. package/dist/cjs/src/graphics/svg.js +4 -3
  12. package/dist/cjs/src/graphics/svg.js.map +1 -1
  13. package/dist/cjs/src/index.js +7 -1
  14. package/dist/cjs/src/index.js.map +1 -1
  15. package/dist/cjs/src/shapes/line.js +5 -8
  16. package/dist/cjs/src/shapes/line.js.map +1 -1
  17. package/dist/cjs/src/shapes/rectAndEllipse.js +12 -5
  18. package/dist/cjs/src/shapes/rectAndEllipse.js.map +1 -1
  19. package/dist/cjs/src/tables/cell.js +125 -0
  20. package/dist/cjs/src/tables/cell.js.map +1 -0
  21. package/dist/cjs/src/tables/classes.js +467 -0
  22. package/dist/cjs/src/tables/classes.js.map +1 -0
  23. package/dist/cjs/src/tables/dynamicTemplate.js +71 -0
  24. package/dist/cjs/src/tables/dynamicTemplate.js.map +1 -0
  25. package/dist/cjs/src/tables/helper.js +171 -0
  26. package/dist/cjs/src/tables/helper.js.map +1 -0
  27. package/dist/cjs/src/tables/index.js +12 -0
  28. package/dist/cjs/src/tables/index.js.map +1 -0
  29. package/dist/cjs/src/tables/pdfRender.js +93 -0
  30. package/dist/cjs/src/tables/pdfRender.js.map +1 -0
  31. package/dist/cjs/src/tables/propPanel.js +88 -0
  32. package/dist/cjs/src/tables/propPanel.js.map +1 -0
  33. package/dist/cjs/src/tables/tableHelper.js +231 -0
  34. package/dist/cjs/src/tables/tableHelper.js.map +1 -0
  35. package/dist/cjs/src/tables/types.js +3 -0
  36. package/dist/cjs/src/tables/types.js.map +1 -0
  37. package/dist/cjs/src/tables/uiRender.js +342 -0
  38. package/dist/cjs/src/tables/uiRender.js.map +1 -0
  39. package/dist/cjs/src/text/helper.js +21 -6
  40. package/dist/cjs/src/text/helper.js.map +1 -1
  41. package/dist/cjs/src/text/index.js +1 -1
  42. package/dist/cjs/src/text/index.js.map +1 -1
  43. package/dist/cjs/src/text/pdfRender.js +6 -9
  44. package/dist/cjs/src/text/pdfRender.js.map +1 -1
  45. package/dist/cjs/src/text/propPanel.js +7 -4
  46. package/dist/cjs/src/text/propPanel.js.map +1 -1
  47. package/dist/cjs/src/text/uiRender.js +18 -17
  48. package/dist/cjs/src/text/uiRender.js.map +1 -1
  49. package/dist/cjs/src/utils.js +3 -1
  50. package/dist/cjs/src/utils.js.map +1 -1
  51. package/dist/esm/__tests__/text.test.js +5 -4
  52. package/dist/esm/__tests__/text.test.js.map +1 -1
  53. package/dist/esm/__tests__/utils.test.js +3 -0
  54. package/dist/esm/__tests__/utils.test.js.map +1 -1
  55. package/dist/esm/src/barcodes/propPanel.js +23 -11
  56. package/dist/esm/src/barcodes/propPanel.js.map +1 -1
  57. package/dist/esm/src/barcodes/uiRender.js +1 -1
  58. package/dist/esm/src/barcodes/uiRender.js.map +1 -1
  59. package/dist/esm/src/graphics/image.js +6 -9
  60. package/dist/esm/src/graphics/image.js.map +1 -1
  61. package/dist/esm/src/graphics/svg.js +4 -3
  62. package/dist/esm/src/graphics/svg.js.map +1 -1
  63. package/dist/esm/src/index.js +4 -1
  64. package/dist/esm/src/index.js.map +1 -1
  65. package/dist/esm/src/shapes/line.js +5 -8
  66. package/dist/esm/src/shapes/line.js.map +1 -1
  67. package/dist/esm/src/shapes/rectAndEllipse.js +12 -5
  68. package/dist/esm/src/shapes/rectAndEllipse.js.map +1 -1
  69. package/dist/esm/src/tables/cell.js +120 -0
  70. package/dist/esm/src/tables/cell.js.map +1 -0
  71. package/dist/esm/src/tables/classes.js +460 -0
  72. package/dist/esm/src/tables/classes.js.map +1 -0
  73. package/dist/esm/src/tables/dynamicTemplate.js +66 -0
  74. package/dist/esm/src/tables/dynamicTemplate.js.map +1 -0
  75. package/dist/esm/src/tables/helper.js +163 -0
  76. package/dist/esm/src/tables/helper.js.map +1 -0
  77. package/dist/esm/src/tables/index.js +10 -0
  78. package/dist/esm/src/tables/index.js.map +1 -0
  79. package/dist/esm/src/tables/pdfRender.js +86 -0
  80. package/dist/esm/src/tables/pdfRender.js.map +1 -0
  81. package/dist/esm/src/tables/propPanel.js +85 -0
  82. package/dist/esm/src/tables/propPanel.js.map +1 -0
  83. package/dist/esm/src/tables/tableHelper.js +226 -0
  84. package/dist/esm/src/tables/tableHelper.js.map +1 -0
  85. package/dist/esm/src/tables/types.js +2 -0
  86. package/dist/esm/src/tables/types.js.map +1 -0
  87. package/dist/esm/src/tables/uiRender.js +335 -0
  88. package/dist/esm/src/tables/uiRender.js.map +1 -0
  89. package/dist/esm/src/text/helper.js +19 -5
  90. package/dist/esm/src/text/helper.js.map +1 -1
  91. package/dist/esm/src/text/index.js +1 -1
  92. package/dist/esm/src/text/index.js.map +1 -1
  93. package/dist/esm/src/text/pdfRender.js +7 -10
  94. package/dist/esm/src/text/pdfRender.js.map +1 -1
  95. package/dist/esm/src/text/propPanel.js +7 -4
  96. package/dist/esm/src/text/propPanel.js.map +1 -1
  97. package/dist/esm/src/text/uiRender.js +19 -18
  98. package/dist/esm/src/text/uiRender.js.map +1 -1
  99. package/dist/esm/src/utils.js +1 -0
  100. package/dist/esm/src/utils.js.map +1 -1
  101. package/dist/types/src/index.d.ts +3 -1
  102. package/dist/types/src/shapes/rectAndEllipse.d.ts +19 -27
  103. package/dist/types/src/tables/cell.d.ts +4 -0
  104. package/dist/types/src/tables/classes.d.ts +68 -0
  105. package/dist/types/src/tables/dynamicTemplate.d.ts +13 -0
  106. package/dist/types/src/tables/helper.d.ts +252 -0
  107. package/dist/types/src/tables/index.d.ts +4 -0
  108. package/dist/types/src/tables/pdfRender.d.ts +3 -0
  109. package/dist/types/src/tables/propPanel.d.ts +3 -0
  110. package/dist/types/src/tables/tableHelper.d.ts +11 -0
  111. package/dist/types/src/tables/types.d.ts +91 -0
  112. package/dist/types/src/tables/uiRender.d.ts +3 -0
  113. package/dist/types/src/text/helper.d.ts +8 -1
  114. package/dist/types/src/utils.d.ts +1 -0
  115. package/package.json +1 -1
  116. package/src/barcodes/propPanel.ts +26 -12
  117. package/src/barcodes/uiRender.ts +1 -1
  118. package/src/graphics/image.ts +8 -11
  119. package/src/graphics/svg.ts +4 -3
  120. package/src/index.ts +7 -0
  121. package/src/shapes/line.ts +4 -8
  122. package/src/shapes/rectAndEllipse.ts +16 -7
  123. package/src/tables/cell.ts +157 -0
  124. package/src/tables/classes.ts +398 -0
  125. package/src/tables/dynamicTemplate.ts +81 -0
  126. package/src/tables/helper.ts +198 -0
  127. package/src/tables/index.ts +12 -0
  128. package/src/tables/pdfRender.ts +113 -0
  129. package/src/tables/propPanel.ts +91 -0
  130. package/src/tables/tableHelper.ts +322 -0
  131. package/src/tables/types.ts +88 -0
  132. package/src/tables/uiRender.ts +393 -0
  133. package/src/text/helper.ts +29 -6
  134. package/src/text/index.ts +1 -1
  135. package/src/text/pdfRender.ts +9 -13
  136. package/src/text/propPanel.ts +7 -4
  137. package/src/text/uiRender.ts +18 -18
  138. package/src/utils.ts +1 -0
@@ -0,0 +1,113 @@
1
+ import type { TableSchema } from './types';
2
+ import type { PDFRenderProps } from '@pdfme/common';
3
+ import { Cell, Table, Row, Column } from './classes';
4
+ import { rectangle } from '../shapes/rectAndEllipse';
5
+ import cell from './cell';
6
+ import { getBodyWithRange } from './helper';
7
+ import { createSingleTable } from './tableHelper';
8
+
9
+ type Pos = { x: number; y: number };
10
+
11
+ const rectanglePdfRender = rectangle.pdf;
12
+ const cellPdfRender = cell.pdf;
13
+
14
+ async function drawCell(arg: PDFRenderProps<TableSchema>, cell: Cell) {
15
+ await cellPdfRender({
16
+ ...arg,
17
+ value: cell.raw,
18
+ schema: {
19
+ type: 'cell',
20
+ position: { x: cell.x, y: cell.y },
21
+ width: cell.width,
22
+ height: cell.height,
23
+ fontName: cell.styles.fontName,
24
+ alignment: cell.styles.alignment,
25
+ verticalAlignment: cell.styles.verticalAlignment,
26
+ fontSize: cell.styles.fontSize,
27
+ lineHeight: cell.styles.lineHeight,
28
+ characterSpacing: cell.styles.characterSpacing,
29
+ backgroundColor: cell.styles.backgroundColor,
30
+ fontColor: cell.styles.textColor,
31
+ borderColor: cell.styles.lineColor,
32
+ borderWidth: cell.styles.lineWidth,
33
+ padding: cell.styles.cellPadding,
34
+ },
35
+ });
36
+ }
37
+
38
+ async function drawRow(
39
+ arg: PDFRenderProps<TableSchema>,
40
+ table: Table,
41
+ row: Row,
42
+ cursor: Pos,
43
+ columns: Column[]
44
+ ) {
45
+ cursor.x = table.settings.margin.left;
46
+ for (const column of columns) {
47
+ const cell = row.cells[column.index];
48
+ if (!cell) {
49
+ cursor.x += column.width;
50
+ continue;
51
+ }
52
+
53
+ cell.x = cursor.x;
54
+ cell.y = cursor.y;
55
+
56
+ await drawCell(arg, cell);
57
+
58
+ cursor.x += column.width;
59
+ }
60
+ cursor.y += row.height;
61
+ }
62
+
63
+ async function drawTableBorder(
64
+ arg: PDFRenderProps<TableSchema>,
65
+ table: Table,
66
+ startPos: Pos,
67
+ cursor: Pos
68
+ ) {
69
+ const lineWidth = table.settings.tableLineWidth;
70
+ const lineColor = table.settings.tableLineColor;
71
+ if (!lineWidth || !lineColor) return;
72
+ await rectanglePdfRender({
73
+ ...arg,
74
+ schema: {
75
+ type: 'rectangle',
76
+ borderWidth: lineWidth,
77
+ borderColor: lineColor,
78
+ color: '',
79
+ position: { x: startPos.x, y: startPos.y },
80
+ width: table.getWidth(),
81
+ height: cursor.y - startPos.y,
82
+ readOnly: true,
83
+ },
84
+ });
85
+ }
86
+
87
+ async function drawTable(arg: PDFRenderProps<TableSchema>, table: Table): Promise<void> {
88
+ const settings = table.settings;
89
+ const startY = settings.startY;
90
+ const margin = settings.margin;
91
+ const cursor = { x: margin.left, y: startY };
92
+
93
+ const startPos = Object.assign({}, cursor);
94
+
95
+ if (settings.showHead) {
96
+ for (const row of table.head) {
97
+ await drawRow(arg, table, row, cursor, table.columns);
98
+ }
99
+ }
100
+
101
+ for (const row of table.body) {
102
+ await drawRow(arg, table, row, cursor, table.columns);
103
+ }
104
+
105
+ await drawTableBorder(arg, table, startPos, cursor);
106
+ }
107
+
108
+ export const pdfRender = async (arg: PDFRenderProps<TableSchema>) => {
109
+ const { value, schema } = arg;
110
+ const body = getBodyWithRange(value, schema.__bodyRange);
111
+ const table = await createSingleTable(body, arg);
112
+ await drawTable(arg, table);
113
+ };
@@ -0,0 +1,91 @@
1
+ import type { PropPanel } from '@pdfme/common';
2
+ import type { TableSchema } from './types';
3
+ import { getFallbackFontName, DEFAULT_FONT_NAME } from '@pdfme/common';
4
+ import {
5
+ getDefaultCellStyles,
6
+ getCellPropPanelSchema,
7
+ getColumnStylesPropPanelSchema,
8
+ } from './helper.js';
9
+ import { HEX_COLOR_PATTERN } from '../constants.js';
10
+
11
+ export const propPanel: PropPanel<TableSchema> = {
12
+ schema: ({ activeSchema, options, i18n }) => {
13
+ // @ts-ignore
14
+ const head = (activeSchema.head || []) as string[];
15
+ const font = options.font || { [DEFAULT_FONT_NAME]: { data: '', fallback: true } };
16
+ const fontNames = Object.keys(font);
17
+ const fallbackFontName = getFallbackFontName(font);
18
+ return {
19
+ tableStyles: {
20
+ title: i18n('schemas.table.tableStyle'),
21
+ type: 'object',
22
+ widget: 'Card',
23
+ span: 24,
24
+ properties: {
25
+ borderWidth: {
26
+ title: i18n('schemas.borderWidth'),
27
+ type: 'number',
28
+ widget: 'inputNumber',
29
+ props: { min: 0, step: 0.1 },
30
+ step: 1,
31
+ },
32
+ borderColor: {
33
+ title: i18n('schemas.borderColor'),
34
+ type: 'string',
35
+ widget: 'color',
36
+ rules: [{ pattern: HEX_COLOR_PATTERN, message: i18n('hexColorPrompt') }],
37
+ },
38
+ },
39
+ },
40
+ headStyles: {
41
+ title: i18n('schemas.table.headStyle'),
42
+ type: 'object',
43
+ widget: 'Card',
44
+ span: 24,
45
+ properties: getCellPropPanelSchema({ i18n, fallbackFontName, fontNames }),
46
+ },
47
+ bodyStyles: {
48
+ title: i18n('schemas.table.bodyStyle'),
49
+ type: 'object',
50
+ widget: 'Card',
51
+ span: 24,
52
+ properties: getCellPropPanelSchema({ i18n, fallbackFontName, fontNames, isBody: true }),
53
+ },
54
+ columnStyles: {
55
+ title: i18n('schemas.table.columnStyle'),
56
+ type: 'object',
57
+ widget: 'Card',
58
+ span: 24,
59
+ properties: getColumnStylesPropPanelSchema({ head, i18n }),
60
+ },
61
+ };
62
+ },
63
+ defaultSchema: {
64
+ type: 'table',
65
+ icon:'<svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="lucide lucide-table"><path d="M12 3v18"/><rect width="18" height="18" x="3" y="3" rx="2"/><path d="M3 9h18"/><path d="M3 15h18"/></svg>',
66
+ position: { x: 0, y: 0 },
67
+ width: 150,
68
+ height: 20,
69
+ content: JSON.stringify([
70
+ ['Alice', 'New York', 'Alice is a freelance web designer and developer'],
71
+ ['Bob', 'Paris', 'Bob is a freelance illustrator and graphic designer'],
72
+ ]),
73
+ showHead: true,
74
+ head: ['Name', 'City', 'Description'],
75
+ headWidthPercentages: [30, 30, 40],
76
+ tableStyles: {
77
+ borderColor: '#000000',
78
+ borderWidth: 0.3,
79
+ },
80
+ headStyles: Object.assign(getDefaultCellStyles(), {
81
+ fontColor: '#ffffff',
82
+ backgroundColor: '#2980ba',
83
+ borderColor: '',
84
+ borderWidth: { top: 0, right: 0, bottom: 0, left: 0 },
85
+ }),
86
+ bodyStyles: Object.assign(getDefaultCellStyles(), {
87
+ alternateBackgroundColor: '#f5f5f5',
88
+ }),
89
+ columnStyles: {},
90
+ },
91
+ };
@@ -0,0 +1,322 @@
1
+ import {
2
+ Schema,
3
+ isBlankPdf,
4
+ BasePdf,
5
+ CommonOptions,
6
+ getDefaultFont,
7
+ getFallbackFontName,
8
+ } from '@pdfme/common';
9
+ import type {
10
+ TableSchema,
11
+ CellStyle,
12
+ Styles,
13
+ Spacing,
14
+ TableInput,
15
+ StylesProps,
16
+ Section,
17
+ } from './types';
18
+ import { cloneDeep } from '../utils';
19
+ import { Cell, Column, Row, Table } from './classes';
20
+
21
+ type StyleProp = 'styles' | 'headStyles' | 'bodyStyles' | 'alternateRowStyles' | 'columnStyles';
22
+
23
+ interface CreateTableArgs {
24
+ schema: Schema;
25
+ basePdf: BasePdf;
26
+ options: CommonOptions;
27
+ _cache: Map<any, any>;
28
+ }
29
+
30
+ interface UserOptions {
31
+ startY: number;
32
+ tableWidth: number;
33
+ margin: Spacing;
34
+ showHead: boolean;
35
+ tableLineWidth?: number;
36
+ tableLineColor?: string;
37
+ head?: string[][];
38
+ body?: string[][];
39
+
40
+ styles?: Partial<Styles>;
41
+ bodyStyles?: Partial<Styles>;
42
+ headStyles?: Partial<Styles>;
43
+ alternateRowStyles?: Partial<Styles>;
44
+ columnStyles?: {
45
+ [key: string]: Partial<Styles>;
46
+ };
47
+ }
48
+
49
+ function parseSection(
50
+ sectionName: Section,
51
+ sectionRows: string[][],
52
+ columns: Column[],
53
+ styleProps: StylesProps,
54
+ fallbackFontName: string
55
+ ): Row[] {
56
+ const rowSpansLeftForColumn: { [key: string]: { left: number; times: number } } = {};
57
+ const result = sectionRows.map((rawRow, rowIndex) => {
58
+ let skippedRowForRowSpans = 0;
59
+ const cells: { [key: string]: Cell } = {};
60
+
61
+ let colSpansAdded = 0;
62
+ let columnSpansLeft = 0;
63
+ for (const column of columns) {
64
+ if (
65
+ rowSpansLeftForColumn[column.index] == null ||
66
+ rowSpansLeftForColumn[column.index].left === 0
67
+ ) {
68
+ if (columnSpansLeft === 0) {
69
+ let rawCell;
70
+ if (Array.isArray(rawRow)) {
71
+ rawCell = rawRow[column.index - colSpansAdded - skippedRowForRowSpans];
72
+ } else {
73
+ rawCell = rawRow[column.index];
74
+ }
75
+ const styles = cellStyles(sectionName, column, rowIndex, styleProps, fallbackFontName);
76
+ const cell = new Cell(rawCell, styles, sectionName);
77
+ cells[column.index] = cell;
78
+
79
+ columnSpansLeft = 0;
80
+ rowSpansLeftForColumn[column.index] = {
81
+ left: 0,
82
+ times: columnSpansLeft,
83
+ };
84
+ } else {
85
+ columnSpansLeft--;
86
+ colSpansAdded++;
87
+ }
88
+ } else {
89
+ rowSpansLeftForColumn[column.index].left--;
90
+ columnSpansLeft = rowSpansLeftForColumn[column.index].times;
91
+ skippedRowForRowSpans++;
92
+ }
93
+ }
94
+ return new Row(rawRow, rowIndex, sectionName, cells);
95
+ });
96
+ return result;
97
+ }
98
+
99
+ function parseContent4Table(input: TableInput, fallbackFontName: string) {
100
+ const content = input.content;
101
+ const columns = content.columns.map((index) => new Column(index));
102
+ const styles = input.styles;
103
+ return {
104
+ columns,
105
+ head: parseSection('head', content.head, columns, styles, fallbackFontName),
106
+ body: parseSection('body', content.body, columns, styles, fallbackFontName),
107
+ };
108
+ }
109
+
110
+ function cellStyles(
111
+ sectionName: Section,
112
+ column: Column,
113
+ rowIndex: number,
114
+ styles: StylesProps,
115
+ fallbackFontName: string
116
+ ) {
117
+ let sectionStyles;
118
+ if (sectionName === 'head') {
119
+ sectionStyles = styles.headStyles;
120
+ } else if (sectionName === 'body') {
121
+ sectionStyles = styles.bodyStyles;
122
+ }
123
+ const otherStyles = Object.assign({}, styles.styles, sectionStyles);
124
+
125
+ const colStyles = styles.columnStyles[column.index] || styles.columnStyles[column.index] || {};
126
+
127
+ const rowStyles =
128
+ sectionName === 'body' && rowIndex % 2 === 0
129
+ ? Object.assign({}, styles.alternateRowStyles)
130
+ : {};
131
+
132
+ const defaultStyle = {
133
+ fontName: fallbackFontName,
134
+ backgroundColor: '',
135
+ textColor: '#000000',
136
+ lineHeight: 1,
137
+ characterSpacing: 0,
138
+ alignment: 'left',
139
+ verticalAlignment: 'middle',
140
+ fontSize: 10,
141
+ cellPadding: 5,
142
+ lineColor: '#000000',
143
+ lineWidth: 0,
144
+ minCellHeight: 0,
145
+ minCellWidth: 0,
146
+ };
147
+ return Object.assign(defaultStyle, otherStyles, rowStyles, colStyles) as Styles;
148
+ }
149
+
150
+ function mapCellStyle(style: CellStyle): Partial<Styles> {
151
+ return {
152
+ fontName: style.fontName,
153
+ alignment: style.alignment,
154
+ verticalAlignment: style.verticalAlignment,
155
+ fontSize: style.fontSize,
156
+ lineHeight: style.lineHeight,
157
+ characterSpacing: style.characterSpacing,
158
+ backgroundColor: style.backgroundColor,
159
+ // ---
160
+ textColor: style.fontColor,
161
+ lineColor: style.borderColor,
162
+ lineWidth: style.borderWidth,
163
+ cellPadding: style.padding,
164
+ };
165
+ }
166
+
167
+ function createTableWithAvailableHeight(
168
+ tableBody: Row[],
169
+ availableHeight: number,
170
+ args: CreateTableArgs
171
+ ) {
172
+ let limit = availableHeight;
173
+ const newTableBody: string[][] = [];
174
+ let index = 0;
175
+ while (limit > 0 && index < tableBody.length) {
176
+ const row = tableBody.slice(0, index + 1).pop();
177
+ if (!row) break;
178
+ const rowHeight = row.height;
179
+ if (limit - rowHeight < 0) {
180
+ break;
181
+ }
182
+ newTableBody.push(row.raw);
183
+ limit -= rowHeight;
184
+ index++;
185
+ }
186
+ return createSingleTable(newTableBody, args);
187
+ }
188
+
189
+ function getTableOptions(schema: TableSchema, body: string[][]): UserOptions {
190
+ const columnStylesWidth = schema.headWidthPercentages.reduce(
191
+ (acc, cur, i) => ({ ...acc, [i]: { cellWidth: schema.width * (cur / 100) } }),
192
+ {} as Record<number, Partial<Styles>>
193
+ );
194
+
195
+ const columnStylesAlignment = Object.entries(schema.columnStyles.alignment || {}).reduce(
196
+ (acc, [key, value]) => ({ ...acc, [key]: { alignment: value } }),
197
+ {} as Record<number, Partial<Styles>>
198
+ );
199
+
200
+ const allKeys = new Set([
201
+ ...Object.keys(columnStylesWidth).map(Number),
202
+ ...Object.keys(columnStylesAlignment).map(Number),
203
+ ]);
204
+ const columnStyles = Array.from(allKeys).reduce((acc, key) => {
205
+ const widthStyle = columnStylesWidth[key] || {};
206
+ const alignmentStyle = columnStylesAlignment[key] || {};
207
+ return { ...acc, [key]: { ...widthStyle, ...alignmentStyle } };
208
+ }, {} as Record<number, Partial<Styles>>);
209
+
210
+ return {
211
+ head: [schema.head],
212
+ body,
213
+ showHead: schema.showHead,
214
+ startY: schema.position.y,
215
+ tableWidth: schema.width,
216
+ tableLineColor: schema.tableStyles.borderColor,
217
+ tableLineWidth: schema.tableStyles.borderWidth,
218
+ headStyles: mapCellStyle(schema.headStyles),
219
+ bodyStyles: mapCellStyle(schema.bodyStyles),
220
+ alternateRowStyles: { backgroundColor: schema.bodyStyles.alternateBackgroundColor },
221
+ columnStyles,
222
+ margin: { top: 0, right: 0, left: schema.position.x, bottom: 0 },
223
+ };
224
+ }
225
+
226
+ function parseStyles(cInput: UserOptions) {
227
+ const styleOptions: StylesProps = {
228
+ styles: {},
229
+ headStyles: {},
230
+ bodyStyles: {},
231
+ alternateRowStyles: {},
232
+ columnStyles: {},
233
+ };
234
+ for (const prop of Object.keys(styleOptions) as StyleProp[]) {
235
+ if (prop === 'columnStyles') {
236
+ const current = cInput[prop];
237
+ styleOptions.columnStyles = Object.assign({}, current);
238
+ } else {
239
+ const allOptions = [cInput];
240
+ const styles = allOptions.map((opts) => opts[prop] || {});
241
+ styleOptions[prop] = Object.assign({}, styles[0], styles[1], styles[2]);
242
+ }
243
+ }
244
+ return styleOptions;
245
+ }
246
+
247
+ function parseContent4Input(options: UserOptions) {
248
+ const head = options.head || [];
249
+ const body = options.body || [];
250
+ const columns = (head[0] || body[0] || []).map((_, index) => index);
251
+ return { columns, head, body };
252
+ }
253
+
254
+ function parseInput(schema: TableSchema, body: string[][]): TableInput {
255
+ const options = getTableOptions(schema, body);
256
+ const styles = parseStyles(options);
257
+ const settings = {
258
+ startY: options.startY,
259
+ margin: options.margin,
260
+ tableWidth: options.tableWidth,
261
+ showHead: options.showHead,
262
+ tableLineWidth: options.tableLineWidth ?? 0,
263
+ tableLineColor: options.tableLineColor ?? '',
264
+ };
265
+
266
+ const content = parseContent4Input(options);
267
+
268
+ return { content, styles, settings };
269
+ }
270
+
271
+ export function createSingleTable(body: string[][], args: CreateTableArgs) {
272
+ const { options, _cache, basePdf } = args;
273
+ if (!isBlankPdf(basePdf)) throw new Error('[@pdfme/schema/table] Blank PDF is not supported');
274
+
275
+ const input = parseInput(args.schema as TableSchema, body);
276
+
277
+ const font = options.font || getDefaultFont();
278
+
279
+ const fallbackFontName = getFallbackFontName(font);
280
+
281
+ const content = parseContent4Table(input, fallbackFontName);
282
+
283
+ return Table.create({ input, content, font, _cache });
284
+ }
285
+
286
+ export async function createMultiTables(body: string[][], args: CreateTableArgs): Promise<Table[]> {
287
+ const { basePdf, schema } = args;
288
+
289
+ if (!isBlankPdf(basePdf)) throw new Error('[@pdfme/schema/table] Blank PDF is not supported');
290
+ const pageHeight = basePdf.height;
291
+ const paddingBottom = basePdf.padding[2];
292
+ const paddingTop = basePdf.padding[0];
293
+ let availableHeight = pageHeight - paddingBottom - schema.position.y;
294
+
295
+ const testTable = await createSingleTable(body, args);
296
+ let remainingBody = testTable.body;
297
+ const tables: Table[] = [];
298
+
299
+ while (remainingBody.length > 0) {
300
+ const tableHeight =
301
+ tables.length === 0
302
+ ? availableHeight - testTable.getHeadHeight()
303
+ : availableHeight - paddingTop;
304
+
305
+ const table = await createTableWithAvailableHeight(remainingBody, tableHeight, args);
306
+
307
+ tables.push(table);
308
+
309
+ remainingBody = remainingBody.slice(table.body.length);
310
+
311
+ if (remainingBody.length > 0) {
312
+ const _schema = cloneDeep(schema);
313
+ _schema.showHead = false;
314
+ _schema.position.y = paddingTop;
315
+ args.schema = _schema;
316
+
317
+ availableHeight = pageHeight - paddingTop - paddingBottom;
318
+ }
319
+ }
320
+
321
+ return tables;
322
+ }
@@ -0,0 +1,88 @@
1
+ import type { ALIGNMENT, VERTICAL_ALIGNMENT } from '../text/types';
2
+ import type { Schema } from '@pdfme/common';
3
+
4
+ export type Spacing = { top: number; right: number; bottom: number; left: number };
5
+ type BorderInsets = Spacing;
6
+ type BoxDimensions = Spacing;
7
+
8
+ export interface CellStyle {
9
+ fontName?: string;
10
+ alignment: ALIGNMENT;
11
+ verticalAlignment: VERTICAL_ALIGNMENT;
12
+ fontSize: number;
13
+ lineHeight: number;
14
+ characterSpacing: number;
15
+ fontColor: string;
16
+ backgroundColor: string;
17
+ borderColor: string;
18
+ borderWidth: BoxDimensions;
19
+ padding: BoxDimensions;
20
+ }
21
+
22
+ export type CellSchema = Schema & CellStyle;
23
+
24
+ export interface TableSchema extends Schema {
25
+ showHead: boolean;
26
+ head: string[];
27
+ headWidthPercentages: number[];
28
+
29
+ __bodyRange?: { start: number; end?: number };
30
+
31
+ tableStyles: {
32
+ borderColor: string;
33
+ borderWidth: number;
34
+ };
35
+ headStyles: CellStyle;
36
+ bodyStyles: CellStyle & { alternateBackgroundColor: string };
37
+ columnStyles: {
38
+ alignment?: { [colIndex: number]: ALIGNMENT };
39
+ };
40
+ }
41
+
42
+ export interface Styles {
43
+ fontName: string | undefined;
44
+ backgroundColor: string;
45
+ textColor: string;
46
+ lineHeight: number;
47
+ characterSpacing: number;
48
+ alignment: 'left' | 'center' | 'right';
49
+ verticalAlignment: 'top' | 'middle' | 'bottom';
50
+ fontSize: number;
51
+ cellPadding: Spacing;
52
+ lineColor: string;
53
+ lineWidth: BorderInsets;
54
+ cellWidth: number;
55
+ minCellHeight: number;
56
+ minCellWidth: number;
57
+ }
58
+
59
+ export interface TableInput {
60
+ settings: Settings;
61
+ styles: StylesProps;
62
+ content: ContentInput;
63
+ }
64
+
65
+ interface ContentInput {
66
+ body: string[][];
67
+ head: string[][];
68
+ columns: number[];
69
+ }
70
+
71
+ export interface Settings {
72
+ startY: number;
73
+ margin: Spacing;
74
+ tableWidth: number;
75
+ showHead: boolean;
76
+ tableLineWidth: number;
77
+ tableLineColor: string;
78
+ }
79
+
80
+ export interface StylesProps {
81
+ styles: Partial<Styles>;
82
+ headStyles: Partial<Styles>;
83
+ bodyStyles: Partial<Styles>;
84
+ alternateRowStyles: Partial<Styles>;
85
+ columnStyles: { [key: string]: Partial<Styles> };
86
+ }
87
+
88
+ export type Section = 'head' | 'body';