mirror of
https://github.com/mozilla/pdf.js.git
synced 2025-04-19 14:48:08 +02:00
Merge pull request #18034 from Snuffleupagus/FileSpec-filename-stripPath
[api-minor] Improve the `FileSpec` implementation
This commit is contained in:
commit
1b811ac113
8 changed files with 49 additions and 58 deletions
|
@ -1619,8 +1619,8 @@ class Catalog {
|
|||
/* xref = */ null,
|
||||
/* skipContent = */ true
|
||||
);
|
||||
const { filename } = fs.serializable;
|
||||
url = filename;
|
||||
const { rawFilename } = fs.serializable;
|
||||
url = rawFilename;
|
||||
} else if (typeof urlDict === "string") {
|
||||
url = urlDict;
|
||||
}
|
||||
|
|
|
@ -18,6 +18,9 @@ import { BaseStream } from "./base_stream.js";
|
|||
import { Dict } from "./primitives.js";
|
||||
|
||||
function pickPlatformItem(dict) {
|
||||
if (!(dict instanceof Dict)) {
|
||||
return null;
|
||||
}
|
||||
// Look for the filename in this order:
|
||||
// UF, F, Unix, Mac, DOS
|
||||
if (dict.has("UF")) {
|
||||
|
@ -34,6 +37,10 @@ function pickPlatformItem(dict) {
|
|||
return null;
|
||||
}
|
||||
|
||||
function stripPath(str) {
|
||||
return str.substring(str.lastIndexOf("/") + 1);
|
||||
}
|
||||
|
||||
/**
|
||||
* "A PDF file can refer to the contents of another file by using a File
|
||||
* Specification (PDF 1.1)", see the spec (7.11) for more details.
|
||||
|
@ -66,26 +73,27 @@ class FileSpec {
|
|||
}
|
||||
|
||||
get filename() {
|
||||
if (!this._filename && this.root) {
|
||||
const filename = pickPlatformItem(this.root) || "unnamed";
|
||||
this._filename = stringToPDFString(filename)
|
||||
let filename = "";
|
||||
|
||||
const item = pickPlatformItem(this.root);
|
||||
if (item && typeof item === "string") {
|
||||
filename = stringToPDFString(item)
|
||||
.replaceAll("\\\\", "\\")
|
||||
.replaceAll("\\/", "/")
|
||||
.replaceAll("\\", "/");
|
||||
}
|
||||
return this._filename;
|
||||
return shadow(this, "filename", filename || "unnamed");
|
||||
}
|
||||
|
||||
get content() {
|
||||
if (!this.#contentAvailable) {
|
||||
return null;
|
||||
}
|
||||
if (!this.contentRef && this.root) {
|
||||
this.contentRef = pickPlatformItem(this.root.get("EF"));
|
||||
}
|
||||
this._contentRef ||= pickPlatformItem(this.root?.get("EF"));
|
||||
|
||||
let content = null;
|
||||
if (this.contentRef) {
|
||||
const fileObj = this.xref.fetchIfRef(this.contentRef);
|
||||
if (this._contentRef) {
|
||||
const fileObj = this.xref.fetchIfRef(this._contentRef);
|
||||
if (fileObj instanceof BaseStream) {
|
||||
content = fileObj.getBytes();
|
||||
} else {
|
||||
|
@ -94,7 +102,7 @@ class FileSpec {
|
|||
);
|
||||
}
|
||||
} else {
|
||||
warn("Embedded file specification does not have a content");
|
||||
warn("Embedded file specification does not have any content");
|
||||
}
|
||||
return content;
|
||||
}
|
||||
|
@ -111,7 +119,8 @@ class FileSpec {
|
|||
|
||||
get serializable() {
|
||||
return {
|
||||
filename: this.filename,
|
||||
rawFilename: this.filename,
|
||||
filename: stripPath(this.filename),
|
||||
content: this.content,
|
||||
description: this.description,
|
||||
};
|
||||
|
|
|
@ -37,7 +37,6 @@ import {
|
|||
} from "../shared/util.js";
|
||||
import {
|
||||
DOMSVGFactory,
|
||||
getFilenameFromUrl,
|
||||
PDFDateString,
|
||||
setLayerDimensions,
|
||||
} from "./display_utils.js";
|
||||
|
@ -2859,15 +2858,13 @@ class FileAttachmentAnnotationElement extends AnnotationElement {
|
|||
constructor(parameters) {
|
||||
super(parameters, { isRenderable: true });
|
||||
|
||||
const { filename, content, description } = this.data.file;
|
||||
this.filename = getFilenameFromUrl(filename, /* onlyStripPath = */ true);
|
||||
this.content = content;
|
||||
const { file } = this.data;
|
||||
this.filename = file.filename;
|
||||
this.content = file.content;
|
||||
|
||||
this.linkService.eventBus?.dispatch("fileattachmentannotation", {
|
||||
source: this,
|
||||
filename,
|
||||
content,
|
||||
description,
|
||||
...file,
|
||||
});
|
||||
}
|
||||
|
||||
|
|
|
@ -803,13 +803,10 @@ function isPdfFile(filename) {
|
|||
/**
|
||||
* Gets the filename from a given URL.
|
||||
* @param {string} url
|
||||
* @param {boolean} [onlyStripPath]
|
||||
* @returns {string}
|
||||
*/
|
||||
function getFilenameFromUrl(url, onlyStripPath = false) {
|
||||
if (!onlyStripPath) {
|
||||
[url] = url.split(/[#?]/, 1);
|
||||
}
|
||||
function getFilenameFromUrl(url) {
|
||||
[url] = url.split(/[#?]/, 1);
|
||||
return url.substring(url.lastIndexOf("/") + 1);
|
||||
}
|
||||
|
||||
|
|
|
@ -4033,9 +4033,12 @@ describe("annotation", function () {
|
|||
idFactoryMock
|
||||
);
|
||||
expect(data.annotationType).toEqual(AnnotationType.FILEATTACHMENT);
|
||||
expect(data.file.filename).toEqual("Test.txt");
|
||||
expect(data.file.content).toEqual(stringToBytes("Test attachment"));
|
||||
expect(data.file.description).toEqual("abc");
|
||||
expect(data.file).toEqual({
|
||||
rawFilename: "Test.txt",
|
||||
filename: "Test.txt",
|
||||
content: stringToBytes("Test attachment"),
|
||||
description: "abc",
|
||||
});
|
||||
});
|
||||
});
|
||||
|
||||
|
|
|
@ -1475,12 +1475,12 @@ describe("api", function () {
|
|||
const pdfDoc = await loadingTask.promise;
|
||||
const attachments = await pdfDoc.getAttachments();
|
||||
|
||||
const { filename, content, description } = attachments["foo.txt"];
|
||||
expect(filename).toEqual("foo.txt");
|
||||
expect(content).toEqual(
|
||||
new Uint8Array([98, 97, 114, 32, 98, 97, 122, 32, 10])
|
||||
);
|
||||
expect(description).toEqual("");
|
||||
expect(attachments["foo.txt"]).toEqual({
|
||||
rawFilename: "foo.txt",
|
||||
filename: "foo.txt",
|
||||
content: new Uint8Array([98, 97, 114, 32, 98, 97, 122, 32, 10]),
|
||||
description: "",
|
||||
});
|
||||
|
||||
await loadingTask.destroy();
|
||||
});
|
||||
|
@ -1490,7 +1490,9 @@ describe("api", function () {
|
|||
const pdfDoc = await loadingTask.promise;
|
||||
const attachments = await pdfDoc.getAttachments();
|
||||
|
||||
const { filename, content, description } = attachments["empty.pdf"];
|
||||
const { rawFilename, filename, content, description } =
|
||||
attachments["empty.pdf"];
|
||||
expect(rawFilename).toEqual("Empty page.pdf");
|
||||
expect(filename).toEqual("Empty page.pdf");
|
||||
expect(content instanceof Uint8Array).toEqual(true);
|
||||
expect(content.length).toEqual(2357);
|
||||
|
|
|
@ -189,13 +189,6 @@ describe("display_utils", function () {
|
|||
const url = "https://server.org/filename.pdf?foo=bar";
|
||||
expect(getFilenameFromUrl(url)).toEqual("filename.pdf");
|
||||
});
|
||||
|
||||
it("should get the filename from a relative URL, keeping the anchor", function () {
|
||||
const url = "../../part1#part2.pdf";
|
||||
expect(getFilenameFromUrl(url, /* onlyStripPath = */ true)).toEqual(
|
||||
"part1#part2.pdf"
|
||||
);
|
||||
});
|
||||
});
|
||||
|
||||
describe("getPdfFilenameFromUrl", function () {
|
||||
|
|
|
@ -18,7 +18,6 @@
|
|||
/** @typedef {import("./download_manager.js").DownloadManager} DownloadManager */
|
||||
|
||||
import { BaseTreeViewer } from "./base_tree_viewer.js";
|
||||
import { getFilenameFromUrl } from "pdfjs-lib";
|
||||
import { waitOnEventOrTimeout } from "./event_utils.js";
|
||||
|
||||
/**
|
||||
|
@ -122,19 +121,13 @@ class PDFAttachmentViewer extends BaseTreeViewer {
|
|||
let attachmentsCount = 0;
|
||||
for (const name in attachments) {
|
||||
const item = attachments[name];
|
||||
const content = item.content,
|
||||
description = item.description,
|
||||
filename = getFilenameFromUrl(
|
||||
item.filename,
|
||||
/* onlyStripPath = */ true
|
||||
);
|
||||
|
||||
const div = document.createElement("div");
|
||||
div.className = "treeItem";
|
||||
|
||||
const element = document.createElement("a");
|
||||
this._bindLink(element, { content, description, filename });
|
||||
element.textContent = this._normalizeTextContent(filename);
|
||||
this._bindLink(element, item);
|
||||
element.textContent = this._normalizeTextContent(item.filename);
|
||||
|
||||
div.append(element);
|
||||
|
||||
|
@ -148,7 +141,7 @@ class PDFAttachmentViewer extends BaseTreeViewer {
|
|||
/**
|
||||
* Used to append FileAttachment annotations to the sidebar.
|
||||
*/
|
||||
#appendAttachment({ filename, content, description }) {
|
||||
#appendAttachment(item) {
|
||||
const renderedPromise = this._renderedCapability.promise;
|
||||
|
||||
renderedPromise.then(() => {
|
||||
|
@ -158,15 +151,12 @@ class PDFAttachmentViewer extends BaseTreeViewer {
|
|||
const attachments = this._attachments || Object.create(null);
|
||||
|
||||
for (const name in attachments) {
|
||||
if (filename === name) {
|
||||
if (item.filename === name) {
|
||||
return; // Ignore the new attachment if it already exists.
|
||||
}
|
||||
}
|
||||
attachments[filename] = {
|
||||
filename,
|
||||
content,
|
||||
description,
|
||||
};
|
||||
attachments[item.filename] = item;
|
||||
|
||||
this.render({
|
||||
attachments,
|
||||
keepRenderedCapability: true,
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue