| /* |
| * Copyright (C) 2008 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.settings.accounts; |
| |
| import com.android.settings.R; |
| import com.google.android.collect.Lists; |
| import com.google.android.collect.Maps; |
| |
| import android.accounts.Account; |
| import android.accounts.AccountManager; |
| import android.accounts.AccountManagerCallback; |
| import android.accounts.AccountManagerFuture; |
| import android.accounts.AuthenticatorException; |
| import android.accounts.OperationCanceledException; |
| import android.app.Activity; |
| import android.app.AlertDialog; |
| import android.app.Dialog; |
| import android.content.ContentResolver; |
| import android.content.Context; |
| import android.content.DialogInterface; |
| import android.content.Intent; |
| import android.content.SyncAdapterType; |
| import android.content.SyncInfo; |
| import android.content.SyncStatusInfo; |
| import android.content.pm.ProviderInfo; |
| import android.net.ConnectivityManager; |
| import android.os.Bundle; |
| import android.preference.Preference; |
| import android.preference.PreferenceScreen; |
| import android.text.TextUtils; |
| import android.text.format.DateFormat; |
| import android.util.Log; |
| import android.view.LayoutInflater; |
| import android.view.Menu; |
| import android.view.MenuInflater; |
| import android.view.MenuItem; |
| import android.view.View; |
| import android.view.ViewGroup; |
| import android.widget.ImageView; |
| import android.widget.TextView; |
| |
| import java.io.IOException; |
| import java.util.ArrayList; |
| import java.util.Date; |
| import java.util.HashMap; |
| import java.util.List; |
| |
| public class AccountSyncSettings extends AccountPreferenceBase { |
| |
| public static final String ACCOUNT_KEY = "account"; |
| protected static final int MENU_REMOVE_ACCOUNT_ID = Menu.FIRST; |
| private static final int MENU_SYNC_NOW_ID = Menu.FIRST + 1; |
| private static final int MENU_SYNC_CANCEL_ID = Menu.FIRST + 2; |
| private static final int REALLY_REMOVE_DIALOG = 100; |
| private static final int FAILED_REMOVAL_DIALOG = 101; |
| private static final int CANT_DO_ONETIME_SYNC_DIALOG = 102; |
| private TextView mUserId; |
| private TextView mProviderId; |
| private ImageView mProviderIcon; |
| private TextView mErrorInfoView; |
| private java.text.DateFormat mDateFormat; |
| private java.text.DateFormat mTimeFormat; |
| private Account mAccount; |
| // List of all accounts, updated when accounts are added/removed |
| // We need to re-scan the accounts on sync events, in case sync state changes. |
| private Account[] mAccounts; |
| private ArrayList<SyncStateCheckBoxPreference> mCheckBoxes = |
| new ArrayList<SyncStateCheckBoxPreference>(); |
| private ArrayList<String> mInvisibleAdapters = Lists.newArrayList(); |
| |
| @Override |
| public Dialog onCreateDialog(final int id) { |
| Dialog dialog = null; |
| if (id == REALLY_REMOVE_DIALOG) { |
| dialog = new AlertDialog.Builder(getActivity()) |
| .setTitle(R.string.really_remove_account_title) |
| .setMessage(R.string.really_remove_account_message) |
| .setNegativeButton(android.R.string.cancel, null) |
| .setPositiveButton(R.string.remove_account_label, |
| new DialogInterface.OnClickListener() { |
| public void onClick(DialogInterface dialog, int which) { |
| AccountManager.get(AccountSyncSettings.this.getActivity()) |
| .removeAccount(mAccount, |
| new AccountManagerCallback<Boolean>() { |
| public void run(AccountManagerFuture<Boolean> future) { |
| boolean failed = true; |
| try { |
| if (future.getResult() == true) { |
| failed = false; |
| } |
| } catch (OperationCanceledException e) { |
| // handled below |
| } catch (IOException e) { |
| // handled below |
| } catch (AuthenticatorException e) { |
| // handled below |
| } |
| if (failed) { |
| showDialog(FAILED_REMOVAL_DIALOG); |
| } else { |
| finish(); |
| } |
| } |
| }, null); |
| } |
| }) |
| .create(); |
| } else if (id == FAILED_REMOVAL_DIALOG) { |
| dialog = new AlertDialog.Builder(getActivity()) |
| .setTitle(R.string.really_remove_account_title) |
| .setPositiveButton(android.R.string.ok, null) |
| .setMessage(R.string.remove_account_failed) |
| .create(); |
| } else if (id == CANT_DO_ONETIME_SYNC_DIALOG) { |
| dialog = new AlertDialog.Builder(getActivity()) |
| .setTitle(R.string.cant_sync_dialog_title) |
| .setMessage(R.string.cant_sync_dialog_message) |
| .setPositiveButton(android.R.string.ok, null) |
| .create(); |
| } |
| return dialog; |
| } |
| |
| @Override |
| public void onCreate(Bundle icicle) { |
| super.onCreate(icicle); |
| |
| setHasOptionsMenu(true); |
| } |
| |
| @Override |
| public View onCreateView(LayoutInflater inflater, ViewGroup container, |
| Bundle savedInstanceState) { |
| final View view = inflater.inflate(R.layout.account_sync_screen, container, false); |
| |
| initializeUi(view); |
| |
| return view; |
| } |
| |
| protected void initializeUi(final View rootView) { |
| addPreferencesFromResource(R.xml.account_sync_settings); |
| |
| mErrorInfoView = (TextView) rootView.findViewById(R.id.sync_settings_error_info); |
| mErrorInfoView.setVisibility(View.GONE); |
| mErrorInfoView.setCompoundDrawablesWithIntrinsicBounds( |
| getResources().getDrawable(R.drawable.ic_list_syncerror), null, null, null); |
| |
| mUserId = (TextView) rootView.findViewById(R.id.user_id); |
| mProviderId = (TextView) rootView.findViewById(R.id.provider_id); |
| mProviderIcon = (ImageView) rootView.findViewById(R.id.provider_icon); |
| } |
| |
| @Override |
| public void onActivityCreated(Bundle savedInstanceState) { |
| super.onActivityCreated(savedInstanceState); |
| |
| final Activity activity = getActivity(); |
| |
| mDateFormat = DateFormat.getDateFormat(activity); |
| mTimeFormat = DateFormat.getTimeFormat(activity); |
| |
| mAccount = (Account) getArguments().getParcelable(ACCOUNT_KEY); |
| if (mAccount != null) { |
| if (Log.isLoggable(TAG, Log.VERBOSE)) Log.v(TAG, "Got account: " + mAccount); |
| mUserId.setText(mAccount.name); |
| mProviderId.setText(mAccount.type); |
| } |
| } |
| |
| @Override |
| public void onResume() { |
| final Activity activity = getActivity(); |
| AccountManager.get(activity).addOnAccountsUpdatedListener(this, null, false); |
| updateAuthDescriptions(); |
| onAccountsUpdated(AccountManager.get(activity).getAccounts()); |
| |
| super.onResume(); |
| } |
| |
| @Override |
| public void onPause() { |
| super.onPause(); |
| AccountManager.get(getActivity()).removeOnAccountsUpdatedListener(this); |
| } |
| |
| private void addSyncStateCheckBox(Account account, String authority) { |
| SyncStateCheckBoxPreference item = |
| new SyncStateCheckBoxPreference(getActivity(), account, authority); |
| item.setPersistent(false); |
| final ProviderInfo providerInfo = getPackageManager().resolveContentProvider(authority, 0); |
| CharSequence providerLabel = providerInfo != null |
| ? providerInfo.loadLabel(getPackageManager()) : null; |
| if (TextUtils.isEmpty(providerLabel)) { |
| Log.e(TAG, "Provider needs a label for authority '" + authority + "'"); |
| providerLabel = authority; |
| } |
| String title = getString(R.string.sync_item_title, providerLabel); |
| item.setTitle(title); |
| item.setKey(authority); |
| getPreferenceScreen().addPreference(item); |
| mCheckBoxes.add(item); |
| } |
| |
| @Override |
| public void onCreateOptionsMenu(Menu menu, MenuInflater inflater) { |
| super.onCreateOptionsMenu(menu, inflater); |
| |
| MenuItem removeAccount = menu.add(0, MENU_REMOVE_ACCOUNT_ID, 0, |
| getString(R.string.remove_account_label)) |
| .setIcon(com.android.internal.R.drawable.ic_menu_delete); |
| MenuItem syncNow = menu.add(0, MENU_SYNC_NOW_ID, 0, |
| getString(R.string.sync_menu_sync_now)) |
| .setIcon(com.android.internal.R.drawable.ic_menu_refresh); |
| MenuItem syncCancel = menu.add(0, MENU_SYNC_CANCEL_ID, 0, |
| getString(R.string.sync_menu_sync_cancel)) |
| .setIcon(com.android.internal.R.drawable.ic_menu_close_clear_cancel); |
| |
| removeAccount.setShowAsAction(MenuItem.SHOW_AS_ACTION_ALWAYS |
| | MenuItem.SHOW_AS_ACTION_WITH_TEXT); |
| syncNow.setShowAsAction(MenuItem.SHOW_AS_ACTION_ALWAYS); |
| syncCancel.setShowAsAction(MenuItem.SHOW_AS_ACTION_ALWAYS); |
| } |
| |
| @Override |
| public void onPrepareOptionsMenu(Menu menu) { |
| super.onPrepareOptionsMenu(menu); |
| boolean syncActive = ContentResolver.getCurrentSync() != null; |
| menu.findItem(MENU_SYNC_NOW_ID).setVisible(!syncActive); |
| menu.findItem(MENU_SYNC_CANCEL_ID).setVisible(syncActive); |
| } |
| |
| @Override |
| public boolean onOptionsItemSelected(MenuItem item) { |
| switch (item.getItemId()) { |
| case MENU_SYNC_NOW_ID: |
| startSyncForEnabledProviders(); |
| return true; |
| case MENU_SYNC_CANCEL_ID: |
| cancelSyncForEnabledProviders(); |
| return true; |
| case MENU_REMOVE_ACCOUNT_ID: |
| showDialog(REALLY_REMOVE_DIALOG); |
| return true; |
| } |
| return super.onOptionsItemSelected(item); |
| } |
| |
| @Override |
| public boolean onPreferenceTreeClick(PreferenceScreen preferences, Preference preference) { |
| if (preference instanceof SyncStateCheckBoxPreference) { |
| SyncStateCheckBoxPreference syncPref = (SyncStateCheckBoxPreference) preference; |
| String authority = syncPref.getAuthority(); |
| Account account = syncPref.getAccount(); |
| boolean syncAutomatically = ContentResolver.getSyncAutomatically(account, authority); |
| if (syncPref.isOneTimeSyncMode()) { |
| requestOrCancelSync(account, authority, true); |
| } else { |
| boolean syncOn = syncPref.isChecked(); |
| boolean oldSyncState = syncAutomatically; |
| if (syncOn != oldSyncState) { |
| // if we're enabling sync, this will request a sync as well |
| ContentResolver.setSyncAutomatically(account, authority, syncOn); |
| // 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.getMasterSyncAutomatically() || !syncOn) { |
| requestOrCancelSync(account, authority, syncOn); |
| } |
| } |
| } |
| return true; |
| } else { |
| return super.onPreferenceTreeClick(preferences, preference); |
| } |
| } |
| |
| private void startSyncForEnabledProviders() { |
| requestOrCancelSyncForEnabledProviders(true /* start them */); |
| getActivity().invalidateOptionsMenu(); |
| } |
| |
| private void cancelSyncForEnabledProviders() { |
| requestOrCancelSyncForEnabledProviders(false /* cancel them */); |
| getActivity().invalidateOptionsMenu(); |
| } |
| |
| private void requestOrCancelSyncForEnabledProviders(boolean startSync) { |
| // sync everything that the user has enabled |
| int count = getPreferenceScreen().getPreferenceCount(); |
| for (int i = 0; i < count; i++) { |
| Preference pref = getPreferenceScreen().getPreference(i); |
| if (! (pref instanceof SyncStateCheckBoxPreference)) { |
| continue; |
| } |
| SyncStateCheckBoxPreference syncPref = (SyncStateCheckBoxPreference) 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 (String authority : mInvisibleAdapters) { |
| requestOrCancelSync(mAccount, 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.requestSync(account, authority, extras); |
| } else { |
| ContentResolver.cancelSync(account, authority); |
| } |
| } |
| |
| 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; |
| } |
| |
| @Override |
| protected void onSyncStateUpdated() { |
| if (!isResumed()) return; |
| setFeedsState(); |
| } |
| |
| private void setFeedsState() { |
| // iterate over all the preferences, setting the state properly for each |
| Date date = new Date(); |
| List<SyncInfo> currentSyncs = ContentResolver.getCurrentSyncs(); |
| boolean syncIsFailing = false; |
| |
| // Refresh the sync status checkboxes - some syncs may have become active. |
| updateAccountCheckboxes(mAccounts); |
| |
| for (int i = 0, count = getPreferenceScreen().getPreferenceCount(); i < count; i++) { |
| Preference pref = getPreferenceScreen().getPreference(i); |
| if (! (pref instanceof SyncStateCheckBoxPreference)) { |
| continue; |
| } |
| SyncStateCheckBoxPreference syncPref = (SyncStateCheckBoxPreference) pref; |
| |
| String authority = syncPref.getAuthority(); |
| Account account = syncPref.getAccount(); |
| |
| SyncStatusInfo status = ContentResolver.getSyncStatus(account, authority); |
| boolean syncEnabled = ContentResolver.getSyncAutomatically(account, authority); |
| boolean authorityIsPending = status == null ? false : status.pending; |
| boolean initialSync = status == null ? false : 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.d(TAG, "Update sync status: " + account + " " + authority + |
| " active = " + activelySyncing + " pend =" + authorityIsPending); |
| } |
| |
| final long successEndTime = (status == null) ? 0 : status.lastSuccessTime; |
| if (successEndTime != 0) { |
| date.setTime(successEndTime); |
| final String timeString = mDateFormat.format(date) + " " |
| + mTimeFormat.format(date); |
| syncPref.setSummary(timeString); |
| } else { |
| syncPref.setSummary(""); |
| } |
| int syncState = ContentResolver.getIsSyncable(account, authority); |
| |
| syncPref.setActive(activelySyncing && (syncState >= 0) && |
| !initialSync); |
| syncPref.setPending(authorityIsPending && (syncState >= 0) && |
| !initialSync); |
| |
| syncPref.setFailed(lastSyncFailed); |
| ConnectivityManager connManager = |
| (ConnectivityManager) getSystemService(Context.CONNECTIVITY_SERVICE); |
| final boolean masterSyncAutomatically = ContentResolver.getMasterSyncAutomatically(); |
| final boolean backgroundDataEnabled = connManager.getBackgroundDataSetting(); |
| final boolean oneTimeSyncMode = !masterSyncAutomatically || !backgroundDataEnabled; |
| syncPref.setOneTimeSyncMode(oneTimeSyncMode); |
| syncPref.setChecked(oneTimeSyncMode || syncEnabled); |
| } |
| mErrorInfoView.setVisibility(syncIsFailing ? View.VISIBLE : View.GONE); |
| getActivity().invalidateOptionsMenu(); |
| } |
| |
| @Override |
| public void onAccountsUpdated(Account[] accounts) { |
| super.onAccountsUpdated(accounts); |
| mAccounts = accounts; |
| updateAccountCheckboxes(accounts); |
| onSyncStateUpdated(); |
| } |
| |
| private void updateAccountCheckboxes(Account[] accounts) { |
| mInvisibleAdapters.clear(); |
| |
| SyncAdapterType[] syncAdapters = ContentResolver.getSyncAdapterTypes(); |
| HashMap<String, ArrayList<String>> accountTypeToAuthorities = |
| Maps.newHashMap(); |
| for (int i = 0, n = syncAdapters.length; i < n; i++) { |
| final SyncAdapterType sa = syncAdapters[i]; |
| if (sa.isUserVisible()) { |
| ArrayList<String> authorities = accountTypeToAuthorities.get(sa.accountType); |
| if (authorities == null) { |
| authorities = new ArrayList<String>(); |
| accountTypeToAuthorities.put(sa.accountType, authorities); |
| } |
| if (Log.isLoggable(TAG, Log.VERBOSE)) { |
| Log.d(TAG, "onAccountUpdated: 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.authority); |
| } |
| } |
| |
| for (int i = 0, n = mCheckBoxes.size(); i < n; i++) { |
| getPreferenceScreen().removePreference(mCheckBoxes.get(i)); |
| } |
| mCheckBoxes.clear(); |
| |
| for (int i = 0, n = accounts.length; i < n; i++) { |
| final Account account = accounts[i]; |
| if (Log.isLoggable(TAG, Log.VERBOSE)) { |
| Log.d(TAG, "looking for sync adapters that match account " + account); |
| } |
| final ArrayList<String> authorities = accountTypeToAuthorities.get(account.type); |
| if (authorities != null && (mAccount == null || mAccount.equals(account))) { |
| for (int j = 0, m = authorities.size(); j < m; j++) { |
| final String authority = authorities.get(j); |
| // We could check services here.... |
| int syncState = ContentResolver.getIsSyncable(account, authority); |
| if (Log.isLoggable(TAG, Log.VERBOSE)) { |
| Log.d(TAG, " found authority " + authority + " " + syncState); |
| } |
| if (syncState > 0) { |
| addSyncStateCheckBox(account, authority); |
| } |
| } |
| } |
| } |
| } |
| |
| /** |
| * Updates the titlebar with an icon for the provider type. |
| */ |
| @Override |
| protected void onAuthDescriptionsUpdated() { |
| super.onAuthDescriptionsUpdated(); |
| getPreferenceScreen().removeAll(); |
| mProviderIcon.setImageDrawable(getDrawableForType(mAccount.type)); |
| mProviderId.setText(getLabelForType(mAccount.type)); |
| PreferenceScreen prefs = addPreferencesForType(mAccount.type); |
| if (prefs != null) { |
| updatePreferenceIntents(prefs); |
| } |
| addPreferencesFromResource(R.xml.account_sync_settings); |
| } |
| |
| private void updatePreferenceIntents(PreferenceScreen prefs) { |
| for (int i = 0; i < prefs.getPreferenceCount(); i++) { |
| Intent intent = prefs.getPreference(i).getIntent(); |
| if (intent != null) { |
| intent.putExtra(ACCOUNT_KEY, mAccount); |
| // This is somewhat of a hack. Since the preference screen we're accessing comes |
| // from another package, we need to modify the intent to launch it with |
| // FLAG_ACTIVITY_NEW_TASK. |
| // TODO: Do something smarter if we ever have PreferenceScreens of our own. |
| intent.setFlags(intent.getFlags() | Intent.FLAG_ACTIVITY_NEW_TASK); |
| } |
| } |
| } |
| } |