Some checks failed
		
		
	
	Types tests / Test (lts/*) (push) Has been cancelled
				
			Lint / Lint (lts/*) (push) Has been cancelled
				
			CodeQL / Analyze (javascript) (push) Has been cancelled
				
			CI / Test (20) (push) Has been cancelled
				
			CI / Test (22) (push) Has been cancelled
				
			CI / Test (24) (push) Has been cancelled
				
			
		
			
				
	
	
		
			338 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			338 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
/* Copyright 2012 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.
 | 
						|
 */
 | 
						|
 | 
						|
/** @typedef {import("../src/display/api").PDFPageProxy} PDFPageProxy */
 | 
						|
// eslint-disable-next-line max-len
 | 
						|
/** @typedef {import("../src/display/display_utils").PageViewport} PageViewport */
 | 
						|
/** @typedef {import("./text_highlighter").TextHighlighter} TextHighlighter */
 | 
						|
// eslint-disable-next-line max-len
 | 
						|
/** @typedef {import("./text_accessibility.js").TextAccessibilityManager} TextAccessibilityManager */
 | 
						|
 | 
						|
import { normalizeUnicode, stopEvent, TextLayer } from "pdfjs-lib";
 | 
						|
import { removeNullCharacters } from "./ui_utils.js";
 | 
						|
 | 
						|
/**
 | 
						|
 * @typedef {Object} TextLayerBuilderOptions
 | 
						|
 * @property {PDFPageProxy} pdfPage
 | 
						|
 * @property {TextHighlighter} [highlighter] - Optional object that will handle
 | 
						|
 *   highlighting text from the find controller.
 | 
						|
 * @property {TextAccessibilityManager} [accessibilityManager]
 | 
						|
 * @property {boolean} [enablePermissions]
 | 
						|
 * @property {function} [onAppend]
 | 
						|
 */
 | 
						|
 | 
						|
/**
 | 
						|
 * @typedef {Object} TextLayerBuilderRenderOptions
 | 
						|
 * @property {PageViewport} viewport
 | 
						|
 * @property {Object} [textContentParams]
 | 
						|
 */
 | 
						|
 | 
						|
/**
 | 
						|
 * The text layer builder provides text selection functionality for the PDF.
 | 
						|
 * It does this by creating overlay divs over the PDF's text. These divs
 | 
						|
 * contain text that matches the PDF text they are overlaying.
 | 
						|
 */
 | 
						|
class TextLayerBuilder {
 | 
						|
  #enablePermissions = false;
 | 
						|
 | 
						|
  #onAppend = null;
 | 
						|
 | 
						|
  #renderingDone = false;
 | 
						|
 | 
						|
  #textLayer = null;
 | 
						|
 | 
						|
  static #textLayers = new Map();
 | 
						|
 | 
						|
  static #selectionChangeAbortController = null;
 | 
						|
 | 
						|
  /**
 | 
						|
   * @param {TextLayerBuilderOptions} options
 | 
						|
   */
 | 
						|
  constructor({
 | 
						|
    pdfPage,
 | 
						|
    highlighter = null,
 | 
						|
    accessibilityManager = null,
 | 
						|
    enablePermissions = false,
 | 
						|
    onAppend = null,
 | 
						|
  }) {
 | 
						|
    this.pdfPage = pdfPage;
 | 
						|
    this.highlighter = highlighter;
 | 
						|
    this.accessibilityManager = accessibilityManager;
 | 
						|
    this.#enablePermissions = enablePermissions === true;
 | 
						|
    this.#onAppend = onAppend;
 | 
						|
 | 
						|
    this.div = document.createElement("div");
 | 
						|
    this.div.tabIndex = 0;
 | 
						|
    this.div.className = "textLayer";
 | 
						|
  }
 | 
						|
 | 
						|
  /**
 | 
						|
   * Renders the text layer.
 | 
						|
   * @param {TextLayerBuilderRenderOptions} options
 | 
						|
   * @returns {Promise<void>}
 | 
						|
   */
 | 
						|
  async render({ viewport, textContentParams = null }) {
 | 
						|
    if (this.#renderingDone && this.#textLayer) {
 | 
						|
      this.#textLayer.update({
 | 
						|
        viewport,
 | 
						|
        onBefore: this.hide.bind(this),
 | 
						|
      });
 | 
						|
      this.show();
 | 
						|
      return;
 | 
						|
    }
 | 
						|
 | 
						|
    this.cancel();
 | 
						|
    this.#textLayer = new TextLayer({
 | 
						|
      textContentSource: this.pdfPage.streamTextContent(
 | 
						|
        textContentParams || {
 | 
						|
          includeMarkedContent: true,
 | 
						|
          disableNormalization: true,
 | 
						|
        }
 | 
						|
      ),
 | 
						|
      container: this.div,
 | 
						|
      viewport,
 | 
						|
    });
 | 
						|
 | 
						|
    const { textDivs, textContentItemsStr } = this.#textLayer;
 | 
						|
    this.highlighter?.setTextMapping(textDivs, textContentItemsStr);
 | 
						|
    this.accessibilityManager?.setTextMapping(textDivs);
 | 
						|
 | 
						|
    await this.#textLayer.render();
 | 
						|
    this.#renderingDone = true;
 | 
						|
 | 
						|
    const endOfContent = document.createElement("div");
 | 
						|
    endOfContent.className = "endOfContent";
 | 
						|
    this.div.append(endOfContent);
 | 
						|
 | 
						|
    this.#bindMouse(endOfContent);
 | 
						|
    // Ensure that the textLayer is appended to the DOM *before* handling
 | 
						|
    // e.g. a pending search operation.
 | 
						|
    this.#onAppend?.(this.div);
 | 
						|
    this.highlighter?.enable();
 | 
						|
    this.accessibilityManager?.enable();
 | 
						|
  }
 | 
						|
 | 
						|
  hide() {
 | 
						|
    if (!this.div.hidden && this.#renderingDone) {
 | 
						|
      // We turn off the highlighter in order to avoid to scroll into view an
 | 
						|
      // element of the text layer which could be hidden.
 | 
						|
      this.highlighter?.disable();
 | 
						|
      this.div.hidden = true;
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  show() {
 | 
						|
    if (this.div.hidden && this.#renderingDone) {
 | 
						|
      this.div.hidden = false;
 | 
						|
      this.highlighter?.enable();
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  /**
 | 
						|
   * Cancel rendering of the text layer.
 | 
						|
   */
 | 
						|
  cancel() {
 | 
						|
    this.#textLayer?.cancel();
 | 
						|
    this.#textLayer = null;
 | 
						|
 | 
						|
    this.highlighter?.disable();
 | 
						|
    this.accessibilityManager?.disable();
 | 
						|
    TextLayerBuilder.#removeGlobalSelectionListener(this.div);
 | 
						|
  }
 | 
						|
 | 
						|
  /**
 | 
						|
   * Improves text selection by adding an additional div where the mouse was
 | 
						|
   * clicked. This reduces flickering of the content if the mouse is slowly
 | 
						|
   * dragged up or down.
 | 
						|
   */
 | 
						|
  #bindMouse(end) {
 | 
						|
    const { div } = this;
 | 
						|
 | 
						|
    div.addEventListener("mousedown", () => {
 | 
						|
      div.classList.add("selecting");
 | 
						|
    });
 | 
						|
 | 
						|
    div.addEventListener("copy", event => {
 | 
						|
      if (!this.#enablePermissions) {
 | 
						|
        const selection = document.getSelection();
 | 
						|
        event.clipboardData.setData(
 | 
						|
          "text/plain",
 | 
						|
          removeNullCharacters(normalizeUnicode(selection.toString()))
 | 
						|
        );
 | 
						|
      }
 | 
						|
      stopEvent(event);
 | 
						|
    });
 | 
						|
 | 
						|
    TextLayerBuilder.#textLayers.set(div, end);
 | 
						|
    TextLayerBuilder.#enableGlobalSelectionListener();
 | 
						|
  }
 | 
						|
 | 
						|
  static #removeGlobalSelectionListener(textLayerDiv) {
 | 
						|
    this.#textLayers.delete(textLayerDiv);
 | 
						|
 | 
						|
    if (this.#textLayers.size === 0) {
 | 
						|
      this.#selectionChangeAbortController?.abort();
 | 
						|
      this.#selectionChangeAbortController = null;
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  static #enableGlobalSelectionListener() {
 | 
						|
    if (this.#selectionChangeAbortController) {
 | 
						|
      // document-level event listeners already installed
 | 
						|
      return;
 | 
						|
    }
 | 
						|
    this.#selectionChangeAbortController = new AbortController();
 | 
						|
    const { signal } = this.#selectionChangeAbortController;
 | 
						|
 | 
						|
    const reset = (end, textLayer) => {
 | 
						|
      if (typeof PDFJSDev === "undefined" || !PDFJSDev.test("MOZCENTRAL")) {
 | 
						|
        textLayer.append(end);
 | 
						|
        end.style.width = "";
 | 
						|
        end.style.height = "";
 | 
						|
      }
 | 
						|
      textLayer.classList.remove("selecting");
 | 
						|
    };
 | 
						|
 | 
						|
    let isPointerDown = false;
 | 
						|
    document.addEventListener(
 | 
						|
      "pointerdown",
 | 
						|
      () => {
 | 
						|
        isPointerDown = true;
 | 
						|
      },
 | 
						|
      { signal }
 | 
						|
    );
 | 
						|
    document.addEventListener(
 | 
						|
      "pointerup",
 | 
						|
      () => {
 | 
						|
        isPointerDown = false;
 | 
						|
        this.#textLayers.forEach(reset);
 | 
						|
      },
 | 
						|
      { signal }
 | 
						|
    );
 | 
						|
    window.addEventListener(
 | 
						|
      "blur",
 | 
						|
      () => {
 | 
						|
        isPointerDown = false;
 | 
						|
        this.#textLayers.forEach(reset);
 | 
						|
      },
 | 
						|
      { signal }
 | 
						|
    );
 | 
						|
    document.addEventListener(
 | 
						|
      "keyup",
 | 
						|
      () => {
 | 
						|
        if (!isPointerDown) {
 | 
						|
          this.#textLayers.forEach(reset);
 | 
						|
        }
 | 
						|
      },
 | 
						|
      { signal }
 | 
						|
    );
 | 
						|
 | 
						|
    if (typeof PDFJSDev === "undefined" || !PDFJSDev.test("MOZCENTRAL")) {
 | 
						|
      // eslint-disable-next-line no-var
 | 
						|
      var isFirefox, prevRange;
 | 
						|
    }
 | 
						|
 | 
						|
    document.addEventListener(
 | 
						|
      "selectionchange",
 | 
						|
      () => {
 | 
						|
        const selection = document.getSelection();
 | 
						|
        if (selection.rangeCount === 0) {
 | 
						|
          this.#textLayers.forEach(reset);
 | 
						|
          return;
 | 
						|
        }
 | 
						|
 | 
						|
        // Even though the spec says that .rangeCount should be 0 or 1, Firefox
 | 
						|
        // creates multiple ranges when selecting across multiple pages.
 | 
						|
        // Make sure to collect all the .textLayer elements where the selection
 | 
						|
        // is happening.
 | 
						|
        const activeTextLayers = new Set();
 | 
						|
        for (let i = 0; i < selection.rangeCount; i++) {
 | 
						|
          const range = selection.getRangeAt(i);
 | 
						|
          for (const textLayerDiv of this.#textLayers.keys()) {
 | 
						|
            if (
 | 
						|
              !activeTextLayers.has(textLayerDiv) &&
 | 
						|
              range.intersectsNode(textLayerDiv)
 | 
						|
            ) {
 | 
						|
              activeTextLayers.add(textLayerDiv);
 | 
						|
            }
 | 
						|
          }
 | 
						|
        }
 | 
						|
 | 
						|
        for (const [textLayerDiv, endDiv] of this.#textLayers) {
 | 
						|
          if (activeTextLayers.has(textLayerDiv)) {
 | 
						|
            textLayerDiv.classList.add("selecting");
 | 
						|
          } else {
 | 
						|
            reset(endDiv, textLayerDiv);
 | 
						|
          }
 | 
						|
        }
 | 
						|
 | 
						|
        if (typeof PDFJSDev !== "undefined" && PDFJSDev.test("MOZCENTRAL")) {
 | 
						|
          return;
 | 
						|
        }
 | 
						|
        if (typeof PDFJSDev === "undefined" || !PDFJSDev.test("CHROME")) {
 | 
						|
          isFirefox ??=
 | 
						|
            getComputedStyle(
 | 
						|
              this.#textLayers.values().next().value
 | 
						|
            ).getPropertyValue("-moz-user-select") === "none";
 | 
						|
 | 
						|
          if (isFirefox) {
 | 
						|
            return;
 | 
						|
          }
 | 
						|
        }
 | 
						|
        // In non-Firefox browsers, when hovering over an empty space (thus,
 | 
						|
        // on .endOfContent), the selection will expand to cover all the
 | 
						|
        // text between the current selection and .endOfContent. By moving
 | 
						|
        // .endOfContent to right after (or before, depending on which side
 | 
						|
        // of the selection the user is moving), we limit the selection jump
 | 
						|
        // to at most cover the enteirety of the <span> where the selection
 | 
						|
        // is being modified.
 | 
						|
        const range = selection.getRangeAt(0);
 | 
						|
        const modifyStart =
 | 
						|
          prevRange &&
 | 
						|
          (range.compareBoundaryPoints(Range.END_TO_END, prevRange) === 0 ||
 | 
						|
            range.compareBoundaryPoints(Range.START_TO_END, prevRange) === 0);
 | 
						|
        let anchor = modifyStart ? range.startContainer : range.endContainer;
 | 
						|
        if (anchor.nodeType === Node.TEXT_NODE) {
 | 
						|
          anchor = anchor.parentNode;
 | 
						|
        }
 | 
						|
        if (!modifyStart && range.endOffset === 0) {
 | 
						|
          do {
 | 
						|
            while (!anchor.previousSibling) {
 | 
						|
              anchor = anchor.parentNode;
 | 
						|
            }
 | 
						|
            anchor = anchor.previousSibling;
 | 
						|
          } while (!anchor.childNodes.length);
 | 
						|
        }
 | 
						|
 | 
						|
        const parentTextLayer = anchor.parentElement?.closest(".textLayer");
 | 
						|
        const endDiv = this.#textLayers.get(parentTextLayer);
 | 
						|
        if (endDiv) {
 | 
						|
          endDiv.style.width = parentTextLayer.style.width;
 | 
						|
          endDiv.style.height = parentTextLayer.style.height;
 | 
						|
          anchor.parentElement.insertBefore(
 | 
						|
            endDiv,
 | 
						|
            modifyStart ? anchor : anchor.nextSibling
 | 
						|
          );
 | 
						|
        }
 | 
						|
 | 
						|
        prevRange = range.cloneRange();
 | 
						|
      },
 | 
						|
      { signal }
 | 
						|
    );
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
export { TextLayerBuilder };
 |