blob: 626f4f494a8c0bfd549dd644506d96d154e8b2ea [file] [log] [blame]
// Copyright 2017 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
package org.chromium.chrome.browser.autofill.settings;
import android.content.Context;
import android.content.res.Configuration;
import android.graphics.PorterDuff;
import android.graphics.drawable.Drawable;
import android.os.Bundle;
import android.view.Menu;
import android.view.MenuInflater;
import android.view.MenuItem;
import androidx.annotation.NonNull;
import androidx.annotation.Nullable;
import androidx.annotation.VisibleForTesting;
import androidx.preference.Preference;
import androidx.preference.PreferenceFragmentCompat;
import androidx.preference.PreferenceScreen;
import org.chromium.base.ApiCompatibilityUtils;
import org.chromium.base.StrictModeContext;
import org.chromium.chrome.R;
import org.chromium.chrome.browser.autofill.AutofillEditorBase;
import org.chromium.chrome.browser.autofill.PersonalDataManager;
import org.chromium.chrome.browser.autofill.PersonalDataManager.AutofillProfile;
import org.chromium.chrome.browser.autofill.Source;
import org.chromium.chrome.browser.autofill.prefeditor.EditorDialog;
import org.chromium.chrome.browser.feedback.FragmentHelpAndFeedbackLauncher;
import org.chromium.chrome.browser.feedback.HelpAndFeedbackLauncher;
import org.chromium.chrome.browser.payments.AutofillAddress;
import org.chromium.chrome.browser.payments.SettingsAutofillAndPaymentsObserver;
import org.chromium.chrome.browser.profiles.Profile;
import org.chromium.chrome.browser.settings.ChromeManagedPreferenceDelegate;
import org.chromium.chrome.browser.sync.SyncService;
import org.chromium.components.autofill.prefeditor.EditorObserverForTest;
import org.chromium.components.browser_ui.settings.ChromeSwitchPreference;
import org.chromium.components.browser_ui.styles.SemanticColorUtils;
import org.chromium.components.sync.UserSelectableType;
/**
* Autofill profiles fragment, which allows the user to edit autofill profiles.
*/
public class AutofillProfilesFragment
extends PreferenceFragmentCompat implements PersonalDataManager.PersonalDataManagerObserver,
FragmentHelpAndFeedbackLauncher {
private static EditorObserverForTest sObserverForTest;
static final String PREF_NEW_PROFILE = "new_profile";
private @Nullable EditorDialog mEditorDialog;
private HelpAndFeedbackLauncher mHelpAndFeedbackLauncher;
@Override
public void onCreatePreferences(Bundle savedInstanceState, String rootKey) {
getActivity().setTitle(R.string.autofill_addresses_settings_title);
setHasOptionsMenu(true);
PreferenceScreen screen = getPreferenceManager().createPreferenceScreen(getStyledContext());
// Suppresses unwanted animations while Preferences are removed from and re-added to the
// screen.
screen.setShouldUseGeneratedIds(false);
setPreferenceScreen(screen);
}
@Override
public void onConfigurationChanged(@NonNull Configuration newConfig) {
super.onConfigurationChanged(newConfig);
if (mEditorDialog != null) {
mEditorDialog.onConfigurationChanged();
}
}
@Override
public void onCreateOptionsMenu(Menu menu, MenuInflater inflater) {
menu.clear();
MenuItem help =
menu.add(Menu.NONE, R.id.menu_id_targeted_help, Menu.NONE, R.string.menu_help);
help.setIcon(R.drawable.ic_help_and_feedback);
}
@Override
public boolean onOptionsItemSelected(MenuItem item) {
if (item.getItemId() == R.id.menu_id_targeted_help) {
mHelpAndFeedbackLauncher.show(
getActivity(), getActivity().getString(R.string.help_context_autofill), null);
return true;
}
return super.onOptionsItemSelected(item);
}
@Override
public void onResume() {
super.onResume();
// Always rebuild our list of profiles. Although we could detect if profiles are added or
// deleted (GUID list changes), the profile summary (name+addr) might be different. To be
// safe, we update all.
rebuildProfileList();
if (sObserverForTest != null) sObserverForTest.onEditorDismiss();
}
private void rebuildProfileList() {
getPreferenceScreen().removeAll();
getPreferenceScreen().setOrderingAsAdded(true);
ChromeSwitchPreference autofillSwitch =
new ChromeSwitchPreference(getStyledContext(), null);
autofillSwitch.setTitle(R.string.autofill_enable_profiles_toggle_label);
autofillSwitch.setSummary(R.string.autofill_enable_profiles_toggle_sublabel);
autofillSwitch.setChecked(PersonalDataManager.isAutofillProfileEnabled());
autofillSwitch.setOnPreferenceChangeListener((preference, newValue) -> {
PersonalDataManager.setAutofillProfileEnabled((boolean) newValue);
return true;
});
autofillSwitch.setManagedPreferenceDelegate(new ChromeManagedPreferenceDelegate() {
@Override
public boolean isPreferenceControlledByPolicy(Preference preference) {
return PersonalDataManager.isAutofillProfileManaged();
}
@Override
public boolean isPreferenceClickDisabled(Preference preference) {
return PersonalDataManager.isAutofillProfileManaged()
&& !PersonalDataManager.isAutofillProfileEnabled();
}
});
getPreferenceScreen().addPreference(autofillSwitch);
final boolean addressSyncEnabled = isAddressSyncEnabled();
for (AutofillProfile profile : PersonalDataManager.getInstance().getProfilesForSettings()) {
assert profile.getIsLocal();
// Add a preference for the profile.
Preference pref = new AutofillProfileEditorPreference(getStyledContext());
pref.setTitle(profile.getFullName());
pref.setSummary(profile.getLabel());
pref.setKey(pref.getTitle().toString()); // For testing.
if (!addressSyncEnabled && profile.getSource() != Source.ACCOUNT) {
// Conditionally set local profile icon for address profiles that are neither
// synced, nor saved in the account.
pref.setWidgetLayoutResource(R.layout.autofill_local_profile_icon);
}
Bundle args = pref.getExtras();
args.putString(AutofillEditorBase.AUTOFILL_GUID, profile.getGUID());
try (StrictModeContext ignored = StrictModeContext.allowDiskWrites()) {
getPreferenceScreen().addPreference(pref);
}
}
// Add 'Add address' button. Tap of it brings up address editor which allows users type in
// new addresses.
if (PersonalDataManager.isAutofillProfileEnabled()) {
AutofillProfileEditorPreference pref =
new AutofillProfileEditorPreference(getStyledContext());
Drawable plusIcon = ApiCompatibilityUtils.getDrawable(getResources(), R.drawable.plus);
plusIcon.mutate();
plusIcon.setColorFilter(SemanticColorUtils.getDefaultControlColorActive(getContext()),
PorterDuff.Mode.SRC_IN);
pref.setIcon(plusIcon);
pref.setTitle(R.string.autofill_create_profile);
pref.setKey(PREF_NEW_PROFILE); // For testing.
try (StrictModeContext ignored = StrictModeContext.allowDiskWrites()) {
getPreferenceScreen().addPreference(pref);
}
}
}
@Override
public void onPersonalDataChanged() {
rebuildProfileList();
if (sObserverForTest != null) sObserverForTest.onEditorDismiss();
}
@Override
public void onActivityCreated(Bundle savedInstanceState) {
super.onActivityCreated(savedInstanceState);
PersonalDataManager.getInstance().registerDataObserver(this);
}
@Override
public void onDestroyView() {
PersonalDataManager.getInstance().unregisterDataObserver(this);
super.onDestroyView();
}
@VisibleForTesting
public static void setObserverForTest(EditorObserverForTest observerForTest) {
sObserverForTest = observerForTest;
EditorDialog.setEditorObserverForTest(sObserverForTest);
}
@Override
public void onDisplayPreferenceDialog(Preference preference) {
if (preference instanceof AutofillProfileEditorPreference) {
String guid = ((AutofillProfileEditorPreference) preference).getGUID();
mEditorDialog = prepareEditorDialog(guid);
AutofillAddress autofillAddress = null;
if (guid != null) {
AutofillProfile profile = PersonalDataManager.getInstance().getProfile(guid);
if (profile != null) {
autofillAddress = new AutofillAddress(getActivity(), profile);
}
}
editAddress(mEditorDialog, autofillAddress);
return;
}
super.onDisplayPreferenceDialog(preference);
}
@VisibleForTesting
EditorDialog prepareEditorDialog(String guid) {
Runnable runnable = guid == null ? null : () -> {
PersonalDataManager.getInstance().deleteProfile(guid);
SettingsAutofillAndPaymentsObserver.getInstance().notifyOnAddressDeleted(guid);
if (sObserverForTest != null) {
sObserverForTest.onEditorReadyToEdit();
}
};
return new EditorDialog(
getActivity(), runnable, Profile.getLastUsedRegularProfile(), false);
}
private void editAddress(EditorDialog dialog, AutofillAddress autofillAddress) {
AddressEditor.Delegate delegate = new AddressEditor.Delegate() {
// User has either created a new address, or edited an existing address.
// We should save changes in any case.
@Override
public void onDone(AutofillAddress address) {
PersonalDataManager.getInstance().setProfile(address.getProfile());
SettingsAutofillAndPaymentsObserver.getInstance().notifyOnAddressUpdated(address);
if (sObserverForTest != null) {
sObserverForTest.onEditorReadyToEdit();
}
}
// User canceled edited meaning that |autofillAddress| has stayed intact.
@Override
public void onCancel() {
if (sObserverForTest != null) {
sObserverForTest.onEditorReadyToEdit();
}
}
};
AddressEditor addressEditor = new AddressEditor(dialog, delegate,
/*saveToDisk=*/true, /*isUpdate=*/autofillAddress != null,
/*isMigrationToAccount=*/false);
addressEditor.edit(autofillAddress);
}
private boolean isAddressSyncEnabled() {
SyncService syncService = SyncService.get();
return syncService != null && syncService.isSyncFeatureEnabled()
&& syncService.getSelectedTypes().contains(UserSelectableType.AUTOFILL);
}
private Context getStyledContext() {
return getPreferenceManager().getContext();
}
@VisibleForTesting
EditorDialog getEditorDialogForTest() {
return mEditorDialog;
}
@Override
public void setHelpAndFeedbackLauncher(HelpAndFeedbackLauncher helpAndFeedbackLauncher) {
mHelpAndFeedbackLauncher = helpAndFeedbackLauncher;
}
}