@pdfme/schemas 3.4.3 → 4.0.0-dev.3
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,393 @@
|
|
1
|
+
import type { UIRenderProps, Mode } from '@pdfme/common';
|
2
|
+
import type { TableSchema, CellStyle, Styles } from './types.js';
|
3
|
+
import { px2mm } from '@pdfme/common';
|
4
|
+
import { createSingleTable } from './tableHelper.js';
|
5
|
+
import { getBody, getBodyWithRange } from './helper.js';
|
6
|
+
import cell from './cell.js';
|
7
|
+
import { Row } from './classes';
|
8
|
+
|
9
|
+
type RowType = InstanceType<typeof Row>;
|
10
|
+
|
11
|
+
const cellUiRender = cell.ui;
|
12
|
+
|
13
|
+
const convertToCellStyle = (styles: Styles): CellStyle => ({
|
14
|
+
fontName: styles.fontName,
|
15
|
+
alignment: styles.alignment,
|
16
|
+
verticalAlignment: styles.verticalAlignment,
|
17
|
+
fontSize: styles.fontSize,
|
18
|
+
lineHeight: styles.lineHeight,
|
19
|
+
characterSpacing: styles.characterSpacing,
|
20
|
+
backgroundColor: styles.backgroundColor,
|
21
|
+
// ---
|
22
|
+
fontColor: styles.textColor,
|
23
|
+
borderColor: styles.lineColor,
|
24
|
+
borderWidth: styles.lineWidth,
|
25
|
+
padding: styles.cellPadding,
|
26
|
+
});
|
27
|
+
|
28
|
+
const calcResizedHeadWidthPercentages = (arg: {
|
29
|
+
currentHeadWidthPercentages: number[];
|
30
|
+
currentHeadWidths: number[];
|
31
|
+
changedHeadWidth: number;
|
32
|
+
changedHeadIndex: number;
|
33
|
+
}) => {
|
34
|
+
const { currentHeadWidthPercentages, currentHeadWidths, changedHeadWidth, changedHeadIndex } =
|
35
|
+
arg;
|
36
|
+
const headWidthPercentages = [...currentHeadWidthPercentages];
|
37
|
+
const totalWidth = currentHeadWidths.reduce((a, b) => a + b, 0);
|
38
|
+
const changedWidthPercentage = (changedHeadWidth / totalWidth) * 100;
|
39
|
+
const originalNextWidthPercentage = headWidthPercentages[changedHeadIndex + 1] ?? 0;
|
40
|
+
const adjustment = headWidthPercentages[changedHeadIndex] - changedWidthPercentage;
|
41
|
+
headWidthPercentages[changedHeadIndex] = changedWidthPercentage;
|
42
|
+
if (changedHeadIndex + 1 < headWidthPercentages.length) {
|
43
|
+
headWidthPercentages[changedHeadIndex + 1] = originalNextWidthPercentage + adjustment;
|
44
|
+
}
|
45
|
+
return headWidthPercentages;
|
46
|
+
};
|
47
|
+
|
48
|
+
const setBorder = (
|
49
|
+
div: HTMLDivElement,
|
50
|
+
borderPosition: 'Top' | 'Left' | 'Right' | 'Bottom',
|
51
|
+
arg: UIRenderProps<TableSchema>
|
52
|
+
) => {
|
53
|
+
div.style[`border${borderPosition}`] = `${String(arg.schema.tableStyles.borderWidth)}mm solid ${
|
54
|
+
arg.schema.tableStyles.borderColor
|
55
|
+
}`;
|
56
|
+
};
|
57
|
+
|
58
|
+
const drawBorder = (
|
59
|
+
div: HTMLDivElement,
|
60
|
+
row: RowType,
|
61
|
+
colIndex: number,
|
62
|
+
rowIndex: number,
|
63
|
+
rowsLength: number,
|
64
|
+
arg: UIRenderProps<TableSchema>
|
65
|
+
) => {
|
66
|
+
const isFirstColumn = colIndex === 0;
|
67
|
+
const isLastColumn = colIndex === Object.values(row.cells).length - 1;
|
68
|
+
const isLastRow = rowIndex === rowsLength - 1;
|
69
|
+
|
70
|
+
if (row.section === 'head') {
|
71
|
+
setBorder(div, 'Top', arg);
|
72
|
+
if (isFirstColumn) setBorder(div, 'Left', arg);
|
73
|
+
if (isLastColumn) setBorder(div, 'Right', arg);
|
74
|
+
if ((JSON.parse(arg.value || '[]') as string[][]).length === 0) {
|
75
|
+
setBorder(div, 'Bottom', arg);
|
76
|
+
}
|
77
|
+
} else if (row.section === 'body') {
|
78
|
+
if (!arg.schema.showHead && rowIndex === 0) {
|
79
|
+
setBorder(div, 'Top', arg);
|
80
|
+
}
|
81
|
+
if (isFirstColumn) setBorder(div, 'Left', arg);
|
82
|
+
if (isLastColumn) setBorder(div, 'Right', arg);
|
83
|
+
if (isLastRow) setBorder(div, 'Bottom', arg);
|
84
|
+
}
|
85
|
+
};
|
86
|
+
|
87
|
+
const renderRowUi = (args: {
|
88
|
+
rows: RowType[];
|
89
|
+
arg: UIRenderProps<TableSchema>;
|
90
|
+
editingPosition: { rowIndex: number; colIndex: number };
|
91
|
+
onChangeEditingPosition: (position: { rowIndex: number; colIndex: number }) => void;
|
92
|
+
offsetY?: number;
|
93
|
+
}) => {
|
94
|
+
const { rows, arg, onChangeEditingPosition, offsetY = 0, editingPosition } = args;
|
95
|
+
const value = JSON.parse(arg.value || '[]') as string[][];
|
96
|
+
|
97
|
+
let rowOffsetY = offsetY;
|
98
|
+
rows.forEach((row, rowIndex) => {
|
99
|
+
const { cells, height, section } = row;
|
100
|
+
let colOffsetX = 0;
|
101
|
+
Object.values(cells).forEach((cell, colIndex) => {
|
102
|
+
const div = document.createElement('div');
|
103
|
+
div.style.position = 'absolute';
|
104
|
+
div.style.top = `${rowOffsetY}mm`;
|
105
|
+
div.style.left = `${colOffsetX}mm`;
|
106
|
+
div.style.width = `${cell.width}mm`;
|
107
|
+
div.style.height = `${cell.height}mm`;
|
108
|
+
div.style.boxSizing = 'border-box';
|
109
|
+
|
110
|
+
drawBorder(div, row, colIndex, rowIndex, rows.length, arg);
|
111
|
+
|
112
|
+
div.style.cursor =
|
113
|
+
arg.mode === 'designer' || (arg.mode === 'form' && section === 'body') ? 'text' : 'default';
|
114
|
+
|
115
|
+
div.addEventListener('click', () => {
|
116
|
+
if (arg.mode === 'viewer') return;
|
117
|
+
onChangeEditingPosition({ rowIndex, colIndex });
|
118
|
+
});
|
119
|
+
arg.rootElement.appendChild(div);
|
120
|
+
const isEditing =
|
121
|
+
editingPosition.rowIndex === rowIndex && editingPosition.colIndex === colIndex;
|
122
|
+
let mode: Mode = 'viewer';
|
123
|
+
if (arg.mode === 'form') {
|
124
|
+
mode = section === 'body' && isEditing ? 'designer' : 'viewer';
|
125
|
+
} else if (arg.mode === 'designer') {
|
126
|
+
mode = isEditing ? 'designer' : 'form';
|
127
|
+
}
|
128
|
+
|
129
|
+
void cellUiRender({
|
130
|
+
...arg,
|
131
|
+
stopEditing: () => {
|
132
|
+
if (arg.mode === 'form') {
|
133
|
+
resetEditingPosition();
|
134
|
+
}
|
135
|
+
},
|
136
|
+
mode,
|
137
|
+
onChange: (v) => {
|
138
|
+
if (!arg.onChange) return;
|
139
|
+
const newValue = (Array.isArray(v) ? v[0].value : v.value) as string;
|
140
|
+
if (section === 'body') {
|
141
|
+
const startRange = arg.schema.__bodyRange?.start ?? 0;
|
142
|
+
value[rowIndex + startRange][colIndex] = newValue;
|
143
|
+
arg.onChange({ key: 'content', value: JSON.stringify(value) });
|
144
|
+
} else {
|
145
|
+
const newHead = [...arg.schema.head];
|
146
|
+
newHead[colIndex] = newValue;
|
147
|
+
arg.onChange({ key: 'head', value: newHead });
|
148
|
+
}
|
149
|
+
},
|
150
|
+
value: cell.raw,
|
151
|
+
placeholder: '',
|
152
|
+
rootElement: div,
|
153
|
+
schema: {
|
154
|
+
type: 'cell',
|
155
|
+
content: cell.raw,
|
156
|
+
position: { x: colOffsetX, y: rowOffsetY },
|
157
|
+
width: cell.width,
|
158
|
+
height: cell.height,
|
159
|
+
...convertToCellStyle(cell.styles),
|
160
|
+
},
|
161
|
+
});
|
162
|
+
colOffsetX += cell.width;
|
163
|
+
});
|
164
|
+
rowOffsetY += height;
|
165
|
+
});
|
166
|
+
};
|
167
|
+
|
168
|
+
const headEditingPosition = { rowIndex: -1, colIndex: -1 };
|
169
|
+
const bodyEditingPosition = { rowIndex: -1, colIndex: -1 };
|
170
|
+
const resetEditingPosition = () => {
|
171
|
+
headEditingPosition.rowIndex = -1;
|
172
|
+
headEditingPosition.colIndex = -1;
|
173
|
+
bodyEditingPosition.rowIndex = -1;
|
174
|
+
bodyEditingPosition.colIndex = -1;
|
175
|
+
};
|
176
|
+
|
177
|
+
export const uiRender = async (arg: UIRenderProps<TableSchema>) => {
|
178
|
+
const { rootElement, onChange, schema, value, mode } = arg;
|
179
|
+
const body = getBody(value);
|
180
|
+
const bodyWidthRange = getBodyWithRange(value, schema.__bodyRange);
|
181
|
+
const table = await createSingleTable(bodyWidthRange, arg);
|
182
|
+
|
183
|
+
rootElement.innerHTML = '';
|
184
|
+
|
185
|
+
const handleChangeEditingPosition = (
|
186
|
+
newPosition: { rowIndex: number; colIndex: number },
|
187
|
+
editingPosition: { rowIndex: number; colIndex: number }
|
188
|
+
) => {
|
189
|
+
resetEditingPosition();
|
190
|
+
editingPosition.rowIndex = newPosition.rowIndex;
|
191
|
+
editingPosition.colIndex = newPosition.colIndex;
|
192
|
+
void uiRender(arg);
|
193
|
+
};
|
194
|
+
|
195
|
+
if (schema.showHead) {
|
196
|
+
renderRowUi({
|
197
|
+
rows: table.head,
|
198
|
+
arg,
|
199
|
+
editingPosition: headEditingPosition,
|
200
|
+
onChangeEditingPosition: (p) => handleChangeEditingPosition(p, headEditingPosition),
|
201
|
+
});
|
202
|
+
}
|
203
|
+
|
204
|
+
const offsetY = schema.showHead ? table.getHeadHeight() : 0;
|
205
|
+
renderRowUi({
|
206
|
+
rows: table.body,
|
207
|
+
arg,
|
208
|
+
editingPosition: bodyEditingPosition,
|
209
|
+
onChangeEditingPosition: (p) => {
|
210
|
+
handleChangeEditingPosition(p, bodyEditingPosition);
|
211
|
+
},
|
212
|
+
offsetY,
|
213
|
+
});
|
214
|
+
|
215
|
+
if (mode === 'form' && onChange) {
|
216
|
+
if (
|
217
|
+
schema.__bodyRange?.end === undefined ||
|
218
|
+
schema.__bodyRange.end >= (JSON.parse(value || '[]') as string[][]).length
|
219
|
+
) {
|
220
|
+
const addRowButton = document.createElement('button');
|
221
|
+
addRowButton.style.width = '30px';
|
222
|
+
addRowButton.style.height = '30px';
|
223
|
+
addRowButton.style.position = 'absolute';
|
224
|
+
addRowButton.style.top = `${table.getHeight()}mm`;
|
225
|
+
addRowButton.style.left = 'calc(50% - 15px)';
|
226
|
+
addRowButton.innerText = '+';
|
227
|
+
addRowButton.onclick = () => {
|
228
|
+
const newRow = Array(schema.head.length).fill('') as string[];
|
229
|
+
onChange({ key: 'content', value: JSON.stringify(body.concat([newRow])) });
|
230
|
+
};
|
231
|
+
rootElement.appendChild(addRowButton);
|
232
|
+
}
|
233
|
+
|
234
|
+
let offsetY = schema.showHead ? table.getHeadHeight() : 0;
|
235
|
+
table.body.forEach((row, i) => {
|
236
|
+
offsetY = offsetY + row.height;
|
237
|
+
const removeRowButton = document.createElement('button');
|
238
|
+
removeRowButton.style.width = '30px';
|
239
|
+
removeRowButton.style.height = '30px';
|
240
|
+
removeRowButton.style.position = 'absolute';
|
241
|
+
removeRowButton.style.top = `${offsetY - px2mm(30)}mm`;
|
242
|
+
removeRowButton.style.right = '-30px';
|
243
|
+
removeRowButton.innerText = '-';
|
244
|
+
removeRowButton.onclick = () => {
|
245
|
+
const newTableBody = body.filter((_, j) => j !== i + (schema.__bodyRange?.start ?? 0));
|
246
|
+
onChange({ key: 'content', value: JSON.stringify(newTableBody) });
|
247
|
+
};
|
248
|
+
rootElement.appendChild(removeRowButton);
|
249
|
+
});
|
250
|
+
}
|
251
|
+
|
252
|
+
if (mode === 'designer' && onChange) {
|
253
|
+
const addColumnButton = document.createElement('button');
|
254
|
+
addColumnButton.style.width = '30px';
|
255
|
+
addColumnButton.style.height = '30px';
|
256
|
+
addColumnButton.style.position = 'absolute';
|
257
|
+
addColumnButton.style.top = `${table.getHeadHeight() - px2mm(30)}mm`;
|
258
|
+
addColumnButton.style.right = '-30px';
|
259
|
+
addColumnButton.innerText = '+';
|
260
|
+
addColumnButton.onclick = (e) => {
|
261
|
+
e.preventDefault();
|
262
|
+
const newColumnWidthPercentage = 25;
|
263
|
+
const totalCurrentWidth = schema.headWidthPercentages.reduce((acc, width) => acc + width, 0);
|
264
|
+
const scalingRatio = (100 - newColumnWidthPercentage) / totalCurrentWidth;
|
265
|
+
const scaledWidths = schema.headWidthPercentages.map((width) => width * scalingRatio);
|
266
|
+
onChange([
|
267
|
+
{ key: 'head', value: schema.head.concat(`Head ${schema.head.length + 1}`) },
|
268
|
+
{ key: 'headWidthPercentages', value: scaledWidths.concat(newColumnWidthPercentage) },
|
269
|
+
{
|
270
|
+
key: 'content',
|
271
|
+
value: JSON.stringify(bodyWidthRange.map((row, i) => row.concat(`Row ${i + 1}`))),
|
272
|
+
},
|
273
|
+
]);
|
274
|
+
};
|
275
|
+
rootElement.appendChild(addColumnButton);
|
276
|
+
|
277
|
+
let offsetX = 0;
|
278
|
+
table.columns.forEach((column, i) => {
|
279
|
+
offsetX = offsetX + column.width;
|
280
|
+
const removeColumnButton = document.createElement('button');
|
281
|
+
removeColumnButton.style.width = '30px';
|
282
|
+
removeColumnButton.style.height = '30px';
|
283
|
+
removeColumnButton.style.position = 'absolute';
|
284
|
+
removeColumnButton.style.top = '-30px';
|
285
|
+
removeColumnButton.style.left = `${offsetX - px2mm(30)}mm`;
|
286
|
+
removeColumnButton.innerText = '-';
|
287
|
+
removeColumnButton.onclick = (e) => {
|
288
|
+
e.preventDefault();
|
289
|
+
const totalWidthMinusRemoved = schema.headWidthPercentages.reduce(
|
290
|
+
(sum, width, j) => (j !== i ? sum + width : sum),
|
291
|
+
0
|
292
|
+
);
|
293
|
+
|
294
|
+
// TODO Should also remove the deleted columnStyles when deleting
|
295
|
+
onChange([
|
296
|
+
{ key: 'head', value: schema.head.filter((_, j) => j !== i) },
|
297
|
+
{
|
298
|
+
key: 'headWidthPercentages',
|
299
|
+
value: schema.headWidthPercentages
|
300
|
+
.filter((_, j) => j !== i)
|
301
|
+
.map((width) => (width / totalWidthMinusRemoved) * 100),
|
302
|
+
},
|
303
|
+
{
|
304
|
+
key: 'content',
|
305
|
+
value: JSON.stringify(bodyWidthRange.map((row) => row.filter((_, j) => j !== i))),
|
306
|
+
},
|
307
|
+
]);
|
308
|
+
};
|
309
|
+
rootElement.appendChild(removeColumnButton);
|
310
|
+
|
311
|
+
if (i === table.columns.length - 1) return;
|
312
|
+
|
313
|
+
const dragHandle = document.createElement('div');
|
314
|
+
const lineWidth = 5;
|
315
|
+
dragHandle.style.width = `${lineWidth}px`;
|
316
|
+
dragHandle.style.height = '100%';
|
317
|
+
dragHandle.style.backgroundColor = '#eee';
|
318
|
+
dragHandle.style.opacity = '0.5';
|
319
|
+
dragHandle.style.cursor = 'col-resize';
|
320
|
+
dragHandle.style.position = 'absolute';
|
321
|
+
dragHandle.style.zIndex = '10';
|
322
|
+
dragHandle.style.left = `${offsetX - px2mm(lineWidth) / 2}mm`;
|
323
|
+
dragHandle.style.top = '0';
|
324
|
+
const setColor = (e: MouseEvent) => {
|
325
|
+
const handle = e.target as HTMLDivElement;
|
326
|
+
handle.style.backgroundColor = '#2196f3';
|
327
|
+
};
|
328
|
+
const resetColor = (e: MouseEvent) => {
|
329
|
+
const handle = e.target as HTMLDivElement;
|
330
|
+
handle.style.backgroundColor = '#eee';
|
331
|
+
};
|
332
|
+
dragHandle.addEventListener('mouseover', setColor);
|
333
|
+
dragHandle.addEventListener('mouseout', resetColor);
|
334
|
+
|
335
|
+
const prevColumnLeft = offsetX - column.width;
|
336
|
+
const nextColumnRight = offsetX - px2mm(lineWidth) + table.columns[i + 1].width;
|
337
|
+
|
338
|
+
dragHandle.addEventListener('mousedown', (e) => {
|
339
|
+
resetEditingPosition();
|
340
|
+
const handle = e.target as HTMLDivElement;
|
341
|
+
dragHandle.removeEventListener('mouseover', setColor);
|
342
|
+
dragHandle.removeEventListener('mouseout', resetColor);
|
343
|
+
|
344
|
+
let move = 0;
|
345
|
+
const mouseMove = (e: MouseEvent) => {
|
346
|
+
// TODO There is an issue where newLeft gets displaced with drag & drop
|
347
|
+
let moveX = e.movementX;
|
348
|
+
const currentLeft = Number(handle.style.left.replace('mm', ''));
|
349
|
+
let newLeft = currentLeft + moveX;
|
350
|
+
if (newLeft < prevColumnLeft) {
|
351
|
+
newLeft = prevColumnLeft;
|
352
|
+
moveX = newLeft - currentLeft;
|
353
|
+
}
|
354
|
+
if (newLeft >= nextColumnRight) {
|
355
|
+
newLeft = nextColumnRight;
|
356
|
+
moveX = newLeft - currentLeft;
|
357
|
+
}
|
358
|
+
handle.style.left = `${newLeft}mm`;
|
359
|
+
move += moveX;
|
360
|
+
};
|
361
|
+
rootElement.addEventListener('mousemove', mouseMove);
|
362
|
+
|
363
|
+
const commitResize = () => {
|
364
|
+
if (move !== 0) {
|
365
|
+
const newHeadWidthPercentages = calcResizedHeadWidthPercentages({
|
366
|
+
currentHeadWidthPercentages: schema.headWidthPercentages,
|
367
|
+
currentHeadWidths: table.columns.map((column) => column.width),
|
368
|
+
changedHeadWidth: table.columns[i].width + move,
|
369
|
+
changedHeadIndex: i,
|
370
|
+
});
|
371
|
+
onChange({ key: 'headWidthPercentages', value: newHeadWidthPercentages });
|
372
|
+
}
|
373
|
+
move = 0;
|
374
|
+
dragHandle.addEventListener('mouseover', setColor);
|
375
|
+
dragHandle.addEventListener('mouseout', resetColor);
|
376
|
+
rootElement.removeEventListener('mousemove', mouseMove);
|
377
|
+
rootElement.removeEventListener('mouseup', commitResize);
|
378
|
+
};
|
379
|
+
rootElement.addEventListener('mouseup', commitResize);
|
380
|
+
});
|
381
|
+
rootElement.appendChild(dragHandle);
|
382
|
+
});
|
383
|
+
}
|
384
|
+
|
385
|
+
if (mode === 'viewer') {
|
386
|
+
resetEditingPosition();
|
387
|
+
}
|
388
|
+
|
389
|
+
const tableHeight = schema.showHead ? table.getHeight() : table.getBodyHeight();
|
390
|
+
if (schema.height !== tableHeight && onChange) {
|
391
|
+
onChange({ key: 'height', value: tableHeight });
|
392
|
+
}
|
393
|
+
};
|
package/src/text/helper.ts
CHANGED
@@ -107,15 +107,18 @@ const getFallbackFont = (font: Font) => {
|
|
107
107
|
|
108
108
|
const getCacheKey = (fontName: string) => `getFontKitFont-${fontName}`;
|
109
109
|
|
110
|
-
export const getFontKitFont = async (
|
111
|
-
|
112
|
-
|
110
|
+
export const getFontKitFont = async (
|
111
|
+
fontName: string | undefined,
|
112
|
+
font: Font,
|
113
|
+
_cache: Map<any, any>
|
114
|
+
) => {
|
115
|
+
const fntNm = fontName || getFallbackFontName(font);
|
116
|
+
const cacheKey = getCacheKey(fntNm);
|
113
117
|
if (_cache.has(cacheKey)) {
|
114
118
|
return _cache.get(cacheKey) as fontkit.Font;
|
115
119
|
}
|
116
120
|
|
117
|
-
const currentFont =
|
118
|
-
font[fontName] || getFallbackFont(font) || getDefaultFont()[DEFAULT_FONT_NAME];
|
121
|
+
const currentFont = font[fntNm] || getFallbackFont(font) || getDefaultFont()[DEFAULT_FONT_NAME];
|
119
122
|
let fontData = currentFont.data;
|
120
123
|
if (typeof fontData === 'string') {
|
121
124
|
fontData = fontData.startsWith('http')
|
@@ -244,7 +247,7 @@ export const calculateDynamicFontSize = async ({
|
|
244
247
|
if (dynamicFontSizeSetting.max < dynamicFontSizeSetting.min) return fontSize;
|
245
248
|
|
246
249
|
const characterSpacing = schemaCharacterSpacing ?? DEFAULT_CHARACTER_SPACING;
|
247
|
-
const fontKitFont = await getFontKitFont(textSchema, font, _cache);
|
250
|
+
const fontKitFont = await getFontKitFont(textSchema.fontName, font, _cache);
|
248
251
|
const paragraphs = value.split('\n');
|
249
252
|
|
250
253
|
let dynamicFontSize = fontSize;
|
@@ -339,4 +342,24 @@ export const calculateDynamicFontSize = async ({
|
|
339
342
|
return dynamicFontSize;
|
340
343
|
};
|
341
344
|
|
345
|
+
export const splitTextToSize = (arg: {
|
346
|
+
value: string;
|
347
|
+
characterSpacing: number;
|
348
|
+
boxWidthInPt: number;
|
349
|
+
fontSize: number;
|
350
|
+
fontKitFont: fontkit.Font;
|
351
|
+
}) => {
|
352
|
+
const { value, characterSpacing, fontSize, fontKitFont, boxWidthInPt } = arg;
|
353
|
+
const fontWidthCalcValues: FontWidthCalcValues = {
|
354
|
+
font: fontKitFont,
|
355
|
+
fontSize,
|
356
|
+
characterSpacing,
|
357
|
+
boxWidthInPt,
|
358
|
+
};
|
359
|
+
let lines: string[] = [];
|
360
|
+
value.split(/\r\n|\r|\n|\f|\u000B/g).forEach((line: string) => {
|
361
|
+
lines = lines.concat(getSplittedLines(line, fontWidthCalcValues));
|
362
|
+
});
|
363
|
+
return lines;
|
364
|
+
};
|
342
365
|
export const isFirefox = () => navigator.userAgent.toLowerCase().indexOf('firefox') > -1;
|
package/src/text/index.ts
CHANGED
@@ -16,8 +16,8 @@ export const readOnlyText: Plugin<TextSchema> = {
|
|
16
16
|
defaultSchema: {
|
17
17
|
...textSchema.propPanel.defaultSchema,
|
18
18
|
type: 'readOnlyText',
|
19
|
+
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-type"><polyline points="4 7 4 4 20 4 20 7"/><line x1="9" x2="15" y1="20" y2="20"/><line x1="12" x2="12" y1="4" y2="20"/></svg>',
|
19
20
|
readOnly: true,
|
20
|
-
readOnlyValue: textSchema.propPanel.defaultValue,
|
21
21
|
},
|
22
22
|
},
|
23
23
|
};
|
package/src/text/pdfRender.ts
CHANGED
@@ -1,12 +1,12 @@
|
|
1
1
|
import { PDFFont, PDFDocument } from '@pdfme/pdf-lib';
|
2
|
-
import type { TextSchema
|
2
|
+
import type { TextSchema } from './types';
|
3
3
|
import {
|
4
4
|
PDFRenderProps,
|
5
|
+
ColorType,
|
5
6
|
Font,
|
6
7
|
getDefaultFont,
|
7
8
|
getFallbackFontName,
|
8
9
|
mm2pt,
|
9
|
-
ColorType,
|
10
10
|
} from '@pdfme/common';
|
11
11
|
import {
|
12
12
|
VERTICAL_ALIGN_TOP,
|
@@ -24,8 +24,8 @@ import {
|
|
24
24
|
heightOfFontAtSize,
|
25
25
|
getFontDescentInPt,
|
26
26
|
getFontKitFont,
|
27
|
-
getSplittedLines,
|
28
27
|
widthOfTextAtSize,
|
28
|
+
splitTextToSize,
|
29
29
|
} from './helper.js';
|
30
30
|
import { convertForPdfLayoutProps, rotatePoint, hex2PrintingColor } from '../utils.js';
|
31
31
|
|
@@ -96,8 +96,8 @@ export const pdfRender = async (arg: PDFRenderProps<TextSchema>) => {
|
|
96
96
|
|
97
97
|
const [pdfFontObj, fontKitFont, fontProp] = await Promise.all([
|
98
98
|
embedAndGetFontObj({ pdfDoc, font, _cache }),
|
99
|
-
getFontKitFont(schema, font, _cache),
|
100
|
-
getFontProp({ value, font, schema, _cache
|
99
|
+
getFontKitFont(schema.fontName, font, _cache),
|
100
|
+
getFontProp({ value, font, schema, _cache }),
|
101
101
|
]);
|
102
102
|
|
103
103
|
const { fontSize, color, alignment, verticalAlignment, lineHeight, characterSpacing } = fontProp;
|
@@ -127,16 +127,12 @@ export const pdfRender = async (arg: PDFRenderProps<TextSchema>) => {
|
|
127
127
|
const descent = getFontDescentInPt(fontKitFont, fontSize);
|
128
128
|
const halfLineHeightAdjustment = lineHeight === 0 ? 0 : ((lineHeight - 1) * fontSize) / 2;
|
129
129
|
|
130
|
-
const
|
131
|
-
|
132
|
-
fontSize,
|
130
|
+
const lines = splitTextToSize({
|
131
|
+
value,
|
133
132
|
characterSpacing,
|
133
|
+
fontSize,
|
134
|
+
fontKitFont,
|
134
135
|
boxWidthInPt: width,
|
135
|
-
};
|
136
|
-
|
137
|
-
let lines: string[] = [];
|
138
|
-
value.split(/\r\n|\r|\n|\f|\u000B/g).forEach((line: string) => {
|
139
|
-
lines = lines.concat(getSplittedLines(line, fontWidthCalcValues));
|
140
136
|
});
|
141
137
|
|
142
138
|
// Text lines are rendered from the bottom upwards, we need to adjust the position down
|
package/src/text/propPanel.ts
CHANGED
@@ -72,12 +72,14 @@ export const propPanel: PropPanel<TextSchema> = {
|
|
72
72
|
widget: 'inputNumber',
|
73
73
|
span: 6,
|
74
74
|
disabled: enableDynamicFont,
|
75
|
+
props: { min: 0 },
|
75
76
|
},
|
76
77
|
characterSpacing: {
|
77
78
|
title: i18n('schemas.text.spacing'),
|
78
79
|
type: 'number',
|
79
80
|
widget: 'inputNumber',
|
80
81
|
span: 6,
|
82
|
+
props: { min: 0 },
|
81
83
|
},
|
82
84
|
alignment: {
|
83
85
|
title: i18n('schemas.text.textAlign'),
|
@@ -109,9 +111,7 @@ export const propPanel: PropPanel<TextSchema> = {
|
|
109
111
|
title: i18n('schemas.text.lineHeight'),
|
110
112
|
type: 'number',
|
111
113
|
widget: 'inputNumber',
|
112
|
-
props: {
|
113
|
-
step: 0.1,
|
114
|
-
},
|
114
|
+
props: { step: 0.1, min: 0 },
|
115
115
|
span: 8,
|
116
116
|
},
|
117
117
|
useDynamicFontSize: { type: 'boolean', widget: 'UseDynamicFontSize', bind: false, span: 16 },
|
@@ -125,12 +125,14 @@ export const propPanel: PropPanel<TextSchema> = {
|
|
125
125
|
type: 'number',
|
126
126
|
widget: 'inputNumber',
|
127
127
|
hidden: !enableDynamicFont,
|
128
|
+
props: { min: 0 },
|
128
129
|
},
|
129
130
|
max: {
|
130
131
|
title: i18n('schemas.text.max'),
|
131
132
|
type: 'number',
|
132
133
|
widget: 'inputNumber',
|
133
134
|
hidden: !enableDynamicFont,
|
135
|
+
props: { min: 0 },
|
134
136
|
},
|
135
137
|
fit: {
|
136
138
|
title: i18n('schemas.text.fit'),
|
@@ -173,9 +175,10 @@ export const propPanel: PropPanel<TextSchema> = {
|
|
173
175
|
return textSchema;
|
174
176
|
},
|
175
177
|
widgets: { UseDynamicFontSize },
|
176
|
-
defaultValue: 'Type Something...',
|
177
178
|
defaultSchema: {
|
178
179
|
type: 'text',
|
180
|
+
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-text-cursor-input"><path d="M5 4h1a3 3 0 0 1 3 3 3 3 0 0 1 3-3h1"/><path d="M13 20h-1a3 3 0 0 1-3-3 3 3 0 0 1-3 3H5"/><path d="M5 16H4a2 2 0 0 1-2-2v-4a2 2 0 0 1 2-2h1"/><path d="M13 8h7a2 2 0 0 1 2 2v4a2 2 0 0 1-2 2h-7"/><path d="M9 7v10"/></svg>',
|
181
|
+
content: 'Type Something...',
|
179
182
|
position: { x: 0, y: 0 },
|
180
183
|
width: 45,
|
181
184
|
height: 10,
|
package/src/text/uiRender.ts
CHANGED
@@ -19,7 +19,7 @@ import {
|
|
19
19
|
getBrowserVerticalFontAdjustments,
|
20
20
|
isFirefox,
|
21
21
|
} from './helper.js';
|
22
|
-
import {
|
22
|
+
import { isEditable } from '../utils.js';
|
23
23
|
|
24
24
|
const mapVerticalAlignToFlex = (verticalAlignmentValue: string | undefined) => {
|
25
25
|
switch (verticalAlignmentValue) {
|
@@ -33,10 +33,9 @@ const mapVerticalAlignToFlex = (verticalAlignmentValue: string | undefined) => {
|
|
33
33
|
return 'flex-start';
|
34
34
|
};
|
35
35
|
|
36
|
-
const getBackgroundColor = (value: string, schema: Schema
|
37
|
-
if (!value) return 'transparent';
|
38
|
-
|
39
|
-
return defaultBackgroundColor;
|
36
|
+
const getBackgroundColor = (value: string, schema: Schema) => {
|
37
|
+
if (!value || !schema.backgroundColor) return 'transparent';
|
38
|
+
return schema.backgroundColor as string;
|
40
39
|
};
|
41
40
|
|
42
41
|
export const uiRender = async (arg: UIRenderProps<TextSchema>) => {
|
@@ -50,7 +49,6 @@ export const uiRender = async (arg: UIRenderProps<TextSchema>) => {
|
|
50
49
|
tabIndex,
|
51
50
|
placeholder,
|
52
51
|
options,
|
53
|
-
theme,
|
54
52
|
_cache,
|
55
53
|
} = arg;
|
56
54
|
const font = options?.font || getDefaultFont();
|
@@ -67,7 +65,7 @@ export const uiRender = async (arg: UIRenderProps<TextSchema>) => {
|
|
67
65
|
dynamicFontSize = await calculateDynamicFontSize(getCdfArg(value));
|
68
66
|
}
|
69
67
|
|
70
|
-
const fontKitFont = await getFontKitFont(schema, font, _cache);
|
68
|
+
const fontKitFont = await getFontKitFont(schema.fontName, font, _cache);
|
71
69
|
// Depending on vertical alignment, we need to move the top or bottom of the font to keep
|
72
70
|
// it within it's defined box and align it with the generated pdf.
|
73
71
|
const { topAdj, bottomAdj } = getBrowserVerticalFontAdjustments(
|
@@ -85,13 +83,14 @@ export const uiRender = async (arg: UIRenderProps<TextSchema>) => {
|
|
85
83
|
const containerStyle: CSS.Properties = {
|
86
84
|
padding: 0,
|
87
85
|
resize: 'none',
|
88
|
-
backgroundColor: getBackgroundColor(value, schema
|
86
|
+
backgroundColor: getBackgroundColor(value, schema),
|
89
87
|
border: 'none',
|
90
88
|
display: 'flex',
|
91
89
|
flexDirection: 'column',
|
92
90
|
justifyContent: mapVerticalAlignToFlex(schema.verticalAlignment),
|
93
91
|
width: '100%',
|
94
92
|
height: '100%',
|
93
|
+
cursor: isEditable(mode, schema) ? 'text' : 'default',
|
95
94
|
};
|
96
95
|
Object.assign(container.style, containerStyle);
|
97
96
|
rootElement.innerHTML = '';
|
@@ -143,7 +142,7 @@ export const uiRender = async (arg: UIRenderProps<TextSchema>) => {
|
|
143
142
|
textBlock.tabIndex = tabIndex || 0;
|
144
143
|
textBlock.innerText = value;
|
145
144
|
textBlock.addEventListener('blur', (e: Event) => {
|
146
|
-
onChange && onChange((e.target as HTMLDivElement).innerText);
|
145
|
+
onChange && onChange({ key: 'content', value: (e.target as HTMLDivElement).innerText });
|
147
146
|
stopEditing && stopEditing();
|
148
147
|
});
|
149
148
|
|
@@ -187,15 +186,16 @@ export const uiRender = async (arg: UIRenderProps<TextSchema>) => {
|
|
187
186
|
container.appendChild(textBlock);
|
188
187
|
|
189
188
|
if (mode === 'designer') {
|
190
|
-
|
191
|
-
|
192
|
-
|
193
|
-
|
194
|
-
|
195
|
-
|
196
|
-
|
197
|
-
|
198
|
-
|
189
|
+
setTimeout(() => {
|
190
|
+
textBlock.focus();
|
191
|
+
// Set the focus to the end of the editable element when you focus, as we would for a textarea
|
192
|
+
const selection = window.getSelection();
|
193
|
+
const range = document.createRange();
|
194
|
+
range.selectNodeContents(textBlock);
|
195
|
+
range.collapse(false); // Collapse range to the end
|
196
|
+
selection?.removeAllRanges();
|
197
|
+
selection?.addRange(range);
|
198
|
+
});
|
199
199
|
}
|
200
200
|
} else {
|
201
201
|
textBlock.innerHTML = value
|
package/src/utils.ts
CHANGED