mirror of
https://github.com/bitwarden/browser.git
synced 2025-01-04 18:37:45 +01:00
[PM-10420] Autofill focus jumps around after autofilling identity (#10361)
* [PM-10420] Autofill focus jumps around after autofilling identity ciphers * [PM-10420] Autofill focus jumps around after autofilling identity ciphers * [PM-10420] Autofill focus jumps around after autofilling identity ciphers * [PM-10420] Incorporating the feature flag within jest to test the validity of both implementations * [PM-10420] Refactoring how we compile the combined list of keywords * [PM-10420] Adding JSDocs to the implemented methods
This commit is contained in:
parent
c50a9063bc
commit
76351ce750
@ -76,6 +76,11 @@ export class AutoFillConstants {
|
|||||||
"textarea",
|
"textarea",
|
||||||
...AutoFillConstants.ExcludedAutofillTypes,
|
...AutoFillConstants.ExcludedAutofillTypes,
|
||||||
];
|
];
|
||||||
|
|
||||||
|
static readonly ExcludedIdentityAutocompleteTypes: Set<string> = new Set([
|
||||||
|
"current-password",
|
||||||
|
"new-password",
|
||||||
|
]);
|
||||||
}
|
}
|
||||||
|
|
||||||
export class CreditCardAutoFillConstants {
|
export class CreditCardAutoFillConstants {
|
||||||
|
@ -60,7 +60,7 @@ import {
|
|||||||
GenerateFillScriptOptions,
|
GenerateFillScriptOptions,
|
||||||
PageDetail,
|
PageDetail,
|
||||||
} from "./abstractions/autofill.service";
|
} from "./abstractions/autofill.service";
|
||||||
import { AutoFillConstants, IdentityAutoFillConstants } from "./autofill-constants";
|
import { AutoFillConstants } from "./autofill-constants";
|
||||||
import AutofillService from "./autofill.service";
|
import AutofillService from "./autofill.service";
|
||||||
|
|
||||||
const mockEquivalentDomains = [
|
const mockEquivalentDomains = [
|
||||||
@ -3056,12 +3056,12 @@ describe("AutofillService", () => {
|
|||||||
options.cipher.identity = mock<IdentityView>();
|
options.cipher.identity = mock<IdentityView>();
|
||||||
});
|
});
|
||||||
|
|
||||||
it("returns null if an identify is not found within the cipher", () => {
|
it("returns null if an identify is not found within the cipher", async () => {
|
||||||
options.cipher.identity = null;
|
options.cipher.identity = null;
|
||||||
jest.spyOn(autofillService as any, "makeScriptAction");
|
jest.spyOn(autofillService as any, "makeScriptAction");
|
||||||
jest.spyOn(autofillService as any, "makeScriptActionWithValue");
|
jest.spyOn(autofillService as any, "makeScriptActionWithValue");
|
||||||
|
|
||||||
const value = autofillService["generateIdentityFillScript"](
|
const value = await autofillService["generateIdentityFillScript"](
|
||||||
fillScript,
|
fillScript,
|
||||||
pageDetails,
|
pageDetails,
|
||||||
filledFields,
|
filledFields,
|
||||||
@ -3087,11 +3087,22 @@ describe("AutofillService", () => {
|
|||||||
jest.spyOn(autofillService as any, "makeScriptActionWithValue");
|
jest.spyOn(autofillService as any, "makeScriptActionWithValue");
|
||||||
});
|
});
|
||||||
|
|
||||||
it("will not attempt to match custom fields", () => {
|
let isRefactorFeatureFlagSet = false;
|
||||||
|
for (let index = 0; index < 2; index++) {
|
||||||
|
describe(`when the isRefactorFeatureFlagSet is ${isRefactorFeatureFlagSet}`, () => {
|
||||||
|
beforeEach(() => {
|
||||||
|
configService.getFeatureFlag.mockResolvedValue(isRefactorFeatureFlagSet);
|
||||||
|
});
|
||||||
|
|
||||||
|
afterAll(() => {
|
||||||
|
isRefactorFeatureFlagSet = true;
|
||||||
|
});
|
||||||
|
|
||||||
|
it("will not attempt to match custom fields", async () => {
|
||||||
const customField = createAutofillFieldMock({ tagName: "span" });
|
const customField = createAutofillFieldMock({ tagName: "span" });
|
||||||
pageDetails.fields.push(customField);
|
pageDetails.fields.push(customField);
|
||||||
|
|
||||||
const value = autofillService["generateIdentityFillScript"](
|
const value = await autofillService["generateIdentityFillScript"](
|
||||||
fillScript,
|
fillScript,
|
||||||
pageDetails,
|
pageDetails,
|
||||||
filledFields,
|
filledFields,
|
||||||
@ -3104,11 +3115,11 @@ describe("AutofillService", () => {
|
|||||||
expect(value.script).toStrictEqual([]);
|
expect(value.script).toStrictEqual([]);
|
||||||
});
|
});
|
||||||
|
|
||||||
it("will not attempt to match a field that is of an excluded type", () => {
|
it("will not attempt to match a field that is of an excluded type", async () => {
|
||||||
const excludedField = createAutofillFieldMock({ type: "hidden" });
|
const excludedField = createAutofillFieldMock({ type: "hidden" });
|
||||||
pageDetails.fields.push(excludedField);
|
pageDetails.fields.push(excludedField);
|
||||||
|
|
||||||
const value = autofillService["generateIdentityFillScript"](
|
const value = await autofillService["generateIdentityFillScript"](
|
||||||
fillScript,
|
fillScript,
|
||||||
pageDetails,
|
pageDetails,
|
||||||
filledFields,
|
filledFields,
|
||||||
@ -3124,11 +3135,11 @@ describe("AutofillService", () => {
|
|||||||
expect(value.script).toStrictEqual([]);
|
expect(value.script).toStrictEqual([]);
|
||||||
});
|
});
|
||||||
|
|
||||||
it("will not attempt to match a field that is not viewable", () => {
|
it("will not attempt to match a field that is not viewable", async () => {
|
||||||
const viewableField = createAutofillFieldMock({ viewable: false });
|
const viewableField = createAutofillFieldMock({ viewable: false });
|
||||||
pageDetails.fields.push(viewableField);
|
pageDetails.fields.push(viewableField);
|
||||||
|
|
||||||
const value = autofillService["generateIdentityFillScript"](
|
const value = await autofillService["generateIdentityFillScript"](
|
||||||
fillScript,
|
fillScript,
|
||||||
pageDetails,
|
pageDetails,
|
||||||
filledFields,
|
filledFields,
|
||||||
@ -3141,25 +3152,23 @@ describe("AutofillService", () => {
|
|||||||
expect(value.script).toStrictEqual([]);
|
expect(value.script).toStrictEqual([]);
|
||||||
});
|
});
|
||||||
|
|
||||||
it("will match a full name field to the vault item identity value", () => {
|
it("will match a full name field to the vault item identity value", async () => {
|
||||||
const fullNameField = createAutofillFieldMock({ opid: "fullName", htmlName: "full-name" });
|
const fullNameField = createAutofillFieldMock({
|
||||||
|
opid: "fullName",
|
||||||
|
htmlName: "full-name",
|
||||||
|
});
|
||||||
pageDetails.fields = [fullNameField];
|
pageDetails.fields = [fullNameField];
|
||||||
options.cipher.identity.firstName = firstName;
|
options.cipher.identity.firstName = firstName;
|
||||||
options.cipher.identity.middleName = middleName;
|
options.cipher.identity.middleName = middleName;
|
||||||
options.cipher.identity.lastName = lastName;
|
options.cipher.identity.lastName = lastName;
|
||||||
|
|
||||||
const value = autofillService["generateIdentityFillScript"](
|
const value = await autofillService["generateIdentityFillScript"](
|
||||||
fillScript,
|
fillScript,
|
||||||
pageDetails,
|
pageDetails,
|
||||||
filledFields,
|
filledFields,
|
||||||
options,
|
options,
|
||||||
);
|
);
|
||||||
|
|
||||||
expect(AutofillService["isFieldMatch"]).toHaveBeenCalledWith(
|
|
||||||
fullNameField.htmlName,
|
|
||||||
IdentityAutoFillConstants.FullNameFieldNames,
|
|
||||||
IdentityAutoFillConstants.FullNameFieldNameValues,
|
|
||||||
);
|
|
||||||
expect(autofillService["makeScriptActionWithValue"]).toHaveBeenCalledWith(
|
expect(autofillService["makeScriptActionWithValue"]).toHaveBeenCalledWith(
|
||||||
fillScript,
|
fillScript,
|
||||||
`${firstName} ${middleName} ${lastName}`,
|
`${firstName} ${middleName} ${lastName}`,
|
||||||
@ -3173,25 +3182,23 @@ describe("AutofillService", () => {
|
|||||||
]);
|
]);
|
||||||
});
|
});
|
||||||
|
|
||||||
it("will match a full name field to the a vault item that only has a last name", () => {
|
it("will match a full name field to the a vault item that only has a last name", async () => {
|
||||||
const fullNameField = createAutofillFieldMock({ opid: "fullName", htmlName: "full-name" });
|
const fullNameField = createAutofillFieldMock({
|
||||||
|
opid: "fullName",
|
||||||
|
htmlName: "full-name",
|
||||||
|
});
|
||||||
pageDetails.fields = [fullNameField];
|
pageDetails.fields = [fullNameField];
|
||||||
options.cipher.identity.firstName = "";
|
options.cipher.identity.firstName = "";
|
||||||
options.cipher.identity.middleName = "";
|
options.cipher.identity.middleName = "";
|
||||||
options.cipher.identity.lastName = lastName;
|
options.cipher.identity.lastName = lastName;
|
||||||
|
|
||||||
const value = autofillService["generateIdentityFillScript"](
|
const value = await autofillService["generateIdentityFillScript"](
|
||||||
fillScript,
|
fillScript,
|
||||||
pageDetails,
|
pageDetails,
|
||||||
filledFields,
|
filledFields,
|
||||||
options,
|
options,
|
||||||
);
|
);
|
||||||
|
|
||||||
expect(AutofillService["isFieldMatch"]).toHaveBeenCalledWith(
|
|
||||||
fullNameField.htmlName,
|
|
||||||
IdentityAutoFillConstants.FullNameFieldNames,
|
|
||||||
IdentityAutoFillConstants.FullNameFieldNameValues,
|
|
||||||
);
|
|
||||||
expect(autofillService["makeScriptActionWithValue"]).toHaveBeenCalledWith(
|
expect(autofillService["makeScriptActionWithValue"]).toHaveBeenCalledWith(
|
||||||
fillScript,
|
fillScript,
|
||||||
lastName,
|
lastName,
|
||||||
@ -3201,7 +3208,7 @@ describe("AutofillService", () => {
|
|||||||
expect(value.script[2]).toStrictEqual(["fill_by_opid", fullNameField.opid, lastName]);
|
expect(value.script[2]).toStrictEqual(["fill_by_opid", fullNameField.opid, lastName]);
|
||||||
});
|
});
|
||||||
|
|
||||||
it("will match first name, middle name, and last name fields to the vault item identity value", () => {
|
it("will match first name, middle name, and last name fields to the vault item identity value", async () => {
|
||||||
const firstNameField = createAutofillFieldMock({
|
const firstNameField = createAutofillFieldMock({
|
||||||
opid: "firstName",
|
opid: "firstName",
|
||||||
htmlName: "first-name",
|
htmlName: "first-name",
|
||||||
@ -3210,58 +3217,50 @@ describe("AutofillService", () => {
|
|||||||
opid: "middleName",
|
opid: "middleName",
|
||||||
htmlName: "middle-name",
|
htmlName: "middle-name",
|
||||||
});
|
});
|
||||||
const lastNameField = createAutofillFieldMock({ opid: "lastName", htmlName: "last-name" });
|
const lastNameField = createAutofillFieldMock({
|
||||||
|
opid: "lastName",
|
||||||
|
htmlName: "last-name",
|
||||||
|
});
|
||||||
pageDetails.fields = [firstNameField, middleNameField, lastNameField];
|
pageDetails.fields = [firstNameField, middleNameField, lastNameField];
|
||||||
options.cipher.identity.firstName = firstName;
|
options.cipher.identity.firstName = firstName;
|
||||||
options.cipher.identity.middleName = middleName;
|
options.cipher.identity.middleName = middleName;
|
||||||
options.cipher.identity.lastName = lastName;
|
options.cipher.identity.lastName = lastName;
|
||||||
|
|
||||||
const value = autofillService["generateIdentityFillScript"](
|
const value = await autofillService["generateIdentityFillScript"](
|
||||||
fillScript,
|
fillScript,
|
||||||
pageDetails,
|
pageDetails,
|
||||||
filledFields,
|
filledFields,
|
||||||
options,
|
options,
|
||||||
);
|
);
|
||||||
|
|
||||||
expect(AutofillService["isFieldMatch"]).toHaveBeenCalledWith(
|
expect(autofillService["makeScriptActionWithValue"]).toHaveBeenCalledWith(
|
||||||
firstNameField.htmlName,
|
|
||||||
IdentityAutoFillConstants.FirstnameFieldNames,
|
|
||||||
);
|
|
||||||
expect(AutofillService["isFieldMatch"]).toHaveBeenCalledWith(
|
|
||||||
middleNameField.htmlName,
|
|
||||||
IdentityAutoFillConstants.MiddlenameFieldNames,
|
|
||||||
);
|
|
||||||
expect(AutofillService["isFieldMatch"]).toHaveBeenCalledWith(
|
|
||||||
lastNameField.htmlName,
|
|
||||||
IdentityAutoFillConstants.LastnameFieldNames,
|
|
||||||
);
|
|
||||||
expect(autofillService["makeScriptAction"]).toHaveBeenCalledWith(
|
|
||||||
fillScript,
|
fillScript,
|
||||||
options.cipher.identity,
|
options.cipher.identity.firstName,
|
||||||
expect.anything(),
|
firstNameField,
|
||||||
filledFields,
|
filledFields,
|
||||||
firstNameField.opid,
|
|
||||||
);
|
);
|
||||||
expect(autofillService["makeScriptAction"]).toHaveBeenCalledWith(
|
expect(autofillService["makeScriptActionWithValue"]).toHaveBeenCalledWith(
|
||||||
fillScript,
|
fillScript,
|
||||||
options.cipher.identity,
|
options.cipher.identity.middleName,
|
||||||
expect.anything(),
|
middleNameField,
|
||||||
filledFields,
|
filledFields,
|
||||||
middleNameField.opid,
|
|
||||||
);
|
);
|
||||||
expect(autofillService["makeScriptAction"]).toHaveBeenCalledWith(
|
expect(autofillService["makeScriptActionWithValue"]).toHaveBeenCalledWith(
|
||||||
fillScript,
|
fillScript,
|
||||||
options.cipher.identity,
|
options.cipher.identity.lastName,
|
||||||
expect.anything(),
|
lastNameField,
|
||||||
filledFields,
|
filledFields,
|
||||||
lastNameField.opid,
|
|
||||||
);
|
);
|
||||||
expect(value.script[2]).toStrictEqual(["fill_by_opid", firstNameField.opid, firstName]);
|
expect(value.script[2]).toStrictEqual(["fill_by_opid", firstNameField.opid, firstName]);
|
||||||
expect(value.script[5]).toStrictEqual(["fill_by_opid", middleNameField.opid, middleName]);
|
expect(value.script[5]).toStrictEqual([
|
||||||
|
"fill_by_opid",
|
||||||
|
middleNameField.opid,
|
||||||
|
middleName,
|
||||||
|
]);
|
||||||
expect(value.script[8]).toStrictEqual(["fill_by_opid", lastNameField.opid, lastName]);
|
expect(value.script[8]).toStrictEqual(["fill_by_opid", lastNameField.opid, lastName]);
|
||||||
});
|
});
|
||||||
|
|
||||||
it("will match title and email fields to the vault item identity value", () => {
|
it("will match title and email fields to the vault item identity value", async () => {
|
||||||
const titleField = createAutofillFieldMock({ opid: "title", htmlName: "title" });
|
const titleField = createAutofillFieldMock({ opid: "title", htmlName: "title" });
|
||||||
const emailField = createAutofillFieldMock({ opid: "email", htmlName: "email" });
|
const emailField = createAutofillFieldMock({ opid: "email", htmlName: "email" });
|
||||||
pageDetails.fields = [titleField, emailField];
|
pageDetails.fields = [titleField, emailField];
|
||||||
@ -3270,40 +3269,30 @@ describe("AutofillService", () => {
|
|||||||
options.cipher.identity.title = title;
|
options.cipher.identity.title = title;
|
||||||
options.cipher.identity.email = email;
|
options.cipher.identity.email = email;
|
||||||
|
|
||||||
const value = autofillService["generateIdentityFillScript"](
|
const value = await autofillService["generateIdentityFillScript"](
|
||||||
fillScript,
|
fillScript,
|
||||||
pageDetails,
|
pageDetails,
|
||||||
filledFields,
|
filledFields,
|
||||||
options,
|
options,
|
||||||
);
|
);
|
||||||
|
|
||||||
expect(AutofillService["isFieldMatch"]).toHaveBeenCalledWith(
|
expect(autofillService["makeScriptActionWithValue"]).toHaveBeenCalledWith(
|
||||||
titleField.htmlName,
|
|
||||||
IdentityAutoFillConstants.TitleFieldNames,
|
|
||||||
);
|
|
||||||
expect(AutofillService["isFieldMatch"]).toHaveBeenCalledWith(
|
|
||||||
emailField.htmlName,
|
|
||||||
IdentityAutoFillConstants.EmailFieldNames,
|
|
||||||
);
|
|
||||||
expect(autofillService["makeScriptAction"]).toHaveBeenCalledWith(
|
|
||||||
fillScript,
|
fillScript,
|
||||||
options.cipher.identity,
|
options.cipher.identity.title,
|
||||||
expect.anything(),
|
titleField,
|
||||||
filledFields,
|
filledFields,
|
||||||
titleField.opid,
|
|
||||||
);
|
);
|
||||||
expect(autofillService["makeScriptAction"]).toHaveBeenCalledWith(
|
expect(autofillService["makeScriptActionWithValue"]).toHaveBeenCalledWith(
|
||||||
fillScript,
|
fillScript,
|
||||||
options.cipher.identity,
|
options.cipher.identity.email,
|
||||||
expect.anything(),
|
emailField,
|
||||||
filledFields,
|
filledFields,
|
||||||
emailField.opid,
|
|
||||||
);
|
);
|
||||||
expect(value.script[2]).toStrictEqual(["fill_by_opid", titleField.opid, title]);
|
expect(value.script[2]).toStrictEqual(["fill_by_opid", titleField.opid, title]);
|
||||||
expect(value.script[5]).toStrictEqual(["fill_by_opid", emailField.opid, email]);
|
expect(value.script[5]).toStrictEqual(["fill_by_opid", emailField.opid, email]);
|
||||||
});
|
});
|
||||||
|
|
||||||
it("will match a full address field to the vault item identity values", () => {
|
it("will match a full address field to the vault item identity values", async () => {
|
||||||
const fullAddressField = createAutofillFieldMock({
|
const fullAddressField = createAutofillFieldMock({
|
||||||
opid: "fullAddress",
|
opid: "fullAddress",
|
||||||
htmlName: "address",
|
htmlName: "address",
|
||||||
@ -3316,18 +3305,13 @@ describe("AutofillService", () => {
|
|||||||
options.cipher.identity.address2 = address2;
|
options.cipher.identity.address2 = address2;
|
||||||
options.cipher.identity.address3 = address3;
|
options.cipher.identity.address3 = address3;
|
||||||
|
|
||||||
const value = autofillService["generateIdentityFillScript"](
|
const value = await autofillService["generateIdentityFillScript"](
|
||||||
fillScript,
|
fillScript,
|
||||||
pageDetails,
|
pageDetails,
|
||||||
filledFields,
|
filledFields,
|
||||||
options,
|
options,
|
||||||
);
|
);
|
||||||
|
|
||||||
expect(AutofillService["isFieldMatch"]).toHaveBeenCalledWith(
|
|
||||||
fullAddressField.htmlName,
|
|
||||||
IdentityAutoFillConstants.AddressFieldNames,
|
|
||||||
IdentityAutoFillConstants.AddressFieldNameValues,
|
|
||||||
);
|
|
||||||
expect(autofillService["makeScriptActionWithValue"]).toHaveBeenCalledWith(
|
expect(autofillService["makeScriptActionWithValue"]).toHaveBeenCalledWith(
|
||||||
fillScript,
|
fillScript,
|
||||||
`${address1}, ${address2}, ${address3}`,
|
`${address1}, ${address2}, ${address3}`,
|
||||||
@ -3341,10 +3325,19 @@ describe("AutofillService", () => {
|
|||||||
]);
|
]);
|
||||||
});
|
});
|
||||||
|
|
||||||
it("will match address1, address2, address3, postalCode, city, state, country, phone, username, and company fields to their corresponding vault item identity values", () => {
|
it("will match address1, address2, address3, postalCode, city, state, country, phone, username, and company fields to their corresponding vault item identity values", async () => {
|
||||||
const address1Field = createAutofillFieldMock({ opid: "address1", htmlName: "address-1" });
|
const address1Field = createAutofillFieldMock({
|
||||||
const address2Field = createAutofillFieldMock({ opid: "address2", htmlName: "address-2" });
|
opid: "address1",
|
||||||
const address3Field = createAutofillFieldMock({ opid: "address3", htmlName: "address-3" });
|
htmlName: "address-1",
|
||||||
|
});
|
||||||
|
const address2Field = createAutofillFieldMock({
|
||||||
|
opid: "address2",
|
||||||
|
htmlName: "address-2",
|
||||||
|
});
|
||||||
|
const address3Field = createAutofillFieldMock({
|
||||||
|
opid: "address3",
|
||||||
|
htmlName: "address-3",
|
||||||
|
});
|
||||||
const postalCodeField = createAutofillFieldMock({
|
const postalCodeField = createAutofillFieldMock({
|
||||||
opid: "postalCode",
|
opid: "postalCode",
|
||||||
htmlName: "postal-code",
|
htmlName: "postal-code",
|
||||||
@ -3353,7 +3346,10 @@ describe("AutofillService", () => {
|
|||||||
const stateField = createAutofillFieldMock({ opid: "state", htmlName: "state" });
|
const stateField = createAutofillFieldMock({ opid: "state", htmlName: "state" });
|
||||||
const countryField = createAutofillFieldMock({ opid: "country", htmlName: "country" });
|
const countryField = createAutofillFieldMock({ opid: "country", htmlName: "country" });
|
||||||
const phoneField = createAutofillFieldMock({ opid: "phone", htmlName: "phone" });
|
const phoneField = createAutofillFieldMock({ opid: "phone", htmlName: "phone" });
|
||||||
const usernameField = createAutofillFieldMock({ opid: "username", htmlName: "username" });
|
const usernameField = createAutofillFieldMock({
|
||||||
|
opid: "username",
|
||||||
|
htmlName: "username",
|
||||||
|
});
|
||||||
const companyField = createAutofillFieldMock({ opid: "company", htmlName: "company" });
|
const companyField = createAutofillFieldMock({ opid: "company", htmlName: "company" });
|
||||||
pageDetails.fields = [
|
pageDetails.fields = [
|
||||||
address1Field,
|
address1Field,
|
||||||
@ -3372,8 +3368,8 @@ describe("AutofillService", () => {
|
|||||||
const address3 = "P.O. Box 123";
|
const address3 = "P.O. Box 123";
|
||||||
const postalCode = "12345";
|
const postalCode = "12345";
|
||||||
const city = "City";
|
const city = "City";
|
||||||
const state = "State";
|
const state = "TX";
|
||||||
const country = "Country";
|
const country = "US";
|
||||||
const phone = "123-456-7890";
|
const phone = "123-456-7890";
|
||||||
const username = "username";
|
const username = "username";
|
||||||
const company = "Company";
|
const company = "Company";
|
||||||
@ -3388,73 +3384,32 @@ describe("AutofillService", () => {
|
|||||||
options.cipher.identity.username = username;
|
options.cipher.identity.username = username;
|
||||||
options.cipher.identity.company = company;
|
options.cipher.identity.company = company;
|
||||||
|
|
||||||
const value = autofillService["generateIdentityFillScript"](
|
const value = await autofillService["generateIdentityFillScript"](
|
||||||
fillScript,
|
fillScript,
|
||||||
pageDetails,
|
pageDetails,
|
||||||
filledFields,
|
filledFields,
|
||||||
options,
|
options,
|
||||||
);
|
);
|
||||||
|
|
||||||
expect(AutofillService["isFieldMatch"]).toHaveBeenCalledWith(
|
expect(value.script).toContainEqual(["fill_by_opid", address1Field.opid, address1]);
|
||||||
address1Field.htmlName,
|
expect(value.script).toContainEqual(["fill_by_opid", address2Field.opid, address2]);
|
||||||
IdentityAutoFillConstants.Address1FieldNames,
|
expect(value.script).toContainEqual(["fill_by_opid", address3Field.opid, address3]);
|
||||||
);
|
expect(value.script).toContainEqual(["fill_by_opid", postalCodeField.opid, postalCode]);
|
||||||
expect(AutofillService["isFieldMatch"]).toHaveBeenCalledWith(
|
expect(value.script).toContainEqual(["fill_by_opid", cityField.opid, city]);
|
||||||
address2Field.htmlName,
|
expect(value.script).toContainEqual(["fill_by_opid", stateField.opid, state]);
|
||||||
IdentityAutoFillConstants.Address2FieldNames,
|
expect(value.script).toContainEqual(["fill_by_opid", countryField.opid, country]);
|
||||||
);
|
expect(value.script).toContainEqual(["fill_by_opid", phoneField.opid, phone]);
|
||||||
expect(AutofillService["isFieldMatch"]).toHaveBeenCalledWith(
|
expect(value.script).toContainEqual(["fill_by_opid", usernameField.opid, username]);
|
||||||
address3Field.htmlName,
|
expect(value.script).toContainEqual(["fill_by_opid", companyField.opid, company]);
|
||||||
IdentityAutoFillConstants.Address3FieldNames,
|
|
||||||
);
|
|
||||||
expect(AutofillService["isFieldMatch"]).toHaveBeenCalledWith(
|
|
||||||
postalCodeField.htmlName,
|
|
||||||
IdentityAutoFillConstants.PostalCodeFieldNames,
|
|
||||||
);
|
|
||||||
expect(AutofillService["isFieldMatch"]).toHaveBeenCalledWith(
|
|
||||||
cityField.htmlName,
|
|
||||||
IdentityAutoFillConstants.CityFieldNames,
|
|
||||||
);
|
|
||||||
expect(AutofillService["isFieldMatch"]).toHaveBeenCalledWith(
|
|
||||||
stateField.htmlName,
|
|
||||||
IdentityAutoFillConstants.StateFieldNames,
|
|
||||||
);
|
|
||||||
expect(AutofillService["isFieldMatch"]).toHaveBeenCalledWith(
|
|
||||||
countryField.htmlName,
|
|
||||||
IdentityAutoFillConstants.CountryFieldNames,
|
|
||||||
);
|
|
||||||
expect(AutofillService["isFieldMatch"]).toHaveBeenCalledWith(
|
|
||||||
phoneField.htmlName,
|
|
||||||
IdentityAutoFillConstants.PhoneFieldNames,
|
|
||||||
);
|
|
||||||
expect(AutofillService["isFieldMatch"]).toHaveBeenCalledWith(
|
|
||||||
usernameField.htmlName,
|
|
||||||
IdentityAutoFillConstants.UserNameFieldNames,
|
|
||||||
);
|
|
||||||
expect(AutofillService["isFieldMatch"]).toHaveBeenCalledWith(
|
|
||||||
companyField.htmlName,
|
|
||||||
IdentityAutoFillConstants.CompanyFieldNames,
|
|
||||||
);
|
|
||||||
expect(autofillService["makeScriptAction"]).toHaveBeenCalled();
|
|
||||||
expect(value.script[2]).toStrictEqual(["fill_by_opid", address1Field.opid, address1]);
|
|
||||||
expect(value.script[5]).toStrictEqual(["fill_by_opid", address2Field.opid, address2]);
|
|
||||||
expect(value.script[8]).toStrictEqual(["fill_by_opid", address3Field.opid, address3]);
|
|
||||||
expect(value.script[11]).toStrictEqual(["fill_by_opid", cityField.opid, city]);
|
|
||||||
expect(value.script[14]).toStrictEqual(["fill_by_opid", postalCodeField.opid, postalCode]);
|
|
||||||
expect(value.script[17]).toStrictEqual(["fill_by_opid", companyField.opid, company]);
|
|
||||||
expect(value.script[20]).toStrictEqual(["fill_by_opid", phoneField.opid, phone]);
|
|
||||||
expect(value.script[23]).toStrictEqual(["fill_by_opid", usernameField.opid, username]);
|
|
||||||
expect(value.script[26]).toStrictEqual(["fill_by_opid", stateField.opid, state]);
|
|
||||||
expect(value.script[29]).toStrictEqual(["fill_by_opid", countryField.opid, country]);
|
|
||||||
});
|
});
|
||||||
|
|
||||||
it("will find the two character IsoState value for an identity cipher that contains the full name of a state", () => {
|
it("will find the two character IsoState value for an identity cipher that contains the full name of a state", async () => {
|
||||||
const stateField = createAutofillFieldMock({ opid: "state", htmlName: "state" });
|
const stateField = createAutofillFieldMock({ opid: "state", htmlName: "state" });
|
||||||
pageDetails.fields = [stateField];
|
pageDetails.fields = [stateField];
|
||||||
const state = "California";
|
const state = "California";
|
||||||
options.cipher.identity.state = state;
|
options.cipher.identity.state = state;
|
||||||
|
|
||||||
const value = autofillService["generateIdentityFillScript"](
|
const value = await autofillService["generateIdentityFillScript"](
|
||||||
fillScript,
|
fillScript,
|
||||||
pageDetails,
|
pageDetails,
|
||||||
filledFields,
|
filledFields,
|
||||||
@ -3470,13 +3425,13 @@ describe("AutofillService", () => {
|
|||||||
expect(value.script[2]).toStrictEqual(["fill_by_opid", stateField.opid, "CA"]);
|
expect(value.script[2]).toStrictEqual(["fill_by_opid", stateField.opid, "CA"]);
|
||||||
});
|
});
|
||||||
|
|
||||||
it("will find the two character IsoProvince value for an identity cipher that contains the full name of a province", () => {
|
it("will find the two character IsoProvince value for an identity cipher that contains the full name of a province", async () => {
|
||||||
const stateField = createAutofillFieldMock({ opid: "state", htmlName: "state" });
|
const stateField = createAutofillFieldMock({ opid: "state", htmlName: "state" });
|
||||||
pageDetails.fields = [stateField];
|
pageDetails.fields = [stateField];
|
||||||
const state = "Ontario";
|
const state = "Ontario";
|
||||||
options.cipher.identity.state = state;
|
options.cipher.identity.state = state;
|
||||||
|
|
||||||
const value = autofillService["generateIdentityFillScript"](
|
const value = await autofillService["generateIdentityFillScript"](
|
||||||
fillScript,
|
fillScript,
|
||||||
pageDetails,
|
pageDetails,
|
||||||
filledFields,
|
filledFields,
|
||||||
@ -3492,13 +3447,13 @@ describe("AutofillService", () => {
|
|||||||
expect(value.script[2]).toStrictEqual(["fill_by_opid", stateField.opid, "ON"]);
|
expect(value.script[2]).toStrictEqual(["fill_by_opid", stateField.opid, "ON"]);
|
||||||
});
|
});
|
||||||
|
|
||||||
it("will find the two character IsoCountry value for an identity cipher that contains the full name of a country", () => {
|
it("will find the two character IsoCountry value for an identity cipher that contains the full name of a country", async () => {
|
||||||
const countryField = createAutofillFieldMock({ opid: "country", htmlName: "country" });
|
const countryField = createAutofillFieldMock({ opid: "country", htmlName: "country" });
|
||||||
pageDetails.fields = [countryField];
|
pageDetails.fields = [countryField];
|
||||||
const country = "Somalia";
|
const country = "Somalia";
|
||||||
options.cipher.identity.country = country;
|
options.cipher.identity.country = country;
|
||||||
|
|
||||||
const value = autofillService["generateIdentityFillScript"](
|
const value = await autofillService["generateIdentityFillScript"](
|
||||||
fillScript,
|
fillScript,
|
||||||
pageDetails,
|
pageDetails,
|
||||||
filledFields,
|
filledFields,
|
||||||
@ -3514,6 +3469,8 @@ describe("AutofillService", () => {
|
|||||||
expect(value.script[2]).toStrictEqual(["fill_by_opid", countryField.opid, "SO"]);
|
expect(value.script[2]).toStrictEqual(["fill_by_opid", countryField.opid, "SO"]);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
describe("isExcludedType", () => {
|
describe("isExcludedType", () => {
|
||||||
|
@ -26,6 +26,7 @@ import { FieldType, CipherType } from "@bitwarden/common/vault/enums";
|
|||||||
import { CipherRepromptType } from "@bitwarden/common/vault/enums/cipher-reprompt-type";
|
import { CipherRepromptType } from "@bitwarden/common/vault/enums/cipher-reprompt-type";
|
||||||
import { CipherView } from "@bitwarden/common/vault/models/view/cipher.view";
|
import { CipherView } from "@bitwarden/common/vault/models/view/cipher.view";
|
||||||
import { FieldView } from "@bitwarden/common/vault/models/view/field.view";
|
import { FieldView } from "@bitwarden/common/vault/models/view/field.view";
|
||||||
|
import { IdentityView } from "@bitwarden/common/vault/models/view/identity.view";
|
||||||
|
|
||||||
import { BrowserApi } from "../../platform/browser/browser-api";
|
import { BrowserApi } from "../../platform/browser/browser-api";
|
||||||
import { ScriptInjectorService } from "../../platform/services/abstractions/script-injector.service";
|
import { ScriptInjectorService } from "../../platform/services/abstractions/script-injector.service";
|
||||||
@ -478,6 +479,12 @@ export default class AutofillService implements AutofillServiceInterface {
|
|||||||
return totpCode;
|
return totpCode;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Checks if the cipher requires password reprompt and opens the password reprompt popout if necessary.
|
||||||
|
*
|
||||||
|
* @param cipher - The cipher to autofill
|
||||||
|
* @param tab - The tab to autofill
|
||||||
|
*/
|
||||||
async isPasswordRepromptRequired(cipher: CipherView, tab: chrome.tabs.Tab): Promise<boolean> {
|
async isPasswordRepromptRequired(cipher: CipherView, tab: chrome.tabs.Tab): Promise<boolean> {
|
||||||
const userHasMasterPasswordAndKeyHash =
|
const userHasMasterPasswordAndKeyHash =
|
||||||
await this.userVerificationService.hasMasterPasswordAndMasterKeyHash();
|
await this.userVerificationService.hasMasterPasswordAndMasterKeyHash();
|
||||||
@ -654,7 +661,7 @@ export default class AutofillService implements AutofillServiceInterface {
|
|||||||
fillScript = this.generateCardFillScript(fillScript, pageDetails, filledFields, options);
|
fillScript = this.generateCardFillScript(fillScript, pageDetails, filledFields, options);
|
||||||
break;
|
break;
|
||||||
case CipherType.Identity:
|
case CipherType.Identity:
|
||||||
fillScript = this.generateIdentityFillScript(
|
fillScript = await this.generateIdentityFillScript(
|
||||||
fillScript,
|
fillScript,
|
||||||
pageDetails,
|
pageDetails,
|
||||||
filledFields,
|
filledFields,
|
||||||
@ -1243,12 +1250,16 @@ export default class AutofillService implements AutofillServiceInterface {
|
|||||||
* @returns {AutofillScript}
|
* @returns {AutofillScript}
|
||||||
* @private
|
* @private
|
||||||
*/
|
*/
|
||||||
private generateIdentityFillScript(
|
private async generateIdentityFillScript(
|
||||||
fillScript: AutofillScript,
|
fillScript: AutofillScript,
|
||||||
pageDetails: AutofillPageDetails,
|
pageDetails: AutofillPageDetails,
|
||||||
filledFields: { [id: string]: AutofillField },
|
filledFields: { [id: string]: AutofillField },
|
||||||
options: GenerateFillScriptOptions,
|
options: GenerateFillScriptOptions,
|
||||||
): AutofillScript {
|
): Promise<AutofillScript> {
|
||||||
|
if (await this.configService.getFeatureFlag(FeatureFlag.GenerateIdentityFillScriptRefactor)) {
|
||||||
|
return this._generateIdentityFillScript(fillScript, pageDetails, filledFields, options);
|
||||||
|
}
|
||||||
|
|
||||||
if (!options.cipher.identity) {
|
if (!options.cipher.identity) {
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
@ -1476,6 +1487,589 @@ export default class AutofillService implements AutofillServiceInterface {
|
|||||||
return fillScript;
|
return fillScript;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Generates the autofill script for the specified page details and identity cipher item.
|
||||||
|
*
|
||||||
|
* @param fillScript - Object to store autofill script, passed between method references
|
||||||
|
* @param pageDetails - The details of the page to autofill
|
||||||
|
* @param filledFields - The fields that have already been filled, passed between method references
|
||||||
|
* @param options - Contains data used to fill cipher items
|
||||||
|
*/
|
||||||
|
private _generateIdentityFillScript(
|
||||||
|
fillScript: AutofillScript,
|
||||||
|
pageDetails: AutofillPageDetails,
|
||||||
|
filledFields: { [id: string]: AutofillField },
|
||||||
|
options: GenerateFillScriptOptions,
|
||||||
|
): AutofillScript {
|
||||||
|
const identity = options.cipher.identity;
|
||||||
|
if (!identity) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (let fieldsIndex = 0; fieldsIndex < pageDetails.fields.length; fieldsIndex++) {
|
||||||
|
const field = pageDetails.fields[fieldsIndex];
|
||||||
|
if (this.excludeFieldFromIdentityFill(field)) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
const keywordsList = this.getIdentityAutofillFieldKeywords(field);
|
||||||
|
const keywordsCombined = keywordsList.join(",");
|
||||||
|
if (this.shouldMakeIdentityTitleFillScript(filledFields, keywordsCombined)) {
|
||||||
|
this.makeScriptActionWithValue(fillScript, identity.title, field, filledFields);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (this.shouldMakeIdentityNameFillScript(filledFields, keywordsList)) {
|
||||||
|
this.makeIdentityNameFillScript(fillScript, filledFields, field, identity);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (this.shouldMakeIdentityFirstNameFillScript(filledFields, keywordsCombined)) {
|
||||||
|
this.makeScriptActionWithValue(fillScript, identity.firstName, field, filledFields);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (this.shouldMakeIdentityMiddleNameFillScript(filledFields, keywordsCombined)) {
|
||||||
|
this.makeScriptActionWithValue(fillScript, identity.middleName, field, filledFields);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (this.shouldMakeIdentityLastNameFillScript(filledFields, keywordsCombined)) {
|
||||||
|
this.makeScriptActionWithValue(fillScript, identity.lastName, field, filledFields);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (this.shouldMakeIdentityEmailFillScript(filledFields, keywordsCombined)) {
|
||||||
|
this.makeScriptActionWithValue(fillScript, identity.email, field, filledFields);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (this.shouldMakeIdentityAddressFillScript(filledFields, keywordsList)) {
|
||||||
|
this.makeIdentityAddressFillScript(fillScript, filledFields, field, identity);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (this.shouldMakeIdentityAddress1FillScript(filledFields, keywordsCombined)) {
|
||||||
|
this.makeScriptActionWithValue(fillScript, identity.address1, field, filledFields);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (this.shouldMakeIdentityAddress2FillScript(filledFields, keywordsCombined)) {
|
||||||
|
this.makeScriptActionWithValue(fillScript, identity.address2, field, filledFields);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (this.shouldMakeIdentityAddress3FillScript(filledFields, keywordsCombined)) {
|
||||||
|
this.makeScriptActionWithValue(fillScript, identity.address3, field, filledFields);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (this.shouldMakeIdentityPostalCodeFillScript(filledFields, keywordsCombined)) {
|
||||||
|
this.makeScriptActionWithValue(fillScript, identity.postalCode, field, filledFields);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (this.shouldMakeIdentityCityFillScript(filledFields, keywordsCombined)) {
|
||||||
|
this.makeScriptActionWithValue(fillScript, identity.city, field, filledFields);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (this.shouldMakeIdentityStateFillScript(filledFields, keywordsCombined)) {
|
||||||
|
this.makeIdentityStateFillScript(fillScript, filledFields, field, identity);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (this.shouldMakeIdentityCountryFillScript(filledFields, keywordsCombined)) {
|
||||||
|
this.makeIdentityCountryFillScript(fillScript, filledFields, field, identity);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (this.shouldMakeIdentityPhoneFillScript(filledFields, keywordsCombined)) {
|
||||||
|
this.makeScriptActionWithValue(fillScript, identity.phone, field, filledFields);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (this.shouldMakeIdentityUserNameFillScript(filledFields, keywordsCombined)) {
|
||||||
|
this.makeScriptActionWithValue(fillScript, identity.username, field, filledFields);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (this.shouldMakeIdentityCompanyFillScript(filledFields, keywordsCombined)) {
|
||||||
|
this.makeScriptActionWithValue(fillScript, identity.company, field, filledFields);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return fillScript;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Identifies if the current field should be excluded from triggering autofill of the identity cipher.
|
||||||
|
*
|
||||||
|
* @param field - The field to check
|
||||||
|
*/
|
||||||
|
private excludeFieldFromIdentityFill(field: AutofillField): boolean {
|
||||||
|
return (
|
||||||
|
AutofillService.isExcludedFieldType(field, AutoFillConstants.ExcludedAutofillTypes) ||
|
||||||
|
AutoFillConstants.ExcludedIdentityAutocompleteTypes.has(field.autoCompleteType) ||
|
||||||
|
!field.viewable
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Gathers all unique keyword identifiers from a field that can be used to determine what
|
||||||
|
* identity value should be filled.
|
||||||
|
*
|
||||||
|
* @param field - The field to gather keywords from
|
||||||
|
*/
|
||||||
|
private getIdentityAutofillFieldKeywords(field: AutofillField): string[] {
|
||||||
|
const keywords: Set<string> = new Set();
|
||||||
|
for (let index = 0; index < IdentityAutoFillConstants.IdentityAttributes.length; index++) {
|
||||||
|
const attribute = IdentityAutoFillConstants.IdentityAttributes[index];
|
||||||
|
if (field[attribute]) {
|
||||||
|
keywords.add(
|
||||||
|
field[attribute]
|
||||||
|
.trim()
|
||||||
|
.toLowerCase()
|
||||||
|
.replace(/[^a-zA-Z0-9]+/g, ""),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return Array.from(keywords);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Identifies if a fill script action for the identity title
|
||||||
|
* field should be created for the provided field.
|
||||||
|
*
|
||||||
|
* @param filledFields - The fields that have already been filled
|
||||||
|
* @param keywords - The keywords from the field
|
||||||
|
*/
|
||||||
|
private shouldMakeIdentityTitleFillScript(
|
||||||
|
filledFields: Record<string, AutofillField>,
|
||||||
|
keywords: string,
|
||||||
|
): boolean {
|
||||||
|
return (
|
||||||
|
!filledFields.title &&
|
||||||
|
AutofillService.isFieldMatch(keywords, IdentityAutoFillConstants.TitleFieldNames)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Identifies if a fill script action for the identity name
|
||||||
|
* field should be created for the provided field.
|
||||||
|
*
|
||||||
|
* @param filledFields - The fields that have already been filled
|
||||||
|
* @param keywords - The keywords from the field
|
||||||
|
*/
|
||||||
|
private shouldMakeIdentityNameFillScript(
|
||||||
|
filledFields: Record<string, AutofillField>,
|
||||||
|
keywords: string[],
|
||||||
|
): boolean {
|
||||||
|
return (
|
||||||
|
!filledFields.name &&
|
||||||
|
keywords.some((keyword) =>
|
||||||
|
AutofillService.isFieldMatch(
|
||||||
|
keyword,
|
||||||
|
IdentityAutoFillConstants.FullNameFieldNames,
|
||||||
|
IdentityAutoFillConstants.FullNameFieldNameValues,
|
||||||
|
),
|
||||||
|
)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Identifies if a fill script action for the identity first name
|
||||||
|
* field should be created for the provided field.
|
||||||
|
*
|
||||||
|
* @param filledFields - The fields that have already been filled
|
||||||
|
* @param keywords - The keywords from the field
|
||||||
|
*/
|
||||||
|
private shouldMakeIdentityFirstNameFillScript(
|
||||||
|
filledFields: Record<string, AutofillField>,
|
||||||
|
keywords: string,
|
||||||
|
): boolean {
|
||||||
|
return (
|
||||||
|
!filledFields.firstName &&
|
||||||
|
AutofillService.isFieldMatch(keywords, IdentityAutoFillConstants.FirstnameFieldNames)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Identifies if a fill script action for the identity middle name
|
||||||
|
* field should be created for the provided field.
|
||||||
|
*
|
||||||
|
* @param filledFields - The fields that have already been filled
|
||||||
|
* @param keywords - The keywords from the field
|
||||||
|
*/
|
||||||
|
private shouldMakeIdentityMiddleNameFillScript(
|
||||||
|
filledFields: Record<string, AutofillField>,
|
||||||
|
keywords: string,
|
||||||
|
): boolean {
|
||||||
|
return (
|
||||||
|
!filledFields.middleName &&
|
||||||
|
AutofillService.isFieldMatch(keywords, IdentityAutoFillConstants.MiddlenameFieldNames)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Identifies if a fill script action for the identity last name
|
||||||
|
* field should be created for the provided field.
|
||||||
|
*
|
||||||
|
* @param filledFields - The fields that have already been filled
|
||||||
|
* @param keywords - The keywords from the field
|
||||||
|
*/
|
||||||
|
private shouldMakeIdentityLastNameFillScript(
|
||||||
|
filledFields: Record<string, AutofillField>,
|
||||||
|
keywords: string,
|
||||||
|
): boolean {
|
||||||
|
return (
|
||||||
|
!filledFields.lastName &&
|
||||||
|
AutofillService.isFieldMatch(keywords, IdentityAutoFillConstants.LastnameFieldNames)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Identifies if a fill script action for the identity email
|
||||||
|
* field should be created for the provided field.
|
||||||
|
*
|
||||||
|
* @param filledFields - The fields that have already been filled
|
||||||
|
* @param keywords - The keywords from the field
|
||||||
|
*/
|
||||||
|
private shouldMakeIdentityEmailFillScript(
|
||||||
|
filledFields: Record<string, AutofillField>,
|
||||||
|
keywords: string,
|
||||||
|
): boolean {
|
||||||
|
return (
|
||||||
|
!filledFields.email &&
|
||||||
|
AutofillService.isFieldMatch(keywords, IdentityAutoFillConstants.EmailFieldNames)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Identifies if a fill script action for the identity address
|
||||||
|
* field should be created for the provided field.
|
||||||
|
*
|
||||||
|
* @param filledFields - The fields that have already been filled
|
||||||
|
* @param keywords - The keywords from the field
|
||||||
|
*/
|
||||||
|
private shouldMakeIdentityAddressFillScript(
|
||||||
|
filledFields: Record<string, AutofillField>,
|
||||||
|
keywords: string[],
|
||||||
|
): boolean {
|
||||||
|
return (
|
||||||
|
!filledFields.address &&
|
||||||
|
keywords.some((keyword) =>
|
||||||
|
AutofillService.isFieldMatch(
|
||||||
|
keyword,
|
||||||
|
IdentityAutoFillConstants.AddressFieldNames,
|
||||||
|
IdentityAutoFillConstants.AddressFieldNameValues,
|
||||||
|
),
|
||||||
|
)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Identifies if a fill script action for the identity address1
|
||||||
|
* field should be created for the provided field.
|
||||||
|
*
|
||||||
|
* @param filledFields - The fields that have already been filled
|
||||||
|
* @param keywords - The keywords from the field
|
||||||
|
*/
|
||||||
|
private shouldMakeIdentityAddress1FillScript(
|
||||||
|
filledFields: Record<string, AutofillField>,
|
||||||
|
keywords: string,
|
||||||
|
): boolean {
|
||||||
|
return (
|
||||||
|
!filledFields.address1 &&
|
||||||
|
AutofillService.isFieldMatch(keywords, IdentityAutoFillConstants.Address1FieldNames)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Identifies if a fill script action for the identity address2
|
||||||
|
* field should be created for the provided field.
|
||||||
|
*
|
||||||
|
* @param filledFields - The fields that have already been filled
|
||||||
|
* @param keywords - The keywords from the field
|
||||||
|
*/
|
||||||
|
private shouldMakeIdentityAddress2FillScript(
|
||||||
|
filledFields: Record<string, AutofillField>,
|
||||||
|
keywords: string,
|
||||||
|
): boolean {
|
||||||
|
return (
|
||||||
|
!filledFields.address2 &&
|
||||||
|
AutofillService.isFieldMatch(keywords, IdentityAutoFillConstants.Address2FieldNames)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Identifies if a fill script action for the identity address3
|
||||||
|
* field should be created for the provided field.
|
||||||
|
*
|
||||||
|
* @param filledFields - The fields that have already been filled
|
||||||
|
* @param keywords - The keywords from the field
|
||||||
|
*/
|
||||||
|
private shouldMakeIdentityAddress3FillScript(
|
||||||
|
filledFields: Record<string, AutofillField>,
|
||||||
|
keywords: string,
|
||||||
|
): boolean {
|
||||||
|
return (
|
||||||
|
!filledFields.address3 &&
|
||||||
|
AutofillService.isFieldMatch(keywords, IdentityAutoFillConstants.Address3FieldNames)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Identifies if a fill script action for the identity postal code
|
||||||
|
* field should be created for the provided field.
|
||||||
|
*
|
||||||
|
* @param filledFields - The fields that have already been filled
|
||||||
|
* @param keywords - The keywords from the field
|
||||||
|
*/
|
||||||
|
private shouldMakeIdentityPostalCodeFillScript(
|
||||||
|
filledFields: Record<string, AutofillField>,
|
||||||
|
keywords: string,
|
||||||
|
): boolean {
|
||||||
|
return (
|
||||||
|
!filledFields.postalCode &&
|
||||||
|
AutofillService.isFieldMatch(keywords, IdentityAutoFillConstants.PostalCodeFieldNames)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Identifies if a fill script action for the identity city
|
||||||
|
* field should be created for the provided field.
|
||||||
|
*
|
||||||
|
* @param filledFields - The fields that have already been filled
|
||||||
|
* @param keywords - The keywords from the field
|
||||||
|
*/
|
||||||
|
private shouldMakeIdentityCityFillScript(
|
||||||
|
filledFields: Record<string, AutofillField>,
|
||||||
|
keywords: string,
|
||||||
|
): boolean {
|
||||||
|
return (
|
||||||
|
!filledFields.city &&
|
||||||
|
AutofillService.isFieldMatch(keywords, IdentityAutoFillConstants.CityFieldNames)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Identifies if a fill script action for the identity state
|
||||||
|
* field should be created for the provided field.
|
||||||
|
*
|
||||||
|
* @param filledFields - The fields that have already been filled
|
||||||
|
* @param keywords - The keywords from the field
|
||||||
|
*/
|
||||||
|
private shouldMakeIdentityStateFillScript(
|
||||||
|
filledFields: Record<string, AutofillField>,
|
||||||
|
keywords: string,
|
||||||
|
): boolean {
|
||||||
|
return (
|
||||||
|
!filledFields.state &&
|
||||||
|
AutofillService.isFieldMatch(keywords, IdentityAutoFillConstants.StateFieldNames)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Identifies if a fill script action for the identity country
|
||||||
|
* field should be created for the provided field.
|
||||||
|
*
|
||||||
|
* @param filledFields - The fields that have already been filled
|
||||||
|
* @param keywords - The keywords from the field
|
||||||
|
*/
|
||||||
|
private shouldMakeIdentityCountryFillScript(
|
||||||
|
filledFields: Record<string, AutofillField>,
|
||||||
|
keywords: string,
|
||||||
|
): boolean {
|
||||||
|
return (
|
||||||
|
!filledFields.country &&
|
||||||
|
AutofillService.isFieldMatch(keywords, IdentityAutoFillConstants.CountryFieldNames)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Identifies if a fill script action for the identity phone
|
||||||
|
* field should be created for the provided field.
|
||||||
|
*
|
||||||
|
* @param filledFields - The fields that have already been filled
|
||||||
|
* @param keywords - The keywords from the field
|
||||||
|
*/
|
||||||
|
private shouldMakeIdentityPhoneFillScript(
|
||||||
|
filledFields: Record<string, AutofillField>,
|
||||||
|
keywords: string,
|
||||||
|
): boolean {
|
||||||
|
return (
|
||||||
|
!filledFields.phone &&
|
||||||
|
AutofillService.isFieldMatch(keywords, IdentityAutoFillConstants.PhoneFieldNames)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Identifies if a fill script action for the identity username
|
||||||
|
* field should be created for the provided field.
|
||||||
|
*
|
||||||
|
* @param filledFields - The fields that have already been filled
|
||||||
|
* @param keywords - The keywords from the field
|
||||||
|
*/
|
||||||
|
private shouldMakeIdentityUserNameFillScript(
|
||||||
|
filledFields: Record<string, AutofillField>,
|
||||||
|
keywords: string,
|
||||||
|
): boolean {
|
||||||
|
return (
|
||||||
|
!filledFields.username &&
|
||||||
|
AutofillService.isFieldMatch(keywords, IdentityAutoFillConstants.UserNameFieldNames)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Identifies if a fill script action for the identity company
|
||||||
|
* field should be created for the provided field.
|
||||||
|
*
|
||||||
|
* @param filledFields - The fields that have already been filled
|
||||||
|
* @param keywords - The keywords from the field
|
||||||
|
*/
|
||||||
|
private shouldMakeIdentityCompanyFillScript(
|
||||||
|
filledFields: Record<string, AutofillField>,
|
||||||
|
keywords: string,
|
||||||
|
): boolean {
|
||||||
|
return (
|
||||||
|
!filledFields.company &&
|
||||||
|
AutofillService.isFieldMatch(keywords, IdentityAutoFillConstants.CompanyFieldNames)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Creates an identity name fill script action for the provided field. This is used
|
||||||
|
* when filling a `full name` field, using the first, middle, and last name from the
|
||||||
|
* identity cipher item.
|
||||||
|
*
|
||||||
|
* @param fillScript - The autofill script to add the action to
|
||||||
|
* @param filledFields - The fields that have already been filled
|
||||||
|
* @param field - The field to fill
|
||||||
|
* @param identity - The identity cipher item
|
||||||
|
*/
|
||||||
|
private makeIdentityNameFillScript(
|
||||||
|
fillScript: AutofillScript,
|
||||||
|
filledFields: Record<string, AutofillField>,
|
||||||
|
field: AutofillField,
|
||||||
|
identity: IdentityView,
|
||||||
|
) {
|
||||||
|
let name = "";
|
||||||
|
if (identity.firstName) {
|
||||||
|
name += identity.firstName;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (identity.middleName) {
|
||||||
|
name += !name ? identity.middleName : ` ${identity.middleName}`;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (identity.lastName) {
|
||||||
|
name += !name ? identity.lastName : ` ${identity.lastName}`;
|
||||||
|
}
|
||||||
|
|
||||||
|
this.makeScriptActionWithValue(fillScript, name, field, filledFields);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Creates an identity address fill script action for the provided field. This is used
|
||||||
|
* when filling a generic `address` field, using the address1, address2, and address3
|
||||||
|
* from the identity cipher item.
|
||||||
|
*
|
||||||
|
* @param fillScript - The autofill script to add the action to
|
||||||
|
* @param filledFields - The fields that have already been filled
|
||||||
|
* @param field - The field to fill
|
||||||
|
* @param identity - The identity cipher item
|
||||||
|
*/
|
||||||
|
private makeIdentityAddressFillScript(
|
||||||
|
fillScript: AutofillScript,
|
||||||
|
filledFields: Record<string, AutofillField>,
|
||||||
|
field: AutofillField,
|
||||||
|
identity: IdentityView,
|
||||||
|
) {
|
||||||
|
if (!identity.address1) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
let address = identity.address1;
|
||||||
|
|
||||||
|
if (identity.address2) {
|
||||||
|
address += `, ${identity.address2}`;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (identity.address3) {
|
||||||
|
address += `, ${identity.address3}`;
|
||||||
|
}
|
||||||
|
|
||||||
|
this.makeScriptActionWithValue(fillScript, address, field, filledFields);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Creates an identity state fill script action for the provided field. This is used
|
||||||
|
* when filling a `state` field, using the state value from the identity cipher item.
|
||||||
|
* If the state value is a full name, it will be converted to an ISO code.
|
||||||
|
*
|
||||||
|
* @param fillScript - The autofill script to add the action to
|
||||||
|
* @param filledFields - The fields that have already been filled
|
||||||
|
* @param field - The field to fill
|
||||||
|
* @param identity - The identity cipher item
|
||||||
|
*/
|
||||||
|
private makeIdentityStateFillScript(
|
||||||
|
fillScript: AutofillScript,
|
||||||
|
filledFields: Record<string, AutofillField>,
|
||||||
|
field: AutofillField,
|
||||||
|
identity: IdentityView,
|
||||||
|
) {
|
||||||
|
if (!identity.state) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (identity.state.length <= 2) {
|
||||||
|
this.makeScriptActionWithValue(fillScript, identity.state, field, filledFields);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const stateLower = identity.state.toLowerCase();
|
||||||
|
const isoState =
|
||||||
|
IdentityAutoFillConstants.IsoStates[stateLower] ||
|
||||||
|
IdentityAutoFillConstants.IsoProvinces[stateLower];
|
||||||
|
if (isoState) {
|
||||||
|
this.makeScriptActionWithValue(fillScript, isoState, field, filledFields);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Creates an identity country fill script action for the provided field. This is used
|
||||||
|
* when filling a `country` field, using the country value from the identity cipher item.
|
||||||
|
* If the country value is a full name, it will be converted to an ISO code.
|
||||||
|
*
|
||||||
|
* @param fillScript - The autofill script to add the action to
|
||||||
|
* @param filledFields - The fields that have already been filled
|
||||||
|
* @param field - The field to fill
|
||||||
|
* @param identity - The identity cipher item
|
||||||
|
*/
|
||||||
|
private makeIdentityCountryFillScript(
|
||||||
|
fillScript: AutofillScript,
|
||||||
|
filledFields: Record<string, AutofillField>,
|
||||||
|
field: AutofillField,
|
||||||
|
identity: IdentityView,
|
||||||
|
) {
|
||||||
|
if (!identity.country) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (identity.country.length <= 2) {
|
||||||
|
this.makeScriptActionWithValue(fillScript, identity.country, field, filledFields);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const countryLower = identity.country.toLowerCase();
|
||||||
|
const isoCountry = IdentityAutoFillConstants.IsoCountries[countryLower];
|
||||||
|
if (isoCountry) {
|
||||||
|
this.makeScriptActionWithValue(fillScript, isoCountry, field, filledFields);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Accepts an HTMLInputElement type value and a list of
|
* Accepts an HTMLInputElement type value and a list of
|
||||||
* excluded types and returns true if the type is excluded.
|
* excluded types and returns true if the type is excluded.
|
||||||
|
@ -1057,7 +1057,7 @@ export class InlineMenuFieldQualificationService
|
|||||||
returnStringValue: boolean,
|
returnStringValue: boolean,
|
||||||
) {
|
) {
|
||||||
if (!this.autofillFieldKeywordsMap.has(autofillFieldData)) {
|
if (!this.autofillFieldKeywordsMap.has(autofillFieldData)) {
|
||||||
const keywords = [
|
const keywordsSet = new Set<string>([
|
||||||
autofillFieldData.htmlID,
|
autofillFieldData.htmlID,
|
||||||
autofillFieldData.htmlName,
|
autofillFieldData.htmlName,
|
||||||
autofillFieldData.htmlClass,
|
autofillFieldData.htmlClass,
|
||||||
@ -1071,9 +1071,8 @@ export class InlineMenuFieldQualificationService
|
|||||||
autofillFieldData["label-right"],
|
autofillFieldData["label-right"],
|
||||||
autofillFieldData["label-tag"],
|
autofillFieldData["label-tag"],
|
||||||
autofillFieldData["label-top"],
|
autofillFieldData["label-top"],
|
||||||
];
|
]);
|
||||||
const keywordsSet = new Set<string>(keywords);
|
const stringValue = Array.from(keywordsSet).join(",").toLowerCase();
|
||||||
const stringValue = keywords.join(",").toLowerCase();
|
|
||||||
this.autofillFieldKeywordsMap.set(autofillFieldData, { keywordsSet, stringValue });
|
this.autofillFieldKeywordsMap.set(autofillFieldData, { keywordsSet, stringValue });
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -29,6 +29,7 @@ export enum FeatureFlag {
|
|||||||
AuthenticatorTwoFactorToken = "authenticator-2fa-token",
|
AuthenticatorTwoFactorToken = "authenticator-2fa-token",
|
||||||
UnauthenticatedExtensionUIRefresh = "unauth-ui-refresh",
|
UnauthenticatedExtensionUIRefresh = "unauth-ui-refresh",
|
||||||
EnableUpgradePasswordManagerSub = "AC-2708-upgrade-password-manager-sub",
|
EnableUpgradePasswordManagerSub = "AC-2708-upgrade-password-manager-sub",
|
||||||
|
GenerateIdentityFillScriptRefactor = "generate-identity-fill-script-refactor",
|
||||||
}
|
}
|
||||||
|
|
||||||
export type AllowedFeatureFlagTypes = boolean | number | string;
|
export type AllowedFeatureFlagTypes = boolean | number | string;
|
||||||
@ -68,6 +69,7 @@ export const DefaultFeatureFlagValue = {
|
|||||||
[FeatureFlag.AuthenticatorTwoFactorToken]: FALSE,
|
[FeatureFlag.AuthenticatorTwoFactorToken]: FALSE,
|
||||||
[FeatureFlag.UnauthenticatedExtensionUIRefresh]: FALSE,
|
[FeatureFlag.UnauthenticatedExtensionUIRefresh]: FALSE,
|
||||||
[FeatureFlag.EnableUpgradePasswordManagerSub]: FALSE,
|
[FeatureFlag.EnableUpgradePasswordManagerSub]: FALSE,
|
||||||
|
[FeatureFlag.GenerateIdentityFillScriptRefactor]: FALSE,
|
||||||
} satisfies Record<FeatureFlag, AllowedFeatureFlagTypes>;
|
} satisfies Record<FeatureFlag, AllowedFeatureFlagTypes>;
|
||||||
|
|
||||||
export type DefaultFeatureFlagValueType = typeof DefaultFeatureFlagValue;
|
export type DefaultFeatureFlagValueType = typeof DefaultFeatureFlagValue;
|
||||||
|
Loading…
Reference in New Issue
Block a user