@pdfme/schemas 3.1.3 → 3.1.5

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 (66) hide show
  1. package/dist/cjs/__tests__/text.test.js +47 -25
  2. package/dist/cjs/__tests__/text.test.js.map +1 -1
  3. package/dist/cjs/src/barcodes/constants.js +1 -2
  4. package/dist/cjs/src/barcodes/constants.js.map +1 -1
  5. package/dist/cjs/src/barcodes/propPanel.js +34 -13
  6. package/dist/cjs/src/barcodes/propPanel.js.map +1 -1
  7. package/dist/cjs/src/constants.js +6 -0
  8. package/dist/cjs/src/constants.js.map +1 -0
  9. package/dist/cjs/src/image/propPanel.js +2 -2
  10. package/dist/cjs/src/image/propPanel.js.map +1 -1
  11. package/dist/cjs/src/text/constants.js +2 -2
  12. package/dist/cjs/src/text/constants.js.map +1 -1
  13. package/dist/cjs/src/text/helper.js +8 -7
  14. package/dist/cjs/src/text/helper.js.map +1 -1
  15. package/dist/cjs/src/text/pdfRender.js +14 -11
  16. package/dist/cjs/src/text/pdfRender.js.map +1 -1
  17. package/dist/cjs/src/text/propPanel.js +24 -3
  18. package/dist/cjs/src/text/propPanel.js.map +1 -1
  19. package/dist/cjs/src/text/uiRender.js +72 -42
  20. package/dist/cjs/src/text/uiRender.js.map +1 -1
  21. package/dist/esm/__tests__/text.test.js +47 -25
  22. package/dist/esm/__tests__/text.test.js.map +1 -1
  23. package/dist/esm/src/barcodes/constants.js +0 -1
  24. package/dist/esm/src/barcodes/constants.js.map +1 -1
  25. package/dist/esm/src/barcodes/propPanel.js +24 -3
  26. package/dist/esm/src/barcodes/propPanel.js.map +1 -1
  27. package/dist/esm/src/constants.js +3 -0
  28. package/dist/esm/src/constants.js.map +1 -0
  29. package/dist/esm/src/image/propPanel.js +1 -1
  30. package/dist/esm/src/image/propPanel.js.map +1 -1
  31. package/dist/esm/src/text/constants.js +1 -1
  32. package/dist/esm/src/text/constants.js.map +1 -1
  33. package/dist/esm/src/text/helper.js +8 -7
  34. package/dist/esm/src/text/helper.js.map +1 -1
  35. package/dist/esm/src/text/pdfRender.js +15 -12
  36. package/dist/esm/src/text/pdfRender.js.map +1 -1
  37. package/dist/esm/src/text/propPanel.js +24 -3
  38. package/dist/esm/src/text/propPanel.js.map +1 -1
  39. package/dist/esm/src/text/uiRender.js +73 -43
  40. package/dist/esm/src/text/uiRender.js.map +1 -1
  41. package/dist/types/src/barcodes/constants.d.ts +0 -1
  42. package/dist/types/src/constants.d.ts +2 -0
  43. package/dist/types/src/text/constants.d.ts +1 -1
  44. package/dist/types/src/text/helper.d.ts +4 -3
  45. package/package.json +1 -1
  46. package/src/barcodes/constants.ts +0 -2
  47. package/src/barcodes/propPanel.ts +24 -3
  48. package/src/constants.ts +2 -0
  49. package/src/image/propPanel.ts +1 -1
  50. package/src/text/constants.ts +1 -1
  51. package/src/text/helper.ts +11 -7
  52. package/src/text/pdfRender.ts +31 -13
  53. package/src/text/propPanel.ts +23 -3
  54. package/src/text/uiRender.ts +83 -46
  55. package/dist/cjs/src/image/constants.js +0 -5
  56. package/dist/cjs/src/image/constants.js.map +0 -1
  57. package/dist/cjs/src/image/helper.js +0 -3
  58. package/dist/cjs/src/image/helper.js.map +0 -1
  59. package/dist/esm/src/image/constants.js +0 -2
  60. package/dist/esm/src/image/constants.js.map +0 -1
  61. package/dist/esm/src/image/helper.js +0 -2
  62. package/dist/esm/src/image/helper.js.map +0 -1
  63. package/dist/types/src/image/constants.d.ts +0 -1
  64. package/dist/types/src/image/helper.d.ts +0 -1
  65. package/src/image/constants.ts +0 -1
  66. package/src/image/helper.ts +0 -1
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@pdfme/schemas",
3
- "version": "3.1.3",
3
+ "version": "3.1.5",
4
4
  "sideEffects": false,
5
5
  "author": "hand-dot",
6
6
  "license": "MIT",
@@ -15,5 +15,3 @@ export const BARCODE_TYPES = [
15
15
  export const DEFAULT_BARCODE_BG_COLOR = '#ffffff';
16
16
 
17
17
  export const DEFAULT_BARCODE_COLOR = '#000000';
18
-
19
- export const DEFAULT_OPACITY = 1;
@@ -1,6 +1,7 @@
1
1
  import type { PropPanel } from '@pdfme/common';
2
2
  import type { BarcodeSchema } from './types';
3
- import { DEFAULT_BARCODE_COLOR, DEFAULT_BARCODE_BG_COLOR, DEFAULT_OPACITY } from './constants.js';
3
+ import { DEFAULT_BARCODE_COLOR, DEFAULT_BARCODE_BG_COLOR } from './constants.js';
4
+ import { DEFAULT_OPACITY, HEX_COLOR_PATTERN } from '../constants.js';
4
5
 
5
6
  const defaultColors = {
6
7
  backgroundColor: DEFAULT_BARCODE_BG_COLOR,
@@ -158,8 +159,28 @@ export const getPropPanelByBarcodeType = (barcodeType: string): PropPanel<Barcod
158
159
  return {
159
160
  schema: ({ i18n }) => {
160
161
  return {
161
- barColor: { title: i18n('schemas.barcodes.barColor'), type: 'string', widget: 'color' },
162
- backgroundColor: { title: i18n('schemas.bgColor'), type: 'string', widget: 'color' },
162
+ barColor: {
163
+ title: i18n('schemas.barcodes.barColor'),
164
+ type: 'string',
165
+ widget: 'color',
166
+ rules: [
167
+ {
168
+ pattern: HEX_COLOR_PATTERN,
169
+ message: 'Please enter a valid hex color code.',
170
+ },
171
+ ],
172
+ },
173
+ backgroundColor: {
174
+ title: i18n('schemas.bgColor'),
175
+ type: 'string',
176
+ widget: 'color',
177
+ rules: [
178
+ {
179
+ pattern: HEX_COLOR_PATTERN,
180
+ message: 'Please enter a valid hex color code.',
181
+ },
182
+ ],
183
+ },
163
184
  ...(barcodeHasText
164
185
  ? { textColor: { title: i18n('schemas.textColor'), type: 'string', widget: 'color' } }
165
186
  : {}),
@@ -0,0 +1,2 @@
1
+ export const DEFAULT_OPACITY = 1;
2
+ export const HEX_COLOR_PATTERN = '^#(?:[A-Fa-f0-9]{3,4}|[A-Fa-f0-9]{6}|[A-Fa-f0-9]{8})$';
@@ -1,6 +1,6 @@
1
1
  import { PropPanel } from '@pdfme/common';
2
2
  import type { ImageSchema } from './types';
3
- import { DEFAULT_OPACITY } from '../image/constants';
3
+ import { DEFAULT_OPACITY } from '../constants.js';
4
4
 
5
5
  export const propPanel: PropPanel<ImageSchema> = {
6
6
  schema: {},
@@ -13,6 +13,7 @@ export const DEFAULT_VERTICAL_ALIGNMENT = VERTICAL_ALIGN_TOP;
13
13
  export const DEFAULT_LINE_HEIGHT = 1;
14
14
  export const DEFAULT_CHARACTER_SPACING = 0;
15
15
  export const DEFAULT_FONT_COLOR = '#000000';
16
+ export const PLACEHOLDER_FONT_COLOR = '#A0A0A0';
16
17
  export const DYNAMIC_FIT_VERTICAL = 'vertical' as DYNAMIC_FONT_SIZE_FIT;
17
18
  export const DYNAMIC_FIT_HORIZONTAL = 'horizontal' as DYNAMIC_FONT_SIZE_FIT;
18
19
  export const DEFAULT_DYNAMIC_FIT = DYNAMIC_FIT_VERTICAL;
@@ -20,4 +21,3 @@ export const DEFAULT_DYNAMIC_MIN_FONT_SIZE = 4;
20
21
 
21
22
  export const DEFAULT_DYNAMIC_MAX_FONT_SIZE = 72;
22
23
  export const FONT_SIZE_ADJUSTMENT = 0.25;
23
- export const DEFAULT_OPACITY = 1;
@@ -106,11 +106,13 @@ const getFallbackFont = (font: Font) => {
106
106
  return font[fallbackFontName];
107
107
  };
108
108
 
109
- const fontKitFontCache: { [fontName: string]: FontKitFont } = {};
110
- export const getFontKitFont = async (textSchema: Schema, font: Font) => {
111
- const fontName = (textSchema.fontName as string) || getFallbackFontName(font);
112
- if (fontKitFontCache[fontName]) {
113
- return fontKitFontCache[fontName];
109
+ const getCacheKey = (fontName: string) => `getFontKitFont-${fontName}`;
110
+
111
+ export const getFontKitFont = async (textSchema: TextSchema, font: Font, _cache: Map<any, any>) => {
112
+ const fontName = textSchema.fontName || getFallbackFontName(font);
113
+ const cacheKey = getCacheKey(fontName);
114
+ if (_cache.has(cacheKey)) {
115
+ return _cache.get(cacheKey) as fontkit.Font;
114
116
  }
115
117
 
116
118
  const currentFont =
@@ -125,7 +127,7 @@ export const getFontKitFont = async (textSchema: Schema, font: Font) => {
125
127
  const fontKitFont = fontkit.create(
126
128
  fontData instanceof Buffer ? fontData : Buffer.from(fontData as ArrayBuffer)
127
129
  );
128
- fontKitFontCache[fontName] = fontKitFont;
130
+ _cache.set(cacheKey, fontKitFont);
129
131
 
130
132
  return fontKitFont;
131
133
  };
@@ -202,11 +204,13 @@ export const calculateDynamicFontSize = async ({
202
204
  font,
203
205
  value,
204
206
  startingFontSize,
207
+ _cache,
205
208
  }: {
206
209
  textSchema: TextSchema;
207
210
  font: Font;
208
211
  value: string;
209
212
  startingFontSize?: number | undefined;
213
+ _cache: Map<any, any>;
210
214
  }) => {
211
215
  const {
212
216
  fontSize: schemaFontSize,
@@ -221,7 +225,7 @@ export const calculateDynamicFontSize = async ({
221
225
  if (dynamicFontSizeSetting.max < dynamicFontSizeSetting.min) return fontSize;
222
226
 
223
227
  const characterSpacing = schemaCharacterSpacing ?? DEFAULT_CHARACTER_SPACING;
224
- const fontKitFont = await getFontKitFont(textSchema, font);
228
+ const fontKitFont = await getFontKitFont(textSchema, font, _cache);
225
229
  const paragraphs = value.split('\n');
226
230
 
227
231
  let dynamicFontSize = fontSize;
@@ -1,5 +1,12 @@
1
1
  import { PDFFont, PDFDocument, rgb } from '@pdfme/pdf-lib';
2
- import { PDFRenderProps, Font, getDefaultFont, getFallbackFontName, mm2pt } from '@pdfme/common';
2
+ import {
3
+ PDFRenderProps,
4
+ Font,
5
+ getDefaultFont,
6
+ getFallbackFontName,
7
+ mm2pt,
8
+ isHexValid,
9
+ } from '@pdfme/common';
3
10
  import type { TextSchema, FontWidthCalcValues } from './types';
4
11
  import {
5
12
  VERTICAL_ALIGN_TOP,
@@ -38,6 +45,12 @@ const hex2rgb = (hex: string) => {
38
45
 
39
46
  const hex2RgbColor = (hexString: string | undefined) => {
40
47
  if (hexString) {
48
+ const isValid = isHexValid(hexString);
49
+
50
+ if (!isValid) {
51
+ throw new Error(`Invalid hex color value ${hexString}`);
52
+ }
53
+
41
54
  const [r, g, b] = hex2rgb(hexString);
42
55
 
43
56
  return rgb(r / 255, g / 255, b / 255);
@@ -46,11 +59,14 @@ const hex2RgbColor = (hexString: string | undefined) => {
46
59
  return undefined;
47
60
  };
48
61
 
49
- const embedAndGetFontObjCache = new WeakMap();
50
- const embedAndGetFontObj = async (arg: { pdfDoc: PDFDocument; font: Font }) => {
51
- const { pdfDoc, font } = arg;
52
- if (embedAndGetFontObjCache.has(pdfDoc)) {
53
- return embedAndGetFontObjCache.get(pdfDoc);
62
+ const embedAndGetFontObj = async (arg: {
63
+ pdfDoc: PDFDocument;
64
+ font: Font;
65
+ _cache: Map<any, any>;
66
+ }) => {
67
+ const { pdfDoc, font, _cache } = arg;
68
+ if (_cache.has(pdfDoc)) {
69
+ return _cache.get(pdfDoc) as { [key: string]: PDFFont };
54
70
  }
55
71
 
56
72
  const fontValues = await Promise.all(
@@ -70,7 +86,7 @@ const embedAndGetFontObj = async (arg: { pdfDoc: PDFDocument; font: Font }) => {
70
86
  {} as { [key: string]: PDFFont }
71
87
  );
72
88
 
73
- embedAndGetFontObjCache.set(pdfDoc, fontObj);
89
+ _cache.set(pdfDoc, fontObj);
74
90
  return fontObj;
75
91
  };
76
92
 
@@ -78,13 +94,15 @@ const getFontProp = async ({
78
94
  value,
79
95
  font,
80
96
  schema,
97
+ _cache,
81
98
  }: {
82
99
  value: string;
83
100
  font: Font;
84
101
  schema: TextSchema;
102
+ _cache: Map<any, any>;
85
103
  }) => {
86
104
  const fontSize = schema.dynamicFontSize
87
- ? await calculateDynamicFontSize({ textSchema: schema, font, value })
105
+ ? await calculateDynamicFontSize({ textSchema: schema, font, value, _cache })
88
106
  : schema.fontSize ?? DEFAULT_FONT_SIZE;
89
107
  const color = hex2RgbColor(schema.fontColor || DEFAULT_FONT_COLOR);
90
108
 
@@ -99,14 +117,14 @@ const getFontProp = async ({
99
117
  };
100
118
 
101
119
  export const pdfRender = async (arg: PDFRenderProps<TextSchema>) => {
102
- const { value, pdfDoc, pdfLib, page, options, schema } = arg;
120
+ const { value, pdfDoc, pdfLib, page, options, schema, _cache } = arg;
103
121
 
104
122
  const { font = getDefaultFont() } = options;
105
123
 
106
124
  const [pdfFontObj, fontKitFont, fontProp] = await Promise.all([
107
- embedAndGetFontObj({ pdfDoc, font }),
108
- getFontKitFont(schema, font),
109
- getFontProp({ value, font, schema }),
125
+ embedAndGetFontObj({ pdfDoc, font, _cache }),
126
+ getFontKitFont(schema, font, _cache),
127
+ getFontProp({ value, font, schema, _cache }),
110
128
  ]);
111
129
 
112
130
  const { fontSize, color, alignment, verticalAlignment, lineHeight, characterSpacing } = fontProp;
@@ -126,7 +144,7 @@ export const pdfRender = async (arg: PDFRenderProps<TextSchema>) => {
126
144
  } = convertForPdfLayoutProps({ schema, pageHeight, applyRotateTranslate: false });
127
145
 
128
146
  if (schema.backgroundColor) {
129
- const color = hex2RgbColor(schema.backgroundColor as string);
147
+ const color = hex2RgbColor(schema.backgroundColor);
130
148
  page.drawRectangle({ x, y, width, height, rotate, color });
131
149
  }
132
150
 
@@ -23,8 +23,8 @@ import {
23
23
  DEFAULT_DYNAMIC_MAX_FONT_SIZE,
24
24
  ALIGN_RIGHT,
25
25
  ALIGN_CENTER,
26
- DEFAULT_OPACITY,
27
26
  } from './constants.js';
27
+ import { DEFAULT_OPACITY, HEX_COLOR_PATTERN } from '../constants.js';
28
28
 
29
29
  const UseDynamicFontSize = (props: PropPanelWidgetProps) => {
30
30
  const { rootElement, changeSchemas, activeSchema, i18n } = props;
@@ -146,8 +146,28 @@ export const propPanel: PropPanel<TextSchema> = {
146
146
  },
147
147
  },
148
148
  },
149
- fontColor: { title: i18n('schemas.textColor'), type: 'string', widget: 'color' },
150
- backgroundColor: { title: i18n('schemas.bgColor'), type: 'string', widget: 'color' },
149
+ fontColor: {
150
+ title: i18n('schemas.textColor'),
151
+ type: 'string',
152
+ widget: 'color',
153
+ rules: [
154
+ {
155
+ pattern: HEX_COLOR_PATTERN,
156
+ message: 'Please enter a valid hex color code.',
157
+ },
158
+ ],
159
+ },
160
+ backgroundColor: {
161
+ title: i18n('schemas.bgColor'),
162
+ type: 'string',
163
+ widget: 'color',
164
+ rules: [
165
+ {
166
+ pattern: HEX_COLOR_PATTERN,
167
+ message: 'Please enter a valid hex color code.',
168
+ },
169
+ ],
170
+ },
151
171
  };
152
172
 
153
173
  return textSchema;
@@ -11,7 +11,7 @@ import {
11
11
  DEFAULT_LINE_HEIGHT,
12
12
  DEFAULT_CHARACTER_SPACING,
13
13
  DEFAULT_FONT_COLOR,
14
- DEFAULT_OPACITY,
14
+ PLACEHOLDER_FONT_COLOR,
15
15
  } from './constants.js';
16
16
  import {
17
17
  calculateDynamicFontSize,
@@ -19,6 +19,7 @@ import {
19
19
  getBrowserVerticalFontAdjustments,
20
20
  } from './helper.js';
21
21
  import { addAlphaToHex } from '../renderUtils.js';
22
+ import { DEFAULT_OPACITY } from '../constants.js';
22
23
 
23
24
  const mapVerticalAlignToFlex = (verticalAlignmentValue: string | undefined) => {
24
25
  switch (verticalAlignmentValue) {
@@ -59,20 +60,23 @@ export const uiRender = async (arg: UIRenderProps<TextSchema>) => {
59
60
  placeholder,
60
61
  options,
61
62
  theme,
63
+ _cache,
62
64
  } = arg;
63
65
  const font = options?.font || getDefaultFont();
64
66
 
65
67
  let dynamicFontSize: undefined | number = undefined;
68
+ const getCdfArg = (v: string) => ({
69
+ textSchema: schema,
70
+ font,
71
+ value: v,
72
+ startingFontSize: dynamicFontSize,
73
+ _cache,
74
+ });
66
75
  if (schema.dynamicFontSize && value) {
67
- dynamicFontSize = await calculateDynamicFontSize({
68
- textSchema: schema,
69
- font,
70
- value,
71
- startingFontSize: dynamicFontSize,
72
- });
76
+ dynamicFontSize = await calculateDynamicFontSize(getCdfArg(value));
73
77
  }
74
78
 
75
- const fontKitFont = await getFontKitFont(schema, font);
79
+ const fontKitFont = await getFontKitFont(schema, font, _cache);
76
80
  // Depending on vertical alignment, we need to move the top or bottom of the font to keep
77
81
  // it within it's defined box and align it with the generated pdf.
78
82
  const { topAdj, bottomAdj } = getBrowserVerticalFontAdjustments(
@@ -82,8 +86,8 @@ export const uiRender = async (arg: UIRenderProps<TextSchema>) => {
82
86
  schema.verticalAlignment ?? DEFAULT_VERTICAL_ALIGNMENT
83
87
  );
84
88
 
85
- const topAdjustment = topAdj;
86
- const bottomAdjustment = bottomAdj;
89
+ const topAdjustment = topAdj.toString();
90
+ const bottomAdjustment = bottomAdj.toString();
87
91
 
88
92
  const container = document.createElement('div');
89
93
 
@@ -102,13 +106,14 @@ export const uiRender = async (arg: UIRenderProps<TextSchema>) => {
102
106
  justifyContent: mapVerticalAlignToFlex(schema.verticalAlignment),
103
107
  width: '100%',
104
108
  height: '100%',
105
- opacity: schema.opacity,
109
+ opacity: schema.opacity ?? DEFAULT_OPACITY,
106
110
  };
107
111
  Object.assign(container.style, containerStyle);
108
112
  rootElement.innerHTML = '';
109
113
  rootElement.appendChild(container);
110
114
 
111
- const fontStyles: CSS.Properties = {
115
+ const textBlockStyle: CSS.Properties = {
116
+ // Font formatting styles
112
117
  fontFamily: schema.fontName ? `'${schema.fontName}'` : 'inherit',
113
118
  color: schema.fontColor ? schema.fontColor : DEFAULT_FONT_COLOR,
114
119
  fontSize: `${dynamicFontSize ?? schema.fontSize ?? DEFAULT_FONT_SIZE}pt`,
@@ -117,45 +122,77 @@ export const uiRender = async (arg: UIRenderProps<TextSchema>) => {
117
122
  textAlign: schema.alignment ?? DEFAULT_ALIGNMENT,
118
123
  whiteSpace: 'pre-wrap',
119
124
  wordBreak: 'break-word',
125
+ // Block layout styles
126
+ resize: 'none',
127
+ border: 'none',
128
+ outline: 'none',
129
+ marginBottom: bottomAdjustment + 'px',
130
+ paddingTop: topAdjustment + 'px',
131
+ backgroundColor: 'transparent',
120
132
  };
133
+ const textBlock = document.createElement('div');
134
+ Object.assign(textBlock.style, textBlockStyle);
121
135
 
122
136
  if (mode === 'form' || mode === 'designer') {
123
- const textarea = document.createElement('textarea');
124
- const textareaStyle: CSS.Properties = {
125
- padding: 0,
126
- resize: 'none',
127
- border: 'none',
128
- outline: 'none',
129
- paddingTop: topAdjustment + 'px',
130
- backgroundColor: 'transparent',
131
- width: '100%',
132
- height: '100%',
133
- };
134
- Object.assign(textarea.style, textareaStyle, fontStyles);
135
- textarea.rows = 1;
136
- textarea.placeholder = placeholder || '';
137
- textarea.tabIndex = tabIndex || 0;
138
-
139
- textarea.addEventListener(
140
- 'change',
141
- (e: Event) => onChange && onChange((e.target as HTMLTextAreaElement).value)
142
- );
143
- textarea.addEventListener('blur', () => stopEditing && stopEditing());
144
- textarea.value = value;
145
- container.appendChild(textarea);
137
+ textBlock.contentEditable = 'plaintext-only';
138
+ textBlock.tabIndex = tabIndex || 0;
139
+ textBlock.innerText = value;
140
+ textBlock.addEventListener('blur', (e: Event) => {
141
+ onChange && onChange((e.target as HTMLDivElement).innerText);
142
+ stopEditing && stopEditing();
143
+ });
144
+
145
+ if (schema.dynamicFontSize) {
146
+ textBlock.addEventListener('keyup', () => {
147
+ setTimeout(() => {
148
+ void (async () => {
149
+ if (!textBlock.textContent) return;
150
+ dynamicFontSize = await calculateDynamicFontSize(getCdfArg(textBlock.textContent));
151
+ textBlock.style.fontSize = `${dynamicFontSize}pt`;
152
+
153
+ const { topAdj: newTopAdj, bottomAdj: newBottomAdj } = getBrowserVerticalFontAdjustments(
154
+ fontKitFont,
155
+ dynamicFontSize ?? schema.fontSize ?? DEFAULT_FONT_SIZE,
156
+ schema.lineHeight ?? DEFAULT_LINE_HEIGHT,
157
+ schema.verticalAlignment ?? DEFAULT_VERTICAL_ALIGNMENT
158
+ );
159
+ textBlock.style.paddingTop = newTopAdj.toString() + 'px';
160
+ textBlock.style.marginBottom = newBottomAdj.toString() + 'px';
161
+ })();
162
+ }, 0);
163
+ });
164
+ }
165
+
166
+ if (placeholder && !value) {
167
+ textBlock.innerText = placeholder;
168
+ textBlock.style.color = PLACEHOLDER_FONT_COLOR;
169
+ if (schema.dynamicFontSize) {
170
+ const fontSize = await calculateDynamicFontSize(getCdfArg(placeholder));
171
+ textBlock.style.fontSize = `${fontSize}pt`;
172
+ }
173
+ textBlock.addEventListener('focus', () => {
174
+ if (textBlock.innerText === placeholder) {
175
+ textBlock.innerText = '';
176
+ textBlock.style.color = schema.fontColor ?? DEFAULT_FONT_COLOR;
177
+ }
178
+ });
179
+ }
180
+
181
+ container.appendChild(textBlock);
182
+
146
183
  if (mode === 'designer') {
147
- textarea.setSelectionRange(value.length, value.length);
148
- textarea.focus();
184
+ textBlock.focus();
185
+
186
+ // Set the focus to the end of the editable element when you focus, as we would for a textarea
187
+ const selection = window.getSelection();
188
+ const range = document.createRange();
189
+ range.selectNodeContents(textBlock);
190
+ range.collapse(false); // Collapse range to the end
191
+ selection?.removeAllRanges();
192
+ selection?.addRange(range);
149
193
  }
150
194
  } else {
151
- const div = document.createElement('div');
152
- const divStyle: CSS.Properties = {
153
- ...fontStyles,
154
- marginBottom: bottomAdjustment + 'px',
155
- paddingTop: topAdjustment + 'px',
156
- };
157
- Object.assign(div.style, divStyle);
158
- div.innerHTML = value
195
+ textBlock.innerHTML = value
159
196
  .split('')
160
197
  .map(
161
198
  (l: string, i: number) =>
@@ -165,6 +202,6 @@ export const uiRender = async (arg: UIRenderProps<TextSchema>) => {
165
202
  )
166
203
  .join('');
167
204
 
168
- container.appendChild(div);
205
+ container.appendChild(textBlock);
169
206
  }
170
207
  };
@@ -1,5 +0,0 @@
1
- "use strict";
2
- Object.defineProperty(exports, "__esModule", { value: true });
3
- exports.DEFAULT_OPACITY = void 0;
4
- exports.DEFAULT_OPACITY = 1;
5
- //# sourceMappingURL=constants.js.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"constants.js","sourceRoot":"","sources":["../../../../src/image/constants.ts"],"names":[],"mappings":";;;AAAa,QAAA,eAAe,GAAG,CAAC,CAAC"}
@@ -1,3 +0,0 @@
1
- "use strict";
2
- Object.defineProperty(exports, "__esModule", { value: true });
3
- //# sourceMappingURL=helper.js.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"helper.js","sourceRoot":"","sources":["../../../../src/image/helper.ts"],"names":[],"mappings":""}
@@ -1,2 +0,0 @@
1
- export const DEFAULT_OPACITY = 1;
2
- //# sourceMappingURL=constants.js.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"constants.js","sourceRoot":"","sources":["../../../../src/image/constants.ts"],"names":[],"mappings":"AAAA,MAAM,CAAC,MAAM,eAAe,GAAG,CAAC,CAAC"}
@@ -1,2 +0,0 @@
1
- export {};
2
- //# sourceMappingURL=helper.js.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"helper.js","sourceRoot":"","sources":["../../../../src/image/helper.ts"],"names":[],"mappings":""}
@@ -1 +0,0 @@
1
- export declare const DEFAULT_OPACITY = 1;
@@ -1 +0,0 @@
1
- export {};
@@ -1 +0,0 @@
1
- export const DEFAULT_OPACITY = 1;
@@ -1 +0,0 @@
1
- export {};