A powerful JavaScript SDK for rendering HTML content as paginated PDFs using Paged.js and html2pdf.js. Perfect for generating reports, invoices, documentation, and any print-ready content in the browser.
Try the interactive demo: https://eka-care.github.io/Pagify-sdk/
- ✨ Client-side PDF generation - No server dependencies
- 📄 Professional pagination - Powered by Paged.js
- 🎨 Rich styling support - CSS @page rules, headers, footers
- 🖼️ Base64 image support - Offline-compatible image handling
- 📱 Cross-platform - Works on desktop and mobile browsers
- 🔧 Easy integration - Simple API with TypeScript support
- 👁️ Preview mode - View paginated layout without generating PDF
- 🧹 Automatic cleanup - Prevents memory leaks from zombie iframes
npm install @eka-care/pagify-sdkOr via CDN:
<script src="https://unpkg.com/@eka-care/pagify-sdk/dist/pagify.standalone.js"></script>import pagify from '@eka-care/pagify-sdk';
// Generate a PDF
await pagify.render({
body_html: '<h1>Hello World!</h1><p>This is my first PDF.</p>',
header_html: '<div>Page Header</div>',
footer_html: '<div>Page <span class="pageNumber"></span></div>',
onPdfReady: (blobUrl) => {
// Display PDF in iframe or create download link
document.getElementById('pdf-viewer').src = blobUrl;
},
onPdfError: (error) => {
console.error('PDF generation failed:', error);
}
});<script src="https://unpkg.com/@eka-care/pagify-sdk/dist/pagify.standalone.js"></script>
<script>
window.pagify.render({
body_html: '<h1>Hello World!</h1>',
onPdfReady: (blobUrl) => {
window.open(blobUrl, '_blank');
}
});
</script>// Show preview without generating PDF (faster)
await pagify.render({
body_html: '<h1>Preview Content</h1>',
containerSelector: '#preview-container',
isViewOnlySkipMakingPDF: true,
onPreviewReady: (result) => {
if (result.success) {
console.log('Preview ready');
}
}
});Renders HTML content as a paginated PDF.
| Parameter | Type | Default | Description |
|---|---|---|---|
body_html |
string |
"" |
Main HTML content for the PDF body |
header_html |
string |
"" |
HTML content for page headers |
footer_html |
string |
"" |
HTML content for page footers |
head_html |
string |
"" |
Additional HTML for the <head> section |
page_size |
string |
"A4" |
Page size (A4, Letter, etc.) |
margin_left |
string |
"0mm" |
Left page margin |
margin_right |
string |
"0mm" |
Right page margin |
header_height |
string |
"0mm" |
Height reserved for header |
footer_height |
string |
"0mm" |
Height reserved for footer |
containerSelector |
string |
null |
CSS selector for preview container |
isViewOnlySkipMakingPDF |
boolean |
false |
If true, only renders preview without generating PDF |
onPdfReady |
function |
null |
Callback when PDF is ready (receives blobUrl) |
onPdfError |
function |
null |
Callback when PDF generation fails (receives error) |
onPreviewReady |
function |
null |
Callback when preview completes (receives {success, error?}) |
await pagify.render({
head_html: `
<style>
@page {
size: A4;
margin: 25mm 20mm;
}
body {
font-family: Arial, sans-serif;
line-height: 1.6;
}
.page-break {
page-break-before: always;
}
</style>
`,
header_html: `
<div style="text-align: center; font-size: 12px;">
Document Header - Page <span class="pageNumber"></span>
</div>
`,
body_html: `
<h1>Professional Report</h1>
<p>This demonstrates advanced PDF generation capabilities.</p>
<table>
<tr><th>Item</th><th>Value</th></tr>
<tr><td>Revenue</td><td>$100,000</td></tr>
<tr><td>Profit</td><td>$25,000</td></tr>
</table>
<div class="page-break"></div>
<h2>Page 2 Content</h2>
<p>This content appears on the second page.</p>
`,
footer_html: `
<div style="text-align: center; font-size: 10px;">
© 2024 Company Name | Page <span class="pageNumber"></span> of <span class="totalPages"></span>
</div>
`,
header_height: "15mm",
footer_height: "15mm",
onPdfReady: (blobUrl) => {
// Create download link
const link = document.createElement('a');
link.href = blobUrl;
link.download = 'report.pdf';
link.textContent = 'Download PDF';
document.body.appendChild(link);
}
});// Step 1: Show only preview and do not make PDF
await pagify.render({
body_html: '<h1>Invoice #12345</h1>',
header_html: '<div>Company Header</div>',
footer_html: '<div>Page <span class="pageNumber"></span></div>',
containerSelector: '#preview-container',
isViewOnlySkipMakingPDF: true,
onPreviewReady: ({ success, error }) => {
if (success) {
// caller/invoker/application layer does their flows
} else {
console.error('Preview failed:', error);
}
}
});
document.getElementById('download-btn').onclick = async () => {
await pagify.render({
body_html: '<h1>Invoice #12345</h1>',
header_html: '<div>Company Header</div>',
footer_html: '<div>Page <span class="pageNumber"></span></div>',
// isViewOnlySkipMakingPDF: true, --> not passing this makes the PDF and the cb to watch for is onPdfReady when this is not passed and when its passed onPreviewReady.
onPdfReady: (blobUrl) => {
const link = document.createElement('a');
link.href = blobUrl;
link.download = 'invoice.pdf';
link.click();
}
});
};Direct PDF generation that returns a Promise with the PDF blob.
try {
const pdfBlob = await pagify.generatePDF({
body_html: '<h1>Direct PDF Generation</h1>',
header_html: '<div>Header</div>'
});
// Use the blob directly
const url = URL.createObjectURL(pdfBlob);
window.open(url, '_blank');
} catch (error) {
console.error('PDF generation failed:', error);
}@page {
size: A4 portrait;
margin: 25mm 20mm;
}Use pageNumber and totalPages classes for dynamic content:
<div class="footer">
Page <span class="pageNumber"></span> of <span class="totalPages"></span>
</div>.page-break-before { page-break-before: always; }
.page-break-after { page-break-after: always; }
.page-break-avoid { page-break-inside: avoid; }For offline compatibility, use base64 encoded images:
<img src="..." alt="Logo">- Chrome 60+
- Firefox 55+
- Safari 12+
- Edge 79+
- Mobile browsers (iOS Safari, Chrome Mobile)
- Paged.js - CSS paged media polyfill
- html2pdf.js - HTML to PDF conversion
Full TypeScript definitions included:
import pagify, { PagifyOptions, PagifySDK } from '@eka-care/pagify-sdk';
const options: PagifyOptions = {
body_html: '<h1>TypeScript Support</h1>',
onPdfReady: (blobUrl: string) => {
console.log('PDF ready:', blobUrl);
}
};
await pagify.render(options);# Install dependencies
npm install
# Build the package
npm run build
# Watch mode for development
npm run dev
# Run linting
npm run lint
# Run tests
npm testMIT License - see LICENSE file for details.
- Fork the repository
- Create your feature branch (
git checkout -b feature/amazing-feature) - Commit your changes (
git commit -m 'Add some amazing feature') - Push to the branch (
git push origin feature/amazing-feature) - Open a Pull Request