import { showLoader, hideLoader, showAlert } from '../ui.js'; import { downloadFile } from '../utils/helpers.js'; import { state } from '../state.js'; import Sortable from 'sortablejs'; import { icons, createIcons } from 'lucide'; import { PDFDocument as PDFLibDocument } from 'pdf-lib'; const duplicateOrganizeState = { sortableInstances: {}, }; function initializePageGridSortable() { const grid = document.getElementById('page-grid'); if (!grid) return; // @ts-expect-error TS(2339) FIXME: Property 'pageGrid' does not exist on type '{}'. if (duplicateOrganizeState.sortableInstances.pageGrid) { // @ts-expect-error TS(2339) FIXME: Property 'pageGrid' does not exist on type '{}'. duplicateOrganizeState.sortableInstances.pageGrid.destroy(); } // @ts-expect-error TS(2339) FIXME: Property 'pageGrid' does not exist on type '{}'. duplicateOrganizeState.sortableInstances.pageGrid = Sortable.create(grid, { animation: 150, ghostClass: 'sortable-ghost', chosenClass: 'sortable-chosen', dragClass: 'sortable-drag', filter: '.duplicate-btn, .delete-btn', preventOnFilter: true, onStart: function (evt: any) { evt.item.style.opacity = '0.5'; }, onEnd: function (evt: any) { evt.item.style.opacity = '1'; }, }); } /** * Attaches event listeners for duplicate and delete to a page thumbnail element. * @param {HTMLElement} element The thumbnail element to attach listeners to. */ function attachEventListeners(element: any) { // Re-number all visible page labels const renumberPages = () => { const grid = document.getElementById('page-grid'); const pages = grid.querySelectorAll('.page-number'); pages.forEach((label, index) => { // @ts-expect-error TS(2322) FIXME: Type 'number' is not assignable to type 'string'. label.textContent = index + 1; }); }; // Duplicate button listener element .querySelector('.duplicate-btn') .addEventListener('click', (e: any) => { e.stopPropagation(); const clone = element.cloneNode(true); element.after(clone); attachEventListeners(clone); renumberPages(); initializePageGridSortable(); }); element.querySelector('.delete-btn').addEventListener('click', (e: any) => { e.stopPropagation(); if (document.getElementById('page-grid').children.length > 1) { element.remove(); renumberPages(); initializePageGridSortable(); } else { showAlert( 'Cannot Delete', 'You cannot delete the last page of the document.' ); } }); } export async function renderDuplicateOrganizeThumbnails() { const grid = document.getElementById('page-grid'); if (!grid) return; showLoader('Rendering page previews...'); const pdfData = await state.pdfDoc.save(); // @ts-expect-error TS(2304) FIXME: Cannot find name 'pdfjsLib'. const pdfjsDoc = await pdfjsLib.getDocument({ data: pdfData }).promise; grid.textContent = ''; for (let i = 1; i <= pdfjsDoc.numPages; i++) { const page = await pdfjsDoc.getPage(i); const viewport = page.getViewport({ scale: 0.5 }); const canvas = document.createElement('canvas'); canvas.height = viewport.height; canvas.width = viewport.width; await page.render({ canvasContext: canvas.getContext('2d'), viewport }) .promise; const wrapper = document.createElement('div'); wrapper.className = 'page-thumbnail relative cursor-move flex flex-col items-center gap-2'; // @ts-expect-error TS(2322) FIXME: Type 'number' is not assignable to type 'string'. wrapper.dataset.originalPageIndex = i - 1; const imgContainer = document.createElement('div'); imgContainer.className = 'w-full h-36 bg-gray-900 rounded-lg flex items-center justify-center overflow-hidden border-2 border-gray-600'; const img = document.createElement('img'); img.src = canvas.toDataURL(); img.className = 'max-w-full max-h-full object-contain'; imgContainer.appendChild(img); const pageNumberSpan = document.createElement('span'); pageNumberSpan.className = 'page-number absolute top-1 left-1 bg-gray-900 bg-opacity-75 text-white text-xs rounded-full px-2 py-1'; pageNumberSpan.textContent = i.toString(); const controlsDiv = document.createElement('div'); controlsDiv.className = 'flex items-center justify-center gap-4'; const duplicateBtn = document.createElement('button'); duplicateBtn.className = 'duplicate-btn bg-green-600 hover:bg-green-700 text-white rounded-full w-8 h-8 flex items-center justify-center'; duplicateBtn.title = 'Duplicate Page'; const duplicateIcon = document.createElement('i'); duplicateIcon.setAttribute('data-lucide', 'copy-plus'); duplicateIcon.className = 'w-5 h-5'; duplicateBtn.appendChild(duplicateIcon); const deleteBtn = document.createElement('button'); deleteBtn.className = 'delete-btn bg-red-600 hover:bg-red-700 text-white rounded-full w-8 h-8 flex items-center justify-center'; deleteBtn.title = 'Delete Page'; const deleteIcon = document.createElement('i'); deleteIcon.setAttribute('data-lucide', 'x-circle'); deleteIcon.className = 'w-5 h-5'; deleteBtn.appendChild(deleteIcon); controlsDiv.append(duplicateBtn, deleteBtn); wrapper.append(imgContainer, pageNumberSpan, controlsDiv); grid.appendChild(wrapper); attachEventListeners(wrapper); } initializePageGridSortable(); createIcons({ icons }); hideLoader(); } export async function processAndSave() { showLoader('Building new PDF...'); try { const grid = document.getElementById('page-grid'); const finalPageElements = grid.querySelectorAll('.page-thumbnail'); // @ts-expect-error TS(2339) FIXME: Property 'dataset' does not exist on type 'Element... Remove this comment to see the full error message const finalIndices = Array.from(finalPageElements).map((el) => parseInt(el.dataset.originalPageIndex) ); const newPdfDoc = await PDFLibDocument.create(); const copiedPages = await newPdfDoc.copyPages(state.pdfDoc, finalIndices); copiedPages.forEach((page: any) => newPdfDoc.addPage(page)); const newPdfBytes = await newPdfDoc.save(); downloadFile( new Blob([new Uint8Array(newPdfBytes)], { type: 'application/pdf' }), 'organized.pdf' ); } catch (e) { console.error(e); showAlert('Error', 'Failed to save the new PDF.'); } finally { hideLoader(); } }