fune/browser/extensions/formautofill/content/FormAutofillFrameScript.js
Neil Deakin ec8b2c4f13 Bug 1573836, make autocomplete component fission compatible, r=mak,MattN
Differential Revision: https://phabricator.services.mozilla.com/D47093

--HG--
rename : toolkit/modules/AutoCompletePopupContent.jsm => toolkit/actors/AutoCompleteChild.jsm
rename : toolkit/components/satchel/AutoCompletePopup.jsm => toolkit/actors/AutoCompleteParent.jsm
extra : moz-landing-system : lando
2019-10-09 19:02:57 +00:00

199 lines
4.8 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/. */
/**
* Form Autofill frame script.
*/
"use strict";
/* eslint-env mozilla/frame-script */
var { Services } = ChromeUtils.import("resource://gre/modules/Services.jsm");
ChromeUtils.defineModuleGetter(
this,
"setTimeout",
"resource://gre/modules/Timer.jsm"
);
ChromeUtils.defineModuleGetter(
this,
"FormAutofill",
"resource://formautofill/FormAutofill.jsm"
);
ChromeUtils.defineModuleGetter(
this,
"FormAutofillContent",
"resource://formautofill/FormAutofillContent.jsm"
);
ChromeUtils.defineModuleGetter(
this,
"FormAutofillUtils",
"resource://formautofill/FormAutofillUtils.jsm"
);
ChromeUtils.defineModuleGetter(
this,
"AutoCompleteChild",
"resource://gre/actors/AutoCompleteChild.jsm"
);
/**
* Handles content's interactions for the frame.
*/
var FormAutofillFrameScript = {
_nextHandleElement: null,
_alreadyDOMContentLoaded: false,
_hasDOMContentLoadedHandler: false,
_hasPendingTask: false,
popupStateListener(messageName, data, target) {
if (!content || !FormAutofill.isAutofillEnabled) {
return;
}
const doc = target.document;
const { chromeEventHandler } = doc.ownerGlobal.docShell;
switch (messageName) {
case "FormAutoComplete:PopupClosed": {
FormAutofillContent.onPopupClosed(data.selectedRowStyle);
Services.tm.dispatchToMainThread(() => {
chromeEventHandler.removeEventListener(
"keydown",
FormAutofillContent._onKeyDown,
true
);
});
break;
}
case "FormAutoComplete:PopupOpened": {
chromeEventHandler.addEventListener(
"keydown",
FormAutofillContent._onKeyDown,
true
);
break;
}
}
},
_doIdentifyAutofillFields() {
if (this._hasPendingTask) {
return;
}
this._hasPendingTask = true;
setTimeout(() => {
FormAutofillContent.identifyAutofillFields(this._nextHandleElement);
this._hasPendingTask = false;
this._nextHandleElement = null;
// This is for testing purpose only which sends a message to indicate that the
// form has been identified, and ready to open popup.
sendAsyncMessage("FormAutofill:FieldsIdentified");
FormAutofillContent.updateActiveInput();
});
},
init() {
addEventListener("focusin", this);
addEventListener("DOMFormBeforeSubmit", this);
addEventListener("unload", this, { once: true });
addMessageListener("FormAutofill:PreviewProfile", this);
addMessageListener("FormAutofill:ClearForm", this);
AutoCompleteChild.addPopupStateListener(this.popupStateListener);
},
handleEvent(evt) {
if (!evt.isTrusted) {
return;
}
switch (evt.type) {
case "focusin": {
if (FormAutofill.isAutofillEnabled) {
this.onFocusIn(evt);
}
break;
}
case "DOMFormBeforeSubmit": {
if (FormAutofill.isAutofillEnabled) {
this.onDOMFormBeforeSubmit(evt);
}
break;
}
case "unload": {
AutoCompleteChild.removePopupStateListener(this.popupStateListener);
break;
}
default: {
throw new Error("Unexpected event type");
}
}
},
onFocusIn(evt) {
FormAutofillContent.updateActiveInput();
let element = evt.target;
if (!FormAutofillUtils.isFieldEligibleForAutofill(element)) {
return;
}
this._nextHandleElement = element;
if (!this._alreadyDOMContentLoaded) {
let doc = element.ownerDocument;
if (doc.readyState === "loading") {
if (!this._hasDOMContentLoadedHandler) {
this._hasDOMContentLoadedHandler = true;
doc.addEventListener(
"DOMContentLoaded",
() => this._doIdentifyAutofillFields(),
{ once: true }
);
}
return;
}
this._alreadyDOMContentLoaded = true;
}
this._doIdentifyAutofillFields();
},
/**
* Handle the DOMFormBeforeSubmit event.
* @param {Event} evt
*/
onDOMFormBeforeSubmit(evt) {
let formElement = evt.target;
if (!FormAutofill.isAutofillEnabled) {
return;
}
FormAutofillContent.formSubmitted(formElement);
},
receiveMessage(message) {
if (!FormAutofill.isAutofillEnabled) {
return;
}
const doc = content.document;
switch (message.name) {
case "FormAutofill:PreviewProfile": {
FormAutofillContent.previewProfile(doc);
break;
}
case "FormAutofill:ClearForm": {
FormAutofillContent.clearForm();
break;
}
}
},
};
FormAutofillFrameScript.init();