@pdfme/common 2.2.1 → 3.0.0-beta.1

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 (65) hide show
  1. package/dist/cjs/__tests__/helper.test.js +292 -233
  2. package/dist/cjs/__tests__/helper.test.js.map +1 -1
  3. package/dist/cjs/src/constants.js +3 -17
  4. package/dist/cjs/src/constants.js.map +1 -1
  5. package/dist/cjs/src/helper.js +72 -99
  6. package/dist/cjs/src/helper.js.map +1 -1
  7. package/dist/cjs/src/index.js +7 -37
  8. package/dist/cjs/src/index.js.map +1 -1
  9. package/dist/cjs/src/schema.js +17 -71
  10. package/dist/cjs/src/schema.js.map +1 -1
  11. package/dist/cjs/src/types.js +3 -0
  12. package/dist/cjs/src/types.js.map +1 -0
  13. package/dist/esm/__tests__/helper.test.js +270 -234
  14. package/dist/esm/__tests__/helper.test.js.map +1 -1
  15. package/dist/esm/src/constants.js +2 -16
  16. package/dist/esm/src/constants.js.map +1 -1
  17. package/dist/esm/src/helper.js +68 -98
  18. package/dist/esm/src/helper.js.map +1 -1
  19. package/dist/esm/src/index.js +3 -6
  20. package/dist/esm/src/index.js.map +1 -1
  21. package/dist/esm/src/schema.js +14 -68
  22. package/dist/esm/src/schema.js.map +1 -1
  23. package/dist/esm/src/types.js +2 -0
  24. package/dist/esm/src/types.js.map +1 -0
  25. package/dist/types/src/constants.d.ts +2 -16
  26. package/dist/types/src/helper.d.ts +11 -2
  27. package/dist/types/src/index.d.ts +5 -8
  28. package/dist/types/src/schema.d.ts +696 -3574
  29. package/dist/types/src/types.d.ts +127 -0
  30. package/package.json +5 -11
  31. package/src/constants.ts +2 -18
  32. package/src/helper.ts +108 -116
  33. package/src/index.ts +28 -87
  34. package/src/schema.ts +18 -84
  35. package/src/types.ts +124 -0
  36. package/tsconfig.cjs.json +3 -2
  37. package/tsconfig.esm.json +3 -2
  38. package/dist/cjs/__tests__/barcode.test.js +0 -107
  39. package/dist/cjs/__tests__/barcode.test.js.map +0 -1
  40. package/dist/cjs/__tests__/font.test.js +0 -465
  41. package/dist/cjs/__tests__/font.test.js.map +0 -1
  42. package/dist/cjs/src/barcode.js +0 -53
  43. package/dist/cjs/src/barcode.js.map +0 -1
  44. package/dist/cjs/src/font.js +0 -310
  45. package/dist/cjs/src/font.js.map +0 -1
  46. package/dist/cjs/src/type.js +0 -12
  47. package/dist/cjs/src/type.js.map +0 -1
  48. package/dist/esm/__tests__/barcode.test.js +0 -102
  49. package/dist/esm/__tests__/barcode.test.js.map +0 -1
  50. package/dist/esm/__tests__/font.test.js +0 -440
  51. package/dist/esm/__tests__/font.test.js.map +0 -1
  52. package/dist/esm/src/barcode.js +0 -44
  53. package/dist/esm/src/barcode.js.map +0 -1
  54. package/dist/esm/src/font.js +0 -274
  55. package/dist/esm/src/font.js.map +0 -1
  56. package/dist/esm/src/type.js +0 -6
  57. package/dist/esm/src/type.js.map +0 -1
  58. package/dist/types/__tests__/barcode.test.d.ts +0 -1
  59. package/dist/types/__tests__/font.test.d.ts +0 -1
  60. package/dist/types/src/barcode.d.ts +0 -19
  61. package/dist/types/src/font.d.ts +0 -34
  62. package/dist/types/src/type.d.ts +0 -81
  63. package/src/barcode.ts +0 -51
  64. package/src/font.ts +0 -352
  65. package/src/type.ts +0 -69
package/src/font.ts DELETED
@@ -1,352 +0,0 @@
1
- import * as fontkit from 'fontkit';
2
- import type { Font as FontKitFont } from 'fontkit';
3
- import { FontWidthCalcValues, Template, Schema, Font, isTextSchema, TextSchema } from './type';
4
- import { Buffer } from 'buffer';
5
- import {
6
- DEFAULT_FONT_VALUE,
7
- DEFAULT_FONT_NAME,
8
- DEFAULT_FONT_SIZE,
9
- DEFAULT_CHARACTER_SPACING,
10
- DEFAULT_LINE_HEIGHT,
11
- FONT_SIZE_ADJUSTMENT,
12
- DEFAULT_DYNAMIC_FIT,
13
- DYNAMIC_FIT_HORIZONTAL,
14
- DYNAMIC_FIT_VERTICAL,
15
- VERTICAL_ALIGN_TOP,
16
- } from './constants';
17
- import { mm2pt, pt2mm, pt2px } from './helper';
18
- import { b64toUint8Array } from "."
19
-
20
- export const getFallbackFontName = (font: Font) => {
21
- const initial = '';
22
- const fallbackFontName = Object.entries(font).reduce((acc, cur) => {
23
- const [fontName, fontValue] = cur;
24
-
25
- return !acc && fontValue.fallback ? fontName : acc;
26
- }, initial);
27
- if (fallbackFontName === initial) {
28
- throw Error(`fallback flag is not found in font. true fallback flag must be only one.`);
29
- }
30
-
31
- return fallbackFontName;
32
- };
33
-
34
- const getFallbackFont = (font: Font) => {
35
- const fallbackFontName = getFallbackFontName(font);
36
- return font[fallbackFontName];
37
- };
38
-
39
- export const getDefaultFont = (): Font => ({
40
- [DEFAULT_FONT_NAME]: { data: b64toUint8Array(DEFAULT_FONT_VALUE), fallback: true },
41
- });
42
-
43
- const uniq = <T>(array: Array<T>) => Array.from(new Set(array));
44
-
45
- const getFontNamesInSchemas = (schemas: { [key: string]: Schema }[]) =>
46
- uniq(
47
- schemas
48
- .map((s) => Object.values(s).map((v) => (isTextSchema(v) ? v.fontName : '')))
49
- .reduce((acc, cur) => acc.concat(cur), [] as (string | undefined)[])
50
- .filter(Boolean) as string[]
51
- );
52
-
53
- export const checkFont = (arg: { font: Font; template: Template }) => {
54
- const { font, template: { schemas } } = arg;
55
- const fontValues = Object.values(font);
56
- const fallbackFontNum = fontValues.reduce((acc, cur) => (cur.fallback ? acc + 1 : acc), 0);
57
- if (fallbackFontNum === 0) {
58
- throw Error(`fallback flag is not found in font. true fallback flag must be only one.`);
59
- }
60
- if (fallbackFontNum > 1) {
61
- throw Error(
62
- `${fallbackFontNum} fallback flags found in font. true fallback flag must be only one.`
63
- );
64
- }
65
-
66
- const fontNamesInSchemas = getFontNamesInSchemas(schemas);
67
- const fontNames = Object.keys(font);
68
- if (fontNamesInSchemas.some((f) => !fontNames.includes(f))) {
69
- throw Error(
70
- `${fontNamesInSchemas
71
- .filter((f) => !fontNames.includes(f))
72
- .join()} of template.schemas is not found in font.`
73
- );
74
- }
75
- };
76
-
77
- export const getBrowserVerticalFontAdjustments = (
78
- fontKitFont: FontKitFont,
79
- fontSize: number,
80
- lineHeight: number,
81
- verticalAlignment: string
82
- ) => {
83
- const { ascent, descent, unitsPerEm } = fontKitFont;
84
-
85
- // Fonts have a designed line height that the browser renders when using `line-height: normal`
86
- const fontBaseLineHeight = (ascent - descent) / unitsPerEm;
87
-
88
- // For vertical alignment top
89
- // To achieve consistent positioning between browser and PDF, we apply the difference between
90
- // the font's actual height and the font size in pixels.
91
- // Browsers middle the font within this height, so we only need half of it to apply to the top.
92
- // This means the font renders a bit lower in the browser, but achieves PDF alignment
93
- const topAdjustment = (fontBaseLineHeight * fontSize - fontSize) / 2;
94
-
95
- if (verticalAlignment === VERTICAL_ALIGN_TOP) {
96
- return { topAdj: pt2px(topAdjustment), bottomAdj: 0 };
97
- }
98
-
99
- // For vertical alignment bottom and middle
100
- // When browsers render text in a non-form element (such as a <div>), some of the text may be
101
- // lowered below and outside the containing element if the line height used is less than
102
- // the base line-height of the font.
103
- // This behaviour does not happen in a <textarea> though, so we need to adjust the positioning
104
- // for consistency between editing and viewing to stop text jumping up and down.
105
- // This portion of text is half of the difference between the base line height and the used
106
- // line height. If using the same or higher line-height than the base font, then line-height
107
- // takes over in the browser and this adjustment is not needed.
108
- // Unlike the top adjustment - this is only driven by browser behaviour, not PDF alignment.
109
- let bottomAdjustment = 0;
110
- if (lineHeight < fontBaseLineHeight) {
111
- bottomAdjustment = ((fontBaseLineHeight - lineHeight) * fontSize) / 2;
112
- }
113
-
114
- return { topAdj: 0, bottomAdj: pt2px(bottomAdjustment) };
115
- };
116
-
117
- export const getFontDescentInPt = (fontKitFont: FontKitFont, fontSize: number) => {
118
- const { descent, unitsPerEm } = fontKitFont;
119
-
120
- return (descent / unitsPerEm) * fontSize;
121
- };
122
-
123
- export const heightOfFontAtSize = (fontKitFont: FontKitFont, fontSize: number) => {
124
- const { ascent, descent, bbox, unitsPerEm } = fontKitFont;
125
-
126
- const scale = 1000 / unitsPerEm;
127
- const yTop = (ascent || bbox.maxY) * scale;
128
- const yBottom = (descent || bbox.minY) * scale;
129
-
130
- let height = yTop - yBottom;
131
- height -= Math.abs(descent * scale) || 0;
132
-
133
- return (height / 1000) * fontSize;
134
- };
135
-
136
- const calculateCharacterSpacing = (textContent: string, textCharacterSpacing: number) => {
137
- return (textContent.length - 1) * textCharacterSpacing;
138
- };
139
-
140
- export const widthOfTextAtSize = (text: string, fontKitFont: FontKitFont, fontSize: number, characterSpacing: number) => {
141
- const { glyphs } = fontKitFont.layout(text);
142
- const scale = 1000 / fontKitFont.unitsPerEm;
143
- const standardWidth =
144
- glyphs.reduce((totalWidth, glyph) => totalWidth + glyph.advanceWidth * scale, 0) *
145
- (fontSize / 1000);
146
- return standardWidth + calculateCharacterSpacing(text, characterSpacing);
147
- };
148
-
149
- const fontKitFontCache: { [fontName: string]: FontKitFont } = {};
150
- export const getFontKitFont = async (textSchema: TextSchema, font: Font) => {
151
- const fontName = textSchema.fontName || getFallbackFontName(font);
152
- if (fontKitFontCache[fontName]) {
153
- return fontKitFontCache[fontName];
154
- }
155
-
156
- const currentFont = font[fontName] || getFallbackFont(font) || getDefaultFont()[DEFAULT_FONT_NAME];
157
- let fontData = currentFont.data;
158
- if (typeof fontData === 'string') {
159
- fontData = fontData.startsWith('http') ? await fetch(fontData).then((res) => res.arrayBuffer()) : b64toUint8Array(fontData);
160
- }
161
-
162
- const fontKitFont = fontkit.create(fontData instanceof Buffer ? fontData : Buffer.from(fontData as ArrayBuffer));
163
- fontKitFontCache[fontName] = fontKitFont;
164
-
165
- return fontKitFont;
166
- };
167
-
168
-
169
- const isTextExceedingBoxWidth = (text: string, calcValues: FontWidthCalcValues) => {
170
- const { font, fontSize, characterSpacing, boxWidthInPt } = calcValues;
171
- const textWidth = widthOfTextAtSize(text, font, fontSize, characterSpacing);
172
- return textWidth > boxWidthInPt;
173
- };
174
-
175
- /**
176
- * Incrementally checks the current line for its real length
177
- * and returns the position where it exceeds the box width.
178
- * Returns `null` to indicate if textLine is shorter than the available box.
179
- */
180
- const getOverPosition = (textLine: string, calcValues: FontWidthCalcValues) => {
181
- for (let i = 0; i <= textLine.length; i++) {
182
- if (isTextExceedingBoxWidth(textLine.slice(0, i + 1), calcValues)) {
183
- return i;
184
- }
185
- }
186
-
187
- return null;
188
- };
189
-
190
- /**
191
- * Gets the position of the split. Splits the exceeding line at
192
- * the last whitespace prior to it exceeding the bounding box width.
193
- */
194
- const getSplitPosition = (textLine: string, calcValues: FontWidthCalcValues) => {
195
- const overPos = getOverPosition(textLine, calcValues);
196
- if (overPos === null) return textLine.length; // input line is shorter than the available space
197
-
198
- let overPosTmp = overPos;
199
- while (textLine[overPosTmp] !== ' ' && overPosTmp >= 0) {
200
- overPosTmp--;
201
- }
202
-
203
- // For very long lines with no whitespace use the original overPos
204
- return overPosTmp > 0 ? overPosTmp : overPos;
205
- };
206
-
207
- /**
208
- * Recursively splits the line at getSplitPosition.
209
- * If there is some leftover, split the rest again in the same manner.
210
- */
211
- export const getSplittedLines = (textLine: string, calcValues: FontWidthCalcValues): string[] => {
212
- const splitPos = getSplitPosition(textLine, calcValues);
213
- const splittedLine = textLine.substring(0, splitPos);
214
- const rest = textLine.substring(splitPos).trimStart();
215
-
216
- if (rest === textLine) {
217
- // if we went so small that we want to split on the first char
218
- // then end recursion to avoid infinite loop
219
- return [textLine];
220
- }
221
-
222
- if (rest.length === 0) {
223
- // end recursion if there is no leftover
224
- return [splittedLine];
225
- }
226
-
227
- return [splittedLine, ...getSplittedLines(rest, calcValues)];
228
- };
229
-
230
- /**
231
- * If using dynamic font size, iteratively increment or decrement the
232
- * font size to fit the containing box.
233
- * Calculating space usage involves splitting lines where they exceed
234
- * the box width based on the proposed size.
235
- */
236
- export const calculateDynamicFontSize = async ({
237
- textSchema,
238
- font,
239
- value,
240
- startingFontSize,
241
- }: {
242
- textSchema: TextSchema;
243
- font: Font;
244
- value: string;
245
- startingFontSize?: number | undefined;
246
- }) => {
247
- const {
248
- fontSize: schemaFontSize,
249
- dynamicFontSize: dynamicFontSizeSetting,
250
- characterSpacing: schemaCharacterSpacing,
251
- width: boxWidth,
252
- height: boxHeight,
253
- lineHeight = DEFAULT_LINE_HEIGHT,
254
- } = textSchema;
255
- const fontSize = startingFontSize || schemaFontSize || DEFAULT_FONT_SIZE;
256
- if (!dynamicFontSizeSetting) return fontSize;
257
- if (dynamicFontSizeSetting.max < dynamicFontSizeSetting.min) return fontSize;
258
-
259
- const characterSpacing = schemaCharacterSpacing ?? DEFAULT_CHARACTER_SPACING;
260
- const fontKitFont = await getFontKitFont(textSchema, font);
261
- const paragraphs = value.split('\n');
262
-
263
- let dynamicFontSize = fontSize;
264
- if (dynamicFontSize < dynamicFontSizeSetting.min) {
265
- dynamicFontSize = dynamicFontSizeSetting.min;
266
- } else if (dynamicFontSize > dynamicFontSizeSetting.max) {
267
- dynamicFontSize = dynamicFontSizeSetting.max;
268
- }
269
- const dynamicFontFit = dynamicFontSizeSetting.fit ?? DEFAULT_DYNAMIC_FIT;
270
-
271
- const calculateConstraints = (size: number) => {
272
- let totalWidthInMm = 0;
273
- let totalHeightInMm = 0;
274
-
275
- const boxWidthInPt = mm2pt(boxWidth);
276
- const firstLineTextHeight = heightOfFontAtSize(fontKitFont, size);
277
- const firstLineHeightInMm = pt2mm(firstLineTextHeight * lineHeight);
278
- const otherRowHeightInMm = pt2mm(size * lineHeight);
279
-
280
- paragraphs.forEach((paragraph, paraIndex) => {
281
- const lines = getSplittedLines(paragraph, {
282
- font: fontKitFont,
283
- fontSize: size,
284
- characterSpacing,
285
- boxWidthInPt,
286
- });
287
- lines.forEach((line, lineIndex) => {
288
- if (dynamicFontFit === DYNAMIC_FIT_VERTICAL) {
289
- // For vertical fit we want to consider the width of text lines where we detect a split
290
- const textWidth = widthOfTextAtSize(line, fontKitFont, size, characterSpacing);
291
- const textWidthInMm = pt2mm(textWidth);
292
- totalWidthInMm = Math.max(totalWidthInMm, textWidthInMm);
293
- }
294
-
295
- if (paraIndex + lineIndex === 0) {
296
- totalHeightInMm += firstLineHeightInMm;
297
- } else {
298
- totalHeightInMm += otherRowHeightInMm;
299
- }
300
- });
301
- if (dynamicFontFit === DYNAMIC_FIT_HORIZONTAL) {
302
- // For horizontal fit we want to consider the line's width 'unsplit'
303
- const textWidth = widthOfTextAtSize(paragraph, fontKitFont, size, characterSpacing);
304
- const textWidthInMm = pt2mm(textWidth);
305
- totalWidthInMm = Math.max(totalWidthInMm, textWidthInMm);
306
- }
307
- });
308
-
309
- return { totalWidthInMm, totalHeightInMm };
310
- };
311
-
312
- const shouldFontGrowToFit = (totalWidthInMm: number, totalHeightInMm: number) => {
313
- if (dynamicFontSize >= dynamicFontSizeSetting.max) {
314
- return false;
315
- }
316
- if (dynamicFontFit === DYNAMIC_FIT_HORIZONTAL) {
317
- return totalWidthInMm < boxWidth;
318
- }
319
- return totalHeightInMm < boxHeight;
320
- };
321
-
322
- const shouldFontShrinkToFit = (totalWidthInMm: number, totalHeightInMm: number) => {
323
- if (dynamicFontSize <= dynamicFontSizeSetting.min || dynamicFontSize <= 0) {
324
- return false;
325
- }
326
- return totalWidthInMm > boxWidth || totalHeightInMm > boxHeight;
327
- };
328
-
329
- let { totalWidthInMm, totalHeightInMm } = calculateConstraints(dynamicFontSize);
330
-
331
- // Attempt to increase the font size up to desired fit
332
- while (shouldFontGrowToFit(totalWidthInMm, totalHeightInMm)) {
333
- dynamicFontSize += FONT_SIZE_ADJUSTMENT;
334
- const { totalWidthInMm: newWidth, totalHeightInMm: newHeight } = calculateConstraints(dynamicFontSize);
335
-
336
- if (newHeight < boxHeight) {
337
- totalWidthInMm = newWidth;
338
- totalHeightInMm = newHeight;
339
- } else {
340
- dynamicFontSize -= FONT_SIZE_ADJUSTMENT;
341
- break;
342
- }
343
- }
344
-
345
- // Attempt to decrease the font size down to desired fit
346
- while (shouldFontShrinkToFit(totalWidthInMm, totalHeightInMm)) {
347
- dynamicFontSize -= FONT_SIZE_ADJUSTMENT;
348
- ({ totalWidthInMm, totalHeightInMm } = calculateConstraints(dynamicFontSize));
349
- }
350
-
351
- return dynamicFontSize;
352
- };
package/src/type.ts DELETED
@@ -1,69 +0,0 @@
1
- import { z } from 'zod';
2
- import type { Font as FontKitFont } from 'fontkit';
3
-
4
- import {
5
- Lang,
6
- Size,
7
- Alignment,
8
- barcodeSchemaTypes,
9
- schemaTypes as _schemaTypes,
10
- BarcodeSchemaType,
11
- SchemaType,
12
- CommonSchema as _CommonSchema,
13
- TextSchema,
14
- ImageSchema,
15
- BarcodeSchema,
16
- Schema,
17
- SchemaInputs,
18
- SchemaForUI,
19
- Font,
20
- BasePdf,
21
- Template,
22
- CommonProps,
23
- GeneratorOptions,
24
- GenerateProps,
25
- UIOptions,
26
- UIProps,
27
- PreviewProps,
28
- PreviewReactProps,
29
- DesignerProps,
30
- DesignerReactProps,
31
- } from './schema.js';
32
-
33
- export type FontWidthCalcValues = {
34
- font: FontKitFont;
35
- fontSize: number;
36
- characterSpacing: number;
37
- boxWidthInPt: number;
38
- };
39
-
40
- type CommonSchema = z.infer<typeof _CommonSchema>;
41
- export const schemaTypes = _schemaTypes;
42
- export const isTextSchema = (arg: CommonSchema): arg is TextSchema => arg.type === 'text';
43
- export const isImageSchema = (arg: CommonSchema): arg is ImageSchema => arg.type === 'image';
44
- export const isBarcodeSchema = (arg: CommonSchema): arg is BarcodeSchema =>
45
- barcodeSchemaTypes.map((t) => t as string).includes(arg.type);
46
-
47
- export type Lang = z.infer<typeof Lang>;
48
- export type Size = z.infer<typeof Size>;
49
- export type Alignment = z.infer<typeof Alignment>;
50
- export type SchemaType = z.infer<typeof SchemaType>;
51
- export type BarCodeType = z.infer<typeof BarcodeSchemaType>;
52
- export type TextSchema = z.infer<typeof TextSchema>;
53
- export type ImageSchema = z.infer<typeof ImageSchema>;
54
- export type BarcodeSchema = z.infer<typeof BarcodeSchema>;
55
- export type Schema = z.infer<typeof Schema>;
56
- export type SchemaInputs = z.infer<typeof SchemaInputs>;
57
- export type SchemaForUI = z.infer<typeof SchemaForUI>;
58
- export type Font = z.infer<typeof Font>;
59
- export type BasePdf = z.infer<typeof BasePdf>;
60
- export type Template = z.infer<typeof Template>;
61
- export type CommonProps = z.infer<typeof CommonProps>;
62
- export type GeneratorOptions = z.infer<typeof GeneratorOptions>;
63
- export type GenerateProps = z.infer<typeof GenerateProps>;
64
- export type UIOptions = z.infer<typeof UIOptions>;
65
- export type UIProps = z.infer<typeof UIProps>;
66
- export type PreviewProps = z.infer<typeof PreviewProps>;
67
- export type PreviewReactProps = z.infer<typeof PreviewReactProps>;
68
- export type DesignerProps = z.infer<typeof DesignerProps>;
69
- export type DesignerReactProps = z.infer<typeof DesignerReactProps>;