@lichens-innovation/ts-common 1.7.0 → 1.9.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/mime.js ADDED
@@ -0,0 +1,14 @@
1
+ import mime from 'mime';
2
+
3
+ // src/mime/index.ts
4
+ var getMimeType = (extension) => {
5
+ const mimeType = mime.getType(extension);
6
+ if (!mimeType) {
7
+ throw new Error(`[getMimeType] Mime type not found for extension: "${extension}"`);
8
+ }
9
+ return mimeType;
10
+ };
11
+
12
+ export { getMimeType };
13
+ //# sourceMappingURL=mime.js.map
14
+ //# sourceMappingURL=mime.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/mime/index.ts"],"names":[],"mappings":";;;AAEO,IAAM,WAAA,GAAc,CAAC,SAAA,KAA8B;AACxD,EAAA,MAAM,QAAA,GAAW,IAAA,CAAK,OAAA,CAAQ,SAAS,CAAA;AACvC,EAAA,IAAI,CAAC,QAAA,EAAU;AACb,IAAA,MAAM,IAAI,KAAA,CAAM,CAAA,kDAAA,EAAqD,SAAS,CAAA,CAAA,CAAG,CAAA;AAAA,EACnF;AAEA,EAAA,OAAO,QAAA;AACT","file":"mime.js","sourcesContent":["import mime from \"mime\";\n\nexport const getMimeType = (extension: string): string => {\n const mimeType = mime.getType(extension);\n if (!mimeType) {\n throw new Error(`[getMimeType] Mime type not found for extension: \"${extension}\"`);\n }\n\n return mimeType;\n};\n"]}
package/dist/pdf.cjs ADDED
@@ -0,0 +1,273 @@
1
+ 'use strict';
2
+
3
+ var jspdf = require('jspdf');
4
+ var jspdfAutotable = require('jspdf-autotable');
5
+ var dateFns = require('date-fns');
6
+
7
+ // src/utils/types.utils.ts
8
+ var isNullish = (value) => value === null || value === void 0;
9
+ var DEFAULT_REPORT_FILENAME = "report.pdf";
10
+ var IMAGE_GAP = 0.5;
11
+ var DEFAULT_THEME = {
12
+ primaryColor: { r: 0, g: 0, b: 0 },
13
+ // black
14
+ fontSize: {
15
+ title: 16,
16
+ header: 10,
17
+ normal: 9,
18
+ small: 8,
19
+ tiny: 7
20
+ }
21
+ };
22
+ var DEFAULT_FOOTER_CELL_BUILDER = ({ align, currentPage, totalPages }) => {
23
+ if (align === "left") return dateFns.format(/* @__PURE__ */ new Date(), "yyyy-MM-dd HH:mm");
24
+ if (align === "center") return "";
25
+ return `Page ${currentPage}/${totalPages}`;
26
+ };
27
+ var DEFAULT_OPTIONS = {
28
+ orientation: "portrait",
29
+ paperFormat: "letter",
30
+ filename: DEFAULT_REPORT_FILENAME,
31
+ theme: DEFAULT_THEME,
32
+ margin: 0.5,
33
+ displayAvailableAreaRectangle: false,
34
+ hasFooter: true,
35
+ footerCellBuilder: DEFAULT_FOOTER_CELL_BUILDER
36
+ };
37
+ var Gaps = {
38
+ TINY: 0.02,
39
+ SMALL: 0.04,
40
+ MEDIUM: 0.08,
41
+ LARGE: 0.16,
42
+ X_LARGE: 0.32
43
+ };
44
+ var PdfColors = {
45
+ LIGHT_GRAY: [240, 240, 240]
46
+ };
47
+
48
+ // src/pdf/pdf-generator.ts
49
+ var PdfGenerator = class {
50
+ doc;
51
+ options;
52
+ _currentY;
53
+ constructor(options = {}) {
54
+ this.options = {
55
+ ...DEFAULT_OPTIONS,
56
+ ...options
57
+ };
58
+ const { orientation, paperFormat } = this.options;
59
+ this.doc = new jspdf.jsPDF({ orientation, format: paperFormat, unit: "in" });
60
+ this.doc.setFont("helvetica");
61
+ this._currentY = this.margin;
62
+ }
63
+ get font() {
64
+ return this.doc.getFont().fontName;
65
+ }
66
+ set font(font) {
67
+ this.doc.setFont(font);
68
+ }
69
+ get theme() {
70
+ return this.options.theme;
71
+ }
72
+ get filename() {
73
+ return this.options.filename;
74
+ }
75
+ get margin() {
76
+ return this.options.margin;
77
+ }
78
+ get pageSize() {
79
+ return this.doc.internal.pageSize;
80
+ }
81
+ get pageWidth() {
82
+ return this.pageSize.width;
83
+ }
84
+ get pageHeight() {
85
+ return this.pageSize.height;
86
+ }
87
+ get fullImageHeight() {
88
+ return this.availableHeight - this.footerHeight - IMAGE_GAP;
89
+ }
90
+ get availableWidth() {
91
+ return this.pageWidth - 2 * this.margin;
92
+ }
93
+ get availableHeight() {
94
+ return this.pageHeight - 2 * this.margin;
95
+ }
96
+ get footerFontSizePoints() {
97
+ return this.theme.fontSize.tiny;
98
+ }
99
+ /** Current Y position (in doc units) for layout. */
100
+ get currentY() {
101
+ return this._currentY;
102
+ }
103
+ /** Set current Y position (e.g. after custom content like header or chart). */
104
+ setCurrentY(y) {
105
+ this._currentY = y;
106
+ }
107
+ addVerticalGap(delta) {
108
+ this._currentY += delta;
109
+ }
110
+ checkTableOverflow(columnWidths) {
111
+ const computedSum = columnWidths.reduce((acc, width) => acc + width, 0);
112
+ if (computedSum <= this.availableWidth) return;
113
+ const infos = {
114
+ availableWidth: this.availableWidth,
115
+ computedSum,
116
+ columnWidths: JSON.stringify(columnWidths)
117
+ };
118
+ console.warn("[checkTableOverflow] table overflow", infos);
119
+ }
120
+ addTable(addTableArgs) {
121
+ const {
122
+ lineWidth = 5e-3,
123
+ lineColor = 0,
124
+ drawLineForTable = true,
125
+ drawLineForCells = true,
126
+ head,
127
+ body,
128
+ startY = this._currentY,
129
+ columnWidths
130
+ } = addTableArgs;
131
+ if (body.length === 0) {
132
+ this._currentY = startY;
133
+ return startY;
134
+ }
135
+ const cellPadding = 0.05;
136
+ const fontSize = this.theme.fontSize.small;
137
+ const colCount = columnWidths?.length ?? Math.max(...body.map((row) => row.length), 0);
138
+ const widths = columnWidths ?? Array(colCount).fill(this.availableWidth / colCount);
139
+ const columnStyles = {};
140
+ widths.forEach((cellWidth, i) => {
141
+ columnStyles[i] = { cellWidth, ...addTableArgs.columnStyles?.[i] };
142
+ });
143
+ const tableLineWidth = drawLineForTable ? lineWidth : 0;
144
+ const cellLineWidth = drawLineForCells ? lineWidth : 0;
145
+ jspdfAutotable.autoTable(this.doc, {
146
+ startY,
147
+ head: !isNullish(head) ? [head] : void 0,
148
+ body,
149
+ theme: "grid",
150
+ tableLineWidth,
151
+ tableLineColor: lineColor,
152
+ margin: { left: this.margin, right: this.margin },
153
+ bodyStyles: { fontSize, font: this.font, cellPadding, lineWidth: cellLineWidth, lineColor },
154
+ headStyles: { fontSize, font: this.font, cellPadding, lineWidth: cellLineWidth, lineColor },
155
+ columnStyles
156
+ });
157
+ const finalY = this.doc.lastAutoTable?.finalY ?? startY;
158
+ this._currentY = finalY;
159
+ }
160
+ /**
161
+ * Add an image at the given position and size. Does not update currentY.
162
+ */
163
+ addImage({ dataUri, x, y, width, height }) {
164
+ const imageFormat = "PNG";
165
+ const imageCompression = "NONE";
166
+ this.doc.addImage(dataUri, imageFormat, x, y, width, height, void 0, imageCompression);
167
+ }
168
+ /** Exposes the jsPDF document for custom drawing (e.g. header text, status badge, certification). */
169
+ getDoc() {
170
+ return this.doc;
171
+ }
172
+ addPage() {
173
+ this.doc.addPage();
174
+ }
175
+ addFullPagePNG(dataURI, aspectRatio) {
176
+ const imageFormat = "PNG";
177
+ const imageCompression = "NONE";
178
+ const alias = void 0;
179
+ const x = this.margin;
180
+ const y = this.margin;
181
+ const { width, height } = this.calculateImageDimensions(aspectRatio);
182
+ this.doc.addImage(dataURI, imageFormat, x, y, width, height, alias, imageCompression);
183
+ this._currentY = this.margin + height + IMAGE_GAP;
184
+ }
185
+ calculateImageDimensions(aspectRatio) {
186
+ const width = this.availableWidth;
187
+ const height = width / aspectRatio;
188
+ const maxHeight = this.availableHeight - this.footerHeight - IMAGE_GAP;
189
+ const finalHeight = Math.min(height, maxHeight);
190
+ const finalWidth = finalHeight < height ? finalHeight * aspectRatio : width;
191
+ return { width: finalWidth, height: finalHeight };
192
+ }
193
+ save() {
194
+ this.doc.save(this.filename);
195
+ }
196
+ buildFooterColumnStyles() {
197
+ const cellWidth = this.availableWidth / 3;
198
+ return {
199
+ 0: { cellWidth, halign: "left" },
200
+ 1: { cellWidth, halign: "center" },
201
+ 2: { cellWidth, halign: "right" }
202
+ };
203
+ }
204
+ renderFooters() {
205
+ if (!this.options.hasFooter) {
206
+ console.warn("[renderFooters] Footer is disabled");
207
+ return;
208
+ }
209
+ const totalPages = this.doc.getNumberOfPages();
210
+ const columnStyles = this.buildFooterColumnStyles();
211
+ const startY = this.pageHeight - this.margin - this.footerHeight;
212
+ const footerCellBuilder = this.options.footerCellBuilder ?? DEFAULT_FOOTER_CELL_BUILDER;
213
+ for (let currentPage = 1; currentPage <= totalPages; currentPage++) {
214
+ this.doc.setPage(currentPage);
215
+ this.drawAvailableAreaRectangle();
216
+ const footerCells = ["left", "center", "right"].map((align) => ({
217
+ content: footerCellBuilder({ align, currentPage, totalPages })
218
+ }));
219
+ jspdfAutotable.autoTable(this.doc, {
220
+ startY,
221
+ body: [footerCells],
222
+ theme: "plain",
223
+ bodyStyles: { fontSize: this.footerFontSizePoints, font: this.font },
224
+ margin: { left: this.margin, right: this.margin },
225
+ columnStyles
226
+ });
227
+ }
228
+ }
229
+ /** Footer is a single line; height derived from font size and table padding. */
230
+ get footerHeight() {
231
+ if (!this.options.hasFooter) {
232
+ return 0;
233
+ }
234
+ const fontSizeInches = this.footerFontSizePoints / 72;
235
+ const lineHeight = fontSizeInches * 1.5;
236
+ return lineHeight + 0.2;
237
+ }
238
+ drawAvailableAreaRectangle() {
239
+ if (!this.options.displayAvailableAreaRectangle) {
240
+ return;
241
+ }
242
+ const currentDrawColor = this.doc.getDrawColor();
243
+ const currentLineWidth = this.doc.getLineWidth();
244
+ try {
245
+ this.doc.setDrawColor(200, 200, 200);
246
+ this.doc.setLineWidth(0.01);
247
+ this.doc.rect(
248
+ this.margin,
249
+ this.margin,
250
+ this.availableWidth,
251
+ this.availableHeight,
252
+ "S"
253
+ // Stroke only, no fill
254
+ );
255
+ } catch (error) {
256
+ console.error("[drawAvailableAreaRectangle] Error drawing available area rect", error);
257
+ } finally {
258
+ this.doc.setDrawColor(currentDrawColor);
259
+ this.doc.setLineWidth(currentLineWidth);
260
+ }
261
+ }
262
+ };
263
+
264
+ exports.DEFAULT_FOOTER_CELL_BUILDER = DEFAULT_FOOTER_CELL_BUILDER;
265
+ exports.DEFAULT_OPTIONS = DEFAULT_OPTIONS;
266
+ exports.DEFAULT_REPORT_FILENAME = DEFAULT_REPORT_FILENAME;
267
+ exports.DEFAULT_THEME = DEFAULT_THEME;
268
+ exports.Gaps = Gaps;
269
+ exports.IMAGE_GAP = IMAGE_GAP;
270
+ exports.PdfColors = PdfColors;
271
+ exports.PdfGenerator = PdfGenerator;
272
+ //# sourceMappingURL=pdf.cjs.map
273
+ //# sourceMappingURL=pdf.cjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/utils/types.utils.ts","../src/pdf/pdf-generator.types.ts","../src/pdf/pdf-generator.ts"],"names":["format","jsPDF","autoTable"],"mappings":"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","file":"pdf.cjs","sourcesContent":["export const NO_OP: () => void = () => {};\n\nexport const isNullish = (value: unknown): value is null | undefined => value === null || value === undefined;\n\nexport const isNumber = (value?: unknown | null): value is number => {\n if (isNullish(value)) {\n return false;\n }\n\n if (typeof value !== 'number') {\n return false;\n }\n\n if (isNaN(value)) {\n return false;\n }\n\n return true;\n};\n\nexport const isString = (value?: unknown | null): value is string => {\n if (isNullish(value)) {\n return false;\n }\n\n if (typeof value !== 'string') {\n return false;\n }\n\n return true;\n};\n","import { format } from \"date-fns\";\nimport type { jsPDF, jsPDFOptions } from \"jspdf\";\n\ninterface FontSize {\n title: number;\n header: number;\n normal: number;\n small: number;\n tiny: number;\n}\n\ninterface RgbColor {\n r: number;\n g: number;\n b: number;\n}\n\nexport interface ThemeConfig {\n primaryColor: RgbColor;\n fontSize: FontSize;\n}\n\nexport type PageSize = jsPDF[\"internal\"][\"pageSize\"];\nexport type PageOrientation = jsPDFOptions[\"orientation\"];\nexport type PaperFormat = jsPDFOptions[\"format\"];\n\nexport const DEFAULT_REPORT_FILENAME = \"report.pdf\";\nexport const IMAGE_GAP = 0.5; // space in inches between image and footer\n\nexport const DEFAULT_THEME: ThemeConfig = {\n primaryColor: { r: 0, g: 0, b: 0 }, // black\n fontSize: {\n title: 16,\n header: 10,\n normal: 9,\n small: 8,\n tiny: 7,\n },\n};\n\nexport interface FooterCellArgs {\n align: TableCellHalign;\n currentPage: number;\n totalPages: number;\n}\nexport type FooterCellBuilder = (footerCellArgs: FooterCellArgs) => string;\n\nexport interface PdfOptions {\n orientation?: PageOrientation;\n paperFormat?: PaperFormat;\n hasFooter?: boolean;\n filename?: string;\n theme?: ThemeConfig;\n margin?: number;\n displayAvailableAreaRectangle?: boolean;\n footerCellBuilder?: FooterCellBuilder;\n}\n\nexport const DEFAULT_FOOTER_CELL_BUILDER = ({ align, currentPage, totalPages }: FooterCellArgs): string => {\n if (align === \"left\") return format(new Date(), \"yyyy-MM-dd HH:mm\");\n if (align === \"center\") return \"\";\n return `Page ${currentPage}/${totalPages}`;\n};\n\nexport const DEFAULT_OPTIONS: Readonly<Required<PdfOptions>> = {\n orientation: \"portrait\",\n paperFormat: \"letter\",\n filename: DEFAULT_REPORT_FILENAME,\n theme: DEFAULT_THEME,\n margin: 0.5,\n displayAvailableAreaRectangle: false,\n hasFooter: true,\n footerCellBuilder: DEFAULT_FOOTER_CELL_BUILDER,\n};\n\nexport const Gaps = {\n TINY: 0.02,\n SMALL: 0.04,\n MEDIUM: 0.08,\n LARGE: 0.16,\n X_LARGE: 0.32,\n};\n\nexport const PdfColors: Record<string, FillColorRgbTuple> = {\n LIGHT_GRAY: [240, 240, 240],\n};\n\nexport type TableCellHalign = \"left\" | \"center\" | \"right\";\nexport type TableCellValign = \"top\" | \"middle\" | \"bottom\";\nexport type FontStyle = \"normal\" | \"bold\";\nexport type FillColorRgbTuple = [number, number, number];\n\nexport type CellStyle = {\n cellWidth?: number;\n halign?: TableCellHalign;\n valign?: TableCellValign;\n fontStyle?: FontStyle;\n fillColor?: FillColorRgbTuple;\n};\n\nexport type CellValue = string | number;\n\n/** Cell with rowSpan/colSpan for merged cells (passed through to autoTable). */\nexport interface TableCellDef {\n content: string | string[] | number;\n rowSpan?: number;\n colSpan?: number;\n}\n\nexport type TableCellInput = CellValue | TableCellDef;\n\n/** Parameters for adding a table (generic API, no jspdf-autotable leak). */\nexport interface AddTableArgs {\n /** Each row is an array of cell values or cell defs (with rowSpan/colSpan). */\n body: TableCellInput[][];\n /** Optional header row (single row of cell values). */\n head?: CellValue[];\n /** Y position to start the table (uses generator currentY if omitted). */\n startY?: number;\n\n /** Optional column widths (in same units as doc, e.g. inches). Sum should match available width. */\n columnWidths?: number[];\n /** Optional per-column styles (e.g. halign, fontStyle). Merged with auto-generated cellWidth. */\n columnStyles?: Record<number, CellStyle>;\n\n /** Whether to draw the table outer border (default true). */\n drawLineForTable?: boolean;\n /** Whether to draw cell borders (default true). */\n drawLineForCells?: boolean;\n /** Table border line width (default 0.005). */\n lineWidth?: number;\n /** Table border color as a grayscale value: 0 = black, 200 = light gray, 255 = white (default 0). */\n lineColor?: number;\n}\n\nexport interface AddImageArgs {\n dataUri: string;\n x: number;\n y: number;\n width: number;\n height: number;\n}\n","import { isNullish, type Dimensions } from \"../utils\";\nimport { jsPDF, type ImageCompression, type ImageFormat } from \"jspdf\";\nimport { autoTable } from \"jspdf-autotable\";\nimport {\n DEFAULT_FOOTER_CELL_BUILDER,\n DEFAULT_OPTIONS,\n IMAGE_GAP,\n type AddImageArgs,\n type AddTableArgs,\n type CellStyle,\n type PageSize,\n type PdfOptions,\n type ThemeConfig,\n} from \"./pdf-generator.types\";\n\nexport class PdfGenerator {\n private doc: jsPDF;\n private options: Required<PdfOptions>;\n private _currentY: number;\n\n constructor(options: PdfOptions = {}) {\n this.options = {\n ...DEFAULT_OPTIONS,\n ...options,\n };\n\n const { orientation, paperFormat } = this.options;\n this.doc = new jsPDF({ orientation, format: paperFormat, unit: \"in\" });\n this.doc.setFont(\"helvetica\");\n\n this._currentY = this.margin;\n }\n\n get font(): string {\n return this.doc.getFont().fontName;\n }\n\n set font(font: string) {\n this.doc.setFont(font);\n }\n\n get theme(): ThemeConfig {\n return this.options.theme;\n }\n\n get filename(): string {\n return this.options.filename;\n }\n\n get margin(): number {\n return this.options.margin;\n }\n\n get pageSize(): PageSize {\n return this.doc.internal.pageSize;\n }\n\n get pageWidth(): number {\n return this.pageSize.width;\n }\n\n get pageHeight(): number {\n return this.pageSize.height;\n }\n\n get fullImageHeight(): number {\n return this.availableHeight - this.footerHeight - IMAGE_GAP;\n }\n\n get availableWidth(): number {\n return this.pageWidth - 2 * this.margin;\n }\n\n get availableHeight(): number {\n return this.pageHeight - 2 * this.margin;\n }\n\n get footerFontSizePoints(): number {\n return this.theme.fontSize.tiny;\n }\n\n /** Current Y position (in doc units) for layout. */\n get currentY(): number {\n return this._currentY;\n }\n\n /** Set current Y position (e.g. after custom content like header or chart). */\n public setCurrentY(y: number): void {\n this._currentY = y;\n }\n\n public addVerticalGap(delta: number): void {\n this._currentY += delta;\n }\n\n public checkTableOverflow(columnWidths: number[]): void {\n const computedSum = columnWidths.reduce((acc, width) => acc + width, 0);\n if (computedSum <= this.availableWidth) return;\n\n const infos = {\n availableWidth: this.availableWidth,\n computedSum,\n columnWidths: JSON.stringify(columnWidths),\n };\n console.warn(\"[checkTableOverflow] table overflow\", infos);\n }\n\n public addTable(addTableArgs: AddTableArgs) {\n const {\n lineWidth = 0.005,\n lineColor = 0,\n drawLineForTable = true,\n drawLineForCells = true,\n head,\n body,\n startY = this._currentY,\n columnWidths,\n } = addTableArgs;\n\n if (body.length === 0) {\n this._currentY = startY;\n return startY;\n }\n\n const cellPadding = 0.05;\n const fontSize = this.theme.fontSize.small;\n const colCount = columnWidths?.length ?? Math.max(...body.map((row) => row.length), 0);\n const widths = columnWidths ?? Array(colCount).fill(this.availableWidth / colCount);\n const columnStyles: Record<number, CellStyle> = {};\n widths.forEach((cellWidth, i) => {\n columnStyles[i] = { cellWidth, ...addTableArgs.columnStyles?.[i] };\n });\n\n const tableLineWidth = drawLineForTable ? lineWidth : 0;\n const cellLineWidth = drawLineForCells ? lineWidth : 0;\n\n autoTable(this.doc, {\n startY,\n head: !isNullish(head) ? [head] : undefined,\n body,\n theme: \"grid\",\n tableLineWidth,\n tableLineColor: lineColor,\n margin: { left: this.margin, right: this.margin },\n bodyStyles: { fontSize, font: this.font, cellPadding, lineWidth: cellLineWidth, lineColor },\n headStyles: { fontSize, font: this.font, cellPadding, lineWidth: cellLineWidth, lineColor },\n columnStyles,\n });\n\n const finalY = (this.doc as unknown as { lastAutoTable: { finalY: number } }).lastAutoTable?.finalY ?? startY;\n this._currentY = finalY;\n }\n\n /**\n * Add an image at the given position and size. Does not update currentY.\n */\n public addImage({ dataUri, x, y, width, height }: AddImageArgs): void {\n const imageFormat: ImageFormat = \"PNG\";\n const imageCompression: ImageCompression = \"NONE\";\n this.doc.addImage(dataUri, imageFormat, x, y, width, height, undefined, imageCompression);\n }\n\n /** Exposes the jsPDF document for custom drawing (e.g. header text, status badge, certification). */\n public getDoc(): jsPDF {\n return this.doc;\n }\n\n public addPage() {\n this.doc.addPage();\n }\n\n public addFullPagePNG(dataURI: string, aspectRatio: number): void {\n const imageFormat: ImageFormat = \"PNG\";\n const imageCompression: ImageCompression = \"NONE\";\n\n const alias = undefined;\n const x = this.margin;\n const y = this.margin;\n const { width, height } = this.calculateImageDimensions(aspectRatio);\n\n this.doc.addImage(dataURI, imageFormat, x, y, width, height, alias, imageCompression);\n\n this._currentY = this.margin + height + IMAGE_GAP;\n }\n\n private calculateImageDimensions(aspectRatio: number): Dimensions {\n const width = this.availableWidth;\n\n // Calculate height based on canvas aspect ratio to maintain proper proportions\n const height = width / aspectRatio;\n\n // Ensure the image doesn't exceed available space (accounting for footer)\n const maxHeight = this.availableHeight - this.footerHeight - IMAGE_GAP;\n const finalHeight = Math.min(height, maxHeight);\n\n // If image is reduced, adjust width to maintain aspect ratio\n const finalWidth = finalHeight < height ? finalHeight * aspectRatio : width;\n\n return { width: finalWidth, height: finalHeight };\n }\n\n public save(): void {\n this.doc.save(this.filename);\n }\n\n private buildFooterColumnStyles() {\n const cellWidth = this.availableWidth / 3;\n\n return {\n 0: { cellWidth, halign: \"left\" as const },\n 1: { cellWidth, halign: \"center\" as const },\n 2: { cellWidth, halign: \"right\" as const },\n };\n }\n\n public renderFooters(): void {\n if (!this.options.hasFooter) {\n console.warn(\"[renderFooters] Footer is disabled\");\n return;\n }\n\n const totalPages = this.doc.getNumberOfPages();\n const columnStyles = this.buildFooterColumnStyles();\n const startY = this.pageHeight - this.margin - this.footerHeight;\n\n const footerCellBuilder = this.options.footerCellBuilder ?? DEFAULT_FOOTER_CELL_BUILDER;\n\n for (let currentPage = 1; currentPage <= totalPages; currentPage++) {\n this.doc.setPage(currentPage);\n this.drawAvailableAreaRectangle();\n const footerCells = ([\"left\", \"center\", \"right\"] as const).map((align) => ({\n content: footerCellBuilder({ align, currentPage, totalPages }),\n }));\n\n autoTable(this.doc, {\n startY,\n body: [footerCells],\n theme: \"plain\",\n bodyStyles: { fontSize: this.footerFontSizePoints, font: this.font },\n margin: { left: this.margin, right: this.margin },\n columnStyles,\n });\n }\n }\n\n /** Footer is a single line; height derived from font size and table padding. */\n get footerHeight(): number {\n if (!this.options.hasFooter) {\n return 0;\n }\n\n const fontSizeInches = this.footerFontSizePoints / 72; // 1 inch = 72 points\n const lineHeight = fontSizeInches * 1.5; // autoTable line height\n return lineHeight + 0.2; // table borders and minimal padding\n }\n\n private drawAvailableAreaRectangle(): void {\n if (!this.options.displayAvailableAreaRectangle) {\n return;\n }\n\n // Save current style state\n const currentDrawColor = this.doc.getDrawColor();\n const currentLineWidth = this.doc.getLineWidth();\n\n try {\n this.doc.setDrawColor(200, 200, 200); // Light gray\n this.doc.setLineWidth(0.01);\n this.doc.rect(\n this.margin,\n this.margin,\n this.availableWidth,\n this.availableHeight,\n \"S\" // Stroke only, no fill\n );\n } catch (error) {\n console.error(\"[drawAvailableAreaRectangle] Error drawing available area rect\", error);\n } finally {\n // Restore previous state\n this.doc.setDrawColor(currentDrawColor);\n this.doc.setLineWidth(currentLineWidth);\n }\n }\n}\n"]}
package/dist/pdf.d.cts ADDED
@@ -0,0 +1,140 @@
1
+ import { jsPDFOptions, jsPDF } from 'jspdf';
2
+
3
+ interface FontSize {
4
+ title: number;
5
+ header: number;
6
+ normal: number;
7
+ small: number;
8
+ tiny: number;
9
+ }
10
+ interface RgbColor {
11
+ r: number;
12
+ g: number;
13
+ b: number;
14
+ }
15
+ interface ThemeConfig {
16
+ primaryColor: RgbColor;
17
+ fontSize: FontSize;
18
+ }
19
+ type PageSize = jsPDF["internal"]["pageSize"];
20
+ type PageOrientation = jsPDFOptions["orientation"];
21
+ type PaperFormat = jsPDFOptions["format"];
22
+ declare const DEFAULT_REPORT_FILENAME = "report.pdf";
23
+ declare const IMAGE_GAP = 0.5;
24
+ declare const DEFAULT_THEME: ThemeConfig;
25
+ interface FooterCellArgs {
26
+ align: TableCellHalign;
27
+ currentPage: number;
28
+ totalPages: number;
29
+ }
30
+ type FooterCellBuilder = (footerCellArgs: FooterCellArgs) => string;
31
+ interface PdfOptions {
32
+ orientation?: PageOrientation;
33
+ paperFormat?: PaperFormat;
34
+ hasFooter?: boolean;
35
+ filename?: string;
36
+ theme?: ThemeConfig;
37
+ margin?: number;
38
+ displayAvailableAreaRectangle?: boolean;
39
+ footerCellBuilder?: FooterCellBuilder;
40
+ }
41
+ declare const DEFAULT_FOOTER_CELL_BUILDER: ({ align, currentPage, totalPages }: FooterCellArgs) => string;
42
+ declare const DEFAULT_OPTIONS: Readonly<Required<PdfOptions>>;
43
+ declare const Gaps: {
44
+ TINY: number;
45
+ SMALL: number;
46
+ MEDIUM: number;
47
+ LARGE: number;
48
+ X_LARGE: number;
49
+ };
50
+ declare const PdfColors: Record<string, FillColorRgbTuple>;
51
+ type TableCellHalign = "left" | "center" | "right";
52
+ type TableCellValign = "top" | "middle" | "bottom";
53
+ type FontStyle = "normal" | "bold";
54
+ type FillColorRgbTuple = [number, number, number];
55
+ type CellStyle = {
56
+ cellWidth?: number;
57
+ halign?: TableCellHalign;
58
+ valign?: TableCellValign;
59
+ fontStyle?: FontStyle;
60
+ fillColor?: FillColorRgbTuple;
61
+ };
62
+ type CellValue = string | number;
63
+ /** Cell with rowSpan/colSpan for merged cells (passed through to autoTable). */
64
+ interface TableCellDef {
65
+ content: string | string[] | number;
66
+ rowSpan?: number;
67
+ colSpan?: number;
68
+ }
69
+ type TableCellInput = CellValue | TableCellDef;
70
+ /** Parameters for adding a table (generic API, no jspdf-autotable leak). */
71
+ interface AddTableArgs {
72
+ /** Each row is an array of cell values or cell defs (with rowSpan/colSpan). */
73
+ body: TableCellInput[][];
74
+ /** Optional header row (single row of cell values). */
75
+ head?: CellValue[];
76
+ /** Y position to start the table (uses generator currentY if omitted). */
77
+ startY?: number;
78
+ /** Optional column widths (in same units as doc, e.g. inches). Sum should match available width. */
79
+ columnWidths?: number[];
80
+ /** Optional per-column styles (e.g. halign, fontStyle). Merged with auto-generated cellWidth. */
81
+ columnStyles?: Record<number, CellStyle>;
82
+ /** Whether to draw the table outer border (default true). */
83
+ drawLineForTable?: boolean;
84
+ /** Whether to draw cell borders (default true). */
85
+ drawLineForCells?: boolean;
86
+ /** Table border line width (default 0.005). */
87
+ lineWidth?: number;
88
+ /** Table border color as a grayscale value: 0 = black, 200 = light gray, 255 = white (default 0). */
89
+ lineColor?: number;
90
+ }
91
+ interface AddImageArgs {
92
+ dataUri: string;
93
+ x: number;
94
+ y: number;
95
+ width: number;
96
+ height: number;
97
+ }
98
+
99
+ declare class PdfGenerator {
100
+ private doc;
101
+ private options;
102
+ private _currentY;
103
+ constructor(options?: PdfOptions);
104
+ get font(): string;
105
+ set font(font: string);
106
+ get theme(): ThemeConfig;
107
+ get filename(): string;
108
+ get margin(): number;
109
+ get pageSize(): PageSize;
110
+ get pageWidth(): number;
111
+ get pageHeight(): number;
112
+ get fullImageHeight(): number;
113
+ get availableWidth(): number;
114
+ get availableHeight(): number;
115
+ get footerFontSizePoints(): number;
116
+ /** Current Y position (in doc units) for layout. */
117
+ get currentY(): number;
118
+ /** Set current Y position (e.g. after custom content like header or chart). */
119
+ setCurrentY(y: number): void;
120
+ addVerticalGap(delta: number): void;
121
+ checkTableOverflow(columnWidths: number[]): void;
122
+ addTable(addTableArgs: AddTableArgs): number | undefined;
123
+ /**
124
+ * Add an image at the given position and size. Does not update currentY.
125
+ */
126
+ addImage({ dataUri, x, y, width, height }: AddImageArgs): void;
127
+ /** Exposes the jsPDF document for custom drawing (e.g. header text, status badge, certification). */
128
+ getDoc(): jsPDF;
129
+ addPage(): void;
130
+ addFullPagePNG(dataURI: string, aspectRatio: number): void;
131
+ private calculateImageDimensions;
132
+ save(): void;
133
+ private buildFooterColumnStyles;
134
+ renderFooters(): void;
135
+ /** Footer is a single line; height derived from font size and table padding. */
136
+ get footerHeight(): number;
137
+ private drawAvailableAreaRectangle;
138
+ }
139
+
140
+ export { type AddImageArgs, type AddTableArgs, type CellStyle, type CellValue, DEFAULT_FOOTER_CELL_BUILDER, DEFAULT_OPTIONS, DEFAULT_REPORT_FILENAME, DEFAULT_THEME, type FillColorRgbTuple, type FontStyle, type FooterCellArgs, type FooterCellBuilder, Gaps, IMAGE_GAP, type PageOrientation, type PageSize, type PaperFormat, PdfColors, PdfGenerator, type PdfOptions, type TableCellDef, type TableCellHalign, type TableCellInput, type TableCellValign, type ThemeConfig };
package/dist/pdf.d.ts ADDED
@@ -0,0 +1,140 @@
1
+ import { jsPDFOptions, jsPDF } from 'jspdf';
2
+
3
+ interface FontSize {
4
+ title: number;
5
+ header: number;
6
+ normal: number;
7
+ small: number;
8
+ tiny: number;
9
+ }
10
+ interface RgbColor {
11
+ r: number;
12
+ g: number;
13
+ b: number;
14
+ }
15
+ interface ThemeConfig {
16
+ primaryColor: RgbColor;
17
+ fontSize: FontSize;
18
+ }
19
+ type PageSize = jsPDF["internal"]["pageSize"];
20
+ type PageOrientation = jsPDFOptions["orientation"];
21
+ type PaperFormat = jsPDFOptions["format"];
22
+ declare const DEFAULT_REPORT_FILENAME = "report.pdf";
23
+ declare const IMAGE_GAP = 0.5;
24
+ declare const DEFAULT_THEME: ThemeConfig;
25
+ interface FooterCellArgs {
26
+ align: TableCellHalign;
27
+ currentPage: number;
28
+ totalPages: number;
29
+ }
30
+ type FooterCellBuilder = (footerCellArgs: FooterCellArgs) => string;
31
+ interface PdfOptions {
32
+ orientation?: PageOrientation;
33
+ paperFormat?: PaperFormat;
34
+ hasFooter?: boolean;
35
+ filename?: string;
36
+ theme?: ThemeConfig;
37
+ margin?: number;
38
+ displayAvailableAreaRectangle?: boolean;
39
+ footerCellBuilder?: FooterCellBuilder;
40
+ }
41
+ declare const DEFAULT_FOOTER_CELL_BUILDER: ({ align, currentPage, totalPages }: FooterCellArgs) => string;
42
+ declare const DEFAULT_OPTIONS: Readonly<Required<PdfOptions>>;
43
+ declare const Gaps: {
44
+ TINY: number;
45
+ SMALL: number;
46
+ MEDIUM: number;
47
+ LARGE: number;
48
+ X_LARGE: number;
49
+ };
50
+ declare const PdfColors: Record<string, FillColorRgbTuple>;
51
+ type TableCellHalign = "left" | "center" | "right";
52
+ type TableCellValign = "top" | "middle" | "bottom";
53
+ type FontStyle = "normal" | "bold";
54
+ type FillColorRgbTuple = [number, number, number];
55
+ type CellStyle = {
56
+ cellWidth?: number;
57
+ halign?: TableCellHalign;
58
+ valign?: TableCellValign;
59
+ fontStyle?: FontStyle;
60
+ fillColor?: FillColorRgbTuple;
61
+ };
62
+ type CellValue = string | number;
63
+ /** Cell with rowSpan/colSpan for merged cells (passed through to autoTable). */
64
+ interface TableCellDef {
65
+ content: string | string[] | number;
66
+ rowSpan?: number;
67
+ colSpan?: number;
68
+ }
69
+ type TableCellInput = CellValue | TableCellDef;
70
+ /** Parameters for adding a table (generic API, no jspdf-autotable leak). */
71
+ interface AddTableArgs {
72
+ /** Each row is an array of cell values or cell defs (with rowSpan/colSpan). */
73
+ body: TableCellInput[][];
74
+ /** Optional header row (single row of cell values). */
75
+ head?: CellValue[];
76
+ /** Y position to start the table (uses generator currentY if omitted). */
77
+ startY?: number;
78
+ /** Optional column widths (in same units as doc, e.g. inches). Sum should match available width. */
79
+ columnWidths?: number[];
80
+ /** Optional per-column styles (e.g. halign, fontStyle). Merged with auto-generated cellWidth. */
81
+ columnStyles?: Record<number, CellStyle>;
82
+ /** Whether to draw the table outer border (default true). */
83
+ drawLineForTable?: boolean;
84
+ /** Whether to draw cell borders (default true). */
85
+ drawLineForCells?: boolean;
86
+ /** Table border line width (default 0.005). */
87
+ lineWidth?: number;
88
+ /** Table border color as a grayscale value: 0 = black, 200 = light gray, 255 = white (default 0). */
89
+ lineColor?: number;
90
+ }
91
+ interface AddImageArgs {
92
+ dataUri: string;
93
+ x: number;
94
+ y: number;
95
+ width: number;
96
+ height: number;
97
+ }
98
+
99
+ declare class PdfGenerator {
100
+ private doc;
101
+ private options;
102
+ private _currentY;
103
+ constructor(options?: PdfOptions);
104
+ get font(): string;
105
+ set font(font: string);
106
+ get theme(): ThemeConfig;
107
+ get filename(): string;
108
+ get margin(): number;
109
+ get pageSize(): PageSize;
110
+ get pageWidth(): number;
111
+ get pageHeight(): number;
112
+ get fullImageHeight(): number;
113
+ get availableWidth(): number;
114
+ get availableHeight(): number;
115
+ get footerFontSizePoints(): number;
116
+ /** Current Y position (in doc units) for layout. */
117
+ get currentY(): number;
118
+ /** Set current Y position (e.g. after custom content like header or chart). */
119
+ setCurrentY(y: number): void;
120
+ addVerticalGap(delta: number): void;
121
+ checkTableOverflow(columnWidths: number[]): void;
122
+ addTable(addTableArgs: AddTableArgs): number | undefined;
123
+ /**
124
+ * Add an image at the given position and size. Does not update currentY.
125
+ */
126
+ addImage({ dataUri, x, y, width, height }: AddImageArgs): void;
127
+ /** Exposes the jsPDF document for custom drawing (e.g. header text, status badge, certification). */
128
+ getDoc(): jsPDF;
129
+ addPage(): void;
130
+ addFullPagePNG(dataURI: string, aspectRatio: number): void;
131
+ private calculateImageDimensions;
132
+ save(): void;
133
+ private buildFooterColumnStyles;
134
+ renderFooters(): void;
135
+ /** Footer is a single line; height derived from font size and table padding. */
136
+ get footerHeight(): number;
137
+ private drawAvailableAreaRectangle;
138
+ }
139
+
140
+ export { type AddImageArgs, type AddTableArgs, type CellStyle, type CellValue, DEFAULT_FOOTER_CELL_BUILDER, DEFAULT_OPTIONS, DEFAULT_REPORT_FILENAME, DEFAULT_THEME, type FillColorRgbTuple, type FontStyle, type FooterCellArgs, type FooterCellBuilder, Gaps, IMAGE_GAP, type PageOrientation, type PageSize, type PaperFormat, PdfColors, PdfGenerator, type PdfOptions, type TableCellDef, type TableCellHalign, type TableCellInput, type TableCellValign, type ThemeConfig };