@pdfme/generator 3.4.3 → 4.0.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 (33) hide show
  1. package/dist/cjs/__tests__/generate.test.js +9 -0
  2. package/dist/cjs/__tests__/generate.test.js.map +1 -1
  3. package/dist/cjs/__tests__/integration1.test.js +2 -1
  4. package/dist/cjs/__tests__/integration1.test.js.map +1 -1
  5. package/dist/cjs/__tests__/integration2.test.js +2 -1
  6. package/dist/cjs/__tests__/integration2.test.js.map +1 -1
  7. package/dist/cjs/__tests__/integration3.test.js +2 -1
  8. package/dist/cjs/__tests__/integration3.test.js.map +1 -1
  9. package/dist/cjs/__tests__/integration4.test.js +2 -1
  10. package/dist/cjs/__tests__/integration4.test.js.map +1 -1
  11. package/dist/cjs/src/generate.js +29 -14
  12. package/dist/cjs/src/generate.js.map +1 -1
  13. package/dist/cjs/src/helper.js +55 -41
  14. package/dist/cjs/src/helper.js.map +1 -1
  15. package/dist/esm/__tests__/generate.test.js +9 -0
  16. package/dist/esm/__tests__/generate.test.js.map +1 -1
  17. package/dist/esm/__tests__/integration1.test.js +2 -1
  18. package/dist/esm/__tests__/integration1.test.js.map +1 -1
  19. package/dist/esm/__tests__/integration2.test.js +2 -1
  20. package/dist/esm/__tests__/integration2.test.js.map +1 -1
  21. package/dist/esm/__tests__/integration3.test.js +2 -1
  22. package/dist/esm/__tests__/integration3.test.js.map +1 -1
  23. package/dist/esm/__tests__/integration4.test.js +2 -1
  24. package/dist/esm/__tests__/integration4.test.js.map +1 -1
  25. package/dist/esm/src/generate.js +31 -16
  26. package/dist/esm/src/generate.js.map +1 -1
  27. package/dist/esm/src/helper.js +53 -39
  28. package/dist/esm/src/helper.js.map +1 -1
  29. package/dist/types/__tests__/assets/templates/index.d.ts +2656 -406
  30. package/dist/types/src/helper.d.ts +11 -17
  31. package/package.json +1 -1
  32. package/src/generate.ts +32 -19
  33. package/src/helper.ts +74 -64
@@ -1,32 +1,26 @@
1
- import * as pdfLib from '@pdfme/pdf-lib';
2
- import type { GeneratorOptions, Template, PDFRenderProps } from '@pdfme/common';
3
- import type { Schema, Plugins } from '@pdfme/common';
1
+ import { Schema, Plugins, GeneratorOptions, Template, PDFRenderProps } from '@pdfme/common';
4
2
  import { PDFPage, PDFDocument, PDFEmbeddedPage } from '@pdfme/pdf-lib';
5
- import { BasePdf } from '@pdfme/common';
6
3
  import type { EmbedPdfBox } from './types';
7
- export declare const getEmbeddedPagesAndEmbedPdfBoxes: (arg: {
4
+ export declare const getEmbedPdfPages: (arg: {
5
+ template: Template;
8
6
  pdfDoc: PDFDocument;
9
- basePdf: BasePdf;
10
7
  }) => Promise<{
11
- embeddedPages: pdfLib.PDFEmbeddedPage[];
8
+ basePages: (PDFEmbeddedPage | PDFPage)[];
12
9
  embedPdfBoxes: EmbedPdfBox[];
13
10
  }>;
14
- export declare const drawEmbeddedPage: (arg: {
15
- page: PDFPage;
16
- embeddedPage: PDFEmbeddedPage;
17
- embedPdfBox: EmbedPdfBox;
18
- }) => void;
19
11
  export declare const preprocessing: (arg: {
20
12
  template: Template;
21
13
  userPlugins: Plugins;
22
14
  }) => Promise<{
23
- pdfDoc: pdfLib.PDFDocument;
24
- embeddedPages: pdfLib.PDFEmbeddedPage[];
25
- embedPdfBoxes: EmbedPdfBox[];
15
+ pdfDoc: PDFDocument;
26
16
  renderObj: Record<string, (arg: PDFRenderProps<Schema>) => Promise<void> | void>;
27
- readOnlySchemaKeys: string[];
28
17
  }>;
29
18
  export declare const postProcessing: (props: {
30
- pdfDoc: pdfLib.PDFDocument;
19
+ pdfDoc: PDFDocument;
31
20
  options: GeneratorOptions;
32
21
  }) => void;
22
+ export declare const insertPage: (arg: {
23
+ basePage: PDFEmbeddedPage | PDFPage;
24
+ embedPdfBox: EmbedPdfBox;
25
+ pdfDoc: PDFDocument;
26
+ }) => PDFPage;
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@pdfme/generator",
3
- "version": "3.4.3",
3
+ "version": "4.0.0",
4
4
  "sideEffects": false,
5
5
  "author": "hand-dot",
6
6
  "license": "MIT",
package/src/generate.ts CHANGED
@@ -1,38 +1,51 @@
1
1
  import * as pdfLib from '@pdfme/pdf-lib';
2
2
  import type { GenerateProps } from '@pdfme/common';
3
- import { checkGenerateProps } from '@pdfme/common';
4
- import { drawEmbeddedPage, preprocessing, postProcessing } from './helper.js';
3
+ import { checkGenerateProps, getDynamicTemplate } from '@pdfme/common';
4
+ import { modifyTemplateForTable, getDynamicHeightForTable } from '@pdfme/schemas';
5
+ import { insertPage, preprocessing, postProcessing, getEmbedPdfPages } from './helper.js';
5
6
 
6
7
  const generate = async (props: GenerateProps) => {
7
8
  checkGenerateProps(props);
8
9
  const { inputs, template, options = {}, plugins: userPlugins = {} } = props;
10
+ const basePdf = template.basePdf;
9
11
 
10
12
  if (inputs.length === 0) {
11
13
  throw new Error('inputs should not be empty');
12
14
  }
13
15
 
14
- const { pdfDoc, embeddedPages, embedPdfBoxes, renderObj, readOnlySchemaKeys } =
15
- await preprocessing({ template, userPlugins });
16
-
17
- const keys = readOnlySchemaKeys.concat(Object.keys(inputs[0]));
18
- if (template.columns) {
19
- keys.sort((a, b) => (template.columns ?? []).indexOf(a) - (template.columns ?? []).indexOf(b));
20
- }
16
+ const { pdfDoc, renderObj } = await preprocessing({ template, userPlugins });
21
17
 
22
18
  const _cache = new Map();
19
+
23
20
  for (let i = 0; i < inputs.length; i += 1) {
24
- const inputObj = inputs[i];
25
- for (let j = 0; j < embeddedPages.length; j += 1) {
26
- const embeddedPage = embeddedPages[j];
27
- const { width: pageWidth, height: pageHeight } = embeddedPage;
28
- const embedPdfBox = embedPdfBoxes[j];
21
+ const input = inputs[i];
29
22
 
30
- const page = pdfDoc.addPage([pageWidth, pageHeight]);
23
+ const dynamicTemplate = await getDynamicTemplate({
24
+ template,
25
+ input,
26
+ options,
27
+ _cache,
28
+ modifyTemplate: (arg) => {
29
+ return modifyTemplateForTable(arg);
30
+ },
31
+ getDynamicHeight: (value, args) => {
32
+ if (args.schema.type !== 'table') return Promise.resolve(args.schema.height);
33
+ return getDynamicHeightForTable(value, args);
34
+ },
35
+ });
36
+ const { basePages, embedPdfBoxes } = await getEmbedPdfPages({
37
+ template: dynamicTemplate,
38
+ pdfDoc,
39
+ });
40
+ const keys = dynamicTemplate.schemas.flatMap((schemaObj) => Object.keys(schemaObj));
31
41
 
32
- drawEmbeddedPage({ page, embeddedPage, embedPdfBox });
42
+ for (let j = 0; j < basePages.length; j += 1) {
43
+ const basePage = basePages[j];
44
+ const embedPdfBox = embedPdfBoxes[j];
45
+ const page = insertPage({ basePage, embedPdfBox, pdfDoc });
33
46
  for (let l = 0; l < keys.length; l += 1) {
34
47
  const key = keys[l];
35
- const schemaObj = template.schemas[j];
48
+ const schemaObj = dynamicTemplate.schemas[j] || {};
36
49
  const schema = schemaObj[key];
37
50
  if (!schema) {
38
51
  continue;
@@ -42,8 +55,8 @@ const generate = async (props: GenerateProps) => {
42
55
  if (!render) {
43
56
  continue;
44
57
  }
45
- const value = schema.readOnly ? schema.readOnlyValue || '' : inputObj[key];
46
- await render({ key, value, schema, pdfLib, pdfDoc, page, options, _cache });
58
+ const value = schema.readOnly ? schema.content || '' : input[key];
59
+ await render({ key, value, schema, basePdf, pdfLib, pdfDoc, page, options, _cache });
47
60
  }
48
61
  }
49
62
  }
package/src/helper.ts CHANGED
@@ -1,69 +1,71 @@
1
1
  import * as fontkit from 'fontkit';
2
- import * as pdfLib from '@pdfme/pdf-lib';
3
- import type { GeneratorOptions, Template, PDFRenderProps, Plugin } from '@pdfme/common';
4
- import type { Schema, Plugins } from '@pdfme/common';
2
+ import {
3
+ Schema,
4
+ Plugins,
5
+ GeneratorOptions,
6
+ Template,
7
+ PDFRenderProps,
8
+ Plugin,
9
+ getB64BasePdf,
10
+ isBlankPdf,
11
+ mm2pt,
12
+ } from '@pdfme/common';
5
13
  import { builtInPlugins } from '@pdfme/schemas';
6
14
  import { PDFPage, PDFDocument, PDFEmbeddedPage, TransformationMatrix } from '@pdfme/pdf-lib';
7
- import { getB64BasePdf, BasePdf } from '@pdfme/common';
8
15
  import { TOOL_NAME } from './constants.js';
9
16
  import type { EmbedPdfBox } from './types';
10
17
 
11
- export const getEmbeddedPagesAndEmbedPdfBoxes = async (arg: {
12
- pdfDoc: PDFDocument;
13
- basePdf: BasePdf;
14
- }) => {
15
- const { pdfDoc, basePdf } = arg;
16
- let embeddedPages: PDFEmbeddedPage[] = [];
18
+ export const getEmbedPdfPages = async (arg: { template: Template; pdfDoc: PDFDocument }) => {
19
+ const {
20
+ template: { schemas, basePdf },
21
+ pdfDoc,
22
+ } = arg;
23
+ let basePages: (PDFEmbeddedPage | PDFPage)[] = [];
17
24
  let embedPdfBoxes: EmbedPdfBox[] = [];
18
- const willLoadPdf = typeof basePdf === 'string' ? await getB64BasePdf(basePdf) : basePdf;
19
- const embedPdf = await PDFDocument.load(willLoadPdf);
20
- const embedPdfPages = embedPdf.getPages();
21
-
22
- embedPdfBoxes = embedPdfPages.map((p) => ({
23
- mediaBox: p.getMediaBox(),
24
- bleedBox: p.getBleedBox(),
25
- trimBox: p.getTrimBox(),
26
- }));
27
-
28
- const boundingBoxes = embedPdfPages.map((p) => {
29
- const { x, y, width, height } = p.getMediaBox();
30
-
31
- return { left: x, bottom: y, right: width, top: height + y };
32
- });
33
25
 
34
- const transformationMatrices = embedPdfPages.map(
35
- () => [1, 0, 0, 1, 0, 0] as TransformationMatrix
36
- );
37
-
38
- embeddedPages = await pdfDoc.embedPages(embedPdfPages, boundingBoxes, transformationMatrices);
39
-
40
- return { embeddedPages, embedPdfBoxes };
41
- };
42
-
43
- export const drawEmbeddedPage = (arg: {
44
- page: PDFPage;
45
- embeddedPage: PDFEmbeddedPage;
46
- embedPdfBox: EmbedPdfBox;
47
- }) => {
48
- const { page, embeddedPage, embedPdfBox } = arg;
49
- page.drawPage(embeddedPage);
50
- const { mediaBox: mb, bleedBox: bb, trimBox: tb } = embedPdfBox;
51
- page.setMediaBox(mb.x, mb.y, mb.width, mb.height);
52
- page.setBleedBox(bb.x, bb.y, bb.width, bb.height);
53
- page.setTrimBox(tb.x, tb.y, tb.width, tb.height);
26
+ if (isBlankPdf(basePdf)) {
27
+ const { width: _width, height: _height } = basePdf;
28
+ const width = mm2pt(_width);
29
+ const height = mm2pt(_height);
30
+ basePages = schemas.map(() => {
31
+ const page = PDFPage.create(pdfDoc);
32
+ page.setSize(width, height);
33
+ return page;
34
+ });
35
+ embedPdfBoxes = schemas.map(() => ({
36
+ mediaBox: { x: 0, y: 0, width, height },
37
+ bleedBox: { x: 0, y: 0, width, height },
38
+ trimBox: { x: 0, y: 0, width, height },
39
+ }));
40
+ } else {
41
+ const willLoadPdf = typeof basePdf === 'string' ? await getB64BasePdf(basePdf) : basePdf;
42
+ const embedPdf = await PDFDocument.load(willLoadPdf as ArrayBuffer | Uint8Array | string);
43
+ const embedPdfPages = embedPdf.getPages();
44
+ embedPdfBoxes = embedPdfPages.map((p) => ({
45
+ mediaBox: p.getMediaBox(),
46
+ bleedBox: p.getBleedBox(),
47
+ trimBox: p.getTrimBox(),
48
+ }));
49
+ const boundingBoxes = embedPdfPages.map((p) => {
50
+ const { x, y, width, height } = p.getMediaBox();
51
+ return { left: x, bottom: y, right: width, top: height + y };
52
+ });
53
+ const transformationMatrices = embedPdfPages.map(
54
+ () => [1, 0, 0, 1, 0, 0] as TransformationMatrix
55
+ );
56
+ basePages = await pdfDoc.embedPages(embedPdfPages, boundingBoxes, transformationMatrices);
57
+ }
58
+ return { basePages, embedPdfBoxes };
54
59
  };
55
60
 
56
61
  export const preprocessing = async (arg: { template: Template; userPlugins: Plugins }) => {
57
62
  const { template, userPlugins } = arg;
58
- const { basePdf, schemas } = template;
63
+ const { schemas } = template;
59
64
 
60
- const pdfDoc = await pdfLib.PDFDocument.create();
65
+ const pdfDoc = await PDFDocument.create();
61
66
  // @ts-ignore
62
67
  pdfDoc.registerFontkit(fontkit);
63
68
 
64
- const pagesAndBoxes = await getEmbeddedPagesAndEmbedPdfBoxes({ pdfDoc, basePdf });
65
- const { embeddedPages, embedPdfBoxes } = pagesAndBoxes;
66
-
67
69
  const pluginValues = (
68
70
  Object.values(userPlugins).length > 0
69
71
  ? Object.values(userPlugins)
@@ -84,22 +86,10 @@ Check this document: https://pdfme.com/docs/custom-schemas`);
84
86
  return { ...acc, [type]: render.pdf };
85
87
  }, {} as Record<string, (arg: PDFRenderProps<Schema>) => Promise<void> | void>);
86
88
 
87
- const readOnlySchemaKeys = schemas.reduce((acc, schema) => {
88
- const entries = Object.entries(schema);
89
- const keys = entries.reduce(
90
- (acc, [key, value]) => (value.readOnly ? [...acc, key] : acc),
91
- [] as string[]
92
- );
93
- return [...acc, ...keys];
94
- }, [] as string[]);
95
-
96
- return { pdfDoc, embeddedPages, embedPdfBoxes, renderObj, readOnlySchemaKeys };
89
+ return { pdfDoc, renderObj };
97
90
  };
98
91
 
99
- export const postProcessing = (props: {
100
- pdfDoc: pdfLib.PDFDocument;
101
- options: GeneratorOptions;
102
- }) => {
92
+ export const postProcessing = (props: { pdfDoc: PDFDocument; options: GeneratorOptions }) => {
103
93
  const { pdfDoc, options } = props;
104
94
  const {
105
95
  author = TOOL_NAME,
@@ -122,3 +112,23 @@ export const postProcessing = (props: {
122
112
  pdfDoc.setSubject(subject);
123
113
  pdfDoc.setTitle(title);
124
114
  };
115
+
116
+ export const insertPage = (arg: {
117
+ basePage: PDFEmbeddedPage | PDFPage;
118
+ embedPdfBox: EmbedPdfBox;
119
+ pdfDoc: PDFDocument;
120
+ }) => {
121
+ const { basePage, embedPdfBox, pdfDoc } = arg;
122
+ const size = basePage instanceof PDFEmbeddedPage ? basePage.size() : basePage.getSize();
123
+ const insertedPage = pdfDoc.addPage([size.width, size.height]);
124
+
125
+ if (basePage instanceof PDFEmbeddedPage) {
126
+ insertedPage.drawPage(basePage);
127
+ const { mediaBox, bleedBox, trimBox } = embedPdfBox;
128
+ insertedPage.setMediaBox(mediaBox.x, mediaBox.y, mediaBox.width, mediaBox.height);
129
+ insertedPage.setBleedBox(bleedBox.x, bleedBox.y, bleedBox.width, bleedBox.height);
130
+ insertedPage.setTrimBox(trimBox.x, trimBox.y, trimBox.width, trimBox.height);
131
+ }
132
+
133
+ return insertedPage;
134
+ };