@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.
- package/dist/cjs/__tests__/text.test.js +5 -4
- package/dist/cjs/__tests__/text.test.js.map +1 -1
- package/dist/cjs/__tests__/utils.test.js +3 -0
- package/dist/cjs/__tests__/utils.test.js.map +1 -1
- package/dist/cjs/src/barcodes/propPanel.js +23 -11
- package/dist/cjs/src/barcodes/propPanel.js.map +1 -1
- package/dist/cjs/src/barcodes/uiRender.js +1 -1
- package/dist/cjs/src/barcodes/uiRender.js.map +1 -1
- package/dist/cjs/src/graphics/image.js +8 -11
- package/dist/cjs/src/graphics/image.js.map +1 -1
- package/dist/cjs/src/graphics/svg.js +4 -3
- package/dist/cjs/src/graphics/svg.js.map +1 -1
- package/dist/cjs/src/index.js +7 -1
- package/dist/cjs/src/index.js.map +1 -1
- package/dist/cjs/src/shapes/line.js +5 -8
- package/dist/cjs/src/shapes/line.js.map +1 -1
- package/dist/cjs/src/shapes/rectAndEllipse.js +12 -5
- package/dist/cjs/src/shapes/rectAndEllipse.js.map +1 -1
- package/dist/cjs/src/tables/cell.js +125 -0
- package/dist/cjs/src/tables/cell.js.map +1 -0
- package/dist/cjs/src/tables/classes.js +467 -0
- package/dist/cjs/src/tables/classes.js.map +1 -0
- package/dist/cjs/src/tables/dynamicTemplate.js +71 -0
- package/dist/cjs/src/tables/dynamicTemplate.js.map +1 -0
- package/dist/cjs/src/tables/helper.js +171 -0
- package/dist/cjs/src/tables/helper.js.map +1 -0
- package/dist/cjs/src/tables/index.js +12 -0
- package/dist/cjs/src/tables/index.js.map +1 -0
- package/dist/cjs/src/tables/pdfRender.js +93 -0
- package/dist/cjs/src/tables/pdfRender.js.map +1 -0
- package/dist/cjs/src/tables/propPanel.js +88 -0
- package/dist/cjs/src/tables/propPanel.js.map +1 -0
- package/dist/cjs/src/tables/tableHelper.js +231 -0
- package/dist/cjs/src/tables/tableHelper.js.map +1 -0
- package/dist/cjs/src/tables/types.js +3 -0
- package/dist/cjs/src/tables/types.js.map +1 -0
- package/dist/cjs/src/tables/uiRender.js +342 -0
- package/dist/cjs/src/tables/uiRender.js.map +1 -0
- package/dist/cjs/src/text/helper.js +21 -6
- package/dist/cjs/src/text/helper.js.map +1 -1
- package/dist/cjs/src/text/index.js +1 -1
- package/dist/cjs/src/text/index.js.map +1 -1
- package/dist/cjs/src/text/pdfRender.js +6 -9
- package/dist/cjs/src/text/pdfRender.js.map +1 -1
- package/dist/cjs/src/text/propPanel.js +7 -4
- package/dist/cjs/src/text/propPanel.js.map +1 -1
- package/dist/cjs/src/text/uiRender.js +18 -17
- package/dist/cjs/src/text/uiRender.js.map +1 -1
- package/dist/cjs/src/utils.js +3 -1
- package/dist/cjs/src/utils.js.map +1 -1
- package/dist/esm/__tests__/text.test.js +5 -4
- package/dist/esm/__tests__/text.test.js.map +1 -1
- package/dist/esm/__tests__/utils.test.js +3 -0
- package/dist/esm/__tests__/utils.test.js.map +1 -1
- package/dist/esm/src/barcodes/propPanel.js +23 -11
- package/dist/esm/src/barcodes/propPanel.js.map +1 -1
- package/dist/esm/src/barcodes/uiRender.js +1 -1
- package/dist/esm/src/barcodes/uiRender.js.map +1 -1
- package/dist/esm/src/graphics/image.js +6 -9
- package/dist/esm/src/graphics/image.js.map +1 -1
- package/dist/esm/src/graphics/svg.js +4 -3
- package/dist/esm/src/graphics/svg.js.map +1 -1
- package/dist/esm/src/index.js +4 -1
- package/dist/esm/src/index.js.map +1 -1
- package/dist/esm/src/shapes/line.js +5 -8
- package/dist/esm/src/shapes/line.js.map +1 -1
- package/dist/esm/src/shapes/rectAndEllipse.js +12 -5
- package/dist/esm/src/shapes/rectAndEllipse.js.map +1 -1
- package/dist/esm/src/tables/cell.js +120 -0
- package/dist/esm/src/tables/cell.js.map +1 -0
- package/dist/esm/src/tables/classes.js +460 -0
- package/dist/esm/src/tables/classes.js.map +1 -0
- package/dist/esm/src/tables/dynamicTemplate.js +66 -0
- package/dist/esm/src/tables/dynamicTemplate.js.map +1 -0
- package/dist/esm/src/tables/helper.js +163 -0
- package/dist/esm/src/tables/helper.js.map +1 -0
- package/dist/esm/src/tables/index.js +10 -0
- package/dist/esm/src/tables/index.js.map +1 -0
- package/dist/esm/src/tables/pdfRender.js +86 -0
- package/dist/esm/src/tables/pdfRender.js.map +1 -0
- package/dist/esm/src/tables/propPanel.js +85 -0
- package/dist/esm/src/tables/propPanel.js.map +1 -0
- package/dist/esm/src/tables/tableHelper.js +226 -0
- package/dist/esm/src/tables/tableHelper.js.map +1 -0
- package/dist/esm/src/tables/types.js +2 -0
- package/dist/esm/src/tables/types.js.map +1 -0
- package/dist/esm/src/tables/uiRender.js +335 -0
- package/dist/esm/src/tables/uiRender.js.map +1 -0
- package/dist/esm/src/text/helper.js +19 -5
- package/dist/esm/src/text/helper.js.map +1 -1
- package/dist/esm/src/text/index.js +1 -1
- package/dist/esm/src/text/index.js.map +1 -1
- package/dist/esm/src/text/pdfRender.js +7 -10
- package/dist/esm/src/text/pdfRender.js.map +1 -1
- package/dist/esm/src/text/propPanel.js +7 -4
- package/dist/esm/src/text/propPanel.js.map +1 -1
- package/dist/esm/src/text/uiRender.js +19 -18
- package/dist/esm/src/text/uiRender.js.map +1 -1
- package/dist/esm/src/utils.js +1 -0
- package/dist/esm/src/utils.js.map +1 -1
- package/dist/types/src/index.d.ts +3 -1
- package/dist/types/src/shapes/rectAndEllipse.d.ts +19 -27
- package/dist/types/src/tables/cell.d.ts +4 -0
- package/dist/types/src/tables/classes.d.ts +68 -0
- package/dist/types/src/tables/dynamicTemplate.d.ts +13 -0
- package/dist/types/src/tables/helper.d.ts +252 -0
- package/dist/types/src/tables/index.d.ts +4 -0
- package/dist/types/src/tables/pdfRender.d.ts +3 -0
- package/dist/types/src/tables/propPanel.d.ts +3 -0
- package/dist/types/src/tables/tableHelper.d.ts +11 -0
- package/dist/types/src/tables/types.d.ts +91 -0
- package/dist/types/src/tables/uiRender.d.ts +3 -0
- package/dist/types/src/text/helper.d.ts +8 -1
- package/dist/types/src/utils.d.ts +1 -0
- package/package.json +1 -1
- package/src/barcodes/propPanel.ts +26 -12
- package/src/barcodes/uiRender.ts +1 -1
- package/src/graphics/image.ts +8 -11
- package/src/graphics/svg.ts +4 -3
- package/src/index.ts +7 -0
- package/src/shapes/line.ts +4 -8
- package/src/shapes/rectAndEllipse.ts +16 -7
- package/src/tables/cell.ts +157 -0
- package/src/tables/classes.ts +398 -0
- package/src/tables/dynamicTemplate.ts +81 -0
- package/src/tables/helper.ts +198 -0
- package/src/tables/index.ts +12 -0
- package/src/tables/pdfRender.ts +113 -0
- package/src/tables/propPanel.ts +91 -0
- package/src/tables/tableHelper.ts +322 -0
- package/src/tables/types.ts +88 -0
- package/src/tables/uiRender.ts +393 -0
- package/src/text/helper.ts +29 -6
- package/src/text/index.ts +1 -1
- package/src/text/pdfRender.ts +9 -13
- package/src/text/propPanel.ts +7 -4
- package/src/text/uiRender.ts +18 -18
- package/src/utils.ts +1 -0
@@ -0,0 +1,398 @@
|
|
1
|
+
import { Font, mm2pt, pt2mm } from '@pdfme/common';
|
2
|
+
import type { Font as FontKitFont } from 'fontkit';
|
3
|
+
import { splitTextToSize, getFontKitFont, widthOfTextAtSize } from '../text/helper';
|
4
|
+
import type { Styles, TableInput, Settings, Section, StylesProps } from './types';
|
5
|
+
|
6
|
+
type ContentSettings = { body: Row[]; head: Row[]; columns: Column[] };
|
7
|
+
|
8
|
+
export class Cell {
|
9
|
+
raw: string;
|
10
|
+
text: string[];
|
11
|
+
styles: Styles;
|
12
|
+
section: Section;
|
13
|
+
contentHeight = 0;
|
14
|
+
contentWidth = 0;
|
15
|
+
wrappedWidth = 0;
|
16
|
+
minReadableWidth = 0;
|
17
|
+
minWidth = 0;
|
18
|
+
|
19
|
+
width = 0;
|
20
|
+
height = 0;
|
21
|
+
x = 0;
|
22
|
+
y = 0;
|
23
|
+
|
24
|
+
constructor(raw: string, styles: Styles, section: Section) {
|
25
|
+
this.styles = styles;
|
26
|
+
this.section = section;
|
27
|
+
this.raw = raw;
|
28
|
+
const splitRegex = /\r\n|\r|\n/g;
|
29
|
+
this.text = raw.split(splitRegex);
|
30
|
+
}
|
31
|
+
|
32
|
+
getContentHeight() {
|
33
|
+
const lineCount = Array.isArray(this.text) ? this.text.length : 1;
|
34
|
+
const lineHeight = pt2mm(this.styles.fontSize) * this.styles.lineHeight;
|
35
|
+
const vPadding = this.padding('top') + this.padding('bottom');
|
36
|
+
const height = lineCount * lineHeight + vPadding;
|
37
|
+
return Math.max(height, this.styles.minCellHeight);
|
38
|
+
}
|
39
|
+
|
40
|
+
padding(name: 'top' | 'bottom' | 'left' | 'right') {
|
41
|
+
return this.styles.cellPadding[name];
|
42
|
+
}
|
43
|
+
}
|
44
|
+
|
45
|
+
export class Column {
|
46
|
+
index: number;
|
47
|
+
wrappedWidth = 0;
|
48
|
+
minReadableWidth = 0;
|
49
|
+
minWidth = 0;
|
50
|
+
width = 0;
|
51
|
+
|
52
|
+
constructor(index: number) {
|
53
|
+
this.index = index;
|
54
|
+
}
|
55
|
+
|
56
|
+
getMaxCustomCellWidth(table: Table) {
|
57
|
+
let max = 0;
|
58
|
+
for (const row of table.allRows()) {
|
59
|
+
const cell: Cell = row.cells[this.index];
|
60
|
+
max = Math.max(max, cell.styles.cellWidth);
|
61
|
+
}
|
62
|
+
return max;
|
63
|
+
}
|
64
|
+
}
|
65
|
+
|
66
|
+
export class Row {
|
67
|
+
readonly raw: string[];
|
68
|
+
readonly index: number;
|
69
|
+
readonly section: Section;
|
70
|
+
readonly cells: { [key: string]: Cell };
|
71
|
+
|
72
|
+
height = 0;
|
73
|
+
|
74
|
+
constructor(raw: string[], index: number, section: Section, cells: { [key: string]: Cell }) {
|
75
|
+
this.raw = raw;
|
76
|
+
this.index = index;
|
77
|
+
this.section = section;
|
78
|
+
this.cells = cells;
|
79
|
+
}
|
80
|
+
|
81
|
+
getMaxCellHeight(columns: Column[]) {
|
82
|
+
return columns.reduce((acc, column) => Math.max(acc, this.cells[column.index]?.height || 0), 0);
|
83
|
+
}
|
84
|
+
|
85
|
+
getMinimumRowHeight(columns: Column[]) {
|
86
|
+
return columns.reduce((acc: number, column: Column) => {
|
87
|
+
const cell = this.cells[column.index];
|
88
|
+
if (!cell) return 0;
|
89
|
+
const vPadding = cell.padding('top') + cell.padding('bottom');
|
90
|
+
const oneRowHeight = vPadding + cell.styles.lineHeight;
|
91
|
+
return oneRowHeight > acc ? oneRowHeight : acc;
|
92
|
+
}, 0);
|
93
|
+
}
|
94
|
+
}
|
95
|
+
|
96
|
+
export class Table {
|
97
|
+
readonly settings: Settings;
|
98
|
+
readonly styles: StylesProps;
|
99
|
+
|
100
|
+
readonly columns: Column[];
|
101
|
+
readonly head: Row[];
|
102
|
+
readonly body: Row[];
|
103
|
+
|
104
|
+
constructor(input: TableInput, content: ContentSettings) {
|
105
|
+
this.settings = input.settings;
|
106
|
+
this.styles = input.styles;
|
107
|
+
|
108
|
+
this.columns = content.columns;
|
109
|
+
this.head = content.head;
|
110
|
+
this.body = content.body;
|
111
|
+
}
|
112
|
+
|
113
|
+
static async create(arg: {
|
114
|
+
input: TableInput;
|
115
|
+
content: ContentSettings;
|
116
|
+
font: Font;
|
117
|
+
_cache: Map<any, any>;
|
118
|
+
}) {
|
119
|
+
const { input, content, font, _cache } = arg;
|
120
|
+
const table = new Table(input, content);
|
121
|
+
|
122
|
+
await calculateWidths({ table, font, _cache });
|
123
|
+
|
124
|
+
return table;
|
125
|
+
}
|
126
|
+
|
127
|
+
getHeadHeight() {
|
128
|
+
return this.head.reduce((acc, row) => acc + row.getMaxCellHeight(this.columns), 0);
|
129
|
+
}
|
130
|
+
|
131
|
+
getBodyHeight() {
|
132
|
+
return this.body.reduce((acc, row) => acc + row.getMaxCellHeight(this.columns), 0);
|
133
|
+
}
|
134
|
+
|
135
|
+
allRows() {
|
136
|
+
return this.head.concat(this.body);
|
137
|
+
}
|
138
|
+
|
139
|
+
getWidth() {
|
140
|
+
return this.settings.tableWidth;
|
141
|
+
}
|
142
|
+
|
143
|
+
getHeight() {
|
144
|
+
return (this.settings.showHead ? this.getHeadHeight() : 0) + this.getBodyHeight();
|
145
|
+
}
|
146
|
+
}
|
147
|
+
|
148
|
+
async function calculateWidths(arg: { table: Table; font: Font; _cache: Map<any, any> }) {
|
149
|
+
const { table, font, _cache } = arg;
|
150
|
+
|
151
|
+
const getFontKitFontByFontName = (fontName: string | undefined) =>
|
152
|
+
getFontKitFont(fontName, font, _cache);
|
153
|
+
|
154
|
+
await calculate(table, getFontKitFontByFontName);
|
155
|
+
|
156
|
+
const resizableColumns: Column[] = [];
|
157
|
+
let initialTableWidth = 0;
|
158
|
+
|
159
|
+
table.columns.forEach((column) => {
|
160
|
+
const customWidth = column.getMaxCustomCellWidth(table);
|
161
|
+
if (customWidth) {
|
162
|
+
// final column width
|
163
|
+
column.width = customWidth;
|
164
|
+
} else {
|
165
|
+
// initial column width (will be resized)
|
166
|
+
column.width = column.wrappedWidth;
|
167
|
+
resizableColumns.push(column);
|
168
|
+
}
|
169
|
+
initialTableWidth += column.width;
|
170
|
+
});
|
171
|
+
|
172
|
+
// width difference that needs to be distributed
|
173
|
+
let resizeWidth = table.getWidth() - initialTableWidth;
|
174
|
+
|
175
|
+
// first resize attempt: with respect to minReadableWidth and minWidth
|
176
|
+
if (resizeWidth) {
|
177
|
+
resizeWidth = resizeColumns(resizableColumns, resizeWidth, (column) =>
|
178
|
+
Math.max(column.minReadableWidth, column.minWidth)
|
179
|
+
);
|
180
|
+
}
|
181
|
+
|
182
|
+
// second resize attempt: ignore minReadableWidth but respect minWidth
|
183
|
+
if (resizeWidth) {
|
184
|
+
resizeWidth = resizeColumns(resizableColumns, resizeWidth, (column) => column.minWidth);
|
185
|
+
}
|
186
|
+
|
187
|
+
resizeWidth = Math.abs(resizeWidth);
|
188
|
+
|
189
|
+
applyColSpans(table);
|
190
|
+
await fitContent(table, getFontKitFontByFontName);
|
191
|
+
applyRowSpans(table);
|
192
|
+
}
|
193
|
+
|
194
|
+
function applyRowSpans(table: Table) {
|
195
|
+
const rowSpanCells: {
|
196
|
+
[key: string]: { cell: Cell; left: number; row: Row };
|
197
|
+
} = {};
|
198
|
+
let colRowSpansLeft = 1;
|
199
|
+
const all = table.allRows();
|
200
|
+
for (let rowIndex = 0; rowIndex < all.length; rowIndex++) {
|
201
|
+
const row = all[rowIndex];
|
202
|
+
for (const column of table.columns) {
|
203
|
+
const data = rowSpanCells[column.index];
|
204
|
+
if (colRowSpansLeft > 1) {
|
205
|
+
colRowSpansLeft--;
|
206
|
+
delete row.cells[column.index];
|
207
|
+
} else if (data) {
|
208
|
+
data.cell.height += row.height;
|
209
|
+
colRowSpansLeft = 1;
|
210
|
+
delete row.cells[column.index];
|
211
|
+
data.left--;
|
212
|
+
if (data.left <= 1) {
|
213
|
+
delete rowSpanCells[column.index];
|
214
|
+
}
|
215
|
+
} else {
|
216
|
+
const cell = row.cells[column.index];
|
217
|
+
if (!cell) {
|
218
|
+
continue;
|
219
|
+
}
|
220
|
+
cell.height = row.height;
|
221
|
+
}
|
222
|
+
}
|
223
|
+
}
|
224
|
+
}
|
225
|
+
|
226
|
+
function applyColSpans(table: Table) {
|
227
|
+
const all = table.allRows();
|
228
|
+
for (let rowIndex = 0; rowIndex < all.length; rowIndex++) {
|
229
|
+
const row = all[rowIndex];
|
230
|
+
|
231
|
+
let colSpanCell = null;
|
232
|
+
let combinedColSpanWidth = 0;
|
233
|
+
let colSpansLeft = 0;
|
234
|
+
for (let columnIndex = 0; columnIndex < table.columns.length; columnIndex++) {
|
235
|
+
const column = table.columns[columnIndex];
|
236
|
+
|
237
|
+
// Width and colspan
|
238
|
+
colSpansLeft -= 1;
|
239
|
+
if (colSpansLeft > 1 && table.columns[columnIndex + 1]) {
|
240
|
+
combinedColSpanWidth += column.width;
|
241
|
+
delete row.cells[column.index];
|
242
|
+
} else if (colSpanCell) {
|
243
|
+
const cell: Cell = colSpanCell;
|
244
|
+
delete row.cells[column.index];
|
245
|
+
colSpanCell = null;
|
246
|
+
cell.width = column.width + combinedColSpanWidth;
|
247
|
+
} else {
|
248
|
+
const cell = row.cells[column.index];
|
249
|
+
if (!cell) continue;
|
250
|
+
colSpansLeft = 1;
|
251
|
+
combinedColSpanWidth = 0;
|
252
|
+
cell.width = column.width + combinedColSpanWidth;
|
253
|
+
}
|
254
|
+
}
|
255
|
+
}
|
256
|
+
}
|
257
|
+
|
258
|
+
async function fitContent(
|
259
|
+
table: Table,
|
260
|
+
getFontKitFontByFontName: (fontName: string | undefined) => Promise<FontKitFont>
|
261
|
+
) {
|
262
|
+
const rowSpanHeight = { count: 0, height: 0 };
|
263
|
+
for (const row of table.allRows()) {
|
264
|
+
for (const column of table.columns) {
|
265
|
+
const cell: Cell = row.cells[column.index];
|
266
|
+
if (!cell) continue;
|
267
|
+
|
268
|
+
const fontKitFont = await getFontKitFontByFontName(cell.styles.fontName);
|
269
|
+
cell.text = splitTextToSize({
|
270
|
+
value: cell.raw,
|
271
|
+
characterSpacing: cell.styles.characterSpacing,
|
272
|
+
boxWidthInPt: mm2pt(cell.width),
|
273
|
+
fontSize: cell.styles.fontSize,
|
274
|
+
fontKitFont,
|
275
|
+
});
|
276
|
+
|
277
|
+
cell.contentHeight = cell.getContentHeight();
|
278
|
+
|
279
|
+
let realContentHeight = cell.contentHeight;
|
280
|
+
if (rowSpanHeight && rowSpanHeight.count > 0) {
|
281
|
+
if (rowSpanHeight.height > realContentHeight) {
|
282
|
+
realContentHeight = rowSpanHeight.height;
|
283
|
+
}
|
284
|
+
}
|
285
|
+
if (realContentHeight > row.height) {
|
286
|
+
row.height = realContentHeight;
|
287
|
+
}
|
288
|
+
}
|
289
|
+
rowSpanHeight.count--;
|
290
|
+
}
|
291
|
+
}
|
292
|
+
|
293
|
+
function resizeColumns(
|
294
|
+
columns: Column[],
|
295
|
+
resizeWidth: number,
|
296
|
+
getMinWidth: (column: Column) => number
|
297
|
+
) {
|
298
|
+
const initialResizeWidth = resizeWidth;
|
299
|
+
const sumWrappedWidth = columns.reduce((acc, column) => acc + column.wrappedWidth, 0);
|
300
|
+
|
301
|
+
for (let i = 0; i < columns.length; i++) {
|
302
|
+
const column = columns[i];
|
303
|
+
|
304
|
+
const ratio = column.wrappedWidth / sumWrappedWidth;
|
305
|
+
const suggestedChange = initialResizeWidth * ratio;
|
306
|
+
const suggestedWidth = column.width + suggestedChange;
|
307
|
+
|
308
|
+
const minWidth = getMinWidth(column);
|
309
|
+
const newWidth = suggestedWidth < minWidth ? minWidth : suggestedWidth;
|
310
|
+
|
311
|
+
resizeWidth -= newWidth - column.width;
|
312
|
+
column.width = newWidth;
|
313
|
+
}
|
314
|
+
|
315
|
+
resizeWidth = Math.round(resizeWidth * 1e10) / 1e10;
|
316
|
+
|
317
|
+
// Run the resizer again if there's remaining width needs
|
318
|
+
// to be distributed and there're columns that can be resized
|
319
|
+
if (resizeWidth) {
|
320
|
+
const resizableColumns = columns.filter((column) => {
|
321
|
+
return resizeWidth < 0
|
322
|
+
? column.width > getMinWidth(column) // check if column can shrink
|
323
|
+
: true; // check if column can grow
|
324
|
+
});
|
325
|
+
|
326
|
+
if (resizableColumns.length) {
|
327
|
+
resizeWidth = resizeColumns(resizableColumns, resizeWidth, getMinWidth);
|
328
|
+
}
|
329
|
+
}
|
330
|
+
|
331
|
+
return resizeWidth;
|
332
|
+
}
|
333
|
+
|
334
|
+
async function calculate(
|
335
|
+
table: Table,
|
336
|
+
getFontKitFontByFontName: (fontName: string | undefined) => Promise<FontKitFont>
|
337
|
+
) {
|
338
|
+
for (const row of table.allRows()) {
|
339
|
+
for (const column of table.columns) {
|
340
|
+
const cell = row.cells[column.index];
|
341
|
+
if (!cell) continue;
|
342
|
+
|
343
|
+
const hPadding = cell.padding('right') + cell.padding('left');
|
344
|
+
const fontKitFont = await getFontKitFontByFontName(cell.styles.fontName);
|
345
|
+
|
346
|
+
cell.contentWidth = getStringWidth(cell, fontKitFont) + hPadding;
|
347
|
+
|
348
|
+
const longestWordWidth = getStringWidth(
|
349
|
+
Object.assign(cell, { text: cell.text.join(' ').split(/\s+/) }),
|
350
|
+
fontKitFont
|
351
|
+
);
|
352
|
+
cell.minReadableWidth = longestWordWidth + hPadding;
|
353
|
+
|
354
|
+
cell.minWidth = cell.styles.cellWidth;
|
355
|
+
cell.wrappedWidth = cell.styles.cellWidth;
|
356
|
+
}
|
357
|
+
}
|
358
|
+
|
359
|
+
for (const row of table.allRows()) {
|
360
|
+
for (const column of table.columns) {
|
361
|
+
const cell = row.cells[column.index];
|
362
|
+
|
363
|
+
// For now we ignore the minWidth and wrappedWidth of colspan cells when calculating colspan widths.
|
364
|
+
// Could probably be improved upon however.
|
365
|
+
if (cell) {
|
366
|
+
column.wrappedWidth = Math.max(column.wrappedWidth, cell.wrappedWidth);
|
367
|
+
column.minWidth = Math.max(column.minWidth, cell.minWidth);
|
368
|
+
column.minReadableWidth = Math.max(column.minReadableWidth, cell.minReadableWidth);
|
369
|
+
} else {
|
370
|
+
// Respect cellWidth set in columnStyles even if there is no cells for this column
|
371
|
+
// or if the column only have colspan cells. Since the width of colspan cells
|
372
|
+
// does not affect the width of columns, setting columnStyles cellWidth enables the
|
373
|
+
// user to at least do it manually.
|
374
|
+
|
375
|
+
// Note that this is not perfect for now since for example row and table styles are
|
376
|
+
// not accounted for
|
377
|
+
const columnStyles = table.styles.columnStyles[column.index] || {};
|
378
|
+
const cellWidth = columnStyles.cellWidth || columnStyles.minCellWidth;
|
379
|
+
if (cellWidth) {
|
380
|
+
column.minWidth = cellWidth;
|
381
|
+
column.wrappedWidth = cellWidth;
|
382
|
+
}
|
383
|
+
}
|
384
|
+
}
|
385
|
+
}
|
386
|
+
}
|
387
|
+
|
388
|
+
function getStringWidth(cell: Cell, fontKitFont: FontKitFont) {
|
389
|
+
const text = cell.text;
|
390
|
+
const textArr: string[] = Array.isArray(text) ? text : [text];
|
391
|
+
const fontSize = cell.styles.fontSize;
|
392
|
+
const characterSpacing = cell.styles.characterSpacing;
|
393
|
+
const widestLineWidth = textArr
|
394
|
+
.map((text) => widthOfTextAtSize(text, fontKitFont, fontSize, characterSpacing))
|
395
|
+
.reduce((a, b) => Math.max(a, b), 0);
|
396
|
+
|
397
|
+
return widestLineWidth;
|
398
|
+
}
|
@@ -0,0 +1,81 @@
|
|
1
|
+
import { Template, Schema, BasePdf, CommonOptions } from '@pdfme/common';
|
2
|
+
import { createMultiTables, createSingleTable } from './tableHelper';
|
3
|
+
import { cloneDeep } from '../utils';
|
4
|
+
import { getBodyWithRange, getBody } from './helper.js';
|
5
|
+
import { TableSchema } from './types';
|
6
|
+
export const modifyTemplateForTable = async (arg: {
|
7
|
+
template: Template;
|
8
|
+
input: Record<string, string>;
|
9
|
+
_cache: Map<any, any>;
|
10
|
+
options: CommonOptions;
|
11
|
+
}): Promise<Template> => {
|
12
|
+
const { template: t, input, options, _cache } = arg;
|
13
|
+
const template: Template = Object.assign(cloneDeep(t), { schemas: [] });
|
14
|
+
let pageIndex = 0;
|
15
|
+
for (const schemaObj of t.schemas) {
|
16
|
+
const additionalSchemaObjs: (typeof schemaObj)[] = [];
|
17
|
+
for (const [key, schema] of Object.entries(schemaObj)) {
|
18
|
+
if (schema.type === 'table') {
|
19
|
+
schema.__bodyRange = undefined;
|
20
|
+
const body = JSON.parse(input?.[key] || '[]') as string[][];
|
21
|
+
const tables = await createMultiTables(body, {
|
22
|
+
schema,
|
23
|
+
basePdf: template.basePdf,
|
24
|
+
options,
|
25
|
+
_cache,
|
26
|
+
});
|
27
|
+
if (tables.length > 1) {
|
28
|
+
const firstTable = tables[0];
|
29
|
+
schema.__bodyRange = { start: 0, end: firstTable.body.length };
|
30
|
+
const allBodies = tables.map((table) => table.body);
|
31
|
+
const from2ndTable = tables.slice(1);
|
32
|
+
from2ndTable.forEach((table, i) => {
|
33
|
+
const additionalPageIndex = pageIndex + i + 1;
|
34
|
+
|
35
|
+
const additionalSchemaObj = {
|
36
|
+
[key]: {
|
37
|
+
...schema,
|
38
|
+
position: { x: schema.position.x, y: table.settings.startY },
|
39
|
+
height: table.getHeight(),
|
40
|
+
showHead: false,
|
41
|
+
__bodyRange: {
|
42
|
+
start: allBodies.slice(0, i + 1).reduce((acc, cur) => acc + cur.length, 0),
|
43
|
+
end: allBodies.slice(0, i + 2).reduce((acc, cur) => acc + cur.length, 0),
|
44
|
+
},
|
45
|
+
content: input[key],
|
46
|
+
},
|
47
|
+
};
|
48
|
+
additionalSchemaObjs[additionalPageIndex] = additionalSchemaObj;
|
49
|
+
});
|
50
|
+
}
|
51
|
+
}
|
52
|
+
}
|
53
|
+
template.schemas.push(schemaObj);
|
54
|
+
additionalSchemaObjs.forEach((obj, index) => {
|
55
|
+
if (!template.schemas[index]) {
|
56
|
+
template.schemas[index] = obj;
|
57
|
+
} else {
|
58
|
+
template.schemas[index] = { ...template.schemas[index], ...obj };
|
59
|
+
}
|
60
|
+
});
|
61
|
+
pageIndex++;
|
62
|
+
}
|
63
|
+
return template;
|
64
|
+
};
|
65
|
+
|
66
|
+
export const getDynamicHeightForTable = async (
|
67
|
+
value: string,
|
68
|
+
args: {
|
69
|
+
schema: Schema;
|
70
|
+
basePdf: BasePdf;
|
71
|
+
options: CommonOptions;
|
72
|
+
_cache: Map<any, any>;
|
73
|
+
}
|
74
|
+
): Promise<number> => {
|
75
|
+
if (args.schema.type !== 'table') return Promise.resolve(args.schema.height);
|
76
|
+
const schema = args.schema as TableSchema;
|
77
|
+
const body =
|
78
|
+
schema.__bodyRange?.start === 0 ? getBody(value) : getBodyWithRange(value, schema.__bodyRange);
|
79
|
+
const table = await createSingleTable(body, args);
|
80
|
+
return table.getHeight();
|
81
|
+
};
|
@@ -0,0 +1,198 @@
|
|
1
|
+
import {
|
2
|
+
DEFAULT_ALIGNMENT,
|
3
|
+
DEFAULT_FONT_SIZE,
|
4
|
+
DEFAULT_LINE_HEIGHT,
|
5
|
+
DEFAULT_CHARACTER_SPACING,
|
6
|
+
DEFAULT_FONT_COLOR,
|
7
|
+
ALIGN_RIGHT,
|
8
|
+
ALIGN_CENTER,
|
9
|
+
ALIGN_LEFT,
|
10
|
+
VERTICAL_ALIGN_TOP,
|
11
|
+
VERTICAL_ALIGN_MIDDLE,
|
12
|
+
VERTICAL_ALIGN_BOTTOM,
|
13
|
+
} from '../text/constants';
|
14
|
+
import { HEX_COLOR_PATTERN } from '../constants.js';
|
15
|
+
|
16
|
+
export const getDefaultCellStyles = () => ({
|
17
|
+
fontName: undefined,
|
18
|
+
alignment: DEFAULT_ALIGNMENT,
|
19
|
+
verticalAlignment: VERTICAL_ALIGN_MIDDLE,
|
20
|
+
fontSize: DEFAULT_FONT_SIZE,
|
21
|
+
lineHeight: DEFAULT_LINE_HEIGHT,
|
22
|
+
characterSpacing: DEFAULT_CHARACTER_SPACING,
|
23
|
+
fontColor: DEFAULT_FONT_COLOR,
|
24
|
+
backgroundColor: '',
|
25
|
+
borderColor: '#888888',
|
26
|
+
borderWidth: { top: 0.1, bottom: 0.1, left: 0.1, right: 0.1 },
|
27
|
+
padding: { top: 5, bottom: 5, left: 5, right: 5 },
|
28
|
+
});
|
29
|
+
|
30
|
+
const getBoxDimensionProp = (step = 1) => {
|
31
|
+
const getCommonProp = () => ({
|
32
|
+
type: 'number',
|
33
|
+
widget: 'inputNumber',
|
34
|
+
props: { min: 0, step },
|
35
|
+
span: 6,
|
36
|
+
});
|
37
|
+
return {
|
38
|
+
top: { title: 'Top', ...getCommonProp() },
|
39
|
+
right: { title: 'Right', ...getCommonProp() },
|
40
|
+
bottom: { title: 'Bottom', ...getCommonProp() },
|
41
|
+
left: { title: 'Left', ...getCommonProp() },
|
42
|
+
};
|
43
|
+
};
|
44
|
+
|
45
|
+
export const getCellPropPanelSchema = (arg: {
|
46
|
+
i18n: (key: string) => string;
|
47
|
+
fallbackFontName: string;
|
48
|
+
fontNames: string[];
|
49
|
+
isBody?: boolean;
|
50
|
+
}) => {
|
51
|
+
const { i18n, fallbackFontName, fontNames, isBody } = arg;
|
52
|
+
|
53
|
+
return {
|
54
|
+
fontName: {
|
55
|
+
title: i18n('schemas.text.fontName'),
|
56
|
+
type: 'string',
|
57
|
+
widget: 'select',
|
58
|
+
default: fallbackFontName,
|
59
|
+
props: { options: fontNames.map((name) => ({ label: name, value: name })) },
|
60
|
+
span: 12,
|
61
|
+
},
|
62
|
+
fontSize: {
|
63
|
+
title: i18n('schemas.text.size'),
|
64
|
+
type: 'number',
|
65
|
+
widget: 'inputNumber',
|
66
|
+
props: { min: 0 },
|
67
|
+
span: 6,
|
68
|
+
},
|
69
|
+
characterSpacing: {
|
70
|
+
title: i18n('schemas.text.spacing'),
|
71
|
+
type: 'number',
|
72
|
+
widget: 'inputNumber',
|
73
|
+
props: { min: 0 },
|
74
|
+
span: 6,
|
75
|
+
},
|
76
|
+
alignment: {
|
77
|
+
title: i18n('schemas.text.textAlign'),
|
78
|
+
type: 'string',
|
79
|
+
widget: 'select',
|
80
|
+
props: {
|
81
|
+
options: [
|
82
|
+
{ label: i18n('schemas.left'), value: ALIGN_LEFT },
|
83
|
+
{ label: i18n('schemas.center'), value: ALIGN_CENTER },
|
84
|
+
{ label: i18n('schemas.right'), value: ALIGN_RIGHT },
|
85
|
+
],
|
86
|
+
},
|
87
|
+
span: 8,
|
88
|
+
},
|
89
|
+
verticalAlignment: {
|
90
|
+
title: i18n('schemas.text.verticalAlign'),
|
91
|
+
type: 'string',
|
92
|
+
widget: 'select',
|
93
|
+
props: {
|
94
|
+
options: [
|
95
|
+
{ label: i18n('schemas.top'), value: VERTICAL_ALIGN_TOP },
|
96
|
+
{ label: i18n('schemas.middle'), value: VERTICAL_ALIGN_MIDDLE },
|
97
|
+
{ label: i18n('schemas.bottom'), value: VERTICAL_ALIGN_BOTTOM },
|
98
|
+
],
|
99
|
+
},
|
100
|
+
span: 8,
|
101
|
+
},
|
102
|
+
lineHeight: {
|
103
|
+
title: i18n('schemas.text.lineHeight'),
|
104
|
+
type: 'number',
|
105
|
+
widget: 'inputNumber',
|
106
|
+
props: { step: 0.1, min: 0 },
|
107
|
+
span: 8,
|
108
|
+
},
|
109
|
+
fontColor: {
|
110
|
+
title: i18n('schemas.textColor'),
|
111
|
+
type: 'string',
|
112
|
+
widget: 'color',
|
113
|
+
rules: [{ pattern: HEX_COLOR_PATTERN, message: i18n('hexColorPrompt') }],
|
114
|
+
},
|
115
|
+
borderColor: {
|
116
|
+
title: i18n('schemas.borderColor'),
|
117
|
+
type: 'string',
|
118
|
+
widget: 'color',
|
119
|
+
rules: [{ pattern: HEX_COLOR_PATTERN, message: i18n('hexColorPrompt') }],
|
120
|
+
},
|
121
|
+
backgroundColor: {
|
122
|
+
title: i18n('schemas.backgroundColor'),
|
123
|
+
type: 'string',
|
124
|
+
widget: 'color',
|
125
|
+
rules: [{ pattern: HEX_COLOR_PATTERN, message: i18n('hexColorPrompt') }],
|
126
|
+
},
|
127
|
+
...(isBody
|
128
|
+
? {
|
129
|
+
alternateBackgroundColor: {
|
130
|
+
title: i18n('schemas.table.alternateBackgroundColor'),
|
131
|
+
type: 'string',
|
132
|
+
widget: 'color',
|
133
|
+
rules: [{ pattern: HEX_COLOR_PATTERN, message: i18n('hexColorPrompt') }],
|
134
|
+
},
|
135
|
+
}
|
136
|
+
: {}),
|
137
|
+
'-': { type: 'void', widget: 'Divider' },
|
138
|
+
borderWidth: {
|
139
|
+
title: i18n('schemas.borderWidth'),
|
140
|
+
type: 'object',
|
141
|
+
widget: 'lineTitle',
|
142
|
+
span: 24,
|
143
|
+
properties: getBoxDimensionProp(0.1),
|
144
|
+
},
|
145
|
+
'--': { type: 'void', widget: 'Divider' },
|
146
|
+
padding: {
|
147
|
+
title: i18n('schemas.padding'),
|
148
|
+
type: 'object',
|
149
|
+
widget: 'lineTitle',
|
150
|
+
span: 24,
|
151
|
+
properties: getBoxDimensionProp(),
|
152
|
+
},
|
153
|
+
};
|
154
|
+
};
|
155
|
+
|
156
|
+
export const getColumnStylesPropPanelSchema = ({
|
157
|
+
head,
|
158
|
+
i18n,
|
159
|
+
}: {
|
160
|
+
head: string[];
|
161
|
+
i18n: (key: string) => string;
|
162
|
+
}) => ({
|
163
|
+
alignment: {
|
164
|
+
type: 'object',
|
165
|
+
widget: 'lineTitle',
|
166
|
+
title: i18n('schemas.text.textAlign'),
|
167
|
+
column: 3,
|
168
|
+
properties: head.reduce(
|
169
|
+
(acc, cur, i) =>
|
170
|
+
Object.assign(acc, {
|
171
|
+
[i]: {
|
172
|
+
title: cur || 'Column ' + String(i + 1),
|
173
|
+
type: 'string',
|
174
|
+
widget: 'select',
|
175
|
+
props: {
|
176
|
+
options: [
|
177
|
+
{ label: i18n('schemas.left'), value: ALIGN_LEFT },
|
178
|
+
{ label: i18n('schemas.center'), value: ALIGN_CENTER },
|
179
|
+
{ label: i18n('schemas.right'), value: ALIGN_RIGHT },
|
180
|
+
],
|
181
|
+
},
|
182
|
+
},
|
183
|
+
}),
|
184
|
+
{}
|
185
|
+
),
|
186
|
+
},
|
187
|
+
});
|
188
|
+
|
189
|
+
export const getBody = (value: string) => JSON.parse(value || '[]') as string[][];
|
190
|
+
|
191
|
+
export const getBodyWithRange = (
|
192
|
+
value: string,
|
193
|
+
range?: { start: number; end?: number | undefined }
|
194
|
+
) => {
|
195
|
+
const body = getBody(value);
|
196
|
+
if (!range) return body;
|
197
|
+
return body.slice(range.start, range.end);
|
198
|
+
};
|
@@ -0,0 +1,12 @@
|
|
1
|
+
import type { Plugin } from '@pdfme/common';
|
2
|
+
import type { TableSchema } from './types.js';
|
3
|
+
import { pdfRender } from './pdfRender.js';
|
4
|
+
import { uiRender } from './uiRender.js';
|
5
|
+
import { propPanel } from './propPanel.js';
|
6
|
+
|
7
|
+
const tableSchema: Plugin<TableSchema> = {
|
8
|
+
pdf: pdfRender,
|
9
|
+
ui: uiRender,
|
10
|
+
propPanel,
|
11
|
+
};
|
12
|
+
export default tableSchema;
|