Refactor color input fields and enhance watermark functionality

- Updated color input fields in various HTML pages to remove unnecessary classes for improved styling consistency.
- Modified the watermark node to include options for positioning and flattening watermarks.
- Enhanced the addTextWatermark function to support customizable positioning and page selection for watermarks.
- Added new controls for text and image watermarks in the UI, allowing users to specify text, font size, color, opacity, angle, and image scaling.
- Updated the WASM provider to use the latest version of pymupdf-wasm.
This commit is contained in:
alam00000
2026-02-26 19:40:08 +05:30
parent 2e64e8e8d4
commit 88260c26ab
41 changed files with 1512 additions and 383 deletions

View File

@@ -187,6 +187,9 @@ export interface TextWatermarkOptions {
color: { r: number; g: number; b: number };
opacity: number;
angle: number;
x?: number;
y?: number;
pageIndices?: number[];
}
export async function addTextWatermark(
@@ -194,19 +197,60 @@ export async function addTextWatermark(
options: TextWatermarkOptions
): Promise<Uint8Array> {
const pdfDoc = await PDFDocument.load(pdfBytes);
const font = await pdfDoc.embedFont(StandardFonts.Helvetica);
const canvas = document.createElement('canvas');
const ctx = canvas.getContext('2d');
if (!ctx) throw new Error('Failed to create canvas context');
const dpr = 2;
const colorR = Math.round(options.color.r * 255);
const colorG = Math.round(options.color.g * 255);
const colorB = Math.round(options.color.b * 255);
const fontStr = `bold ${options.fontSize * dpr}px "Noto Sans SC", "Noto Sans JP", "Noto Sans KR", "Noto Sans Arabic", Arial, sans-serif`;
ctx.font = fontStr;
const metrics = ctx.measureText(options.text);
canvas.width = Math.ceil(metrics.width) + 4;
canvas.height = Math.ceil(options.fontSize * dpr * 1.4);
ctx.font = fontStr;
ctx.fillStyle = `rgb(${colorR}, ${colorG}, ${colorB})`;
ctx.textBaseline = 'middle';
ctx.fillText(options.text, 2, canvas.height / 2);
const blob = await new Promise<Blob>((resolve, reject) => {
canvas.toBlob(
(b) => (b ? resolve(b) : reject(new Error('Canvas toBlob failed'))),
'image/png'
);
});
const imageBytes = new Uint8Array(await blob.arrayBuffer());
const image = await pdfDoc.embedPng(imageBytes);
const pages = pdfDoc.getPages();
const posX = options.x ?? 0.5;
const posY = options.y ?? 0.5;
const imgWidth = image.width / dpr;
const imgHeight = image.height / dpr;
for (const page of pages) {
const rad = (options.angle * Math.PI) / 180;
const halfW = imgWidth / 2;
const halfH = imgHeight / 2;
const targetIndices = options.pageIndices ?? pages.map((_, i) => i);
for (const idx of targetIndices) {
const page = pages[idx];
if (!page) continue;
const { width, height } = page.getSize();
const textWidth = font.widthOfTextAtSize(options.text, options.fontSize);
const cx = posX * width;
const cy = posY * height;
page.drawText(options.text, {
x: (width - textWidth) / 2,
y: height / 2,
font,
size: options.fontSize,
color: rgb(options.color.r, options.color.g, options.color.b),
page.drawImage(image, {
x: cx - Math.cos(rad) * halfW + Math.sin(rad) * halfH,
y: cy - Math.sin(rad) * halfW - Math.cos(rad) * halfH,
width: imgWidth,
height: imgHeight,
opacity: options.opacity,
rotate: degrees(options.angle),
});
@@ -221,6 +265,9 @@ export interface ImageWatermarkOptions {
opacity: number;
angle: number;
scale: number;
x?: number;
y?: number;
pageIndices?: number[];
}
export async function addImageWatermark(
@@ -233,15 +280,26 @@ export async function addImageWatermark(
? await pdfDoc.embedPng(options.imageBytes)
: await pdfDoc.embedJpg(options.imageBytes);
const pages = pdfDoc.getPages();
const posX = options.x ?? 0.5;
const posY = options.y ?? 0.5;
for (const page of pages) {
const imgWidth = image.width * options.scale;
const imgHeight = image.height * options.scale;
const rad = (options.angle * Math.PI) / 180;
const halfW = imgWidth / 2;
const halfH = imgHeight / 2;
const targetIndices = options.pageIndices ?? pages.map((_, i) => i);
for (const idx of targetIndices) {
const page = pages[idx];
if (!page) continue;
const { width, height } = page.getSize();
const imgWidth = image.width * options.scale;
const imgHeight = image.height * options.scale;
const cx = posX * width;
const cy = posY * height;
page.drawImage(image, {
x: (width - imgWidth) / 2,
y: (height - imgHeight) / 2,
x: cx - Math.cos(rad) * halfW + Math.sin(rad) * halfH,
y: cy - Math.sin(rad) * halfW - Math.cos(rad) * halfH,
width: imgWidth,
height: imgHeight,
opacity: options.opacity,