2016-09-07 20:30:26 +09:00
|
|
|
/* Copyright 2017 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.
|
|
|
|
*/
|
|
|
|
|
2023-06-28 19:23:02 +09:00
|
|
|
import { AnnotationEditorType, shadow } from "pdfjs-lib";
|
2023-02-04 21:55:12 +09:00
|
|
|
import { CursorTool, PresentationModeState } from "./ui_utils.js";
|
2020-01-02 20:00:16 +09:00
|
|
|
import { GrabToPan } from "./grab_to_pan.js";
|
2016-09-07 20:30:26 +09:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @typedef {Object} PDFCursorToolsOptions
|
|
|
|
* @property {HTMLDivElement} container - The document container.
|
|
|
|
* @property {EventBus} eventBus - The application event bus.
|
2019-10-12 23:30:32 +09:00
|
|
|
* @property {number} [cursorToolOnLoad] - The cursor tool that will be enabled
|
|
|
|
* on load; the constants from {CursorTool} should be used. The default value
|
|
|
|
* is `CursorTool.SELECT`.
|
2016-09-07 20:30:26 +09:00
|
|
|
*/
|
|
|
|
|
|
|
|
class PDFCursorTools {
|
2023-06-28 19:16:46 +09:00
|
|
|
#active = CursorTool.SELECT;
|
|
|
|
|
|
|
|
#prevActive = null;
|
|
|
|
|
2016-09-07 20:30:26 +09:00
|
|
|
/**
|
|
|
|
* @param {PDFCursorToolsOptions} options
|
|
|
|
*/
|
2018-03-20 07:24:56 +09:00
|
|
|
constructor({ container, eventBus, cursorToolOnLoad = CursorTool.SELECT }) {
|
2016-09-07 20:30:26 +09:00
|
|
|
this.container = container;
|
|
|
|
this.eventBus = eventBus;
|
|
|
|
|
2022-03-06 23:36:32 +09:00
|
|
|
this.#addEventListeners();
|
2016-09-07 20:30:26 +09:00
|
|
|
|
2018-07-08 17:55:56 +09:00
|
|
|
// Defer the initial `switchTool` call, to give other viewer components
|
|
|
|
// time to initialize *and* register 'cursortoolchanged' event listeners.
|
2018-03-20 07:24:56 +09:00
|
|
|
Promise.resolve().then(() => {
|
|
|
|
this.switchTool(cursorToolOnLoad);
|
|
|
|
});
|
2016-09-07 20:30:26 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2019-10-13 00:02:54 +09:00
|
|
|
* @type {number} One of the values in {CursorTool}.
|
2016-09-07 20:30:26 +09:00
|
|
|
*/
|
|
|
|
get activeTool() {
|
2023-06-28 19:16:46 +09:00
|
|
|
return this.#active;
|
2016-09-07 20:30:26 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param {number} tool - The cursor mode that should be switched to,
|
|
|
|
* must be one of the values in {CursorTool}.
|
|
|
|
*/
|
|
|
|
switchTool(tool) {
|
2023-06-28 19:16:46 +09:00
|
|
|
if (this.#prevActive !== null) {
|
2022-09-29 06:25:51 +09:00
|
|
|
// Cursor tools cannot be used in PresentationMode/AnnotationEditor.
|
|
|
|
return;
|
2016-09-07 20:30:26 +09:00
|
|
|
}
|
2023-06-28 19:16:46 +09:00
|
|
|
if (tool === this.#active) {
|
2016-09-07 20:30:26 +09:00
|
|
|
return; // The requested tool is already active.
|
|
|
|
}
|
|
|
|
|
2019-12-27 08:22:32 +09:00
|
|
|
const disableActiveTool = () => {
|
2023-06-28 19:16:46 +09:00
|
|
|
switch (this.#active) {
|
2016-09-07 20:30:26 +09:00
|
|
|
case CursorTool.SELECT:
|
|
|
|
break;
|
|
|
|
case CursorTool.HAND:
|
2023-06-28 19:23:02 +09:00
|
|
|
this._handTool.deactivate();
|
2016-09-07 20:30:26 +09:00
|
|
|
break;
|
|
|
|
case CursorTool.ZOOM:
|
|
|
|
/* falls through */
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2019-12-26 04:03:46 +09:00
|
|
|
// Enable the new cursor tool.
|
|
|
|
switch (tool) {
|
2016-09-07 20:30:26 +09:00
|
|
|
case CursorTool.SELECT:
|
|
|
|
disableActiveTool();
|
|
|
|
break;
|
|
|
|
case CursorTool.HAND:
|
|
|
|
disableActiveTool();
|
2023-06-28 19:23:02 +09:00
|
|
|
this._handTool.activate();
|
2016-09-07 20:30:26 +09:00
|
|
|
break;
|
|
|
|
case CursorTool.ZOOM:
|
|
|
|
/* falls through */
|
|
|
|
default:
|
|
|
|
console.error(`switchTool: "${tool}" is an unsupported value.`);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
// Update the active tool *after* it has been validated above,
|
|
|
|
// in order to prevent setting it to an invalid state.
|
2023-06-28 19:16:46 +09:00
|
|
|
this.#active = tool;
|
2016-09-07 20:30:26 +09:00
|
|
|
|
|
|
|
this.eventBus.dispatch("cursortoolchanged", {
|
|
|
|
source: this,
|
2023-06-28 19:16:46 +09:00
|
|
|
tool,
|
2016-09-07 20:30:26 +09:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2022-03-06 23:36:32 +09:00
|
|
|
#addEventListeners() {
|
Re-factor the `EventBus` to allow servicing of "external" event listeners *after* the viewer components have updated
Since the goal has always been, essentially since the `EventBus` abstraction was added, to remove all dispatching of DOM events[1] from the viewer components this patch tries to address one thing that came up when updating the examples:
The DOM events are always dispatched last, and it's thus guaranteed that all internal event listeners have been invoked first.
However, there's no such guarantees with the general `EventBus` functionality and the order in which event listeners are invoked is *not* specified. With the promotion of the `EventBus` in the examples, over DOM events, it seems like a good idea to at least *try* to keep this ordering invariant[2] intact.
Obviously this won't prevent anyone from manually calling the new *internal* viewer component methods on the `EventBus`, but hopefully that won't be too common since any existing third-party code would obviously use the `on`/`off` methods and that all of the examples shows the *correct* usage (which should be similarily documented on the "Third party viewer usage" Wiki-page).
---
[1] Looking at the various Firefox-tests, I'm not sure that it'll be possible to (easily) re-write all of them to not rely on DOM events (since getting access to `PDFViewerApplication` might be generally difficult/messy depending on scopes).
In any case, even if technically feasible, it would most likely add *a lot* of complication that may not be desireable in the various Firefox-tests. All-in-all, I'd be fine with keeping the DOM events only for the `MOZCENTRAL` target and gated on `Cu.isInAutomation` (or similar) rather than a preference.
[2] I wouldn't expect any *real* bugs in a custom implementation, simply based on event ordering, but it nonetheless seem like a good idea if any "external" events are still handled last.
2020-02-27 07:33:27 +09:00
|
|
|
this.eventBus._on("switchcursortool", evt => {
|
2016-09-07 20:30:26 +09:00
|
|
|
this.switchTool(evt.tool);
|
|
|
|
});
|
|
|
|
|
2022-09-29 06:25:51 +09:00
|
|
|
let annotationEditorMode = AnnotationEditorType.NONE,
|
|
|
|
presentationModeState = PresentationModeState.NORMAL;
|
2016-09-07 20:30:26 +09:00
|
|
|
|
2022-09-29 06:25:51 +09:00
|
|
|
const disableActive = () => {
|
2023-06-28 19:16:46 +09:00
|
|
|
const prevActive = this.#active;
|
2022-09-29 06:25:51 +09:00
|
|
|
|
|
|
|
this.switchTool(CursorTool.SELECT);
|
2023-06-28 19:16:46 +09:00
|
|
|
this.#prevActive ??= prevActive; // Keep track of the first one.
|
2022-09-29 06:25:51 +09:00
|
|
|
};
|
|
|
|
const enableActive = () => {
|
2023-06-28 19:16:46 +09:00
|
|
|
const prevActive = this.#prevActive;
|
2022-09-29 06:25:51 +09:00
|
|
|
|
|
|
|
if (
|
2023-06-28 19:16:46 +09:00
|
|
|
prevActive !== null &&
|
2022-09-29 06:25:51 +09:00
|
|
|
annotationEditorMode === AnnotationEditorType.NONE &&
|
|
|
|
presentationModeState === PresentationModeState.NORMAL
|
|
|
|
) {
|
2023-06-28 19:16:46 +09:00
|
|
|
this.#prevActive = null;
|
|
|
|
this.switchTool(prevActive);
|
2022-09-29 06:25:51 +09:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
this.eventBus._on("secondarytoolbarreset", evt => {
|
2023-06-28 19:16:46 +09:00
|
|
|
if (this.#prevActive !== null) {
|
2022-09-29 06:25:51 +09:00
|
|
|
annotationEditorMode = AnnotationEditorType.NONE;
|
|
|
|
presentationModeState = PresentationModeState.NORMAL;
|
|
|
|
|
|
|
|
enableActive();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
this.eventBus._on("annotationeditormodechanged", ({ mode }) => {
|
|
|
|
annotationEditorMode = mode;
|
|
|
|
|
|
|
|
if (mode === AnnotationEditorType.NONE) {
|
|
|
|
enableActive();
|
|
|
|
} else {
|
|
|
|
disableActive();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
this.eventBus._on("presentationmodechanged", ({ state }) => {
|
|
|
|
presentationModeState = state;
|
|
|
|
|
|
|
|
if (state === PresentationModeState.NORMAL) {
|
|
|
|
enableActive();
|
|
|
|
} else if (state === PresentationModeState.FULLSCREEN) {
|
|
|
|
disableActive();
|
2016-09-07 20:30:26 +09:00
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
2023-06-28 19:23:02 +09:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @private
|
|
|
|
*/
|
|
|
|
get _handTool() {
|
|
|
|
return shadow(
|
|
|
|
this,
|
|
|
|
"_handTool",
|
|
|
|
new GrabToPan({
|
|
|
|
element: this.container,
|
|
|
|
})
|
|
|
|
);
|
|
|
|
}
|
2016-09-07 20:30:26 +09:00
|
|
|
}
|
|
|
|
|
2023-02-04 21:55:12 +09:00
|
|
|
export { PDFCursorTools };
|