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