filestash/public/assets/pages/viewerpage/application_pdf.js
2025-06-28 01:17:57 +10:00

106 lines
4.3 KiB
JavaScript

import { createElement } from "../../lib/skeleton/index.js";
import rxjs, { effect, onLoad } from "../../lib/rx.js";
import { qs, safe } from "../../lib/dom.js";
import { createLoader } from "../../components/loader.js";
import { loadCSS, loadJS } from "../../helpers/loader.js";
import { join } from "../../lib/path.js";
import ctrlError from "../ctrl_error.js";
import { transition } from "./common.js";
import { renderMenubar, buttonDownload } from "./component_menubar.js";
import "../../components/icon.js";
const hasNativePDF = "application/pdf" in window.navigator.mimeTypes && !!window.chrome;
export default async function(render, opts) {
const ctrl = hasNativePDF ? ctrlPDFNative : ctrlPDFJs;
ctrl(render, opts);
}
function ctrlPDFNative(render, { getFilename, getDownloadUrl }) {
const $page = createElement(`
<div class="component_pdfviewer">
<component-menubar filename="${safe(getFilename())}"></component-menubar>
<div data-bind="pdf">
<embed
class="hidden"
src="${safe(getDownloadUrl())}#toolbar=0"
type="application/pdf"
/>
</div>
</div>
`);
render($page);
renderMenubar(qs($page, "component-menubar"), buttonDownload(getFilename(), getDownloadUrl()));
const removeLoader = createLoader(qs($page, `[data-bind="pdf"]`));
effect(onLoad(qs($page, "embed")).pipe(
removeLoader,
rxjs.tap(($embed) => $embed.classList.remove("hidden")),
rxjs.tap(($embed) => transition($embed)),
rxjs.catchError(ctrlError()),
));
}
async function ctrlPDFJs(render, { getFilename, getDownloadUrl }) {
const $page = createElement(`
<div class="component_pdfviewer">
<component-menubar filename="${safe(getFilename())}"></component-menubar>
<div data-bind="pdf"></div>
</div>
`);
render($page);
const $container = qs($page, `[data-bind="pdf"]`);
const createBr = () => $container.appendChild(createElement(`<div style="height:${document.body.clientWidth > 600 ? 20 : 5}px">&nbsp;</div>`));
const removeLoader = createLoader($container);
const base = qs(document.head, "base").getAttribute("href");
effect(rxjs.from(window.pdfjsLib.getDocument(base + getDownloadUrl()).promise).pipe(
removeLoader,
rxjs.mergeMap(async(pdf) => {
createBr();
for (let i=0; i<pdf.numPages; i++) {
const page = await pdf.getPage(i + 1);
const marginLeftRight = (document.body.clientWidth > 600 ? 50 : 15);
const ratio = window.devicePixelRatio || 1;
const viewport = page.getViewport({
scale: Math.min(
Math.max(
document.body.clientWidth - marginLeftRight,
0,
),
800,
) / page.getViewport({ scale: 1 / ratio }).width,
});
const $canvas = document.createElement("canvas");
$canvas.height = viewport.height;
$canvas.width = viewport.width;
$canvas.style.width = Math.floor(viewport.width / ratio) + "px";
$canvas.style.height = Math.floor(viewport.height / ratio) + "px";
$container.appendChild($canvas);
if (window.env === "test") $canvas.getContext = () => null;
await page.render({
canvasContext: $canvas.getContext("2d"),
viewport,
});
await new Promise((done) => window.requestAnimationFrame(done));
}
createBr();
}),
rxjs.catchError(ctrlError()),
));
}
export function init() {
const deps = [
loadCSS(import.meta.url, "./application_pdf.css"),
];
if (!hasNativePDF) {
deps.push(loadJS(import.meta.url, "../../lib/vendor/pdfjs/pdf.js", { type: "module" }));
deps.push(loadJS(import.meta.url, "../../lib/vendor/pdfjs/pdf.worker.js", { type: "module" }).then(() => {
window.pdfjsLib.GlobalWorkerOptions.workerSrc = join(import.meta.url, "../../lib/vendor/pdfjs/pdf.worker.js");
}));
}
return Promise.all(deps);
}