@pdfme/schemas 3.2.3 → 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.
Files changed (138) hide show
  1. package/dist/cjs/__tests__/text.test.js +5 -4
  2. package/dist/cjs/__tests__/text.test.js.map +1 -1
  3. package/dist/cjs/__tests__/utils.test.js +3 -0
  4. package/dist/cjs/__tests__/utils.test.js.map +1 -1
  5. package/dist/cjs/src/barcodes/propPanel.js +11 -11
  6. package/dist/cjs/src/barcodes/propPanel.js.map +1 -1
  7. package/dist/cjs/src/barcodes/uiRender.js +1 -1
  8. package/dist/cjs/src/barcodes/uiRender.js.map +1 -1
  9. package/dist/cjs/src/graphics/image.js +5 -11
  10. package/dist/cjs/src/graphics/image.js.map +1 -1
  11. package/dist/cjs/src/graphics/svg.js +2 -3
  12. package/dist/cjs/src/graphics/svg.js.map +1 -1
  13. package/dist/cjs/src/index.js +6 -1
  14. package/dist/cjs/src/index.js.map +1 -1
  15. package/dist/cjs/src/shapes/line.js +5 -9
  16. package/dist/cjs/src/shapes/line.js.map +1 -1
  17. package/dist/cjs/src/shapes/rectAndEllipse.js +3 -2
  18. package/dist/cjs/src/shapes/rectAndEllipse.js.map +1 -1
  19. package/dist/cjs/src/tables/cell.js +125 -0
  20. package/dist/cjs/src/tables/cell.js.map +1 -0
  21. package/dist/cjs/src/tables/classes.js +467 -0
  22. package/dist/cjs/src/tables/classes.js.map +1 -0
  23. package/dist/cjs/src/tables/dynamicTemplate.js +71 -0
  24. package/dist/cjs/src/tables/dynamicTemplate.js.map +1 -0
  25. package/dist/cjs/src/tables/helper.js +171 -0
  26. package/dist/cjs/src/tables/helper.js.map +1 -0
  27. package/dist/cjs/src/tables/index.js +12 -0
  28. package/dist/cjs/src/tables/index.js.map +1 -0
  29. package/dist/cjs/src/tables/pdfRender.js +95 -0
  30. package/dist/cjs/src/tables/pdfRender.js.map +1 -0
  31. package/dist/cjs/src/tables/propPanel.js +87 -0
  32. package/dist/cjs/src/tables/propPanel.js.map +1 -0
  33. package/dist/cjs/src/tables/tableHelper.js +231 -0
  34. package/dist/cjs/src/tables/tableHelper.js.map +1 -0
  35. package/dist/cjs/src/tables/types.js +3 -0
  36. package/dist/cjs/src/tables/types.js.map +1 -0
  37. package/dist/cjs/src/tables/uiRender.js +321 -0
  38. package/dist/cjs/src/tables/uiRender.js.map +1 -0
  39. package/dist/cjs/src/text/helper.js +21 -6
  40. package/dist/cjs/src/text/helper.js.map +1 -1
  41. package/dist/cjs/src/text/index.js +0 -1
  42. package/dist/cjs/src/text/index.js.map +1 -1
  43. package/dist/cjs/src/text/pdfRender.js +5 -8
  44. package/dist/cjs/src/text/pdfRender.js.map +1 -1
  45. package/dist/cjs/src/text/propPanel.js +6 -4
  46. package/dist/cjs/src/text/propPanel.js.map +1 -1
  47. package/dist/cjs/src/text/uiRender.js +18 -17
  48. package/dist/cjs/src/text/uiRender.js.map +1 -1
  49. package/dist/cjs/src/utils.js +9 -1
  50. package/dist/cjs/src/utils.js.map +1 -1
  51. package/dist/esm/__tests__/text.test.js +5 -4
  52. package/dist/esm/__tests__/text.test.js.map +1 -1
  53. package/dist/esm/__tests__/utils.test.js +3 -0
  54. package/dist/esm/__tests__/utils.test.js.map +1 -1
  55. package/dist/esm/src/barcodes/propPanel.js +11 -11
  56. package/dist/esm/src/barcodes/propPanel.js.map +1 -1
  57. package/dist/esm/src/barcodes/uiRender.js +1 -1
  58. package/dist/esm/src/barcodes/uiRender.js.map +1 -1
  59. package/dist/esm/src/graphics/image.js +4 -10
  60. package/dist/esm/src/graphics/image.js.map +1 -1
  61. package/dist/esm/src/graphics/svg.js +2 -3
  62. package/dist/esm/src/graphics/svg.js.map +1 -1
  63. package/dist/esm/src/index.js +3 -1
  64. package/dist/esm/src/index.js.map +1 -1
  65. package/dist/esm/src/shapes/line.js +5 -9
  66. package/dist/esm/src/shapes/line.js.map +1 -1
  67. package/dist/esm/src/shapes/rectAndEllipse.js +3 -2
  68. package/dist/esm/src/shapes/rectAndEllipse.js.map +1 -1
  69. package/dist/esm/src/tables/cell.js +120 -0
  70. package/dist/esm/src/tables/cell.js.map +1 -0
  71. package/dist/esm/src/tables/classes.js +460 -0
  72. package/dist/esm/src/tables/classes.js.map +1 -0
  73. package/dist/esm/src/tables/dynamicTemplate.js +66 -0
  74. package/dist/esm/src/tables/dynamicTemplate.js.map +1 -0
  75. package/dist/esm/src/tables/helper.js +163 -0
  76. package/dist/esm/src/tables/helper.js.map +1 -0
  77. package/dist/esm/src/tables/index.js +10 -0
  78. package/dist/esm/src/tables/index.js.map +1 -0
  79. package/dist/esm/src/tables/pdfRender.js +88 -0
  80. package/dist/esm/src/tables/pdfRender.js.map +1 -0
  81. package/dist/esm/src/tables/propPanel.js +84 -0
  82. package/dist/esm/src/tables/propPanel.js.map +1 -0
  83. package/dist/esm/src/tables/tableHelper.js +226 -0
  84. package/dist/esm/src/tables/tableHelper.js.map +1 -0
  85. package/dist/esm/src/tables/types.js +2 -0
  86. package/dist/esm/src/tables/types.js.map +1 -0
  87. package/dist/esm/src/tables/uiRender.js +314 -0
  88. package/dist/esm/src/tables/uiRender.js.map +1 -0
  89. package/dist/esm/src/text/helper.js +19 -5
  90. package/dist/esm/src/text/helper.js.map +1 -1
  91. package/dist/esm/src/text/index.js +0 -1
  92. package/dist/esm/src/text/index.js.map +1 -1
  93. package/dist/esm/src/text/pdfRender.js +6 -9
  94. package/dist/esm/src/text/pdfRender.js.map +1 -1
  95. package/dist/esm/src/text/propPanel.js +6 -4
  96. package/dist/esm/src/text/propPanel.js.map +1 -1
  97. package/dist/esm/src/text/uiRender.js +19 -18
  98. package/dist/esm/src/text/uiRender.js.map +1 -1
  99. package/dist/esm/src/utils.js +6 -0
  100. package/dist/esm/src/utils.js.map +1 -1
  101. package/dist/types/src/index.d.ts +3 -1
  102. package/dist/types/src/shapes/rectAndEllipse.d.ts +13 -27
  103. package/dist/types/src/tables/cell.d.ts +4 -0
  104. package/dist/types/src/tables/classes.d.ts +68 -0
  105. package/dist/types/src/tables/dynamicTemplate.d.ts +13 -0
  106. package/dist/types/src/tables/helper.d.ts +252 -0
  107. package/dist/types/src/tables/index.d.ts +4 -0
  108. package/dist/types/src/tables/pdfRender.d.ts +3 -0
  109. package/dist/types/src/tables/propPanel.d.ts +3 -0
  110. package/dist/types/src/tables/tableHelper.d.ts +11 -0
  111. package/dist/types/src/tables/types.d.ts +91 -0
  112. package/dist/types/src/tables/uiRender.d.ts +3 -0
  113. package/dist/types/src/text/helper.d.ts +8 -1
  114. package/dist/types/src/utils.d.ts +2 -0
  115. package/package.json +1 -1
  116. package/src/barcodes/propPanel.ts +12 -12
  117. package/src/barcodes/uiRender.ts +1 -1
  118. package/src/graphics/image.ts +6 -11
  119. package/src/graphics/svg.ts +2 -3
  120. package/src/index.ts +5 -0
  121. package/src/shapes/line.ts +4 -9
  122. package/src/shapes/rectAndEllipse.ts +4 -4
  123. package/src/tables/cell.ts +157 -0
  124. package/src/tables/classes.ts +398 -0
  125. package/src/tables/dynamicTemplate.ts +81 -0
  126. package/src/tables/helper.ts +198 -0
  127. package/src/tables/index.ts +12 -0
  128. package/src/tables/pdfRender.ts +115 -0
  129. package/src/tables/propPanel.ts +90 -0
  130. package/src/tables/tableHelper.ts +322 -0
  131. package/src/tables/types.ts +88 -0
  132. package/src/tables/uiRender.ts +362 -0
  133. package/src/text/helper.ts +30 -6
  134. package/src/text/index.ts +0 -1
  135. package/src/text/pdfRender.ts +7 -11
  136. package/src/text/propPanel.ts +6 -4
  137. package/src/text/uiRender.ts +18 -18
  138. 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
+ };
@@ -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 (textSchema: TextSchema, font: Font, _cache: Map<any, any>) => {
111
- const fontName = textSchema.fontName || getFallbackFontName(font);
112
- const cacheKey = getCacheKey(fontName);
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
@@ -17,7 +17,6 @@ export const readOnlyText: Plugin<TextSchema> = {
17
17
  ...textSchema.propPanel.defaultSchema,
18
18
  type: 'readOnlyText',
19
19
  readOnly: true,
20
- readOnlyValue: textSchema.propPanel.defaultValue,
21
20
  },
22
21
  },
23
22
  };
@@ -1,5 +1,5 @@
1
1
  import { PDFFont, PDFDocument } from '@pdfme/pdf-lib';
2
- import type { TextSchema, FontWidthCalcValues } from './types';
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 fontWidthCalcValues: FontWidthCalcValues = {
122
- font: fontKitFont,
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
@@ -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,
@@ -18,7 +18,7 @@ import {
18
18
  getFontKitFont,
19
19
  getBrowserVerticalFontAdjustments,
20
20
  } from './helper.js';
21
- import { addAlphaToHex, isEditable } from '../utils.js';
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, defaultBackgroundColor: string) => {
36
- if (!value) return 'transparent';
37
- if (schema.backgroundColor) return schema.backgroundColor as string;
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, addAlphaToHex(theme.colorPrimaryBg, 30)),
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
- textBlock.focus();
170
-
171
- // Set the focus to the end of the editable element when you focus, as we would for a textarea
172
- const selection = window.getSelection();
173
- const range = document.createRange();
174
- range.selectNodeContents(textBlock);
175
- range.collapse(false); // Collapse range to the end
176
- selection?.removeAllRanges();
177
- selection?.addRange(range);
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));