hrms-report-template/libs/html-templates-lib.ts
2025-02-28 11:42:24 +07:00

223 lines
7.6 KiB
TypeScript

import express from "express"
export const htmlTemplateRoute = express.Router()
import { mimeToExtension, templateOption } from "./report-template"
// import fs from "fs"
//import { chromium } from 'playwright'
import puppeteer, { PDFOptions } from "puppeteer"
import Handlebars from "handlebars"
//import { createReport } from "docx-templates"
// แก้ package.json ของ LibreOfficeFileConverter
// https://github.com/microsoft/TypeScript/issues/52363#issuecomment-1659179354
//import { LibreOfficeFileConverter } from "libreoffice-file-converter"
const TEMPLATE_FOLDER_NAME = "templates/html"
function wait(ms: number) {
return new Promise((resolve) => setTimeout(resolve, ms))
}
/**
* docxTemplate Uses docx-template to convert input data and template to output buffer.
* SPA and lazy load page may not fully render(eg. pantip.com).
* You have to handle exception throw by function
* handlebars template support only content from Buffer
* @param {Buffer|String} t template in buffer format or url to web page
* @param {templateOption} templOpt Template Information in JSON format
* @param {String} outputMediaType output extension, support pdf, jpeg, png
* @return {Promise<Uint8Array>} output buffer after apply template.
*/
export async function htmlTemplateX(
t: Buffer | String,
templOpt: templateOption,
outputMediaType: string = "pdf"
): Promise<Uint8Array> {
try {
if (!["pdf", "jpeg", "png"].find((e) => e === outputMediaType)) {
throw "FormatError"
}
const browser = await puppeteer.launch({
headless: true,
args: ["--no-sandbox"],
})
const page = await browser.newPage()
if (templOpt.htmlOption?.navigationTimeout)
page.setDefaultNavigationTimeout(120000)
if (typeof t === "string") {
switch (templOpt.htmlOption?.waitUntil) {
case "networkidle0":
await page.goto(t, { waitUntil: "networkidle0" })
break
case "networkidle2":
await page.goto(t, { waitUntil: "networkidle2" })
break
default:
await page.goto(t)
}
} else {
if (templOpt.data) {
const template = Handlebars.compile(t.toString())
const html = template(templOpt.data)
await page.setContent(html)
} else {
await page.setContent(t.toString())
}
}
const totalHeight = await page.evaluate(async (_templOpt) => {
//force scroll for lazy load page
const _scroll = _templOpt.htmlOption?.preloadScroll ?? 1000
const _wait = _templOpt.htmlOption?.preloadWait ?? 400
const _loop = _templOpt.htmlOption?.preloadLoop ?? 4
for (let i = 0; i < _loop; i++) {
window.scrollBy(0, _scroll)
await new Promise((resolve) => setTimeout(resolve, _wait))
}
//extra wait
await new Promise((resolve) => setTimeout(resolve, _wait * 2))
let scrollableSection =
_templOpt.htmlOption?.querySelector &&
document.querySelector(_templOpt.htmlOption.querySelector)
? document.querySelector(_templOpt.htmlOption.querySelector)
: null
const childElement = scrollableSection ? scrollableSection : document.body
if (scrollableSection == null) scrollableSection = document.body
//const childElement = scrollableSection.firstElementChild;
let scrollPosition = 0
let viewportHeight = window.innerHeight
while (scrollPosition < childElement.scrollHeight) {
scrollableSection.scrollBy(0, viewportHeight)
await new Promise((resolve) => setTimeout(resolve, 500))
scrollPosition += viewportHeight
}
//return scrollPosition
return childElement.scrollHeight
}, templOpt)
console.log("totalHeight")
if (!totalHeight) {
throw new Error(
`Unable to determine the page height ${totalHeight}. The selector may not correct or no body tag`
)
} else {
console.log("Page height adjusted to:", totalHeight)
}
await page.setViewport({
width: Number(templOpt.htmlOption?.pdfOption?.width ?? 1200),
height: totalHeight,
deviceScaleFactor: 2,
isMobile: false,
})
///// output to photo end here
if (outputMediaType === "png" || outputMediaType === "jpeg") {
const photoBuffer = await page.screenshot({
// path: 'url_pup.png',
fullPage: true,
type: outputMediaType, // 'webp'
})
await browser.close()
return photoBuffer
}
///// output to PDF
//TODO overide option from htmlTemplateOption
let o: PDFOptions = {
// path: './url_prop.pdf',
// format:"A4",
width: 1200,
height: totalHeight,
printBackground: true,
scale: 1,
displayHeaderFooter: false,
margin: { top: 5, right: 5, bottom: 5, left: 5 },
}
//Murge with default
if (templOpt.htmlOption?.pdfOption)
o = { ...o, ...templOpt.htmlOption.pdfOption }
const { path, ...pdfOption } = o //remove path if exists
console.log("pdfOption:", pdfOption)
const buffer = await page.pdf(pdfOption)
await browser.close()
return buffer
} catch (e) {
//console.log(e)
throw e
}
}
/** javascript-obfuscator:disable
* @swagger
* /api/v1/report-template/html:
* post:
* summary: แปลหน้าเวปไปเป็น pdf , png, jpeg (ฟีเจอร์ html template ด้วย handlebars ยังไม่เสร็จ) ค่า template เป็น url ของหน้าเวปที่ต้องการ, reportName เป็นชื่อไฟล์ที่ต้องการ
* tags: [report-template]
* requestBody:
* required: true
* content:
* application/json:
* schema:
* $ref: '#/components/schemas/templateOption'
* example:
* template: https://bma-dashboard.frappet.synology.me/d/ANtkJay4z/4Lic4Li54LmJ4Lie4Li04LiB4Liy4Lij?orgId=1&kiosk
* reportName: html-report
* htmlOption: {"querySelector": ".scrollbar-view"}
* responses:
* 201:
* description: เอกสารถูกสร้างขึ้น
* content:
* application/pdf:
* schema:
* type: string
* format: binary
* image/png:
* schema:
* type: string
* format: binary
* image/jpeg:
* schema:
* type: string
* format: binary
* 400:
* description: format error
* 500:
* description: Server error
*
*/
htmlTemplateRoute.post("/", async function (req, res) {
try {
if (!req.headers["content-type"] || !req.headers["accept"])
throw new Error("Require header content-type, accept")
let inputType = mimeToExtension(req.headers["content-type"]) // application/json
let outputMediaType = mimeToExtension(req.headers["accept"])
let buffer = await htmlTemplateX(
req.body.template,
req.body,
outputMediaType
)
res.statusCode = 201
res.setHeader("Content-Type", req.headers["accept"])
res.setHeader(
"Content-Disposition",
`attachment;filename=${req.body.reportName}.${outputMediaType}`
)
res.setHeader("Content-Length", buffer.length)
res.end(buffer)
} catch (ex) {
if (ex instanceof SyntaxError) {
res.statusCode = 400
res.statusMessage = ex.message
res.end(res.statusMessage)
console.error("report-template/html: ", ex)
} else {
res.statusCode = 500
res.statusMessage =
"Internal Server Error during POST report-template/html"
res.end(res.statusMessage)
console.error("report-template/html: ", ex)
}
}
})