@pdfme/common 2.0.2 → 2.2.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 (43) hide show
  1. package/dist/cjs/__tests__/font.test.js +237 -27
  2. package/dist/cjs/__tests__/font.test.js.map +1 -1
  3. package/dist/cjs/__tests__/helper.test.js +25 -0
  4. package/dist/cjs/__tests__/helper.test.js.map +1 -1
  5. package/dist/cjs/src/constants.js +12 -5
  6. package/dist/cjs/src/constants.js.map +1 -1
  7. package/dist/cjs/src/font.js +180 -48
  8. package/dist/cjs/src/font.js.map +1 -1
  9. package/dist/cjs/src/helper.js +14 -1
  10. package/dist/cjs/src/helper.js.map +1 -1
  11. package/dist/cjs/src/index.js +18 -6
  12. package/dist/cjs/src/index.js.map +1 -1
  13. package/dist/cjs/src/schema.js +10 -1
  14. package/dist/cjs/src/schema.js.map +1 -1
  15. package/dist/cjs/src/type.js.map +1 -1
  16. package/dist/esm/__tests__/font.test.js +238 -28
  17. package/dist/esm/__tests__/font.test.js.map +1 -1
  18. package/dist/esm/__tests__/helper.test.js +26 -1
  19. package/dist/esm/__tests__/helper.test.js.map +1 -1
  20. package/dist/esm/src/constants.js +11 -4
  21. package/dist/esm/src/constants.js.map +1 -1
  22. package/dist/esm/src/font.js +176 -47
  23. package/dist/esm/src/font.js.map +1 -1
  24. package/dist/esm/src/helper.js +11 -1
  25. package/dist/esm/src/helper.js.map +1 -1
  26. package/dist/esm/src/index.js +4 -4
  27. package/dist/esm/src/index.js.map +1 -1
  28. package/dist/esm/src/schema.js +9 -0
  29. package/dist/esm/src/schema.js.map +1 -1
  30. package/dist/esm/src/type.js.map +1 -1
  31. package/dist/types/src/constants.d.ts +11 -4
  32. package/dist/types/src/font.d.ts +20 -3
  33. package/dist/types/src/helper.d.ts +3 -0
  34. package/dist/types/src/index.d.ts +6 -6
  35. package/dist/types/src/schema.d.ts +165 -0
  36. package/dist/types/src/type.d.ts +9 -0
  37. package/package.json +1 -1
  38. package/src/constants.ts +11 -4
  39. package/src/font.ts +228 -72
  40. package/src/helper.ts +14 -1
  41. package/src/index.ts +38 -11
  42. package/src/schema.ts +10 -0
  43. package/src/type.ts +9 -0
@@ -1,5 +1,12 @@
1
1
  import { z } from 'zod';
2
+ import type { Font as FontKitFont } from 'fontkit';
2
3
  import { Lang, Size, Alignment, BarcodeSchemaType, SchemaType, CommonSchema as _CommonSchema, TextSchema, ImageSchema, BarcodeSchema, Schema, SchemaInputs, SchemaForUI, Font, BasePdf, Template, CommonProps, GeneratorOptions, GenerateProps, UIOptions, UIProps, PreviewProps, PreviewReactProps, DesignerProps, DesignerReactProps } from './schema.js';
4
+ export declare type FontWidthCalcValues = {
5
+ font: FontKitFont;
6
+ fontSize: number;
7
+ characterSpacing: number;
8
+ boxWidthInPt: number;
9
+ };
3
10
  declare type CommonSchema = z.infer<typeof _CommonSchema>;
4
11
  export declare const schemaTypes: readonly ["text", "image", "qrcode", "japanpost", "ean13", "ean8", "code39", "code128", "nw7", "itf14", "upca", "upce", "gs1datamatrix"];
5
12
  export declare const isTextSchema: (arg: CommonSchema) => arg is {
@@ -12,6 +19,7 @@ export declare const isTextSchema: (arg: CommonSchema) => arg is {
12
19
  };
13
20
  rotate?: number | undefined;
14
21
  alignment?: "center" | "left" | "right" | undefined;
22
+ verticalAlignment?: "top" | "bottom" | "middle" | undefined;
15
23
  fontSize?: number | undefined;
16
24
  fontName?: string | undefined;
17
25
  fontColor?: string | undefined;
@@ -21,6 +29,7 @@ export declare const isTextSchema: (arg: CommonSchema) => arg is {
21
29
  dynamicFontSize?: {
22
30
  max: number;
23
31
  min: number;
32
+ fit?: string | undefined;
24
33
  } | undefined;
25
34
  };
26
35
  export declare const isImageSchema: (arg: CommonSchema) => arg is {
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@pdfme/common",
3
- "version": "2.0.2",
3
+ "version": "2.2.0",
4
4
  "sideEffects": false,
5
5
  "author": "hand-dot",
6
6
  "license": "MIT",
package/src/constants.ts CHANGED
@@ -1,13 +1,20 @@
1
1
  export const DEFAULT_FONT_NAME = 'Roboto';
2
2
  export const DEFAULT_FONT_SIZE = 13;
3
3
  export const DEFAULT_ALIGNMENT = 'left';
4
+ export const VERTICAL_ALIGN_TOP = 'top';
5
+ export const VERTICAL_ALIGN_MIDDLE = 'middle';
6
+ export const VERTICAL_ALIGN_BOTTOM = 'bottom';
7
+ export const DEFAULT_VERTICAL_ALIGNMENT = VERTICAL_ALIGN_TOP;
4
8
  export const DEFAULT_LINE_HEIGHT = 1;
5
9
  export const DEFAULT_CHARACTER_SPACING = 0;
6
10
  export const DEFAULT_FONT_COLOR = '#000';
7
- export const DEFAULT_TOLERANCE = 1;
8
- export const DEFAULT_FONT_SIZE_ADJUSTMENT = 0.25;
9
- export const DEFAULT_PT_TO_PX_RATIO = 1.333;
10
- export const DEFAULT_PT_TO_MM_RATIO = 0.3528;
11
+ export const DYNAMIC_FIT_VERTICAL = 'vertical';
12
+ export const DYNAMIC_FIT_HORIZONTAL = 'horizontal';
13
+ export const DEFAULT_DYNAMIC_FIT = DYNAMIC_FIT_VERTICAL;
14
+ export const FONT_SIZE_ADJUSTMENT = 0.25;
15
+ export const PT_TO_PX_RATIO = 1.333;
16
+ export const PT_TO_MM_RATIO = 0.3528;
17
+ export const MM_TO_PT_RATIO = 2.8346; // https://www.ddc.co.jp/words/archives/20090701114500.html
11
18
 
12
19
  export const BLANK_PDF =
13
20
  'data:application/pdf;base64,JVBERi0xLjcKJeLjz9MKNSAwIG9iago8PAovRmlsdGVyIC9GbGF0ZURlY29kZQovTGVuZ3RoIDM4Cj4+CnN0cmVhbQp4nCvkMlAwUDC1NNUzMVGwMDHUszRSKErlCtfiyuMK5AIAXQ8GCgplbmRzdHJlYW0KZW5kb2JqCjQgMCBvYmoKPDwKL1R5cGUgL1BhZ2UKL01lZGlhQm94IFswIDAgNTk1LjQ0IDg0MS45Ml0KL1Jlc291cmNlcyA8PAo+PgovQ29udGVudHMgNSAwIFIKL1BhcmVudCAyIDAgUgo+PgplbmRvYmoKMiAwIG9iago8PAovVHlwZSAvUGFnZXMKL0tpZHMgWzQgMCBSXQovQ291bnQgMQo+PgplbmRvYmoKMSAwIG9iago8PAovVHlwZSAvQ2F0YWxvZwovUGFnZXMgMiAwIFIKPj4KZW5kb2JqCjMgMCBvYmoKPDwKL3RyYXBwZWQgKGZhbHNlKQovQ3JlYXRvciAoU2VyaWYgQWZmaW5pdHkgRGVzaWduZXIgMS4xMC40KQovVGl0bGUgKFVudGl0bGVkLnBkZikKL0NyZWF0aW9uRGF0ZSAoRDoyMDIyMDEwNjE0MDg1OCswOScwMCcpCi9Qcm9kdWNlciAoaUxvdmVQREYpCi9Nb2REYXRlIChEOjIwMjIwMTA2MDUwOTA5WikKPj4KZW5kb2JqCjYgMCBvYmoKPDwKL1NpemUgNwovUm9vdCAxIDAgUgovSW5mbyAzIDAgUgovSUQgWzwyODhCM0VENTAyOEU0MDcyNERBNzNCOUE0Nzk4OUEwQT4gPEY1RkJGNjg4NkVERDZBQUNBNDRCNEZDRjBBRDUxRDlDPl0KL1R5cGUgL1hSZWYKL1cgWzEgMiAyXQovRmlsdGVyIC9GbGF0ZURlY29kZQovSW5kZXggWzAgN10KL0xlbmd0aCAzNgo+PgpzdHJlYW0KeJxjYGD4/5+RUZmBgZHhFZBgDAGxakAEP5BgEmFgAABlRwQJCmVuZHN0cmVhbQplbmRvYmoKc3RhcnR4cmVmCjUzMgolJUVPRgo=';
package/src/font.ts CHANGED
@@ -1,17 +1,20 @@
1
1
  import * as fontkit from 'fontkit';
2
2
  import type { Font as FontKitFont } from 'fontkit';
3
- import { Template, Schema, Font, isTextSchema, TextSchema } from './type';
3
+ import { FontWidthCalcValues, Template, Schema, Font, isTextSchema, TextSchema } from './type';
4
4
  import { Buffer } from 'buffer';
5
5
  import {
6
6
  DEFAULT_FONT_VALUE,
7
7
  DEFAULT_FONT_NAME,
8
8
  DEFAULT_FONT_SIZE,
9
9
  DEFAULT_CHARACTER_SPACING,
10
- DEFAULT_TOLERANCE,
11
- DEFAULT_FONT_SIZE_ADJUSTMENT,
12
- DEFAULT_PT_TO_MM_RATIO,
13
- DEFAULT_PT_TO_PX_RATIO,
10
+ DEFAULT_LINE_HEIGHT,
11
+ FONT_SIZE_ADJUSTMENT,
12
+ DEFAULT_DYNAMIC_FIT,
13
+ DYNAMIC_FIT_HORIZONTAL,
14
+ DYNAMIC_FIT_VERTICAL,
15
+ VERTICAL_ALIGN_TOP,
14
16
  } from './constants';
17
+ import { mm2pt, pt2mm, pt2px } from './helper';
15
18
  import { b64toUint8Array } from "."
16
19
 
17
20
  export const getFallbackFontName = (font: Font) => {
@@ -31,7 +34,7 @@ export const getFallbackFontName = (font: Font) => {
31
34
  const getFallbackFont = (font: Font) => {
32
35
  const fallbackFontName = getFallbackFontName(font);
33
36
  return font[fallbackFontName];
34
- }
37
+ };
35
38
 
36
39
  export const getDefaultFont = (): Font => ({
37
40
  [DEFAULT_FONT_NAME]: { data: b64toUint8Array(DEFAULT_FONT_VALUE), fallback: true },
@@ -74,21 +77,51 @@ export const checkFont = (arg: { font: Font; template: Template }) => {
74
77
  }
75
78
  };
76
79
 
77
- export const getFontAlignmentValue = (fontKitFont: FontKitFont, fontSize: number) => {
80
+ export const getBrowserVerticalFontAdjustments = (
81
+ fontKitFont: FontKitFont,
82
+ fontSize: number,
83
+ lineHeight: number,
84
+ verticalAlignment: string
85
+ ) => {
78
86
  const { ascent, descent, unitsPerEm } = fontKitFont;
79
87
 
80
- const fontSizeInPx = fontSize * DEFAULT_PT_TO_PX_RATIO;
88
+ // Fonts have a designed line height that the browser renders when using `line-height: normal`
89
+ const fontBaseLineHeight = (ascent - descent) / unitsPerEm;
81
90
 
82
- // Convert ascent and descent to px values
83
- const ascentInPixels = (ascent / unitsPerEm) * fontSizeInPx;
84
- const descentInPixels = (descent / unitsPerEm) * fontSizeInPx;
91
+ // For vertical alignment top
92
+ // To achieve consistent positioning between browser and PDF, we apply the difference between
93
+ // the font's actual height and the font size in pixels.
94
+ // Browsers middle the font within this height, so we only need half of it to apply to the top.
95
+ // This means the font renders a bit lower in the browser, but achieves PDF alignment
96
+ const topAdjustment = (fontBaseLineHeight * fontSize - fontSize) / 2;
85
97
 
86
- // Calculate the single line height in px
87
- const singleLineHeight = ((ascentInPixels + Math.abs(descentInPixels)) / fontSizeInPx);
98
+ if (verticalAlignment === VERTICAL_ALIGN_TOP) {
99
+ return { topAdj: pt2px(topAdjustment), bottomAdj: 0 };
100
+ }
88
101
 
89
- // Calculate the top margin/padding in px
90
- return ((singleLineHeight * fontSizeInPx) - fontSizeInPx) / 2
91
- }
102
+ // For vertical alignment bottom and middle
103
+ // When browsers render text in a non-form element (such as a <div>), some of the text may be
104
+ // lowered below and outside the containing element if the line height used is less than
105
+ // the base line-height of the font.
106
+ // This behaviour does not happen in a <textarea> though, so we need to adjust the positioning
107
+ // for consistency between editing and viewing to stop text jumping up and down.
108
+ // This portion of text is half of the difference between the base line height and the used
109
+ // line height. If using the same or higher line-height than the base font, then line-height
110
+ // takes over in the browser and this adjustment is not needed.
111
+ // Unlike the top adjustment - this is only driven by browser behaviour, not PDF alignment.
112
+ let bottomAdjustment = 0;
113
+ if (lineHeight < fontBaseLineHeight) {
114
+ bottomAdjustment = ((fontBaseLineHeight - lineHeight) * fontSize) / 2;
115
+ }
116
+
117
+ return { topAdj: 0, bottomAdj: pt2px(bottomAdjustment) };
118
+ };
119
+
120
+ export const getFontDescentInPt = (fontKitFont: FontKitFont, fontSize: number) => {
121
+ const { descent, unitsPerEm } = fontKitFont;
122
+
123
+ return (descent / unitsPerEm) * fontSize;
124
+ };
92
125
 
93
126
  export const heightOfFontAtSize = (fontKitFont: FontKitFont, fontSize: number) => {
94
127
  const { ascent, descent, bbox, unitsPerEm } = fontKitFont;
@@ -103,46 +136,19 @@ export const heightOfFontAtSize = (fontKitFont: FontKitFont, fontSize: number) =
103
136
  return (height / 1000) * fontSize;
104
137
  };
105
138
 
106
- const widthOfTextAtSize = (input: string, fontKitFont: FontKitFont, fontSize: number) => {
107
- const { glyphs } = fontKitFont.layout(input);
108
- const scale = 1000 / fontKitFont.unitsPerEm;
109
- return glyphs.reduce((totalWidth, glyph) => totalWidth + glyph.advanceWidth * scale, 0) * (fontSize / 1000);
110
- }
111
-
112
- const calculateCharacterSpacing = (
113
- textContent: string,
114
- textCharacterSpacing: number
115
- ) => {
116
- const numberOfCharacters = textContent.length;
117
- return (numberOfCharacters - 1) * textCharacterSpacing;
139
+ const calculateCharacterSpacing = (textContent: string, textCharacterSpacing: number) => {
140
+ return (textContent.length - 1) * textCharacterSpacing;
118
141
  };
119
142
 
120
- const calculateTextWidthInMm = (textContent: string, textWidth: number, textCharacterSpacing: number) =>
121
- (textWidth + calculateCharacterSpacing(textContent, textCharacterSpacing)) * DEFAULT_PT_TO_MM_RATIO;
122
-
123
- const getLongestLine = (
124
- textContentRows: string[],
125
- fontKitFont: FontKitFont,
126
- fontSize: number,
127
- characterSpacingCount: number
128
- ) => {
129
- let longestLine = '';
130
- let maxLineWidth = 0;
131
-
132
- textContentRows.forEach((line) => {
133
- const textWidth = widthOfTextAtSize(line, fontKitFont, fontSize);
134
- const lineWidth = calculateTextWidthInMm(line, textWidth, characterSpacingCount);
135
-
136
- if (lineWidth > maxLineWidth) {
137
- longestLine = line;
138
- maxLineWidth = lineWidth;
139
- }
140
- });
141
-
142
- return longestLine;
143
+ export const widthOfTextAtSize = (text: string, fontKitFont: FontKitFont, fontSize: number, characterSpacing: number) => {
144
+ const { glyphs } = fontKitFont.layout(text);
145
+ const scale = 1000 / fontKitFont.unitsPerEm;
146
+ const standardWidth =
147
+ glyphs.reduce((totalWidth, glyph) => totalWidth + glyph.advanceWidth * scale, 0) *
148
+ (fontSize / 1000);
149
+ return standardWidth + calculateCharacterSpacing(text, characterSpacing);
143
150
  };
144
151
 
145
-
146
152
  const fontKitFontCache: { [fontName: string]: FontKitFont } = {};
147
153
  export const getFontKitFont = async (textSchema: TextSchema, font: Font) => {
148
154
  const fontName = textSchema.fontName || getFallbackFontName(font);
@@ -157,38 +163,188 @@ export const getFontKitFont = async (textSchema: TextSchema, font: Font) => {
157
163
  }
158
164
 
159
165
  const fontKitFont = fontkit.create(fontData instanceof Buffer ? fontData : Buffer.from(fontData as ArrayBuffer));
160
- fontKitFontCache[fontName] = fontKitFont
166
+ fontKitFontCache[fontName] = fontKitFont;
161
167
 
162
168
  return fontKitFont;
163
- }
169
+ };
164
170
 
165
- const getTextContent = (input: string, fontKitFont: FontKitFont, fontSize: number, characterSpacingCount: number): string => {
166
- const textContentRows = input.split('\n');
167
- return textContentRows.length > 1 ? getLongestLine(textContentRows, fontKitFont, fontSize, characterSpacingCount) : input;
168
- }
169
171
 
170
- export const calculateDynamicFontSize = async ({ textSchema, font, input }: { textSchema: TextSchema, font: Font, input: string }) => {
171
- const { fontSize: _fontSize, dynamicFontSize: dynamicFontSizeSetting, characterSpacing, width } = textSchema;
172
- const fontSize = _fontSize || DEFAULT_FONT_SIZE;
172
+ const isTextExceedingBoxWidth = (text: string, calcValues: FontWidthCalcValues) => {
173
+ const { font, fontSize, characterSpacing, boxWidthInPt } = calcValues;
174
+ const textWidth = widthOfTextAtSize(text, font, fontSize, characterSpacing);
175
+ return textWidth > boxWidthInPt;
176
+ };
177
+
178
+ /**
179
+ * Incrementally checks the current line for its real length
180
+ * and returns the position where it exceeds the box width.
181
+ * Returns `null` to indicate if textLine is shorter than the available box.
182
+ */
183
+ const getOverPosition = (textLine: string, calcValues: FontWidthCalcValues) => {
184
+ for (let i = 0; i <= textLine.length; i++) {
185
+ if (isTextExceedingBoxWidth(textLine.slice(0, i + 1), calcValues)) {
186
+ return i;
187
+ }
188
+ }
189
+
190
+ return null;
191
+ };
192
+
193
+ /**
194
+ * Gets the position of the split. Splits the exceeding line at
195
+ * the last whitespace prior to it exceeding the bounding box width.
196
+ */
197
+ const getSplitPosition = (textLine: string, calcValues: FontWidthCalcValues) => {
198
+ const overPos = getOverPosition(textLine, calcValues);
199
+ if (overPos === null) return textLine.length; // input line is shorter than the available space
200
+
201
+ let overPosTmp = overPos;
202
+ while (textLine[overPosTmp] !== ' ' && overPosTmp >= 0) {
203
+ overPosTmp--;
204
+ }
205
+
206
+ // For very long lines with no whitespace use the original overPos
207
+ return overPosTmp > 0 ? overPosTmp : overPos;
208
+ };
209
+
210
+ /**
211
+ * Recursively splits the line at getSplitPosition.
212
+ * If there is some leftover, split the rest again in the same manner.
213
+ */
214
+ export const getSplittedLines = (textLine: string, calcValues: FontWidthCalcValues): string[] => {
215
+ const splitPos = getSplitPosition(textLine, calcValues);
216
+ const splittedLine = textLine.substring(0, splitPos);
217
+ const rest = textLine.substring(splitPos).trimStart();
218
+
219
+ if (rest === textLine) {
220
+ // if we went so small that we want to split on the first char
221
+ // then end recursion to avoid infinite loop
222
+ return [textLine];
223
+ }
224
+
225
+ if (rest.length === 0) {
226
+ // end recursion if there is no leftover
227
+ return [splittedLine];
228
+ }
229
+
230
+ return [splittedLine, ...getSplittedLines(rest, calcValues)];
231
+ };
232
+
233
+ /**
234
+ * If using dynamic font size, iteratively increment or decrement the
235
+ * font size to fit the containing box.
236
+ * Calculating space usage involves splitting lines where they exceed
237
+ * the box width based on the proposed size.
238
+ */
239
+ export const calculateDynamicFontSize = async ({
240
+ textSchema,
241
+ font,
242
+ input,
243
+ startingFontSize,
244
+ }: {
245
+ textSchema: TextSchema;
246
+ font: Font;
247
+ input: string;
248
+ startingFontSize?: number | undefined;
249
+ }) => {
250
+ const {
251
+ fontSize: schemaFontSize,
252
+ dynamicFontSize: dynamicFontSizeSetting,
253
+ characterSpacing: schemaCharacterSpacing,
254
+ width: boxWidth,
255
+ height: boxHeight,
256
+ lineHeight = DEFAULT_LINE_HEIGHT,
257
+ } = textSchema;
258
+ const fontSize = startingFontSize || schemaFontSize || DEFAULT_FONT_SIZE;
173
259
  if (!dynamicFontSizeSetting) return fontSize;
260
+ if (dynamicFontSizeSetting.max < dynamicFontSizeSetting.min) return fontSize;
174
261
 
175
- const characterSpacingCount = characterSpacing ?? DEFAULT_CHARACTER_SPACING;
262
+ const characterSpacing = schemaCharacterSpacing ?? DEFAULT_CHARACTER_SPACING;
176
263
  const fontKitFont = await getFontKitFont(textSchema, font);
177
- const textContent = getTextContent(input, fontKitFont, fontSize, characterSpacingCount);
178
- const textWidth = widthOfTextAtSize(textContent, fontKitFont, fontSize);
264
+ const textContentRows = input.split('\n');
179
265
 
180
266
  let dynamicFontSize = fontSize;
181
- let textWidthInMm = calculateTextWidthInMm(textContent, textWidth, characterSpacingCount);
267
+ if (dynamicFontSize < dynamicFontSizeSetting.min) {
268
+ dynamicFontSize = dynamicFontSizeSetting.min;
269
+ } else if (dynamicFontSize > dynamicFontSizeSetting.max) {
270
+ dynamicFontSize = dynamicFontSizeSetting.max;
271
+ }
272
+ const dynamicFontFit = dynamicFontSizeSetting.fit ?? DEFAULT_DYNAMIC_FIT;
273
+
274
+ const calculateConstraints = (size: number) => {
275
+ let totalWidthInMm = 0;
276
+ let totalHeightInMm = 0;
277
+
278
+ const boxWidthInPt = mm2pt(boxWidth);
279
+ const textHeight = heightOfFontAtSize(fontKitFont, size);
280
+ const textHeightInMm = pt2mm(textHeight * lineHeight);
281
+
282
+ textContentRows.forEach((paragraph) => {
283
+ const lines = getSplittedLines(paragraph, {
284
+ font: fontKitFont,
285
+ fontSize: size,
286
+ characterSpacing,
287
+ boxWidthInPt,
288
+ });
289
+ lines.forEach((line) => {
290
+ if (dynamicFontFit === DYNAMIC_FIT_VERTICAL) {
291
+ // For vertical fit we want to consider the width of text lines where we detect a split
292
+ const textWidth = widthOfTextAtSize(line, fontKitFont, size, characterSpacing);
293
+ const textWidthInMm = pt2mm(textWidth);
294
+ totalWidthInMm = Math.max(totalWidthInMm, textWidthInMm);
295
+ }
296
+
297
+ totalHeightInMm += textHeightInMm;
298
+ });
299
+ if (dynamicFontFit === DYNAMIC_FIT_HORIZONTAL) {
300
+ // For horizontal fit we want to consider the line's width 'unsplit'
301
+ const textWidth = widthOfTextAtSize(paragraph, fontKitFont, size, characterSpacing);
302
+ const textWidthInMm = pt2mm(textWidth);
303
+ totalWidthInMm = Math.max(totalWidthInMm, textWidthInMm);
304
+ }
305
+ });
306
+
307
+ return { totalWidthInMm, totalHeightInMm };
308
+ };
309
+
310
+ const shouldFontGrowToFit = (totalWidthInMm: number, totalHeightInMm: number) => {
311
+ if (dynamicFontSize >= dynamicFontSizeSetting.max) {
312
+ return false;
313
+ }
314
+ if (dynamicFontFit === DYNAMIC_FIT_HORIZONTAL) {
315
+ return totalWidthInMm < boxWidth;
316
+ }
317
+ return totalHeightInMm < boxHeight;
318
+ };
182
319
 
183
- while (textWidthInMm > width - DEFAULT_TOLERANCE && dynamicFontSize > dynamicFontSizeSetting.min) {
184
- dynamicFontSize -= DEFAULT_FONT_SIZE_ADJUSTMENT;
185
- textWidthInMm = calculateTextWidthInMm(textContent, widthOfTextAtSize(textContent, fontKitFont, dynamicFontSize), characterSpacingCount);
320
+ const shouldFontShrinkToFit = (totalWidthInMm: number, totalHeightInMm: number) => {
321
+ if (dynamicFontSize <= dynamicFontSizeSetting.min || dynamicFontSize <= 0) {
322
+ return false;
323
+ }
324
+ return totalWidthInMm > boxWidth || totalHeightInMm > boxHeight;
325
+ };
326
+
327
+ let { totalWidthInMm, totalHeightInMm } = calculateConstraints(dynamicFontSize);
328
+
329
+ // Attempt to increase the font size up to desired fit
330
+ while (shouldFontGrowToFit(totalWidthInMm, totalHeightInMm)) {
331
+ dynamicFontSize += FONT_SIZE_ADJUSTMENT;
332
+ const { totalWidthInMm: newWidth, totalHeightInMm: newHeight } = calculateConstraints(dynamicFontSize);
333
+
334
+ if (newHeight < boxHeight) {
335
+ totalWidthInMm = newWidth;
336
+ totalHeightInMm = newHeight;
337
+ } else {
338
+ dynamicFontSize -= FONT_SIZE_ADJUSTMENT;
339
+ break;
340
+ }
186
341
  }
187
342
 
188
- while (textWidthInMm < width - DEFAULT_TOLERANCE && dynamicFontSize < dynamicFontSizeSetting.max) {
189
- dynamicFontSize += DEFAULT_FONT_SIZE_ADJUSTMENT;
190
- textWidthInMm = calculateTextWidthInMm(textContent, widthOfTextAtSize(textContent, fontKitFont, dynamicFontSize), characterSpacingCount);
343
+ // Attempt to decrease the font size down to desired fit
344
+ while (shouldFontShrinkToFit(totalWidthInMm, totalHeightInMm)) {
345
+ dynamicFontSize -= FONT_SIZE_ADJUSTMENT;
346
+ ({ totalWidthInMm, totalHeightInMm } = calculateConstraints(dynamicFontSize));
191
347
  }
192
348
 
193
349
  return dynamicFontSize;
194
- };
350
+ };
package/src/helper.ts CHANGED
@@ -10,7 +10,20 @@ import {
10
10
  GenerateProps as GeneratePropsSchema,
11
11
  UIProps as UIPropsSchema,
12
12
  } from './schema';
13
- import { checkFont } from "./font"
13
+ import { MM_TO_PT_RATIO, PT_TO_MM_RATIO, PT_TO_PX_RATIO } from './constants';
14
+ import { checkFont } from './font';
15
+
16
+ export const mm2pt = (mm: number): number => {
17
+ return parseFloat(String(mm)) * MM_TO_PT_RATIO;
18
+ };
19
+
20
+ export const pt2mm = (pt: number): number => {
21
+ return pt * PT_TO_MM_RATIO;
22
+ };
23
+
24
+ export const pt2px = (pt: number): number => {
25
+ return pt * PT_TO_PX_RATIO;
26
+ };
14
27
 
15
28
  const blob2Base64Pdf = (blob: Blob) => {
16
29
  return new Promise<string>((resolve, reject) => {
package/src/index.ts CHANGED
@@ -2,18 +2,26 @@ import {
2
2
  DEFAULT_FONT_NAME,
3
3
  DEFAULT_FONT_SIZE,
4
4
  DEFAULT_ALIGNMENT,
5
+ VERTICAL_ALIGN_TOP,
6
+ VERTICAL_ALIGN_MIDDLE,
7
+ VERTICAL_ALIGN_BOTTOM,
8
+ DEFAULT_VERTICAL_ALIGNMENT,
5
9
  DEFAULT_LINE_HEIGHT,
6
10
  DEFAULT_CHARACTER_SPACING,
7
11
  DEFAULT_FONT_COLOR,
8
- DEFAULT_TOLERANCE,
9
- DEFAULT_FONT_SIZE_ADJUSTMENT,
10
- DEFAULT_PT_TO_MM_RATIO,
11
- DEFAULT_PT_TO_PX_RATIO,
12
+ DYNAMIC_FIT_VERTICAL,
13
+ DYNAMIC_FIT_HORIZONTAL,
14
+ DEFAULT_DYNAMIC_FIT,
15
+ FONT_SIZE_ADJUSTMENT,
16
+ MM_TO_PT_RATIO,
17
+ PT_TO_MM_RATIO,
18
+ PT_TO_PX_RATIO,
12
19
  BLANK_PDF,
13
20
  DEFAULT_FONT_VALUE,
14
21
  } from './constants.js';
15
22
  import { schemaTypes, isImageSchema, isBarcodeSchema, isTextSchema } from './type.js';
16
23
  import type {
24
+ FontWidthCalcValues,
17
25
  Lang,
18
26
  Size,
19
27
  Alignment,
@@ -48,28 +56,41 @@ import {
48
56
  checkPreviewProps,
49
57
  checkDesignerProps,
50
58
  checkGenerateProps,
59
+ mm2pt,
60
+ pt2px,
51
61
  validateBarcodeInput,
52
62
  } from './helper.js';
53
63
  import {
54
- calculateDynamicFontSize, getFallbackFontName,
64
+ calculateDynamicFontSize,
65
+ getFallbackFontName,
55
66
  getDefaultFont,
56
67
  heightOfFontAtSize,
68
+ widthOfTextAtSize,
57
69
  checkFont,
58
70
  getFontKitFont,
59
- getFontAlignmentValue,
71
+ getBrowserVerticalFontAdjustments,
72
+ getFontDescentInPt,
73
+ getSplittedLines,
60
74
  } from './font.js';
61
75
 
62
76
  export {
63
77
  DEFAULT_FONT_NAME,
64
78
  DEFAULT_FONT_SIZE,
65
79
  DEFAULT_ALIGNMENT,
80
+ VERTICAL_ALIGN_TOP,
81
+ VERTICAL_ALIGN_MIDDLE,
82
+ VERTICAL_ALIGN_BOTTOM,
83
+ DEFAULT_VERTICAL_ALIGNMENT,
66
84
  DEFAULT_LINE_HEIGHT,
67
85
  DEFAULT_CHARACTER_SPACING,
68
86
  DEFAULT_FONT_COLOR,
69
- DEFAULT_TOLERANCE,
70
- DEFAULT_FONT_SIZE_ADJUSTMENT,
71
- DEFAULT_PT_TO_MM_RATIO,
72
- DEFAULT_PT_TO_PX_RATIO,
87
+ DYNAMIC_FIT_VERTICAL,
88
+ DYNAMIC_FIT_HORIZONTAL,
89
+ DEFAULT_DYNAMIC_FIT,
90
+ FONT_SIZE_ADJUSTMENT,
91
+ MM_TO_PT_RATIO,
92
+ PT_TO_MM_RATIO,
93
+ PT_TO_PX_RATIO,
73
94
  BLANK_PDF,
74
95
  DEFAULT_FONT_VALUE,
75
96
  schemaTypes,
@@ -80,10 +101,15 @@ export {
80
101
  b64toUint8Array,
81
102
  getFallbackFontName,
82
103
  getDefaultFont,
104
+ getSplittedLines,
83
105
  heightOfFontAtSize,
106
+ widthOfTextAtSize,
107
+ mm2pt,
108
+ pt2px,
84
109
  checkFont,
110
+ getBrowserVerticalFontAdjustments,
111
+ getFontDescentInPt,
85
112
  getFontKitFont,
86
- getFontAlignmentValue,
87
113
  checkInputs,
88
114
  checkUIOptions,
89
115
  checkTemplate,
@@ -119,4 +145,5 @@ export type {
119
145
  PreviewReactProps,
120
146
  DesignerProps,
121
147
  DesignerReactProps,
148
+ FontWidthCalcValues,
122
149
  };
package/src/schema.ts CHANGED
@@ -1,5 +1,6 @@
1
1
  /* eslint dot-notation: "off"*/
2
2
  import { z } from 'zod';
3
+ import { VERTICAL_ALIGN_TOP, VERTICAL_ALIGN_MIDDLE, VERTICAL_ALIGN_BOTTOM } from "./constants";
3
4
 
4
5
  const langs = ['en', 'ja', 'ar', 'th', 'pl'] as const;
5
6
  export const Lang = z.enum(langs);
@@ -9,6 +10,13 @@ export const Size = z.object({ height: z.number(), width: z.number() });
9
10
  const alignments = ['left', 'center', 'right'] as const;
10
11
  export const Alignment = z.enum(alignments);
11
12
 
13
+ const verticalAlignments = [
14
+ VERTICAL_ALIGN_TOP,
15
+ VERTICAL_ALIGN_MIDDLE,
16
+ VERTICAL_ALIGN_BOTTOM,
17
+ ] as const;
18
+ export const VerticalAlignment = z.enum(verticalAlignments);
19
+
12
20
  // prettier-ignore
13
21
  export const barcodeSchemaTypes = ['qrcode', 'japanpost', 'ean13', 'ean8', 'code39', 'code128', 'nw7', 'itf14', 'upca', 'upce', 'gs1datamatrix'] as const;
14
22
  const notBarcodeSchemaTypes = ['text', 'image'] as const;
@@ -28,6 +36,7 @@ export const CommonSchema = z.object({
28
36
  export const TextSchema = CommonSchema.extend({
29
37
  type: z.literal(SchemaType.Enum.text),
30
38
  alignment: Alignment.optional(),
39
+ verticalAlignment: VerticalAlignment.optional(),
31
40
  fontSize: z.number().optional(),
32
41
  fontName: z.string().optional(),
33
42
  fontColor: z.string().optional(),
@@ -37,6 +46,7 @@ export const TextSchema = CommonSchema.extend({
37
46
  dynamicFontSize: z.object({
38
47
  max: z.number(),
39
48
  min: z.number(),
49
+ fit: z.string().optional(),
40
50
  }).optional(),
41
51
  });
42
52
 
package/src/type.ts CHANGED
@@ -1,4 +1,6 @@
1
1
  import { z } from 'zod';
2
+ import type { Font as FontKitFont } from 'fontkit';
3
+
2
4
  import {
3
5
  Lang,
4
6
  Size,
@@ -28,6 +30,13 @@ import {
28
30
  DesignerReactProps,
29
31
  } from './schema.js';
30
32
 
33
+ export type FontWidthCalcValues = {
34
+ font: FontKitFont;
35
+ fontSize: number;
36
+ characterSpacing: number;
37
+ boxWidthInPt: number;
38
+ };
39
+
31
40
  type CommonSchema = z.infer<typeof _CommonSchema>;
32
41
  export const schemaTypes = _schemaTypes;
33
42
  export const isTextSchema = (arg: CommonSchema): arg is TextSchema => arg.type === 'text';