2020-07-22 20:55:52 +09:00
|
|
|
/* Copyright 2020 Mozilla Foundation
|
|
|
|
*
|
|
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
* you may not use this file except in compliance with the License.
|
|
|
|
* You may obtain a copy of the License at
|
|
|
|
*
|
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
*
|
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
* See the License for the specific language governing permissions and
|
|
|
|
* limitations under the License.
|
|
|
|
*/
|
|
|
|
|
2022-06-13 20:35:58 +09:00
|
|
|
import { objectFromMap, unreachable } from "../shared/util.js";
|
2022-06-01 17:38:08 +09:00
|
|
|
import { AnnotationEditor } from "./editor/editor.js";
|
2022-05-04 22:03:46 +09:00
|
|
|
import { MurmurHash3_64 } from "../shared/murmurhash3.js";
|
2020-10-28 22:14:27 +09:00
|
|
|
|
2020-07-23 05:38:04 +09:00
|
|
|
/**
|
|
|
|
* Key/value storage for annotation data in forms.
|
|
|
|
*/
|
2020-07-22 20:55:52 +09:00
|
|
|
class AnnotationStorage {
|
2022-11-11 19:59:11 +09:00
|
|
|
#modified = false;
|
|
|
|
|
|
|
|
#storage = new Map();
|
2020-08-19 05:50:23 +09:00
|
|
|
|
2022-11-11 19:59:11 +09:00
|
|
|
constructor() {
|
2020-08-19 05:50:23 +09:00
|
|
|
// Callbacks to signal when the modification state is set or reset.
|
|
|
|
// This is used by the viewer to only bind on `beforeunload` if forms
|
|
|
|
// are actually edited to prevent doing so unconditionally since that
|
2021-04-06 20:09:55 +09:00
|
|
|
// can have undesirable effects.
|
2020-08-19 05:50:23 +09:00
|
|
|
this.onSetModified = null;
|
|
|
|
this.onResetModified = null;
|
2022-08-03 01:29:01 +09:00
|
|
|
this.onAnnotationEditor = null;
|
2020-07-22 20:55:52 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2021-03-12 00:25:53 +09:00
|
|
|
* Get the value for a given key if it exists, or return the default value.
|
2020-07-22 20:55:52 +09:00
|
|
|
* @param {string} key
|
|
|
|
* @param {Object} defaultValue
|
|
|
|
* @returns {Object}
|
|
|
|
*/
|
2020-12-20 02:12:44 +09:00
|
|
|
getValue(key, defaultValue) {
|
2022-11-11 19:59:11 +09:00
|
|
|
const value = this.#storage.get(key);
|
2021-09-18 21:55:38 +09:00
|
|
|
if (value === undefined) {
|
|
|
|
return defaultValue;
|
|
|
|
}
|
|
|
|
|
|
|
|
return Object.assign(defaultValue, value);
|
2020-12-20 02:12:44 +09:00
|
|
|
}
|
|
|
|
|
2022-05-03 02:28:00 +09:00
|
|
|
/**
|
|
|
|
* Get the value for a given key.
|
|
|
|
* @param {string} key
|
|
|
|
* @returns {Object}
|
|
|
|
*/
|
|
|
|
getRawValue(key) {
|
2022-11-11 19:59:11 +09:00
|
|
|
return this.#storage.get(key);
|
2022-05-03 02:28:00 +09:00
|
|
|
}
|
|
|
|
|
2022-06-01 17:38:08 +09:00
|
|
|
/**
|
|
|
|
* Remove a value from the storage.
|
|
|
|
* @param {string} key
|
|
|
|
*/
|
2022-08-26 07:03:19 +09:00
|
|
|
remove(key) {
|
2022-11-11 19:59:11 +09:00
|
|
|
this.#storage.delete(key);
|
2022-06-05 04:32:27 +09:00
|
|
|
|
2022-11-11 19:59:11 +09:00
|
|
|
if (this.#storage.size === 0) {
|
2022-06-05 04:32:27 +09:00
|
|
|
this.resetModified();
|
|
|
|
}
|
2022-08-26 07:03:19 +09:00
|
|
|
|
|
|
|
if (typeof this.onAnnotationEditor === "function") {
|
2022-11-11 19:59:11 +09:00
|
|
|
for (const value of this.#storage.values()) {
|
2022-08-26 07:03:19 +09:00
|
|
|
if (value instanceof AnnotationEditor) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
this.onAnnotationEditor(null);
|
|
|
|
}
|
2022-06-01 17:38:08 +09:00
|
|
|
}
|
|
|
|
|
2020-07-22 20:55:52 +09:00
|
|
|
/**
|
|
|
|
* Set the value for a given key
|
|
|
|
* @param {string} key
|
|
|
|
* @param {Object} value
|
|
|
|
*/
|
|
|
|
setValue(key, value) {
|
2022-11-11 19:59:11 +09:00
|
|
|
const obj = this.#storage.get(key);
|
2020-11-04 00:04:08 +09:00
|
|
|
let modified = false;
|
|
|
|
if (obj !== undefined) {
|
|
|
|
for (const [entry, val] of Object.entries(value)) {
|
|
|
|
if (obj[entry] !== val) {
|
|
|
|
modified = true;
|
|
|
|
obj[entry] = val;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
modified = true;
|
2022-11-11 19:59:11 +09:00
|
|
|
this.#storage.set(key, value);
|
2020-11-04 00:04:08 +09:00
|
|
|
}
|
|
|
|
if (modified) {
|
2022-06-13 20:35:58 +09:00
|
|
|
this.#setModified();
|
2020-08-19 05:50:23 +09:00
|
|
|
}
|
2022-08-03 01:29:01 +09:00
|
|
|
|
|
|
|
if (
|
|
|
|
value instanceof AnnotationEditor &&
|
|
|
|
typeof this.onAnnotationEditor === "function"
|
|
|
|
) {
|
|
|
|
this.onAnnotationEditor(value.constructor._type);
|
|
|
|
}
|
2020-07-22 20:55:52 +09:00
|
|
|
}
|
|
|
|
|
2022-07-30 01:00:52 +09:00
|
|
|
/**
|
|
|
|
* Check if the storage contains the given key.
|
|
|
|
* @param {string} key
|
|
|
|
* @returns {boolean}
|
|
|
|
*/
|
|
|
|
has(key) {
|
2022-11-11 19:59:11 +09:00
|
|
|
return this.#storage.has(key);
|
2022-07-30 01:00:52 +09:00
|
|
|
}
|
|
|
|
|
2023-02-13 17:56:08 +09:00
|
|
|
/**
|
|
|
|
* @returns {Object | null}
|
|
|
|
*/
|
2020-07-22 20:55:52 +09:00
|
|
|
getAll() {
|
2022-11-11 19:59:11 +09:00
|
|
|
return this.#storage.size > 0 ? objectFromMap(this.#storage) : null;
|
2020-08-10 23:59:16 +09:00
|
|
|
}
|
|
|
|
|
2023-02-13 17:56:08 +09:00
|
|
|
/**
|
|
|
|
* @param {Object} obj
|
|
|
|
*/
|
|
|
|
setAll(obj) {
|
|
|
|
for (const [key, val] of Object.entries(obj)) {
|
|
|
|
this.setValue(key, val);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-08-10 23:59:16 +09:00
|
|
|
get size() {
|
2022-11-11 19:59:11 +09:00
|
|
|
return this.#storage.size;
|
2020-07-22 20:55:52 +09:00
|
|
|
}
|
2020-08-19 05:50:23 +09:00
|
|
|
|
2022-06-13 20:35:58 +09:00
|
|
|
#setModified() {
|
2022-11-11 19:59:11 +09:00
|
|
|
if (!this.#modified) {
|
|
|
|
this.#modified = true;
|
2020-08-19 05:50:23 +09:00
|
|
|
if (typeof this.onSetModified === "function") {
|
|
|
|
this.onSetModified();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
resetModified() {
|
2022-11-11 19:59:11 +09:00
|
|
|
if (this.#modified) {
|
|
|
|
this.#modified = false;
|
2020-08-19 05:50:23 +09:00
|
|
|
if (typeof this.onResetModified === "function") {
|
|
|
|
this.onResetModified();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2021-02-18 21:51:08 +09:00
|
|
|
|
2022-06-13 20:35:58 +09:00
|
|
|
/**
|
|
|
|
* @returns {PrintAnnotationStorage}
|
|
|
|
*/
|
|
|
|
get print() {
|
|
|
|
return new PrintAnnotationStorage(this);
|
|
|
|
}
|
|
|
|
|
2021-02-18 21:51:08 +09:00
|
|
|
/**
|
|
|
|
* PLEASE NOTE: Only intended for usage within the API itself.
|
|
|
|
* @ignore
|
|
|
|
*/
|
|
|
|
get serializable() {
|
2022-11-11 19:59:11 +09:00
|
|
|
if (this.#storage.size === 0) {
|
2022-06-01 17:38:08 +09:00
|
|
|
return null;
|
|
|
|
}
|
|
|
|
const clone = new Map();
|
2022-06-13 20:35:58 +09:00
|
|
|
|
2022-11-11 19:59:11 +09:00
|
|
|
for (const [key, val] of this.#storage) {
|
2022-07-13 00:32:14 +09:00
|
|
|
const serialized =
|
|
|
|
val instanceof AnnotationEditor ? val.serialize() : val;
|
|
|
|
if (serialized) {
|
|
|
|
clone.set(key, serialized);
|
|
|
|
}
|
2022-06-01 17:38:08 +09:00
|
|
|
}
|
|
|
|
return clone;
|
2021-02-18 21:51:08 +09:00
|
|
|
}
|
[Regression] Re-factor the *internal* `includeAnnotationStorage` handling, since it's currently subtly wrong
*This patch is very similar to the recently fixed `renderInteractiveForms`-options, see PR 13867.*
As far as I can tell, this *subtle* bug has existed ever since `AnnotationStorage`-support was first added in PR 12106 (a little over a year ago).
The value of the `includeAnnotationStorage`-option, as passed to the `PDFPageProxy.render` method, will (potentially) affect the size/content of the operatorList that's returned from the worker (for documents with forms).
Given that operatorLists will generally, unless they contain huge images, be cached in the API, repeated `PDFPageProxy.render` calls where the form-data has been changed by the user in between, can thus *wrongly* return a cached operatorList.
In the viewer we're only using the `includeAnnotationStorage`-option when printing, which is probably why this has gone unnoticed for so long. Note that we, for performance reasons, don't cache printing-operatorLists in the API.
However, there's nothing stopping an API-user from using the `includeAnnotationStorage`-option during "normal" rendering, which could thus result in *subtle* (and difficult to understand) rendering bugs.
In order to handle this, we need to know if the `AnnotationStorage`-instance has been updated since the last `PDFPageProxy.render` call. The most "correct" solution would obviously be to create a hash of the `AnnotationStorage` contents, however that would require adding a bunch of code, complexity, and runtime overhead.
Given that operatorList caching in the API doesn't have to be perfect[1], but only have to avoid *false* cache-hits, we can simplify things significantly be only keeping track of the last time that the `AnnotationStorage`-data was modified.
*Please note:* While working on this patch, I also noticed that the `renderInteractiveForms`- and `includeAnnotationStorage`-options in the `PDFPageProxy.render` method are mutually exclusive.[2]
Given that the various Annotation-related options in `PDFPageProxy.render` have been added at different times, this has unfortunately led to the current "messy" situation.[3]
---
[1] Note how we're already not caching operatorLists for pages with *huge* images, in order to save memory, hence there's no guarantee that operatorLists will always be cached.
[2] Setting both to `true` will result in undefined behaviour, since trying to insert `AnnotationStorage`-values into fields that are being excluded from the operatorList-building will obviously not work, which isn't at all clear from the documentation.
[3] My intention is to try and fix this in a follow-up PR, and I've got a WIP patch locally, however it will result in a number of API-observable changes.
2021-08-16 02:57:42 +09:00
|
|
|
|
|
|
|
/**
|
|
|
|
* PLEASE NOTE: Only intended for usage within the API itself.
|
|
|
|
* @ignore
|
|
|
|
*/
|
2022-06-13 20:35:58 +09:00
|
|
|
static getHash(map) {
|
|
|
|
if (!map) {
|
|
|
|
return "";
|
|
|
|
}
|
2022-05-04 22:03:46 +09:00
|
|
|
const hash = new MurmurHash3_64();
|
|
|
|
|
2022-06-13 20:35:58 +09:00
|
|
|
for (const [key, val] of map) {
|
2022-06-05 21:02:16 +09:00
|
|
|
hash.update(`${key}:${JSON.stringify(val)}`);
|
2022-05-04 22:03:46 +09:00
|
|
|
}
|
|
|
|
return hash.hexdigest();
|
[Regression] Re-factor the *internal* `includeAnnotationStorage` handling, since it's currently subtly wrong
*This patch is very similar to the recently fixed `renderInteractiveForms`-options, see PR 13867.*
As far as I can tell, this *subtle* bug has existed ever since `AnnotationStorage`-support was first added in PR 12106 (a little over a year ago).
The value of the `includeAnnotationStorage`-option, as passed to the `PDFPageProxy.render` method, will (potentially) affect the size/content of the operatorList that's returned from the worker (for documents with forms).
Given that operatorLists will generally, unless they contain huge images, be cached in the API, repeated `PDFPageProxy.render` calls where the form-data has been changed by the user in between, can thus *wrongly* return a cached operatorList.
In the viewer we're only using the `includeAnnotationStorage`-option when printing, which is probably why this has gone unnoticed for so long. Note that we, for performance reasons, don't cache printing-operatorLists in the API.
However, there's nothing stopping an API-user from using the `includeAnnotationStorage`-option during "normal" rendering, which could thus result in *subtle* (and difficult to understand) rendering bugs.
In order to handle this, we need to know if the `AnnotationStorage`-instance has been updated since the last `PDFPageProxy.render` call. The most "correct" solution would obviously be to create a hash of the `AnnotationStorage` contents, however that would require adding a bunch of code, complexity, and runtime overhead.
Given that operatorList caching in the API doesn't have to be perfect[1], but only have to avoid *false* cache-hits, we can simplify things significantly be only keeping track of the last time that the `AnnotationStorage`-data was modified.
*Please note:* While working on this patch, I also noticed that the `renderInteractiveForms`- and `includeAnnotationStorage`-options in the `PDFPageProxy.render` method are mutually exclusive.[2]
Given that the various Annotation-related options in `PDFPageProxy.render` have been added at different times, this has unfortunately led to the current "messy" situation.[3]
---
[1] Note how we're already not caching operatorLists for pages with *huge* images, in order to save memory, hence there's no guarantee that operatorLists will always be cached.
[2] Setting both to `true` will result in undefined behaviour, since trying to insert `AnnotationStorage`-values into fields that are being excluded from the operatorList-building will obviously not work, which isn't at all clear from the documentation.
[3] My intention is to try and fix this in a follow-up PR, and I've got a WIP patch locally, however it will result in a number of API-observable changes.
2021-08-16 02:57:42 +09:00
|
|
|
}
|
2020-07-22 20:55:52 +09:00
|
|
|
}
|
|
|
|
|
2022-06-13 20:35:58 +09:00
|
|
|
/**
|
|
|
|
* A special `AnnotationStorage` for use during printing, where the serializable
|
|
|
|
* data is *frozen* upon initialization, to prevent scripting from modifying its
|
|
|
|
* contents. (Necessary since printing is triggered synchronously in browsers.)
|
|
|
|
*/
|
|
|
|
class PrintAnnotationStorage extends AnnotationStorage {
|
|
|
|
#serializable = null;
|
|
|
|
|
|
|
|
constructor(parent) {
|
|
|
|
super();
|
|
|
|
// Create a *copy* of the data, since Objects are passed by reference in JS.
|
|
|
|
this.#serializable = structuredClone(parent.serializable);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @returns {PrintAnnotationStorage}
|
|
|
|
*/
|
|
|
|
// eslint-disable-next-line getter-return
|
|
|
|
get print() {
|
|
|
|
unreachable("Should not call PrintAnnotationStorage.print");
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* PLEASE NOTE: Only intended for usage within the API itself.
|
|
|
|
* @ignore
|
|
|
|
*/
|
|
|
|
get serializable() {
|
|
|
|
return this.#serializable;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
export { AnnotationStorage, PrintAnnotationStorage };
|