Bug 1691819 - [1.7] Extend the Autocomplete API to support credit cards. r=geckoview-reviewers,agi

Differential Revision: https://phabricator.services.mozilla.com/D106695
This commit is contained in:
Eugen Sawin 2021-04-12 21:57:15 +00:00
parent 33cc29d588
commit 434f165a88
9 changed files with 905 additions and 360 deletions

View File

@ -84,3 +84,5 @@ pref("toolkit.autocomplete.delegate", true);
// Android doesn't support the new sync storage yet, we will have our own in // Android doesn't support the new sync storage yet, we will have our own in
// Bug 1625257. // Bug 1625257.
pref("webextensions.storage.sync.kinto", true); pref("webextensions.storage.sync.kinto", true);
pref("browser.formfill.enable", true);

View File

@ -681,6 +681,26 @@ function startup() {
frameScript: "chrome://geckoview/content/GeckoViewMediaControlChild.js", frameScript: "chrome://geckoview/content/GeckoViewMediaControlChild.js",
}, },
}, },
{
name: "GeckoViewAutocomplete",
onInit: {
actors: {
FormAutofill: {
parent: {
moduleURI: "resource://autofill/FormAutofillParent.jsm",
},
child: {
moduleURI: "resource://autofill/FormAutofillChild.jsm",
events: {
focusin: {},
DOMFormBeforeSubmit: {},
},
},
allFrames: true,
},
},
},
},
]); ]);
if (!Services.appinfo.sessionHistoryInParent) { if (!Services.appinfo.sessionHistoryInParent) {

View File

@ -24,7 +24,8 @@ import org.mozilla.geckoview.Autocomplete
import org.mozilla.geckoview.Autocomplete.LoginEntry import org.mozilla.geckoview.Autocomplete.LoginEntry
import org.mozilla.geckoview.Autocomplete.LoginSaveOption import org.mozilla.geckoview.Autocomplete.LoginSaveOption
import org.mozilla.geckoview.Autocomplete.LoginSelectOption import org.mozilla.geckoview.Autocomplete.LoginSelectOption
import org.mozilla.geckoview.Autocomplete.LoginStorageDelegate import org.mozilla.geckoview.Autocomplete.SelectOption
import org.mozilla.geckoview.Autocomplete.StorageDelegate
import org.mozilla.geckoview.test.rule.GeckoSessionTestRule import org.mozilla.geckoview.test.rule.GeckoSessionTestRule
import org.mozilla.geckoview.test.rule.GeckoSessionTestRule.AssertCalled import org.mozilla.geckoview.test.rule.GeckoSessionTestRule.AssertCalled
import org.mozilla.geckoview.test.util.Callbacks import org.mozilla.geckoview.test.util.Callbacks
@ -43,18 +44,18 @@ class AutocompleteTest : BaseSessionTest() {
"signon.autofillForms.http" to true)) "signon.autofillForms.http" to true))
val runtime = sessionRule.runtime val runtime = sessionRule.runtime
val register = { delegate: LoginStorageDelegate -> val register = { delegate: StorageDelegate ->
runtime.loginStorageDelegate = delegate runtime.autocompleteStorageDelegate = delegate
} }
val unregister = { _: LoginStorageDelegate -> val unregister = { _: StorageDelegate ->
runtime.loginStorageDelegate = null runtime.autocompleteStorageDelegate = null
} }
val fetchHandled = GeckoResult<Void>() val fetchHandled = GeckoResult<Void>()
sessionRule.addExternalDelegateDuringNextWait( sessionRule.addExternalDelegateDuringNextWait(
LoginStorageDelegate::class, register, unregister, StorageDelegate::class, register, unregister,
object : LoginStorageDelegate { object : StorageDelegate {
@AssertCalled(count = 1) @AssertCalled(count = 1)
override fun onLoginFetch(domain: String) override fun onLoginFetch(domain: String)
: GeckoResult<Array<LoginEntry>>? { : GeckoResult<Array<LoginEntry>>? {
@ -81,16 +82,16 @@ class AutocompleteTest : BaseSessionTest() {
"signon.userInputRequiredToCapture.enabled" to false)) "signon.userInputRequiredToCapture.enabled" to false))
val runtime = sessionRule.runtime val runtime = sessionRule.runtime
val register = { delegate: LoginStorageDelegate -> val register = { delegate: StorageDelegate ->
runtime.loginStorageDelegate = delegate runtime.autocompleteStorageDelegate = delegate
} }
val unregister = { _: LoginStorageDelegate -> val unregister = { _: StorageDelegate ->
runtime.loginStorageDelegate = null runtime.autocompleteStorageDelegate = null
} }
sessionRule.addExternalDelegateDuringNextWait( sessionRule.addExternalDelegateDuringNextWait(
LoginStorageDelegate::class, register, unregister, StorageDelegate::class, register, unregister,
object : LoginStorageDelegate { object : StorageDelegate {
@AssertCalled(count = 1) @AssertCalled(count = 1)
override fun onLoginFetch(domain: String) override fun onLoginFetch(domain: String)
: GeckoResult<Array<LoginEntry>>? { : GeckoResult<Array<LoginEntry>>? {
@ -104,8 +105,8 @@ class AutocompleteTest : BaseSessionTest() {
mainSession.waitForPageStop() mainSession.waitForPageStop()
sessionRule.addExternalDelegateUntilTestEnd( sessionRule.addExternalDelegateUntilTestEnd(
LoginStorageDelegate::class, register, unregister, StorageDelegate::class, register, unregister,
object : LoginStorageDelegate { object : StorageDelegate {
@AssertCalled(count = 0) @AssertCalled(count = 0)
override fun onLoginSave(login: LoginEntry) {} override fun onLoginSave(login: LoginEntry) {}
}) })
@ -152,11 +153,11 @@ class AutocompleteTest : BaseSessionTest() {
"signon.userInputRequiredToCapture.enabled" to false)) "signon.userInputRequiredToCapture.enabled" to false))
val runtime = sessionRule.runtime val runtime = sessionRule.runtime
val register = { delegate: LoginStorageDelegate -> val register = { delegate: StorageDelegate ->
runtime.loginStorageDelegate = delegate runtime.autocompleteStorageDelegate = delegate
} }
val unregister = { _: LoginStorageDelegate -> val unregister = { _: StorageDelegate ->
runtime.loginStorageDelegate = null runtime.autocompleteStorageDelegate = null
} }
mainSession.loadTestPath(FORMS3_HTML_PATH) mainSession.loadTestPath(FORMS3_HTML_PATH)
@ -165,8 +166,8 @@ class AutocompleteTest : BaseSessionTest() {
val saveHandled = GeckoResult<Void>() val saveHandled = GeckoResult<Void>()
sessionRule.addExternalDelegateUntilTestEnd( sessionRule.addExternalDelegateUntilTestEnd(
LoginStorageDelegate::class, register, unregister, StorageDelegate::class, register, unregister,
object : LoginStorageDelegate { object : StorageDelegate {
@AssertCalled @AssertCalled
override fun onLoginSave(login: LoginEntry) { override fun onLoginSave(login: LoginEntry) {
assertThat( assertThat(
@ -229,11 +230,11 @@ class AutocompleteTest : BaseSessionTest() {
"signon.userInputRequiredToCapture.enabled" to false)) "signon.userInputRequiredToCapture.enabled" to false))
val runtime = sessionRule.runtime val runtime = sessionRule.runtime
val register = { delegate: LoginStorageDelegate -> val register = { delegate: StorageDelegate ->
runtime.loginStorageDelegate = delegate runtime.autocompleteStorageDelegate = delegate
} }
val unregister = { _: LoginStorageDelegate -> val unregister = { _: StorageDelegate ->
runtime.loginStorageDelegate = null runtime.autocompleteStorageDelegate = null
} }
mainSession.loadTestPath(FORMS3_HTML_PATH) mainSession.loadTestPath(FORMS3_HTML_PATH)
@ -242,8 +243,8 @@ class AutocompleteTest : BaseSessionTest() {
val saveHandled = GeckoResult<Void>() val saveHandled = GeckoResult<Void>()
sessionRule.addExternalDelegateUntilTestEnd( sessionRule.addExternalDelegateUntilTestEnd(
LoginStorageDelegate::class, register, unregister, StorageDelegate::class, register, unregister,
object : LoginStorageDelegate { object : StorageDelegate {
@AssertCalled @AssertCalled
override fun onLoginSave(login: LoginEntry) { override fun onLoginSave(login: LoginEntry) {
assertThat( assertThat(
@ -314,11 +315,11 @@ class AutocompleteTest : BaseSessionTest() {
"signon.userInputRequiredToCapture.enabled" to false)) "signon.userInputRequiredToCapture.enabled" to false))
val runtime = sessionRule.runtime val runtime = sessionRule.runtime
val register = { delegate: LoginStorageDelegate -> val register = { delegate: StorageDelegate ->
runtime.loginStorageDelegate = delegate runtime.autocompleteStorageDelegate = delegate
} }
val unregister = { _: LoginStorageDelegate -> val unregister = { _: StorageDelegate ->
runtime.loginStorageDelegate = null runtime.autocompleteStorageDelegate = null
} }
val saveHandled = GeckoResult<Void>() val saveHandled = GeckoResult<Void>()
@ -331,8 +332,8 @@ class AutocompleteTest : BaseSessionTest() {
val savedLogins = mutableListOf<LoginEntry>() val savedLogins = mutableListOf<LoginEntry>()
sessionRule.addExternalDelegateUntilTestEnd( sessionRule.addExternalDelegateUntilTestEnd(
LoginStorageDelegate::class, register, unregister, StorageDelegate::class, register, unregister,
object : LoginStorageDelegate { object : StorageDelegate {
@AssertCalled @AssertCalled
override fun onLoginFetch(domain: String) override fun onLoginFetch(domain: String)
: GeckoResult<Array<LoginEntry>>? { : GeckoResult<Array<LoginEntry>>? {
@ -430,11 +431,11 @@ class AutocompleteTest : BaseSessionTest() {
"signon.userInputRequiredToCapture.enabled" to false)) "signon.userInputRequiredToCapture.enabled" to false))
val runtime = sessionRule.runtime val runtime = sessionRule.runtime
val register = { delegate: LoginStorageDelegate -> val register = { delegate: StorageDelegate ->
runtime.loginStorageDelegate = delegate runtime.autocompleteStorageDelegate = delegate
} }
val unregister = { _: LoginStorageDelegate -> val unregister = { _: StorageDelegate ->
runtime.loginStorageDelegate = null runtime.autocompleteStorageDelegate = null
} }
val usedHandled = GeckoResult<Void>() val usedHandled = GeckoResult<Void>()
@ -454,8 +455,8 @@ class AutocompleteTest : BaseSessionTest() {
if (autofillEnabled) { if (autofillEnabled) {
sessionRule.addExternalDelegateUntilTestEnd( sessionRule.addExternalDelegateUntilTestEnd(
LoginStorageDelegate::class, register, unregister, StorageDelegate::class, register, unregister,
object : LoginStorageDelegate { object : StorageDelegate {
@AssertCalled @AssertCalled
override fun onLoginFetch(domain: String) override fun onLoginFetch(domain: String)
: GeckoResult<Array<LoginEntry>>? { : GeckoResult<Array<LoginEntry>>? {
@ -491,8 +492,8 @@ class AutocompleteTest : BaseSessionTest() {
}) })
} else { } else {
sessionRule.addExternalDelegateUntilTestEnd( sessionRule.addExternalDelegateUntilTestEnd(
LoginStorageDelegate::class, register, unregister, StorageDelegate::class, register, unregister,
object : LoginStorageDelegate { object : StorageDelegate {
@AssertCalled @AssertCalled
override fun onLoginFetch(domain: String) override fun onLoginFetch(domain: String)
: GeckoResult<Array<LoginEntry>>? { : GeckoResult<Array<LoginEntry>>? {
@ -547,11 +548,11 @@ class AutocompleteTest : BaseSessionTest() {
"signon.userInputRequiredToCapture.enabled" to false)) "signon.userInputRequiredToCapture.enabled" to false))
val runtime = sessionRule.runtime val runtime = sessionRule.runtime
val register = { delegate: LoginStorageDelegate -> val register = { delegate: StorageDelegate ->
runtime.loginStorageDelegate = delegate runtime.autocompleteStorageDelegate = delegate
} }
val unregister = { _: LoginStorageDelegate -> val unregister = { _: StorageDelegate ->
runtime.loginStorageDelegate = null runtime.autocompleteStorageDelegate = null
} }
val user1 = "user1x" val user1 = "user1x"
@ -568,8 +569,8 @@ class AutocompleteTest : BaseSessionTest() {
val savedLogins = mutableListOf<LoginEntry>(savedLogin) val savedLogins = mutableListOf<LoginEntry>(savedLogin)
sessionRule.addExternalDelegateUntilTestEnd( sessionRule.addExternalDelegateUntilTestEnd(
LoginStorageDelegate::class, register, unregister, StorageDelegate::class, register, unregister,
object : LoginStorageDelegate { object : StorageDelegate {
@AssertCalled @AssertCalled
override fun onLoginFetch(domain: String) override fun onLoginFetch(domain: String)
: GeckoResult<Array<LoginEntry>>? { : GeckoResult<Array<LoginEntry>>? {
@ -657,11 +658,11 @@ class AutocompleteTest : BaseSessionTest() {
// f. Ensure that onLoginUsed is called. // f. Ensure that onLoginUsed is called.
val runtime = sessionRule.runtime val runtime = sessionRule.runtime
val register = { delegate: LoginStorageDelegate -> val register = { delegate: StorageDelegate ->
runtime.loginStorageDelegate = delegate runtime.autocompleteStorageDelegate = delegate
} }
val unregister = { _: LoginStorageDelegate -> val unregister = { _: StorageDelegate ->
runtime.loginStorageDelegate = null runtime.autocompleteStorageDelegate = null
} }
val user1 = "user1x" val user1 = "user1x"
@ -676,8 +677,8 @@ class AutocompleteTest : BaseSessionTest() {
val usedHandled = GeckoResult<Void>() val usedHandled = GeckoResult<Void>()
sessionRule.addExternalDelegateUntilTestEnd( sessionRule.addExternalDelegateUntilTestEnd(
LoginStorageDelegate::class, register, unregister, StorageDelegate::class, register, unregister,
object : LoginStorageDelegate { object : StorageDelegate {
@AssertCalled @AssertCalled
override fun onLoginFetch(domain: String) override fun onLoginFetch(domain: String)
: GeckoResult<Array<LoginEntry>>? { : GeckoResult<Array<LoginEntry>>? {
@ -929,11 +930,11 @@ class AutocompleteTest : BaseSessionTest() {
// f. Ensure that onLoginUsed is not called. // f. Ensure that onLoginUsed is not called.
val runtime = sessionRule.runtime val runtime = sessionRule.runtime
val register = { delegate: LoginStorageDelegate -> val register = { delegate: StorageDelegate ->
runtime.loginStorageDelegate = delegate runtime.autocompleteStorageDelegate = delegate
} }
val unregister = { _: LoginStorageDelegate -> val unregister = { _: StorageDelegate ->
runtime.loginStorageDelegate = null runtime.autocompleteStorageDelegate = null
} }
val user1 = "user1x" val user1 = "user1x"
@ -949,8 +950,8 @@ class AutocompleteTest : BaseSessionTest() {
val selectHandled = GeckoResult<Void>() val selectHandled = GeckoResult<Void>()
sessionRule.addExternalDelegateUntilTestEnd( sessionRule.addExternalDelegateUntilTestEnd(
LoginStorageDelegate::class, register, unregister, StorageDelegate::class, register, unregister,
object : LoginStorageDelegate { object : StorageDelegate {
@AssertCalled @AssertCalled
override fun onLoginFetch(domain: String) override fun onLoginFetch(domain: String)
: GeckoResult<Array<LoginEntry>>? { : GeckoResult<Array<LoginEntry>>? {
@ -1178,11 +1179,11 @@ class AutocompleteTest : BaseSessionTest() {
// a. Ensure onLoginSave is called with accordingly. // a. Ensure onLoginSave is called with accordingly.
val runtime = sessionRule.runtime val runtime = sessionRule.runtime
val register = { delegate: LoginStorageDelegate -> val register = { delegate: StorageDelegate ->
runtime.loginStorageDelegate = delegate runtime.autocompleteStorageDelegate = delegate
} }
val unregister = { _: LoginStorageDelegate -> val unregister = { _: StorageDelegate ->
runtime.loginStorageDelegate = null runtime.autocompleteStorageDelegate = null
} }
val user1 = "user1x" val user1 = "user1x"
@ -1193,8 +1194,8 @@ class AutocompleteTest : BaseSessionTest() {
var numSelects = 0 var numSelects = 0
sessionRule.addExternalDelegateUntilTestEnd( sessionRule.addExternalDelegateUntilTestEnd(
LoginStorageDelegate::class, register, unregister, StorageDelegate::class, register, unregister,
object : LoginStorageDelegate { object : StorageDelegate {
@AssertCalled @AssertCalled
override fun onLoginFetch(domain: String) override fun onLoginFetch(domain: String)
: GeckoResult<Array<LoginEntry>>? { : GeckoResult<Array<LoginEntry>>? {
@ -1244,7 +1245,7 @@ class AutocompleteTest : BaseSessionTest() {
assertThat( assertThat(
"Hint should match", "Hint should match",
option.hint, option.hint,
equalTo(LoginSelectOption.Hint.GENERATED)) equalTo(SelectOption.Hint.GENERATED))
assertThat("Login should not be null", login, notNullValue()) assertThat("Login should not be null", login, notNullValue())
assertThat( assertThat(

View File

@ -62,7 +62,7 @@ import org.mozilla.gecko.util.GeckoBundle;
* <p> * <p>
* With the document parsed and the login input fields identified, GeckoView * With the document parsed and the login input fields identified, GeckoView
* dispatches a * dispatches a
* <code>LoginStorageDelegate.onLoginFetch(&quot;example.com&quot;)</code> * <code>StorageDelegate.onLoginFetch(&quot;example.com&quot;)</code>
* request to fetch logins for the given domain. * request to fetch logins for the given domain.
* </p> * </p>
* <p> * <p>
@ -82,14 +82,14 @@ import org.mozilla.gecko.util.GeckoBundle;
* <h3>Update API</h3> * <h3>Update API</h3>
* <p> * <p>
* When the user submits some login input fields, GeckoView dispatches another * When the user submits some login input fields, GeckoView dispatches another
* <code>LoginStorageDelegate.onLoginFetch(&quot;example.com&quot;)</code> * <code>StorageDelegate.onLoginFetch(&quot;example.com&quot;)</code>
* request to check whether the submitted login exists or whether it's a new or * request to check whether the submitted login exists or whether it's a new or
* updated login entry. * updated login entry.
* </p> * </p>
* <p> * <p>
* If the submitted login is already contained as-is in the collection returned * If the submitted login is already contained as-is in the collection returned
* by <code>onLoginFetch</code>, then GeckoView dispatches * by <code>onLoginFetch</code>, then GeckoView dispatches
* <code>LoginStorageDelegate.onLoginUsed</code> with the submitted login * <code>StorageDelegate.onLoginUsed</code> with the submitted login
* entry. * entry.
* </p> * </p>
* <p> * <p>
@ -120,12 +120,12 @@ import org.mozilla.gecko.util.GeckoBundle;
* <p> * <p>
* The login entry returned in a confirmed save prompt is used to request for * The login entry returned in a confirmed save prompt is used to request for
* saving in the runtime delegate via * saving in the runtime delegate via
* <code>LoginStorageDelegate.onLoginSave(login)</code>. * <code>StorageDelegate.onLoginSave(login)</code>.
* If the app has already stored the entry during the prompt request handling, * If the app has already stored the entry during the prompt request handling,
* it may ignore this storage saving request. * it may ignore this storage saving request.
* </p> * </p>
* *
* <br>@see GeckoRuntime#setLoginStorageDelegate * <br>@see GeckoRuntime#setAutocompleteStorageDelegate
* <br>@see GeckoSession#setPromptDelegate * <br>@see GeckoSession#setPromptDelegate
* <br>@see GeckoSession.PromptDelegate#onLoginSave * <br>@see GeckoSession.PromptDelegate#onLoginSave
* <br>@see GeckoSession.PromptDelegate#onLoginSelect * <br>@see GeckoSession.PromptDelegate#onLoginSelect
@ -136,6 +136,172 @@ public class Autocomplete {
protected Autocomplete() {} protected Autocomplete() {}
/**
* Holds credit card information for a specific entry.
*/
public static class CreditCard {
private static final String GUID_KEY = "guid";
private static final String NAME_KEY = "name";
private static final String NUMBER_KEY = "number";
private static final String EXP_MONTH_KEY = "expMonth";
private static final String EXP_YEAR_KEY = "expYear";
/**
* The unique identifier for this login entry.
*/
public final @Nullable String guid;
/**
* The full name as it appears on the credit card.
*/
public final @NonNull String name;
/**
* The credit card number.
*/
public final @NonNull String number;
/**
* The expiration month.
*/
public final @NonNull String expirationMonth;
/**
* The expiration year.
*/
public final @NonNull String expirationYear;
// For tests only.
@AnyThread
protected CreditCard() {
guid = null;
name = "";
number = "";
expirationMonth = "";
expirationYear = "";
}
@AnyThread
/* package */ CreditCard(final @NonNull GeckoBundle bundle) {
guid = bundle.getString(GUID_KEY);
name = bundle.getString(NAME_KEY, "");
number = bundle.getString(NUMBER_KEY, "");
expirationMonth = bundle.getString(EXP_MONTH_KEY, "");
expirationYear = bundle.getString(EXP_YEAR_KEY, "");
}
@Override
@AnyThread
public String toString() {
final StringBuilder builder = new StringBuilder("CreditCard {");
builder
.append("guid=").append(guid)
.append(", name=").append(name)
.append(", number=").append(number)
.append(", expirationMonth=").append(expirationMonth)
.append(", expirationYear=").append(expirationYear)
.append("}");
return builder.toString();
}
@AnyThread
/* package */ @NonNull GeckoBundle toBundle() {
final GeckoBundle bundle = new GeckoBundle(7);
bundle.putString(GUID_KEY, guid);
bundle.putString(NAME_KEY, name);
bundle.putString(NUMBER_KEY, number);
bundle.putString(EXP_MONTH_KEY, expirationMonth);
bundle.putString(EXP_YEAR_KEY, expirationYear);
return bundle;
}
public static class Builder {
private final GeckoBundle mBundle;
@AnyThread
/* package */ Builder(final @NonNull GeckoBundle bundle) {
mBundle = new GeckoBundle(bundle);
}
@AnyThread
@SuppressWarnings("checkstyle:javadocmethod")
public Builder() {
mBundle = new GeckoBundle(7);
}
/**
* Finalize the {@link CreditCard} instance.
*
* @return The {@link CreditCard} instance.
*/
@AnyThread
public @NonNull CreditCard build() {
return new CreditCard(mBundle);
}
/**
* Set the unique identifier for this credit card entry.
*
* @param guid The unique identifier string.
* @return This {@link Builder} instance.
*/
@AnyThread
public @NonNull Builder guid(final @Nullable String guid) {
mBundle.putString(GUID_KEY, guid);
return this;
}
/**
* Set the name for this credit card entry.
*
* @param name The full name as it appears on the credit card.
* @return This {@link Builder} instance.
*/
@AnyThread
public @NonNull Builder name(final @Nullable String name) {
mBundle.putString(NAME_KEY, name);
return this;
}
/**
* Set the number for this credit card entry.
*
* @param number The credit card number string.
* @return This {@link Builder} instance.
*/
@AnyThread
public @NonNull Builder number(final @Nullable String number) {
mBundle.putString(NUMBER_KEY, number);
return this;
}
/**
* Set the expiration month for this credit card entry.
*
* @param expMonth The expiration month string.
* @return This {@link Builder} instance.
*/
@AnyThread
public @NonNull Builder expirationMonth(final @Nullable String expMonth) {
mBundle.putString(EXP_MONTH_KEY, expMonth);
return this;
}
/**
* Set the expiration year for this credit card entry.
*
* @param expYear The expiration year string.
* @return This {@link Builder} instance.
*/
@AnyThread
public @NonNull Builder expirationYear(final @Nullable String expYear) {
mBundle.putString(EXP_YEAR_KEY, expYear);
return this;
}
}
}
/** /**
* Holds login information for a specific entry. * Holds login information for a specific entry.
*/ */
@ -337,7 +503,7 @@ public class Autocomplete {
// Sync with UsedField in GeckoViewAutocomplete.jsm. // Sync with UsedField in GeckoViewAutocomplete.jsm.
/** /**
* Possible login entry field types for {@link LoginStorageDelegate#onLoginUsed}. * Possible login entry field types for {@link StorageDelegate#onLoginUsed}.
*/ */
public static class UsedField { public static class UsedField {
/** /**
@ -353,9 +519,9 @@ public class Autocomplete {
* Login storage events include login entry requests for autofill and * Login storage events include login entry requests for autofill and
* autocompletion of login input fields. * autocompletion of login input fields.
* This delegate is attached to the runtime via * This delegate is attached to the runtime via
* {@link GeckoRuntime#setLoginStorageDelegate}. * {@link GeckoRuntime#setAutocompleteStorageDelegate}.
*/ */
public interface LoginStorageDelegate { public interface StorageDelegate {
/** /**
* Request login entries for a given domain. * Request login entries for a given domain.
* While processing the web document, we have identified elements * While processing the web document, we have identified elements
@ -374,6 +540,21 @@ public class Autocomplete {
return null; return null;
} }
/**
* Request credit card entries.
* While processing the web document, we have identified elements
* resembling credit card input fields suitable for autofill.
* We will attempt to match the provided credit card information to the
* identified input fields.
*
* @return A {@link GeckoResult} that completes with an array of
* {@link CreditCard} containing the existing credit cards.
*/
@UiThread
default @Nullable GeckoResult<CreditCard[]> onCreditCardFetch() {
return null;
}
/** /**
* Request saving or updating of the given login entry. * Request saving or updating of the given login entry.
* This is triggered by confirming a * This is triggered by confirming a
@ -401,6 +582,13 @@ public class Autocomplete {
@LSUsedField final int usedFields) {} @LSUsedField final int usedFields) {}
} }
/**
* @deprecated This API has been replaced by {@link StorageDelegate} and
* will be removed in GeckoView 93.
*/
@Deprecated @DeprecationSchedule(version = 93, id = "login-storage")
public interface LoginStorageDelegate extends StorageDelegate {}
/** /**
* Abstract base class for Autocomplete options. * Abstract base class for Autocomplete options.
* Extended by {@link Autocomplete.SaveOption} and * Extended by {@link Autocomplete.SaveOption} and
@ -428,44 +616,10 @@ public class Autocomplete {
* Extended by {@link Autocomplete.LoginSaveOption}. * Extended by {@link Autocomplete.LoginSaveOption}.
*/ */
public abstract static class SaveOption<T> extends Option<T> { public abstract static class SaveOption<T> extends Option<T> {
@SuppressWarnings("checkstyle:javadocmethod")
public SaveOption(final @NonNull T value, final int hint) {
super(value, hint);
}
}
/**
* Abstract base class for saving options.
* Extended by {@link Autocomplete.LoginSelectOption}.
*/
public abstract static class SelectOption<T> extends Option<T> {
@SuppressWarnings("checkstyle:javadocmethod")
public SelectOption(
final @NonNull T value,
final int hint) {
super(value, hint);
}
@Override
public String toString() {
final StringBuilder builder = new StringBuilder("SelectOption {");
builder
.append("value=").append(value).append(", ")
.append("hint=").append(hint)
.append("}");
return builder.toString();
}
}
/**
* Holds information required to process login saving requests.
*/
public static class LoginSaveOption extends SaveOption<LoginEntry> {
@Retention(RetentionPolicy.SOURCE) @Retention(RetentionPolicy.SOURCE)
@IntDef(flag = true, @IntDef(flag = true,
value = { Hint.NONE, Hint.GENERATED, Hint.LOW_CONFIDENCE }) value = { Hint.NONE, Hint.GENERATED, Hint.LOW_CONFIDENCE })
/* package */ @interface LoginSaveHint {} /* package */ @interface SaveOptionHint {}
/** /**
* Hint types for login saving requests. * Hint types for login saving requests.
@ -491,6 +645,84 @@ public class Autocomplete {
protected Hint() {} protected Hint() {}
} }
@SuppressWarnings("checkstyle:javadocmethod")
public SaveOption(
final @NonNull T value,
final @SaveOptionHint int hint) {
super(value, hint);
}
}
/**
* Abstract base class for saving options.
* Extended by {@link Autocomplete.LoginSelectOption}.
*/
public abstract static class SelectOption<T> extends Option<T> {
@Retention(RetentionPolicy.SOURCE)
@IntDef(flag = true,
value = { Hint.NONE, Hint.GENERATED, Hint.INSECURE_FORM,
Hint.DUPLICATE_USERNAME, Hint.MATCHING_ORIGIN })
/* package */ @interface SelectOptionHint {}
/**
* Hint types for selection requests.
*/
public static class Hint {
public static final int NONE = 0;
/**
* Auto-generated password.
* A new password-only login entry containing a secure generated
* password.
*/
public static final int GENERATED = 1 << 0;
/**
* Insecure context.
* The form or transmission mechanics are considered insecure.
* This is the case when the form is served via http or submitted
* insecurely.
*/
public static final int INSECURE_FORM = 1 << 1;
/**
* The username is shared with another login entry.
* There are multiple login entries in the options that share the
* same username. You may have to disambiguate the login entry,
* e.g., using the last date of modification and its origin.
*/
public static final int DUPLICATE_USERNAME = 1 << 2;
/**
* The login entry's origin matches the login form origin.
* The login was saved from the same origin it is being requested
* for, rather than for a subdomain.
*/
public static final int MATCHING_ORIGIN = 1 << 3;
}
@SuppressWarnings("checkstyle:javadocmethod")
public SelectOption(
final @NonNull T value,
final @SelectOptionHint int hint) {
super(value, hint);
}
@Override
public String toString() {
final StringBuilder builder = new StringBuilder("SelectOption {");
builder
.append("value=").append(value).append(", ")
.append("hint=").append(hint)
.append("}");
return builder.toString();
}
}
/**
* Holds information required to process login saving requests.
*/
public static class LoginSaveOption extends SaveOption<LoginEntry> {
/** /**
* Construct a login save option. * Construct a login save option.
* *
@ -499,7 +731,7 @@ public class Autocomplete {
*/ */
/* package */ LoginSaveOption( /* package */ LoginSaveOption(
final @NonNull LoginEntry value, final @NonNull LoginEntry value,
final @LoginSaveHint int hint) { final @SaveOptionHint int hint) {
super(value, hint); super(value, hint);
} }
@ -525,49 +757,6 @@ public class Autocomplete {
* Holds information required to process login selection requests. * Holds information required to process login selection requests.
*/ */
public static class LoginSelectOption extends SelectOption<LoginEntry> { public static class LoginSelectOption extends SelectOption<LoginEntry> {
@Retention(RetentionPolicy.SOURCE)
@IntDef(flag = true,
value = { Hint.NONE, Hint.GENERATED, Hint.INSECURE_FORM,
Hint.DUPLICATE_USERNAME, Hint.MATCHING_ORIGIN })
/* package */ @interface LoginSelectHint {}
/**
* Hint types for login selection requests.
*/
public static class Hint {
public static final int NONE = 0;
/**
* Auto-generated password.
* A new password-only login entry containing a secure generated
* password.
*/
public static final int GENERATED = 1 << 0;
/**
* Insecure login.
* The login form or transmission mechanics are considered insecure.
* This is the case when the form is served via http or submitted
* insecurely.
*/
public static final int INSECURE_FORM = 1 << 1;
/**
* The username is shared with another login entry.
* There are multiple login entries in the options that share the
* same username. You may have to disambiguate the login entry,
* e.g., using the last date of modification and its origin.
*/
public static final int DUPLICATE_USERNAME = 1 << 2;
/**
* The login entry's origin matches the login form origin.
* The login was saved from the same origin it is being requested
* for, rather than for a subdomain.
*/
public static final int MATCHING_ORIGIN = 1 << 3;
}
/** /**
* Construct a login select option. * Construct a login select option.
* *
@ -576,7 +765,7 @@ public class Autocomplete {
*/ */
/* package */ LoginSelectOption( /* package */ LoginSelectOption(
final @NonNull LoginEntry value, final @NonNull LoginEntry value,
final @LoginSelectHint int hint) { final @SelectOptionHint int hint) {
super(value, hint); super(value, hint);
} }
@ -606,24 +795,87 @@ public class Autocomplete {
} }
} }
/* package */ final static class LoginStorageProxy implements BundleEventListener { /**
private static final String LOGTAG = "LoginStorageProxy"; * Holds information required to process credit card selection requests.
*/
public static class CreditCardSelectOption extends SelectOption<CreditCard> {
@Retention(RetentionPolicy.SOURCE)
@IntDef(flag = true,
value = { Hint.NONE, Hint.INSECURE_FORM })
/* package */ @interface CreditCardSelectHint {}
/**
* Hint types for credit card selection requests.
*/
public static class Hint {
public static final int NONE = 0;
/**
* Insecure context.
* The form or transmission mechanics are considered insecure.
* This is the case when the form is served via http or submitted
* insecurely.
*/
public static final int INSECURE_FORM = 1 << 1;
}
/**
* Construct a credit card select option.
*
* @param value The {@link LoginEntry} credit card entry selection option.
* @param hint The {@link Hint} detailing the type of the option.
*/
/* package */ CreditCardSelectOption(
final @NonNull CreditCard value,
final @CreditCardSelectHint int hint) {
super(value, hint);
}
/**
* Construct a credit card select option.
*
* @param value The {@link CreditCard} credit card entry selection option.
*/
public CreditCardSelectOption(final @NonNull CreditCard value) {
this(value, Hint.NONE);
}
/* package */ static @NonNull CreditCardSelectOption fromBundle(
final @NonNull GeckoBundle bundle) {
final int hint = bundle.getInt("hint");
final CreditCard value = new CreditCard(bundle.getBundle("value"));
return new CreditCardSelectOption(value, hint);
}
@Override
/* package */ @NonNull GeckoBundle toBundle() {
final GeckoBundle bundle = new GeckoBundle(2);
bundle.putBundle(VALUE_KEY, value.toBundle());
bundle.putInt(HINT_KEY, hint);
return bundle;
}
}
/* package */ final static class StorageProxy implements BundleEventListener {
private static final String FETCH_LOGIN_EVENT = private static final String FETCH_LOGIN_EVENT =
"GeckoView:Autocomplete:Fetch:Login"; "GeckoView:Autocomplete:Fetch:Login";
private static final String FETCH_CREDIT_CARD_EVENT =
"GeckoView:Autocomplete:Fetch:CreditCard";
private static final String SAVE_LOGIN_EVENT = private static final String SAVE_LOGIN_EVENT =
"GeckoView:Autocomplete:Save:Login"; "GeckoView:Autocomplete:Save:Login";
private static final String USED_LOGIN_EVENT = private static final String USED_LOGIN_EVENT =
"GeckoView:Autocomplete:Used:Login"; "GeckoView:Autocomplete:Used:Login";
private @Nullable LoginStorageDelegate mDelegate; private @Nullable StorageDelegate mDelegate;
public LoginStorageProxy() {} public StorageProxy() {}
private void registerListener() { private void registerListener() {
EventDispatcher.getInstance().registerUiThreadListener( EventDispatcher.getInstance().registerUiThreadListener(
this, this,
FETCH_LOGIN_EVENT, FETCH_LOGIN_EVENT,
FETCH_CREDIT_CARD_EVENT,
SAVE_LOGIN_EVENT, SAVE_LOGIN_EVENT,
USED_LOGIN_EVENT); USED_LOGIN_EVENT);
} }
@ -632,22 +884,28 @@ public class Autocomplete {
EventDispatcher.getInstance().unregisterUiThreadListener( EventDispatcher.getInstance().unregisterUiThreadListener(
this, this,
FETCH_LOGIN_EVENT, FETCH_LOGIN_EVENT,
FETCH_CREDIT_CARD_EVENT,
SAVE_LOGIN_EVENT, SAVE_LOGIN_EVENT,
USED_LOGIN_EVENT); USED_LOGIN_EVENT);
} }
public synchronized void setDelegate( public synchronized void setDelegate(
final @Nullable LoginStorageDelegate delegate) { final @Nullable StorageDelegate delegate) {
if (mDelegate == null && delegate != null) { if (mDelegate == delegate) {
registerListener(); return;
} else if (mDelegate != null && delegate == null) { }
if (mDelegate != null) {
unregisterListener(); unregisterListener();
} }
mDelegate = delegate; mDelegate = delegate;
if (mDelegate != null) {
registerListener();
}
} }
public synchronized @Nullable LoginStorageDelegate getDelegate() { public synchronized @Nullable StorageDelegate getDelegate() {
return mDelegate; return mDelegate;
} }
@ -662,7 +920,7 @@ public class Autocomplete {
if (mDelegate == null) { if (mDelegate == null) {
if (callback != null) { if (callback != null) {
callback.sendError("No LoginStorageDelegate attached"); callback.sendError("No StorageDelegate attached");
} }
return; return;
} }
@ -691,6 +949,29 @@ public class Autocomplete {
return loginBundles; return loginBundles;
})); }));
} else if (FETCH_CREDIT_CARD_EVENT.equals(event)) {
final GeckoResult<Autocomplete.CreditCard[]> result =
mDelegate.onCreditCardFetch();
if (result == null) {
callback.sendSuccess(new GeckoBundle[0]);
return;
}
callback.resolveTo(result.map(creditCards -> {
if (creditCards == null) {
return new GeckoBundle[0];
}
// This is a one-liner with streams (API level 24).
final GeckoBundle[] creditCardBundles =
new GeckoBundle[creditCards.length];
for (int i = 0; i < creditCards.length; ++i) {
creditCardBundles[i] = creditCards[i].toBundle();
}
return creditCardBundles;
}));
} else if (SAVE_LOGIN_EVENT.equals(event)) { } else if (SAVE_LOGIN_EVENT.equals(event)) {
final GeckoBundle loginBundle = message.getBundle("login"); final GeckoBundle loginBundle = message.getBundle("login");
final LoginEntry login = new LoginEntry(loginBundle); final LoginEntry login = new LoginEntry(loginBundle);

View File

@ -173,13 +173,13 @@ public final class GeckoRuntime implements Parcelable {
private final WebExtensionController mWebExtensionController; private final WebExtensionController mWebExtensionController;
private WebPushController mPushController; private WebPushController mPushController;
private final ContentBlockingController mContentBlockingController; private final ContentBlockingController mContentBlockingController;
private final Autocomplete.LoginStorageProxy mLoginStorageProxy; private final Autocomplete.StorageProxy mAutocompleteStorageProxy;
private final ProfilerController mProfilerController; private final ProfilerController mProfilerController;
private GeckoRuntime() { private GeckoRuntime() {
mWebExtensionController = new WebExtensionController(this); mWebExtensionController = new WebExtensionController(this);
mContentBlockingController = new ContentBlockingController(); mContentBlockingController = new ContentBlockingController();
mLoginStorageProxy = new Autocomplete.LoginStorageProxy(); mAutocompleteStorageProxy = new Autocomplete.StorageProxy();
mProfilerController = new ProfilerController(); mProfilerController = new ProfilerController();
if (sRuntime != null) { if (sRuntime != null) {
@ -566,28 +566,63 @@ public final class GeckoRuntime implements Parcelable {
} }
/** /**
* Set the {@link Autocomplete.LoginStorageDelegate} instance on this runtime. * Set the {@link Autocomplete.StorageDelegate} instance on this runtime.
* This delegate is required for handling login storage requests. * This delegate is required for handling autocomplete storage requests.
* *
* @param delegate The {@link Autocomplete.LoginStorageDelegate} handling login storage * @param delegate The {@link Autocomplete.StorageDelegate} handling
* requests. * autocomplete storage requests.
*/ */
@UiThread @UiThread
public void setAutocompleteStorageDelegate(
final @Nullable Autocomplete.StorageDelegate delegate) {
ThreadUtils.assertOnUiThread();
mAutocompleteStorageProxy.setDelegate(delegate);
}
/**
* Set the {@link Autocomplete.LoginStorageDelegate} instance on this runtime.
* This delegate is required for handling autocomplete storage requests.
*
* @param delegate The {@link Autocomplete.LoginStorageDelegate} handling
* autocomplete storage requests.
*
* @deprecated This API has been replaced by
* {@link #setAutocompleteStorageDelegate} and
* will be removed in GeckoView 93.
*/
@Deprecated @DeprecationSchedule(version = 93, id = "login-storage")
@UiThread
public void setLoginStorageDelegate( public void setLoginStorageDelegate(
final @Nullable Autocomplete.LoginStorageDelegate delegate) { final @Nullable Autocomplete.LoginStorageDelegate delegate) {
ThreadUtils.assertOnUiThread(); ThreadUtils.assertOnUiThread();
mLoginStorageProxy.setDelegate(delegate); mAutocompleteStorageProxy.setDelegate(delegate);
}
/**
* Get the {@link Autocomplete.StorageDelegate} instance set on this runtime.
*
* @return The {@link Autocomplete.StorageDelegate} set on this runtime.
*/
@UiThread
public @Nullable Autocomplete.StorageDelegate getAutocompleteStorageDelegate() {
ThreadUtils.assertOnUiThread();
return mAutocompleteStorageProxy.getDelegate();
} }
/** /**
* Get the {@link Autocomplete.LoginStorageDelegate} instance set on this runtime. * Get the {@link Autocomplete.LoginStorageDelegate} instance set on this runtime.
* *
* @return The {@link Autocomplete.LoginStorageDelegate} set on this runtime. * @return The {@link Autocomplete.LoginStorageDelegate} set on this runtime.
*
* @deprecated This API has been replaced by
* {@link #getAutocompleteStorageDelegate} and
* will be removed in GeckoView 93.
*/ */
@Deprecated @DeprecationSchedule(version = 93, id = "login-storage")
@UiThread @UiThread
public @Nullable Autocomplete.LoginStorageDelegate getLoginStorageDelegate() { public @Nullable Autocomplete.LoginStorageDelegate getLoginStorageDelegate() {
ThreadUtils.assertOnUiThread(); ThreadUtils.assertOnUiThread();
return mLoginStorageProxy.getDelegate(); return (Autocomplete.LoginStorageDelegate)mAutocompleteStorageProxy.getDelegate();
} }
@UiThread @UiThread

View File

@ -294,7 +294,7 @@ public final class GeckoRuntimeSettings extends RuntimeSettings {
/** /**
* Set whether login forms should be filled automatically if only one * Set whether login forms should be filled automatically if only one
* viable candidate is provided via * viable candidate is provided via
* {@link Autocomplete.LoginStorageDelegate#onLoginFetch onLoginFetch}. * {@link Autocomplete.StorageDelegate#onLoginFetch onLoginFetch}.
* *
* @param enabled A flag determining whether login autofill should be * @param enabled A flag determining whether login autofill should be
* enabled. * enabled.
@ -1181,7 +1181,7 @@ public final class GeckoRuntimeSettings extends RuntimeSettings {
/** /**
* Set whether login forms should be filled automatically if only one * Set whether login forms should be filled automatically if only one
* viable candidate is provided via * viable candidate is provided via
* {@link Autocomplete.LoginStorageDelegate#onLoginFetch onLoginFetch}. * {@link Autocomplete.StorageDelegate#onLoginFetch onLoginFetch}.
* *
* @param enabled A flag determining whether login autofill should be * @param enabled A flag determining whether login autofill should be
* enabled. * enabled.

View File

@ -2860,7 +2860,29 @@ public class GeckoSession {
new PromptDelegate.AutocompleteRequest<>(options); new PromptDelegate.AutocompleteRequest<>(options);
res = delegate.onLoginSelect(session, request); res = delegate.onLoginSelect(session, request);
break;
}
case "Autocomplete:Select:CreditCard": {
final GeckoBundle[] optionBundles =
message.getBundleArray("options");
if (optionBundles == null) {
break;
}
final Autocomplete.CreditCardSelectOption[] options =
new Autocomplete.CreditCardSelectOption[optionBundles.length];
for (int i = 0; i < options.length; ++i) {
options[i] = Autocomplete.CreditCardSelectOption.fromBundle(
optionBundles[i]);
}
final PromptDelegate.AutocompleteRequest
<Autocomplete.CreditCardSelectOption> request =
new PromptDelegate.AutocompleteRequest<>(options);
res = delegate.onCreditCardSelect(session, request);
break; break;
} }
default: { default: {
@ -5143,7 +5165,7 @@ public class GeckoSession {
* *
* Confirm the request with an {@link Autocomplete.Option} * Confirm the request with an {@link Autocomplete.Option}
* to trigger a * to trigger a
* {@link Autocomplete.LoginStorageDelegate#onLoginSave} request * {@link Autocomplete.StorageDelegate#onLoginSave} request
* to save the given selection. * to save the given selection.
* The confirmed selection may be an entry out of the request's * The confirmed selection may be an entry out of the request's
* options, a modified option, or a freshly created login entry. * options, a modified option, or a freshly created login entry.
@ -5184,6 +5206,34 @@ public class GeckoSession {
request) { request) {
return null; return null;
} }
/**
* Handle a credit card selection prompt request.
* This is triggered by the user focusing on a credit card input field.
*
* @param session The {@link GeckoSession} that triggered the request.
* @param request The {@link AutocompleteRequest} containing the request
* details.
*
* @return A {@link GeckoResult} resolving to a {@link PromptResponse}
*
* Confirm the request with an {@link Autocomplete.Option}
* to let GeckoView fill out the credit card forms with the given
* selection details.
* The confirmed selection may be an entry out of the request's
* options, a modified option, or a freshly created credit
* card entry.
*
* Dismiss the request to deny autocompletion for the detected
* form.
*/
@UiThread
default @Nullable GeckoResult<PromptResponse> onCreditCardSelect(
@NonNull final GeckoSession session,
@NonNull final AutocompleteRequest<Autocomplete.CreditCardSelectOption>
request) {
return null;
}
} }
/** /**

View File

@ -7,6 +7,8 @@
const EXPORTED_SYMBOLS = [ const EXPORTED_SYMBOLS = [
"GeckoViewAutocomplete", "GeckoViewAutocomplete",
"LoginEntry", "LoginEntry",
"CreditCard",
"Address",
"SelectOption", "SelectOption",
]; ];
@ -106,8 +108,189 @@ class LoginEntry {
} }
} }
class Address {
constructor({
name,
givenName,
additionalName,
familyName,
organization,
streetAddress,
addressLevel1,
addressLevel2,
addressLevel3,
postalCode,
country,
tel,
email,
guid,
timeCreated,
timeLastUsed,
timeLastModified,
timesUsed,
version,
}) {
this.name = name ?? null;
this.givenName = givenName ?? null;
this.additionalName = additionalName ?? null;
this.familyName = familyName ?? null;
this.organization = organization ?? null;
this.streetAddress = streetAddress ?? null;
this.addressLevel1 = addressLevel1 ?? null;
this.addressLevel2 = addressLevel2 ?? null;
this.addressLevel3 = addressLevel3 ?? null;
this.postalCode = postalCode ?? null;
this.country = country ?? null;
this.tel = tel ?? null;
this.email = email ?? null;
// Metadata.
this.guid = guid ?? null;
// TODO: Not supported by GV.
this.timeCreated = timeCreated ?? null;
this.timeLastUsed = timeLastUsed ?? null;
this.timeLastModified = timeLastModified ?? null;
this.timesUsed = timesUsed ?? null;
this.version = version ?? null;
}
isValid() {
return (
(this.name ?? this.givenName ?? this.familyName) !== null &&
this.streetAddress !== null &&
this.postalCode !== null
);
}
static fromGecko(aObj) {
return new Address({
version: aObj.version,
name: aObj.name,
givenName: aObj["given-name"],
additionalName: aObj["additional-name"],
familyName: aObj["family-name"],
organization: aObj.organization,
streetAddress: aObj["street-address"],
addressLevel1: aObj["address-level1"],
addressLevel2: aObj["address-level2"],
addressLevel3: aObj["address-level3"],
postalCode: aObj["postal-code"],
country: aObj.country,
tel: aObj.tel,
email: aObj.email,
guid: aObj.guid,
timeCreated: aObj.timeCreated,
timeLastUsed: aObj.timeLastUsed,
timeLastModified: aObj.timeLastModified,
timesUsed: aObj.timesUsed,
});
}
static parse(aObj) {
const entry = new Address({});
Object.assign(entry, aObj);
return entry;
}
toGecko() {
return {
version: this.version,
name: this.name,
"given-name": this.givenName,
"additional-name": this.additionalName,
"family-name": this.familyName,
organization: this.organization,
"street-address": this.streetAddress,
"address-level1": this.addressLevel1,
"address-level2": this.addressLevel2,
"address-level3": this.addressLevel3,
"postal-code": this.postalCode,
country: this.country,
tel: this.tel,
email: this.email,
guid: this.guid,
};
}
}
class CreditCard {
constructor({
name,
number,
expMonth,
expYear,
type,
guid,
timeCreated,
timeLastUsed,
timeLastModified,
timesUsed,
version,
}) {
this.name = name ?? null;
this.number = number ?? null;
this.expMonth = expMonth ?? null;
this.expYear = expYear ?? null;
this.type = type ?? null;
// Metadata.
this.guid = guid ?? null;
// TODO: Not supported by GV.
this.timeCreated = timeCreated ?? null;
this.timeLastUsed = timeLastUsed ?? null;
this.timeLastModified = timeLastModified ?? null;
this.timesUsed = timesUsed ?? null;
this.version = version ?? null;
}
isValid() {
return (
this.name !== null &&
this.number !== null &&
this.expMonth !== null &&
this.expYear !== null
);
}
static fromGecko(aObj) {
return new CreditCard({
version: aObj.version,
name: aObj["cc-name"],
number: aObj["cc-number"],
expMonth: aObj["cc-exp-month"],
expYear: aObj["cc-exp-year"],
type: aObj["cc-type"],
guid: aObj.guid,
timeCreated: aObj.timeCreated,
timeLastUsed: aObj.timeLastUsed,
timeLastModified: aObj.timeLastModified,
timesUsed: aObj.timesUsed,
});
}
static parse(aObj) {
const entry = new CreditCard({});
Object.assign(entry, aObj);
return entry;
}
toGecko() {
return {
version: this.version,
"cc-name": this.name,
"cc-number": this.number,
"cc-exp-month": this.expMonth,
"cc-exp-year": this.expYear,
"cc-type": this.type,
guid: this.guid,
};
}
}
class SelectOption { class SelectOption {
// Sync with Autocomplete.LoginSelectOption.Hint in Autocomplete.java. // Sync with Autocomplete.SelectOption.Hint in Autocomplete.java.
static Hint = { static Hint = {
NONE: 0, NONE: 0,
GENERATED: 1 << 0, GENERATED: 1 << 0,
@ -160,7 +343,9 @@ const GeckoViewAutocomplete = {
fetchCreditCards() { fetchCreditCards() {
debug`fetchCreditCards`; debug`fetchCreditCards`;
return Promise.resolve(null); return EventDispatcher.instance.sendRequestForResult({
type: "GeckoView:Autocomplete:Fetch:CreditCard",
});
}, },
/** /**
@ -189,6 +374,11 @@ const GeckoViewAutocomplete = {
*/ */
onCreditCardSave(aCreditCard) { onCreditCardSave(aCreditCard) {
debug`onLoginSave ${aCreditCard}`; debug`onLoginSave ${aCreditCard}`;
EventDispatcher.instance.sendRequest({
type: "GeckoView:Autocomplete:Save:CreditCard",
creditCard: aCreditCard,
});
}, },
/** /**
@ -235,7 +425,8 @@ const GeckoViewAutocomplete = {
}); });
}, },
_numActiveOnLoginSelect: 0, _numActiveSelections: 0,
/** /**
* Delegates login entry selection. * Delegates login entry selection.
* Call this when there are multiple login entry option for a form to delegate * Call this when there are multiple login entry option for a form to delegate
@ -276,6 +467,60 @@ const GeckoViewAutocomplete = {
}); });
}, },
/**
* Delegates credit card entry selection.
* Call this when there are multiple credit card entry option for a form to delegate
* the selection.
*
* @param aBrowser The browser instance the triggered the selection.
* @param aOptions The list of {SelectOption} depicting viable options.
*/
onCreditCardSelect(aBrowser, aOptions) {
debug`onCreditCardSelect ${aOptions}`;
return new Promise((resolve, reject) => {
if (!aBrowser || !aOptions) {
debug`onCreditCardSelect Rejecting - no browser or options provided`;
reject();
return;
}
const prompt = new GeckoViewPrompter(aBrowser.ownerGlobal);
prompt.asyncShowPrompt(
{
type: "Autocomplete:Select:CreditCard",
options: aOptions,
},
result => {
if (!result || !result.selection) {
reject();
return;
}
const option = new SelectOption({
value: CreditCard.parse(result.selection.value),
hint: result.selection.hint,
});
resolve(option);
}
);
});
},
/**
* Delegates address entry selection.
* Call this when there are multiple address entry option for a form to delegate
* the selection.
*
* @param aBrowser The browser instance the triggered the selection.
* @param aOptions The list of {SelectOption} depicting viable options.
*/
onAddressSelect(aBrowser, aOptions) {
debug`onAddressSelect ${aOptions}`;
return Promise.resolve(null);
},
async delegateSelection({ async delegateSelection({
browsingContext, browsingContext,
options, options,
@ -291,6 +536,8 @@ const GeckoViewAutocomplete = {
let insecureHint = SelectOption.Hint.NONE; let insecureHint = SelectOption.Hint.NONE;
let loginStyle = null; let loginStyle = null;
// TODO: Replace this string with more robust mechanics.
let selectionType = null;
const selectOptions = []; const selectOptions = [];
for (const option of options) { for (const option of options) {
@ -301,6 +548,7 @@ const GeckoViewAutocomplete = {
break; break;
} }
case "generatedPassword": { case "generatedPassword": {
selectionType = "login";
const comment = JSON.parse(option.comment); const comment = JSON.parse(option.comment);
selectOptions.push( selectOptions.push(
new SelectOption({ new SelectOption({
@ -315,6 +563,7 @@ const GeckoViewAutocomplete = {
case "login": case "login":
// Fallthrough. // Fallthrough.
case "loginWithOrigin": { case "loginWithOrigin": {
selectionType = "login";
loginStyle = option.style; loginStyle = option.style;
const comment = JSON.parse(option.comment); const comment = JSON.parse(option.comment);
@ -334,6 +583,32 @@ const GeckoViewAutocomplete = {
); );
break; break;
} }
case "autofill-profile": {
const comment = JSON.parse(option.comment);
debug`delegateSelection ${comment}`;
const creditCard = CreditCard.fromGecko(comment);
const address = Address.fromGecko(comment);
if (creditCard.isValid()) {
selectionType = "creditCard";
selectOptions.push(
new SelectOption({
value: creditCard,
hint: insecureHint,
})
);
} else if (address.isValid()) {
selectionType = "address";
selectOptions.push(
new SelectOption({
value: address,
hint: insecureHint,
})
);
}
break;
}
default:
debug`delegateSelection - ignoring unknown option style ${option.style}`;
} }
} }
@ -342,45 +617,79 @@ const GeckoViewAutocomplete = {
return; return;
} }
if (this._numActiveOnLoginSelect > 0) { if (this._numActiveSelections > 0) {
debug`Abort delegateSelection - there is already one delegation active`; debug`Abort delegateSelection - there is already one delegation active`;
return; return;
} }
++this._numActiveOnLoginSelect; ++this._numActiveSelections;
let selectedOption = null;
const browser = browsingContext.top.embedderElement; const browser = browsingContext.top.embedderElement;
const selectedOption = await this.onLoginSelect( if (selectionType === "login") {
browser, selectedOption = await this.onLoginSelect(browser, selectOptions).catch(
selectOptions _ => {
).catch(_ => { debug`No GV delegate attached`;
debug`No GV delegate attached`; }
}); );
} else if (selectionType === "creditCard") {
--this._numActiveOnLoginSelect; selectedOption = await this.onCreditCardSelect(
browser,
debug`delegateSelection selected option: ${selectedOption}`; selectOptions
const selectedLogin = selectedOption?.value?.toLoginInfo(); ).catch(_ => {
debug`No GV delegate attached`;
if (!selectedLogin) { });
debug`Abort delegateSelection - no login entry selected`; } else if (selectionType === "address") {
return; selectedOption = await this.onAddressSelect(browser, selectOptions).catch(
_ => {
debug`No GV delegate attached`;
}
);
} }
debug`delegateSelection - filling form`; --this._numActiveSelections;
const actor = browsingContext.currentWindowGlobal.getActor("LoginManager"); debug`delegateSelection selected option: ${selectedOption}`;
await actor.fillForm({ if (selectionType === "login") {
browser, const selectedLogin = selectedOption?.value?.toLoginInfo();
inputElementIdentifier,
loginFormOrigin: formOrigin, if (!selectedLogin) {
login: selectedLogin, debug`Abort delegateSelection - no login entry selected`;
style: return;
selectedOption.hint & SelectOption.Hint.GENERATED }
? "generatedPassword"
: loginStyle, debug`delegateSelection - filling form`;
});
const actor = browsingContext.currentWindowGlobal.getActor(
"LoginManager"
);
await actor.fillForm({
browser,
inputElementIdentifier,
loginFormOrigin: formOrigin,
login: selectedLogin,
style:
selectedOption.hint & SelectOption.Hint.GENERATED
? "generatedPassword"
: loginStyle,
});
} else if (selectionType === "creditCard") {
const selectedCreditCard = selectedOption?.value?.toGecko();
const actor = browsingContext.currentWindowGlobal.getActor(
"FormAutofill"
);
actor.sendAsyncMessage("FormAutofill:FillForm", selectedCreditCard);
} else if (selectionType === "address") {
const selectedAddress = selectedOption?.value?.toGecko();
const actor = browsingContext.currentWindowGlobal.getActor(
"FormAutofill"
);
actor.sendAsyncMessage("FormAutofill:FillForm", selectedAddress);
}
debug`delegateSelection - form filled`; debug`delegateSelection - form filled`;
}, },

View File

@ -1,153 +0,0 @@
/* This Source Code Form is subject to the terms of the Mozilla Public
* License, v. 2.0. If a copy of the MPL was not distributed with this file,
* You can obtain one at http://mozilla.org/MPL/2.0/. */
"use strict";
const EXPORTED_SYMBOLS = ["GeckoViewLoginStorage", "LoginEntry"];
const { XPCOMUtils } = ChromeUtils.import(
"resource://gre/modules/XPCOMUtils.jsm"
);
const { GeckoViewUtils } = ChromeUtils.import(
"resource://gre/modules/GeckoViewUtils.jsm"
);
XPCOMUtils.defineLazyModuleGetters(this, {
EventDispatcher: "resource://gre/modules/Messaging.jsm",
});
XPCOMUtils.defineLazyGetter(this, "LoginInfo", () =>
Components.Constructor(
"@mozilla.org/login-manager/loginInfo;1",
"nsILoginInfo",
"init"
)
);
class LoginEntry {
constructor() {
this.origin = null;
this.formActionOrigin = null;
this.httpRealm = null;
this.username = null;
this.password = null;
// Metadata.
this.guid = null;
// TODO: Not supported by GV.
this.timeCreated = null;
this.timeLastUsed = null;
this.timePasswordChanged = null;
this.timesUsed = null;
}
toLoginInfo() {
const info = new LoginInfo(
this.origin,
this.formActionOrigin,
this.httpRealm,
this.username,
this.password
);
// Metadata.
info.QueryInterface(Ci.nsILoginMetaInfo);
info.guid = this.guid;
info.timeCreated = this.timeCreated;
info.timeLastUsed = this.timeLastUsed;
info.timePasswordChanged = this.timePasswordChanged;
info.timesUsed = this.timesUsed;
return info;
}
static fromBundle(aObj) {
const entry = new LoginEntry();
Object.assign(entry, aObj);
return entry;
}
static fromLoginInfo(aInfo) {
const entry = new LoginEntry();
entry.origin = aInfo.origin;
entry.formActionOrigin = aInfo.formActionOrigin;
entry.httpRealm = aInfo.httpRealm;
entry.username = aInfo.username;
entry.password = aInfo.password;
// Metadata.
aInfo.QueryInterface(Ci.nsILoginMetaInfo);
entry.guid = aInfo.guid;
entry.timeCreated = aInfo.timeCreated;
entry.timeLastUsed = aInfo.timeLastUsed;
entry.timePasswordChanged = aInfo.timePasswordChanged;
entry.timesUsed = aInfo.timesUsed;
return entry;
}
}
// Sync with LoginStorage.Delegate.UsedField in LoginStorage.java.
const UsedField = { PASSWORD: 1 };
const GeckoViewLoginStorage = {
/**
* Delegates login entry fetching for the given domain to the attached
* LoginStorage GeckoView delegate.
*
* @param aDomain
* The domain string to fetch login entries for.
* @return {Promise}
* Resolves with an array of login objects or null.
* Rejected if no delegate is attached.
* Login object string properties:
* { guid, origin, formActionOrigin, httpRealm, username, password }
*/
fetchLogins(aDomain) {
debug`fetchLogins for ${aDomain}`;
return EventDispatcher.instance.sendRequestForResult({
type: "GeckoView:LoginStorage:Fetch",
domain: aDomain,
});
},
/**
* Delegates login entry saving to the attached LoginStorage GeckoView delegate.
* Call this when a new login entry or a new password for an existing login
* entry has been submitted.
*
* @param aLogin The {LoginEntry} to be saved.
*/
onLoginSave(aLogin) {
debug`onLoginSave ${aLogin}`;
EventDispatcher.instance.sendRequest({
type: "GeckoView:LoginStorage:Save",
login: aLogin,
});
},
/**
* Delegates login entry password usage to the attached LoginStorage GeckoView
* delegate.
* Call this when the password of an existing login entry, as returned by
* fetchLogins, has been used for autofill.
*
* @param aLogin The {LoginEntry} whose password was used.
*/
onLoginPasswordUsed(aLogin) {
debug`onLoginUsed ${aLogin}`;
EventDispatcher.instance.sendRequest({
type: "GeckoView:LoginStorage:Used",
usedFields: UsedField.PASSWORD,
login: aLogin,
});
},
};
const { debug } = GeckoViewUtils.initLogging("GeckoViewLoginStorage");