@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.
- package/dist/cjs/__tests__/font.test.js +237 -27
- package/dist/cjs/__tests__/font.test.js.map +1 -1
- package/dist/cjs/__tests__/helper.test.js +25 -0
- package/dist/cjs/__tests__/helper.test.js.map +1 -1
- package/dist/cjs/src/constants.js +12 -5
- package/dist/cjs/src/constants.js.map +1 -1
- package/dist/cjs/src/font.js +180 -48
- package/dist/cjs/src/font.js.map +1 -1
- package/dist/cjs/src/helper.js +14 -1
- package/dist/cjs/src/helper.js.map +1 -1
- package/dist/cjs/src/index.js +18 -6
- package/dist/cjs/src/index.js.map +1 -1
- package/dist/cjs/src/schema.js +10 -1
- package/dist/cjs/src/schema.js.map +1 -1
- package/dist/cjs/src/type.js.map +1 -1
- package/dist/esm/__tests__/font.test.js +238 -28
- package/dist/esm/__tests__/font.test.js.map +1 -1
- package/dist/esm/__tests__/helper.test.js +26 -1
- package/dist/esm/__tests__/helper.test.js.map +1 -1
- package/dist/esm/src/constants.js +11 -4
- package/dist/esm/src/constants.js.map +1 -1
- package/dist/esm/src/font.js +176 -47
- package/dist/esm/src/font.js.map +1 -1
- package/dist/esm/src/helper.js +11 -1
- package/dist/esm/src/helper.js.map +1 -1
- package/dist/esm/src/index.js +4 -4
- package/dist/esm/src/index.js.map +1 -1
- package/dist/esm/src/schema.js +9 -0
- package/dist/esm/src/schema.js.map +1 -1
- package/dist/esm/src/type.js.map +1 -1
- package/dist/types/src/constants.d.ts +11 -4
- package/dist/types/src/font.d.ts +20 -3
- package/dist/types/src/helper.d.ts +3 -0
- package/dist/types/src/index.d.ts +6 -6
- package/dist/types/src/schema.d.ts +165 -0
- package/dist/types/src/type.d.ts +9 -0
- package/package.json +1 -1
- package/src/constants.ts +11 -4
- package/src/font.ts +228 -72
- package/src/helper.ts +14 -1
- package/src/index.ts +38 -11
- package/src/schema.ts +10 -0
- package/src/type.ts +9 -0
package/dist/types/src/type.d.ts
CHANGED
@@ -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
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
|
8
|
-
export const
|
9
|
-
export const
|
10
|
-
export const
|
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
|
-
|
11
|
-
|
12
|
-
|
13
|
-
|
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
|
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
|
-
|
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
|
-
//
|
83
|
-
|
84
|
-
|
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
|
-
|
87
|
-
|
98
|
+
if (verticalAlignment === VERTICAL_ALIGN_TOP) {
|
99
|
+
return { topAdj: pt2px(topAdjustment), bottomAdj: 0 };
|
100
|
+
}
|
88
101
|
|
89
|
-
//
|
90
|
-
|
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
|
107
|
-
|
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
|
121
|
-
|
122
|
-
|
123
|
-
const
|
124
|
-
|
125
|
-
|
126
|
-
|
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
|
-
|
171
|
-
const {
|
172
|
-
const
|
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
|
262
|
+
const characterSpacing = schemaCharacterSpacing ?? DEFAULT_CHARACTER_SPACING;
|
176
263
|
const fontKitFont = await getFontKitFont(textSchema, font);
|
177
|
-
const
|
178
|
-
const textWidth = widthOfTextAtSize(textContent, fontKitFont, fontSize);
|
264
|
+
const textContentRows = input.split('\n');
|
179
265
|
|
180
266
|
let dynamicFontSize = fontSize;
|
181
|
-
|
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
|
-
|
184
|
-
dynamicFontSize
|
185
|
-
|
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
|
-
|
189
|
-
|
190
|
-
|
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 {
|
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
|
-
|
9
|
-
|
10
|
-
|
11
|
-
|
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,
|
64
|
+
calculateDynamicFontSize,
|
65
|
+
getFallbackFontName,
|
55
66
|
getDefaultFont,
|
56
67
|
heightOfFontAtSize,
|
68
|
+
widthOfTextAtSize,
|
57
69
|
checkFont,
|
58
70
|
getFontKitFont,
|
59
|
-
|
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
|
-
|
70
|
-
|
71
|
-
|
72
|
-
|
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';
|