mirror of
https://github.com/Stirling-Tools/Stirling-PDF.git
synced 2025-06-06 18:30:57 +00:00
Add Decrypt to all relevant pages
This commit is contained in:
parent
1d6511b043
commit
ef8231de3a
@ -4,7 +4,7 @@ export class DecryptFile {
|
|||||||
const formData = new FormData();
|
const formData = new FormData();
|
||||||
formData.append('fileInput', file);
|
formData.append('fileInput', file);
|
||||||
if (requiresPassword) {
|
if (requiresPassword) {
|
||||||
const password = prompt(`${window.translations.passwordPrompt}`);
|
const password = prompt(`${window.decrypt.passwordPrompt}`);
|
||||||
|
|
||||||
if (password === null) {
|
if (password === null) {
|
||||||
// User cancelled
|
// User cancelled
|
||||||
@ -16,9 +16,9 @@ export class DecryptFile {
|
|||||||
// No password provided
|
// No password provided
|
||||||
console.error(`No password provided for encrypted PDF: ${file.name}`);
|
console.error(`No password provided for encrypted PDF: ${file.name}`);
|
||||||
this.showErrorBanner(
|
this.showErrorBanner(
|
||||||
`${window.translations.noPassword.replace('{0}', file.name)}`,
|
`${window.decrypt.noPassword.replace('{0}', file.name)}`,
|
||||||
'',
|
'',
|
||||||
`${window.translations.unexpectedError}`
|
`${window.decrypt.unexpectedError}`
|
||||||
);
|
);
|
||||||
return null; // No file to return
|
return null; // No file to return
|
||||||
}
|
}
|
||||||
@ -37,11 +37,11 @@ export class DecryptFile {
|
|||||||
return new File([decryptedBlob], file.name, {type: 'application/pdf'});
|
return new File([decryptedBlob], file.name, {type: 'application/pdf'});
|
||||||
} else {
|
} else {
|
||||||
const errorText = await response.text();
|
const errorText = await response.text();
|
||||||
console.error(`${window.translations.invalidPassword} ${errorText}`);
|
console.error(`${window.decrypt.invalidPassword} ${errorText}`);
|
||||||
this.showErrorBanner(
|
this.showErrorBanner(
|
||||||
`${window.translations.invalidPassword}`,
|
`${window.decrypt.invalidPassword}`,
|
||||||
errorText,
|
errorText,
|
||||||
`${window.translations.invalidPasswordHeader.replace('{0}', file.name)}`
|
`${window.decrypt.invalidPasswordHeader.replace('{0}', file.name)}`
|
||||||
);
|
);
|
||||||
return null; // No file to return
|
return null; // No file to return
|
||||||
}
|
}
|
||||||
@ -49,8 +49,8 @@ export class DecryptFile {
|
|||||||
// Handle network or unexpected errors
|
// Handle network or unexpected errors
|
||||||
console.error(`Failed to decrypt PDF: ${file.name}`, error);
|
console.error(`Failed to decrypt PDF: ${file.name}`, error);
|
||||||
this.showErrorBanner(
|
this.showErrorBanner(
|
||||||
`${window.translations.unexpectedError.replace('{0}', file.name)}`,
|
`${window.decrypt.unexpectedError.replace('{0}', file.name)}`,
|
||||||
`${error.message || window.translations.unexpectedError}`,
|
`${error.message || window.decrypt.unexpectedError}`,
|
||||||
error
|
error
|
||||||
);
|
);
|
||||||
return null; // No file to return
|
return null; // No file to return
|
||||||
@ -59,6 +59,10 @@ export class DecryptFile {
|
|||||||
|
|
||||||
async checkFileEncrypted(file) {
|
async checkFileEncrypted(file) {
|
||||||
try {
|
try {
|
||||||
|
if (file.type !== 'application/pdf') {
|
||||||
|
return {isEncrypted: false, requiresPassword: false};
|
||||||
|
}
|
||||||
|
|
||||||
pdfjsLib.GlobalWorkerOptions.workerSrc = './pdfjs-legacy/pdf.worker.mjs';
|
pdfjsLib.GlobalWorkerOptions.workerSrc = './pdfjs-legacy/pdf.worker.mjs';
|
||||||
const arrayBuffer = await file.arrayBuffer();
|
const arrayBuffer = await file.arrayBuffer();
|
||||||
const arrayBufferForPdfLib = arrayBuffer.slice(0);
|
const arrayBufferForPdfLib = arrayBuffer.slice(0);
|
27
src/main/resources/static/js/download.js
Normal file
27
src/main/resources/static/js/download.js
Normal file
@ -0,0 +1,27 @@
|
|||||||
|
document.getElementById('download-pdf').addEventListener('click', async () => {
|
||||||
|
const modifiedPdf = await DraggableUtils.getOverlayedPdfDocument();
|
||||||
|
let decryptedFile = modifiedPdf;
|
||||||
|
let isEncrypted = false;
|
||||||
|
let requiresPassword = false;
|
||||||
|
await this.decryptFile
|
||||||
|
.checkFileEncrypted(decryptedFile)
|
||||||
|
.then((result) => {
|
||||||
|
isEncrypted = result.isEncrypted;
|
||||||
|
requiresPassword = result.requiresPassword;
|
||||||
|
})
|
||||||
|
.catch((error) => {
|
||||||
|
console.error(error);
|
||||||
|
});
|
||||||
|
if (decryptedFile.type === 'application/pdf' && isEncrypted) {
|
||||||
|
decryptedFile = await this.decryptFile.decryptFile(decryptedFile, requiresPassword);
|
||||||
|
if (!decryptedFile) {
|
||||||
|
throw new Error('File decryption failed.');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
const modifiedPdfBytes = await modifiedPdf.save();
|
||||||
|
const blob = new Blob([modifiedPdfBytes], {type: 'application/pdf'});
|
||||||
|
const link = document.createElement('a');
|
||||||
|
link.href = URL.createObjectURL(blob);
|
||||||
|
link.download = originalFileName + '_signed.pdf';
|
||||||
|
link.click();
|
||||||
|
});
|
@ -1,6 +1,6 @@
|
|||||||
const DraggableUtils = {
|
const DraggableUtils = {
|
||||||
boxDragContainer: document.getElementById("box-drag-container"),
|
boxDragContainer: document.getElementById('box-drag-container'),
|
||||||
pdfCanvas: document.getElementById("pdf-canvas"),
|
pdfCanvas: document.getElementById('pdf-canvas'),
|
||||||
nextId: 0,
|
nextId: 0,
|
||||||
pdfDoc: null,
|
pdfDoc: null,
|
||||||
pageIndex: 0,
|
pageIndex: 0,
|
||||||
@ -9,19 +9,17 @@ const DraggableUtils = {
|
|||||||
lastInteracted: null,
|
lastInteracted: null,
|
||||||
|
|
||||||
init() {
|
init() {
|
||||||
interact(".draggable-canvas")
|
interact('.draggable-canvas')
|
||||||
.draggable({
|
.draggable({
|
||||||
listeners: {
|
listeners: {
|
||||||
move: (event) => {
|
move: (event) => {
|
||||||
const target = event.target;
|
const target = event.target;
|
||||||
const x = (parseFloat(target.getAttribute("data-bs-x")) || 0)
|
const x = (parseFloat(target.getAttribute('data-bs-x')) || 0) + event.dx;
|
||||||
+ event.dx;
|
const y = (parseFloat(target.getAttribute('data-bs-y')) || 0) + event.dy;
|
||||||
const y = (parseFloat(target.getAttribute("data-bs-y")) || 0)
|
|
||||||
+ event.dy;
|
|
||||||
|
|
||||||
target.style.transform = `translate(${x}px, ${y}px)`;
|
target.style.transform = `translate(${x}px, ${y}px)`;
|
||||||
target.setAttribute("data-bs-x", x);
|
target.setAttribute('data-bs-x', x);
|
||||||
target.setAttribute("data-bs-y", y);
|
target.setAttribute('data-bs-y', y);
|
||||||
|
|
||||||
this.onInteraction(target);
|
this.onInteraction(target);
|
||||||
//update the last interacted element
|
//update the last interacted element
|
||||||
@ -34,8 +32,8 @@ const DraggableUtils = {
|
|||||||
listeners: {
|
listeners: {
|
||||||
move: (event) => {
|
move: (event) => {
|
||||||
var target = event.target;
|
var target = event.target;
|
||||||
var x = parseFloat(target.getAttribute("data-bs-x")) || 0;
|
var x = parseFloat(target.getAttribute('data-bs-x')) || 0;
|
||||||
var y = parseFloat(target.getAttribute("data-bs-y")) || 0;
|
var y = parseFloat(target.getAttribute('data-bs-y')) || 0;
|
||||||
|
|
||||||
// check if control key is pressed
|
// check if control key is pressed
|
||||||
if (event.ctrlKey) {
|
if (event.ctrlKey) {
|
||||||
@ -44,8 +42,7 @@ const DraggableUtils = {
|
|||||||
let width = event.rect.width;
|
let width = event.rect.width;
|
||||||
let height = event.rect.height;
|
let height = event.rect.height;
|
||||||
|
|
||||||
if (Math.abs(event.deltaRect.width) >= Math.abs(
|
if (Math.abs(event.deltaRect.width) >= Math.abs(event.deltaRect.height)) {
|
||||||
event.deltaRect.height)) {
|
|
||||||
height = width / aspectRatio;
|
height = width / aspectRatio;
|
||||||
} else {
|
} else {
|
||||||
width = height * aspectRatio;
|
width = height * aspectRatio;
|
||||||
@ -55,19 +52,18 @@ const DraggableUtils = {
|
|||||||
event.rect.height = height;
|
event.rect.height = height;
|
||||||
}
|
}
|
||||||
|
|
||||||
target.style.width = event.rect.width + "px";
|
target.style.width = event.rect.width + 'px';
|
||||||
target.style.height = event.rect.height + "px";
|
target.style.height = event.rect.height + 'px';
|
||||||
|
|
||||||
// translate when resizing from top or left edges
|
// translate when resizing from top or left edges
|
||||||
x += event.deltaRect.left;
|
x += event.deltaRect.left;
|
||||||
y += event.deltaRect.top;
|
y += event.deltaRect.top;
|
||||||
|
|
||||||
target.style.transform = "translate(" + x + "px," + y + "px)";
|
target.style.transform = 'translate(' + x + 'px,' + y + 'px)';
|
||||||
|
|
||||||
target.setAttribute("data-bs-x", x);
|
target.setAttribute('data-bs-x', x);
|
||||||
target.setAttribute("data-bs-y", y);
|
target.setAttribute('data-bs-y', y);
|
||||||
target.textContent = Math.round(event.rect.width) + "\u00D7"
|
target.textContent = Math.round(event.rect.width) + '\u00D7' + Math.round(event.rect.height);
|
||||||
+ Math.round(event.rect.height);
|
|
||||||
|
|
||||||
this.onInteraction(target);
|
this.onInteraction(target);
|
||||||
},
|
},
|
||||||
@ -95,8 +91,8 @@ const DraggableUtils = {
|
|||||||
const stepY = target.offsetHeight * 0.05;
|
const stepY = target.offsetHeight * 0.05;
|
||||||
|
|
||||||
// Get the current x and y coordinates
|
// Get the current x and y coordinates
|
||||||
let x = (parseFloat(target.getAttribute('data-bs-x')) || 0);
|
let x = parseFloat(target.getAttribute('data-bs-x')) || 0;
|
||||||
let y = (parseFloat(target.getAttribute('data-bs-y')) || 0);
|
let y = parseFloat(target.getAttribute('data-bs-y')) || 0;
|
||||||
|
|
||||||
// Check which key was pressed and update the coordinates accordingly
|
// Check which key was pressed and update the coordinates accordingly
|
||||||
switch (event.key) {
|
switch (event.key) {
|
||||||
@ -135,15 +131,15 @@ const DraggableUtils = {
|
|||||||
},
|
},
|
||||||
|
|
||||||
createDraggableCanvas() {
|
createDraggableCanvas() {
|
||||||
const createdCanvas = document.createElement("canvas");
|
const createdCanvas = document.createElement('canvas');
|
||||||
createdCanvas.id = `draggable-canvas-${this.nextId++}`;
|
createdCanvas.id = `draggable-canvas-${this.nextId++}`;
|
||||||
createdCanvas.classList.add("draggable-canvas");
|
createdCanvas.classList.add('draggable-canvas');
|
||||||
|
|
||||||
const x = 0;
|
const x = 0;
|
||||||
const y = 20;
|
const y = 20;
|
||||||
createdCanvas.style.transform = `translate(${x}px, ${y}px)`;
|
createdCanvas.style.transform = `translate(${x}px, ${y}px)`;
|
||||||
createdCanvas.setAttribute("data-bs-x", x);
|
createdCanvas.setAttribute('data-bs-x', x);
|
||||||
createdCanvas.setAttribute("data-bs-y", y);
|
createdCanvas.setAttribute('data-bs-y', y);
|
||||||
|
|
||||||
//Click element in order to enable arrow keys
|
//Click element in order to enable arrow keys
|
||||||
createdCanvas.addEventListener('click', () => {
|
createdCanvas.addEventListener('click', () => {
|
||||||
@ -186,29 +182,29 @@ const DraggableUtils = {
|
|||||||
newHeight = newHeight * scaleMultiplier;
|
newHeight = newHeight * scaleMultiplier;
|
||||||
}
|
}
|
||||||
|
|
||||||
createdCanvas.style.width = newWidth + "px";
|
createdCanvas.style.width = newWidth + 'px';
|
||||||
createdCanvas.style.height = newHeight + "px";
|
createdCanvas.style.height = newHeight + 'px';
|
||||||
|
|
||||||
var myContext = createdCanvas.getContext("2d");
|
var myContext = createdCanvas.getContext('2d');
|
||||||
myContext.drawImage(myImage, 0, 0);
|
myContext.drawImage(myImage, 0, 0);
|
||||||
resolve(createdCanvas);
|
resolve(createdCanvas);
|
||||||
};
|
};
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
deleteAllDraggableCanvases() {
|
deleteAllDraggableCanvases() {
|
||||||
this.boxDragContainer.querySelectorAll(".draggable-canvas").forEach((el) => el.remove());
|
this.boxDragContainer.querySelectorAll('.draggable-canvas').forEach((el) => el.remove());
|
||||||
},
|
},
|
||||||
async addAllPagesDraggableCanvas(element) {
|
async addAllPagesDraggableCanvas(element) {
|
||||||
if (element) {
|
if (element) {
|
||||||
let currentPage = this.pageIndex
|
let currentPage = this.pageIndex;
|
||||||
if (!this.elementAllPages.includes(element)) {
|
if (!this.elementAllPages.includes(element)) {
|
||||||
this.elementAllPages.push(element)
|
this.elementAllPages.push(element);
|
||||||
element.style.filter = 'sepia(1) hue-rotate(90deg) brightness(1.2)';
|
element.style.filter = 'sepia(1) hue-rotate(90deg) brightness(1.2)';
|
||||||
let newElement = {
|
let newElement = {
|
||||||
"element": element,
|
element: element,
|
||||||
"offsetWidth": element.width,
|
offsetWidth: element.width,
|
||||||
"offsetHeight": element.height
|
offsetHeight: element.height,
|
||||||
}
|
};
|
||||||
|
|
||||||
let pagesMap = this.documentsMap.get(this.pdfDoc);
|
let pagesMap = this.documentsMap.get(this.pdfDoc);
|
||||||
|
|
||||||
@ -216,21 +212,20 @@ const DraggableUtils = {
|
|||||||
pagesMap = {};
|
pagesMap = {};
|
||||||
this.documentsMap.set(this.pdfDoc, pagesMap);
|
this.documentsMap.set(this.pdfDoc, pagesMap);
|
||||||
}
|
}
|
||||||
let page = this.pageIndex
|
let page = this.pageIndex;
|
||||||
|
|
||||||
for (let pageIndex = 0; pageIndex < this.pdfDoc.numPages; pageIndex++) {
|
for (let pageIndex = 0; pageIndex < this.pdfDoc.numPages; pageIndex++) {
|
||||||
|
|
||||||
if (pagesMap[`${pageIndex}-offsetWidth`]) {
|
if (pagesMap[`${pageIndex}-offsetWidth`]) {
|
||||||
if (!pagesMap[pageIndex].includes(newElement)) {
|
if (!pagesMap[pageIndex].includes(newElement)) {
|
||||||
pagesMap[pageIndex].push(newElement);
|
pagesMap[pageIndex].push(newElement);
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
pagesMap[pageIndex] = []
|
pagesMap[pageIndex] = [];
|
||||||
pagesMap[pageIndex].push(newElement)
|
pagesMap[pageIndex].push(newElement);
|
||||||
pagesMap[`${pageIndex}-offsetWidth`] = pagesMap[`${page}-offsetWidth`];
|
pagesMap[`${pageIndex}-offsetWidth`] = pagesMap[`${page}-offsetWidth`];
|
||||||
pagesMap[`${pageIndex}-offsetHeight`] = pagesMap[`${page}-offsetHeight`];
|
pagesMap[`${pageIndex}-offsetHeight`] = pagesMap[`${page}-offsetHeight`];
|
||||||
}
|
}
|
||||||
await this.goToPage(pageIndex)
|
await this.goToPage(pageIndex);
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
const index = this.elementAllPages.indexOf(element);
|
const index = this.elementAllPages.indexOf(element);
|
||||||
@ -247,17 +242,17 @@ const DraggableUtils = {
|
|||||||
for (let pageIndex = 0; pageIndex < this.pdfDoc.numPages; pageIndex++) {
|
for (let pageIndex = 0; pageIndex < this.pdfDoc.numPages; pageIndex++) {
|
||||||
if (pagesMap[`${pageIndex}-offsetWidth`] && pageIndex != currentPage) {
|
if (pagesMap[`${pageIndex}-offsetWidth`] && pageIndex != currentPage) {
|
||||||
const pageElements = pagesMap[pageIndex];
|
const pageElements = pagesMap[pageIndex];
|
||||||
pageElements.forEach(elementPage => {
|
pageElements.forEach((elementPage) => {
|
||||||
const elementIndex = pageElements.findIndex(elementPage => elementPage['element'].id === element.id);
|
const elementIndex = pageElements.findIndex((elementPage) => elementPage['element'].id === element.id);
|
||||||
if (elementIndex !== -1) {
|
if (elementIndex !== -1) {
|
||||||
pageElements.splice(elementIndex, 1);
|
pageElements.splice(elementIndex, 1);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
await this.goToPage(pageIndex)
|
await this.goToPage(pageIndex);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
await this.goToPage(currentPage)
|
await this.goToPage(currentPage);
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
deleteDraggableCanvas(element) {
|
deleteDraggableCanvas(element) {
|
||||||
@ -271,7 +266,7 @@ const DraggableUtils = {
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
getLastInteracted() {
|
getLastInteracted() {
|
||||||
return this.boxDragContainer.querySelector(".draggable-canvas:last-of-type");
|
return this.boxDragContainer.querySelector('.draggable-canvas:last-of-type');
|
||||||
},
|
},
|
||||||
|
|
||||||
storePageContents() {
|
storePageContents() {
|
||||||
@ -280,7 +275,7 @@ const DraggableUtils = {
|
|||||||
pagesMap = {};
|
pagesMap = {};
|
||||||
}
|
}
|
||||||
|
|
||||||
const elements = [...this.boxDragContainer.querySelectorAll(".draggable-canvas")];
|
const elements = [...this.boxDragContainer.querySelectorAll('.draggable-canvas')];
|
||||||
const draggablesData = elements.map((el) => {
|
const draggablesData = elements.map((el) => {
|
||||||
return {
|
return {
|
||||||
element: el,
|
element: el,
|
||||||
@ -291,8 +286,8 @@ const DraggableUtils = {
|
|||||||
elements.forEach((el) => this.boxDragContainer.removeChild(el));
|
elements.forEach((el) => this.boxDragContainer.removeChild(el));
|
||||||
|
|
||||||
pagesMap[this.pageIndex] = draggablesData;
|
pagesMap[this.pageIndex] = draggablesData;
|
||||||
pagesMap[this.pageIndex + "-offsetWidth"] = this.pdfCanvas.offsetWidth;
|
pagesMap[this.pageIndex + '-offsetWidth'] = this.pdfCanvas.offsetWidth;
|
||||||
pagesMap[this.pageIndex + "-offsetHeight"] = this.pdfCanvas.offsetHeight;
|
pagesMap[this.pageIndex + '-offsetHeight'] = this.pdfCanvas.offsetHeight;
|
||||||
|
|
||||||
this.documentsMap.set(this.pdfDoc, pagesMap);
|
this.documentsMap.set(this.pdfDoc, pagesMap);
|
||||||
},
|
},
|
||||||
@ -329,7 +324,7 @@ const DraggableUtils = {
|
|||||||
|
|
||||||
// render the page onto the canvas
|
// render the page onto the canvas
|
||||||
var renderContext = {
|
var renderContext = {
|
||||||
canvasContext: this.pdfCanvas.getContext("2d"),
|
canvasContext: this.pdfCanvas.getContext('2d'),
|
||||||
viewport: page.getViewport({scale: 1}),
|
viewport: page.getViewport({scale: 1}),
|
||||||
};
|
};
|
||||||
await page.render(renderContext).promise;
|
await page.render(renderContext).promise;
|
||||||
@ -369,7 +364,7 @@ const DraggableUtils = {
|
|||||||
const pagesMap = this.documentsMap.get(this.pdfDoc);
|
const pagesMap = this.documentsMap.get(this.pdfDoc);
|
||||||
|
|
||||||
for (let pageIdx in pagesMap) {
|
for (let pageIdx in pagesMap) {
|
||||||
if (pageIdx.includes("offset")) {
|
if (pageIdx.includes('offset')) {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
console.log(typeof pageIdx);
|
console.log(typeof pageIdx);
|
||||||
@ -377,9 +372,8 @@ const DraggableUtils = {
|
|||||||
const page = pdfDocModified.getPage(parseInt(pageIdx));
|
const page = pdfDocModified.getPage(parseInt(pageIdx));
|
||||||
let draggablesData = pagesMap[pageIdx];
|
let draggablesData = pagesMap[pageIdx];
|
||||||
|
|
||||||
const offsetWidth = pagesMap[pageIdx + "-offsetWidth"];
|
const offsetWidth = pagesMap[pageIdx + '-offsetWidth'];
|
||||||
const offsetHeight = pagesMap[pageIdx + "-offsetHeight"];
|
const offsetHeight = pagesMap[pageIdx + '-offsetHeight'];
|
||||||
|
|
||||||
|
|
||||||
for (const draggableData of draggablesData) {
|
for (const draggableData of draggablesData) {
|
||||||
// embed the draggable canvas
|
// embed the draggable canvas
|
||||||
@ -389,8 +383,8 @@ const DraggableUtils = {
|
|||||||
const pdfImageObject = await pdfDocModified.embedPng(draggableImgBytes);
|
const pdfImageObject = await pdfDocModified.embedPng(draggableImgBytes);
|
||||||
|
|
||||||
// calculate the position in the pdf document
|
// calculate the position in the pdf document
|
||||||
const tansform = draggableElement.style.transform.replace(/[^.,-\d]/g, "");
|
const tansform = draggableElement.style.transform.replace(/[^.,-\d]/g, '');
|
||||||
const transformComponents = tansform.split(",");
|
const transformComponents = tansform.split(',');
|
||||||
const draggablePositionPixels = {
|
const draggablePositionPixels = {
|
||||||
x: parseFloat(transformComponents[0]),
|
x: parseFloat(transformComponents[0]),
|
||||||
y: parseFloat(transformComponents[1]),
|
y: parseFloat(transformComponents[1]),
|
||||||
@ -429,9 +423,8 @@ const DraggableUtils = {
|
|||||||
};
|
};
|
||||||
|
|
||||||
//Defining the image if the page has a 0-degree angle
|
//Defining the image if the page has a 0-degree angle
|
||||||
let x = draggablePositionPdf.x
|
let x = draggablePositionPdf.x;
|
||||||
let y = heightAdjusted - draggablePositionPdf.y - draggablePositionPdf.height
|
let y = heightAdjusted - draggablePositionPdf.y - draggablePositionPdf.height;
|
||||||
|
|
||||||
|
|
||||||
//Defining the image position if it is at other angles
|
//Defining the image position if it is at other angles
|
||||||
if (normalizedAngle === 90) {
|
if (normalizedAngle === 90) {
|
||||||
@ -451,7 +444,7 @@ const DraggableUtils = {
|
|||||||
y: y,
|
y: y,
|
||||||
width: draggablePositionPdf.width,
|
width: draggablePositionPdf.width,
|
||||||
height: draggablePositionPdf.height,
|
height: draggablePositionPdf.height,
|
||||||
rotate: rotation
|
rotate: rotation,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -460,6 +453,6 @@ const DraggableUtils = {
|
|||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
document.addEventListener("DOMContentLoaded", () => {
|
document.addEventListener('DOMContentLoaded', () => {
|
||||||
DraggableUtils.init();
|
DraggableUtils.init();
|
||||||
});
|
});
|
||||||
|
@ -1,20 +1,20 @@
|
|||||||
import FileIconFactory from "./file-icon-factory.js";
|
import FileIconFactory from './file-icon-factory.js';
|
||||||
import FileUtils from "./file-utils.js";
|
import FileUtils from './file-utils.js';
|
||||||
import UUID from './uuid.js';
|
import UUID from './uuid.js';
|
||||||
|
import {DecryptFile} from './DecryptFiles.js';
|
||||||
|
const decryptFile = new DecryptFile();
|
||||||
let isScriptExecuted = false;
|
let isScriptExecuted = false;
|
||||||
if (!isScriptExecuted) {
|
if (!isScriptExecuted) {
|
||||||
isScriptExecuted = true;
|
isScriptExecuted = true;
|
||||||
document.addEventListener("DOMContentLoaded", function () {
|
document.addEventListener('DOMContentLoaded', function () {
|
||||||
document.querySelectorAll(".custom-file-chooser").forEach(setupFileInput);
|
document.querySelectorAll('.custom-file-chooser').forEach(setupFileInput);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
function setupFileInput(chooser) {
|
function setupFileInput(chooser) {
|
||||||
const elementId = chooser.getAttribute("data-bs-element-id");
|
const elementId = chooser.getAttribute('data-bs-element-id');
|
||||||
const filesSelected = chooser.getAttribute("data-bs-files-selected");
|
const filesSelected = chooser.getAttribute('data-bs-files-selected');
|
||||||
const pdfPrompt = chooser.getAttribute("data-bs-pdf-prompt");
|
const pdfPrompt = chooser.getAttribute('data-bs-pdf-prompt');
|
||||||
const inputContainerId = chooser.getAttribute('data-bs-element-container-id');
|
const inputContainerId = chooser.getAttribute('data-bs-element-container-id');
|
||||||
|
|
||||||
let inputContainer = document.getElementById(inputContainerId);
|
let inputContainer = document.getElementById(inputContainerId);
|
||||||
@ -26,7 +26,7 @@ function setupFileInput(chooser) {
|
|||||||
inputContainer.addEventListener('click', (e) => {
|
inputContainer.addEventListener('click', (e) => {
|
||||||
let inputBtn = document.getElementById(elementId);
|
let inputBtn = document.getElementById(elementId);
|
||||||
inputBtn.click();
|
inputBtn.click();
|
||||||
})
|
});
|
||||||
|
|
||||||
const dragenterListener = function () {
|
const dragenterListener = function () {
|
||||||
dragCounter++;
|
dragCounter++;
|
||||||
@ -63,7 +63,7 @@ function setupFileInput(chooser) {
|
|||||||
const files = dt.files;
|
const files = dt.files;
|
||||||
|
|
||||||
const fileInput = document.getElementById(elementId);
|
const fileInput = document.getElementById(elementId);
|
||||||
if (fileInput?.hasAttribute("multiple")) {
|
if (fileInput?.hasAttribute('multiple')) {
|
||||||
pushFileListTo(files, allFiles);
|
pushFileListTo(files, allFiles);
|
||||||
} else if (fileInput) {
|
} else if (fileInput) {
|
||||||
allFiles = [files[0]];
|
allFiles = [files[0]];
|
||||||
@ -78,7 +78,7 @@ function setupFileInput(chooser) {
|
|||||||
|
|
||||||
dragCounter = 0;
|
dragCounter = 0;
|
||||||
|
|
||||||
fileInput.dispatchEvent(new CustomEvent("change", { bubbles: true, detail: {source: 'drag-drop'} }));
|
fileInput.dispatchEvent(new CustomEvent('change', {bubbles: true, detail: {source: 'drag-drop'}}));
|
||||||
};
|
};
|
||||||
|
|
||||||
function pushFileListTo(fileList, container) {
|
function pushFileListTo(fileList, container) {
|
||||||
@ -87,7 +87,7 @@ function setupFileInput(chooser) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
["dragenter", "dragover", "dragleave", "drop"].forEach((eventName) => {
|
['dragenter', 'dragover', 'dragleave', 'drop'].forEach((eventName) => {
|
||||||
document.body.addEventListener(eventName, preventDefaults, false);
|
document.body.addEventListener(eventName, preventDefaults, false);
|
||||||
});
|
});
|
||||||
|
|
||||||
@ -96,37 +96,49 @@ function setupFileInput(chooser) {
|
|||||||
e.stopPropagation();
|
e.stopPropagation();
|
||||||
}
|
}
|
||||||
|
|
||||||
document.body.addEventListener("dragenter", dragenterListener);
|
document.body.addEventListener('dragenter', dragenterListener);
|
||||||
document.body.addEventListener("dragleave", dragleaveListener);
|
document.body.addEventListener('dragleave', dragleaveListener);
|
||||||
document.body.addEventListener("drop", dropListener);
|
document.body.addEventListener('drop', dropListener);
|
||||||
|
|
||||||
$("#" + elementId).on("change", function (e) {
|
$('#' + elementId).on('change', async function (e) {
|
||||||
let element = e.target;
|
let element = e.target;
|
||||||
const isDragAndDrop = e.detail?.source == 'drag-drop';
|
const isDragAndDrop = e.detail?.source == 'drag-drop';
|
||||||
|
|
||||||
if (element instanceof HTMLInputElement && element.hasAttribute("multiple")) {
|
if (element instanceof HTMLInputElement && element.hasAttribute('multiple')) {
|
||||||
allFiles = isDragAndDrop ? allFiles : [...allFiles, ...element.files];
|
allFiles = isDragAndDrop ? allFiles : [...allFiles, ...element.files];
|
||||||
} else {
|
} else {
|
||||||
allFiles = Array.from(isDragAndDrop ? allFiles : [element.files[0]]);
|
allFiles = Array.from(isDragAndDrop ? allFiles : [element.files[0]]);
|
||||||
}
|
}
|
||||||
|
allFiles = await Promise.all(
|
||||||
allFiles = allFiles.map(file => {
|
allFiles.map(async (file) => {
|
||||||
if (!file.uniqueId) file.uniqueId = UUID.uuidv4();
|
let decryptedFile = file;
|
||||||
|
try {
|
||||||
|
const decryptFile = new DecryptFile();
|
||||||
|
const {isEncrypted, requiresPassword} = await decryptFile.checkFileEncrypted(file);
|
||||||
|
if (file.type === 'application/pdf' && isEncrypted) {
|
||||||
|
decryptedFile = await decryptFile.decryptFile(file, requiresPassword);
|
||||||
|
if (!decryptedFile) throw new Error('File decryption failed.');
|
||||||
|
}
|
||||||
|
decryptedFile.uniqueId = UUID.uuidv4();
|
||||||
|
return decryptedFile;
|
||||||
|
} catch (error) {
|
||||||
|
console.error(`Error decrypting file: ${file.name}`, error);
|
||||||
return file;
|
return file;
|
||||||
});
|
}
|
||||||
|
})
|
||||||
|
);
|
||||||
if (!isDragAndDrop) {
|
if (!isDragAndDrop) {
|
||||||
let dataTransfer = toDataTransfer(allFiles);
|
let dataTransfer = toDataTransfer(allFiles);
|
||||||
element.files = dataTransfer.files;
|
element.files = dataTransfer.files;
|
||||||
}
|
}
|
||||||
|
|
||||||
handleFileInputChange(this);
|
handleFileInputChange(this);
|
||||||
this.dispatchEvent(new CustomEvent("file-input-change", { bubbles: true }));
|
this.dispatchEvent(new CustomEvent('file-input-change', {bubbles: true, detail: {elementId, allFiles}}));
|
||||||
});
|
});
|
||||||
|
|
||||||
function toDataTransfer(files) {
|
function toDataTransfer(files) {
|
||||||
let dataTransfer = new DataTransfer();
|
let dataTransfer = new DataTransfer();
|
||||||
files.forEach(file => dataTransfer.items.add(file));
|
files.forEach((file) => dataTransfer.items.add(file));
|
||||||
return dataTransfer;
|
return dataTransfer;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -136,7 +148,7 @@ function setupFileInput(chooser) {
|
|||||||
|
|
||||||
const filesInfo = files.map((f) => ({name: f.name, size: f.size, uniqueId: f.uniqueId}));
|
const filesInfo = files.map((f) => ({name: f.name, size: f.size, uniqueId: f.uniqueId}));
|
||||||
|
|
||||||
const selectedFilesContainer = $(inputContainer).siblings(".selected-files");
|
const selectedFilesContainer = $(inputContainer).siblings('.selected-files');
|
||||||
selectedFilesContainer.empty();
|
selectedFilesContainer.empty();
|
||||||
filesInfo.forEach((info) => {
|
filesInfo.forEach((info) => {
|
||||||
let fileContainerClasses = 'small-file-container d-flex flex-column justify-content-center align-items-center';
|
let fileContainerClasses = 'small-file-container d-flex flex-column justify-content-center align-items-center';
|
||||||
@ -167,28 +179,26 @@ function setupFileInput(chooser) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
function showOrHideSelectedFilesContainer(files) {
|
function showOrHideSelectedFilesContainer(files) {
|
||||||
if (files && files.length > 0)
|
if (files && files.length > 0) chooser.style.setProperty('--selected-files-display', 'flex');
|
||||||
chooser.style.setProperty('--selected-files-display', 'flex');
|
else chooser.style.setProperty('--selected-files-display', 'none');
|
||||||
else
|
|
||||||
chooser.style.setProperty('--selected-files-display', 'none');
|
|
||||||
}
|
}
|
||||||
|
|
||||||
function removeFileListener(e) {
|
function removeFileListener(e) {
|
||||||
const fileId = (e.target).getAttribute('data-file-id');
|
const fileId = e.target.getAttribute('data-file-id');
|
||||||
|
|
||||||
let inputElement = document.getElementById(elementId);
|
let inputElement = document.getElementById(elementId);
|
||||||
removeFileById(fileId, inputElement);
|
removeFileById(fileId, inputElement);
|
||||||
|
|
||||||
showOrHideSelectedFilesContainer(allFiles);
|
showOrHideSelectedFilesContainer(allFiles);
|
||||||
|
|
||||||
inputElement.dispatchEvent(new CustomEvent("file-input-change", { bubbles: true }));
|
inputElement.dispatchEvent(new CustomEvent('file-input-change', {bubbles: true}));
|
||||||
}
|
}
|
||||||
|
|
||||||
function removeFileById(fileId, inputElement) {
|
function removeFileById(fileId, inputElement) {
|
||||||
let fileContainer = document.getElementById(fileId);
|
let fileContainer = document.getElementById(fileId);
|
||||||
fileContainer.remove();
|
fileContainer.remove();
|
||||||
|
|
||||||
allFiles = allFiles.filter(v => v.uniqueId != fileId);
|
allFiles = allFiles.filter((v) => v.uniqueId != fileId);
|
||||||
let dataTransfer = toDataTransfer(allFiles);
|
let dataTransfer = toDataTransfer(allFiles);
|
||||||
|
|
||||||
if (inputElement) inputElement.files = dataTransfer.files;
|
if (inputElement) inputElement.files = dataTransfer.files;
|
||||||
@ -207,23 +217,19 @@ function setupFileInput(chooser) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
function createFileInfoContainer(info) {
|
function createFileInfoContainer(info) {
|
||||||
let fileInfoContainer = document.createElement("div");
|
let fileInfoContainer = document.createElement('div');
|
||||||
let fileInfoContainerClasses = 'file-info d-flex flex-column align-items-center justify-content-center';
|
let fileInfoContainerClasses = 'file-info d-flex flex-column align-items-center justify-content-center';
|
||||||
|
|
||||||
$(fileInfoContainer).addClass(fileInfoContainerClasses);
|
$(fileInfoContainer).addClass(fileInfoContainerClasses);
|
||||||
|
|
||||||
$(fileInfoContainer).append(
|
$(fileInfoContainer).append(`<div title="${info.name}">${info.name}</div>`);
|
||||||
`<div title="${info.name}">${info.name}</div>`
|
|
||||||
);
|
|
||||||
let fileSizeWithUnits = FileUtils.transformFileSize(info.size);
|
let fileSizeWithUnits = FileUtils.transformFileSize(info.size);
|
||||||
$(fileInfoContainer).append(
|
$(fileInfoContainer).append(`<div title="${info.size}">${fileSizeWithUnits}</div>`);
|
||||||
`<div title="${info.size}">${fileSizeWithUnits}</div>`
|
|
||||||
);
|
|
||||||
return fileInfoContainer;
|
return fileInfoContainer;
|
||||||
}
|
}
|
||||||
|
|
||||||
//Listen for event of file being removed and the filter it out of the allFiles array
|
//Listen for event of file being removed and the filter it out of the allFiles array
|
||||||
document.addEventListener("fileRemoved", function (e) {
|
document.addEventListener('fileRemoved', function (e) {
|
||||||
const fileId = e.detail;
|
const fileId = e.detail;
|
||||||
let inputElement = document.getElementById(elementId);
|
let inputElement = document.getElementById(elementId);
|
||||||
removeFileById(fileId, inputElement);
|
removeFileById(fileId, inputElement);
|
||||||
|
@ -5,7 +5,7 @@ import {SplitAllCommand} from './commands/split.js';
|
|||||||
import {UndoManager} from './UndoManager.js';
|
import {UndoManager} from './UndoManager.js';
|
||||||
import {PageBreakCommand} from './commands/page-break.js';
|
import {PageBreakCommand} from './commands/page-break.js';
|
||||||
import {AddFilesCommand} from './commands/add-page.js';
|
import {AddFilesCommand} from './commands/add-page.js';
|
||||||
import {DecryptFile} from './DecryptFiles.js';
|
import {DecryptFile} from '../DecryptFiles.js';
|
||||||
|
|
||||||
class PdfContainer {
|
class PdfContainer {
|
||||||
fileName;
|
fileName;
|
||||||
|
253
src/main/resources/static/js/pages/adjust-contrast.js
Normal file
253
src/main/resources/static/js/pages/adjust-contrast.js
Normal file
@ -0,0 +1,253 @@
|
|||||||
|
var canvas = document.getElementById('contrast-pdf-canvas');
|
||||||
|
var context = canvas.getContext('2d');
|
||||||
|
var originalImageData = null;
|
||||||
|
var allPages = [];
|
||||||
|
var pdfDoc = null;
|
||||||
|
var pdf = null; // This is the current PDF document
|
||||||
|
|
||||||
|
async function renderPDFAndSaveOriginalImageData(file) {
|
||||||
|
var fileReader = new FileReader();
|
||||||
|
fileReader.onload = async function () {
|
||||||
|
var data = new Uint8Array(this.result);
|
||||||
|
pdfjsLib.GlobalWorkerOptions.workerSrc = './pdfjs-legacy/pdf.worker.mjs';
|
||||||
|
pdf = await pdfjsLib.getDocument({data: data}).promise;
|
||||||
|
|
||||||
|
// Get the number of pages in the PDF
|
||||||
|
var numPages = pdf.numPages;
|
||||||
|
allPages = Array.from({length: numPages}, (_, i) => i + 1);
|
||||||
|
|
||||||
|
// Create a new PDF document
|
||||||
|
pdfDoc = await PDFLib.PDFDocument.create();
|
||||||
|
// Render the first page in the viewer
|
||||||
|
await renderPageAndAdjustImageProperties(1);
|
||||||
|
document.getElementById('sliders-container').style.display = 'block';
|
||||||
|
};
|
||||||
|
fileReader.readAsArrayBuffer(file);
|
||||||
|
}
|
||||||
|
|
||||||
|
// This function is now async and returns a promise
|
||||||
|
function renderPageAndAdjustImageProperties(pageNum) {
|
||||||
|
return new Promise(async function (resolve, reject) {
|
||||||
|
var page = await pdf.getPage(pageNum);
|
||||||
|
var scale = 1.5;
|
||||||
|
var viewport = page.getViewport({scale: scale});
|
||||||
|
|
||||||
|
canvas.height = viewport.height;
|
||||||
|
canvas.width = viewport.width;
|
||||||
|
|
||||||
|
var renderContext = {
|
||||||
|
canvasContext: context,
|
||||||
|
viewport: viewport,
|
||||||
|
};
|
||||||
|
|
||||||
|
var renderTask = page.render(renderContext);
|
||||||
|
renderTask.promise.then(function () {
|
||||||
|
originalImageData = context.getImageData(0, 0, canvas.width, canvas.height);
|
||||||
|
adjustImageProperties();
|
||||||
|
resolve();
|
||||||
|
});
|
||||||
|
canvas.classList.add('fixed-shadow-canvas');
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
function adjustImageProperties() {
|
||||||
|
var contrast = parseFloat(document.getElementById('contrast-slider').value);
|
||||||
|
var brightness = parseFloat(document.getElementById('brightness-slider').value);
|
||||||
|
var saturation = parseFloat(document.getElementById('saturation-slider').value);
|
||||||
|
|
||||||
|
contrast /= 100; // normalize to range [0, 2]
|
||||||
|
brightness /= 100; // normalize to range [0, 2]
|
||||||
|
saturation /= 100; // normalize to range [0, 2]
|
||||||
|
|
||||||
|
if (originalImageData) {
|
||||||
|
var newImageData = context.createImageData(originalImageData.width, originalImageData.height);
|
||||||
|
newImageData.data.set(originalImageData.data);
|
||||||
|
|
||||||
|
for (var i = 0; i < newImageData.data.length; i += 4) {
|
||||||
|
var r = newImageData.data[i];
|
||||||
|
var g = newImageData.data[i + 1];
|
||||||
|
var b = newImageData.data[i + 2];
|
||||||
|
// Adjust contrast
|
||||||
|
r = adjustContrastForPixel(r, contrast);
|
||||||
|
g = adjustContrastForPixel(g, contrast);
|
||||||
|
b = adjustContrastForPixel(b, contrast);
|
||||||
|
// Adjust brightness
|
||||||
|
r = adjustBrightnessForPixel(r, brightness);
|
||||||
|
g = adjustBrightnessForPixel(g, brightness);
|
||||||
|
b = adjustBrightnessForPixel(b, brightness);
|
||||||
|
// Adjust saturation
|
||||||
|
var rgb = adjustSaturationForPixel(r, g, b, saturation);
|
||||||
|
newImageData.data[i] = rgb[0];
|
||||||
|
newImageData.data[i + 1] = rgb[1];
|
||||||
|
newImageData.data[i + 2] = rgb[2];
|
||||||
|
}
|
||||||
|
context.putImageData(newImageData, 0, 0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function rgbToHsl(r, g, b) {
|
||||||
|
(r /= 255), (g /= 255), (b /= 255);
|
||||||
|
|
||||||
|
var max = Math.max(r, g, b),
|
||||||
|
min = Math.min(r, g, b);
|
||||||
|
var h,
|
||||||
|
s,
|
||||||
|
l = (max + min) / 2;
|
||||||
|
|
||||||
|
if (max === min) {
|
||||||
|
h = s = 0; // achromatic
|
||||||
|
} else {
|
||||||
|
var d = max - min;
|
||||||
|
s = l > 0.5 ? d / (2 - max - min) : d / (max + min);
|
||||||
|
|
||||||
|
switch (max) {
|
||||||
|
case r:
|
||||||
|
h = (g - b) / d + (g < b ? 6 : 0);
|
||||||
|
break;
|
||||||
|
case g:
|
||||||
|
h = (b - r) / d + 2;
|
||||||
|
break;
|
||||||
|
case b:
|
||||||
|
h = (r - g) / d + 4;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
h /= 6;
|
||||||
|
}
|
||||||
|
|
||||||
|
return [h, s, l];
|
||||||
|
}
|
||||||
|
|
||||||
|
function hslToRgb(h, s, l) {
|
||||||
|
var r, g, b;
|
||||||
|
|
||||||
|
if (s === 0) {
|
||||||
|
r = g = b = l; // achromatic
|
||||||
|
} else {
|
||||||
|
var hue2rgb = function hue2rgb(p, q, t) {
|
||||||
|
if (t < 0) t += 1;
|
||||||
|
if (t > 1) t -= 1;
|
||||||
|
if (t < 1 / 6) return p + (q - p) * 6 * t;
|
||||||
|
if (t < 1 / 2) return q;
|
||||||
|
if (t < 2 / 3) return p + (q - p) * (2 / 3 - t) * 6;
|
||||||
|
return p;
|
||||||
|
};
|
||||||
|
|
||||||
|
var q = l < 0.5 ? l * (1 + s) : l + s - l * s;
|
||||||
|
var p = 2 * l - q;
|
||||||
|
|
||||||
|
r = hue2rgb(p, q, h + 1 / 3);
|
||||||
|
g = hue2rgb(p, q, h);
|
||||||
|
b = hue2rgb(p, q, h - 1 / 3);
|
||||||
|
}
|
||||||
|
|
||||||
|
return [r * 255, g * 255, b * 255];
|
||||||
|
}
|
||||||
|
|
||||||
|
function adjustContrastForPixel(pixel, contrast) {
|
||||||
|
// Normalize to range [-0.5, 0.5]
|
||||||
|
var normalized = pixel / 255 - 0.5;
|
||||||
|
|
||||||
|
// Apply contrast
|
||||||
|
normalized *= contrast;
|
||||||
|
|
||||||
|
// Denormalize back to [0, 255]
|
||||||
|
return (normalized + 0.5) * 255;
|
||||||
|
}
|
||||||
|
|
||||||
|
function clamp(value, min, max) {
|
||||||
|
return Math.min(Math.max(value, min), max);
|
||||||
|
}
|
||||||
|
|
||||||
|
function adjustSaturationForPixel(r, g, b, saturation) {
|
||||||
|
var hsl = rgbToHsl(r, g, b);
|
||||||
|
|
||||||
|
// Adjust saturation
|
||||||
|
hsl[1] = clamp(hsl[1] * saturation, 0, 1);
|
||||||
|
|
||||||
|
// Convert back to RGB
|
||||||
|
var rgb = hslToRgb(hsl[0], hsl[1], hsl[2]);
|
||||||
|
|
||||||
|
// Return adjusted RGB values
|
||||||
|
return rgb;
|
||||||
|
}
|
||||||
|
|
||||||
|
function adjustBrightnessForPixel(pixel, brightness) {
|
||||||
|
return Math.max(0, Math.min(255, pixel * brightness));
|
||||||
|
}
|
||||||
|
let inputFileName = '';
|
||||||
|
async function downloadPDF() {
|
||||||
|
for (var i = 0; i < allPages.length; i++) {
|
||||||
|
await renderPageAndAdjustImageProperties(allPages[i]);
|
||||||
|
const pngImageBytes = canvas.toDataURL('image/png');
|
||||||
|
const pngImage = await pdfDoc.embedPng(pngImageBytes);
|
||||||
|
const pngDims = pngImage.scale(1);
|
||||||
|
|
||||||
|
// Create a blank page matching the dimensions of the image
|
||||||
|
const page = pdfDoc.addPage([pngDims.width, pngDims.height]);
|
||||||
|
|
||||||
|
// Draw the PNG image
|
||||||
|
page.drawImage(pngImage, {
|
||||||
|
x: 0,
|
||||||
|
y: 0,
|
||||||
|
width: pngDims.width,
|
||||||
|
height: pngDims.height,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
// Serialize the PDFDocument to bytes (a Uint8Array)
|
||||||
|
const pdfBytes = await pdfDoc.save();
|
||||||
|
|
||||||
|
// Create a Blob
|
||||||
|
const blob = new Blob([pdfBytes.buffer], {type: 'application/pdf'});
|
||||||
|
|
||||||
|
// Create download link
|
||||||
|
const downloadLink = document.createElement('a');
|
||||||
|
downloadLink.href = URL.createObjectURL(blob);
|
||||||
|
let newFileName = inputFileName ? inputFileName.replace('.pdf', '') : 'download';
|
||||||
|
newFileName += '_adjusted_color.pdf';
|
||||||
|
|
||||||
|
downloadLink.download = newFileName;
|
||||||
|
downloadLink.click();
|
||||||
|
|
||||||
|
// After download, reset the viewer and clear stored data
|
||||||
|
allPages = []; // Clear the pages
|
||||||
|
originalImageData = null; // Clear the image data
|
||||||
|
|
||||||
|
// Go back to page 1 and render it in the viewer
|
||||||
|
if (pdf !== null) {
|
||||||
|
renderPageAndAdjustImageProperties(1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Event listeners
|
||||||
|
document.getElementById('fileInput-input').addEventListener('change', function (e) {
|
||||||
|
const fileInput = event.target;
|
||||||
|
fileInput.addEventListener('file-input-change', async (e) => {
|
||||||
|
const {allFiles} = e.detail;
|
||||||
|
if (allFiles && allFiles.length > 0) {
|
||||||
|
const file = allFiles[0];
|
||||||
|
inputFileName = file.name;
|
||||||
|
renderPDFAndSaveOriginalImageData(file);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
document.getElementById('contrast-slider').addEventListener('input', function () {
|
||||||
|
document.getElementById('contrast-val').textContent = this.value;
|
||||||
|
adjustImageProperties();
|
||||||
|
});
|
||||||
|
|
||||||
|
document.getElementById('brightness-slider').addEventListener('input', function () {
|
||||||
|
document.getElementById('brightness-val').textContent = this.value;
|
||||||
|
adjustImageProperties();
|
||||||
|
});
|
||||||
|
|
||||||
|
document.getElementById('saturation-slider').addEventListener('input', function () {
|
||||||
|
document.getElementById('saturation-val').textContent = this.value;
|
||||||
|
adjustImageProperties();
|
||||||
|
});
|
||||||
|
|
||||||
|
document.getElementById('download-button').addEventListener('click', function () {
|
||||||
|
downloadPDF();
|
||||||
|
});
|
215
src/main/resources/static/js/pages/sign.js
Normal file
215
src/main/resources/static/js/pages/sign.js
Normal file
@ -0,0 +1,215 @@
|
|||||||
|
window.toggleSignatureView = toggleSignatureView;
|
||||||
|
window.previewSignature = previewSignature;
|
||||||
|
window.addSignatureFromPreview = addSignatureFromPreview;
|
||||||
|
window.addDraggableFromPad = addDraggableFromPad;
|
||||||
|
window.addDraggableFromText = addDraggableFromText;
|
||||||
|
window.goToFirstOrLastPage = goToFirstOrLastPage;
|
||||||
|
|
||||||
|
let currentPreviewSrc = null;
|
||||||
|
|
||||||
|
function toggleSignatureView() {
|
||||||
|
const gridView = document.getElementById('gridView');
|
||||||
|
const listView = document.getElementById('listView');
|
||||||
|
const gridText = document.querySelector('.grid-view-text');
|
||||||
|
const listText = document.querySelector('.list-view-text');
|
||||||
|
|
||||||
|
if (gridView.style.display !== 'none') {
|
||||||
|
gridView.style.display = 'none';
|
||||||
|
listView.style.display = 'block';
|
||||||
|
gridText.style.display = 'none';
|
||||||
|
listText.style.display = 'inline';
|
||||||
|
} else {
|
||||||
|
gridView.style.display = 'block';
|
||||||
|
listView.style.display = 'none';
|
||||||
|
gridText.style.display = 'inline';
|
||||||
|
listText.style.display = 'none';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function previewSignature(element) {
|
||||||
|
const src = element.dataset.src;
|
||||||
|
currentPreviewSrc = src;
|
||||||
|
|
||||||
|
const filename = element.querySelector('.signature-list-name').textContent;
|
||||||
|
|
||||||
|
const previewImage = document.getElementById('previewImage');
|
||||||
|
const previewFileName = document.getElementById('previewFileName');
|
||||||
|
|
||||||
|
previewImage.src = src;
|
||||||
|
previewFileName.textContent = filename;
|
||||||
|
|
||||||
|
const modal = new bootstrap.Modal(document.getElementById('signaturePreview'));
|
||||||
|
modal.show();
|
||||||
|
}
|
||||||
|
|
||||||
|
function addSignatureFromPreview() {
|
||||||
|
if (currentPreviewSrc) {
|
||||||
|
DraggableUtils.createDraggableCanvasFromUrl(currentPreviewSrc);
|
||||||
|
bootstrap.Modal.getInstance(document.getElementById('signaturePreview')).hide();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
let originalFileName = '';
|
||||||
|
document.querySelector('input[name=pdf-upload]').addEventListener('change', async (event) => {
|
||||||
|
const fileInput = event.target;
|
||||||
|
|
||||||
|
// Wait for the second function to complete
|
||||||
|
fileInput.addEventListener('file-input-change', async (e) => {
|
||||||
|
const {allFiles} = e.detail;
|
||||||
|
if (allFiles && allFiles.length > 0) {
|
||||||
|
const file = allFiles[0];
|
||||||
|
originalFileName = file.name.replace(/\.[^/.]+$/, '');
|
||||||
|
const pdfData = await file.arrayBuffer();
|
||||||
|
pdfjsLib.GlobalWorkerOptions.workerSrc = './pdfjs-legacy/pdf.worker.mjs';
|
||||||
|
const pdfDoc = await pdfjsLib.getDocument({data: pdfData}).promise;
|
||||||
|
await DraggableUtils.renderPage(pdfDoc, 0);
|
||||||
|
|
||||||
|
document.querySelectorAll('.show-on-file-selected').forEach((el) => {
|
||||||
|
el.style.cssText = '';
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
document.addEventListener('DOMContentLoaded', () => {
|
||||||
|
document.querySelectorAll('.show-on-file-selected').forEach((el) => {
|
||||||
|
el.style.cssText = 'display:none !important';
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
const imageUpload = document.querySelector('input[name=image-upload]');
|
||||||
|
imageUpload.addEventListener('change', (e) => {
|
||||||
|
if (!e.target.files) return;
|
||||||
|
for (const imageFile of e.target.files) {
|
||||||
|
var reader = new FileReader();
|
||||||
|
reader.readAsDataURL(imageFile);
|
||||||
|
reader.onloadend = function (e) {
|
||||||
|
DraggableUtils.createDraggableCanvasFromUrl(e.target.result);
|
||||||
|
};
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
const signaturePadCanvas = document.getElementById('drawing-pad-canvas');
|
||||||
|
const signaturePad = new SignaturePad(signaturePadCanvas, {
|
||||||
|
minWidth: 1,
|
||||||
|
maxWidth: 2,
|
||||||
|
penColor: 'black',
|
||||||
|
});
|
||||||
|
|
||||||
|
function addDraggableFromPad() {
|
||||||
|
if (signaturePad.isEmpty()) return;
|
||||||
|
const startTime = Date.now();
|
||||||
|
const croppedDataUrl = getCroppedCanvasDataUrl(signaturePadCanvas);
|
||||||
|
console.log(Date.now() - startTime);
|
||||||
|
DraggableUtils.createDraggableCanvasFromUrl(croppedDataUrl);
|
||||||
|
}
|
||||||
|
|
||||||
|
function getCroppedCanvasDataUrl(canvas) {
|
||||||
|
let originalCtx = canvas.getContext('2d');
|
||||||
|
let originalWidth = canvas.width;
|
||||||
|
let originalHeight = canvas.height;
|
||||||
|
let imageData = originalCtx.getImageData(0, 0, originalWidth, originalHeight);
|
||||||
|
|
||||||
|
let minX = originalWidth + 1,
|
||||||
|
maxX = -1,
|
||||||
|
minY = originalHeight + 1,
|
||||||
|
maxY = -1,
|
||||||
|
x = 0,
|
||||||
|
y = 0,
|
||||||
|
currentPixelColorValueIndex;
|
||||||
|
|
||||||
|
for (y = 0; y < originalHeight; y++) {
|
||||||
|
for (x = 0; x < originalWidth; x++) {
|
||||||
|
currentPixelColorValueIndex = (y * originalWidth + x) * 4;
|
||||||
|
let currentPixelAlphaValue = imageData.data[currentPixelColorValueIndex + 3];
|
||||||
|
if (currentPixelAlphaValue > 0) {
|
||||||
|
if (minX > x) minX = x;
|
||||||
|
if (maxX < x) maxX = x;
|
||||||
|
if (minY > y) minY = y;
|
||||||
|
if (maxY < y) maxY = y;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
let croppedWidth = maxX - minX;
|
||||||
|
let croppedHeight = maxY - minY;
|
||||||
|
if (croppedWidth < 0 || croppedHeight < 0) return null;
|
||||||
|
let cuttedImageData = originalCtx.getImageData(minX, minY, croppedWidth, croppedHeight);
|
||||||
|
|
||||||
|
let croppedCanvas = document.createElement('canvas'),
|
||||||
|
croppedCtx = croppedCanvas.getContext('2d');
|
||||||
|
|
||||||
|
croppedCanvas.width = croppedWidth;
|
||||||
|
croppedCanvas.height = croppedHeight;
|
||||||
|
croppedCtx.putImageData(cuttedImageData, 0, 0);
|
||||||
|
|
||||||
|
return croppedCanvas.toDataURL();
|
||||||
|
}
|
||||||
|
|
||||||
|
function resizeCanvas() {
|
||||||
|
var ratio = Math.max(window.devicePixelRatio || 1, 1);
|
||||||
|
var additionalFactor = 10;
|
||||||
|
|
||||||
|
signaturePadCanvas.width = signaturePadCanvas.offsetWidth * ratio * additionalFactor;
|
||||||
|
signaturePadCanvas.height = signaturePadCanvas.offsetHeight * ratio * additionalFactor;
|
||||||
|
signaturePadCanvas.getContext('2d').scale(ratio * additionalFactor, ratio * additionalFactor);
|
||||||
|
|
||||||
|
signaturePad.clear();
|
||||||
|
}
|
||||||
|
|
||||||
|
new IntersectionObserver((entries, observer) => {
|
||||||
|
if (entries.some((entry) => entry.intersectionRatio > 0)) {
|
||||||
|
resizeCanvas();
|
||||||
|
}
|
||||||
|
}).observe(signaturePadCanvas);
|
||||||
|
|
||||||
|
new ResizeObserver(resizeCanvas).observe(signaturePadCanvas);
|
||||||
|
|
||||||
|
function addDraggableFromText() {
|
||||||
|
const sigText = document.getElementById('sigText').value;
|
||||||
|
const font = document.querySelector('select[name=font]').value;
|
||||||
|
const fontSize = 100;
|
||||||
|
|
||||||
|
const canvas = document.createElement('canvas');
|
||||||
|
const ctx = canvas.getContext('2d');
|
||||||
|
ctx.font = `${fontSize}px ${font}`;
|
||||||
|
const textWidth = ctx.measureText(sigText).width;
|
||||||
|
const textHeight = fontSize;
|
||||||
|
|
||||||
|
let paragraphs = sigText.split(/\r?\n/);
|
||||||
|
|
||||||
|
canvas.width = textWidth;
|
||||||
|
canvas.height = paragraphs.length * textHeight * 1.35; // for tails
|
||||||
|
ctx.font = `${fontSize}px ${font}`;
|
||||||
|
|
||||||
|
ctx.textBaseline = 'top';
|
||||||
|
|
||||||
|
let y = 0;
|
||||||
|
|
||||||
|
paragraphs.forEach((paragraph) => {
|
||||||
|
ctx.fillText(paragraph, 0, y);
|
||||||
|
y += fontSize;
|
||||||
|
});
|
||||||
|
|
||||||
|
const dataURL = canvas.toDataURL();
|
||||||
|
DraggableUtils.createDraggableCanvasFromUrl(dataURL);
|
||||||
|
}
|
||||||
|
|
||||||
|
async function goToFirstOrLastPage(page) {
|
||||||
|
if (page) {
|
||||||
|
const lastPage = DraggableUtils.pdfDoc.numPages;
|
||||||
|
await DraggableUtils.goToPage(lastPage - 1);
|
||||||
|
} else {
|
||||||
|
await DraggableUtils.goToPage(0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
document.getElementById('download-pdf').addEventListener('click', async () => {
|
||||||
|
const modifiedPdf = await DraggableUtils.getOverlayedPdfDocument();
|
||||||
|
const modifiedPdfBytes = await modifiedPdf.save();
|
||||||
|
const blob = new Blob([modifiedPdfBytes], {type: 'application/pdf'});
|
||||||
|
const link = document.createElement('a');
|
||||||
|
link.href = URL.createObjectURL(blob);
|
||||||
|
link.download = originalFileName + '_signed.pdf';
|
||||||
|
link.click();
|
||||||
|
});
|
@ -204,8 +204,7 @@
|
|||||||
<script type="module" th:src="@{'/pdfjs-legacy/pdf.mjs'}"></script>
|
<script type="module" th:src="@{'/pdfjs-legacy/pdf.mjs'}"></script>
|
||||||
<script th:src="@{'/js/downloader.js'}"></script>
|
<script th:src="@{'/js/downloader.js'}"></script>
|
||||||
<script>
|
<script>
|
||||||
window.translations = {
|
window.decrypt = {
|
||||||
decrypt: {
|
|
||||||
passwordPrompt: '[[#{decrypt.passwordPrompt}]]',
|
passwordPrompt: '[[#{decrypt.passwordPrompt}]]',
|
||||||
cancelled: '[[#{decrypt.cancelled}]]',
|
cancelled: '[[#{decrypt.cancelled}]]',
|
||||||
noPassword: '[[#{decrypt.noPassword}]]',
|
noPassword: '[[#{decrypt.noPassword}]]',
|
||||||
@ -214,7 +213,6 @@
|
|||||||
unexpectedError: '[[#{decrypt.unexpectedError}]]',
|
unexpectedError: '[[#{decrypt.unexpectedError}]]',
|
||||||
serverError: '[[#{decrypt.serverError}]]',
|
serverError: '[[#{decrypt.serverError}]]',
|
||||||
success: '[[#{decrypt.success}]]',
|
success: '[[#{decrypt.success}]]',
|
||||||
}
|
|
||||||
};</script>
|
};</script>
|
||||||
<div class="custom-file-chooser mb-3" th:attr="data-bs-unique-id=${name}, data-bs-element-id=${name+'-input'}, data-bs-element-container-id=${name+'-input-container'}, data-bs-files-selected=#{filesSelected}, data-bs-pdf-prompt=#{pdfPrompt}">
|
<div class="custom-file-chooser mb-3" th:attr="data-bs-unique-id=${name}, data-bs-element-id=${name+'-input'}, data-bs-element-container-id=${name+'-input-container'}, data-bs-files-selected=#{filesSelected}, data-bs-pdf-prompt=#{pdfPrompt}">
|
||||||
<div class="mb-3 d-flex flex-row justify-content-center align-items-center flex-wrap input-container" th:name="${name}+'-input'" th:id="${name}+'-input-container'" th:data-text="#{fileChooser.hoveredDragAndDrop}">
|
<div class="mb-3 d-flex flex-row justify-content-center align-items-center flex-wrap input-container" th:name="${name}+'-input'" th:id="${name}+'-input-container'" th:data-text="#{fileChooser.hoveredDragAndDrop}">
|
||||||
|
@ -55,247 +55,7 @@
|
|||||||
</form>
|
</form>
|
||||||
<script type="module" th:src="@{'/pdfjs-legacy/pdf.mjs'}"></script>
|
<script type="module" th:src="@{'/pdfjs-legacy/pdf.mjs'}"></script>
|
||||||
<script th:src="@{'/js/thirdParty/pdf-lib.min.js'}"></script>
|
<script th:src="@{'/js/thirdParty/pdf-lib.min.js'}"></script>
|
||||||
<script>
|
<script type="module" th:src="@{'/js/pages/adjust-contrast.js'}"></script>
|
||||||
var canvas = document.getElementById('contrast-pdf-canvas');
|
|
||||||
var context = canvas.getContext('2d');
|
|
||||||
var originalImageData = null;
|
|
||||||
var allPages = [];
|
|
||||||
var pdfDoc = null;
|
|
||||||
var pdf = null; // This is the current PDF document
|
|
||||||
|
|
||||||
async function renderPDFAndSaveOriginalImageData(file) {
|
|
||||||
var fileReader = new FileReader();
|
|
||||||
fileReader.onload = async function() {
|
|
||||||
var data = new Uint8Array(this.result);
|
|
||||||
pdfjsLib.GlobalWorkerOptions.workerSrc = './pdfjs-legacy/pdf.worker.mjs'
|
|
||||||
pdf = await pdfjsLib.getDocument({data: data}).promise;
|
|
||||||
|
|
||||||
// Get the number of pages in the PDF
|
|
||||||
var numPages = pdf.numPages;
|
|
||||||
allPages = Array.from({length: numPages}, (_, i) => i + 1);
|
|
||||||
|
|
||||||
// Create a new PDF document
|
|
||||||
pdfDoc = await PDFLib.PDFDocument.create();
|
|
||||||
// Render the first page in the viewer
|
|
||||||
await renderPageAndAdjustImageProperties(1);
|
|
||||||
document.getElementById("sliders-container").style.display = "block";
|
|
||||||
};
|
|
||||||
fileReader.readAsArrayBuffer(file);
|
|
||||||
}
|
|
||||||
|
|
||||||
// This function is now async and returns a promise
|
|
||||||
function renderPageAndAdjustImageProperties(pageNum) {
|
|
||||||
return new Promise(async function(resolve, reject) {
|
|
||||||
var page = await pdf.getPage(pageNum);
|
|
||||||
var scale = 1.5;
|
|
||||||
var viewport = page.getViewport({ scale: scale });
|
|
||||||
|
|
||||||
canvas.height = viewport.height;
|
|
||||||
canvas.width = viewport.width;
|
|
||||||
|
|
||||||
var renderContext = {
|
|
||||||
canvasContext: context,
|
|
||||||
viewport: viewport
|
|
||||||
};
|
|
||||||
|
|
||||||
var renderTask = page.render(renderContext);
|
|
||||||
renderTask.promise.then(function () {
|
|
||||||
originalImageData = context.getImageData(0, 0, canvas.width, canvas.height);
|
|
||||||
adjustImageProperties();
|
|
||||||
resolve();
|
|
||||||
});
|
|
||||||
canvas.classList.add("fixed-shadow-canvas");
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
function adjustImageProperties() {
|
|
||||||
var contrast = parseFloat(document.getElementById('contrast-slider').value);
|
|
||||||
var brightness = parseFloat(document.getElementById('brightness-slider').value);
|
|
||||||
var saturation = parseFloat(document.getElementById('saturation-slider').value);
|
|
||||||
|
|
||||||
contrast /= 100; // normalize to range [0, 2]
|
|
||||||
brightness /= 100; // normalize to range [0, 2]
|
|
||||||
saturation /= 100; // normalize to range [0, 2]
|
|
||||||
|
|
||||||
if (originalImageData) {
|
|
||||||
var newImageData = context.createImageData(originalImageData.width, originalImageData.height);
|
|
||||||
newImageData.data.set(originalImageData.data);
|
|
||||||
|
|
||||||
for(var i=0; i<newImageData.data.length; i+=4) {
|
|
||||||
var r = newImageData.data[i];
|
|
||||||
var g = newImageData.data[i+1];
|
|
||||||
var b = newImageData.data[i+2];
|
|
||||||
// Adjust contrast
|
|
||||||
r = adjustContrastForPixel(r, contrast);
|
|
||||||
g = adjustContrastForPixel(g, contrast);
|
|
||||||
b = adjustContrastForPixel(b, contrast);
|
|
||||||
// Adjust brightness
|
|
||||||
r = adjustBrightnessForPixel(r, brightness);
|
|
||||||
g = adjustBrightnessForPixel(g, brightness);
|
|
||||||
b = adjustBrightnessForPixel(b, brightness);
|
|
||||||
// Adjust saturation
|
|
||||||
var rgb = adjustSaturationForPixel(r, g, b, saturation);
|
|
||||||
newImageData.data[i] = rgb[0];
|
|
||||||
newImageData.data[i+1] = rgb[1];
|
|
||||||
newImageData.data[i+2] = rgb[2];
|
|
||||||
}
|
|
||||||
context.putImageData(newImageData, 0, 0);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
function rgbToHsl(r, g, b) {
|
|
||||||
r /= 255, g /= 255, b /= 255;
|
|
||||||
|
|
||||||
var max = Math.max(r, g, b), min = Math.min(r, g, b);
|
|
||||||
var h, s, l = (max + min) / 2;
|
|
||||||
|
|
||||||
if (max === min) {
|
|
||||||
h = s = 0; // achromatic
|
|
||||||
} else {
|
|
||||||
var d = max - min;
|
|
||||||
s = l > 0.5 ? d / (2 - max - min) : d / (max + min);
|
|
||||||
|
|
||||||
switch (max) {
|
|
||||||
case r: h = (g - b) / d + (g < b ? 6 : 0); break;
|
|
||||||
case g: h = (b - r) / d + 2; break;
|
|
||||||
case b: h = (r - g) / d + 4; break;
|
|
||||||
}
|
|
||||||
|
|
||||||
h /= 6;
|
|
||||||
}
|
|
||||||
|
|
||||||
return [h, s, l];
|
|
||||||
}
|
|
||||||
|
|
||||||
function hslToRgb(h, s, l) {
|
|
||||||
var r, g, b;
|
|
||||||
|
|
||||||
if (s === 0) {
|
|
||||||
r = g = b = l; // achromatic
|
|
||||||
} else {
|
|
||||||
var hue2rgb = function hue2rgb(p, q, t) {
|
|
||||||
if (t < 0) t += 1;
|
|
||||||
if (t > 1) t -= 1;
|
|
||||||
if (t < 1 / 6) return p + (q - p) * 6 * t;
|
|
||||||
if (t < 1 / 2) return q;
|
|
||||||
if (t < 2 / 3) return p + (q - p) * (2 / 3 - t) * 6;
|
|
||||||
return p;
|
|
||||||
};
|
|
||||||
|
|
||||||
var q = l < 0.5 ? l * (1 + s) : l + s - l * s;
|
|
||||||
var p = 2 * l - q;
|
|
||||||
|
|
||||||
r = hue2rgb(p, q, h + 1 / 3);
|
|
||||||
g = hue2rgb(p, q, h);
|
|
||||||
b = hue2rgb(p, q, h - 1 / 3);
|
|
||||||
}
|
|
||||||
|
|
||||||
return [r * 255, g * 255, b * 255];
|
|
||||||
}
|
|
||||||
|
|
||||||
function adjustContrastForPixel(pixel, contrast) {
|
|
||||||
// Normalize to range [-0.5, 0.5]
|
|
||||||
var normalized = pixel / 255 - 0.5;
|
|
||||||
|
|
||||||
// Apply contrast
|
|
||||||
normalized *= contrast;
|
|
||||||
|
|
||||||
// Denormalize back to [0, 255]
|
|
||||||
return (normalized + 0.5) * 255;
|
|
||||||
}
|
|
||||||
|
|
||||||
function clamp(value, min, max) {
|
|
||||||
return Math.min(Math.max(value, min), max);
|
|
||||||
}
|
|
||||||
|
|
||||||
function adjustSaturationForPixel(r, g, b, saturation) {
|
|
||||||
var hsl = rgbToHsl(r, g, b);
|
|
||||||
|
|
||||||
// Adjust saturation
|
|
||||||
hsl[1] = clamp(hsl[1] * saturation, 0, 1);
|
|
||||||
|
|
||||||
// Convert back to RGB
|
|
||||||
var rgb = hslToRgb(hsl[0], hsl[1], hsl[2]);
|
|
||||||
|
|
||||||
// Return adjusted RGB values
|
|
||||||
return rgb;
|
|
||||||
}
|
|
||||||
|
|
||||||
function adjustBrightnessForPixel(pixel, brightness) {
|
|
||||||
return Math.max(0, Math.min(255, pixel * brightness));
|
|
||||||
}
|
|
||||||
let inputFileName = '';
|
|
||||||
async function downloadPDF() {
|
|
||||||
for (var i = 0; i < allPages.length; i++) {
|
|
||||||
await renderPageAndAdjustImageProperties(allPages[i]);
|
|
||||||
const pngImageBytes = canvas.toDataURL('image/png');
|
|
||||||
const pngImage = await pdfDoc.embedPng(pngImageBytes);
|
|
||||||
const pngDims = pngImage.scale(1);
|
|
||||||
|
|
||||||
// Create a blank page matching the dimensions of the image
|
|
||||||
const page = pdfDoc.addPage([pngDims.width, pngDims.height]);
|
|
||||||
|
|
||||||
// Draw the PNG image
|
|
||||||
page.drawImage(pngImage, {
|
|
||||||
x: 0,
|
|
||||||
y: 0,
|
|
||||||
width: pngDims.width,
|
|
||||||
height: pngDims.height
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
// Serialize the PDFDocument to bytes (a Uint8Array)
|
|
||||||
const pdfBytes = await pdfDoc.save();
|
|
||||||
|
|
||||||
// Create a Blob
|
|
||||||
const blob = new Blob([pdfBytes.buffer], {type: "application/pdf"});
|
|
||||||
|
|
||||||
// Create download link
|
|
||||||
const downloadLink = document.createElement('a');
|
|
||||||
downloadLink.href = URL.createObjectURL(blob);
|
|
||||||
let newFileName = inputFileName ? inputFileName.replace('.pdf', '') : 'download';
|
|
||||||
newFileName += '_adjusted_color.pdf';
|
|
||||||
|
|
||||||
downloadLink.download = newFileName;
|
|
||||||
downloadLink.click();
|
|
||||||
|
|
||||||
// After download, reset the viewer and clear stored data
|
|
||||||
allPages = []; // Clear the pages
|
|
||||||
originalImageData = null; // Clear the image data
|
|
||||||
|
|
||||||
// Go back to page 1 and render it in the viewer
|
|
||||||
if (pdf !== null) {
|
|
||||||
renderPageAndAdjustImageProperties(1);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Event listeners
|
|
||||||
document.getElementById('fileInput-input').addEventListener('change', function(e) {
|
|
||||||
if (e.target.files.length > 0) {
|
|
||||||
inputFileName = e.target.files[0].name;
|
|
||||||
renderPDFAndSaveOriginalImageData(e.target.files[0]);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
document.getElementById('contrast-slider').addEventListener('input', function() {
|
|
||||||
document.getElementById('contrast-val').textContent = this.value;
|
|
||||||
adjustImageProperties();
|
|
||||||
});
|
|
||||||
|
|
||||||
document.getElementById('brightness-slider').addEventListener('input', function() {
|
|
||||||
document.getElementById('brightness-val').textContent = this.value;
|
|
||||||
adjustImageProperties();
|
|
||||||
});
|
|
||||||
|
|
||||||
document.getElementById('saturation-slider').addEventListener('input', function() {
|
|
||||||
document.getElementById('saturation-val').textContent = this.value;
|
|
||||||
adjustImageProperties();
|
|
||||||
});
|
|
||||||
|
|
||||||
document.getElementById('download-button').addEventListener('click', function() {
|
|
||||||
downloadPDF();
|
|
||||||
});
|
|
||||||
</script>
|
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
@ -163,6 +163,9 @@
|
|||||||
dragDropMessage:'[[#{multiTool.dragDropMessage}]]',
|
dragDropMessage:'[[#{multiTool.dragDropMessage}]]',
|
||||||
undo: '[[#{multiTool.undo}]]',
|
undo: '[[#{multiTool.undo}]]',
|
||||||
redo: '[[#{multiTool.redo}]]',
|
redo: '[[#{multiTool.redo}]]',
|
||||||
|
};
|
||||||
|
|
||||||
|
window.decrypt = {
|
||||||
passwordPrompt: '[[#{decrypt.passwordPrompt}]]',
|
passwordPrompt: '[[#{decrypt.passwordPrompt}]]',
|
||||||
cancelled: '[[#{decrypt.cancelled}]]',
|
cancelled: '[[#{decrypt.cancelled}]]',
|
||||||
noPassword: '[[#{decrypt.noPassword}]]',
|
noPassword: '[[#{decrypt.noPassword}]]',
|
||||||
@ -171,9 +174,7 @@
|
|||||||
unexpectedError: '[[#{decrypt.unexpectedError}]]',
|
unexpectedError: '[[#{decrypt.unexpectedError}]]',
|
||||||
serverError: '[[#{decrypt.serverError}]]',
|
serverError: '[[#{decrypt.serverError}]]',
|
||||||
success: '[[#{decrypt.success}]]',
|
success: '[[#{decrypt.success}]]',
|
||||||
};
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
const csvInput = document.getElementById("csv-input");
|
const csvInput = document.getElementById("csv-input");
|
||||||
csvInput.addEventListener("keydown", function (event) {
|
csvInput.addEventListener("keydown", function (event) {
|
||||||
|
@ -19,9 +19,10 @@
|
|||||||
}
|
}
|
||||||
</style>
|
</style>
|
||||||
</th:block>
|
</th:block>
|
||||||
|
|
||||||
<script th:src="@{'/js/thirdParty/signature_pad.umd.min.js'}"></script>
|
<script th:src="@{'/js/thirdParty/signature_pad.umd.min.js'}"></script>
|
||||||
<script th:src="@{'/js/thirdParty/interact.min.js'}"></script>
|
<script th:src="@{'/js/thirdParty/interact.min.js'}"></script>
|
||||||
|
<script type="module" th:src="@{'/js/pages/sign.js'}"></script>
|
||||||
|
|
||||||
</head>
|
</head>
|
||||||
|
|
||||||
<body>
|
<body>
|
||||||
@ -42,75 +43,6 @@
|
|||||||
th:replace="~{fragments/common :: fileSelector(name='pdf-upload', multipleInputsForSingleRequest=false, disableMultipleFiles=true, accept='application/pdf')}">
|
th:replace="~{fragments/common :: fileSelector(name='pdf-upload', multipleInputsForSingleRequest=false, disableMultipleFiles=true, accept='application/pdf')}">
|
||||||
</div>
|
</div>
|
||||||
<script type="module" th:src="@{'/pdfjs-legacy/pdf.mjs'}"></script>
|
<script type="module" th:src="@{'/pdfjs-legacy/pdf.mjs'}"></script>
|
||||||
<script>
|
|
||||||
let currentPreviewSrc = null;
|
|
||||||
|
|
||||||
function toggleSignatureView() {
|
|
||||||
const gridView = document.getElementById('gridView');
|
|
||||||
const listView = document.getElementById('listView');
|
|
||||||
const gridText = document.querySelector('.grid-view-text');
|
|
||||||
const listText = document.querySelector('.list-view-text');
|
|
||||||
|
|
||||||
if (gridView.style.display !== 'none') {
|
|
||||||
gridView.style.display = 'none';
|
|
||||||
listView.style.display = 'block';
|
|
||||||
gridText.style.display = 'none';
|
|
||||||
listText.style.display = 'inline';
|
|
||||||
} else {
|
|
||||||
gridView.style.display = 'block';
|
|
||||||
listView.style.display = 'none';
|
|
||||||
gridText.style.display = 'inline';
|
|
||||||
listText.style.display = 'none';
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
function previewSignature(element) {
|
|
||||||
const src = element.dataset.src;
|
|
||||||
currentPreviewSrc = src;
|
|
||||||
|
|
||||||
// Extract filename from the data source path
|
|
||||||
const filename = element.querySelector('.signature-list-name').textContent;
|
|
||||||
|
|
||||||
// Update preview modal
|
|
||||||
const previewImage = document.getElementById('previewImage');
|
|
||||||
const previewFileName = document.getElementById('previewFileName');
|
|
||||||
|
|
||||||
previewImage.src = src;
|
|
||||||
previewFileName.textContent = filename;
|
|
||||||
|
|
||||||
const modal = new bootstrap.Modal(document.getElementById('signaturePreview'));
|
|
||||||
modal.show();
|
|
||||||
}
|
|
||||||
|
|
||||||
function addSignatureFromPreview() {
|
|
||||||
if (currentPreviewSrc) {
|
|
||||||
DraggableUtils.createDraggableCanvasFromUrl(currentPreviewSrc);
|
|
||||||
bootstrap.Modal.getInstance(document.getElementById('signaturePreview')).hide();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
let originalFileName = '';
|
|
||||||
document.querySelector('input[name=pdf-upload]').addEventListener('change', async (event) => {
|
|
||||||
const file = event.target.files[0];
|
|
||||||
if (file) {
|
|
||||||
originalFileName = file.name.replace(/\.[^/.]+$/, "");
|
|
||||||
const pdfData = await file.arrayBuffer();
|
|
||||||
pdfjsLib.GlobalWorkerOptions.workerSrc = './pdfjs-legacy/pdf.worker.mjs';
|
|
||||||
const pdfDoc = await pdfjsLib.getDocument({ data: pdfData }).promise;
|
|
||||||
await DraggableUtils.renderPage(pdfDoc, 0);
|
|
||||||
|
|
||||||
document.querySelectorAll(".show-on-file-selected").forEach(el => {
|
|
||||||
el.style.cssText = '';
|
|
||||||
});
|
|
||||||
}
|
|
||||||
});
|
|
||||||
document.addEventListener("DOMContentLoaded", () => {
|
|
||||||
document.querySelectorAll(".show-on-file-selected").forEach(el => {
|
|
||||||
el.style.cssText = "display:none !important";
|
|
||||||
});
|
|
||||||
});
|
|
||||||
</script>
|
|
||||||
<div class="tab-group show-on-file-selected">
|
<div class="tab-group show-on-file-selected">
|
||||||
<div class="tab-container" th:title="#{sign.upload}">
|
<div class="tab-container" th:title="#{sign.upload}">
|
||||||
<div
|
<div
|
||||||
@ -237,123 +169,6 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<script>
|
|
||||||
const imageUpload = document.querySelector('input[name=image-upload]');
|
|
||||||
imageUpload.addEventListener('change', e => {
|
|
||||||
if (!e.target.files) return;
|
|
||||||
for (const imageFile of e.target.files) {
|
|
||||||
var reader = new FileReader();
|
|
||||||
reader.readAsDataURL(imageFile);
|
|
||||||
reader.onloadend = function (e) {
|
|
||||||
DraggableUtils.createDraggableCanvasFromUrl(e.target.result);
|
|
||||||
};
|
|
||||||
}
|
|
||||||
});
|
|
||||||
</script>
|
|
||||||
<script>
|
|
||||||
const signaturePadCanvas = document.getElementById('drawing-pad-canvas');
|
|
||||||
const signaturePad = new SignaturePad(signaturePadCanvas, {
|
|
||||||
minWidth: 1,
|
|
||||||
maxWidth: 2,
|
|
||||||
penColor: 'black',
|
|
||||||
});
|
|
||||||
|
|
||||||
function addDraggableFromPad() {
|
|
||||||
if (signaturePad.isEmpty()) return;
|
|
||||||
const startTime = Date.now();
|
|
||||||
const croppedDataUrl = getCroppedCanvasDataUrl(signaturePadCanvas);
|
|
||||||
console.log(Date.now() - startTime);
|
|
||||||
DraggableUtils.createDraggableCanvasFromUrl(croppedDataUrl);
|
|
||||||
}
|
|
||||||
|
|
||||||
function getCroppedCanvasDataUrl(canvas) {
|
|
||||||
let originalCtx = canvas.getContext('2d');
|
|
||||||
let originalWidth = canvas.width;
|
|
||||||
let originalHeight = canvas.height;
|
|
||||||
let imageData = originalCtx.getImageData(0, 0, originalWidth, originalHeight);
|
|
||||||
|
|
||||||
let minX = originalWidth + 1, maxX = -1, minY = originalHeight + 1, maxY = -1, x = 0, y = 0, currentPixelColorValueIndex;
|
|
||||||
|
|
||||||
for (y = 0; y < originalHeight; y++) {
|
|
||||||
for (x = 0; x < originalWidth; x++) {
|
|
||||||
currentPixelColorValueIndex = (y * originalWidth + x) * 4;
|
|
||||||
let currentPixelAlphaValue = imageData.data[currentPixelColorValueIndex + 3];
|
|
||||||
if (currentPixelAlphaValue > 0) {
|
|
||||||
if (minX > x) minX = x;
|
|
||||||
if (maxX < x) maxX = x;
|
|
||||||
if (minY > y) minY = y;
|
|
||||||
if (maxY < y) maxY = y;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
let croppedWidth = maxX - minX;
|
|
||||||
let croppedHeight = maxY - minY;
|
|
||||||
if (croppedWidth < 0 || croppedHeight < 0) return null;
|
|
||||||
let cuttedImageData = originalCtx.getImageData(minX, minY, croppedWidth, croppedHeight);
|
|
||||||
|
|
||||||
let croppedCanvas = document.createElement('canvas'),
|
|
||||||
croppedCtx = croppedCanvas.getContext('2d');
|
|
||||||
|
|
||||||
croppedCanvas.width = croppedWidth;
|
|
||||||
croppedCanvas.height = croppedHeight;
|
|
||||||
croppedCtx.putImageData(cuttedImageData, 0, 0);
|
|
||||||
|
|
||||||
return croppedCanvas.toDataURL();
|
|
||||||
}
|
|
||||||
|
|
||||||
function resizeCanvas() {
|
|
||||||
var ratio = Math.max(window.devicePixelRatio || 1, 1);
|
|
||||||
var additionalFactor = 10;
|
|
||||||
|
|
||||||
signaturePadCanvas.width = signaturePadCanvas.offsetWidth * ratio * additionalFactor;
|
|
||||||
signaturePadCanvas.height = signaturePadCanvas.offsetHeight * ratio * additionalFactor;
|
|
||||||
signaturePadCanvas.getContext("2d").scale(ratio * additionalFactor, ratio * additionalFactor);
|
|
||||||
|
|
||||||
signaturePad.clear();
|
|
||||||
}
|
|
||||||
|
|
||||||
new IntersectionObserver((entries, observer) => {
|
|
||||||
if (entries.some(entry => entry.intersectionRatio > 0)) {
|
|
||||||
resizeCanvas();
|
|
||||||
}
|
|
||||||
}).observe(signaturePadCanvas);
|
|
||||||
|
|
||||||
new ResizeObserver(resizeCanvas).observe(signaturePadCanvas);
|
|
||||||
</script>
|
|
||||||
<script>
|
|
||||||
function addDraggableFromText() {
|
|
||||||
const sigText = document.getElementById('sigText').value;
|
|
||||||
const font = document.querySelector('select[name=font]').value;
|
|
||||||
const fontSize = 100;
|
|
||||||
|
|
||||||
const canvas = document.createElement('canvas');
|
|
||||||
const ctx = canvas.getContext('2d');
|
|
||||||
ctx.font = `${fontSize}px ${font}`;
|
|
||||||
const textWidth = ctx.measureText(sigText).width;
|
|
||||||
const textHeight = fontSize;
|
|
||||||
|
|
||||||
let paragraphs = sigText.split(/\r?\n/);
|
|
||||||
|
|
||||||
canvas.width = textWidth;
|
|
||||||
canvas.height = paragraphs.length * textHeight * 1.35; // for tails
|
|
||||||
ctx.font = `${fontSize}px ${font}`;
|
|
||||||
|
|
||||||
ctx.textBaseline = 'top';
|
|
||||||
|
|
||||||
let y = 0;
|
|
||||||
|
|
||||||
paragraphs.forEach(paragraph => {
|
|
||||||
ctx.fillText(paragraph, 0, y);
|
|
||||||
y += fontSize;
|
|
||||||
});
|
|
||||||
|
|
||||||
const dataURL = canvas.toDataURL();
|
|
||||||
DraggableUtils.createDraggableCanvasFromUrl(dataURL);
|
|
||||||
}
|
|
||||||
</script>
|
|
||||||
|
|
||||||
<!-- draggables box -->
|
<!-- draggables box -->
|
||||||
<div id="box-drag-container" class="show-on-file-selected">
|
<div id="box-drag-container" class="show-on-file-selected">
|
||||||
<canvas id="pdf-canvas"></canvas>
|
<canvas id="pdf-canvas"></canvas>
|
||||||
@ -410,35 +225,11 @@
|
|||||||
</button>
|
</button>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<!-- download button -->
|
<!-- download button -->
|
||||||
<div class="margin-auto-parent">
|
<div class="margin-auto-parent">
|
||||||
<button id="download-pdf" class="btn btn-primary mb-2 show-on-file-selected margin-center"
|
<button id="download-pdf" class="btn btn-primary mb-2 show-on-file-selected margin-center"
|
||||||
th:text="#{downloadPdf}"></button>
|
th:text="#{downloadPdf}"></button>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<script>
|
|
||||||
async function goToFirstOrLastPage(page) {
|
|
||||||
if (page) {
|
|
||||||
const lastPage = DraggableUtils.pdfDoc.numPages
|
|
||||||
await DraggableUtils.goToPage(lastPage - 1)
|
|
||||||
} else {
|
|
||||||
await DraggableUtils.goToPage(0)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
</script>
|
|
||||||
|
|
||||||
<script>
|
|
||||||
document.getElementById("download-pdf").addEventListener('click', async () => {
|
|
||||||
const modifiedPdf = await DraggableUtils.getOverlayedPdfDocument();
|
|
||||||
const modifiedPdfBytes = await modifiedPdf.save();
|
|
||||||
const blob = new Blob([modifiedPdfBytes], { type: 'application/pdf' });
|
|
||||||
const link = document.createElement('a');
|
|
||||||
link.href = URL.createObjectURL(blob);
|
|
||||||
link.download = originalFileName + '_signed.pdf';
|
|
||||||
link.click();
|
|
||||||
});
|
|
||||||
</script>
|
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
@ -447,6 +238,7 @@
|
|||||||
</div>
|
</div>
|
||||||
<!-- Link the draggable.js file -->
|
<!-- Link the draggable.js file -->
|
||||||
<script th:src="@{'/js/draggable.js'}"></script>
|
<script th:src="@{'/js/draggable.js'}"></script>
|
||||||
|
|
||||||
</body>
|
</body>
|
||||||
|
|
||||||
</html>
|
</html>
|
Loading…
x
Reference in New Issue
Block a user