2025-02-05 19:34:11 +01:00
|
|
|
/* Copyright 2025 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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
import { getUuid } from "pdfjs-lib";
|
|
|
|
|
2025-02-13 21:15:27 +01:00
|
|
|
const KEY_STORAGE = "pdfjs.signature";
|
|
|
|
|
2025-02-05 19:34:11 +01:00
|
|
|
class SignatureStorage {
|
|
|
|
// TODO: Encrypt the data in using a password and add a UI for entering it.
|
|
|
|
// We could use the Web Crypto API for this (see https://bradyjoslin.com/blog/encryption-webcrypto/
|
|
|
|
// for an example).
|
|
|
|
|
2025-02-13 21:15:27 +01:00
|
|
|
#eventBus;
|
|
|
|
|
2025-02-05 19:34:11 +01:00
|
|
|
#signatures = null;
|
|
|
|
|
2025-02-13 21:15:27 +01:00
|
|
|
#signal = null;
|
|
|
|
|
|
|
|
constructor(eventBus, signal) {
|
|
|
|
this.#eventBus = eventBus;
|
|
|
|
this.#signal = signal;
|
|
|
|
}
|
|
|
|
|
2025-02-05 19:34:11 +01:00
|
|
|
#save() {
|
2025-02-12 19:29:16 +01:00
|
|
|
localStorage.setItem(
|
2025-02-13 21:15:27 +01:00
|
|
|
KEY_STORAGE,
|
2025-04-09 12:06:40 +02:00
|
|
|
JSON.stringify(Object.fromEntries(this.#signatures))
|
2025-02-12 19:29:16 +01:00
|
|
|
);
|
2025-02-05 19:34:11 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
async getAll() {
|
2025-02-13 21:15:27 +01:00
|
|
|
if (this.#signal) {
|
|
|
|
window.addEventListener(
|
|
|
|
"storage",
|
|
|
|
({ key }) => {
|
|
|
|
if (key === KEY_STORAGE) {
|
|
|
|
this.#signatures = null;
|
|
|
|
this.#eventBus?.dispatch("storedsignatureschanged", {
|
|
|
|
source: this,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
},
|
|
|
|
{ signal: this.#signal }
|
|
|
|
);
|
|
|
|
this.#signal = null;
|
|
|
|
}
|
2025-02-05 19:34:11 +01:00
|
|
|
if (!this.#signatures) {
|
2025-02-12 19:29:16 +01:00
|
|
|
this.#signatures = new Map();
|
2025-02-13 21:15:27 +01:00
|
|
|
const data = localStorage.getItem(KEY_STORAGE);
|
2025-02-12 19:29:16 +01:00
|
|
|
if (data) {
|
|
|
|
for (const [key, value] of Object.entries(JSON.parse(data))) {
|
|
|
|
this.#signatures.set(key, value);
|
|
|
|
}
|
|
|
|
}
|
2025-02-05 19:34:11 +01:00
|
|
|
}
|
|
|
|
return this.#signatures;
|
|
|
|
}
|
|
|
|
|
|
|
|
async isFull() {
|
2025-02-12 19:29:16 +01:00
|
|
|
// Only allow 5 signatures to be saved.
|
2025-02-24 19:50:40 +01:00
|
|
|
return (await this.size()) === 5;
|
|
|
|
}
|
|
|
|
|
|
|
|
async size() {
|
|
|
|
return (await this.getAll()).size;
|
2025-02-05 19:34:11 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
async create(data) {
|
|
|
|
if (await this.isFull()) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
const uuid = getUuid();
|
2025-02-12 19:29:16 +01:00
|
|
|
this.#signatures.set(uuid, data);
|
2025-02-05 19:34:11 +01:00
|
|
|
this.#save();
|
|
|
|
|
|
|
|
return uuid;
|
|
|
|
}
|
|
|
|
|
|
|
|
async delete(uuid) {
|
|
|
|
const signatures = await this.getAll();
|
2025-02-12 19:29:16 +01:00
|
|
|
if (!signatures.has(uuid)) {
|
2025-02-05 19:34:11 +01:00
|
|
|
return false;
|
|
|
|
}
|
2025-02-12 19:29:16 +01:00
|
|
|
signatures.delete(uuid);
|
2025-02-05 19:34:11 +01:00
|
|
|
this.#save();
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
export { SignatureStorage };
|