blob: ebf705a7ccea49fd0aeae16960894ff79f3932f5 [file] [log] [blame]
/*
* Copyright (C) 2015 The Android Open Source Project
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License
*/
package com.android.tv.settings.accounts;
import android.accounts.Account;
import android.accounts.AccountManager;
import android.app.Activity;
import android.content.ContentResolver;
import android.content.Context;
import android.content.Intent;
import android.content.SyncAdapterType;
import android.content.SyncInfo;
import android.content.SyncStatusInfo;
import android.content.SyncStatusObserver;
import android.content.pm.PackageManager;
import android.content.pm.ProviderInfo;
import android.os.Bundle;
import android.os.Handler;
import android.os.UserHandle;
import androidx.preference.Preference;
import androidx.preference.PreferenceGroup;
import android.text.TextUtils;
import android.text.format.DateUtils;
import android.util.Log;
import com.android.internal.logging.nano.MetricsProto;
import com.android.settingslib.accounts.AuthenticatorHelper;
import com.android.tv.settings.R;
import com.android.tv.settings.SettingsPreferenceFragment;
import com.google.android.collect.Lists;
import java.util.ArrayList;
import java.util.Collections;
import java.util.List;
/**
* The account sync settings screen in TV Settings.
*/
public class AccountSyncFragment extends SettingsPreferenceFragment implements
AuthenticatorHelper.OnAccountsUpdateListener {
private static final String TAG = "AccountSyncFragment";
private static final String ARG_ACCOUNT = "account";
private static final String KEY_REMOVE_ACCOUNT = "remove_account";
private static final String KEY_SYNC_NOW = "sync_now";
private static final String KEY_SYNC_ADAPTERS = "sync_adapters";
private Object mStatusChangeListenerHandle;
private UserHandle mUserHandle;
private Account mAccount;
private ArrayList<SyncAdapterType> mInvisibleAdapters = Lists.newArrayList();
private PreferenceGroup mSyncCategory;
private final Handler mHandler = new Handler();
private SyncStatusObserver mSyncStatusObserver = new SyncStatusObserver() {
public void onStatusChanged(int which) {
mHandler.post(new Runnable() {
public void run() {
if (isResumed()) {
onSyncStateUpdated();
}
}
});
}
};
private AuthenticatorHelper mAuthenticatorHelper;
public static AccountSyncFragment newInstance(Account account) {
final Bundle b = new Bundle(1);
prepareArgs(b, account);
final AccountSyncFragment f = new AccountSyncFragment();
f.setArguments(b);
return f;
}
public static void prepareArgs(Bundle b, Account account) {
b.putParcelable(ARG_ACCOUNT, account);
}
@Override
public void onCreate(Bundle savedInstanceState) {
mUserHandle = new UserHandle(UserHandle.myUserId());
mAccount = getArguments().getParcelable(ARG_ACCOUNT);
mAuthenticatorHelper = new AuthenticatorHelper(getActivity(), mUserHandle, this);
super.onCreate(savedInstanceState);
if (Log.isLoggable(TAG, Log.VERBOSE)) {
Log.v(TAG, "Got account: " + mAccount);
}
}
@Override
public void onStart() {
super.onStart();
mStatusChangeListenerHandle = ContentResolver.addStatusChangeListener(
ContentResolver.SYNC_OBSERVER_TYPE_ACTIVE
| ContentResolver.SYNC_OBSERVER_TYPE_STATUS
| ContentResolver.SYNC_OBSERVER_TYPE_SETTINGS,
mSyncStatusObserver);
onSyncStateUpdated();
mAuthenticatorHelper.listenToAccountUpdates();
mAuthenticatorHelper.updateAuthDescriptions(getActivity());
}
@Override
public void onResume() {
super.onResume();
mHandler.post(() -> onAccountsUpdate(mUserHandle));
}
@Override
public void onStop() {
super.onStop();
ContentResolver.removeStatusChangeListener(mStatusChangeListenerHandle);
mAuthenticatorHelper.stopListeningToAccountUpdates();
}
@Override
public void onCreatePreferences(Bundle savedInstanceState, String rootKey) {
setPreferencesFromResource(R.xml.account_preference, null);
getPreferenceScreen().setTitle(mAccount.name);
final Preference removeAccountPref = findPreference(KEY_REMOVE_ACCOUNT);
removeAccountPref.setIntent(new Intent(getActivity(), RemoveAccountDialog.class)
.putExtra(AccountSyncActivity.EXTRA_ACCOUNT, mAccount.name));
mSyncCategory = (PreferenceGroup) findPreference(KEY_SYNC_ADAPTERS);
}
@Override
public boolean onPreferenceTreeClick(Preference preference) {
if (preference instanceof SyncStateSwitchPreference) {
SyncStateSwitchPreference syncPref = (SyncStateSwitchPreference) preference;
String authority = syncPref.getAuthority();
Account account = syncPref.getAccount();
final int userId = mUserHandle.getIdentifier();
if (syncPref.isOneTimeSyncMode()) {
requestOrCancelSync(account, authority, true);
} else {
boolean syncOn = syncPref.isChecked();
boolean oldSyncState = ContentResolver.getSyncAutomaticallyAsUser(account,
authority, userId);
if (syncOn != oldSyncState) {
// if we're enabling sync, this will request a sync as well
ContentResolver.setSyncAutomaticallyAsUser(account, authority, syncOn, userId);
// if the master sync switch is off, the request above will
// get dropped. when the user clicks on this toggle,
// we want to force the sync, however.
if (!ContentResolver.getMasterSyncAutomaticallyAsUser(userId) || !syncOn) {
requestOrCancelSync(account, authority, syncOn);
}
}
}
return true;
} else if (TextUtils.equals(preference.getKey(), KEY_SYNC_NOW)) {
boolean syncActive = !ContentResolver.getCurrentSyncsAsUser(
mUserHandle.getIdentifier()).isEmpty();
if (syncActive) {
cancelSyncForEnabledProviders();
} else {
startSyncForEnabledProviders();
}
return true;
} else {
return super.onPreferenceTreeClick(preference);
}
}
private void startSyncForEnabledProviders() {
requestOrCancelSyncForEnabledProviders(true /* start them */);
final Activity activity = getActivity();
if (activity != null) {
activity.invalidateOptionsMenu();
}
}
private void cancelSyncForEnabledProviders() {
requestOrCancelSyncForEnabledProviders(false /* cancel them */);
final Activity activity = getActivity();
if (activity != null) {
activity.invalidateOptionsMenu();
}
}
private void requestOrCancelSyncForEnabledProviders(boolean startSync) {
// sync everything that the user has enabled
int count = mSyncCategory.getPreferenceCount();
for (int i = 0; i < count; i++) {
Preference pref = mSyncCategory.getPreference(i);
if (! (pref instanceof SyncStateSwitchPreference)) {
continue;
}
SyncStateSwitchPreference syncPref = (SyncStateSwitchPreference) pref;
if (!syncPref.isChecked()) {
continue;
}
requestOrCancelSync(syncPref.getAccount(), syncPref.getAuthority(), startSync);
}
// plus whatever the system needs to sync, e.g., invisible sync adapters
if (mAccount != null) {
for (SyncAdapterType syncAdapter : mInvisibleAdapters) {
requestOrCancelSync(mAccount, syncAdapter.authority, startSync);
}
}
}
private void requestOrCancelSync(Account account, String authority, boolean flag) {
if (flag) {
Bundle extras = new Bundle();
extras.putBoolean(ContentResolver.SYNC_EXTRAS_MANUAL, true);
ContentResolver.requestSyncAsUser(account, authority, mUserHandle.getIdentifier(),
extras);
} else {
ContentResolver.cancelSyncAsUser(account, authority, mUserHandle.getIdentifier());
}
}
private boolean isSyncing(List<SyncInfo> currentSyncs, Account account, String authority) {
for (SyncInfo syncInfo : currentSyncs) {
if (syncInfo.account.equals(account) && syncInfo.authority.equals(authority)) {
return true;
}
}
return false;
}
private boolean accountExists(Account account) {
if (account == null) return false;
Account[] accounts = AccountManager.get(getActivity()).getAccountsByTypeAsUser(
account.type, mUserHandle);
for (final Account other : accounts) {
if (other.equals(account)) {
return true;
}
}
return false;
}
@Override
public void onAccountsUpdate(UserHandle userHandle) {
if (!isResumed()) {
return;
}
if (!accountExists(mAccount)) {
// The account was deleted
if (!getFragmentManager().popBackStackImmediate()) {
getActivity().finish();
}
return;
}
updateAccountSwitches();
onSyncStateUpdated();
}
private void onSyncStateUpdated() {
// iterate over all the preferences, setting the state properly for each
final int userId = mUserHandle.getIdentifier();
List<SyncInfo> currentSyncs = ContentResolver.getCurrentSyncsAsUser(userId);
// boolean syncIsFailing = false;
// Refresh the sync status switches - some syncs may have become active.
updateAccountSwitches();
for (int i = 0, count = mSyncCategory.getPreferenceCount(); i < count; i++) {
Preference pref = mSyncCategory.getPreference(i);
if (! (pref instanceof SyncStateSwitchPreference)) {
continue;
}
SyncStateSwitchPreference syncPref = (SyncStateSwitchPreference) pref;
String authority = syncPref.getAuthority();
Account account = syncPref.getAccount();
SyncStatusInfo status = ContentResolver.getSyncStatusAsUser(account, authority, userId);
boolean syncEnabled = ContentResolver.getSyncAutomaticallyAsUser(account, authority,
userId);
boolean authorityIsPending = status != null && status.pending;
boolean initialSync = status != null && status.initialize;
boolean activelySyncing = isSyncing(currentSyncs, account, authority);
boolean lastSyncFailed = status != null
&& status.lastFailureTime != 0
&& status.getLastFailureMesgAsInt(0)
!= ContentResolver.SYNC_ERROR_SYNC_ALREADY_IN_PROGRESS;
if (!syncEnabled) lastSyncFailed = false;
// if (lastSyncFailed && !activelySyncing && !authorityIsPending) {
// syncIsFailing = true;
// }
if (Log.isLoggable(TAG, Log.VERBOSE)) {
Log.v(TAG, "Update sync status: " + account + " " + authority +
" active = " + activelySyncing + " pend =" + authorityIsPending);
}
final long successEndTime = (status == null) ? 0 : status.lastSuccessTime;
if (!syncEnabled) {
syncPref.setSummary(R.string.sync_disabled);
} else if (activelySyncing) {
syncPref.setSummary(R.string.sync_in_progress);
} else if (successEndTime != 0) {
final String timeString = DateUtils.formatDateTime(getActivity(), successEndTime,
DateUtils.FORMAT_SHOW_DATE | DateUtils.FORMAT_SHOW_TIME);
syncPref.setSummary(getResources().getString(R.string.last_synced, timeString));
} else {
syncPref.setSummary("");
}
int syncState = ContentResolver.getIsSyncableAsUser(account, authority, userId);
syncPref.setActive(activelySyncing && (syncState >= 0) &&
!initialSync);
syncPref.setPending(authorityIsPending && (syncState >= 0) &&
!initialSync);
syncPref.setFailed(lastSyncFailed);
final boolean oneTimeSyncMode = !ContentResolver.getMasterSyncAutomaticallyAsUser(
userId);
syncPref.setOneTimeSyncMode(oneTimeSyncMode);
syncPref.setChecked(oneTimeSyncMode || syncEnabled);
}
}
private void updateAccountSwitches() {
mInvisibleAdapters.clear();
SyncAdapterType[] syncAdapters = ContentResolver.getSyncAdapterTypesAsUser(
mUserHandle.getIdentifier());
ArrayList<String> authorities = new ArrayList<>(syncAdapters.length);
for (SyncAdapterType sa : syncAdapters) {
// Only keep track of sync adapters for this account
if (!sa.accountType.equals(mAccount.type)) continue;
if (sa.isUserVisible()) {
if (Log.isLoggable(TAG, Log.VERBOSE)) {
Log.v(TAG, "updateAccountSwitches: added authority " + sa.authority
+ " to accountType " + sa.accountType);
}
authorities.add(sa.authority);
} else {
// keep track of invisible sync adapters, so sync now forces
// them to sync as well.
mInvisibleAdapters.add(sa);
}
}
mSyncCategory.removeAll();
final List<Preference> switches = new ArrayList<>(authorities.size());
if (Log.isLoggable(TAG, Log.VERBOSE)) {
Log.v(TAG, "looking for sync adapters that match account " + mAccount);
}
for (final String authority : authorities) {
// We could check services here....
int syncState = ContentResolver.getIsSyncableAsUser(mAccount, authority,
mUserHandle.getIdentifier());
if (Log.isLoggable(TAG, Log.VERBOSE)) {
Log.v(TAG, " found authority " + authority + " " + syncState);
}
if (syncState > 0) {
final Preference pref = createSyncStateSwitch(mAccount, authority);
switches.add(pref);
}
}
Collections.sort(switches);
for (final Preference pref : switches) {
mSyncCategory.addPreference(pref);
}
}
private Preference createSyncStateSwitch(Account account, String authority) {
final Context themedContext = getPreferenceManager().getContext();
SyncStateSwitchPreference preference =
new SyncStateSwitchPreference(themedContext, account, authority);
preference.setPersistent(false);
final PackageManager packageManager = getActivity().getPackageManager();
final ProviderInfo providerInfo = packageManager.resolveContentProviderAsUser(
authority, 0, mUserHandle.getIdentifier());
if (providerInfo == null) {
return null;
}
CharSequence providerLabel = providerInfo.loadLabel(packageManager);
if (TextUtils.isEmpty(providerLabel)) {
Log.e(TAG, "Provider needs a label for authority '" + authority + "'");
return null;
}
String title = getString(R.string.sync_item_title, providerLabel);
preference.setTitle(title);
preference.setKey(authority);
return preference;
}
@Override
public int getMetricsCategory() {
return MetricsProto.MetricsEvent.ACCOUNTS_ACCOUNT_SYNC;
}
}