forked from mirrors/gecko-dev
		
	
		
			
				
	
	
		
			121 lines
		
	
	
	
		
			3 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			121 lines
		
	
	
	
		
			3 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
/* This Source Code Form is subject to the terms of the Mozilla Public
 | 
						|
 * License, v. 2.0. If a copy of the MPL was not distributed with this
 | 
						|
 * file, You can obtain one at http://mozilla.org/MPL/2.0/. */
 | 
						|
 | 
						|
import { XPCOMUtils } from "resource://gre/modules/XPCOMUtils.sys.mjs";
 | 
						|
 | 
						|
const lazy = {};
 | 
						|
 | 
						|
ChromeUtils.defineESModuleGetters(lazy, {
 | 
						|
  PageDataSchema: "resource:///modules/pagedata/PageDataSchema.sys.mjs",
 | 
						|
  PrivateBrowsingUtils: "resource://gre/modules/PrivateBrowsingUtils.sys.mjs",
 | 
						|
});
 | 
						|
 | 
						|
// We defer any attempt to check for page data for a short time after a page
 | 
						|
// loads to allow JS to operate.
 | 
						|
XPCOMUtils.defineLazyPreferenceGetter(
 | 
						|
  lazy,
 | 
						|
  "READY_DELAY",
 | 
						|
  "browser.pagedata.readyDelay",
 | 
						|
  500
 | 
						|
);
 | 
						|
 | 
						|
/**
 | 
						|
 * The actor responsible for monitoring a page for page data.
 | 
						|
 */
 | 
						|
export class PageDataChild extends JSWindowActorChild {
 | 
						|
  #isContentWindowPrivate = true;
 | 
						|
  /**
 | 
						|
   * Used to debounce notifications about a page being ready.
 | 
						|
   *
 | 
						|
   * @type {Timer | null}
 | 
						|
   */
 | 
						|
  #deferTimer = null;
 | 
						|
 | 
						|
  /**
 | 
						|
   * Called when the actor is created for a new page.
 | 
						|
   */
 | 
						|
  actorCreated() {
 | 
						|
    this.#isContentWindowPrivate =
 | 
						|
      lazy.PrivateBrowsingUtils.isContentWindowPrivate(this.contentWindow);
 | 
						|
  }
 | 
						|
 | 
						|
  /**
 | 
						|
   * Called when the page is destroyed.
 | 
						|
   */
 | 
						|
  didDestroy() {
 | 
						|
    if (this.#deferTimer) {
 | 
						|
      this.#deferTimer.cancel();
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  /**
 | 
						|
   * Called when the page has signalled it is done loading. This signal is
 | 
						|
   * debounced by READY_DELAY.
 | 
						|
   */
 | 
						|
  #deferReady() {
 | 
						|
    if (!this.#deferTimer) {
 | 
						|
      this.#deferTimer = Cc["@mozilla.org/timer;1"].createInstance(Ci.nsITimer);
 | 
						|
    }
 | 
						|
 | 
						|
    // If the timer was already running this re-starts it.
 | 
						|
    this.#deferTimer.initWithCallback(
 | 
						|
      () => {
 | 
						|
        this.#deferTimer = null;
 | 
						|
        this.sendAsyncMessage("PageData:DocumentReady", {
 | 
						|
          url: this.document.documentURI,
 | 
						|
        });
 | 
						|
      },
 | 
						|
      lazy.READY_DELAY,
 | 
						|
      Ci.nsITimer.TYPE_ONE_SHOT_LOW_PRIORITY
 | 
						|
    );
 | 
						|
  }
 | 
						|
 | 
						|
  /**
 | 
						|
   * Called when a message is received from the parent process.
 | 
						|
   *
 | 
						|
   * @param {ReceiveMessageArgument} msg
 | 
						|
   *   The received message.
 | 
						|
   *
 | 
						|
   * @returns {Promise | undefined}
 | 
						|
   *   A promise for the requested data or undefined if no data was requested.
 | 
						|
   */
 | 
						|
  receiveMessage(msg) {
 | 
						|
    if (this.#isContentWindowPrivate) {
 | 
						|
      return undefined;
 | 
						|
    }
 | 
						|
 | 
						|
    switch (msg.name) {
 | 
						|
      case "PageData:CheckLoaded":
 | 
						|
        // The service just started in the parent. Check if this document is
 | 
						|
        // already loaded.
 | 
						|
        if (this.document.readystate == "complete") {
 | 
						|
          this.#deferReady();
 | 
						|
        }
 | 
						|
        break;
 | 
						|
      case "PageData:Collect":
 | 
						|
        return lazy.PageDataSchema.collectPageData(this.document);
 | 
						|
    }
 | 
						|
 | 
						|
    return undefined;
 | 
						|
  }
 | 
						|
 | 
						|
  /**
 | 
						|
   * DOM event handler.
 | 
						|
   *
 | 
						|
   * @param {Event} event
 | 
						|
   *   The DOM event.
 | 
						|
   */
 | 
						|
  handleEvent(event) {
 | 
						|
    if (this.#isContentWindowPrivate) {
 | 
						|
      return;
 | 
						|
    }
 | 
						|
 | 
						|
    switch (event.type) {
 | 
						|
      case "DOMContentLoaded":
 | 
						|
      case "pageshow":
 | 
						|
        this.#deferReady();
 | 
						|
        break;
 | 
						|
    }
 | 
						|
  }
 | 
						|
}
 |