@pdfme/schemas 3.2.3-dev.1 → 4.0.0-alpha.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 +11 -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 +5 -11
- package/dist/cjs/src/graphics/image.js.map +1 -1
- package/dist/cjs/src/graphics/svg.js +2 -3
- package/dist/cjs/src/graphics/svg.js.map +1 -1
- package/dist/cjs/src/index.js +6 -1
- package/dist/cjs/src/index.js.map +1 -1
- package/dist/cjs/src/shapes/line.js +5 -9
- package/dist/cjs/src/shapes/line.js.map +1 -1
- package/dist/cjs/src/shapes/rectAndEllipse.js +3 -2
- 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 +95 -0
- package/dist/cjs/src/tables/pdfRender.js.map +1 -0
- package/dist/cjs/src/tables/propPanel.js +87 -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 +321 -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 +0 -1
- package/dist/cjs/src/text/index.js.map +1 -1
- package/dist/cjs/src/text/pdfRender.js +5 -8
- package/dist/cjs/src/text/pdfRender.js.map +1 -1
- package/dist/cjs/src/text/propPanel.js +6 -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 +9 -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 +11 -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 +4 -10
- package/dist/esm/src/graphics/image.js.map +1 -1
- package/dist/esm/src/graphics/svg.js +2 -3
- package/dist/esm/src/graphics/svg.js.map +1 -1
- package/dist/esm/src/index.js +3 -1
- package/dist/esm/src/index.js.map +1 -1
- package/dist/esm/src/shapes/line.js +5 -9
- package/dist/esm/src/shapes/line.js.map +1 -1
- package/dist/esm/src/shapes/rectAndEllipse.js +3 -2
- 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 +88 -0
- package/dist/esm/src/tables/pdfRender.js.map +1 -0
- package/dist/esm/src/tables/propPanel.js +84 -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 +314 -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 +0 -1
- package/dist/esm/src/text/index.js.map +1 -1
- package/dist/esm/src/text/pdfRender.js +6 -9
- package/dist/esm/src/text/pdfRender.js.map +1 -1
- package/dist/esm/src/text/propPanel.js +6 -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 +6 -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 +13 -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 +2 -0
- package/package.json +1 -1
- package/src/barcodes/propPanel.ts +12 -12
- package/src/barcodes/uiRender.ts +1 -1
- package/src/graphics/image.ts +6 -11
- package/src/graphics/svg.ts +2 -3
- package/src/index.ts +5 -0
- package/src/shapes/line.ts +4 -9
- package/src/shapes/rectAndEllipse.ts +4 -4
- 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 +115 -0
- package/src/tables/propPanel.ts +90 -0
- package/src/tables/tableHelper.ts +322 -0
- package/src/tables/types.ts +88 -0
- package/src/tables/uiRender.ts +362 -0
- package/src/text/helper.ts +30 -6
- package/src/text/index.ts +0 -1
- package/src/text/pdfRender.ts +7 -11
- package/src/text/propPanel.ts +6 -4
- package/src/text/uiRender.ts +18 -18
- package/src/utils.ts +8 -0
@@ -0,0 +1,362 @@
|
|
1
|
+
import type { UIRenderProps, Mode } from '@pdfme/common';
|
2
|
+
import type { TableSchema, CellStyle, Styles } from './types.js';
|
3
|
+
import { createSingleTable } from './tableHelper.js';
|
4
|
+
import { getBody, getBodyWithRange } from './helper.js';
|
5
|
+
import cell from './cell.js';
|
6
|
+
import { px2mm } from '../utils';
|
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 renderRowUi = (args: {
|
49
|
+
rows: RowType[];
|
50
|
+
arg: UIRenderProps<TableSchema>;
|
51
|
+
editingPosition: { rowIndex: number; colIndex: number };
|
52
|
+
onChangeEditingPosition: (position: { rowIndex: number; colIndex: number }) => void;
|
53
|
+
offsetY?: number;
|
54
|
+
}) => {
|
55
|
+
const { rows, arg, onChangeEditingPosition, offsetY = 0, editingPosition } = args;
|
56
|
+
const value = JSON.parse(arg.value || '[]') as string[][];
|
57
|
+
|
58
|
+
// TODO Need to adjust the inner size when the outer border increases
|
59
|
+
// Want to have the same style as border-collapse: collapse; (overlapping borders should merge into one)
|
60
|
+
// This should apply to the table itself as well as the cells, which should behave similarly.
|
61
|
+
let rowOffsetY = offsetY;
|
62
|
+
rows.forEach((row, rowIndex) => {
|
63
|
+
const { cells, height, section } = row;
|
64
|
+
let colWidth = 0;
|
65
|
+
Object.values(cells).forEach((cell, colIndex) => {
|
66
|
+
const div = document.createElement('div');
|
67
|
+
div.style.position = 'absolute';
|
68
|
+
div.style.top = `${rowOffsetY}mm`;
|
69
|
+
div.style.left = `${colWidth}mm`;
|
70
|
+
div.style.width = `${cell.width}mm`;
|
71
|
+
div.style.height = `${cell.height}mm`;
|
72
|
+
|
73
|
+
div.style.cursor =
|
74
|
+
arg.mode === 'designer' || (arg.mode === 'form' && section === 'body') ? 'text' : 'default';
|
75
|
+
|
76
|
+
div.addEventListener('click', () => {
|
77
|
+
if (arg.mode === 'viewer') return;
|
78
|
+
onChangeEditingPosition({ rowIndex, colIndex });
|
79
|
+
});
|
80
|
+
arg.rootElement.appendChild(div);
|
81
|
+
const isEditing =
|
82
|
+
editingPosition.rowIndex === rowIndex && editingPosition.colIndex === colIndex;
|
83
|
+
let mode: Mode = 'viewer';
|
84
|
+
if (arg.mode === 'form') {
|
85
|
+
mode = section === 'body' && isEditing ? 'designer' : 'viewer';
|
86
|
+
} else if (arg.mode === 'designer') {
|
87
|
+
mode = isEditing ? 'designer' : 'form';
|
88
|
+
}
|
89
|
+
|
90
|
+
void cellUiRender({
|
91
|
+
...arg,
|
92
|
+
stopEditing: () => {
|
93
|
+
if (arg.mode === 'form') {
|
94
|
+
resetEditingPosition();
|
95
|
+
}
|
96
|
+
},
|
97
|
+
mode,
|
98
|
+
onChange: (v) => {
|
99
|
+
if (!arg.onChange) return;
|
100
|
+
const newValue = (Array.isArray(v) ? v[0].value : v.value) as string;
|
101
|
+
if (section === 'body') {
|
102
|
+
const startRange = arg.schema.__bodyRange?.start ?? 0;
|
103
|
+
value[rowIndex + startRange][colIndex] = newValue;
|
104
|
+
// TODO Calling onChange triggers re-rendering, causing the focus to be lost
|
105
|
+
arg.onChange({ key: 'content', value: JSON.stringify(value) });
|
106
|
+
} else {
|
107
|
+
const newHead = [...arg.schema.head];
|
108
|
+
newHead[colIndex] = newValue;
|
109
|
+
arg.onChange({ key: 'head', value: newHead });
|
110
|
+
}
|
111
|
+
},
|
112
|
+
value: cell.raw,
|
113
|
+
placeholder: '',
|
114
|
+
rootElement: div,
|
115
|
+
schema: {
|
116
|
+
type: 'cell',
|
117
|
+
content: cell.raw,
|
118
|
+
position: { x: colWidth, y: rowOffsetY },
|
119
|
+
width: cell.width,
|
120
|
+
height: cell.height,
|
121
|
+
...convertToCellStyle(cell.styles),
|
122
|
+
},
|
123
|
+
});
|
124
|
+
colWidth += cell.width;
|
125
|
+
});
|
126
|
+
rowOffsetY += height;
|
127
|
+
});
|
128
|
+
};
|
129
|
+
|
130
|
+
const headEditingPosition = { rowIndex: -1, colIndex: -1 };
|
131
|
+
const bodyEditingPosition = { rowIndex: -1, colIndex: -1 };
|
132
|
+
const resetEditingPosition = () => {
|
133
|
+
headEditingPosition.rowIndex = -1;
|
134
|
+
headEditingPosition.colIndex = -1;
|
135
|
+
bodyEditingPosition.rowIndex = -1;
|
136
|
+
bodyEditingPosition.colIndex = -1;
|
137
|
+
};
|
138
|
+
|
139
|
+
export const uiRender = async (arg: UIRenderProps<TableSchema>) => {
|
140
|
+
const { rootElement, onChange, schema, value, mode } = arg;
|
141
|
+
const body = getBody(value);
|
142
|
+
const bodyWidthRange = getBodyWithRange(value, schema.__bodyRange);
|
143
|
+
const table = await createSingleTable(bodyWidthRange, arg);
|
144
|
+
|
145
|
+
rootElement.innerHTML = '';
|
146
|
+
|
147
|
+
rootElement.style.borderColor = schema.tableStyles.borderColor;
|
148
|
+
rootElement.style.borderWidth = String(schema.tableStyles.borderWidth) + 'mm';
|
149
|
+
rootElement.style.borderStyle = 'solid';
|
150
|
+
rootElement.style.boxSizing = 'border-box';
|
151
|
+
|
152
|
+
const handleChangeEditingPosition = (
|
153
|
+
newPosition: { rowIndex: number; colIndex: number },
|
154
|
+
editingPosition: { rowIndex: number; colIndex: number }
|
155
|
+
) => {
|
156
|
+
resetEditingPosition();
|
157
|
+
editingPosition.rowIndex = newPosition.rowIndex;
|
158
|
+
editingPosition.colIndex = newPosition.colIndex;
|
159
|
+
void uiRender(arg);
|
160
|
+
};
|
161
|
+
|
162
|
+
if (schema.showHead) {
|
163
|
+
renderRowUi({
|
164
|
+
rows: table.head,
|
165
|
+
arg,
|
166
|
+
editingPosition: headEditingPosition,
|
167
|
+
onChangeEditingPosition: (p) => handleChangeEditingPosition(p, headEditingPosition),
|
168
|
+
});
|
169
|
+
}
|
170
|
+
|
171
|
+
const offsetY = schema.showHead ? table.getHeadHeight() : 0;
|
172
|
+
renderRowUi({
|
173
|
+
rows: table.body,
|
174
|
+
arg,
|
175
|
+
editingPosition: bodyEditingPosition,
|
176
|
+
onChangeEditingPosition: (p) => {
|
177
|
+
handleChangeEditingPosition(p, bodyEditingPosition);
|
178
|
+
},
|
179
|
+
offsetY,
|
180
|
+
});
|
181
|
+
|
182
|
+
if (mode === 'form' && onChange) {
|
183
|
+
if (
|
184
|
+
schema.__bodyRange?.end === undefined ||
|
185
|
+
schema.__bodyRange.end >= (JSON.parse(value || '[]') as string[][]).length
|
186
|
+
) {
|
187
|
+
const addRowButton = document.createElement('button');
|
188
|
+
addRowButton.style.width = '30px';
|
189
|
+
addRowButton.style.height = '30px';
|
190
|
+
addRowButton.style.position = 'absolute';
|
191
|
+
addRowButton.style.top = `${table.getHeight()}mm`;
|
192
|
+
addRowButton.style.left = 'calc(50% - 15px)';
|
193
|
+
addRowButton.innerText = '+';
|
194
|
+
addRowButton.onclick = () => {
|
195
|
+
const newRow = Array(schema.head.length).fill('') as string[];
|
196
|
+
onChange({ key: 'content', value: JSON.stringify(body.concat([newRow])) });
|
197
|
+
};
|
198
|
+
rootElement.appendChild(addRowButton);
|
199
|
+
}
|
200
|
+
|
201
|
+
let offsetY = schema.showHead ? table.getHeadHeight() : 0;
|
202
|
+
table.body.forEach((row, i) => {
|
203
|
+
offsetY = offsetY + row.height;
|
204
|
+
const removeRowButton = document.createElement('button');
|
205
|
+
removeRowButton.style.width = '30px';
|
206
|
+
removeRowButton.style.height = '30px';
|
207
|
+
removeRowButton.style.position = 'absolute';
|
208
|
+
removeRowButton.style.top = `${offsetY - px2mm(30)}mm`;
|
209
|
+
removeRowButton.style.right = '-30px';
|
210
|
+
removeRowButton.innerText = '-';
|
211
|
+
removeRowButton.onclick = () => {
|
212
|
+
const newTableBody = body.filter((_, j) => j !== i + (schema.__bodyRange?.start ?? 0));
|
213
|
+
onChange({ key: 'content', value: JSON.stringify(newTableBody) });
|
214
|
+
};
|
215
|
+
rootElement.appendChild(removeRowButton);
|
216
|
+
});
|
217
|
+
}
|
218
|
+
|
219
|
+
if (mode === 'designer' && onChange) {
|
220
|
+
const addColumnButton = document.createElement('button');
|
221
|
+
addColumnButton.style.width = '30px';
|
222
|
+
addColumnButton.style.height = '30px';
|
223
|
+
addColumnButton.style.position = 'absolute';
|
224
|
+
addColumnButton.style.top = `${table.getHeadHeight() - px2mm(30)}mm`;
|
225
|
+
addColumnButton.style.right = '-30px';
|
226
|
+
addColumnButton.innerText = '+';
|
227
|
+
addColumnButton.onclick = (e) => {
|
228
|
+
e.preventDefault();
|
229
|
+
const newColumnWidthPercentage = 25;
|
230
|
+
const totalCurrentWidth = schema.headWidthPercentages.reduce((acc, width) => acc + width, 0);
|
231
|
+
const scalingRatio = (100 - newColumnWidthPercentage) / totalCurrentWidth;
|
232
|
+
const scaledWidths = schema.headWidthPercentages.map((width) => width * scalingRatio);
|
233
|
+
onChange([
|
234
|
+
{ key: 'head', value: schema.head.concat(`Head ${schema.head.length + 1}`) },
|
235
|
+
{ key: 'headWidthPercentages', value: scaledWidths.concat(newColumnWidthPercentage) },
|
236
|
+
{
|
237
|
+
key: 'content',
|
238
|
+
value: JSON.stringify(bodyWidthRange.map((row, i) => row.concat(`Row ${i + 1}`))),
|
239
|
+
},
|
240
|
+
]);
|
241
|
+
};
|
242
|
+
rootElement.appendChild(addColumnButton);
|
243
|
+
|
244
|
+
let offsetX = 0;
|
245
|
+
table.columns.forEach((column, i) => {
|
246
|
+
offsetX = offsetX + column.width;
|
247
|
+
const removeColumnButton = document.createElement('button');
|
248
|
+
removeColumnButton.style.width = '30px';
|
249
|
+
removeColumnButton.style.height = '30px';
|
250
|
+
removeColumnButton.style.position = 'absolute';
|
251
|
+
removeColumnButton.style.top = '-30px';
|
252
|
+
removeColumnButton.style.left = `${offsetX - px2mm(30)}mm`;
|
253
|
+
removeColumnButton.innerText = '-';
|
254
|
+
removeColumnButton.onclick = (e) => {
|
255
|
+
e.preventDefault();
|
256
|
+
const totalWidthMinusRemoved = schema.headWidthPercentages.reduce(
|
257
|
+
(sum, width, j) => (j !== i ? sum + width : sum),
|
258
|
+
0
|
259
|
+
);
|
260
|
+
|
261
|
+
// TODO Should also remove the deleted columnStyles when deleting
|
262
|
+
onChange([
|
263
|
+
{ key: 'head', value: schema.head.filter((_, j) => j !== i) },
|
264
|
+
{
|
265
|
+
key: 'headWidthPercentages',
|
266
|
+
value: schema.headWidthPercentages
|
267
|
+
.filter((_, j) => j !== i)
|
268
|
+
.map((width) => (width / totalWidthMinusRemoved) * 100),
|
269
|
+
},
|
270
|
+
{
|
271
|
+
key: 'content',
|
272
|
+
value: JSON.stringify(bodyWidthRange.map((row) => row.filter((_, j) => j !== i))),
|
273
|
+
},
|
274
|
+
]);
|
275
|
+
};
|
276
|
+
rootElement.appendChild(removeColumnButton);
|
277
|
+
|
278
|
+
if (i === table.columns.length - 1) return;
|
279
|
+
|
280
|
+
const dragHandle = document.createElement('div');
|
281
|
+
const lineWidth = 5;
|
282
|
+
dragHandle.style.width = `${lineWidth}px`;
|
283
|
+
dragHandle.style.height = '100%';
|
284
|
+
dragHandle.style.backgroundColor = '#eee';
|
285
|
+
dragHandle.style.opacity = '0.5';
|
286
|
+
dragHandle.style.cursor = 'col-resize';
|
287
|
+
dragHandle.style.position = 'absolute';
|
288
|
+
dragHandle.style.zIndex = '10';
|
289
|
+
dragHandle.style.left = `${offsetX - px2mm(lineWidth) / 2}mm`;
|
290
|
+
dragHandle.style.top = '0';
|
291
|
+
const setColor = (e: MouseEvent) => {
|
292
|
+
const handle = e.target as HTMLDivElement;
|
293
|
+
handle.style.backgroundColor = '#2196f3';
|
294
|
+
};
|
295
|
+
const resetColor = (e: MouseEvent) => {
|
296
|
+
const handle = e.target as HTMLDivElement;
|
297
|
+
handle.style.backgroundColor = '#eee';
|
298
|
+
};
|
299
|
+
dragHandle.addEventListener('mouseover', setColor);
|
300
|
+
dragHandle.addEventListener('mouseout', resetColor);
|
301
|
+
|
302
|
+
const prevColumnLeft = offsetX - column.width;
|
303
|
+
const nextColumnRight = offsetX - px2mm(lineWidth) + table.columns[i + 1].width;
|
304
|
+
|
305
|
+
dragHandle.addEventListener('mousedown', (e) => {
|
306
|
+
resetEditingPosition();
|
307
|
+
const handle = e.target as HTMLDivElement;
|
308
|
+
dragHandle.removeEventListener('mouseover', setColor);
|
309
|
+
dragHandle.removeEventListener('mouseout', resetColor);
|
310
|
+
|
311
|
+
let move = 0;
|
312
|
+
const mouseMove = (e: MouseEvent) => {
|
313
|
+
// TODO There is an issue where newLeft gets displaced with drag & drop
|
314
|
+
let moveX = e.movementX;
|
315
|
+
const currentLeft = Number(handle.style.left.replace('mm', ''));
|
316
|
+
let newLeft = currentLeft + moveX;
|
317
|
+
if (newLeft < prevColumnLeft) {
|
318
|
+
newLeft = prevColumnLeft;
|
319
|
+
moveX = newLeft - currentLeft;
|
320
|
+
}
|
321
|
+
if (newLeft >= nextColumnRight) {
|
322
|
+
newLeft = nextColumnRight;
|
323
|
+
moveX = newLeft - currentLeft;
|
324
|
+
}
|
325
|
+
handle.style.left = `${newLeft}mm`;
|
326
|
+
move += moveX;
|
327
|
+
};
|
328
|
+
rootElement.addEventListener('mousemove', mouseMove);
|
329
|
+
|
330
|
+
const commitResize = () => {
|
331
|
+
if (move !== 0) {
|
332
|
+
const newHeadWidthPercentages = calcResizedHeadWidthPercentages({
|
333
|
+
currentHeadWidthPercentages: schema.headWidthPercentages,
|
334
|
+
currentHeadWidths: table.columns.map((column) => column.width),
|
335
|
+
changedHeadWidth: table.columns[i].width + move,
|
336
|
+
changedHeadIndex: i,
|
337
|
+
});
|
338
|
+
onChange({ key: 'headWidthPercentages', value: newHeadWidthPercentages });
|
339
|
+
}
|
340
|
+
move = 0;
|
341
|
+
dragHandle.addEventListener('mouseover', setColor);
|
342
|
+
dragHandle.addEventListener('mouseout', resetColor);
|
343
|
+
rootElement.removeEventListener('mousemove', mouseMove);
|
344
|
+
rootElement.removeEventListener('mouseup', commitResize);
|
345
|
+
};
|
346
|
+
rootElement.addEventListener('mouseup', commitResize);
|
347
|
+
});
|
348
|
+
rootElement.appendChild(dragHandle);
|
349
|
+
});
|
350
|
+
}
|
351
|
+
|
352
|
+
if (mode === 'viewer') {
|
353
|
+
resetEditingPosition();
|
354
|
+
}
|
355
|
+
|
356
|
+
if (mode !== 'form') {
|
357
|
+
const tableHeight = schema.showHead ? table.getHeight() : table.getBodyHeight();
|
358
|
+
if (schema.height !== tableHeight && onChange) {
|
359
|
+
onChange({ key: 'height', value: tableHeight });
|
360
|
+
}
|
361
|
+
}
|
362
|
+
};
|
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')
|
@@ -224,7 +227,7 @@ export const calculateDynamicFontSize = async ({
|
|
224
227
|
if (dynamicFontSizeSetting.max < dynamicFontSizeSetting.min) return fontSize;
|
225
228
|
|
226
229
|
const characterSpacing = schemaCharacterSpacing ?? DEFAULT_CHARACTER_SPACING;
|
227
|
-
const fontKitFont = await getFontKitFont(textSchema, font, _cache);
|
230
|
+
const fontKitFont = await getFontKitFont(textSchema.fontName, font, _cache);
|
228
231
|
const paragraphs = value.split('\n');
|
229
232
|
|
230
233
|
let dynamicFontSize = fontSize;
|
@@ -318,3 +321,24 @@ export const calculateDynamicFontSize = async ({
|
|
318
321
|
|
319
322
|
return dynamicFontSize;
|
320
323
|
};
|
324
|
+
|
325
|
+
export const splitTextToSize = (arg: {
|
326
|
+
value: string;
|
327
|
+
characterSpacing: number;
|
328
|
+
boxWidthInPt: number;
|
329
|
+
fontSize: number;
|
330
|
+
fontKitFont: fontkit.Font;
|
331
|
+
}) => {
|
332
|
+
const { value, characterSpacing, fontSize, fontKitFont, boxWidthInPt } = arg;
|
333
|
+
const fontWidthCalcValues: FontWidthCalcValues = {
|
334
|
+
font: fontKitFont,
|
335
|
+
fontSize,
|
336
|
+
characterSpacing,
|
337
|
+
boxWidthInPt,
|
338
|
+
};
|
339
|
+
let lines: string[] = [];
|
340
|
+
value.split(/\r\n|\r|\n|\f|\u000B/g).forEach((line: string) => {
|
341
|
+
lines = lines.concat(getSplittedLines(line, fontWidthCalcValues));
|
342
|
+
});
|
343
|
+
return lines;
|
344
|
+
};
|
package/src/text/index.ts
CHANGED
package/src/text/pdfRender.ts
CHANGED
@@ -1,5 +1,5 @@
|
|
1
1
|
import { PDFFont, PDFDocument } from '@pdfme/pdf-lib';
|
2
|
-
import type { TextSchema
|
2
|
+
import type { TextSchema } from './types';
|
3
3
|
import { PDFRenderProps, Font, getDefaultFont, getFallbackFontName, mm2pt } from '@pdfme/common';
|
4
4
|
import {
|
5
5
|
VERTICAL_ALIGN_TOP,
|
@@ -17,8 +17,8 @@ import {
|
|
17
17
|
heightOfFontAtSize,
|
18
18
|
getFontDescentInPt,
|
19
19
|
getFontKitFont,
|
20
|
-
getSplittedLines,
|
21
20
|
widthOfTextAtSize,
|
21
|
+
splitTextToSize,
|
22
22
|
} from './helper.js';
|
23
23
|
import { convertForPdfLayoutProps, rotatePoint, hex2RgbColor } from '../utils.js';
|
24
24
|
|
@@ -87,7 +87,7 @@ export const pdfRender = async (arg: PDFRenderProps<TextSchema>) => {
|
|
87
87
|
|
88
88
|
const [pdfFontObj, fontKitFont, fontProp] = await Promise.all([
|
89
89
|
embedAndGetFontObj({ pdfDoc, font, _cache }),
|
90
|
-
getFontKitFont(schema, font, _cache),
|
90
|
+
getFontKitFont(schema.fontName, font, _cache),
|
91
91
|
getFontProp({ value, font, schema, _cache }),
|
92
92
|
]);
|
93
93
|
|
@@ -118,16 +118,12 @@ export const pdfRender = async (arg: PDFRenderProps<TextSchema>) => {
|
|
118
118
|
const descent = getFontDescentInPt(fontKitFont, fontSize);
|
119
119
|
const halfLineHeightAdjustment = lineHeight === 0 ? 0 : ((lineHeight - 1) * fontSize) / 2;
|
120
120
|
|
121
|
-
const
|
122
|
-
|
123
|
-
fontSize,
|
121
|
+
const lines = splitTextToSize({
|
122
|
+
value,
|
124
123
|
characterSpacing,
|
124
|
+
fontSize,
|
125
|
+
fontKitFont,
|
125
126
|
boxWidthInPt: width,
|
126
|
-
};
|
127
|
-
|
128
|
-
let lines: string[] = [];
|
129
|
-
value.split(/\r\n|\r|\n|\f|\u000B/g).forEach((line: string) => {
|
130
|
-
lines = lines.concat(getSplittedLines(line, fontWidthCalcValues));
|
131
127
|
});
|
132
128
|
|
133
129
|
// 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,9 @@ 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
|
+
content: 'Type Something...',
|
179
181
|
position: { x: 0, y: 0 },
|
180
182
|
width: 45,
|
181
183
|
height: 10,
|
package/src/text/uiRender.ts
CHANGED
@@ -18,7 +18,7 @@ import {
|
|
18
18
|
getFontKitFont,
|
19
19
|
getBrowserVerticalFontAdjustments,
|
20
20
|
} from './helper.js';
|
21
|
-
import {
|
21
|
+
import { isEditable } from '../utils.js';
|
22
22
|
|
23
23
|
const mapVerticalAlignToFlex = (verticalAlignmentValue: string | undefined) => {
|
24
24
|
switch (verticalAlignmentValue) {
|
@@ -32,10 +32,9 @@ const mapVerticalAlignToFlex = (verticalAlignmentValue: string | undefined) => {
|
|
32
32
|
return 'flex-start';
|
33
33
|
};
|
34
34
|
|
35
|
-
const getBackgroundColor = (value: string, schema: Schema
|
36
|
-
if (!value) return 'transparent';
|
37
|
-
|
38
|
-
return defaultBackgroundColor;
|
35
|
+
const getBackgroundColor = (value: string, schema: Schema) => {
|
36
|
+
if (!value || !schema.backgroundColor) return 'transparent';
|
37
|
+
return schema.backgroundColor as string;
|
39
38
|
};
|
40
39
|
|
41
40
|
export const uiRender = async (arg: UIRenderProps<TextSchema>) => {
|
@@ -49,7 +48,6 @@ export const uiRender = async (arg: UIRenderProps<TextSchema>) => {
|
|
49
48
|
tabIndex,
|
50
49
|
placeholder,
|
51
50
|
options,
|
52
|
-
theme,
|
53
51
|
_cache,
|
54
52
|
} = arg;
|
55
53
|
const font = options?.font || getDefaultFont();
|
@@ -66,7 +64,7 @@ export const uiRender = async (arg: UIRenderProps<TextSchema>) => {
|
|
66
64
|
dynamicFontSize = await calculateDynamicFontSize(getCdfArg(value));
|
67
65
|
}
|
68
66
|
|
69
|
-
const fontKitFont = await getFontKitFont(schema, font, _cache);
|
67
|
+
const fontKitFont = await getFontKitFont(schema.fontName, font, _cache);
|
70
68
|
// Depending on vertical alignment, we need to move the top or bottom of the font to keep
|
71
69
|
// it within it's defined box and align it with the generated pdf.
|
72
70
|
const { topAdj, bottomAdj } = getBrowserVerticalFontAdjustments(
|
@@ -84,13 +82,14 @@ export const uiRender = async (arg: UIRenderProps<TextSchema>) => {
|
|
84
82
|
const containerStyle: CSS.Properties = {
|
85
83
|
padding: 0,
|
86
84
|
resize: 'none',
|
87
|
-
backgroundColor: getBackgroundColor(value, schema
|
85
|
+
backgroundColor: getBackgroundColor(value, schema),
|
88
86
|
border: 'none',
|
89
87
|
display: 'flex',
|
90
88
|
flexDirection: 'column',
|
91
89
|
justifyContent: mapVerticalAlignToFlex(schema.verticalAlignment),
|
92
90
|
width: '100%',
|
93
91
|
height: '100%',
|
92
|
+
cursor: isEditable(mode, schema) ? 'text' : 'default',
|
94
93
|
};
|
95
94
|
Object.assign(container.style, containerStyle);
|
96
95
|
rootElement.innerHTML = '';
|
@@ -122,7 +121,7 @@ export const uiRender = async (arg: UIRenderProps<TextSchema>) => {
|
|
122
121
|
textBlock.tabIndex = tabIndex || 0;
|
123
122
|
textBlock.innerText = value;
|
124
123
|
textBlock.addEventListener('blur', (e: Event) => {
|
125
|
-
onChange && onChange((e.target as HTMLDivElement).innerText);
|
124
|
+
onChange && onChange({ key: 'content', value: (e.target as HTMLDivElement).innerText });
|
126
125
|
stopEditing && stopEditing();
|
127
126
|
});
|
128
127
|
|
@@ -166,15 +165,16 @@ export const uiRender = async (arg: UIRenderProps<TextSchema>) => {
|
|
166
165
|
container.appendChild(textBlock);
|
167
166
|
|
168
167
|
if (mode === 'designer') {
|
169
|
-
|
170
|
-
|
171
|
-
|
172
|
-
|
173
|
-
|
174
|
-
|
175
|
-
|
176
|
-
|
177
|
-
|
168
|
+
setTimeout(() => {
|
169
|
+
textBlock.focus();
|
170
|
+
// Set the focus to the end of the editable element when you focus, as we would for a textarea
|
171
|
+
const selection = window.getSelection();
|
172
|
+
const range = document.createRange();
|
173
|
+
range.selectNodeContents(textBlock);
|
174
|
+
range.collapse(false); // Collapse range to the end
|
175
|
+
selection?.removeAllRanges();
|
176
|
+
selection?.addRange(range);
|
177
|
+
});
|
178
178
|
}
|
179
179
|
} else {
|
180
180
|
textBlock.innerHTML = value
|
package/src/utils.ts
CHANGED
@@ -159,3 +159,11 @@ export const createErrorElm = () => {
|
|
159
159
|
|
160
160
|
return container;
|
161
161
|
};
|
162
|
+
|
163
|
+
export const px2mm = (px: number): number => {
|
164
|
+
// http://www.endmemo.com/sconvert/millimeterpixel.php
|
165
|
+
const ratio = 0.26458333333333;
|
166
|
+
return parseFloat(String(px)) * ratio;
|
167
|
+
};
|
168
|
+
|
169
|
+
export const cloneDeep = <T>(value: T): T => JSON.parse(JSON.stringify(value));
|