From ca3755f2238881ddb22998af747a544635640a53 Mon Sep 17 00:00:00 2001
From: Emanuele Feliziani
Date: Wed, 12 Jul 2023 10:21:00 +0200
Subject: [PATCH 1/6] Minor accuracy improvements
Signed-off-by: Emanuele Feliziani
---
src/Form/FormAnalyzer.js | 10 +++++++---
src/autofill-utils.js | 1 +
2 files changed, 8 insertions(+), 3 deletions(-)
diff --git a/src/Form/FormAnalyzer.js b/src/Form/FormAnalyzer.js
index 504aacb92..b643e508a 100644
--- a/src/Form/FormAnalyzer.js
+++ b/src/Form/FormAnalyzer.js
@@ -9,6 +9,8 @@ const signupRegex = new RegExp(
)
const conservativeSignupRegex = new RegExp(/sign.?up|join|register|enroll|newsletter|subscri(be|ption)|settings|preferences|profile|update/i)
const strictSignupRegex = new RegExp(/sign.?up|join|register|(create|new).+account|enroll|settings|preferences|profile|update/i)
+const resetPasswordLink = new RegExp(/(forgot(ten)?|reset|don't remember) (your )?password|password forgotten/i)
+const loginProvidersRegex = new RegExp(/ with /i)
class FormAnalyzer {
/** @type HTMLElement */
@@ -124,7 +126,7 @@ class FormAnalyzer {
shouldCheckUnifiedForm = false,
shouldBeConservative = false
}) {
- const matchesLogin = /current.?password/i.test(string) || loginRegex.test(string)
+ const matchesLogin = /current.?password/i.test(string) || loginRegex.test(string) || resetPasswordLink.test(string)
// Check explicitly for unified login/signup forms
if (shouldCheckUnifiedForm && matchesLogin && strictSignupRegex.test(string)) {
@@ -230,9 +232,11 @@ class FormAnalyzer {
(el.getAttribute('role') || '').toUpperCase() === 'LINK' ||
el.matches('button[class*=secondary]')
) {
- // Unless it's a forgotten password link, we don't flip those links
let shouldFlip = true
- if (/(forgot(ten)?|reset) (your )?password|password forgotten| with /i.test(string)) {
+ if (
+ resetPasswordLink.test(string) || // Don't flip forgotten password links
+ loginProvidersRegex.test(string) // Don't flip login providers links
+ ) {
shouldFlip = false
}
this.updateSignal({string, strength: 1, signalType: `external link: ${string}`, shouldFlip})
diff --git a/src/autofill-utils.js b/src/autofill-utils.js
index 608ace6b2..c33106960 100644
--- a/src/autofill-utils.js
+++ b/src/autofill-utils.js
@@ -326,6 +326,7 @@ const isLikelyASubmitButton = (el) => {
return (
el.getAttribute('type') === 'submit' || // is explicitly set as "submit"
+ el.getAttribute('name') === 'submit' || // is called "submit"
/primary|submit/i.test(el.className) || // has high-signal submit classes
/submit/i.test(dataTestId) ||
SUBMIT_BUTTON_REGEX.test(contentExcludingLabel) || // has high-signal text
From d3aba2dda2bf0811eda24552f1082708bb9af24f Mon Sep 17 00:00:00 2001
From: Emanuele Feliziani
Date: Wed, 12 Jul 2023 10:21:46 +0200
Subject: [PATCH 2/6] Use getters for formAnalyzer results
Signed-off-by: Emanuele Feliziani
---
src/Form/Form.js | 15 ++++++++++-----
1 file changed, 10 insertions(+), 5 deletions(-)
diff --git a/src/Form/Form.js b/src/Form/Form.js
index 7759fa0f3..5c66a0ffb 100644
--- a/src/Form/Form.js
+++ b/src/Form/Form.js
@@ -32,8 +32,6 @@ class Form {
form;
/** @type {HTMLInputElement | null} */
activeInput;
- /** @type {boolean | null} */
- isSignup;
/**
* @param {HTMLElement} form
* @param {HTMLInputElement|HTMLSelectElement} input
@@ -45,9 +43,6 @@ class Form {
this.form = form
this.matching = matching || createMatching()
this.formAnalyzer = new FormAnalyzer(form, input, matching)
- this.isLogin = this.formAnalyzer.isLogin
- this.isSignup = this.formAnalyzer.isSignup
- this.isHybrid = this.formAnalyzer.isHybrid
this.device = deviceInterface
/** @type Record<'all' | SupportedMainTypes, Set> */
@@ -94,6 +89,16 @@ class Form {
}
}
+ get isLogin () {
+ return this.formAnalyzer.isLogin
+ }
+ get isSignup () {
+ return this.formAnalyzer.isSignup
+ }
+ get isHybrid () {
+ return this.formAnalyzer.isHybrid
+ }
+
logFormInfo () {
if (!shouldLog()) return
From 2ff48798bb0ec4c1160392cb48ba20fefc23c0e1 Mon Sep 17 00:00:00 2001
From: Emanuele Feliziani
Date: Wed, 12 Jul 2023 11:41:30 +0200
Subject: [PATCH 3/6] Add mutation observer to forms
Signed-off-by: Emanuele Feliziani
---
dist/autofill-debug.js | 63 +++++--
dist/autofill.js | 63 +++++--
integration-test/helpers/mocks.js | 1 +
integration-test/helpers/pages.js | 42 ++++-
integration-test/pages/mutating-form.html | 161 ++++++++++++++++++
.../tests/mutating-form.macos.spec.js | 36 ++++
src/Form/Form.js | 32 +++-
.../Resources/assets/autofill-debug.js | 63 +++++--
swift-package/Resources/assets/autofill.js | 63 +++++--
9 files changed, 469 insertions(+), 55 deletions(-)
create mode 100644 integration-test/pages/mutating-form.html
create mode 100644 integration-test/tests/mutating-form.macos.spec.js
diff --git a/dist/autofill-debug.js b/dist/autofill-debug.js
index 11b484cfe..ef967f0d4 100644
--- a/dist/autofill-debug.js
+++ b/dist/autofill-debug.js
@@ -9990,8 +9990,6 @@ class Form {
/** @type {HTMLInputElement | null} */
- /** @type {boolean | null} */
-
/**
* @param {HTMLElement} form
* @param {HTMLInputElement|HTMLSelectElement} input
@@ -10008,14 +10006,9 @@ class Form {
_defineProperty(this, "activeInput", void 0);
- _defineProperty(this, "isSignup", void 0);
-
this.form = form;
this.matching = matching || (0, _matching.createMatching)();
this.formAnalyzer = new _FormAnalyzer.default(form, input, matching);
- this.isLogin = this.formAnalyzer.isLogin;
- this.isSignup = this.formAnalyzer.isSignup;
- this.isHybrid = this.formAnalyzer.isHybrid;
this.device = deviceInterface;
/** @type Record<'all' | SupportedMainTypes, Set> */
@@ -10042,6 +10035,24 @@ class Form {
for (const entry of entries) {
if (!entry.isIntersecting) this.removeTooltip();
}
+ });
+ this.mutObsConfig = {
+ childList: true,
+ subtree: true
+ };
+ this.mutObs = new MutationObserver(records => {
+ const anythingRemoved = records.some(record => record.removedNodes.length > 0);
+
+ if (anythingRemoved) {
+ // Must check for inputs because a parent may be removed and not show up in record.removedNodes
+ if ([...this.inputs.all].some(input => !input.isConnected)) {
+ // If any known input has been removed from the DOM, reanalyze the whole form
+ window.requestIdleCallback(() => {
+ this.formAnalyzer = new _FormAnalyzer.default(this.form, input, this.matching);
+ this.recategorizeAllInputs();
+ });
+ }
+ }
}); // This ensures we fire the handler again if the form is changed
this.addListener(form, 'input', () => {
@@ -10051,6 +10062,7 @@ class Form {
}
});
this.categorizeInputs();
+ this.mutObs.observe(this.form, this.mutObsConfig);
this.logFormInfo();
if (shouldAutoprompt) {
@@ -10058,6 +10070,18 @@ class Form {
}
}
+ get isLogin() {
+ return this.formAnalyzer.isLogin;
+ }
+
+ get isSignup() {
+ return this.formAnalyzer.isSignup;
+ }
+
+ get isHybrid() {
+ return this.formAnalyzer.isHybrid;
+ }
+
logFormInfo() {
if (!(0, _autofillUtils.shouldLog)()) return;
console.log("Form type: %c".concat(this.getFormType()), 'font-weight: bold');
@@ -10289,6 +10313,7 @@ class Form {
recategorizeAllInputs() {
+ this.initialScanComplete = false;
this.removeAllDecorations();
this.forgetAllInputs();
this.categorizeInputs();
@@ -10311,6 +10336,7 @@ class Form {
destroy() {
this.removeAllDecorations();
this.removeTooltip();
+ this.mutObs.disconnect();
this.matching.clear();
this.intObs = null;
}
@@ -10323,6 +10349,8 @@ class Form {
} else {
this.form.querySelectorAll(selector).forEach(input => this.addInput(input));
}
+
+ this.initialScanComplete = true;
}
get submitButtons() {
@@ -10378,9 +10406,16 @@ class Form {
addInput(input) {
var _this$device$settings;
- // Nothing to do with 1-character fields
+ if (this.inputs.all.has(input)) return this; // When new inputs are added after the initial scan, reanalyze the whole form
+
+ if (this.initialScanComplete) {
+ this.formAnalyzer = new _FormAnalyzer.default(this.form, input, this.matching);
+ this.recategorizeAllInputs();
+ return this;
+ } // Nothing to do with 1-character fields
+
+
if (input.maxLength === 1) return this;
- if (this.inputs.all.has(input)) return this;
this.inputs.all.add(input);
const opts = {
isLogin: this.isLogin,
@@ -10787,6 +10822,8 @@ const loginRegex = new RegExp(/sign(ing)?.?in(?!g)|log.?(i|o)n|log.?out|unsubscr
const signupRegex = new RegExp(/sign(ing)?.?up|join|\bregist(er|ration)|newsletter|\bsubscri(be|ption)|contact|create|start|enroll|settings|preferences|profile|update|checkout|guest|purchase|buy|order|schedule|estimate|request|new.?customer|(confirm|retype|repeat) password|password confirm?/i);
const conservativeSignupRegex = new RegExp(/sign.?up|join|register|enroll|newsletter|subscri(be|ption)|settings|preferences|profile|update/i);
const strictSignupRegex = new RegExp(/sign.?up|join|register|(create|new).+account|enroll|settings|preferences|profile|update/i);
+const resetPasswordLink = new RegExp(/(forgot(ten)?|reset|don't remember) (your )?password|password forgotten/i);
+const loginProvidersRegex = new RegExp(/ with /i);
class FormAnalyzer {
/** @type HTMLElement */
@@ -10909,7 +10946,7 @@ class FormAnalyzer {
shouldCheckUnifiedForm = false,
shouldBeConservative = false
} = _ref;
- const matchesLogin = /current.?password/i.test(string) || loginRegex.test(string); // Check explicitly for unified login/signup forms
+ const matchesLogin = /current.?password/i.test(string) || loginRegex.test(string) || resetPasswordLink.test(string); // Check explicitly for unified login/signup forms
if (shouldCheckUnifiedForm && matchesLogin && strictSignupRegex.test(string)) {
this.increaseHybridSignal(strength, signalType);
@@ -11024,10 +11061,11 @@ class FormAnalyzer {
if (el instanceof HTMLAnchorElement && el.href && el.getAttribute('href') !== '#' || (el.getAttribute('role') || '').toUpperCase() === 'LINK' || el.matches('button[class*=secondary]')) {
- // Unless it's a forgotten password link, we don't flip those links
let shouldFlip = true;
- if (/(forgot(ten)?|reset) (your )?password|password forgotten| with /i.test(string)) {
+ if (resetPasswordLink.test(string) || // Don't flip forgotten password links
+ loginProvidersRegex.test(string) // Don't flip login providers links
+ ) {
shouldFlip = false;
}
@@ -17845,6 +17883,7 @@ const isLikelyASubmitButton = el => {
const dataTestId = el.getAttribute('data-test-id') || '';
const contentExcludingLabel = text + ' ' + title + ' ' + value;
return (el.getAttribute('type') === 'submit' || // is explicitly set as "submit"
+ el.getAttribute('name') === 'submit' || // is called "submit"
/primary|submit/i.test(el.className) || // has high-signal submit classes
/submit/i.test(dataTestId) || SUBMIT_BUTTON_REGEX.test(contentExcludingLabel) || // has high-signal text
el.offsetHeight * el.offsetWidth >= 10000 && !/secondary/i.test(el.className) // it's a large element 250x40px
diff --git a/dist/autofill.js b/dist/autofill.js
index c94ab37ce..66e3376a2 100644
--- a/dist/autofill.js
+++ b/dist/autofill.js
@@ -6314,8 +6314,6 @@ class Form {
/** @type {HTMLInputElement | null} */
- /** @type {boolean | null} */
-
/**
* @param {HTMLElement} form
* @param {HTMLInputElement|HTMLSelectElement} input
@@ -6332,14 +6330,9 @@ class Form {
_defineProperty(this, "activeInput", void 0);
- _defineProperty(this, "isSignup", void 0);
-
this.form = form;
this.matching = matching || (0, _matching.createMatching)();
this.formAnalyzer = new _FormAnalyzer.default(form, input, matching);
- this.isLogin = this.formAnalyzer.isLogin;
- this.isSignup = this.formAnalyzer.isSignup;
- this.isHybrid = this.formAnalyzer.isHybrid;
this.device = deviceInterface;
/** @type Record<'all' | SupportedMainTypes, Set> */
@@ -6366,6 +6359,24 @@ class Form {
for (const entry of entries) {
if (!entry.isIntersecting) this.removeTooltip();
}
+ });
+ this.mutObsConfig = {
+ childList: true,
+ subtree: true
+ };
+ this.mutObs = new MutationObserver(records => {
+ const anythingRemoved = records.some(record => record.removedNodes.length > 0);
+
+ if (anythingRemoved) {
+ // Must check for inputs because a parent may be removed and not show up in record.removedNodes
+ if ([...this.inputs.all].some(input => !input.isConnected)) {
+ // If any known input has been removed from the DOM, reanalyze the whole form
+ window.requestIdleCallback(() => {
+ this.formAnalyzer = new _FormAnalyzer.default(this.form, input, this.matching);
+ this.recategorizeAllInputs();
+ });
+ }
+ }
}); // This ensures we fire the handler again if the form is changed
this.addListener(form, 'input', () => {
@@ -6375,6 +6386,7 @@ class Form {
}
});
this.categorizeInputs();
+ this.mutObs.observe(this.form, this.mutObsConfig);
this.logFormInfo();
if (shouldAutoprompt) {
@@ -6382,6 +6394,18 @@ class Form {
}
}
+ get isLogin() {
+ return this.formAnalyzer.isLogin;
+ }
+
+ get isSignup() {
+ return this.formAnalyzer.isSignup;
+ }
+
+ get isHybrid() {
+ return this.formAnalyzer.isHybrid;
+ }
+
logFormInfo() {
if (!(0, _autofillUtils.shouldLog)()) return;
console.log("Form type: %c".concat(this.getFormType()), 'font-weight: bold');
@@ -6613,6 +6637,7 @@ class Form {
recategorizeAllInputs() {
+ this.initialScanComplete = false;
this.removeAllDecorations();
this.forgetAllInputs();
this.categorizeInputs();
@@ -6635,6 +6660,7 @@ class Form {
destroy() {
this.removeAllDecorations();
this.removeTooltip();
+ this.mutObs.disconnect();
this.matching.clear();
this.intObs = null;
}
@@ -6647,6 +6673,8 @@ class Form {
} else {
this.form.querySelectorAll(selector).forEach(input => this.addInput(input));
}
+
+ this.initialScanComplete = true;
}
get submitButtons() {
@@ -6702,9 +6730,16 @@ class Form {
addInput(input) {
var _this$device$settings;
- // Nothing to do with 1-character fields
+ if (this.inputs.all.has(input)) return this; // When new inputs are added after the initial scan, reanalyze the whole form
+
+ if (this.initialScanComplete) {
+ this.formAnalyzer = new _FormAnalyzer.default(this.form, input, this.matching);
+ this.recategorizeAllInputs();
+ return this;
+ } // Nothing to do with 1-character fields
+
+
if (input.maxLength === 1) return this;
- if (this.inputs.all.has(input)) return this;
this.inputs.all.add(input);
const opts = {
isLogin: this.isLogin,
@@ -7111,6 +7146,8 @@ const loginRegex = new RegExp(/sign(ing)?.?in(?!g)|log.?(i|o)n|log.?out|unsubscr
const signupRegex = new RegExp(/sign(ing)?.?up|join|\bregist(er|ration)|newsletter|\bsubscri(be|ption)|contact|create|start|enroll|settings|preferences|profile|update|checkout|guest|purchase|buy|order|schedule|estimate|request|new.?customer|(confirm|retype|repeat) password|password confirm?/i);
const conservativeSignupRegex = new RegExp(/sign.?up|join|register|enroll|newsletter|subscri(be|ption)|settings|preferences|profile|update/i);
const strictSignupRegex = new RegExp(/sign.?up|join|register|(create|new).+account|enroll|settings|preferences|profile|update/i);
+const resetPasswordLink = new RegExp(/(forgot(ten)?|reset|don't remember) (your )?password|password forgotten/i);
+const loginProvidersRegex = new RegExp(/ with /i);
class FormAnalyzer {
/** @type HTMLElement */
@@ -7233,7 +7270,7 @@ class FormAnalyzer {
shouldCheckUnifiedForm = false,
shouldBeConservative = false
} = _ref;
- const matchesLogin = /current.?password/i.test(string) || loginRegex.test(string); // Check explicitly for unified login/signup forms
+ const matchesLogin = /current.?password/i.test(string) || loginRegex.test(string) || resetPasswordLink.test(string); // Check explicitly for unified login/signup forms
if (shouldCheckUnifiedForm && matchesLogin && strictSignupRegex.test(string)) {
this.increaseHybridSignal(strength, signalType);
@@ -7348,10 +7385,11 @@ class FormAnalyzer {
if (el instanceof HTMLAnchorElement && el.href && el.getAttribute('href') !== '#' || (el.getAttribute('role') || '').toUpperCase() === 'LINK' || el.matches('button[class*=secondary]')) {
- // Unless it's a forgotten password link, we don't flip those links
let shouldFlip = true;
- if (/(forgot(ten)?|reset) (your )?password|password forgotten| with /i.test(string)) {
+ if (resetPasswordLink.test(string) || // Don't flip forgotten password links
+ loginProvidersRegex.test(string) // Don't flip login providers links
+ ) {
shouldFlip = false;
}
@@ -14169,6 +14207,7 @@ const isLikelyASubmitButton = el => {
const dataTestId = el.getAttribute('data-test-id') || '';
const contentExcludingLabel = text + ' ' + title + ' ' + value;
return (el.getAttribute('type') === 'submit' || // is explicitly set as "submit"
+ el.getAttribute('name') === 'submit' || // is called "submit"
/primary|submit/i.test(el.className) || // has high-signal submit classes
/submit/i.test(dataTestId) || SUBMIT_BUTTON_REGEX.test(contentExcludingLabel) || // has high-signal text
el.offsetHeight * el.offsetWidth >= 10000 && !/secondary/i.test(el.className) // it's a large element 250x40px
diff --git a/integration-test/helpers/mocks.js b/integration-test/helpers/mocks.js
index 354b552b8..afc8b9ce1 100644
--- a/integration-test/helpers/mocks.js
+++ b/integration-test/helpers/mocks.js
@@ -9,6 +9,7 @@ export const constants = {
'emailAtTopLeft': 'pages/email-at-top-left.html',
'iframeContainer': 'pages/iframe-container.html',
'signup': 'pages/signup.html',
+ 'mutatingForm': 'pages/mutating-form.html',
'login': 'pages/login.html',
'loginWithPoorForm': 'pages/login-poor-form.html',
'loginWithText': 'pages/login-with-text.html',
diff --git a/integration-test/helpers/pages.js b/integration-test/helpers/pages.js
index c928afe85..f55b57beb 100644
--- a/integration-test/helpers/pages.js
+++ b/integration-test/helpers/pages.js
@@ -59,6 +59,26 @@ export function incontextSignupPage (page, { platform } = { platform: 'extension
return new IncontextSignupPage()
}
+export function mutatingFormPage (page) {
+ class MutatingFormPage {
+ async navigate () {
+ await page.goto(constants.pages['mutatingForm'])
+ }
+ async toggleLoginOrSignup () {
+ const toggleBtn = page.locator('#toggle-login-signup')
+ await toggleBtn.click()
+ }
+ passwordFieldShowsKey () {
+ return loginPage(page).passwordFieldShowsKey()
+ }
+ assertPasswordHasNoIcon () {
+ return signupPage(page).assertPasswordHasNoIcon()
+ }
+ }
+
+ return new MutatingFormPage()
+}
+
/**
* A wrapper around interactions for `integration-test/pages/signup.html`
*
@@ -68,6 +88,7 @@ export function signupPage (page) {
const decoratedFirstInputSelector = '#email' + constants.fields.email.selectors.identity
const decoratedSecondInputSelector = '#email-2' + constants.fields.email.selectors.identity
const emailStyleAttr = () => page.locator('#email').first().getAttribute('style')
+ const passwordStyleAttr = () => page.locator('#password' + constants.fields.password.selectors.credential).getAttribute('style')
class SignupPage {
async navigate () {
@@ -232,7 +253,10 @@ export function signupPage (page) {
await expect(input).toHaveValue('')
}
async assertEmailHasNoDaxIcon () {
- expect(await emailStyleAttr()).toBeNull()
+ expect(await emailStyleAttr()).toBeFalsy()
+ }
+ async assertPasswordHasNoIcon () {
+ expect(await passwordStyleAttr()).toBeFalsy()
}
}
@@ -293,16 +317,24 @@ export function loginPage (page, opts = {}) {
const emailStyle = await page.locator('#email').getAttribute('style')
expect(emailStyle).toContain(constants.iconMatchers.dax)
}
- async emailHasDaxPasswordNoIcon () {
- await this.emailFieldShowsDax()
+ async passwordFieldShowsKey () {
+ // don't make assertions until the element is both found + has a none-empty 'style' attribute
+ await page.waitForFunction(() => Boolean(document.querySelector('#password')?.getAttribute('style')))
+ const emailStyle = await page.locator('#password').getAttribute('style')
+ expect(emailStyle).toContain(constants.iconMatchers.key)
+ }
+ async passwordHasNoIcon () {
const passwordStyle = await page.locator('#password').getAttribute('style')
expect(passwordStyle || '').not.toContain('data:image/svg+xml;base64,')
}
+ async emailHasDaxPasswordNoIcon () {
+ await this.emailFieldShowsDax()
+ await this.passwordHasNoIcon()
+ }
async onlyPasswordFieldHasIcon () {
const styles1 = await page.locator('#email').getAttribute('style')
- const styles2 = await page.locator('#password').getAttribute('style')
expect(styles1 || '').not.toContain('data:image/svg+xml;base64,')
- expect(styles2 || '').toContain(constants.iconMatchers.key)
+ await this.passwordFieldShowsKey()
}
/**
* @param {string} username
diff --git a/integration-test/pages/mutating-form.html b/integration-test/pages/mutating-form.html
new file mode 100644
index 000000000..cad764bc2
--- /dev/null
+++ b/integration-test/pages/mutating-form.html
@@ -0,0 +1,161 @@
+
+
+
+
+
+
+ Mutating form
+
+
+
+
+
+
+
+[Home]
+
+
+
+
+
+
+
+
+
+
diff --git a/integration-test/tests/mutating-form.macos.spec.js b/integration-test/tests/mutating-form.macos.spec.js
new file mode 100644
index 000000000..3451cdd37
--- /dev/null
+++ b/integration-test/tests/mutating-form.macos.spec.js
@@ -0,0 +1,36 @@
+import {createAutofillScript, forwardConsoleMessages} from '../helpers/harness.js'
+import {createWebkitMocks, macosContentScopeReplacements} from '../helpers/mocks.webkit.js'
+import {mutatingFormPage} from '../helpers/pages.js'
+import {test as base} from '@playwright/test'
+
+/**
+ * Tests for various auto-fill scenarios on macos
+ */
+const test = base.extend({})
+
+test.describe('Mutating form page', () => {
+ async function applyScript (page) {
+ await createAutofillScript()
+ .replaceAll(macosContentScopeReplacements())
+ .platform('macos')
+ .applyTo(page)
+ }
+
+ test('works fine on macOS', async ({page}) => {
+ // enable in-terminal exceptions
+ await forwardConsoleMessages(page)
+ await createWebkitMocks().applyTo(page)
+
+ // Load the autofill.js script with replacements
+ await await applyScript(page)
+
+ const mutatingForm = mutatingFormPage(page)
+ await mutatingForm.navigate()
+
+ await mutatingForm.passwordFieldShowsKey()
+
+ await mutatingForm.toggleLoginOrSignup()
+
+ await mutatingForm.assertPasswordHasNoIcon()
+ })
+})
diff --git a/src/Form/Form.js b/src/Form/Form.js
index 5c66a0ffb..6cefc81f3 100644
--- a/src/Form/Form.js
+++ b/src/Form/Form.js
@@ -72,6 +72,23 @@ class Form {
}
})
+ this.mutObsConfig = { childList: true, subtree: true }
+ this.mutObs = new MutationObserver(
+ (records) => {
+ const anythingRemoved = records.some(record => record.removedNodes.length > 0)
+ if (anythingRemoved) {
+ // Must check for inputs because a parent may be removed and not show up in record.removedNodes
+ if ([...this.inputs.all].some(input => !input.isConnected)) {
+ // If any known input has been removed from the DOM, reanalyze the whole form
+ window.requestIdleCallback(() => {
+ this.formAnalyzer = new FormAnalyzer(this.form, input, this.matching)
+ this.recategorizeAllInputs()
+ })
+ }
+ }
+ }
+ )
+
// This ensures we fire the handler again if the form is changed
this.addListener(form, 'input', () => {
if (!this.isAutofilling) {
@@ -81,6 +98,7 @@ class Form {
})
this.categorizeInputs()
+ this.mutObs.observe(this.form, this.mutObsConfig)
this.logFormInfo()
@@ -300,6 +318,7 @@ class Form {
* Resets our input scoring and starts from scratch
*/
recategorizeAllInputs () {
+ this.initialScanComplete = false
this.removeAllDecorations()
this.forgetAllInputs()
this.categorizeInputs()
@@ -319,6 +338,7 @@ class Form {
destroy () {
this.removeAllDecorations()
this.removeTooltip()
+ this.mutObs.disconnect()
this.matching.clear()
this.intObs = null
}
@@ -330,6 +350,7 @@ class Form {
} else {
this.form.querySelectorAll(selector).forEach(input => this.addInput(input))
}
+ this.initialScanComplete = true
}
get submitButtons () {
@@ -383,11 +404,18 @@ class Form {
}
addInput (input) {
+ if (this.inputs.all.has(input)) return this
+
+ // When new inputs are added after the initial scan, reanalyze the whole form
+ if (this.initialScanComplete) {
+ this.formAnalyzer = new FormAnalyzer(this.form, input, this.matching)
+ this.recategorizeAllInputs()
+ return this
+ }
+
// Nothing to do with 1-character fields
if (input.maxLength === 1) return this
- if (this.inputs.all.has(input)) return this
-
this.inputs.all.add(input)
const opts = {
diff --git a/swift-package/Resources/assets/autofill-debug.js b/swift-package/Resources/assets/autofill-debug.js
index 11b484cfe..ef967f0d4 100644
--- a/swift-package/Resources/assets/autofill-debug.js
+++ b/swift-package/Resources/assets/autofill-debug.js
@@ -9990,8 +9990,6 @@ class Form {
/** @type {HTMLInputElement | null} */
- /** @type {boolean | null} */
-
/**
* @param {HTMLElement} form
* @param {HTMLInputElement|HTMLSelectElement} input
@@ -10008,14 +10006,9 @@ class Form {
_defineProperty(this, "activeInput", void 0);
- _defineProperty(this, "isSignup", void 0);
-
this.form = form;
this.matching = matching || (0, _matching.createMatching)();
this.formAnalyzer = new _FormAnalyzer.default(form, input, matching);
- this.isLogin = this.formAnalyzer.isLogin;
- this.isSignup = this.formAnalyzer.isSignup;
- this.isHybrid = this.formAnalyzer.isHybrid;
this.device = deviceInterface;
/** @type Record<'all' | SupportedMainTypes, Set> */
@@ -10042,6 +10035,24 @@ class Form {
for (const entry of entries) {
if (!entry.isIntersecting) this.removeTooltip();
}
+ });
+ this.mutObsConfig = {
+ childList: true,
+ subtree: true
+ };
+ this.mutObs = new MutationObserver(records => {
+ const anythingRemoved = records.some(record => record.removedNodes.length > 0);
+
+ if (anythingRemoved) {
+ // Must check for inputs because a parent may be removed and not show up in record.removedNodes
+ if ([...this.inputs.all].some(input => !input.isConnected)) {
+ // If any known input has been removed from the DOM, reanalyze the whole form
+ window.requestIdleCallback(() => {
+ this.formAnalyzer = new _FormAnalyzer.default(this.form, input, this.matching);
+ this.recategorizeAllInputs();
+ });
+ }
+ }
}); // This ensures we fire the handler again if the form is changed
this.addListener(form, 'input', () => {
@@ -10051,6 +10062,7 @@ class Form {
}
});
this.categorizeInputs();
+ this.mutObs.observe(this.form, this.mutObsConfig);
this.logFormInfo();
if (shouldAutoprompt) {
@@ -10058,6 +10070,18 @@ class Form {
}
}
+ get isLogin() {
+ return this.formAnalyzer.isLogin;
+ }
+
+ get isSignup() {
+ return this.formAnalyzer.isSignup;
+ }
+
+ get isHybrid() {
+ return this.formAnalyzer.isHybrid;
+ }
+
logFormInfo() {
if (!(0, _autofillUtils.shouldLog)()) return;
console.log("Form type: %c".concat(this.getFormType()), 'font-weight: bold');
@@ -10289,6 +10313,7 @@ class Form {
recategorizeAllInputs() {
+ this.initialScanComplete = false;
this.removeAllDecorations();
this.forgetAllInputs();
this.categorizeInputs();
@@ -10311,6 +10336,7 @@ class Form {
destroy() {
this.removeAllDecorations();
this.removeTooltip();
+ this.mutObs.disconnect();
this.matching.clear();
this.intObs = null;
}
@@ -10323,6 +10349,8 @@ class Form {
} else {
this.form.querySelectorAll(selector).forEach(input => this.addInput(input));
}
+
+ this.initialScanComplete = true;
}
get submitButtons() {
@@ -10378,9 +10406,16 @@ class Form {
addInput(input) {
var _this$device$settings;
- // Nothing to do with 1-character fields
+ if (this.inputs.all.has(input)) return this; // When new inputs are added after the initial scan, reanalyze the whole form
+
+ if (this.initialScanComplete) {
+ this.formAnalyzer = new _FormAnalyzer.default(this.form, input, this.matching);
+ this.recategorizeAllInputs();
+ return this;
+ } // Nothing to do with 1-character fields
+
+
if (input.maxLength === 1) return this;
- if (this.inputs.all.has(input)) return this;
this.inputs.all.add(input);
const opts = {
isLogin: this.isLogin,
@@ -10787,6 +10822,8 @@ const loginRegex = new RegExp(/sign(ing)?.?in(?!g)|log.?(i|o)n|log.?out|unsubscr
const signupRegex = new RegExp(/sign(ing)?.?up|join|\bregist(er|ration)|newsletter|\bsubscri(be|ption)|contact|create|start|enroll|settings|preferences|profile|update|checkout|guest|purchase|buy|order|schedule|estimate|request|new.?customer|(confirm|retype|repeat) password|password confirm?/i);
const conservativeSignupRegex = new RegExp(/sign.?up|join|register|enroll|newsletter|subscri(be|ption)|settings|preferences|profile|update/i);
const strictSignupRegex = new RegExp(/sign.?up|join|register|(create|new).+account|enroll|settings|preferences|profile|update/i);
+const resetPasswordLink = new RegExp(/(forgot(ten)?|reset|don't remember) (your )?password|password forgotten/i);
+const loginProvidersRegex = new RegExp(/ with /i);
class FormAnalyzer {
/** @type HTMLElement */
@@ -10909,7 +10946,7 @@ class FormAnalyzer {
shouldCheckUnifiedForm = false,
shouldBeConservative = false
} = _ref;
- const matchesLogin = /current.?password/i.test(string) || loginRegex.test(string); // Check explicitly for unified login/signup forms
+ const matchesLogin = /current.?password/i.test(string) || loginRegex.test(string) || resetPasswordLink.test(string); // Check explicitly for unified login/signup forms
if (shouldCheckUnifiedForm && matchesLogin && strictSignupRegex.test(string)) {
this.increaseHybridSignal(strength, signalType);
@@ -11024,10 +11061,11 @@ class FormAnalyzer {
if (el instanceof HTMLAnchorElement && el.href && el.getAttribute('href') !== '#' || (el.getAttribute('role') || '').toUpperCase() === 'LINK' || el.matches('button[class*=secondary]')) {
- // Unless it's a forgotten password link, we don't flip those links
let shouldFlip = true;
- if (/(forgot(ten)?|reset) (your )?password|password forgotten| with /i.test(string)) {
+ if (resetPasswordLink.test(string) || // Don't flip forgotten password links
+ loginProvidersRegex.test(string) // Don't flip login providers links
+ ) {
shouldFlip = false;
}
@@ -17845,6 +17883,7 @@ const isLikelyASubmitButton = el => {
const dataTestId = el.getAttribute('data-test-id') || '';
const contentExcludingLabel = text + ' ' + title + ' ' + value;
return (el.getAttribute('type') === 'submit' || // is explicitly set as "submit"
+ el.getAttribute('name') === 'submit' || // is called "submit"
/primary|submit/i.test(el.className) || // has high-signal submit classes
/submit/i.test(dataTestId) || SUBMIT_BUTTON_REGEX.test(contentExcludingLabel) || // has high-signal text
el.offsetHeight * el.offsetWidth >= 10000 && !/secondary/i.test(el.className) // it's a large element 250x40px
diff --git a/swift-package/Resources/assets/autofill.js b/swift-package/Resources/assets/autofill.js
index c94ab37ce..66e3376a2 100644
--- a/swift-package/Resources/assets/autofill.js
+++ b/swift-package/Resources/assets/autofill.js
@@ -6314,8 +6314,6 @@ class Form {
/** @type {HTMLInputElement | null} */
- /** @type {boolean | null} */
-
/**
* @param {HTMLElement} form
* @param {HTMLInputElement|HTMLSelectElement} input
@@ -6332,14 +6330,9 @@ class Form {
_defineProperty(this, "activeInput", void 0);
- _defineProperty(this, "isSignup", void 0);
-
this.form = form;
this.matching = matching || (0, _matching.createMatching)();
this.formAnalyzer = new _FormAnalyzer.default(form, input, matching);
- this.isLogin = this.formAnalyzer.isLogin;
- this.isSignup = this.formAnalyzer.isSignup;
- this.isHybrid = this.formAnalyzer.isHybrid;
this.device = deviceInterface;
/** @type Record<'all' | SupportedMainTypes, Set> */
@@ -6366,6 +6359,24 @@ class Form {
for (const entry of entries) {
if (!entry.isIntersecting) this.removeTooltip();
}
+ });
+ this.mutObsConfig = {
+ childList: true,
+ subtree: true
+ };
+ this.mutObs = new MutationObserver(records => {
+ const anythingRemoved = records.some(record => record.removedNodes.length > 0);
+
+ if (anythingRemoved) {
+ // Must check for inputs because a parent may be removed and not show up in record.removedNodes
+ if ([...this.inputs.all].some(input => !input.isConnected)) {
+ // If any known input has been removed from the DOM, reanalyze the whole form
+ window.requestIdleCallback(() => {
+ this.formAnalyzer = new _FormAnalyzer.default(this.form, input, this.matching);
+ this.recategorizeAllInputs();
+ });
+ }
+ }
}); // This ensures we fire the handler again if the form is changed
this.addListener(form, 'input', () => {
@@ -6375,6 +6386,7 @@ class Form {
}
});
this.categorizeInputs();
+ this.mutObs.observe(this.form, this.mutObsConfig);
this.logFormInfo();
if (shouldAutoprompt) {
@@ -6382,6 +6394,18 @@ class Form {
}
}
+ get isLogin() {
+ return this.formAnalyzer.isLogin;
+ }
+
+ get isSignup() {
+ return this.formAnalyzer.isSignup;
+ }
+
+ get isHybrid() {
+ return this.formAnalyzer.isHybrid;
+ }
+
logFormInfo() {
if (!(0, _autofillUtils.shouldLog)()) return;
console.log("Form type: %c".concat(this.getFormType()), 'font-weight: bold');
@@ -6613,6 +6637,7 @@ class Form {
recategorizeAllInputs() {
+ this.initialScanComplete = false;
this.removeAllDecorations();
this.forgetAllInputs();
this.categorizeInputs();
@@ -6635,6 +6660,7 @@ class Form {
destroy() {
this.removeAllDecorations();
this.removeTooltip();
+ this.mutObs.disconnect();
this.matching.clear();
this.intObs = null;
}
@@ -6647,6 +6673,8 @@ class Form {
} else {
this.form.querySelectorAll(selector).forEach(input => this.addInput(input));
}
+
+ this.initialScanComplete = true;
}
get submitButtons() {
@@ -6702,9 +6730,16 @@ class Form {
addInput(input) {
var _this$device$settings;
- // Nothing to do with 1-character fields
+ if (this.inputs.all.has(input)) return this; // When new inputs are added after the initial scan, reanalyze the whole form
+
+ if (this.initialScanComplete) {
+ this.formAnalyzer = new _FormAnalyzer.default(this.form, input, this.matching);
+ this.recategorizeAllInputs();
+ return this;
+ } // Nothing to do with 1-character fields
+
+
if (input.maxLength === 1) return this;
- if (this.inputs.all.has(input)) return this;
this.inputs.all.add(input);
const opts = {
isLogin: this.isLogin,
@@ -7111,6 +7146,8 @@ const loginRegex = new RegExp(/sign(ing)?.?in(?!g)|log.?(i|o)n|log.?out|unsubscr
const signupRegex = new RegExp(/sign(ing)?.?up|join|\bregist(er|ration)|newsletter|\bsubscri(be|ption)|contact|create|start|enroll|settings|preferences|profile|update|checkout|guest|purchase|buy|order|schedule|estimate|request|new.?customer|(confirm|retype|repeat) password|password confirm?/i);
const conservativeSignupRegex = new RegExp(/sign.?up|join|register|enroll|newsletter|subscri(be|ption)|settings|preferences|profile|update/i);
const strictSignupRegex = new RegExp(/sign.?up|join|register|(create|new).+account|enroll|settings|preferences|profile|update/i);
+const resetPasswordLink = new RegExp(/(forgot(ten)?|reset|don't remember) (your )?password|password forgotten/i);
+const loginProvidersRegex = new RegExp(/ with /i);
class FormAnalyzer {
/** @type HTMLElement */
@@ -7233,7 +7270,7 @@ class FormAnalyzer {
shouldCheckUnifiedForm = false,
shouldBeConservative = false
} = _ref;
- const matchesLogin = /current.?password/i.test(string) || loginRegex.test(string); // Check explicitly for unified login/signup forms
+ const matchesLogin = /current.?password/i.test(string) || loginRegex.test(string) || resetPasswordLink.test(string); // Check explicitly for unified login/signup forms
if (shouldCheckUnifiedForm && matchesLogin && strictSignupRegex.test(string)) {
this.increaseHybridSignal(strength, signalType);
@@ -7348,10 +7385,11 @@ class FormAnalyzer {
if (el instanceof HTMLAnchorElement && el.href && el.getAttribute('href') !== '#' || (el.getAttribute('role') || '').toUpperCase() === 'LINK' || el.matches('button[class*=secondary]')) {
- // Unless it's a forgotten password link, we don't flip those links
let shouldFlip = true;
- if (/(forgot(ten)?|reset) (your )?password|password forgotten| with /i.test(string)) {
+ if (resetPasswordLink.test(string) || // Don't flip forgotten password links
+ loginProvidersRegex.test(string) // Don't flip login providers links
+ ) {
shouldFlip = false;
}
@@ -14169,6 +14207,7 @@ const isLikelyASubmitButton = el => {
const dataTestId = el.getAttribute('data-test-id') || '';
const contentExcludingLabel = text + ' ' + title + ' ' + value;
return (el.getAttribute('type') === 'submit' || // is explicitly set as "submit"
+ el.getAttribute('name') === 'submit' || // is called "submit"
/primary|submit/i.test(el.className) || // has high-signal submit classes
/submit/i.test(dataTestId) || SUBMIT_BUTTON_REGEX.test(contentExcludingLabel) || // has high-signal text
el.offsetHeight * el.offsetWidth >= 10000 && !/secondary/i.test(el.className) // it's a large element 250x40px
From 55f09109a14fd38cbc18ffb35f5e72d5d1afd354 Mon Sep 17 00:00:00 2001
From: Emanuele Feliziani
Date: Fri, 21 Jul 2023 12:05:16 +0200
Subject: [PATCH 4/6] Add performance safeguards
Signed-off-by: Emanuele Feliziani
---
dist/autofill-debug.js | 126 +++++++++++++++---
dist/autofill.js | 126 +++++++++++++++---
integration-test/helpers/pages.js | 2 +-
src/Form/Form.js | 30 ++++-
src/Form/Form.test.js | 41 ++++++
src/Scanner.js | 25 +++-
src/Scanner.test.js | 19 ++-
src/__snapshots__/Scanner.test.js.snap | 124 +++++++++++++++++
src/constants.js | 6 +-
.../Resources/assets/autofill-debug.js | 126 +++++++++++++++---
swift-package/Resources/assets/autofill.js | 126 +++++++++++++++---
11 files changed, 657 insertions(+), 94 deletions(-)
diff --git a/dist/autofill-debug.js b/dist/autofill-debug.js
index 4f2429514..a33be6fe9 100644
--- a/dist/autofill-debug.js
+++ b/dist/autofill-debug.js
@@ -10084,7 +10084,9 @@ function _defineProperty(obj, key, value) { if (key in obj) { Object.definePrope
const {
ATTR_AUTOFILL,
- ATTR_INPUT_TYPE
+ ATTR_INPUT_TYPE,
+ MAX_FORM_MUT_OBS_COUNT,
+ MAX_INPUTS_PER_FORM
} = _constants.constants;
class Form {
@@ -10094,8 +10096,6 @@ class Form {
/** @type {HTMLInputElement | null} */
- /** @type {boolean | null} */
-
/**
* @param {HTMLElement} form
* @param {HTMLInputElement|HTMLSelectElement} input
@@ -10112,14 +10112,9 @@ class Form {
_defineProperty(this, "activeInput", void 0);
- _defineProperty(this, "isSignup", void 0);
-
this.form = form;
this.matching = matching || (0, _matching.createMatching)();
this.formAnalyzer = new _FormAnalyzer.default(form, input, matching);
- this.isLogin = this.formAnalyzer.isLogin;
- this.isSignup = this.formAnalyzer.isSignup;
- this.isHybrid = this.formAnalyzer.isHybrid;
this.device = deviceInterface;
/** @type Record<'all' | SupportedMainTypes, Set> */
@@ -10146,6 +10141,30 @@ class Form {
for (const entry of entries) {
if (!entry.isIntersecting) this.removeTooltip();
}
+ });
+ this.mutObsCount = 0;
+ this.mutObsConfig = {
+ childList: true,
+ subtree: true
+ };
+ this.mutObs = new MutationObserver(records => {
+ const anythingRemoved = records.some(record => record.removedNodes.length > 0);
+
+ if (anythingRemoved) {
+ // Must check for inputs because a parent may be removed and not show up in record.removedNodes
+ if ([...this.inputs.all].some(input => !input.isConnected)) {
+ // If any known input has been removed from the DOM, reanalyze the whole form
+ window.requestIdleCallback(() => {
+ this.formAnalyzer = new _FormAnalyzer.default(this.form, input, this.matching);
+ this.recategorizeAllInputs();
+ });
+ this.mutObsCount++; // If the form mutates too much, disconnect to avoid performance issues
+
+ if (this.mutObsCount >= MAX_FORM_MUT_OBS_COUNT) {
+ this.mutObs.disconnect();
+ }
+ }
+ }
}); // This ensures we fire the handler again if the form is changed
this.addListener(form, 'input', () => {
@@ -10155,6 +10174,7 @@ class Form {
}
});
this.categorizeInputs();
+ this.mutObs.observe(this.form, this.mutObsConfig);
this.logFormInfo();
if (shouldAutoprompt) {
@@ -10162,6 +10182,18 @@ class Form {
}
}
+ get isLogin() {
+ return this.formAnalyzer.isLogin;
+ }
+
+ get isSignup() {
+ return this.formAnalyzer.isSignup;
+ }
+
+ get isHybrid() {
+ return this.formAnalyzer.isHybrid;
+ }
+
logFormInfo() {
if (!(0, _autofillUtils.shouldLog)()) return;
console.log("Form type: %c".concat(this.getFormType()), 'font-weight: bold');
@@ -10353,6 +10385,7 @@ class Form {
(0, _autofillUtils.removeInlineStyles)(input, (0, _inputStyles.getIconStylesBase)(input, this));
(0, _autofillUtils.removeInlineStyles)(input, (0, _inputStyles.getIconStylesAlternate)(input, this));
input.removeAttribute(ATTR_AUTOFILL);
+ input.removeAttribute(ATTR_INPUT_TYPE);
}
removeAllDecorations() {
@@ -10383,6 +10416,7 @@ class Form {
forgetAllInputs() {
this.execOnInputs(input => {
+ input.removeAttribute(ATTR_AUTOFILL);
input.removeAttribute(ATTR_INPUT_TYPE);
});
Object.values(this.inputs).forEach(inputSet => inputSet.clear());
@@ -10393,6 +10427,7 @@ class Form {
recategorizeAllInputs() {
+ this.initialScanComplete = false;
this.removeAllDecorations();
this.forgetAllInputs();
this.categorizeInputs();
@@ -10415,6 +10450,7 @@ class Form {
destroy() {
this.removeAllDecorations();
this.removeTooltip();
+ this.mutObs.disconnect();
this.matching.clear();
this.intObs = null;
}
@@ -10425,8 +10461,16 @@ class Form {
if (this.form.matches(selector)) {
this.addInput(this.form);
} else {
- this.form.querySelectorAll(selector).forEach(input => this.addInput(input));
+ const foundInputs = this.form.querySelectorAll(selector);
+
+ if (foundInputs.length < MAX_INPUTS_PER_FORM) {
+ foundInputs.forEach(input => this.addInput(input));
+ } else {
+ console.log('The form has too many inputs, bailing.');
+ }
}
+
+ this.initialScanComplete = true;
}
get submitButtons() {
@@ -10482,9 +10526,23 @@ class Form {
addInput(input) {
var _this$device$settings;
- // Nothing to do with 1-character fields
+ if (this.inputs.all.has(input)) return this; // If the form has too many inputs, destroy everything to avoid performance issues
+
+ if (this.inputs.all.size > MAX_INPUTS_PER_FORM) {
+ console.log('The form has too many inputs, destroying.');
+ this.destroy();
+ return this;
+ } // When new inputs are added after the initial scan, reanalyze the whole form
+
+
+ if (this.initialScanComplete) {
+ this.formAnalyzer = new _FormAnalyzer.default(this.form, input, this.matching);
+ this.recategorizeAllInputs();
+ return this;
+ } // Nothing to do with 1-character fields
+
+
if (input.maxLength === 1) return this;
- if (this.inputs.all.has(input)) return this;
this.inputs.all.add(input);
const opts = {
isLogin: this.isLogin,
@@ -10891,6 +10949,8 @@ const loginRegex = new RegExp(/sign(ing)?.?in(?!g)|log.?(i|o)n|log.?out|unsubscr
const signupRegex = new RegExp(/sign(ing)?.?up|join|\bregist(er|ration)|newsletter|\bsubscri(be|ption)|contact|create|start|enroll|settings|preferences|profile|update|checkout|guest|purchase|buy|order|schedule|estimate|request|new.?customer|(confirm|retype|repeat) password|password confirm?/i);
const conservativeSignupRegex = new RegExp(/sign.?up|join|register|enroll|newsletter|subscri(be|ption)|settings|preferences|profile|update/i);
const strictSignupRegex = new RegExp(/sign.?up|join|register|(create|new).+account|enroll|settings|preferences|profile|update/i);
+const resetPasswordLink = new RegExp(/(forgot(ten)?|reset|don't remember) (your )?password|password forgotten/i);
+const loginProvidersRegex = new RegExp(/ with /i);
class FormAnalyzer {
/** @type HTMLElement */
@@ -11013,7 +11073,7 @@ class FormAnalyzer {
shouldCheckUnifiedForm = false,
shouldBeConservative = false
} = _ref;
- const matchesLogin = /current.?password/i.test(string) || loginRegex.test(string); // Check explicitly for unified login/signup forms
+ const matchesLogin = /current.?password/i.test(string) || loginRegex.test(string) || resetPasswordLink.test(string); // Check explicitly for unified login/signup forms
if (shouldCheckUnifiedForm && matchesLogin && strictSignupRegex.test(string)) {
this.increaseHybridSignal(strength, signalType);
@@ -11128,10 +11188,11 @@ class FormAnalyzer {
if (el instanceof HTMLAnchorElement && el.href && el.getAttribute('href') !== '#' || (el.getAttribute('role') || '').toUpperCase() === 'LINK' || el.matches('button[class*=secondary]')) {
- // Unless it's a forgotten password link, we don't flip those links
let shouldFlip = true;
- if (/(forgot(ten)?|reset) (your )?password|password forgotten| with /i.test(string)) {
+ if (resetPasswordLink.test(string) || // Don't flip forgotten password links
+ loginProvidersRegex.test(string) // Don't flip login providers links
+ ) {
shouldFlip = false;
}
@@ -15102,29 +15163,40 @@ var _Form = require("./Form/Form.js");
var _selectorsCss = require("./Form/selectors-css.js");
+var _constants = require("./constants.js");
+
var _matching = require("./Form/matching.js");
var _autofillUtils = require("./autofill-utils.js");
function _defineProperty(obj, key, value) { if (key in obj) { Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); } else { obj[key] = value; } return obj; }
+const {
+ MAX_INPUTS_PER_PAGE,
+ MAX_FORMS_PER_PAGE,
+ MAX_INPUTS_PER_FORM
+} = _constants.constants;
/**
* @typedef {{
* forms: Map;
* init(): ()=> void;
* enqueue(elements: (HTMLElement|Document)[]): void;
* findEligibleInputs(context): Scanner;
+ * options: ScannerOptions;
* }} Scanner
*
* @typedef {{
* initialDelay: number,
* bufferSize: number,
* debounceTimePeriod: number,
- * maxInputsOnPage: number,
+ * maxInputsPerPage: number,
+ * maxFormsPerPage: number,
+ * maxInputsPerForm: number
* }} ScannerOptions
*/
/** @type {ScannerOptions} */
+
const defaultScannerOptions = {
// This buffer size is very large because it's an unexpected edge-case that
// a DOM will be continually modified over and over without ever stopping. If we do see 1000 unique
@@ -15137,7 +15209,9 @@ const defaultScannerOptions = {
// How many inputs is too many on the page. If we detect that there's above
// this maximum, then we don't scan the page. This will prevent slowdowns on
// large pages which are unlikely to require autofill anyway.
- maxInputsOnPage: 100
+ maxInputsPerPage: MAX_INPUTS_PER_PAGE,
+ maxFormsPerPage: MAX_FORMS_PER_PAGE,
+ maxInputsPerForm: MAX_INPUTS_PER_FORM
};
/**
* This allows:
@@ -15283,7 +15357,7 @@ class DefaultScanner {
} else {
const inputs = context.querySelectorAll(_selectorsCss.FORM_INPUTS_SELECTOR);
- if (inputs.length > this.options.maxInputsOnPage) {
+ if (inputs.length > this.options.maxInputsPerPage) {
return this;
}
@@ -15360,9 +15434,14 @@ class DefaultScanner {
(_this$forms$get2 = this.forms.get(childForm)) === null || _this$forms$get2 === void 0 ? void 0 : _this$forms$get2.destroy();
this.forms.delete(childForm);
- }
+ } // Only add the form if below the limit of forms per page
+
- this.forms.set(parentForm, new _Form.Form(parentForm, input, this.device, this.matching, this.shouldAutoprompt));
+ if (this.forms.size < this.options.maxFormsPerPage) {
+ this.forms.set(parentForm, new _Form.Form(parentForm, input, this.device, this.matching, this.shouldAutoprompt));
+ } else {
+ console.log('The page has too many forms, stop adding them.');
+ }
}
}
/**
@@ -15430,7 +15509,7 @@ function createScanner(device, scannerOptions) {
});
}
-},{"./Form/Form.js":33,"./Form/matching.js":43,"./Form/selectors-css.js":44,"./autofill-utils.js":63}],52:[function(require,module,exports){
+},{"./Form/Form.js":33,"./Form/matching.js":43,"./Form/selectors-css.js":44,"./autofill-utils.js":63,"./constants.js":66}],52:[function(require,module,exports){
"use strict";
Object.defineProperty(exports, "__esModule", {
@@ -17948,6 +18027,7 @@ const isLikelyASubmitButton = el => {
const dataTestId = el.getAttribute('data-test-id') || '';
const contentExcludingLabel = text + ' ' + title + ' ' + value;
return (el.getAttribute('type') === 'submit' || // is explicitly set as "submit"
+ el.getAttribute('name') === 'submit' || // is called "submit"
/primary|submit/i.test(el.className) || // has high-signal submit classes
/submit/i.test(dataTestId) || SUBMIT_BUTTON_REGEX.test(contentExcludingLabel) || // has high-signal text
el.offsetHeight * el.offsetWidth >= 10000 && !/secondary/i.test(el.className) // it's a large element 250x40px
@@ -18246,7 +18326,11 @@ exports.constants = void 0;
const constants = {
ATTR_INPUT_TYPE: 'data-ddg-inputType',
ATTR_AUTOFILL: 'data-ddg-autofill',
- TEXT_LENGTH_CUTOFF: 50
+ TEXT_LENGTH_CUTOFF: 50,
+ MAX_INPUTS_PER_PAGE: 100,
+ MAX_FORMS_PER_PAGE: 30,
+ MAX_INPUTS_PER_FORM: 80,
+ MAX_FORM_MUT_OBS_COUNT: 50
};
exports.constants = constants;
diff --git a/dist/autofill.js b/dist/autofill.js
index 04a70fc9a..628a7c963 100644
--- a/dist/autofill.js
+++ b/dist/autofill.js
@@ -6408,7 +6408,9 @@ function _defineProperty(obj, key, value) { if (key in obj) { Object.definePrope
const {
ATTR_AUTOFILL,
- ATTR_INPUT_TYPE
+ ATTR_INPUT_TYPE,
+ MAX_FORM_MUT_OBS_COUNT,
+ MAX_INPUTS_PER_FORM
} = _constants.constants;
class Form {
@@ -6418,8 +6420,6 @@ class Form {
/** @type {HTMLInputElement | null} */
- /** @type {boolean | null} */
-
/**
* @param {HTMLElement} form
* @param {HTMLInputElement|HTMLSelectElement} input
@@ -6436,14 +6436,9 @@ class Form {
_defineProperty(this, "activeInput", void 0);
- _defineProperty(this, "isSignup", void 0);
-
this.form = form;
this.matching = matching || (0, _matching.createMatching)();
this.formAnalyzer = new _FormAnalyzer.default(form, input, matching);
- this.isLogin = this.formAnalyzer.isLogin;
- this.isSignup = this.formAnalyzer.isSignup;
- this.isHybrid = this.formAnalyzer.isHybrid;
this.device = deviceInterface;
/** @type Record<'all' | SupportedMainTypes, Set> */
@@ -6470,6 +6465,30 @@ class Form {
for (const entry of entries) {
if (!entry.isIntersecting) this.removeTooltip();
}
+ });
+ this.mutObsCount = 0;
+ this.mutObsConfig = {
+ childList: true,
+ subtree: true
+ };
+ this.mutObs = new MutationObserver(records => {
+ const anythingRemoved = records.some(record => record.removedNodes.length > 0);
+
+ if (anythingRemoved) {
+ // Must check for inputs because a parent may be removed and not show up in record.removedNodes
+ if ([...this.inputs.all].some(input => !input.isConnected)) {
+ // If any known input has been removed from the DOM, reanalyze the whole form
+ window.requestIdleCallback(() => {
+ this.formAnalyzer = new _FormAnalyzer.default(this.form, input, this.matching);
+ this.recategorizeAllInputs();
+ });
+ this.mutObsCount++; // If the form mutates too much, disconnect to avoid performance issues
+
+ if (this.mutObsCount >= MAX_FORM_MUT_OBS_COUNT) {
+ this.mutObs.disconnect();
+ }
+ }
+ }
}); // This ensures we fire the handler again if the form is changed
this.addListener(form, 'input', () => {
@@ -6479,6 +6498,7 @@ class Form {
}
});
this.categorizeInputs();
+ this.mutObs.observe(this.form, this.mutObsConfig);
this.logFormInfo();
if (shouldAutoprompt) {
@@ -6486,6 +6506,18 @@ class Form {
}
}
+ get isLogin() {
+ return this.formAnalyzer.isLogin;
+ }
+
+ get isSignup() {
+ return this.formAnalyzer.isSignup;
+ }
+
+ get isHybrid() {
+ return this.formAnalyzer.isHybrid;
+ }
+
logFormInfo() {
if (!(0, _autofillUtils.shouldLog)()) return;
console.log("Form type: %c".concat(this.getFormType()), 'font-weight: bold');
@@ -6677,6 +6709,7 @@ class Form {
(0, _autofillUtils.removeInlineStyles)(input, (0, _inputStyles.getIconStylesBase)(input, this));
(0, _autofillUtils.removeInlineStyles)(input, (0, _inputStyles.getIconStylesAlternate)(input, this));
input.removeAttribute(ATTR_AUTOFILL);
+ input.removeAttribute(ATTR_INPUT_TYPE);
}
removeAllDecorations() {
@@ -6707,6 +6740,7 @@ class Form {
forgetAllInputs() {
this.execOnInputs(input => {
+ input.removeAttribute(ATTR_AUTOFILL);
input.removeAttribute(ATTR_INPUT_TYPE);
});
Object.values(this.inputs).forEach(inputSet => inputSet.clear());
@@ -6717,6 +6751,7 @@ class Form {
recategorizeAllInputs() {
+ this.initialScanComplete = false;
this.removeAllDecorations();
this.forgetAllInputs();
this.categorizeInputs();
@@ -6739,6 +6774,7 @@ class Form {
destroy() {
this.removeAllDecorations();
this.removeTooltip();
+ this.mutObs.disconnect();
this.matching.clear();
this.intObs = null;
}
@@ -6749,8 +6785,16 @@ class Form {
if (this.form.matches(selector)) {
this.addInput(this.form);
} else {
- this.form.querySelectorAll(selector).forEach(input => this.addInput(input));
+ const foundInputs = this.form.querySelectorAll(selector);
+
+ if (foundInputs.length < MAX_INPUTS_PER_FORM) {
+ foundInputs.forEach(input => this.addInput(input));
+ } else {
+ console.log('The form has too many inputs, bailing.');
+ }
}
+
+ this.initialScanComplete = true;
}
get submitButtons() {
@@ -6806,9 +6850,23 @@ class Form {
addInput(input) {
var _this$device$settings;
- // Nothing to do with 1-character fields
+ if (this.inputs.all.has(input)) return this; // If the form has too many inputs, destroy everything to avoid performance issues
+
+ if (this.inputs.all.size > MAX_INPUTS_PER_FORM) {
+ console.log('The form has too many inputs, destroying.');
+ this.destroy();
+ return this;
+ } // When new inputs are added after the initial scan, reanalyze the whole form
+
+
+ if (this.initialScanComplete) {
+ this.formAnalyzer = new _FormAnalyzer.default(this.form, input, this.matching);
+ this.recategorizeAllInputs();
+ return this;
+ } // Nothing to do with 1-character fields
+
+
if (input.maxLength === 1) return this;
- if (this.inputs.all.has(input)) return this;
this.inputs.all.add(input);
const opts = {
isLogin: this.isLogin,
@@ -7215,6 +7273,8 @@ const loginRegex = new RegExp(/sign(ing)?.?in(?!g)|log.?(i|o)n|log.?out|unsubscr
const signupRegex = new RegExp(/sign(ing)?.?up|join|\bregist(er|ration)|newsletter|\bsubscri(be|ption)|contact|create|start|enroll|settings|preferences|profile|update|checkout|guest|purchase|buy|order|schedule|estimate|request|new.?customer|(confirm|retype|repeat) password|password confirm?/i);
const conservativeSignupRegex = new RegExp(/sign.?up|join|register|enroll|newsletter|subscri(be|ption)|settings|preferences|profile|update/i);
const strictSignupRegex = new RegExp(/sign.?up|join|register|(create|new).+account|enroll|settings|preferences|profile|update/i);
+const resetPasswordLink = new RegExp(/(forgot(ten)?|reset|don't remember) (your )?password|password forgotten/i);
+const loginProvidersRegex = new RegExp(/ with /i);
class FormAnalyzer {
/** @type HTMLElement */
@@ -7337,7 +7397,7 @@ class FormAnalyzer {
shouldCheckUnifiedForm = false,
shouldBeConservative = false
} = _ref;
- const matchesLogin = /current.?password/i.test(string) || loginRegex.test(string); // Check explicitly for unified login/signup forms
+ const matchesLogin = /current.?password/i.test(string) || loginRegex.test(string) || resetPasswordLink.test(string); // Check explicitly for unified login/signup forms
if (shouldCheckUnifiedForm && matchesLogin && strictSignupRegex.test(string)) {
this.increaseHybridSignal(strength, signalType);
@@ -7452,10 +7512,11 @@ class FormAnalyzer {
if (el instanceof HTMLAnchorElement && el.href && el.getAttribute('href') !== '#' || (el.getAttribute('role') || '').toUpperCase() === 'LINK' || el.matches('button[class*=secondary]')) {
- // Unless it's a forgotten password link, we don't flip those links
let shouldFlip = true;
- if (/(forgot(ten)?|reset) (your )?password|password forgotten| with /i.test(string)) {
+ if (resetPasswordLink.test(string) || // Don't flip forgotten password links
+ loginProvidersRegex.test(string) // Don't flip login providers links
+ ) {
shouldFlip = false;
}
@@ -11426,29 +11487,40 @@ var _Form = require("./Form/Form.js");
var _selectorsCss = require("./Form/selectors-css.js");
+var _constants = require("./constants.js");
+
var _matching = require("./Form/matching.js");
var _autofillUtils = require("./autofill-utils.js");
function _defineProperty(obj, key, value) { if (key in obj) { Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); } else { obj[key] = value; } return obj; }
+const {
+ MAX_INPUTS_PER_PAGE,
+ MAX_FORMS_PER_PAGE,
+ MAX_INPUTS_PER_FORM
+} = _constants.constants;
/**
* @typedef {{
* forms: Map;
* init(): ()=> void;
* enqueue(elements: (HTMLElement|Document)[]): void;
* findEligibleInputs(context): Scanner;
+ * options: ScannerOptions;
* }} Scanner
*
* @typedef {{
* initialDelay: number,
* bufferSize: number,
* debounceTimePeriod: number,
- * maxInputsOnPage: number,
+ * maxInputsPerPage: number,
+ * maxFormsPerPage: number,
+ * maxInputsPerForm: number
* }} ScannerOptions
*/
/** @type {ScannerOptions} */
+
const defaultScannerOptions = {
// This buffer size is very large because it's an unexpected edge-case that
// a DOM will be continually modified over and over without ever stopping. If we do see 1000 unique
@@ -11461,7 +11533,9 @@ const defaultScannerOptions = {
// How many inputs is too many on the page. If we detect that there's above
// this maximum, then we don't scan the page. This will prevent slowdowns on
// large pages which are unlikely to require autofill anyway.
- maxInputsOnPage: 100
+ maxInputsPerPage: MAX_INPUTS_PER_PAGE,
+ maxFormsPerPage: MAX_FORMS_PER_PAGE,
+ maxInputsPerForm: MAX_INPUTS_PER_FORM
};
/**
* This allows:
@@ -11607,7 +11681,7 @@ class DefaultScanner {
} else {
const inputs = context.querySelectorAll(_selectorsCss.FORM_INPUTS_SELECTOR);
- if (inputs.length > this.options.maxInputsOnPage) {
+ if (inputs.length > this.options.maxInputsPerPage) {
return this;
}
@@ -11684,9 +11758,14 @@ class DefaultScanner {
(_this$forms$get2 = this.forms.get(childForm)) === null || _this$forms$get2 === void 0 ? void 0 : _this$forms$get2.destroy();
this.forms.delete(childForm);
- }
+ } // Only add the form if below the limit of forms per page
+
- this.forms.set(parentForm, new _Form.Form(parentForm, input, this.device, this.matching, this.shouldAutoprompt));
+ if (this.forms.size < this.options.maxFormsPerPage) {
+ this.forms.set(parentForm, new _Form.Form(parentForm, input, this.device, this.matching, this.shouldAutoprompt));
+ } else {
+ console.log('The page has too many forms, stop adding them.');
+ }
}
}
/**
@@ -11754,7 +11833,7 @@ function createScanner(device, scannerOptions) {
});
}
-},{"./Form/Form.js":25,"./Form/matching.js":35,"./Form/selectors-css.js":36,"./autofill-utils.js":55}],44:[function(require,module,exports){
+},{"./Form/Form.js":25,"./Form/matching.js":35,"./Form/selectors-css.js":36,"./autofill-utils.js":55,"./constants.js":58}],44:[function(require,module,exports){
"use strict";
Object.defineProperty(exports, "__esModule", {
@@ -14272,6 +14351,7 @@ const isLikelyASubmitButton = el => {
const dataTestId = el.getAttribute('data-test-id') || '';
const contentExcludingLabel = text + ' ' + title + ' ' + value;
return (el.getAttribute('type') === 'submit' || // is explicitly set as "submit"
+ el.getAttribute('name') === 'submit' || // is called "submit"
/primary|submit/i.test(el.className) || // has high-signal submit classes
/submit/i.test(dataTestId) || SUBMIT_BUTTON_REGEX.test(contentExcludingLabel) || // has high-signal text
el.offsetHeight * el.offsetWidth >= 10000 && !/secondary/i.test(el.className) // it's a large element 250x40px
@@ -14570,7 +14650,11 @@ exports.constants = void 0;
const constants = {
ATTR_INPUT_TYPE: 'data-ddg-inputType',
ATTR_AUTOFILL: 'data-ddg-autofill',
- TEXT_LENGTH_CUTOFF: 50
+ TEXT_LENGTH_CUTOFF: 50,
+ MAX_INPUTS_PER_PAGE: 100,
+ MAX_FORMS_PER_PAGE: 30,
+ MAX_INPUTS_PER_FORM: 80,
+ MAX_FORM_MUT_OBS_COUNT: 50
};
exports.constants = constants;
diff --git a/integration-test/helpers/pages.js b/integration-test/helpers/pages.js
index 884f71bb1..c1235422c 100644
--- a/integration-test/helpers/pages.js
+++ b/integration-test/helpers/pages.js
@@ -678,7 +678,7 @@ export function emailAutofillPage (page) {
expect(input).toHaveAttribute(ATTR_AUTOFILL, 'true')
}
async assertDaxIconIsHidden ({ checking = 'autofill' } = {}) {
- const input = await page.locator(selectors.identity)
+ const input = await page.getByLabel('Email')
if (checking === 'style') {
const style = await input.getAttribute('style')
expect(style).toBeFalsy()
diff --git a/src/Form/Form.js b/src/Form/Form.js
index 6cefc81f3..3e9eed099 100644
--- a/src/Form/Form.js
+++ b/src/Form/Form.js
@@ -23,7 +23,12 @@ import {
} from './formatters.js'
import {constants} from '../constants.js'
-const {ATTR_AUTOFILL, ATTR_INPUT_TYPE} = constants
+const {
+ ATTR_AUTOFILL,
+ ATTR_INPUT_TYPE,
+ MAX_FORM_MUT_OBS_COUNT,
+ MAX_INPUTS_PER_FORM
+} = constants
class Form {
/** @type {import("../Form/matching").Matching} */
@@ -72,6 +77,7 @@ class Form {
}
})
+ this.mutObsCount = 0
this.mutObsConfig = { childList: true, subtree: true }
this.mutObs = new MutationObserver(
(records) => {
@@ -84,6 +90,12 @@ class Form {
this.formAnalyzer = new FormAnalyzer(this.form, input, this.matching)
this.recategorizeAllInputs()
})
+
+ this.mutObsCount++
+ // If the form mutates too much, disconnect to avoid performance issues
+ if (this.mutObsCount >= MAX_FORM_MUT_OBS_COUNT) {
+ this.mutObs.disconnect()
+ }
}
}
}
@@ -290,6 +302,7 @@ class Form {
removeInlineStyles(input, getIconStylesBase(input, this))
removeInlineStyles(input, getIconStylesAlternate(input, this))
input.removeAttribute(ATTR_AUTOFILL)
+ input.removeAttribute(ATTR_INPUT_TYPE)
}
removeAllDecorations () {
this.execOnInputs((input) => this.removeInputDecoration(input))
@@ -309,6 +322,7 @@ class Form {
*/
forgetAllInputs () {
this.execOnInputs((input) => {
+ input.removeAttribute(ATTR_AUTOFILL)
input.removeAttribute(ATTR_INPUT_TYPE)
})
Object.values(this.inputs).forEach((inputSet) => inputSet.clear())
@@ -348,7 +362,12 @@ class Form {
if (this.form.matches(selector)) {
this.addInput(this.form)
} else {
- this.form.querySelectorAll(selector).forEach(input => this.addInput(input))
+ const foundInputs = this.form.querySelectorAll(selector)
+ if (foundInputs.length < MAX_INPUTS_PER_FORM) {
+ foundInputs.forEach(input => this.addInput(input))
+ } else {
+ console.log('The form has too many inputs, bailing.')
+ }
}
this.initialScanComplete = true
}
@@ -406,6 +425,13 @@ class Form {
addInput (input) {
if (this.inputs.all.has(input)) return this
+ // If the form has too many inputs, destroy everything to avoid performance issues
+ if (this.inputs.all.size > MAX_INPUTS_PER_FORM) {
+ console.log('The form has too many inputs, destroying.')
+ this.destroy()
+ return this
+ }
+
// When new inputs are added after the initial scan, reanalyze the whole form
if (this.initialScanComplete) {
this.formAnalyzer = new FormAnalyzer(this.form, input, this.matching)
diff --git a/src/Form/Form.test.js b/src/Form/Form.test.js
index 42e66ebcc..fb7e06c86 100644
--- a/src/Form/Form.test.js
+++ b/src/Form/Form.test.js
@@ -1,6 +1,7 @@
import InterfacePrototype from '../DeviceInterface/InterfacePrototype.js'
import { createScanner } from '../Scanner.js'
import {attachAndReturnGenericForm} from '../test-utils.js'
+import {constants} from '../constants.js'
afterEach(() => {
document.body.innerHTML = ''
@@ -374,3 +375,43 @@ describe('Attempt form submission when needed', () => {
})
})
})
+
+describe('Form bails', () => {
+ beforeEach(() => {
+ document.body.innerHTML = ''
+ })
+ test('when it has too many fields', async () => {
+ const formEl = attachAndReturnGenericForm()
+ for (let i = 0; i <= constants.MAX_INPUTS_PER_FORM + 10; i++) {
+ const input = document.createElement('input')
+ input.type = 'email'
+ input.placeholder = 'Email address'
+ formEl.appendChild(input)
+ }
+
+ createScanner(InterfacePrototype.default()).findEligibleInputs(document)
+ const decoratedInputs = document.querySelectorAll(`[${constants.ATTR_INPUT_TYPE}]`)
+ expect(decoratedInputs).toHaveLength(0)
+ })
+ test('when too many fields are added after the initial scan', async () => {
+ const formEl = attachAndReturnGenericForm()
+
+ const scanner = createScanner(InterfacePrototype.default()).findEligibleInputs(document)
+ let decoratedInputs = document.querySelectorAll(`[${constants.ATTR_INPUT_TYPE}]`)
+ expect(decoratedInputs).toHaveLength(2)
+
+ const newInputs = []
+ for (let i = 0; i <= constants.MAX_INPUTS_PER_FORM + 10; i++) {
+ const input = document.createElement('input')
+ input.type = 'email'
+ input.placeholder = 'Email address'
+ newInputs.push(input)
+ }
+ formEl.append(...newInputs)
+ // Scan right away without waiting for the queue
+ scanner.findEligibleInputs(formEl)
+
+ decoratedInputs = document.querySelectorAll(`[${constants.ATTR_INPUT_TYPE}]`)
+ expect(decoratedInputs).toHaveLength(0)
+ })
+})
diff --git a/src/Scanner.js b/src/Scanner.js
index 2ee90374a..dc05439f3 100644
--- a/src/Scanner.js
+++ b/src/Scanner.js
@@ -1,21 +1,31 @@
import { Form } from './Form/Form.js'
import { SUBMIT_BUTTON_SELECTOR, FORM_INPUTS_SELECTOR } from './Form/selectors-css.js'
+import { constants } from './constants.js'
import { createMatching } from './Form/matching.js'
import {isFormLikelyToBeUsedAsPageWrapper} from './autofill-utils.js'
+const {
+ MAX_INPUTS_PER_PAGE,
+ MAX_FORMS_PER_PAGE,
+ MAX_INPUTS_PER_FORM
+} = constants
+
/**
* @typedef {{
* forms: Map;
* init(): ()=> void;
* enqueue(elements: (HTMLElement|Document)[]): void;
* findEligibleInputs(context): Scanner;
+ * options: ScannerOptions;
* }} Scanner
*
* @typedef {{
* initialDelay: number,
* bufferSize: number,
* debounceTimePeriod: number,
- * maxInputsOnPage: number,
+ * maxInputsPerPage: number,
+ * maxFormsPerPage: number,
+ * maxInputsPerForm: number
* }} ScannerOptions
*/
@@ -32,7 +42,9 @@ const defaultScannerOptions = {
// How many inputs is too many on the page. If we detect that there's above
// this maximum, then we don't scan the page. This will prevent slowdowns on
// large pages which are unlikely to require autofill anyway.
- maxInputsOnPage: 100
+ maxInputsPerPage: MAX_INPUTS_PER_PAGE,
+ maxFormsPerPage: MAX_FORMS_PER_PAGE,
+ maxInputsPerForm: MAX_INPUTS_PER_FORM
}
/**
@@ -131,7 +143,7 @@ class DefaultScanner {
this.addInput(context)
} else {
const inputs = context.querySelectorAll(FORM_INPUTS_SELECTOR)
- if (inputs.length > this.options.maxInputsOnPage) {
+ if (inputs.length > this.options.maxInputsPerPage) {
return this
}
inputs.forEach((input) => this.addInput(input))
@@ -200,7 +212,12 @@ class DefaultScanner {
this.forms.delete(childForm)
}
- this.forms.set(parentForm, new Form(parentForm, input, this.device, this.matching, this.shouldAutoprompt))
+ // Only add the form if below the limit of forms per page
+ if (this.forms.size < this.options.maxFormsPerPage) {
+ this.forms.set(parentForm, new Form(parentForm, input, this.device, this.matching, this.shouldAutoprompt))
+ } else {
+ console.log('The page has too many forms, stop adding them.')
+ }
}
}
diff --git a/src/Scanner.test.js b/src/Scanner.test.js
index 87eb6b889..79fbfd98b 100644
--- a/src/Scanner.test.js
+++ b/src/Scanner.test.js
@@ -48,7 +48,7 @@ describe('performance', () => {
})
it('should not scan if above maximum inputs', () => {
const scanner = createScanner(InterfacePrototype.default(), {
- maxInputsOnPage: 3
+ maxInputsPerPage: 3
})
scanner.findEligibleInputs(document)
@@ -59,7 +59,7 @@ describe('performance', () => {
})
it('should stop scanning if page grows above maximum inputs', () => {
const scanner = createScanner(InterfacePrototype.default(), {
- maxInputsOnPage: 5,
+ maxInputsPerPage: 5,
bufferSize: 2
})
@@ -77,6 +77,21 @@ describe('performance', () => {
inputs.forEach(input => scanner.enqueue([input]))
jest.advanceTimersByTime(1000)
+ // Confirm that newly added inputs are not scanned
+ expect(document.body).toMatchSnapshot()
+ })
+ it('should stop scanning if page grows above maximum forms', () => {
+ const scanner = createScanner(InterfacePrototype.default(), {
+ maxFormsPerPage: 1
+ })
+ const form = document.querySelector('form')
+ const formClone = form?.cloneNode(true)
+ if (!formClone) throw new Error('unreachable')
+ document.body.appendChild(formClone)
+
+ scanner.findEligibleInputs(document)
+ jest.advanceTimersByTime(1000)
+
// Confirm that newly added inputs are not scanned
expect(document.body).toMatchSnapshot()
})
diff --git a/src/__snapshots__/Scanner.test.js.snap b/src/__snapshots__/Scanner.test.js.snap
index 503e92382..100c91d5e 100644
--- a/src/__snapshots__/Scanner.test.js.snap
+++ b/src/__snapshots__/Scanner.test.js.snap
@@ -199,6 +199,130 @@ exports[`performance should not scan if above maximum inputs 1`] = `
+
+
+
+
+
diff --git a/src/constants.js b/src/constants.js
index 1b6ba596f..25bc641ec 100644
--- a/src/constants.js
+++ b/src/constants.js
@@ -1,5 +1,9 @@
export const constants = {
ATTR_INPUT_TYPE: 'data-ddg-inputType',
ATTR_AUTOFILL: 'data-ddg-autofill',
- TEXT_LENGTH_CUTOFF: 50
+ TEXT_LENGTH_CUTOFF: 50,
+ MAX_INPUTS_PER_PAGE: 100,
+ MAX_FORMS_PER_PAGE: 30,
+ MAX_INPUTS_PER_FORM: 80,
+ MAX_FORM_MUT_OBS_COUNT: 50
}
diff --git a/swift-package/Resources/assets/autofill-debug.js b/swift-package/Resources/assets/autofill-debug.js
index 4f2429514..a33be6fe9 100644
--- a/swift-package/Resources/assets/autofill-debug.js
+++ b/swift-package/Resources/assets/autofill-debug.js
@@ -10084,7 +10084,9 @@ function _defineProperty(obj, key, value) { if (key in obj) { Object.definePrope
const {
ATTR_AUTOFILL,
- ATTR_INPUT_TYPE
+ ATTR_INPUT_TYPE,
+ MAX_FORM_MUT_OBS_COUNT,
+ MAX_INPUTS_PER_FORM
} = _constants.constants;
class Form {
@@ -10094,8 +10096,6 @@ class Form {
/** @type {HTMLInputElement | null} */
- /** @type {boolean | null} */
-
/**
* @param {HTMLElement} form
* @param {HTMLInputElement|HTMLSelectElement} input
@@ -10112,14 +10112,9 @@ class Form {
_defineProperty(this, "activeInput", void 0);
- _defineProperty(this, "isSignup", void 0);
-
this.form = form;
this.matching = matching || (0, _matching.createMatching)();
this.formAnalyzer = new _FormAnalyzer.default(form, input, matching);
- this.isLogin = this.formAnalyzer.isLogin;
- this.isSignup = this.formAnalyzer.isSignup;
- this.isHybrid = this.formAnalyzer.isHybrid;
this.device = deviceInterface;
/** @type Record<'all' | SupportedMainTypes, Set> */
@@ -10146,6 +10141,30 @@ class Form {
for (const entry of entries) {
if (!entry.isIntersecting) this.removeTooltip();
}
+ });
+ this.mutObsCount = 0;
+ this.mutObsConfig = {
+ childList: true,
+ subtree: true
+ };
+ this.mutObs = new MutationObserver(records => {
+ const anythingRemoved = records.some(record => record.removedNodes.length > 0);
+
+ if (anythingRemoved) {
+ // Must check for inputs because a parent may be removed and not show up in record.removedNodes
+ if ([...this.inputs.all].some(input => !input.isConnected)) {
+ // If any known input has been removed from the DOM, reanalyze the whole form
+ window.requestIdleCallback(() => {
+ this.formAnalyzer = new _FormAnalyzer.default(this.form, input, this.matching);
+ this.recategorizeAllInputs();
+ });
+ this.mutObsCount++; // If the form mutates too much, disconnect to avoid performance issues
+
+ if (this.mutObsCount >= MAX_FORM_MUT_OBS_COUNT) {
+ this.mutObs.disconnect();
+ }
+ }
+ }
}); // This ensures we fire the handler again if the form is changed
this.addListener(form, 'input', () => {
@@ -10155,6 +10174,7 @@ class Form {
}
});
this.categorizeInputs();
+ this.mutObs.observe(this.form, this.mutObsConfig);
this.logFormInfo();
if (shouldAutoprompt) {
@@ -10162,6 +10182,18 @@ class Form {
}
}
+ get isLogin() {
+ return this.formAnalyzer.isLogin;
+ }
+
+ get isSignup() {
+ return this.formAnalyzer.isSignup;
+ }
+
+ get isHybrid() {
+ return this.formAnalyzer.isHybrid;
+ }
+
logFormInfo() {
if (!(0, _autofillUtils.shouldLog)()) return;
console.log("Form type: %c".concat(this.getFormType()), 'font-weight: bold');
@@ -10353,6 +10385,7 @@ class Form {
(0, _autofillUtils.removeInlineStyles)(input, (0, _inputStyles.getIconStylesBase)(input, this));
(0, _autofillUtils.removeInlineStyles)(input, (0, _inputStyles.getIconStylesAlternate)(input, this));
input.removeAttribute(ATTR_AUTOFILL);
+ input.removeAttribute(ATTR_INPUT_TYPE);
}
removeAllDecorations() {
@@ -10383,6 +10416,7 @@ class Form {
forgetAllInputs() {
this.execOnInputs(input => {
+ input.removeAttribute(ATTR_AUTOFILL);
input.removeAttribute(ATTR_INPUT_TYPE);
});
Object.values(this.inputs).forEach(inputSet => inputSet.clear());
@@ -10393,6 +10427,7 @@ class Form {
recategorizeAllInputs() {
+ this.initialScanComplete = false;
this.removeAllDecorations();
this.forgetAllInputs();
this.categorizeInputs();
@@ -10415,6 +10450,7 @@ class Form {
destroy() {
this.removeAllDecorations();
this.removeTooltip();
+ this.mutObs.disconnect();
this.matching.clear();
this.intObs = null;
}
@@ -10425,8 +10461,16 @@ class Form {
if (this.form.matches(selector)) {
this.addInput(this.form);
} else {
- this.form.querySelectorAll(selector).forEach(input => this.addInput(input));
+ const foundInputs = this.form.querySelectorAll(selector);
+
+ if (foundInputs.length < MAX_INPUTS_PER_FORM) {
+ foundInputs.forEach(input => this.addInput(input));
+ } else {
+ console.log('The form has too many inputs, bailing.');
+ }
}
+
+ this.initialScanComplete = true;
}
get submitButtons() {
@@ -10482,9 +10526,23 @@ class Form {
addInput(input) {
var _this$device$settings;
- // Nothing to do with 1-character fields
+ if (this.inputs.all.has(input)) return this; // If the form has too many inputs, destroy everything to avoid performance issues
+
+ if (this.inputs.all.size > MAX_INPUTS_PER_FORM) {
+ console.log('The form has too many inputs, destroying.');
+ this.destroy();
+ return this;
+ } // When new inputs are added after the initial scan, reanalyze the whole form
+
+
+ if (this.initialScanComplete) {
+ this.formAnalyzer = new _FormAnalyzer.default(this.form, input, this.matching);
+ this.recategorizeAllInputs();
+ return this;
+ } // Nothing to do with 1-character fields
+
+
if (input.maxLength === 1) return this;
- if (this.inputs.all.has(input)) return this;
this.inputs.all.add(input);
const opts = {
isLogin: this.isLogin,
@@ -10891,6 +10949,8 @@ const loginRegex = new RegExp(/sign(ing)?.?in(?!g)|log.?(i|o)n|log.?out|unsubscr
const signupRegex = new RegExp(/sign(ing)?.?up|join|\bregist(er|ration)|newsletter|\bsubscri(be|ption)|contact|create|start|enroll|settings|preferences|profile|update|checkout|guest|purchase|buy|order|schedule|estimate|request|new.?customer|(confirm|retype|repeat) password|password confirm?/i);
const conservativeSignupRegex = new RegExp(/sign.?up|join|register|enroll|newsletter|subscri(be|ption)|settings|preferences|profile|update/i);
const strictSignupRegex = new RegExp(/sign.?up|join|register|(create|new).+account|enroll|settings|preferences|profile|update/i);
+const resetPasswordLink = new RegExp(/(forgot(ten)?|reset|don't remember) (your )?password|password forgotten/i);
+const loginProvidersRegex = new RegExp(/ with /i);
class FormAnalyzer {
/** @type HTMLElement */
@@ -11013,7 +11073,7 @@ class FormAnalyzer {
shouldCheckUnifiedForm = false,
shouldBeConservative = false
} = _ref;
- const matchesLogin = /current.?password/i.test(string) || loginRegex.test(string); // Check explicitly for unified login/signup forms
+ const matchesLogin = /current.?password/i.test(string) || loginRegex.test(string) || resetPasswordLink.test(string); // Check explicitly for unified login/signup forms
if (shouldCheckUnifiedForm && matchesLogin && strictSignupRegex.test(string)) {
this.increaseHybridSignal(strength, signalType);
@@ -11128,10 +11188,11 @@ class FormAnalyzer {
if (el instanceof HTMLAnchorElement && el.href && el.getAttribute('href') !== '#' || (el.getAttribute('role') || '').toUpperCase() === 'LINK' || el.matches('button[class*=secondary]')) {
- // Unless it's a forgotten password link, we don't flip those links
let shouldFlip = true;
- if (/(forgot(ten)?|reset) (your )?password|password forgotten| with /i.test(string)) {
+ if (resetPasswordLink.test(string) || // Don't flip forgotten password links
+ loginProvidersRegex.test(string) // Don't flip login providers links
+ ) {
shouldFlip = false;
}
@@ -15102,29 +15163,40 @@ var _Form = require("./Form/Form.js");
var _selectorsCss = require("./Form/selectors-css.js");
+var _constants = require("./constants.js");
+
var _matching = require("./Form/matching.js");
var _autofillUtils = require("./autofill-utils.js");
function _defineProperty(obj, key, value) { if (key in obj) { Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); } else { obj[key] = value; } return obj; }
+const {
+ MAX_INPUTS_PER_PAGE,
+ MAX_FORMS_PER_PAGE,
+ MAX_INPUTS_PER_FORM
+} = _constants.constants;
/**
* @typedef {{
* forms: Map;
* init(): ()=> void;
* enqueue(elements: (HTMLElement|Document)[]): void;
* findEligibleInputs(context): Scanner;
+ * options: ScannerOptions;
* }} Scanner
*
* @typedef {{
* initialDelay: number,
* bufferSize: number,
* debounceTimePeriod: number,
- * maxInputsOnPage: number,
+ * maxInputsPerPage: number,
+ * maxFormsPerPage: number,
+ * maxInputsPerForm: number
* }} ScannerOptions
*/
/** @type {ScannerOptions} */
+
const defaultScannerOptions = {
// This buffer size is very large because it's an unexpected edge-case that
// a DOM will be continually modified over and over without ever stopping. If we do see 1000 unique
@@ -15137,7 +15209,9 @@ const defaultScannerOptions = {
// How many inputs is too many on the page. If we detect that there's above
// this maximum, then we don't scan the page. This will prevent slowdowns on
// large pages which are unlikely to require autofill anyway.
- maxInputsOnPage: 100
+ maxInputsPerPage: MAX_INPUTS_PER_PAGE,
+ maxFormsPerPage: MAX_FORMS_PER_PAGE,
+ maxInputsPerForm: MAX_INPUTS_PER_FORM
};
/**
* This allows:
@@ -15283,7 +15357,7 @@ class DefaultScanner {
} else {
const inputs = context.querySelectorAll(_selectorsCss.FORM_INPUTS_SELECTOR);
- if (inputs.length > this.options.maxInputsOnPage) {
+ if (inputs.length > this.options.maxInputsPerPage) {
return this;
}
@@ -15360,9 +15434,14 @@ class DefaultScanner {
(_this$forms$get2 = this.forms.get(childForm)) === null || _this$forms$get2 === void 0 ? void 0 : _this$forms$get2.destroy();
this.forms.delete(childForm);
- }
+ } // Only add the form if below the limit of forms per page
+
- this.forms.set(parentForm, new _Form.Form(parentForm, input, this.device, this.matching, this.shouldAutoprompt));
+ if (this.forms.size < this.options.maxFormsPerPage) {
+ this.forms.set(parentForm, new _Form.Form(parentForm, input, this.device, this.matching, this.shouldAutoprompt));
+ } else {
+ console.log('The page has too many forms, stop adding them.');
+ }
}
}
/**
@@ -15430,7 +15509,7 @@ function createScanner(device, scannerOptions) {
});
}
-},{"./Form/Form.js":33,"./Form/matching.js":43,"./Form/selectors-css.js":44,"./autofill-utils.js":63}],52:[function(require,module,exports){
+},{"./Form/Form.js":33,"./Form/matching.js":43,"./Form/selectors-css.js":44,"./autofill-utils.js":63,"./constants.js":66}],52:[function(require,module,exports){
"use strict";
Object.defineProperty(exports, "__esModule", {
@@ -17948,6 +18027,7 @@ const isLikelyASubmitButton = el => {
const dataTestId = el.getAttribute('data-test-id') || '';
const contentExcludingLabel = text + ' ' + title + ' ' + value;
return (el.getAttribute('type') === 'submit' || // is explicitly set as "submit"
+ el.getAttribute('name') === 'submit' || // is called "submit"
/primary|submit/i.test(el.className) || // has high-signal submit classes
/submit/i.test(dataTestId) || SUBMIT_BUTTON_REGEX.test(contentExcludingLabel) || // has high-signal text
el.offsetHeight * el.offsetWidth >= 10000 && !/secondary/i.test(el.className) // it's a large element 250x40px
@@ -18246,7 +18326,11 @@ exports.constants = void 0;
const constants = {
ATTR_INPUT_TYPE: 'data-ddg-inputType',
ATTR_AUTOFILL: 'data-ddg-autofill',
- TEXT_LENGTH_CUTOFF: 50
+ TEXT_LENGTH_CUTOFF: 50,
+ MAX_INPUTS_PER_PAGE: 100,
+ MAX_FORMS_PER_PAGE: 30,
+ MAX_INPUTS_PER_FORM: 80,
+ MAX_FORM_MUT_OBS_COUNT: 50
};
exports.constants = constants;
diff --git a/swift-package/Resources/assets/autofill.js b/swift-package/Resources/assets/autofill.js
index 04a70fc9a..628a7c963 100644
--- a/swift-package/Resources/assets/autofill.js
+++ b/swift-package/Resources/assets/autofill.js
@@ -6408,7 +6408,9 @@ function _defineProperty(obj, key, value) { if (key in obj) { Object.definePrope
const {
ATTR_AUTOFILL,
- ATTR_INPUT_TYPE
+ ATTR_INPUT_TYPE,
+ MAX_FORM_MUT_OBS_COUNT,
+ MAX_INPUTS_PER_FORM
} = _constants.constants;
class Form {
@@ -6418,8 +6420,6 @@ class Form {
/** @type {HTMLInputElement | null} */
- /** @type {boolean | null} */
-
/**
* @param {HTMLElement} form
* @param {HTMLInputElement|HTMLSelectElement} input
@@ -6436,14 +6436,9 @@ class Form {
_defineProperty(this, "activeInput", void 0);
- _defineProperty(this, "isSignup", void 0);
-
this.form = form;
this.matching = matching || (0, _matching.createMatching)();
this.formAnalyzer = new _FormAnalyzer.default(form, input, matching);
- this.isLogin = this.formAnalyzer.isLogin;
- this.isSignup = this.formAnalyzer.isSignup;
- this.isHybrid = this.formAnalyzer.isHybrid;
this.device = deviceInterface;
/** @type Record<'all' | SupportedMainTypes, Set> */
@@ -6470,6 +6465,30 @@ class Form {
for (const entry of entries) {
if (!entry.isIntersecting) this.removeTooltip();
}
+ });
+ this.mutObsCount = 0;
+ this.mutObsConfig = {
+ childList: true,
+ subtree: true
+ };
+ this.mutObs = new MutationObserver(records => {
+ const anythingRemoved = records.some(record => record.removedNodes.length > 0);
+
+ if (anythingRemoved) {
+ // Must check for inputs because a parent may be removed and not show up in record.removedNodes
+ if ([...this.inputs.all].some(input => !input.isConnected)) {
+ // If any known input has been removed from the DOM, reanalyze the whole form
+ window.requestIdleCallback(() => {
+ this.formAnalyzer = new _FormAnalyzer.default(this.form, input, this.matching);
+ this.recategorizeAllInputs();
+ });
+ this.mutObsCount++; // If the form mutates too much, disconnect to avoid performance issues
+
+ if (this.mutObsCount >= MAX_FORM_MUT_OBS_COUNT) {
+ this.mutObs.disconnect();
+ }
+ }
+ }
}); // This ensures we fire the handler again if the form is changed
this.addListener(form, 'input', () => {
@@ -6479,6 +6498,7 @@ class Form {
}
});
this.categorizeInputs();
+ this.mutObs.observe(this.form, this.mutObsConfig);
this.logFormInfo();
if (shouldAutoprompt) {
@@ -6486,6 +6506,18 @@ class Form {
}
}
+ get isLogin() {
+ return this.formAnalyzer.isLogin;
+ }
+
+ get isSignup() {
+ return this.formAnalyzer.isSignup;
+ }
+
+ get isHybrid() {
+ return this.formAnalyzer.isHybrid;
+ }
+
logFormInfo() {
if (!(0, _autofillUtils.shouldLog)()) return;
console.log("Form type: %c".concat(this.getFormType()), 'font-weight: bold');
@@ -6677,6 +6709,7 @@ class Form {
(0, _autofillUtils.removeInlineStyles)(input, (0, _inputStyles.getIconStylesBase)(input, this));
(0, _autofillUtils.removeInlineStyles)(input, (0, _inputStyles.getIconStylesAlternate)(input, this));
input.removeAttribute(ATTR_AUTOFILL);
+ input.removeAttribute(ATTR_INPUT_TYPE);
}
removeAllDecorations() {
@@ -6707,6 +6740,7 @@ class Form {
forgetAllInputs() {
this.execOnInputs(input => {
+ input.removeAttribute(ATTR_AUTOFILL);
input.removeAttribute(ATTR_INPUT_TYPE);
});
Object.values(this.inputs).forEach(inputSet => inputSet.clear());
@@ -6717,6 +6751,7 @@ class Form {
recategorizeAllInputs() {
+ this.initialScanComplete = false;
this.removeAllDecorations();
this.forgetAllInputs();
this.categorizeInputs();
@@ -6739,6 +6774,7 @@ class Form {
destroy() {
this.removeAllDecorations();
this.removeTooltip();
+ this.mutObs.disconnect();
this.matching.clear();
this.intObs = null;
}
@@ -6749,8 +6785,16 @@ class Form {
if (this.form.matches(selector)) {
this.addInput(this.form);
} else {
- this.form.querySelectorAll(selector).forEach(input => this.addInput(input));
+ const foundInputs = this.form.querySelectorAll(selector);
+
+ if (foundInputs.length < MAX_INPUTS_PER_FORM) {
+ foundInputs.forEach(input => this.addInput(input));
+ } else {
+ console.log('The form has too many inputs, bailing.');
+ }
}
+
+ this.initialScanComplete = true;
}
get submitButtons() {
@@ -6806,9 +6850,23 @@ class Form {
addInput(input) {
var _this$device$settings;
- // Nothing to do with 1-character fields
+ if (this.inputs.all.has(input)) return this; // If the form has too many inputs, destroy everything to avoid performance issues
+
+ if (this.inputs.all.size > MAX_INPUTS_PER_FORM) {
+ console.log('The form has too many inputs, destroying.');
+ this.destroy();
+ return this;
+ } // When new inputs are added after the initial scan, reanalyze the whole form
+
+
+ if (this.initialScanComplete) {
+ this.formAnalyzer = new _FormAnalyzer.default(this.form, input, this.matching);
+ this.recategorizeAllInputs();
+ return this;
+ } // Nothing to do with 1-character fields
+
+
if (input.maxLength === 1) return this;
- if (this.inputs.all.has(input)) return this;
this.inputs.all.add(input);
const opts = {
isLogin: this.isLogin,
@@ -7215,6 +7273,8 @@ const loginRegex = new RegExp(/sign(ing)?.?in(?!g)|log.?(i|o)n|log.?out|unsubscr
const signupRegex = new RegExp(/sign(ing)?.?up|join|\bregist(er|ration)|newsletter|\bsubscri(be|ption)|contact|create|start|enroll|settings|preferences|profile|update|checkout|guest|purchase|buy|order|schedule|estimate|request|new.?customer|(confirm|retype|repeat) password|password confirm?/i);
const conservativeSignupRegex = new RegExp(/sign.?up|join|register|enroll|newsletter|subscri(be|ption)|settings|preferences|profile|update/i);
const strictSignupRegex = new RegExp(/sign.?up|join|register|(create|new).+account|enroll|settings|preferences|profile|update/i);
+const resetPasswordLink = new RegExp(/(forgot(ten)?|reset|don't remember) (your )?password|password forgotten/i);
+const loginProvidersRegex = new RegExp(/ with /i);
class FormAnalyzer {
/** @type HTMLElement */
@@ -7337,7 +7397,7 @@ class FormAnalyzer {
shouldCheckUnifiedForm = false,
shouldBeConservative = false
} = _ref;
- const matchesLogin = /current.?password/i.test(string) || loginRegex.test(string); // Check explicitly for unified login/signup forms
+ const matchesLogin = /current.?password/i.test(string) || loginRegex.test(string) || resetPasswordLink.test(string); // Check explicitly for unified login/signup forms
if (shouldCheckUnifiedForm && matchesLogin && strictSignupRegex.test(string)) {
this.increaseHybridSignal(strength, signalType);
@@ -7452,10 +7512,11 @@ class FormAnalyzer {
if (el instanceof HTMLAnchorElement && el.href && el.getAttribute('href') !== '#' || (el.getAttribute('role') || '').toUpperCase() === 'LINK' || el.matches('button[class*=secondary]')) {
- // Unless it's a forgotten password link, we don't flip those links
let shouldFlip = true;
- if (/(forgot(ten)?|reset) (your )?password|password forgotten| with /i.test(string)) {
+ if (resetPasswordLink.test(string) || // Don't flip forgotten password links
+ loginProvidersRegex.test(string) // Don't flip login providers links
+ ) {
shouldFlip = false;
}
@@ -11426,29 +11487,40 @@ var _Form = require("./Form/Form.js");
var _selectorsCss = require("./Form/selectors-css.js");
+var _constants = require("./constants.js");
+
var _matching = require("./Form/matching.js");
var _autofillUtils = require("./autofill-utils.js");
function _defineProperty(obj, key, value) { if (key in obj) { Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); } else { obj[key] = value; } return obj; }
+const {
+ MAX_INPUTS_PER_PAGE,
+ MAX_FORMS_PER_PAGE,
+ MAX_INPUTS_PER_FORM
+} = _constants.constants;
/**
* @typedef {{
* forms: Map;
* init(): ()=> void;
* enqueue(elements: (HTMLElement|Document)[]): void;
* findEligibleInputs(context): Scanner;
+ * options: ScannerOptions;
* }} Scanner
*
* @typedef {{
* initialDelay: number,
* bufferSize: number,
* debounceTimePeriod: number,
- * maxInputsOnPage: number,
+ * maxInputsPerPage: number,
+ * maxFormsPerPage: number,
+ * maxInputsPerForm: number
* }} ScannerOptions
*/
/** @type {ScannerOptions} */
+
const defaultScannerOptions = {
// This buffer size is very large because it's an unexpected edge-case that
// a DOM will be continually modified over and over without ever stopping. If we do see 1000 unique
@@ -11461,7 +11533,9 @@ const defaultScannerOptions = {
// How many inputs is too many on the page. If we detect that there's above
// this maximum, then we don't scan the page. This will prevent slowdowns on
// large pages which are unlikely to require autofill anyway.
- maxInputsOnPage: 100
+ maxInputsPerPage: MAX_INPUTS_PER_PAGE,
+ maxFormsPerPage: MAX_FORMS_PER_PAGE,
+ maxInputsPerForm: MAX_INPUTS_PER_FORM
};
/**
* This allows:
@@ -11607,7 +11681,7 @@ class DefaultScanner {
} else {
const inputs = context.querySelectorAll(_selectorsCss.FORM_INPUTS_SELECTOR);
- if (inputs.length > this.options.maxInputsOnPage) {
+ if (inputs.length > this.options.maxInputsPerPage) {
return this;
}
@@ -11684,9 +11758,14 @@ class DefaultScanner {
(_this$forms$get2 = this.forms.get(childForm)) === null || _this$forms$get2 === void 0 ? void 0 : _this$forms$get2.destroy();
this.forms.delete(childForm);
- }
+ } // Only add the form if below the limit of forms per page
+
- this.forms.set(parentForm, new _Form.Form(parentForm, input, this.device, this.matching, this.shouldAutoprompt));
+ if (this.forms.size < this.options.maxFormsPerPage) {
+ this.forms.set(parentForm, new _Form.Form(parentForm, input, this.device, this.matching, this.shouldAutoprompt));
+ } else {
+ console.log('The page has too many forms, stop adding them.');
+ }
}
}
/**
@@ -11754,7 +11833,7 @@ function createScanner(device, scannerOptions) {
});
}
-},{"./Form/Form.js":25,"./Form/matching.js":35,"./Form/selectors-css.js":36,"./autofill-utils.js":55}],44:[function(require,module,exports){
+},{"./Form/Form.js":25,"./Form/matching.js":35,"./Form/selectors-css.js":36,"./autofill-utils.js":55,"./constants.js":58}],44:[function(require,module,exports){
"use strict";
Object.defineProperty(exports, "__esModule", {
@@ -14272,6 +14351,7 @@ const isLikelyASubmitButton = el => {
const dataTestId = el.getAttribute('data-test-id') || '';
const contentExcludingLabel = text + ' ' + title + ' ' + value;
return (el.getAttribute('type') === 'submit' || // is explicitly set as "submit"
+ el.getAttribute('name') === 'submit' || // is called "submit"
/primary|submit/i.test(el.className) || // has high-signal submit classes
/submit/i.test(dataTestId) || SUBMIT_BUTTON_REGEX.test(contentExcludingLabel) || // has high-signal text
el.offsetHeight * el.offsetWidth >= 10000 && !/secondary/i.test(el.className) // it's a large element 250x40px
@@ -14570,7 +14650,11 @@ exports.constants = void 0;
const constants = {
ATTR_INPUT_TYPE: 'data-ddg-inputType',
ATTR_AUTOFILL: 'data-ddg-autofill',
- TEXT_LENGTH_CUTOFF: 50
+ TEXT_LENGTH_CUTOFF: 50,
+ MAX_INPUTS_PER_PAGE: 100,
+ MAX_FORMS_PER_PAGE: 30,
+ MAX_INPUTS_PER_FORM: 80,
+ MAX_FORM_MUT_OBS_COUNT: 50
};
exports.constants = constants;
From 12399f7c40659f6cf6d4b2879b01e5650370e5e9 Mon Sep 17 00:00:00 2001
From: Emanuele Feliziani
Date: Fri, 21 Jul 2023 13:03:44 +0200
Subject: [PATCH 5/6] Fix icon sizing
Signed-off-by: Emanuele Feliziani
---
dist/autofill-debug.js | 8 ++++----
dist/autofill.css | 4 ++--
dist/autofill.js | 8 ++++----
src/Form/inputStyles.js | 2 +-
src/UI/HTMLTooltip.js | 6 +++---
src/UI/styles/autofill-tooltip-styles.css | 4 ++--
swift-package/Resources/assets/autofill-debug.js | 8 ++++----
swift-package/Resources/assets/autofill.css | 4 ++--
swift-package/Resources/assets/autofill.js | 8 ++++----
9 files changed, 26 insertions(+), 26 deletions(-)
diff --git a/dist/autofill-debug.js b/dist/autofill-debug.js
index a33be6fe9..7486f3f0b 100644
--- a/dist/autofill-debug.js
+++ b/dist/autofill-debug.js
@@ -12221,7 +12221,7 @@ const getIcon = function (input, form) {
const getBasicStyles = (input, icon) => ({
// Height must be > 0 to account for fields initially hidden
- 'background-size': "auto ".concat(input.offsetHeight <= 30 && input.offsetHeight > 0 ? '100%' : '26px'),
+ 'background-size': "auto ".concat(input.offsetHeight <= 30 && input.offsetHeight > 0 ? '100%' : '24px'),
'background-position': 'center right',
'background-repeat': 'no-repeat',
'background-origin': 'content-box',
@@ -16183,8 +16183,8 @@ function _defineProperty(obj, key, value) { if (key in obj) { Object.definePrope
/** @type {HTMLTooltipOptions} */
const defaultOptions = {
wrapperClass: '',
- tooltipPositionClass: (top, left) => "\n .tooltip {\n transform: translate(".concat(left, "px, ").concat(top, "px) !important;\n }\n "),
- caretPositionClass: (top, left, isAboveInput) => "\n .tooltip--email__caret {\n ".concat(isAboveInput ? "transform: translate(".concat(left, "px, ").concat(top, "px) rotate(180deg); transform-origin: 16px !important;") : "transform: translate(".concat(left, "px, ").concat(top, "px) !important;"), "\n }"),
+ tooltipPositionClass: (top, left) => "\n .tooltip {\n transform: translate(".concat(Math.floor(left), "px, ").concat(Math.floor(top), "px) !important;\n }\n "),
+ caretPositionClass: (top, left, isAboveInput) => "\n .tooltip--email__caret {\n ".concat(isAboveInput ? "transform: translate(".concat(Math.floor(left), "px, ").concat(Math.floor(top), "px) rotate(180deg); transform-origin: 18px !important;") : "transform: translate(".concat(Math.floor(left), "px, ").concat(Math.floor(top), "px) !important;"), "\n }"),
css: ""),
setSize: undefined,
remove: () => {
@@ -17613,7 +17613,7 @@ Object.defineProperty(exports, "__esModule", {
value: true
});
exports.CSS_STYLES = void 0;
-const CSS_STYLES = ":root {\n color-scheme: light dark;\n}\n\n.wrapper *, .wrapper *::before, .wrapper *::after {\n box-sizing: border-box;\n}\n.wrapper {\n position: fixed;\n top: 0;\n left: 0;\n padding: 0;\n font-family: 'DDG_ProximaNova', 'Proxima Nova', system-ui, 'Segoe UI', 'Roboto', 'Oxygen', 'Ubuntu',\n 'Cantarell', 'Fira Sans', 'Droid Sans', 'Helvetica Neue', sans-serif;\n -webkit-font-smoothing: antialiased;\n z-index: 2147483647;\n}\n.wrapper--data {\n font-family: 'SF Pro Text', system-ui, 'Segoe UI', 'Roboto', 'Oxygen', 'Ubuntu',\n 'Cantarell', 'Fira Sans', 'Droid Sans', 'Helvetica Neue', sans-serif;\n}\n:not(.top-autofill) .tooltip {\n position: absolute;\n width: 300px;\n max-width: calc(100vw - 25px);\n transform: translate(-1000px, -1000px);\n z-index: 2147483647;\n}\n.tooltip--data, #topAutofill {\n background-color: rgba(242, 240, 240, 1);\n -webkit-backdrop-filter: blur(40px);\n backdrop-filter: blur(40px);\n}\n@media (prefers-color-scheme: dark) {\n .tooltip--data, #topAutofill {\n background: rgb(100, 98, 102, .9);\n }\n}\n.tooltip--data {\n padding: 6px;\n font-size: 13px;\n line-height: 14px;\n width: 315px;\n max-height: 290px;\n overflow-y: auto;\n}\n.top-autofill .tooltip--data {\n min-height: 100vh;\n}\n.tooltip--data.tooltip--incontext-signup {\n width: 360px;\n}\n:not(.top-autofill) .tooltip--data {\n top: 100%;\n left: 100%;\n border: 0.5px solid rgba(255, 255, 255, 0.2);\n border-radius: 6px;\n box-shadow: 0 10px 20px rgba(0, 0, 0, 0.32);\n}\n@media (prefers-color-scheme: dark) {\n :not(.top-autofill) .tooltip--data {\n border: 1px solid rgba(255, 255, 255, 0.2);\n }\n}\n:not(.top-autofill) .tooltip--email {\n top: calc(100% + 6px);\n right: calc(100% - 46px);\n padding: 8px;\n border: 1px solid #D0D0D0;\n border-radius: 10px;\n background-color: #FFFFFF;\n font-size: 14px;\n line-height: 1.3;\n color: #333333;\n box-shadow: 0 10px 20px rgba(0, 0, 0, 0.15);\n}\n.tooltip--email__caret {\n position: absolute;\n transform: translate(-1000px, -1000px);\n z-index: 2147483647;\n}\n.tooltip--email__caret::before,\n.tooltip--email__caret::after {\n content: \"\";\n width: 0;\n height: 0;\n border-left: 10px solid transparent;\n border-right: 10px solid transparent;\n display: block;\n border-bottom: 8px solid #D0D0D0;\n position: absolute;\n right: -26px;\n}\n.tooltip--email__caret::before {\n border-bottom-color: #D0D0D0;\n top: -1px;\n}\n.tooltip--email__caret::after {\n border-bottom-color: #FFFFFF;\n top: 0px;\n}\n\n/* Buttons */\n.tooltip__button {\n display: flex;\n width: 100%;\n padding: 8px 8px 8px 0px;\n font-family: inherit;\n color: inherit;\n background: transparent;\n border: none;\n border-radius: 6px;\n}\n.tooltip__button.currentFocus,\n.wrapper:not(.top-autofill) .tooltip__button:hover {\n background-color: #3969EF;\n color: #FFFFFF;\n}\n\n/* Data autofill tooltip specific */\n.tooltip__button--data {\n position: relative;\n min-height: 48px;\n flex-direction: row;\n justify-content: flex-start;\n font-size: inherit;\n font-weight: 500;\n line-height: 16px;\n text-align: left;\n border-radius: 3px;\n}\n.tooltip--data__item-container {\n max-height: 220px;\n overflow: auto;\n}\n.tooltip__button--data:first-child {\n margin-top: 0;\n}\n.tooltip__button--data:last-child {\n margin-bottom: 0;\n}\n.tooltip__button--data::before {\n content: '';\n flex-shrink: 0;\n display: block;\n width: 32px;\n height: 32px;\n margin: 0 8px;\n background-size: 20px 20px;\n background-repeat: no-repeat;\n background-position: center 6px;\n}\n#provider_locked::after {\n position: absolute;\n content: '';\n flex-shrink: 0;\n display: block;\n width: 32px;\n height: 32px;\n margin: 0 8px;\n background-size: 11px 13px;\n background-repeat: no-repeat;\n background-position: right bottom;\n}\n.tooltip__button--data.currentFocus:not(.tooltip__button--data--bitwarden)::before,\n.wrapper:not(.top-autofill) .tooltip__button--data:not(.tooltip__button--data--bitwarden):hover::before {\n filter: invert(100%);\n}\n@media (prefers-color-scheme: dark) {\n .tooltip__button--data:not(.tooltip__button--data--bitwarden)::before,\n .tooltip__button--data:not(.tooltip__button--data--bitwarden)::before {\n filter: invert(100%);\n opacity: .9;\n }\n}\n.tooltip__button__text-container {\n margin: auto 0;\n}\n.label {\n display: block;\n font-weight: 400;\n letter-spacing: -0.25px;\n color: rgba(0,0,0,.8);\n font-size: 13px;\n line-height: 1;\n}\n.label + .label {\n margin-top: 2px;\n}\n.label.label--medium {\n font-weight: 500;\n letter-spacing: -0.25px;\n color: rgba(0,0,0,.9);\n}\n.label.label--small {\n font-size: 11px;\n font-weight: 400;\n letter-spacing: 0.06px;\n color: rgba(0,0,0,0.6);\n}\n@media (prefers-color-scheme: dark) {\n .tooltip--data .label {\n color: #ffffff;\n }\n .tooltip--data .label--medium {\n color: #ffffff;\n }\n .tooltip--data .label--small {\n color: #cdcdcd;\n }\n}\n.tooltip__button.currentFocus .label,\n.wrapper:not(.top-autofill) .tooltip__button:hover .label {\n color: #FFFFFF;\n}\n\n.tooltip__button--manage {\n font-size: 13px;\n padding: 5px 9px;\n border-radius: 3px;\n margin: 0;\n}\n\n/* Icons */\n.tooltip__button--data--credentials::before {\n background-image: url('');\n}\n.tooltip__button--data--creditCards::before {\n background-image: url('');\n}\n.tooltip__button--data--identities::before {\n background-image: url('');\n}\n.tooltip__button--data--credentials.tooltip__button--data--bitwarden::before {\n background-image: url('');\n}\n#provider_locked:after {\n background-image: url('');\n}\n\nhr {\n display: block;\n margin: 5px 9px;\n border: none; /* reset the border */\n border-top: 1px solid rgba(0,0,0,.1);\n}\n\nhr:first-child {\n display: none;\n}\n\n@media (prefers-color-scheme: dark) {\n hr {\n border-top: 1px solid rgba(255,255,255,.2);\n }\n}\n\n#privateAddress {\n align-items: flex-start;\n}\n#personalAddress::before,\n#privateAddress::before,\n#incontextSignup::before,\n#personalAddress.currentFocus::before,\n#personalAddress:hover::before,\n#privateAddress.currentFocus::before,\n#privateAddress:hover::before {\n filter: none;\n /* This is the same icon as `daxBase64` in `src/Form/logo-svg.js` */\n background-image: url('');\n}\n\n/* Email tooltip specific */\n.tooltip__button--email {\n flex-direction: column;\n justify-content: center;\n align-items: flex-start;\n font-size: 14px;\n padding: 4px 8px;\n}\n.tooltip__button--email__primary-text {\n font-weight: bold;\n}\n.tooltip__button--email__secondary-text {\n font-size: 12px;\n}\n\n/* Email Protection signup notice */\n:not(.top-autofill) .tooltip--email-signup {\n text-align: left;\n color: #222222;\n padding: 16px 20px;\n width: 380px;\n}\n\n.tooltip--email-signup h1 {\n font-weight: 700;\n font-size: 16px;\n line-height: 1.5;\n margin: 0;\n}\n\n.tooltip--email-signup p {\n font-weight: 400;\n font-size: 14px;\n line-height: 1.4;\n}\n\n.notice-controls {\n display: flex;\n}\n\n.tooltip--email-signup .notice-controls > * {\n border-radius: 8px;\n border: 0;\n cursor: pointer;\n display: inline-block;\n font-family: inherit;\n font-style: normal;\n font-weight: bold;\n padding: 8px 12px;\n text-decoration: none;\n}\n\n.notice-controls .ghost {\n margin-left: 1rem;\n}\n\n.tooltip--email-signup a.primary {\n background: #3969EF;\n color: #fff;\n}\n\n.tooltip--email-signup a.primary:hover,\n.tooltip--email-signup a.primary:focus {\n background: #2b55ca;\n}\n\n.tooltip--email-signup a.primary:active {\n background: #1e42a4;\n}\n\n.tooltip--email-signup button.ghost {\n background: transparent;\n color: #3969EF;\n}\n\n.tooltip--email-signup button.ghost:hover,\n.tooltip--email-signup button.ghost:focus {\n background-color: rgba(0, 0, 0, 0.06);\n color: #2b55ca;\n}\n\n.tooltip--email-signup button.ghost:active {\n background-color: rgba(0, 0, 0, 0.12);\n color: #1e42a4;\n}\n\n.tooltip--email-signup button.close-tooltip {\n background-color: transparent;\n background-image: url();\n background-position: center center;\n background-repeat: no-repeat;\n border: 0;\n cursor: pointer;\n padding: 16px;\n position: absolute;\n right: 12px;\n top: 12px;\n}\n";
+const CSS_STYLES = ":root {\n color-scheme: light dark;\n}\n\n.wrapper *, .wrapper *::before, .wrapper *::after {\n box-sizing: border-box;\n}\n.wrapper {\n position: fixed;\n top: 0;\n left: 0;\n padding: 0;\n font-family: 'DDG_ProximaNova', 'Proxima Nova', system-ui, 'Segoe UI', 'Roboto', 'Oxygen', 'Ubuntu',\n 'Cantarell', 'Fira Sans', 'Droid Sans', 'Helvetica Neue', sans-serif;\n -webkit-font-smoothing: antialiased;\n z-index: 2147483647;\n}\n.wrapper--data {\n font-family: 'SF Pro Text', system-ui, 'Segoe UI', 'Roboto', 'Oxygen', 'Ubuntu',\n 'Cantarell', 'Fira Sans', 'Droid Sans', 'Helvetica Neue', sans-serif;\n}\n:not(.top-autofill) .tooltip {\n position: absolute;\n width: 300px;\n max-width: calc(100vw - 25px);\n transform: translate(-1000px, -1000px);\n z-index: 2147483647;\n}\n.tooltip--data, #topAutofill {\n background-color: rgba(242, 240, 240, 1);\n -webkit-backdrop-filter: blur(40px);\n backdrop-filter: blur(40px);\n}\n@media (prefers-color-scheme: dark) {\n .tooltip--data, #topAutofill {\n background: rgb(100, 98, 102, .9);\n }\n}\n.tooltip--data {\n padding: 6px;\n font-size: 13px;\n line-height: 14px;\n width: 315px;\n max-height: 290px;\n overflow-y: auto;\n}\n.top-autofill .tooltip--data {\n min-height: 100vh;\n}\n.tooltip--data.tooltip--incontext-signup {\n width: 360px;\n}\n:not(.top-autofill) .tooltip--data {\n top: 100%;\n left: 100%;\n border: 0.5px solid rgba(255, 255, 255, 0.2);\n border-radius: 6px;\n box-shadow: 0 10px 20px rgba(0, 0, 0, 0.32);\n}\n@media (prefers-color-scheme: dark) {\n :not(.top-autofill) .tooltip--data {\n border: 1px solid rgba(255, 255, 255, 0.2);\n }\n}\n:not(.top-autofill) .tooltip--email {\n top: calc(100% + 6px);\n right: calc(100% - 48px);\n padding: 8px;\n border: 1px solid #D0D0D0;\n border-radius: 10px;\n background-color: #FFFFFF;\n font-size: 14px;\n line-height: 1.3;\n color: #333333;\n box-shadow: 0 10px 20px rgba(0, 0, 0, 0.15);\n}\n.tooltip--email__caret {\n position: absolute;\n transform: translate(-1000px, -1000px);\n z-index: 2147483647;\n}\n.tooltip--email__caret::before,\n.tooltip--email__caret::after {\n content: \"\";\n width: 0;\n height: 0;\n border-left: 10px solid transparent;\n border-right: 10px solid transparent;\n display: block;\n border-bottom: 8px solid #D0D0D0;\n position: absolute;\n right: -28px;\n}\n.tooltip--email__caret::before {\n border-bottom-color: #D0D0D0;\n top: -1px;\n}\n.tooltip--email__caret::after {\n border-bottom-color: #FFFFFF;\n top: 0px;\n}\n\n/* Buttons */\n.tooltip__button {\n display: flex;\n width: 100%;\n padding: 8px 8px 8px 0px;\n font-family: inherit;\n color: inherit;\n background: transparent;\n border: none;\n border-radius: 6px;\n}\n.tooltip__button.currentFocus,\n.wrapper:not(.top-autofill) .tooltip__button:hover {\n background-color: #3969EF;\n color: #FFFFFF;\n}\n\n/* Data autofill tooltip specific */\n.tooltip__button--data {\n position: relative;\n min-height: 48px;\n flex-direction: row;\n justify-content: flex-start;\n font-size: inherit;\n font-weight: 500;\n line-height: 16px;\n text-align: left;\n border-radius: 3px;\n}\n.tooltip--data__item-container {\n max-height: 220px;\n overflow: auto;\n}\n.tooltip__button--data:first-child {\n margin-top: 0;\n}\n.tooltip__button--data:last-child {\n margin-bottom: 0;\n}\n.tooltip__button--data::before {\n content: '';\n flex-shrink: 0;\n display: block;\n width: 32px;\n height: 32px;\n margin: 0 8px;\n background-size: 20px 20px;\n background-repeat: no-repeat;\n background-position: center 6px;\n}\n#provider_locked::after {\n position: absolute;\n content: '';\n flex-shrink: 0;\n display: block;\n width: 32px;\n height: 32px;\n margin: 0 8px;\n background-size: 11px 13px;\n background-repeat: no-repeat;\n background-position: right bottom;\n}\n.tooltip__button--data.currentFocus:not(.tooltip__button--data--bitwarden)::before,\n.wrapper:not(.top-autofill) .tooltip__button--data:not(.tooltip__button--data--bitwarden):hover::before {\n filter: invert(100%);\n}\n@media (prefers-color-scheme: dark) {\n .tooltip__button--data:not(.tooltip__button--data--bitwarden)::before,\n .tooltip__button--data:not(.tooltip__button--data--bitwarden)::before {\n filter: invert(100%);\n opacity: .9;\n }\n}\n.tooltip__button__text-container {\n margin: auto 0;\n}\n.label {\n display: block;\n font-weight: 400;\n letter-spacing: -0.25px;\n color: rgba(0,0,0,.8);\n font-size: 13px;\n line-height: 1;\n}\n.label + .label {\n margin-top: 2px;\n}\n.label.label--medium {\n font-weight: 500;\n letter-spacing: -0.25px;\n color: rgba(0,0,0,.9);\n}\n.label.label--small {\n font-size: 11px;\n font-weight: 400;\n letter-spacing: 0.06px;\n color: rgba(0,0,0,0.6);\n}\n@media (prefers-color-scheme: dark) {\n .tooltip--data .label {\n color: #ffffff;\n }\n .tooltip--data .label--medium {\n color: #ffffff;\n }\n .tooltip--data .label--small {\n color: #cdcdcd;\n }\n}\n.tooltip__button.currentFocus .label,\n.wrapper:not(.top-autofill) .tooltip__button:hover .label {\n color: #FFFFFF;\n}\n\n.tooltip__button--manage {\n font-size: 13px;\n padding: 5px 9px;\n border-radius: 3px;\n margin: 0;\n}\n\n/* Icons */\n.tooltip__button--data--credentials::before {\n background-image: url('');\n}\n.tooltip__button--data--creditCards::before {\n background-image: url('');\n}\n.tooltip__button--data--identities::before {\n background-image: url('');\n}\n.tooltip__button--data--credentials.tooltip__button--data--bitwarden::before {\n background-image: url('');\n}\n#provider_locked:after {\n background-image: url('');\n}\n\nhr {\n display: block;\n margin: 5px 9px;\n border: none; /* reset the border */\n border-top: 1px solid rgba(0,0,0,.1);\n}\n\nhr:first-child {\n display: none;\n}\n\n@media (prefers-color-scheme: dark) {\n hr {\n border-top: 1px solid rgba(255,255,255,.2);\n }\n}\n\n#privateAddress {\n align-items: flex-start;\n}\n#personalAddress::before,\n#privateAddress::before,\n#incontextSignup::before,\n#personalAddress.currentFocus::before,\n#personalAddress:hover::before,\n#privateAddress.currentFocus::before,\n#privateAddress:hover::before {\n filter: none;\n /* This is the same icon as `daxBase64` in `src/Form/logo-svg.js` */\n background-image: url('');\n}\n\n/* Email tooltip specific */\n.tooltip__button--email {\n flex-direction: column;\n justify-content: center;\n align-items: flex-start;\n font-size: 14px;\n padding: 4px 8px;\n}\n.tooltip__button--email__primary-text {\n font-weight: bold;\n}\n.tooltip__button--email__secondary-text {\n font-size: 12px;\n}\n\n/* Email Protection signup notice */\n:not(.top-autofill) .tooltip--email-signup {\n text-align: left;\n color: #222222;\n padding: 16px 20px;\n width: 380px;\n}\n\n.tooltip--email-signup h1 {\n font-weight: 700;\n font-size: 16px;\n line-height: 1.5;\n margin: 0;\n}\n\n.tooltip--email-signup p {\n font-weight: 400;\n font-size: 14px;\n line-height: 1.4;\n}\n\n.notice-controls {\n display: flex;\n}\n\n.tooltip--email-signup .notice-controls > * {\n border-radius: 8px;\n border: 0;\n cursor: pointer;\n display: inline-block;\n font-family: inherit;\n font-style: normal;\n font-weight: bold;\n padding: 8px 12px;\n text-decoration: none;\n}\n\n.notice-controls .ghost {\n margin-left: 1rem;\n}\n\n.tooltip--email-signup a.primary {\n background: #3969EF;\n color: #fff;\n}\n\n.tooltip--email-signup a.primary:hover,\n.tooltip--email-signup a.primary:focus {\n background: #2b55ca;\n}\n\n.tooltip--email-signup a.primary:active {\n background: #1e42a4;\n}\n\n.tooltip--email-signup button.ghost {\n background: transparent;\n color: #3969EF;\n}\n\n.tooltip--email-signup button.ghost:hover,\n.tooltip--email-signup button.ghost:focus {\n background-color: rgba(0, 0, 0, 0.06);\n color: #2b55ca;\n}\n\n.tooltip--email-signup button.ghost:active {\n background-color: rgba(0, 0, 0, 0.12);\n color: #1e42a4;\n}\n\n.tooltip--email-signup button.close-tooltip {\n background-color: transparent;\n background-image: url();\n background-position: center center;\n background-repeat: no-repeat;\n border: 0;\n cursor: pointer;\n padding: 16px;\n position: absolute;\n right: 12px;\n top: 12px;\n}\n";
exports.CSS_STYLES = CSS_STYLES;
},{}],63:[function(require,module,exports){
diff --git a/dist/autofill.css b/dist/autofill.css
index 3dbe7ac18..7eb2c493f 100644
--- a/dist/autofill.css
+++ b/dist/autofill.css
@@ -64,7 +64,7 @@
}
:not(.top-autofill) .tooltip--email {
top: calc(100% + 6px);
- right: calc(100% - 46px);
+ right: calc(100% - 48px);
padding: 8px;
border: 1px solid #D0D0D0;
border-radius: 10px;
@@ -89,7 +89,7 @@
display: block;
border-bottom: 8px solid #D0D0D0;
position: absolute;
- right: -26px;
+ right: -28px;
}
.tooltip--email__caret::before {
border-bottom-color: #D0D0D0;
diff --git a/dist/autofill.js b/dist/autofill.js
index 628a7c963..a285cfa8a 100644
--- a/dist/autofill.js
+++ b/dist/autofill.js
@@ -8545,7 +8545,7 @@ const getIcon = function (input, form) {
const getBasicStyles = (input, icon) => ({
// Height must be > 0 to account for fields initially hidden
- 'background-size': "auto ".concat(input.offsetHeight <= 30 && input.offsetHeight > 0 ? '100%' : '26px'),
+ 'background-size': "auto ".concat(input.offsetHeight <= 30 && input.offsetHeight > 0 ? '100%' : '24px'),
'background-position': 'center right',
'background-repeat': 'no-repeat',
'background-origin': 'content-box',
@@ -12507,8 +12507,8 @@ function _defineProperty(obj, key, value) { if (key in obj) { Object.definePrope
/** @type {HTMLTooltipOptions} */
const defaultOptions = {
wrapperClass: '',
- tooltipPositionClass: (top, left) => "\n .tooltip {\n transform: translate(".concat(left, "px, ").concat(top, "px) !important;\n }\n "),
- caretPositionClass: (top, left, isAboveInput) => "\n .tooltip--email__caret {\n ".concat(isAboveInput ? "transform: translate(".concat(left, "px, ").concat(top, "px) rotate(180deg); transform-origin: 16px !important;") : "transform: translate(".concat(left, "px, ").concat(top, "px) !important;"), "\n }"),
+ tooltipPositionClass: (top, left) => "\n .tooltip {\n transform: translate(".concat(Math.floor(left), "px, ").concat(Math.floor(top), "px) !important;\n }\n "),
+ caretPositionClass: (top, left, isAboveInput) => "\n .tooltip--email__caret {\n ".concat(isAboveInput ? "transform: translate(".concat(Math.floor(left), "px, ").concat(Math.floor(top), "px) rotate(180deg); transform-origin: 18px !important;") : "transform: translate(".concat(Math.floor(left), "px, ").concat(Math.floor(top), "px) !important;"), "\n }"),
css: ""),
setSize: undefined,
remove: () => {
@@ -13937,7 +13937,7 @@ Object.defineProperty(exports, "__esModule", {
value: true
});
exports.CSS_STYLES = void 0;
-const CSS_STYLES = ":root {\n color-scheme: light dark;\n}\n\n.wrapper *, .wrapper *::before, .wrapper *::after {\n box-sizing: border-box;\n}\n.wrapper {\n position: fixed;\n top: 0;\n left: 0;\n padding: 0;\n font-family: 'DDG_ProximaNova', 'Proxima Nova', system-ui, 'Segoe UI', 'Roboto', 'Oxygen', 'Ubuntu',\n 'Cantarell', 'Fira Sans', 'Droid Sans', 'Helvetica Neue', sans-serif;\n -webkit-font-smoothing: antialiased;\n z-index: 2147483647;\n}\n.wrapper--data {\n font-family: 'SF Pro Text', system-ui, 'Segoe UI', 'Roboto', 'Oxygen', 'Ubuntu',\n 'Cantarell', 'Fira Sans', 'Droid Sans', 'Helvetica Neue', sans-serif;\n}\n:not(.top-autofill) .tooltip {\n position: absolute;\n width: 300px;\n max-width: calc(100vw - 25px);\n transform: translate(-1000px, -1000px);\n z-index: 2147483647;\n}\n.tooltip--data, #topAutofill {\n background-color: rgba(242, 240, 240, 1);\n -webkit-backdrop-filter: blur(40px);\n backdrop-filter: blur(40px);\n}\n@media (prefers-color-scheme: dark) {\n .tooltip--data, #topAutofill {\n background: rgb(100, 98, 102, .9);\n }\n}\n.tooltip--data {\n padding: 6px;\n font-size: 13px;\n line-height: 14px;\n width: 315px;\n max-height: 290px;\n overflow-y: auto;\n}\n.top-autofill .tooltip--data {\n min-height: 100vh;\n}\n.tooltip--data.tooltip--incontext-signup {\n width: 360px;\n}\n:not(.top-autofill) .tooltip--data {\n top: 100%;\n left: 100%;\n border: 0.5px solid rgba(255, 255, 255, 0.2);\n border-radius: 6px;\n box-shadow: 0 10px 20px rgba(0, 0, 0, 0.32);\n}\n@media (prefers-color-scheme: dark) {\n :not(.top-autofill) .tooltip--data {\n border: 1px solid rgba(255, 255, 255, 0.2);\n }\n}\n:not(.top-autofill) .tooltip--email {\n top: calc(100% + 6px);\n right: calc(100% - 46px);\n padding: 8px;\n border: 1px solid #D0D0D0;\n border-radius: 10px;\n background-color: #FFFFFF;\n font-size: 14px;\n line-height: 1.3;\n color: #333333;\n box-shadow: 0 10px 20px rgba(0, 0, 0, 0.15);\n}\n.tooltip--email__caret {\n position: absolute;\n transform: translate(-1000px, -1000px);\n z-index: 2147483647;\n}\n.tooltip--email__caret::before,\n.tooltip--email__caret::after {\n content: \"\";\n width: 0;\n height: 0;\n border-left: 10px solid transparent;\n border-right: 10px solid transparent;\n display: block;\n border-bottom: 8px solid #D0D0D0;\n position: absolute;\n right: -26px;\n}\n.tooltip--email__caret::before {\n border-bottom-color: #D0D0D0;\n top: -1px;\n}\n.tooltip--email__caret::after {\n border-bottom-color: #FFFFFF;\n top: 0px;\n}\n\n/* Buttons */\n.tooltip__button {\n display: flex;\n width: 100%;\n padding: 8px 8px 8px 0px;\n font-family: inherit;\n color: inherit;\n background: transparent;\n border: none;\n border-radius: 6px;\n}\n.tooltip__button.currentFocus,\n.wrapper:not(.top-autofill) .tooltip__button:hover {\n background-color: #3969EF;\n color: #FFFFFF;\n}\n\n/* Data autofill tooltip specific */\n.tooltip__button--data {\n position: relative;\n min-height: 48px;\n flex-direction: row;\n justify-content: flex-start;\n font-size: inherit;\n font-weight: 500;\n line-height: 16px;\n text-align: left;\n border-radius: 3px;\n}\n.tooltip--data__item-container {\n max-height: 220px;\n overflow: auto;\n}\n.tooltip__button--data:first-child {\n margin-top: 0;\n}\n.tooltip__button--data:last-child {\n margin-bottom: 0;\n}\n.tooltip__button--data::before {\n content: '';\n flex-shrink: 0;\n display: block;\n width: 32px;\n height: 32px;\n margin: 0 8px;\n background-size: 20px 20px;\n background-repeat: no-repeat;\n background-position: center 6px;\n}\n#provider_locked::after {\n position: absolute;\n content: '';\n flex-shrink: 0;\n display: block;\n width: 32px;\n height: 32px;\n margin: 0 8px;\n background-size: 11px 13px;\n background-repeat: no-repeat;\n background-position: right bottom;\n}\n.tooltip__button--data.currentFocus:not(.tooltip__button--data--bitwarden)::before,\n.wrapper:not(.top-autofill) .tooltip__button--data:not(.tooltip__button--data--bitwarden):hover::before {\n filter: invert(100%);\n}\n@media (prefers-color-scheme: dark) {\n .tooltip__button--data:not(.tooltip__button--data--bitwarden)::before,\n .tooltip__button--data:not(.tooltip__button--data--bitwarden)::before {\n filter: invert(100%);\n opacity: .9;\n }\n}\n.tooltip__button__text-container {\n margin: auto 0;\n}\n.label {\n display: block;\n font-weight: 400;\n letter-spacing: -0.25px;\n color: rgba(0,0,0,.8);\n font-size: 13px;\n line-height: 1;\n}\n.label + .label {\n margin-top: 2px;\n}\n.label.label--medium {\n font-weight: 500;\n letter-spacing: -0.25px;\n color: rgba(0,0,0,.9);\n}\n.label.label--small {\n font-size: 11px;\n font-weight: 400;\n letter-spacing: 0.06px;\n color: rgba(0,0,0,0.6);\n}\n@media (prefers-color-scheme: dark) {\n .tooltip--data .label {\n color: #ffffff;\n }\n .tooltip--data .label--medium {\n color: #ffffff;\n }\n .tooltip--data .label--small {\n color: #cdcdcd;\n }\n}\n.tooltip__button.currentFocus .label,\n.wrapper:not(.top-autofill) .tooltip__button:hover .label {\n color: #FFFFFF;\n}\n\n.tooltip__button--manage {\n font-size: 13px;\n padding: 5px 9px;\n border-radius: 3px;\n margin: 0;\n}\n\n/* Icons */\n.tooltip__button--data--credentials::before {\n background-image: url('');\n}\n.tooltip__button--data--creditCards::before {\n background-image: url('');\n}\n.tooltip__button--data--identities::before {\n background-image: url('');\n}\n.tooltip__button--data--credentials.tooltip__button--data--bitwarden::before {\n background-image: url('');\n}\n#provider_locked:after {\n background-image: url('');\n}\n\nhr {\n display: block;\n margin: 5px 9px;\n border: none; /* reset the border */\n border-top: 1px solid rgba(0,0,0,.1);\n}\n\nhr:first-child {\n display: none;\n}\n\n@media (prefers-color-scheme: dark) {\n hr {\n border-top: 1px solid rgba(255,255,255,.2);\n }\n}\n\n#privateAddress {\n align-items: flex-start;\n}\n#personalAddress::before,\n#privateAddress::before,\n#incontextSignup::before,\n#personalAddress.currentFocus::before,\n#personalAddress:hover::before,\n#privateAddress.currentFocus::before,\n#privateAddress:hover::before {\n filter: none;\n /* This is the same icon as `daxBase64` in `src/Form/logo-svg.js` */\n background-image: url('');\n}\n\n/* Email tooltip specific */\n.tooltip__button--email {\n flex-direction: column;\n justify-content: center;\n align-items: flex-start;\n font-size: 14px;\n padding: 4px 8px;\n}\n.tooltip__button--email__primary-text {\n font-weight: bold;\n}\n.tooltip__button--email__secondary-text {\n font-size: 12px;\n}\n\n/* Email Protection signup notice */\n:not(.top-autofill) .tooltip--email-signup {\n text-align: left;\n color: #222222;\n padding: 16px 20px;\n width: 380px;\n}\n\n.tooltip--email-signup h1 {\n font-weight: 700;\n font-size: 16px;\n line-height: 1.5;\n margin: 0;\n}\n\n.tooltip--email-signup p {\n font-weight: 400;\n font-size: 14px;\n line-height: 1.4;\n}\n\n.notice-controls {\n display: flex;\n}\n\n.tooltip--email-signup .notice-controls > * {\n border-radius: 8px;\n border: 0;\n cursor: pointer;\n display: inline-block;\n font-family: inherit;\n font-style: normal;\n font-weight: bold;\n padding: 8px 12px;\n text-decoration: none;\n}\n\n.notice-controls .ghost {\n margin-left: 1rem;\n}\n\n.tooltip--email-signup a.primary {\n background: #3969EF;\n color: #fff;\n}\n\n.tooltip--email-signup a.primary:hover,\n.tooltip--email-signup a.primary:focus {\n background: #2b55ca;\n}\n\n.tooltip--email-signup a.primary:active {\n background: #1e42a4;\n}\n\n.tooltip--email-signup button.ghost {\n background: transparent;\n color: #3969EF;\n}\n\n.tooltip--email-signup button.ghost:hover,\n.tooltip--email-signup button.ghost:focus {\n background-color: rgba(0, 0, 0, 0.06);\n color: #2b55ca;\n}\n\n.tooltip--email-signup button.ghost:active {\n background-color: rgba(0, 0, 0, 0.12);\n color: #1e42a4;\n}\n\n.tooltip--email-signup button.close-tooltip {\n background-color: transparent;\n background-image: url();\n background-position: center center;\n background-repeat: no-repeat;\n border: 0;\n cursor: pointer;\n padding: 16px;\n position: absolute;\n right: 12px;\n top: 12px;\n}\n";
+const CSS_STYLES = ":root {\n color-scheme: light dark;\n}\n\n.wrapper *, .wrapper *::before, .wrapper *::after {\n box-sizing: border-box;\n}\n.wrapper {\n position: fixed;\n top: 0;\n left: 0;\n padding: 0;\n font-family: 'DDG_ProximaNova', 'Proxima Nova', system-ui, 'Segoe UI', 'Roboto', 'Oxygen', 'Ubuntu',\n 'Cantarell', 'Fira Sans', 'Droid Sans', 'Helvetica Neue', sans-serif;\n -webkit-font-smoothing: antialiased;\n z-index: 2147483647;\n}\n.wrapper--data {\n font-family: 'SF Pro Text', system-ui, 'Segoe UI', 'Roboto', 'Oxygen', 'Ubuntu',\n 'Cantarell', 'Fira Sans', 'Droid Sans', 'Helvetica Neue', sans-serif;\n}\n:not(.top-autofill) .tooltip {\n position: absolute;\n width: 300px;\n max-width: calc(100vw - 25px);\n transform: translate(-1000px, -1000px);\n z-index: 2147483647;\n}\n.tooltip--data, #topAutofill {\n background-color: rgba(242, 240, 240, 1);\n -webkit-backdrop-filter: blur(40px);\n backdrop-filter: blur(40px);\n}\n@media (prefers-color-scheme: dark) {\n .tooltip--data, #topAutofill {\n background: rgb(100, 98, 102, .9);\n }\n}\n.tooltip--data {\n padding: 6px;\n font-size: 13px;\n line-height: 14px;\n width: 315px;\n max-height: 290px;\n overflow-y: auto;\n}\n.top-autofill .tooltip--data {\n min-height: 100vh;\n}\n.tooltip--data.tooltip--incontext-signup {\n width: 360px;\n}\n:not(.top-autofill) .tooltip--data {\n top: 100%;\n left: 100%;\n border: 0.5px solid rgba(255, 255, 255, 0.2);\n border-radius: 6px;\n box-shadow: 0 10px 20px rgba(0, 0, 0, 0.32);\n}\n@media (prefers-color-scheme: dark) {\n :not(.top-autofill) .tooltip--data {\n border: 1px solid rgba(255, 255, 255, 0.2);\n }\n}\n:not(.top-autofill) .tooltip--email {\n top: calc(100% + 6px);\n right: calc(100% - 48px);\n padding: 8px;\n border: 1px solid #D0D0D0;\n border-radius: 10px;\n background-color: #FFFFFF;\n font-size: 14px;\n line-height: 1.3;\n color: #333333;\n box-shadow: 0 10px 20px rgba(0, 0, 0, 0.15);\n}\n.tooltip--email__caret {\n position: absolute;\n transform: translate(-1000px, -1000px);\n z-index: 2147483647;\n}\n.tooltip--email__caret::before,\n.tooltip--email__caret::after {\n content: \"\";\n width: 0;\n height: 0;\n border-left: 10px solid transparent;\n border-right: 10px solid transparent;\n display: block;\n border-bottom: 8px solid #D0D0D0;\n position: absolute;\n right: -28px;\n}\n.tooltip--email__caret::before {\n border-bottom-color: #D0D0D0;\n top: -1px;\n}\n.tooltip--email__caret::after {\n border-bottom-color: #FFFFFF;\n top: 0px;\n}\n\n/* Buttons */\n.tooltip__button {\n display: flex;\n width: 100%;\n padding: 8px 8px 8px 0px;\n font-family: inherit;\n color: inherit;\n background: transparent;\n border: none;\n border-radius: 6px;\n}\n.tooltip__button.currentFocus,\n.wrapper:not(.top-autofill) .tooltip__button:hover {\n background-color: #3969EF;\n color: #FFFFFF;\n}\n\n/* Data autofill tooltip specific */\n.tooltip__button--data {\n position: relative;\n min-height: 48px;\n flex-direction: row;\n justify-content: flex-start;\n font-size: inherit;\n font-weight: 500;\n line-height: 16px;\n text-align: left;\n border-radius: 3px;\n}\n.tooltip--data__item-container {\n max-height: 220px;\n overflow: auto;\n}\n.tooltip__button--data:first-child {\n margin-top: 0;\n}\n.tooltip__button--data:last-child {\n margin-bottom: 0;\n}\n.tooltip__button--data::before {\n content: '';\n flex-shrink: 0;\n display: block;\n width: 32px;\n height: 32px;\n margin: 0 8px;\n background-size: 20px 20px;\n background-repeat: no-repeat;\n background-position: center 6px;\n}\n#provider_locked::after {\n position: absolute;\n content: '';\n flex-shrink: 0;\n display: block;\n width: 32px;\n height: 32px;\n margin: 0 8px;\n background-size: 11px 13px;\n background-repeat: no-repeat;\n background-position: right bottom;\n}\n.tooltip__button--data.currentFocus:not(.tooltip__button--data--bitwarden)::before,\n.wrapper:not(.top-autofill) .tooltip__button--data:not(.tooltip__button--data--bitwarden):hover::before {\n filter: invert(100%);\n}\n@media (prefers-color-scheme: dark) {\n .tooltip__button--data:not(.tooltip__button--data--bitwarden)::before,\n .tooltip__button--data:not(.tooltip__button--data--bitwarden)::before {\n filter: invert(100%);\n opacity: .9;\n }\n}\n.tooltip__button__text-container {\n margin: auto 0;\n}\n.label {\n display: block;\n font-weight: 400;\n letter-spacing: -0.25px;\n color: rgba(0,0,0,.8);\n font-size: 13px;\n line-height: 1;\n}\n.label + .label {\n margin-top: 2px;\n}\n.label.label--medium {\n font-weight: 500;\n letter-spacing: -0.25px;\n color: rgba(0,0,0,.9);\n}\n.label.label--small {\n font-size: 11px;\n font-weight: 400;\n letter-spacing: 0.06px;\n color: rgba(0,0,0,0.6);\n}\n@media (prefers-color-scheme: dark) {\n .tooltip--data .label {\n color: #ffffff;\n }\n .tooltip--data .label--medium {\n color: #ffffff;\n }\n .tooltip--data .label--small {\n color: #cdcdcd;\n }\n}\n.tooltip__button.currentFocus .label,\n.wrapper:not(.top-autofill) .tooltip__button:hover .label {\n color: #FFFFFF;\n}\n\n.tooltip__button--manage {\n font-size: 13px;\n padding: 5px 9px;\n border-radius: 3px;\n margin: 0;\n}\n\n/* Icons */\n.tooltip__button--data--credentials::before {\n background-image: url('');\n}\n.tooltip__button--data--creditCards::before {\n background-image: url('');\n}\n.tooltip__button--data--identities::before {\n background-image: url('');\n}\n.tooltip__button--data--credentials.tooltip__button--data--bitwarden::before {\n background-image: url('');\n}\n#provider_locked:after {\n background-image: url('');\n}\n\nhr {\n display: block;\n margin: 5px 9px;\n border: none; /* reset the border */\n border-top: 1px solid rgba(0,0,0,.1);\n}\n\nhr:first-child {\n display: none;\n}\n\n@media (prefers-color-scheme: dark) {\n hr {\n border-top: 1px solid rgba(255,255,255,.2);\n }\n}\n\n#privateAddress {\n align-items: flex-start;\n}\n#personalAddress::before,\n#privateAddress::before,\n#incontextSignup::before,\n#personalAddress.currentFocus::before,\n#personalAddress:hover::before,\n#privateAddress.currentFocus::before,\n#privateAddress:hover::before {\n filter: none;\n /* This is the same icon as `daxBase64` in `src/Form/logo-svg.js` */\n background-image: url('');\n}\n\n/* Email tooltip specific */\n.tooltip__button--email {\n flex-direction: column;\n justify-content: center;\n align-items: flex-start;\n font-size: 14px;\n padding: 4px 8px;\n}\n.tooltip__button--email__primary-text {\n font-weight: bold;\n}\n.tooltip__button--email__secondary-text {\n font-size: 12px;\n}\n\n/* Email Protection signup notice */\n:not(.top-autofill) .tooltip--email-signup {\n text-align: left;\n color: #222222;\n padding: 16px 20px;\n width: 380px;\n}\n\n.tooltip--email-signup h1 {\n font-weight: 700;\n font-size: 16px;\n line-height: 1.5;\n margin: 0;\n}\n\n.tooltip--email-signup p {\n font-weight: 400;\n font-size: 14px;\n line-height: 1.4;\n}\n\n.notice-controls {\n display: flex;\n}\n\n.tooltip--email-signup .notice-controls > * {\n border-radius: 8px;\n border: 0;\n cursor: pointer;\n display: inline-block;\n font-family: inherit;\n font-style: normal;\n font-weight: bold;\n padding: 8px 12px;\n text-decoration: none;\n}\n\n.notice-controls .ghost {\n margin-left: 1rem;\n}\n\n.tooltip--email-signup a.primary {\n background: #3969EF;\n color: #fff;\n}\n\n.tooltip--email-signup a.primary:hover,\n.tooltip--email-signup a.primary:focus {\n background: #2b55ca;\n}\n\n.tooltip--email-signup a.primary:active {\n background: #1e42a4;\n}\n\n.tooltip--email-signup button.ghost {\n background: transparent;\n color: #3969EF;\n}\n\n.tooltip--email-signup button.ghost:hover,\n.tooltip--email-signup button.ghost:focus {\n background-color: rgba(0, 0, 0, 0.06);\n color: #2b55ca;\n}\n\n.tooltip--email-signup button.ghost:active {\n background-color: rgba(0, 0, 0, 0.12);\n color: #1e42a4;\n}\n\n.tooltip--email-signup button.close-tooltip {\n background-color: transparent;\n background-image: url();\n background-position: center center;\n background-repeat: no-repeat;\n border: 0;\n cursor: pointer;\n padding: 16px;\n position: absolute;\n right: 12px;\n top: 12px;\n}\n";
exports.CSS_STYLES = CSS_STYLES;
},{}],55:[function(require,module,exports){
diff --git a/src/Form/inputStyles.js b/src/Form/inputStyles.js
index fdf5fa06d..f1a695571 100644
--- a/src/Form/inputStyles.js
+++ b/src/Form/inputStyles.js
@@ -29,7 +29,7 @@ const getIcon = (input, form, type = 'base') => {
*/
const getBasicStyles = (input, icon) => ({
// Height must be > 0 to account for fields initially hidden
- 'background-size': `auto ${input.offsetHeight <= 30 && input.offsetHeight > 0 ? '100%' : '26px'}`,
+ 'background-size': `auto ${input.offsetHeight <= 30 && input.offsetHeight > 0 ? '100%' : '24px'}`,
'background-position': 'center right',
'background-repeat': 'no-repeat',
'background-origin': 'content-box',
diff --git a/src/UI/HTMLTooltip.js b/src/UI/HTMLTooltip.js
index ce4875a61..4c7b7faba 100644
--- a/src/UI/HTMLTooltip.js
+++ b/src/UI/HTMLTooltip.js
@@ -27,14 +27,14 @@ export const defaultOptions = {
wrapperClass: '',
tooltipPositionClass: (top, left) => `
.tooltip {
- transform: translate(${left}px, ${top}px) !important;
+ transform: translate(${Math.floor(left)}px, ${Math.floor(top)}px) !important;
}
`,
caretPositionClass: (top, left, isAboveInput) => `
.tooltip--email__caret {
${isAboveInput
- ? `transform: translate(${left}px, ${top}px) rotate(180deg); transform-origin: 16px !important;`
- : `transform: translate(${left}px, ${top}px) !important;`
+ ? `transform: translate(${Math.floor(left)}px, ${Math.floor(top)}px) rotate(180deg); transform-origin: 18px !important;`
+ : `transform: translate(${Math.floor(left)}px, ${Math.floor(top)}px) !important;`
}
}`,
css: ``,
diff --git a/src/UI/styles/autofill-tooltip-styles.css b/src/UI/styles/autofill-tooltip-styles.css
index 3dbe7ac18..7eb2c493f 100644
--- a/src/UI/styles/autofill-tooltip-styles.css
+++ b/src/UI/styles/autofill-tooltip-styles.css
@@ -64,7 +64,7 @@
}
:not(.top-autofill) .tooltip--email {
top: calc(100% + 6px);
- right: calc(100% - 46px);
+ right: calc(100% - 48px);
padding: 8px;
border: 1px solid #D0D0D0;
border-radius: 10px;
@@ -89,7 +89,7 @@
display: block;
border-bottom: 8px solid #D0D0D0;
position: absolute;
- right: -26px;
+ right: -28px;
}
.tooltip--email__caret::before {
border-bottom-color: #D0D0D0;
diff --git a/swift-package/Resources/assets/autofill-debug.js b/swift-package/Resources/assets/autofill-debug.js
index a33be6fe9..7486f3f0b 100644
--- a/swift-package/Resources/assets/autofill-debug.js
+++ b/swift-package/Resources/assets/autofill-debug.js
@@ -12221,7 +12221,7 @@ const getIcon = function (input, form) {
const getBasicStyles = (input, icon) => ({
// Height must be > 0 to account for fields initially hidden
- 'background-size': "auto ".concat(input.offsetHeight <= 30 && input.offsetHeight > 0 ? '100%' : '26px'),
+ 'background-size': "auto ".concat(input.offsetHeight <= 30 && input.offsetHeight > 0 ? '100%' : '24px'),
'background-position': 'center right',
'background-repeat': 'no-repeat',
'background-origin': 'content-box',
@@ -16183,8 +16183,8 @@ function _defineProperty(obj, key, value) { if (key in obj) { Object.definePrope
/** @type {HTMLTooltipOptions} */
const defaultOptions = {
wrapperClass: '',
- tooltipPositionClass: (top, left) => "\n .tooltip {\n transform: translate(".concat(left, "px, ").concat(top, "px) !important;\n }\n "),
- caretPositionClass: (top, left, isAboveInput) => "\n .tooltip--email__caret {\n ".concat(isAboveInput ? "transform: translate(".concat(left, "px, ").concat(top, "px) rotate(180deg); transform-origin: 16px !important;") : "transform: translate(".concat(left, "px, ").concat(top, "px) !important;"), "\n }"),
+ tooltipPositionClass: (top, left) => "\n .tooltip {\n transform: translate(".concat(Math.floor(left), "px, ").concat(Math.floor(top), "px) !important;\n }\n "),
+ caretPositionClass: (top, left, isAboveInput) => "\n .tooltip--email__caret {\n ".concat(isAboveInput ? "transform: translate(".concat(Math.floor(left), "px, ").concat(Math.floor(top), "px) rotate(180deg); transform-origin: 18px !important;") : "transform: translate(".concat(Math.floor(left), "px, ").concat(Math.floor(top), "px) !important;"), "\n }"),
css: ""),
setSize: undefined,
remove: () => {
@@ -17613,7 +17613,7 @@ Object.defineProperty(exports, "__esModule", {
value: true
});
exports.CSS_STYLES = void 0;
-const CSS_STYLES = ":root {\n color-scheme: light dark;\n}\n\n.wrapper *, .wrapper *::before, .wrapper *::after {\n box-sizing: border-box;\n}\n.wrapper {\n position: fixed;\n top: 0;\n left: 0;\n padding: 0;\n font-family: 'DDG_ProximaNova', 'Proxima Nova', system-ui, 'Segoe UI', 'Roboto', 'Oxygen', 'Ubuntu',\n 'Cantarell', 'Fira Sans', 'Droid Sans', 'Helvetica Neue', sans-serif;\n -webkit-font-smoothing: antialiased;\n z-index: 2147483647;\n}\n.wrapper--data {\n font-family: 'SF Pro Text', system-ui, 'Segoe UI', 'Roboto', 'Oxygen', 'Ubuntu',\n 'Cantarell', 'Fira Sans', 'Droid Sans', 'Helvetica Neue', sans-serif;\n}\n:not(.top-autofill) .tooltip {\n position: absolute;\n width: 300px;\n max-width: calc(100vw - 25px);\n transform: translate(-1000px, -1000px);\n z-index: 2147483647;\n}\n.tooltip--data, #topAutofill {\n background-color: rgba(242, 240, 240, 1);\n -webkit-backdrop-filter: blur(40px);\n backdrop-filter: blur(40px);\n}\n@media (prefers-color-scheme: dark) {\n .tooltip--data, #topAutofill {\n background: rgb(100, 98, 102, .9);\n }\n}\n.tooltip--data {\n padding: 6px;\n font-size: 13px;\n line-height: 14px;\n width: 315px;\n max-height: 290px;\n overflow-y: auto;\n}\n.top-autofill .tooltip--data {\n min-height: 100vh;\n}\n.tooltip--data.tooltip--incontext-signup {\n width: 360px;\n}\n:not(.top-autofill) .tooltip--data {\n top: 100%;\n left: 100%;\n border: 0.5px solid rgba(255, 255, 255, 0.2);\n border-radius: 6px;\n box-shadow: 0 10px 20px rgba(0, 0, 0, 0.32);\n}\n@media (prefers-color-scheme: dark) {\n :not(.top-autofill) .tooltip--data {\n border: 1px solid rgba(255, 255, 255, 0.2);\n }\n}\n:not(.top-autofill) .tooltip--email {\n top: calc(100% + 6px);\n right: calc(100% - 46px);\n padding: 8px;\n border: 1px solid #D0D0D0;\n border-radius: 10px;\n background-color: #FFFFFF;\n font-size: 14px;\n line-height: 1.3;\n color: #333333;\n box-shadow: 0 10px 20px rgba(0, 0, 0, 0.15);\n}\n.tooltip--email__caret {\n position: absolute;\n transform: translate(-1000px, -1000px);\n z-index: 2147483647;\n}\n.tooltip--email__caret::before,\n.tooltip--email__caret::after {\n content: \"\";\n width: 0;\n height: 0;\n border-left: 10px solid transparent;\n border-right: 10px solid transparent;\n display: block;\n border-bottom: 8px solid #D0D0D0;\n position: absolute;\n right: -26px;\n}\n.tooltip--email__caret::before {\n border-bottom-color: #D0D0D0;\n top: -1px;\n}\n.tooltip--email__caret::after {\n border-bottom-color: #FFFFFF;\n top: 0px;\n}\n\n/* Buttons */\n.tooltip__button {\n display: flex;\n width: 100%;\n padding: 8px 8px 8px 0px;\n font-family: inherit;\n color: inherit;\n background: transparent;\n border: none;\n border-radius: 6px;\n}\n.tooltip__button.currentFocus,\n.wrapper:not(.top-autofill) .tooltip__button:hover {\n background-color: #3969EF;\n color: #FFFFFF;\n}\n\n/* Data autofill tooltip specific */\n.tooltip__button--data {\n position: relative;\n min-height: 48px;\n flex-direction: row;\n justify-content: flex-start;\n font-size: inherit;\n font-weight: 500;\n line-height: 16px;\n text-align: left;\n border-radius: 3px;\n}\n.tooltip--data__item-container {\n max-height: 220px;\n overflow: auto;\n}\n.tooltip__button--data:first-child {\n margin-top: 0;\n}\n.tooltip__button--data:last-child {\n margin-bottom: 0;\n}\n.tooltip__button--data::before {\n content: '';\n flex-shrink: 0;\n display: block;\n width: 32px;\n height: 32px;\n margin: 0 8px;\n background-size: 20px 20px;\n background-repeat: no-repeat;\n background-position: center 6px;\n}\n#provider_locked::after {\n position: absolute;\n content: '';\n flex-shrink: 0;\n display: block;\n width: 32px;\n height: 32px;\n margin: 0 8px;\n background-size: 11px 13px;\n background-repeat: no-repeat;\n background-position: right bottom;\n}\n.tooltip__button--data.currentFocus:not(.tooltip__button--data--bitwarden)::before,\n.wrapper:not(.top-autofill) .tooltip__button--data:not(.tooltip__button--data--bitwarden):hover::before {\n filter: invert(100%);\n}\n@media (prefers-color-scheme: dark) {\n .tooltip__button--data:not(.tooltip__button--data--bitwarden)::before,\n .tooltip__button--data:not(.tooltip__button--data--bitwarden)::before {\n filter: invert(100%);\n opacity: .9;\n }\n}\n.tooltip__button__text-container {\n margin: auto 0;\n}\n.label {\n display: block;\n font-weight: 400;\n letter-spacing: -0.25px;\n color: rgba(0,0,0,.8);\n font-size: 13px;\n line-height: 1;\n}\n.label + .label {\n margin-top: 2px;\n}\n.label.label--medium {\n font-weight: 500;\n letter-spacing: -0.25px;\n color: rgba(0,0,0,.9);\n}\n.label.label--small {\n font-size: 11px;\n font-weight: 400;\n letter-spacing: 0.06px;\n color: rgba(0,0,0,0.6);\n}\n@media (prefers-color-scheme: dark) {\n .tooltip--data .label {\n color: #ffffff;\n }\n .tooltip--data .label--medium {\n color: #ffffff;\n }\n .tooltip--data .label--small {\n color: #cdcdcd;\n }\n}\n.tooltip__button.currentFocus .label,\n.wrapper:not(.top-autofill) .tooltip__button:hover .label {\n color: #FFFFFF;\n}\n\n.tooltip__button--manage {\n font-size: 13px;\n padding: 5px 9px;\n border-radius: 3px;\n margin: 0;\n}\n\n/* Icons */\n.tooltip__button--data--credentials::before {\n background-image: url('');\n}\n.tooltip__button--data--creditCards::before {\n background-image: url('');\n}\n.tooltip__button--data--identities::before {\n background-image: url('');\n}\n.tooltip__button--data--credentials.tooltip__button--data--bitwarden::before {\n background-image: url('');\n}\n#provider_locked:after {\n background-image: url('');\n}\n\nhr {\n display: block;\n margin: 5px 9px;\n border: none; /* reset the border */\n border-top: 1px solid rgba(0,0,0,.1);\n}\n\nhr:first-child {\n display: none;\n}\n\n@media (prefers-color-scheme: dark) {\n hr {\n border-top: 1px solid rgba(255,255,255,.2);\n }\n}\n\n#privateAddress {\n align-items: flex-start;\n}\n#personalAddress::before,\n#privateAddress::before,\n#incontextSignup::before,\n#personalAddress.currentFocus::before,\n#personalAddress:hover::before,\n#privateAddress.currentFocus::before,\n#privateAddress:hover::before {\n filter: none;\n /* This is the same icon as `daxBase64` in `src/Form/logo-svg.js` */\n background-image: url('');\n}\n\n/* Email tooltip specific */\n.tooltip__button--email {\n flex-direction: column;\n justify-content: center;\n align-items: flex-start;\n font-size: 14px;\n padding: 4px 8px;\n}\n.tooltip__button--email__primary-text {\n font-weight: bold;\n}\n.tooltip__button--email__secondary-text {\n font-size: 12px;\n}\n\n/* Email Protection signup notice */\n:not(.top-autofill) .tooltip--email-signup {\n text-align: left;\n color: #222222;\n padding: 16px 20px;\n width: 380px;\n}\n\n.tooltip--email-signup h1 {\n font-weight: 700;\n font-size: 16px;\n line-height: 1.5;\n margin: 0;\n}\n\n.tooltip--email-signup p {\n font-weight: 400;\n font-size: 14px;\n line-height: 1.4;\n}\n\n.notice-controls {\n display: flex;\n}\n\n.tooltip--email-signup .notice-controls > * {\n border-radius: 8px;\n border: 0;\n cursor: pointer;\n display: inline-block;\n font-family: inherit;\n font-style: normal;\n font-weight: bold;\n padding: 8px 12px;\n text-decoration: none;\n}\n\n.notice-controls .ghost {\n margin-left: 1rem;\n}\n\n.tooltip--email-signup a.primary {\n background: #3969EF;\n color: #fff;\n}\n\n.tooltip--email-signup a.primary:hover,\n.tooltip--email-signup a.primary:focus {\n background: #2b55ca;\n}\n\n.tooltip--email-signup a.primary:active {\n background: #1e42a4;\n}\n\n.tooltip--email-signup button.ghost {\n background: transparent;\n color: #3969EF;\n}\n\n.tooltip--email-signup button.ghost:hover,\n.tooltip--email-signup button.ghost:focus {\n background-color: rgba(0, 0, 0, 0.06);\n color: #2b55ca;\n}\n\n.tooltip--email-signup button.ghost:active {\n background-color: rgba(0, 0, 0, 0.12);\n color: #1e42a4;\n}\n\n.tooltip--email-signup button.close-tooltip {\n background-color: transparent;\n background-image: url();\n background-position: center center;\n background-repeat: no-repeat;\n border: 0;\n cursor: pointer;\n padding: 16px;\n position: absolute;\n right: 12px;\n top: 12px;\n}\n";
+const CSS_STYLES = ":root {\n color-scheme: light dark;\n}\n\n.wrapper *, .wrapper *::before, .wrapper *::after {\n box-sizing: border-box;\n}\n.wrapper {\n position: fixed;\n top: 0;\n left: 0;\n padding: 0;\n font-family: 'DDG_ProximaNova', 'Proxima Nova', system-ui, 'Segoe UI', 'Roboto', 'Oxygen', 'Ubuntu',\n 'Cantarell', 'Fira Sans', 'Droid Sans', 'Helvetica Neue', sans-serif;\n -webkit-font-smoothing: antialiased;\n z-index: 2147483647;\n}\n.wrapper--data {\n font-family: 'SF Pro Text', system-ui, 'Segoe UI', 'Roboto', 'Oxygen', 'Ubuntu',\n 'Cantarell', 'Fira Sans', 'Droid Sans', 'Helvetica Neue', sans-serif;\n}\n:not(.top-autofill) .tooltip {\n position: absolute;\n width: 300px;\n max-width: calc(100vw - 25px);\n transform: translate(-1000px, -1000px);\n z-index: 2147483647;\n}\n.tooltip--data, #topAutofill {\n background-color: rgba(242, 240, 240, 1);\n -webkit-backdrop-filter: blur(40px);\n backdrop-filter: blur(40px);\n}\n@media (prefers-color-scheme: dark) {\n .tooltip--data, #topAutofill {\n background: rgb(100, 98, 102, .9);\n }\n}\n.tooltip--data {\n padding: 6px;\n font-size: 13px;\n line-height: 14px;\n width: 315px;\n max-height: 290px;\n overflow-y: auto;\n}\n.top-autofill .tooltip--data {\n min-height: 100vh;\n}\n.tooltip--data.tooltip--incontext-signup {\n width: 360px;\n}\n:not(.top-autofill) .tooltip--data {\n top: 100%;\n left: 100%;\n border: 0.5px solid rgba(255, 255, 255, 0.2);\n border-radius: 6px;\n box-shadow: 0 10px 20px rgba(0, 0, 0, 0.32);\n}\n@media (prefers-color-scheme: dark) {\n :not(.top-autofill) .tooltip--data {\n border: 1px solid rgba(255, 255, 255, 0.2);\n }\n}\n:not(.top-autofill) .tooltip--email {\n top: calc(100% + 6px);\n right: calc(100% - 48px);\n padding: 8px;\n border: 1px solid #D0D0D0;\n border-radius: 10px;\n background-color: #FFFFFF;\n font-size: 14px;\n line-height: 1.3;\n color: #333333;\n box-shadow: 0 10px 20px rgba(0, 0, 0, 0.15);\n}\n.tooltip--email__caret {\n position: absolute;\n transform: translate(-1000px, -1000px);\n z-index: 2147483647;\n}\n.tooltip--email__caret::before,\n.tooltip--email__caret::after {\n content: \"\";\n width: 0;\n height: 0;\n border-left: 10px solid transparent;\n border-right: 10px solid transparent;\n display: block;\n border-bottom: 8px solid #D0D0D0;\n position: absolute;\n right: -28px;\n}\n.tooltip--email__caret::before {\n border-bottom-color: #D0D0D0;\n top: -1px;\n}\n.tooltip--email__caret::after {\n border-bottom-color: #FFFFFF;\n top: 0px;\n}\n\n/* Buttons */\n.tooltip__button {\n display: flex;\n width: 100%;\n padding: 8px 8px 8px 0px;\n font-family: inherit;\n color: inherit;\n background: transparent;\n border: none;\n border-radius: 6px;\n}\n.tooltip__button.currentFocus,\n.wrapper:not(.top-autofill) .tooltip__button:hover {\n background-color: #3969EF;\n color: #FFFFFF;\n}\n\n/* Data autofill tooltip specific */\n.tooltip__button--data {\n position: relative;\n min-height: 48px;\n flex-direction: row;\n justify-content: flex-start;\n font-size: inherit;\n font-weight: 500;\n line-height: 16px;\n text-align: left;\n border-radius: 3px;\n}\n.tooltip--data__item-container {\n max-height: 220px;\n overflow: auto;\n}\n.tooltip__button--data:first-child {\n margin-top: 0;\n}\n.tooltip__button--data:last-child {\n margin-bottom: 0;\n}\n.tooltip__button--data::before {\n content: '';\n flex-shrink: 0;\n display: block;\n width: 32px;\n height: 32px;\n margin: 0 8px;\n background-size: 20px 20px;\n background-repeat: no-repeat;\n background-position: center 6px;\n}\n#provider_locked::after {\n position: absolute;\n content: '';\n flex-shrink: 0;\n display: block;\n width: 32px;\n height: 32px;\n margin: 0 8px;\n background-size: 11px 13px;\n background-repeat: no-repeat;\n background-position: right bottom;\n}\n.tooltip__button--data.currentFocus:not(.tooltip__button--data--bitwarden)::before,\n.wrapper:not(.top-autofill) .tooltip__button--data:not(.tooltip__button--data--bitwarden):hover::before {\n filter: invert(100%);\n}\n@media (prefers-color-scheme: dark) {\n .tooltip__button--data:not(.tooltip__button--data--bitwarden)::before,\n .tooltip__button--data:not(.tooltip__button--data--bitwarden)::before {\n filter: invert(100%);\n opacity: .9;\n }\n}\n.tooltip__button__text-container {\n margin: auto 0;\n}\n.label {\n display: block;\n font-weight: 400;\n letter-spacing: -0.25px;\n color: rgba(0,0,0,.8);\n font-size: 13px;\n line-height: 1;\n}\n.label + .label {\n margin-top: 2px;\n}\n.label.label--medium {\n font-weight: 500;\n letter-spacing: -0.25px;\n color: rgba(0,0,0,.9);\n}\n.label.label--small {\n font-size: 11px;\n font-weight: 400;\n letter-spacing: 0.06px;\n color: rgba(0,0,0,0.6);\n}\n@media (prefers-color-scheme: dark) {\n .tooltip--data .label {\n color: #ffffff;\n }\n .tooltip--data .label--medium {\n color: #ffffff;\n }\n .tooltip--data .label--small {\n color: #cdcdcd;\n }\n}\n.tooltip__button.currentFocus .label,\n.wrapper:not(.top-autofill) .tooltip__button:hover .label {\n color: #FFFFFF;\n}\n\n.tooltip__button--manage {\n font-size: 13px;\n padding: 5px 9px;\n border-radius: 3px;\n margin: 0;\n}\n\n/* Icons */\n.tooltip__button--data--credentials::before {\n background-image: url('');\n}\n.tooltip__button--data--creditCards::before {\n background-image: url('');\n}\n.tooltip__button--data--identities::before {\n background-image: url('');\n}\n.tooltip__button--data--credentials.tooltip__button--data--bitwarden::before {\n background-image: url('');\n}\n#provider_locked:after {\n background-image: url('');\n}\n\nhr {\n display: block;\n margin: 5px 9px;\n border: none; /* reset the border */\n border-top: 1px solid rgba(0,0,0,.1);\n}\n\nhr:first-child {\n display: none;\n}\n\n@media (prefers-color-scheme: dark) {\n hr {\n border-top: 1px solid rgba(255,255,255,.2);\n }\n}\n\n#privateAddress {\n align-items: flex-start;\n}\n#personalAddress::before,\n#privateAddress::before,\n#incontextSignup::before,\n#personalAddress.currentFocus::before,\n#personalAddress:hover::before,\n#privateAddress.currentFocus::before,\n#privateAddress:hover::before {\n filter: none;\n /* This is the same icon as `daxBase64` in `src/Form/logo-svg.js` */\n background-image: url('');\n}\n\n/* Email tooltip specific */\n.tooltip__button--email {\n flex-direction: column;\n justify-content: center;\n align-items: flex-start;\n font-size: 14px;\n padding: 4px 8px;\n}\n.tooltip__button--email__primary-text {\n font-weight: bold;\n}\n.tooltip__button--email__secondary-text {\n font-size: 12px;\n}\n\n/* Email Protection signup notice */\n:not(.top-autofill) .tooltip--email-signup {\n text-align: left;\n color: #222222;\n padding: 16px 20px;\n width: 380px;\n}\n\n.tooltip--email-signup h1 {\n font-weight: 700;\n font-size: 16px;\n line-height: 1.5;\n margin: 0;\n}\n\n.tooltip--email-signup p {\n font-weight: 400;\n font-size: 14px;\n line-height: 1.4;\n}\n\n.notice-controls {\n display: flex;\n}\n\n.tooltip--email-signup .notice-controls > * {\n border-radius: 8px;\n border: 0;\n cursor: pointer;\n display: inline-block;\n font-family: inherit;\n font-style: normal;\n font-weight: bold;\n padding: 8px 12px;\n text-decoration: none;\n}\n\n.notice-controls .ghost {\n margin-left: 1rem;\n}\n\n.tooltip--email-signup a.primary {\n background: #3969EF;\n color: #fff;\n}\n\n.tooltip--email-signup a.primary:hover,\n.tooltip--email-signup a.primary:focus {\n background: #2b55ca;\n}\n\n.tooltip--email-signup a.primary:active {\n background: #1e42a4;\n}\n\n.tooltip--email-signup button.ghost {\n background: transparent;\n color: #3969EF;\n}\n\n.tooltip--email-signup button.ghost:hover,\n.tooltip--email-signup button.ghost:focus {\n background-color: rgba(0, 0, 0, 0.06);\n color: #2b55ca;\n}\n\n.tooltip--email-signup button.ghost:active {\n background-color: rgba(0, 0, 0, 0.12);\n color: #1e42a4;\n}\n\n.tooltip--email-signup button.close-tooltip {\n background-color: transparent;\n background-image: url();\n background-position: center center;\n background-repeat: no-repeat;\n border: 0;\n cursor: pointer;\n padding: 16px;\n position: absolute;\n right: 12px;\n top: 12px;\n}\n";
exports.CSS_STYLES = CSS_STYLES;
},{}],63:[function(require,module,exports){
diff --git a/swift-package/Resources/assets/autofill.css b/swift-package/Resources/assets/autofill.css
index 3dbe7ac18..7eb2c493f 100644
--- a/swift-package/Resources/assets/autofill.css
+++ b/swift-package/Resources/assets/autofill.css
@@ -64,7 +64,7 @@
}
:not(.top-autofill) .tooltip--email {
top: calc(100% + 6px);
- right: calc(100% - 46px);
+ right: calc(100% - 48px);
padding: 8px;
border: 1px solid #D0D0D0;
border-radius: 10px;
@@ -89,7 +89,7 @@
display: block;
border-bottom: 8px solid #D0D0D0;
position: absolute;
- right: -26px;
+ right: -28px;
}
.tooltip--email__caret::before {
border-bottom-color: #D0D0D0;
diff --git a/swift-package/Resources/assets/autofill.js b/swift-package/Resources/assets/autofill.js
index 628a7c963..a285cfa8a 100644
--- a/swift-package/Resources/assets/autofill.js
+++ b/swift-package/Resources/assets/autofill.js
@@ -8545,7 +8545,7 @@ const getIcon = function (input, form) {
const getBasicStyles = (input, icon) => ({
// Height must be > 0 to account for fields initially hidden
- 'background-size': "auto ".concat(input.offsetHeight <= 30 && input.offsetHeight > 0 ? '100%' : '26px'),
+ 'background-size': "auto ".concat(input.offsetHeight <= 30 && input.offsetHeight > 0 ? '100%' : '24px'),
'background-position': 'center right',
'background-repeat': 'no-repeat',
'background-origin': 'content-box',
@@ -12507,8 +12507,8 @@ function _defineProperty(obj, key, value) { if (key in obj) { Object.definePrope
/** @type {HTMLTooltipOptions} */
const defaultOptions = {
wrapperClass: '',
- tooltipPositionClass: (top, left) => "\n .tooltip {\n transform: translate(".concat(left, "px, ").concat(top, "px) !important;\n }\n "),
- caretPositionClass: (top, left, isAboveInput) => "\n .tooltip--email__caret {\n ".concat(isAboveInput ? "transform: translate(".concat(left, "px, ").concat(top, "px) rotate(180deg); transform-origin: 16px !important;") : "transform: translate(".concat(left, "px, ").concat(top, "px) !important;"), "\n }"),
+ tooltipPositionClass: (top, left) => "\n .tooltip {\n transform: translate(".concat(Math.floor(left), "px, ").concat(Math.floor(top), "px) !important;\n }\n "),
+ caretPositionClass: (top, left, isAboveInput) => "\n .tooltip--email__caret {\n ".concat(isAboveInput ? "transform: translate(".concat(Math.floor(left), "px, ").concat(Math.floor(top), "px) rotate(180deg); transform-origin: 18px !important;") : "transform: translate(".concat(Math.floor(left), "px, ").concat(Math.floor(top), "px) !important;"), "\n }"),
css: ""),
setSize: undefined,
remove: () => {
@@ -13937,7 +13937,7 @@ Object.defineProperty(exports, "__esModule", {
value: true
});
exports.CSS_STYLES = void 0;
-const CSS_STYLES = ":root {\n color-scheme: light dark;\n}\n\n.wrapper *, .wrapper *::before, .wrapper *::after {\n box-sizing: border-box;\n}\n.wrapper {\n position: fixed;\n top: 0;\n left: 0;\n padding: 0;\n font-family: 'DDG_ProximaNova', 'Proxima Nova', system-ui, 'Segoe UI', 'Roboto', 'Oxygen', 'Ubuntu',\n 'Cantarell', 'Fira Sans', 'Droid Sans', 'Helvetica Neue', sans-serif;\n -webkit-font-smoothing: antialiased;\n z-index: 2147483647;\n}\n.wrapper--data {\n font-family: 'SF Pro Text', system-ui, 'Segoe UI', 'Roboto', 'Oxygen', 'Ubuntu',\n 'Cantarell', 'Fira Sans', 'Droid Sans', 'Helvetica Neue', sans-serif;\n}\n:not(.top-autofill) .tooltip {\n position: absolute;\n width: 300px;\n max-width: calc(100vw - 25px);\n transform: translate(-1000px, -1000px);\n z-index: 2147483647;\n}\n.tooltip--data, #topAutofill {\n background-color: rgba(242, 240, 240, 1);\n -webkit-backdrop-filter: blur(40px);\n backdrop-filter: blur(40px);\n}\n@media (prefers-color-scheme: dark) {\n .tooltip--data, #topAutofill {\n background: rgb(100, 98, 102, .9);\n }\n}\n.tooltip--data {\n padding: 6px;\n font-size: 13px;\n line-height: 14px;\n width: 315px;\n max-height: 290px;\n overflow-y: auto;\n}\n.top-autofill .tooltip--data {\n min-height: 100vh;\n}\n.tooltip--data.tooltip--incontext-signup {\n width: 360px;\n}\n:not(.top-autofill) .tooltip--data {\n top: 100%;\n left: 100%;\n border: 0.5px solid rgba(255, 255, 255, 0.2);\n border-radius: 6px;\n box-shadow: 0 10px 20px rgba(0, 0, 0, 0.32);\n}\n@media (prefers-color-scheme: dark) {\n :not(.top-autofill) .tooltip--data {\n border: 1px solid rgba(255, 255, 255, 0.2);\n }\n}\n:not(.top-autofill) .tooltip--email {\n top: calc(100% + 6px);\n right: calc(100% - 46px);\n padding: 8px;\n border: 1px solid #D0D0D0;\n border-radius: 10px;\n background-color: #FFFFFF;\n font-size: 14px;\n line-height: 1.3;\n color: #333333;\n box-shadow: 0 10px 20px rgba(0, 0, 0, 0.15);\n}\n.tooltip--email__caret {\n position: absolute;\n transform: translate(-1000px, -1000px);\n z-index: 2147483647;\n}\n.tooltip--email__caret::before,\n.tooltip--email__caret::after {\n content: \"\";\n width: 0;\n height: 0;\n border-left: 10px solid transparent;\n border-right: 10px solid transparent;\n display: block;\n border-bottom: 8px solid #D0D0D0;\n position: absolute;\n right: -26px;\n}\n.tooltip--email__caret::before {\n border-bottom-color: #D0D0D0;\n top: -1px;\n}\n.tooltip--email__caret::after {\n border-bottom-color: #FFFFFF;\n top: 0px;\n}\n\n/* Buttons */\n.tooltip__button {\n display: flex;\n width: 100%;\n padding: 8px 8px 8px 0px;\n font-family: inherit;\n color: inherit;\n background: transparent;\n border: none;\n border-radius: 6px;\n}\n.tooltip__button.currentFocus,\n.wrapper:not(.top-autofill) .tooltip__button:hover {\n background-color: #3969EF;\n color: #FFFFFF;\n}\n\n/* Data autofill tooltip specific */\n.tooltip__button--data {\n position: relative;\n min-height: 48px;\n flex-direction: row;\n justify-content: flex-start;\n font-size: inherit;\n font-weight: 500;\n line-height: 16px;\n text-align: left;\n border-radius: 3px;\n}\n.tooltip--data__item-container {\n max-height: 220px;\n overflow: auto;\n}\n.tooltip__button--data:first-child {\n margin-top: 0;\n}\n.tooltip__button--data:last-child {\n margin-bottom: 0;\n}\n.tooltip__button--data::before {\n content: '';\n flex-shrink: 0;\n display: block;\n width: 32px;\n height: 32px;\n margin: 0 8px;\n background-size: 20px 20px;\n background-repeat: no-repeat;\n background-position: center 6px;\n}\n#provider_locked::after {\n position: absolute;\n content: '';\n flex-shrink: 0;\n display: block;\n width: 32px;\n height: 32px;\n margin: 0 8px;\n background-size: 11px 13px;\n background-repeat: no-repeat;\n background-position: right bottom;\n}\n.tooltip__button--data.currentFocus:not(.tooltip__button--data--bitwarden)::before,\n.wrapper:not(.top-autofill) .tooltip__button--data:not(.tooltip__button--data--bitwarden):hover::before {\n filter: invert(100%);\n}\n@media (prefers-color-scheme: dark) {\n .tooltip__button--data:not(.tooltip__button--data--bitwarden)::before,\n .tooltip__button--data:not(.tooltip__button--data--bitwarden)::before {\n filter: invert(100%);\n opacity: .9;\n }\n}\n.tooltip__button__text-container {\n margin: auto 0;\n}\n.label {\n display: block;\n font-weight: 400;\n letter-spacing: -0.25px;\n color: rgba(0,0,0,.8);\n font-size: 13px;\n line-height: 1;\n}\n.label + .label {\n margin-top: 2px;\n}\n.label.label--medium {\n font-weight: 500;\n letter-spacing: -0.25px;\n color: rgba(0,0,0,.9);\n}\n.label.label--small {\n font-size: 11px;\n font-weight: 400;\n letter-spacing: 0.06px;\n color: rgba(0,0,0,0.6);\n}\n@media (prefers-color-scheme: dark) {\n .tooltip--data .label {\n color: #ffffff;\n }\n .tooltip--data .label--medium {\n color: #ffffff;\n }\n .tooltip--data .label--small {\n color: #cdcdcd;\n }\n}\n.tooltip__button.currentFocus .label,\n.wrapper:not(.top-autofill) .tooltip__button:hover .label {\n color: #FFFFFF;\n}\n\n.tooltip__button--manage {\n font-size: 13px;\n padding: 5px 9px;\n border-radius: 3px;\n margin: 0;\n}\n\n/* Icons */\n.tooltip__button--data--credentials::before {\n background-image: url('');\n}\n.tooltip__button--data--creditCards::before {\n background-image: url('');\n}\n.tooltip__button--data--identities::before {\n background-image: url('');\n}\n.tooltip__button--data--credentials.tooltip__button--data--bitwarden::before {\n background-image: url('');\n}\n#provider_locked:after {\n background-image: url('');\n}\n\nhr {\n display: block;\n margin: 5px 9px;\n border: none; /* reset the border */\n border-top: 1px solid rgba(0,0,0,.1);\n}\n\nhr:first-child {\n display: none;\n}\n\n@media (prefers-color-scheme: dark) {\n hr {\n border-top: 1px solid rgba(255,255,255,.2);\n }\n}\n\n#privateAddress {\n align-items: flex-start;\n}\n#personalAddress::before,\n#privateAddress::before,\n#incontextSignup::before,\n#personalAddress.currentFocus::before,\n#personalAddress:hover::before,\n#privateAddress.currentFocus::before,\n#privateAddress:hover::before {\n filter: none;\n /* This is the same icon as `daxBase64` in `src/Form/logo-svg.js` */\n background-image: url('');\n}\n\n/* Email tooltip specific */\n.tooltip__button--email {\n flex-direction: column;\n justify-content: center;\n align-items: flex-start;\n font-size: 14px;\n padding: 4px 8px;\n}\n.tooltip__button--email__primary-text {\n font-weight: bold;\n}\n.tooltip__button--email__secondary-text {\n font-size: 12px;\n}\n\n/* Email Protection signup notice */\n:not(.top-autofill) .tooltip--email-signup {\n text-align: left;\n color: #222222;\n padding: 16px 20px;\n width: 380px;\n}\n\n.tooltip--email-signup h1 {\n font-weight: 700;\n font-size: 16px;\n line-height: 1.5;\n margin: 0;\n}\n\n.tooltip--email-signup p {\n font-weight: 400;\n font-size: 14px;\n line-height: 1.4;\n}\n\n.notice-controls {\n display: flex;\n}\n\n.tooltip--email-signup .notice-controls > * {\n border-radius: 8px;\n border: 0;\n cursor: pointer;\n display: inline-block;\n font-family: inherit;\n font-style: normal;\n font-weight: bold;\n padding: 8px 12px;\n text-decoration: none;\n}\n\n.notice-controls .ghost {\n margin-left: 1rem;\n}\n\n.tooltip--email-signup a.primary {\n background: #3969EF;\n color: #fff;\n}\n\n.tooltip--email-signup a.primary:hover,\n.tooltip--email-signup a.primary:focus {\n background: #2b55ca;\n}\n\n.tooltip--email-signup a.primary:active {\n background: #1e42a4;\n}\n\n.tooltip--email-signup button.ghost {\n background: transparent;\n color: #3969EF;\n}\n\n.tooltip--email-signup button.ghost:hover,\n.tooltip--email-signup button.ghost:focus {\n background-color: rgba(0, 0, 0, 0.06);\n color: #2b55ca;\n}\n\n.tooltip--email-signup button.ghost:active {\n background-color: rgba(0, 0, 0, 0.12);\n color: #1e42a4;\n}\n\n.tooltip--email-signup button.close-tooltip {\n background-color: transparent;\n background-image: url();\n background-position: center center;\n background-repeat: no-repeat;\n border: 0;\n cursor: pointer;\n padding: 16px;\n position: absolute;\n right: 12px;\n top: 12px;\n}\n";
+const CSS_STYLES = ":root {\n color-scheme: light dark;\n}\n\n.wrapper *, .wrapper *::before, .wrapper *::after {\n box-sizing: border-box;\n}\n.wrapper {\n position: fixed;\n top: 0;\n left: 0;\n padding: 0;\n font-family: 'DDG_ProximaNova', 'Proxima Nova', system-ui, 'Segoe UI', 'Roboto', 'Oxygen', 'Ubuntu',\n 'Cantarell', 'Fira Sans', 'Droid Sans', 'Helvetica Neue', sans-serif;\n -webkit-font-smoothing: antialiased;\n z-index: 2147483647;\n}\n.wrapper--data {\n font-family: 'SF Pro Text', system-ui, 'Segoe UI', 'Roboto', 'Oxygen', 'Ubuntu',\n 'Cantarell', 'Fira Sans', 'Droid Sans', 'Helvetica Neue', sans-serif;\n}\n:not(.top-autofill) .tooltip {\n position: absolute;\n width: 300px;\n max-width: calc(100vw - 25px);\n transform: translate(-1000px, -1000px);\n z-index: 2147483647;\n}\n.tooltip--data, #topAutofill {\n background-color: rgba(242, 240, 240, 1);\n -webkit-backdrop-filter: blur(40px);\n backdrop-filter: blur(40px);\n}\n@media (prefers-color-scheme: dark) {\n .tooltip--data, #topAutofill {\n background: rgb(100, 98, 102, .9);\n }\n}\n.tooltip--data {\n padding: 6px;\n font-size: 13px;\n line-height: 14px;\n width: 315px;\n max-height: 290px;\n overflow-y: auto;\n}\n.top-autofill .tooltip--data {\n min-height: 100vh;\n}\n.tooltip--data.tooltip--incontext-signup {\n width: 360px;\n}\n:not(.top-autofill) .tooltip--data {\n top: 100%;\n left: 100%;\n border: 0.5px solid rgba(255, 255, 255, 0.2);\n border-radius: 6px;\n box-shadow: 0 10px 20px rgba(0, 0, 0, 0.32);\n}\n@media (prefers-color-scheme: dark) {\n :not(.top-autofill) .tooltip--data {\n border: 1px solid rgba(255, 255, 255, 0.2);\n }\n}\n:not(.top-autofill) .tooltip--email {\n top: calc(100% + 6px);\n right: calc(100% - 48px);\n padding: 8px;\n border: 1px solid #D0D0D0;\n border-radius: 10px;\n background-color: #FFFFFF;\n font-size: 14px;\n line-height: 1.3;\n color: #333333;\n box-shadow: 0 10px 20px rgba(0, 0, 0, 0.15);\n}\n.tooltip--email__caret {\n position: absolute;\n transform: translate(-1000px, -1000px);\n z-index: 2147483647;\n}\n.tooltip--email__caret::before,\n.tooltip--email__caret::after {\n content: \"\";\n width: 0;\n height: 0;\n border-left: 10px solid transparent;\n border-right: 10px solid transparent;\n display: block;\n border-bottom: 8px solid #D0D0D0;\n position: absolute;\n right: -28px;\n}\n.tooltip--email__caret::before {\n border-bottom-color: #D0D0D0;\n top: -1px;\n}\n.tooltip--email__caret::after {\n border-bottom-color: #FFFFFF;\n top: 0px;\n}\n\n/* Buttons */\n.tooltip__button {\n display: flex;\n width: 100%;\n padding: 8px 8px 8px 0px;\n font-family: inherit;\n color: inherit;\n background: transparent;\n border: none;\n border-radius: 6px;\n}\n.tooltip__button.currentFocus,\n.wrapper:not(.top-autofill) .tooltip__button:hover {\n background-color: #3969EF;\n color: #FFFFFF;\n}\n\n/* Data autofill tooltip specific */\n.tooltip__button--data {\n position: relative;\n min-height: 48px;\n flex-direction: row;\n justify-content: flex-start;\n font-size: inherit;\n font-weight: 500;\n line-height: 16px;\n text-align: left;\n border-radius: 3px;\n}\n.tooltip--data__item-container {\n max-height: 220px;\n overflow: auto;\n}\n.tooltip__button--data:first-child {\n margin-top: 0;\n}\n.tooltip__button--data:last-child {\n margin-bottom: 0;\n}\n.tooltip__button--data::before {\n content: '';\n flex-shrink: 0;\n display: block;\n width: 32px;\n height: 32px;\n margin: 0 8px;\n background-size: 20px 20px;\n background-repeat: no-repeat;\n background-position: center 6px;\n}\n#provider_locked::after {\n position: absolute;\n content: '';\n flex-shrink: 0;\n display: block;\n width: 32px;\n height: 32px;\n margin: 0 8px;\n background-size: 11px 13px;\n background-repeat: no-repeat;\n background-position: right bottom;\n}\n.tooltip__button--data.currentFocus:not(.tooltip__button--data--bitwarden)::before,\n.wrapper:not(.top-autofill) .tooltip__button--data:not(.tooltip__button--data--bitwarden):hover::before {\n filter: invert(100%);\n}\n@media (prefers-color-scheme: dark) {\n .tooltip__button--data:not(.tooltip__button--data--bitwarden)::before,\n .tooltip__button--data:not(.tooltip__button--data--bitwarden)::before {\n filter: invert(100%);\n opacity: .9;\n }\n}\n.tooltip__button__text-container {\n margin: auto 0;\n}\n.label {\n display: block;\n font-weight: 400;\n letter-spacing: -0.25px;\n color: rgba(0,0,0,.8);\n font-size: 13px;\n line-height: 1;\n}\n.label + .label {\n margin-top: 2px;\n}\n.label.label--medium {\n font-weight: 500;\n letter-spacing: -0.25px;\n color: rgba(0,0,0,.9);\n}\n.label.label--small {\n font-size: 11px;\n font-weight: 400;\n letter-spacing: 0.06px;\n color: rgba(0,0,0,0.6);\n}\n@media (prefers-color-scheme: dark) {\n .tooltip--data .label {\n color: #ffffff;\n }\n .tooltip--data .label--medium {\n color: #ffffff;\n }\n .tooltip--data .label--small {\n color: #cdcdcd;\n }\n}\n.tooltip__button.currentFocus .label,\n.wrapper:not(.top-autofill) .tooltip__button:hover .label {\n color: #FFFFFF;\n}\n\n.tooltip__button--manage {\n font-size: 13px;\n padding: 5px 9px;\n border-radius: 3px;\n margin: 0;\n}\n\n/* Icons */\n.tooltip__button--data--credentials::before {\n background-image: url('');\n}\n.tooltip__button--data--creditCards::before {\n background-image: url('');\n}\n.tooltip__button--data--identities::before {\n background-image: url('');\n}\n.tooltip__button--data--credentials.tooltip__button--data--bitwarden::before {\n background-image: url('');\n}\n#provider_locked:after {\n background-image: url('');\n}\n\nhr {\n display: block;\n margin: 5px 9px;\n border: none; /* reset the border */\n border-top: 1px solid rgba(0,0,0,.1);\n}\n\nhr:first-child {\n display: none;\n}\n\n@media (prefers-color-scheme: dark) {\n hr {\n border-top: 1px solid rgba(255,255,255,.2);\n }\n}\n\n#privateAddress {\n align-items: flex-start;\n}\n#personalAddress::before,\n#privateAddress::before,\n#incontextSignup::before,\n#personalAddress.currentFocus::before,\n#personalAddress:hover::before,\n#privateAddress.currentFocus::before,\n#privateAddress:hover::before {\n filter: none;\n /* This is the same icon as `daxBase64` in `src/Form/logo-svg.js` */\n background-image: url('');\n}\n\n/* Email tooltip specific */\n.tooltip__button--email {\n flex-direction: column;\n justify-content: center;\n align-items: flex-start;\n font-size: 14px;\n padding: 4px 8px;\n}\n.tooltip__button--email__primary-text {\n font-weight: bold;\n}\n.tooltip__button--email__secondary-text {\n font-size: 12px;\n}\n\n/* Email Protection signup notice */\n:not(.top-autofill) .tooltip--email-signup {\n text-align: left;\n color: #222222;\n padding: 16px 20px;\n width: 380px;\n}\n\n.tooltip--email-signup h1 {\n font-weight: 700;\n font-size: 16px;\n line-height: 1.5;\n margin: 0;\n}\n\n.tooltip--email-signup p {\n font-weight: 400;\n font-size: 14px;\n line-height: 1.4;\n}\n\n.notice-controls {\n display: flex;\n}\n\n.tooltip--email-signup .notice-controls > * {\n border-radius: 8px;\n border: 0;\n cursor: pointer;\n display: inline-block;\n font-family: inherit;\n font-style: normal;\n font-weight: bold;\n padding: 8px 12px;\n text-decoration: none;\n}\n\n.notice-controls .ghost {\n margin-left: 1rem;\n}\n\n.tooltip--email-signup a.primary {\n background: #3969EF;\n color: #fff;\n}\n\n.tooltip--email-signup a.primary:hover,\n.tooltip--email-signup a.primary:focus {\n background: #2b55ca;\n}\n\n.tooltip--email-signup a.primary:active {\n background: #1e42a4;\n}\n\n.tooltip--email-signup button.ghost {\n background: transparent;\n color: #3969EF;\n}\n\n.tooltip--email-signup button.ghost:hover,\n.tooltip--email-signup button.ghost:focus {\n background-color: rgba(0, 0, 0, 0.06);\n color: #2b55ca;\n}\n\n.tooltip--email-signup button.ghost:active {\n background-color: rgba(0, 0, 0, 0.12);\n color: #1e42a4;\n}\n\n.tooltip--email-signup button.close-tooltip {\n background-color: transparent;\n background-image: url();\n background-position: center center;\n background-repeat: no-repeat;\n border: 0;\n cursor: pointer;\n padding: 16px;\n position: absolute;\n right: 12px;\n top: 12px;\n}\n";
exports.CSS_STYLES = CSS_STYLES;
},{}],55:[function(require,module,exports){
From e4c0181d001bf7fcd78dbac31aca0f1fcdae7f15 Mon Sep 17 00:00:00 2001
From: Emanuele Feliziani
Date: Fri, 21 Jul 2023 14:45:04 +0200
Subject: [PATCH 6/6] Guard logs with shouldLog
Signed-off-by: Emanuele Feliziani
---
dist/autofill-debug.js | 13 ++++++++++---
dist/autofill.js | 13 ++++++++++---
src/Form/Form.js | 8 ++++++--
src/Scanner.js | 6 ++++--
swift-package/Resources/assets/autofill-debug.js | 13 ++++++++++---
swift-package/Resources/assets/autofill.js | 13 ++++++++++---
6 files changed, 50 insertions(+), 16 deletions(-)
diff --git a/dist/autofill-debug.js b/dist/autofill-debug.js
index 7486f3f0b..299c70623 100644
--- a/dist/autofill-debug.js
+++ b/dist/autofill-debug.js
@@ -10466,7 +10466,9 @@ class Form {
if (foundInputs.length < MAX_INPUTS_PER_FORM) {
foundInputs.forEach(input => this.addInput(input));
} else {
- console.log('The form has too many inputs, bailing.');
+ if ((0, _autofillUtils.shouldLog)()) {
+ console.log('The form has too many inputs, bailing.');
+ }
}
}
@@ -10529,7 +10531,10 @@ class Form {
if (this.inputs.all.has(input)) return this; // If the form has too many inputs, destroy everything to avoid performance issues
if (this.inputs.all.size > MAX_INPUTS_PER_FORM) {
- console.log('The form has too many inputs, destroying.');
+ if ((0, _autofillUtils.shouldLog)()) {
+ console.log('The form has too many inputs, destroying.');
+ }
+
this.destroy();
return this;
} // When new inputs are added after the initial scan, reanalyze the whole form
@@ -15440,7 +15445,9 @@ class DefaultScanner {
if (this.forms.size < this.options.maxFormsPerPage) {
this.forms.set(parentForm, new _Form.Form(parentForm, input, this.device, this.matching, this.shouldAutoprompt));
} else {
- console.log('The page has too many forms, stop adding them.');
+ if ((0, _autofillUtils.shouldLog)()) {
+ console.log('The page has too many forms, stop adding them.');
+ }
}
}
}
diff --git a/dist/autofill.js b/dist/autofill.js
index a285cfa8a..1c487d492 100644
--- a/dist/autofill.js
+++ b/dist/autofill.js
@@ -6790,7 +6790,9 @@ class Form {
if (foundInputs.length < MAX_INPUTS_PER_FORM) {
foundInputs.forEach(input => this.addInput(input));
} else {
- console.log('The form has too many inputs, bailing.');
+ if ((0, _autofillUtils.shouldLog)()) {
+ console.log('The form has too many inputs, bailing.');
+ }
}
}
@@ -6853,7 +6855,10 @@ class Form {
if (this.inputs.all.has(input)) return this; // If the form has too many inputs, destroy everything to avoid performance issues
if (this.inputs.all.size > MAX_INPUTS_PER_FORM) {
- console.log('The form has too many inputs, destroying.');
+ if ((0, _autofillUtils.shouldLog)()) {
+ console.log('The form has too many inputs, destroying.');
+ }
+
this.destroy();
return this;
} // When new inputs are added after the initial scan, reanalyze the whole form
@@ -11764,7 +11769,9 @@ class DefaultScanner {
if (this.forms.size < this.options.maxFormsPerPage) {
this.forms.set(parentForm, new _Form.Form(parentForm, input, this.device, this.matching, this.shouldAutoprompt));
} else {
- console.log('The page has too many forms, stop adding them.');
+ if ((0, _autofillUtils.shouldLog)()) {
+ console.log('The page has too many forms, stop adding them.');
+ }
}
}
}
diff --git a/src/Form/Form.js b/src/Form/Form.js
index 3e9eed099..1ad16ef97 100644
--- a/src/Form/Form.js
+++ b/src/Form/Form.js
@@ -366,7 +366,9 @@ class Form {
if (foundInputs.length < MAX_INPUTS_PER_FORM) {
foundInputs.forEach(input => this.addInput(input))
} else {
- console.log('The form has too many inputs, bailing.')
+ if (shouldLog()) {
+ console.log('The form has too many inputs, bailing.')
+ }
}
}
this.initialScanComplete = true
@@ -427,7 +429,9 @@ class Form {
// If the form has too many inputs, destroy everything to avoid performance issues
if (this.inputs.all.size > MAX_INPUTS_PER_FORM) {
- console.log('The form has too many inputs, destroying.')
+ if (shouldLog()) {
+ console.log('The form has too many inputs, destroying.')
+ }
this.destroy()
return this
}
diff --git a/src/Scanner.js b/src/Scanner.js
index dc05439f3..a8c4fad06 100644
--- a/src/Scanner.js
+++ b/src/Scanner.js
@@ -2,7 +2,7 @@ import { Form } from './Form/Form.js'
import { SUBMIT_BUTTON_SELECTOR, FORM_INPUTS_SELECTOR } from './Form/selectors-css.js'
import { constants } from './constants.js'
import { createMatching } from './Form/matching.js'
-import {isFormLikelyToBeUsedAsPageWrapper} from './autofill-utils.js'
+import {isFormLikelyToBeUsedAsPageWrapper, shouldLog} from './autofill-utils.js'
const {
MAX_INPUTS_PER_PAGE,
@@ -216,7 +216,9 @@ class DefaultScanner {
if (this.forms.size < this.options.maxFormsPerPage) {
this.forms.set(parentForm, new Form(parentForm, input, this.device, this.matching, this.shouldAutoprompt))
} else {
- console.log('The page has too many forms, stop adding them.')
+ if (shouldLog()) {
+ console.log('The page has too many forms, stop adding them.')
+ }
}
}
}
diff --git a/swift-package/Resources/assets/autofill-debug.js b/swift-package/Resources/assets/autofill-debug.js
index 7486f3f0b..299c70623 100644
--- a/swift-package/Resources/assets/autofill-debug.js
+++ b/swift-package/Resources/assets/autofill-debug.js
@@ -10466,7 +10466,9 @@ class Form {
if (foundInputs.length < MAX_INPUTS_PER_FORM) {
foundInputs.forEach(input => this.addInput(input));
} else {
- console.log('The form has too many inputs, bailing.');
+ if ((0, _autofillUtils.shouldLog)()) {
+ console.log('The form has too many inputs, bailing.');
+ }
}
}
@@ -10529,7 +10531,10 @@ class Form {
if (this.inputs.all.has(input)) return this; // If the form has too many inputs, destroy everything to avoid performance issues
if (this.inputs.all.size > MAX_INPUTS_PER_FORM) {
- console.log('The form has too many inputs, destroying.');
+ if ((0, _autofillUtils.shouldLog)()) {
+ console.log('The form has too many inputs, destroying.');
+ }
+
this.destroy();
return this;
} // When new inputs are added after the initial scan, reanalyze the whole form
@@ -15440,7 +15445,9 @@ class DefaultScanner {
if (this.forms.size < this.options.maxFormsPerPage) {
this.forms.set(parentForm, new _Form.Form(parentForm, input, this.device, this.matching, this.shouldAutoprompt));
} else {
- console.log('The page has too many forms, stop adding them.');
+ if ((0, _autofillUtils.shouldLog)()) {
+ console.log('The page has too many forms, stop adding them.');
+ }
}
}
}
diff --git a/swift-package/Resources/assets/autofill.js b/swift-package/Resources/assets/autofill.js
index a285cfa8a..1c487d492 100644
--- a/swift-package/Resources/assets/autofill.js
+++ b/swift-package/Resources/assets/autofill.js
@@ -6790,7 +6790,9 @@ class Form {
if (foundInputs.length < MAX_INPUTS_PER_FORM) {
foundInputs.forEach(input => this.addInput(input));
} else {
- console.log('The form has too many inputs, bailing.');
+ if ((0, _autofillUtils.shouldLog)()) {
+ console.log('The form has too many inputs, bailing.');
+ }
}
}
@@ -6853,7 +6855,10 @@ class Form {
if (this.inputs.all.has(input)) return this; // If the form has too many inputs, destroy everything to avoid performance issues
if (this.inputs.all.size > MAX_INPUTS_PER_FORM) {
- console.log('The form has too many inputs, destroying.');
+ if ((0, _autofillUtils.shouldLog)()) {
+ console.log('The form has too many inputs, destroying.');
+ }
+
this.destroy();
return this;
} // When new inputs are added after the initial scan, reanalyze the whole form
@@ -11764,7 +11769,9 @@ class DefaultScanner {
if (this.forms.size < this.options.maxFormsPerPage) {
this.forms.set(parentForm, new _Form.Form(parentForm, input, this.device, this.matching, this.shouldAutoprompt));
} else {
- console.log('The page has too many forms, stop adding them.');
+ if ((0, _autofillUtils.shouldLog)()) {
+ console.log('The page has too many forms, stop adding them.');
+ }
}
}
}