1 /* 2 * Copyright (C) 2009 The Android Open Source Project 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); 5 * you may not use this file except in compliance with the License. 6 * You may obtain a copy of the License at 7 * 8 * http://www.apache.org/licenses/LICENSE-2.0 9 * 10 * Unless required by applicable law or agreed to in writing, software 11 * distributed under the License is distributed on an "AS IS" BASIS, 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 * See the License for the specific language governing permissions and 14 * limitations under the License. 15 */ 16 17 package com.android.contacts.model; 18 19 import android.accounts.Account; 20 import android.accounts.AccountManager; 21 import android.accounts.OnAccountsUpdateListener; 22 import android.content.BroadcastReceiver; 23 import android.content.ContentResolver; 24 import android.content.Context; 25 import android.content.Intent; 26 import android.content.IntentFilter; 27 import android.content.SharedPreferences; 28 import android.content.SyncStatusObserver; 29 import android.content.pm.PackageManager; 30 import android.os.Handler; 31 import android.os.Looper; 32 import android.provider.ContactsContract; 33 import android.text.TextUtils; 34 import android.util.Log; 35 36 import androidx.core.content.ContextCompat; 37 import androidx.localbroadcastmanager.content.LocalBroadcastManager; 38 39 import com.android.contacts.R; 40 import com.android.contacts.list.ContactListFilterController; 41 import com.android.contacts.model.account.AccountInfo; 42 import com.android.contacts.model.account.AccountType; 43 import com.android.contacts.model.account.AccountTypeProvider; 44 import com.android.contacts.model.account.AccountTypeWithDataSet; 45 import com.android.contacts.model.account.AccountWithDataSet; 46 import com.android.contacts.model.account.FallbackAccountType; 47 import com.android.contacts.model.account.GoogleAccountType; 48 import com.android.contacts.model.account.SimAccountType; 49 import com.android.contacts.model.dataitem.DataKind; 50 import com.android.contacts.util.concurrent.ContactsExecutors; 51 52 import com.google.common.base.Function; 53 import com.google.common.base.Objects; 54 import com.google.common.base.Preconditions; 55 import com.google.common.base.Predicate; 56 import com.google.common.collect.Collections2; 57 import com.google.common.util.concurrent.FutureCallback; 58 import com.google.common.util.concurrent.Futures; 59 import com.google.common.util.concurrent.ListenableFuture; 60 import com.google.common.util.concurrent.ListeningExecutorService; 61 import com.google.common.util.concurrent.MoreExecutors; 62 63 import java.util.ArrayList; 64 import java.util.Collections; 65 import java.util.List; 66 import java.util.concurrent.Callable; 67 import java.util.concurrent.Executor; 68 69 import javax.annotation.Nullable; 70 71 /** 72 * Singleton holder for all parsed {@link AccountType} available on the 73 * system, typically filled through {@link PackageManager} queries. 74 */ 75 public abstract class AccountTypeManager { 76 static final String TAG = "AccountTypeManager"; 77 78 private static final Object mInitializationLock = new Object(); 79 private static AccountTypeManager mAccountTypeManager; 80 81 public static final String BROADCAST_ACCOUNTS_CHANGED = AccountTypeManager.class.getName() + 82 ".AccountsChanged"; 83 84 public enum AccountFilter implements Predicate<AccountInfo> { 85 ALL { 86 @Override apply(@ullable AccountInfo input)87 public boolean apply(@Nullable AccountInfo input) { 88 return input != null; 89 } 90 }, 91 CONTACTS_WRITABLE { 92 @Override apply(@ullable AccountInfo input)93 public boolean apply(@Nullable AccountInfo input) { 94 return input != null && input.getType().areContactsWritable(); 95 } 96 }, 97 DRAWER_DISPLAYABLE { 98 @Override apply(@ullable AccountInfo input)99 public boolean apply(@Nullable AccountInfo input) { 100 return input != null && ((input.getType() instanceof SimAccountType) 101 || input.getType().areContactsWritable()); 102 } 103 }, 104 GROUPS_WRITABLE { 105 @Override apply(@ullable AccountInfo input)106 public boolean apply(@Nullable AccountInfo input) { 107 return input != null && input.getType().isGroupMembershipEditable(); 108 } 109 }; 110 } 111 112 /** 113 * Requests the singleton instance of {@link AccountTypeManager} with data bound from 114 * the available authenticators. This method can safely be called from the UI thread. 115 */ getInstance(Context context)116 public static AccountTypeManager getInstance(Context context) { 117 if (!hasRequiredPermissions(context)) { 118 // Hopefully any component that depends on the values returned by this class 119 // will be restarted if the permissions change. 120 return EMPTY; 121 } 122 synchronized (mInitializationLock) { 123 if (mAccountTypeManager == null) { 124 context = context.getApplicationContext(); 125 mAccountTypeManager = new AccountTypeManagerImpl(context); 126 } 127 } 128 return mAccountTypeManager; 129 } 130 131 /** 132 * Set the instance of account type manager. This is only for and should only be used by unit 133 * tests. While having this method is not ideal, it's simpler than the alternative of 134 * holding this as a service in the ContactsApplication context class. 135 * 136 * @param mockManager The mock AccountTypeManager. 137 */ setInstanceForTest(AccountTypeManager mockManager)138 public static void setInstanceForTest(AccountTypeManager mockManager) { 139 synchronized (mInitializationLock) { 140 mAccountTypeManager = mockManager; 141 } 142 } 143 144 private static final AccountTypeManager EMPTY = new AccountTypeManager() { 145 146 @Override 147 public ListenableFuture<List<AccountInfo>> getAccountsAsync() { 148 return Futures.immediateFuture(Collections.<AccountInfo>emptyList()); 149 } 150 151 @Override 152 public ListenableFuture<List<AccountInfo>> filterAccountsAsync( 153 Predicate<AccountInfo> filter) { 154 return Futures.immediateFuture(Collections.<AccountInfo>emptyList()); 155 } 156 157 @Override 158 public AccountInfo getAccountInfoForAccount(AccountWithDataSet account) { 159 return null; 160 } 161 162 @Override 163 public Account getDefaultGoogleAccount() { 164 return null; 165 } 166 167 @Override 168 public AccountType getAccountType(AccountTypeWithDataSet accountTypeWithDataSet) { 169 return null; 170 } 171 }; 172 173 /** 174 * Returns the list of all accounts (if contactWritableOnly is false) or just the list of 175 * contact writable accounts (if contactWritableOnly is true). 176 * 177 * <p>TODO(mhagerott) delete this method. It's left in place to prevent build breakages when 178 * this change is automerged. Usages of this method in downstream branches should be 179 * replaced with an asynchronous account loading pattern</p> 180 */ getAccounts(boolean contactWritableOnly)181 public List<AccountWithDataSet> getAccounts(boolean contactWritableOnly) { 182 return contactWritableOnly 183 ? blockForWritableAccounts() 184 : AccountInfo.extractAccounts(Futures.getUnchecked(getAccountsAsync())); 185 } 186 187 /** 188 * Returns all contact writable accounts 189 * 190 * <p>In general this method should be avoided. It exists to support some legacy usages of 191 * accounts in infrequently used features where refactoring to asynchronous loading is 192 * not justified. The chance that this will actually block is pretty low if the app has been 193 * launched previously</p> 194 */ blockForWritableAccounts()195 public List<AccountWithDataSet> blockForWritableAccounts() { 196 return AccountInfo.extractAccounts( 197 Futures.getUnchecked(filterAccountsAsync(AccountFilter.CONTACTS_WRITABLE))); 198 } 199 200 /** 201 * Loads accounts in background and returns future that will complete with list of all accounts 202 */ getAccountsAsync()203 public abstract ListenableFuture<List<AccountInfo>> getAccountsAsync(); 204 205 /** 206 * Loads accounts and applies the fitler returning only for which the predicate is true 207 */ filterAccountsAsync( Predicate<AccountInfo> filter)208 public abstract ListenableFuture<List<AccountInfo>> filterAccountsAsync( 209 Predicate<AccountInfo> filter); 210 getAccountInfoForAccount(AccountWithDataSet account)211 public abstract AccountInfo getAccountInfoForAccount(AccountWithDataSet account); 212 213 /** 214 * Returns the default google account. 215 */ getDefaultGoogleAccount()216 public abstract Account getDefaultGoogleAccount(); 217 218 /** 219 * Returns the Google Accounts. 220 * 221 * <p>This method exists in addition to filterAccountsByTypeAsync because it should be safe 222 * to call synchronously. 223 * </p> 224 */ getWritableGoogleAccounts()225 public List<AccountInfo> getWritableGoogleAccounts() { 226 // This implementation may block and should be overridden by the Impl class 227 return Futures.getUnchecked(filterAccountsAsync(new Predicate<AccountInfo>() { 228 @Override 229 public boolean apply(@Nullable AccountInfo input) { 230 return input.getType().areContactsWritable() && 231 GoogleAccountType.ACCOUNT_TYPE.equals(input.getType().accountType); 232 } 233 })); 234 } 235 236 /** 237 * Returns true if there are real accounts (not "local" account) in the list of accounts. 238 */ 239 public boolean hasNonLocalAccount() { 240 final List<AccountWithDataSet> allAccounts = 241 AccountInfo.extractAccounts(Futures.getUnchecked(getAccountsAsync())); 242 if (allAccounts == null || allAccounts.size() == 0) { 243 return false; 244 } 245 if (allAccounts.size() > 1) { 246 return true; 247 } 248 return !allAccounts.get(0).isNullAccount(); 249 } 250 251 static Account getDefaultGoogleAccount(AccountManager accountManager, 252 SharedPreferences prefs, String defaultAccountKey) { 253 // Get all the google accounts on the device 254 final Account[] accounts = accountManager.getAccountsByType( 255 GoogleAccountType.ACCOUNT_TYPE); 256 if (accounts == null || accounts.length == 0) { 257 return null; 258 } 259 260 // Get the default account from preferences 261 final String defaultAccount = prefs.getString(defaultAccountKey, null); 262 final AccountWithDataSet accountWithDataSet = defaultAccount == null ? null : 263 AccountWithDataSet.unstringify(defaultAccount); 264 265 // Look for an account matching the one from preferences 266 if (accountWithDataSet != null) { 267 for (int i = 0; i < accounts.length; i++) { 268 if (TextUtils.equals(accountWithDataSet.name, accounts[i].name) 269 && TextUtils.equals(accountWithDataSet.type, accounts[i].type)) { 270 return accounts[i]; 271 } 272 } 273 } 274 275 // Just return the first one 276 return accounts[0]; 277 } 278 279 public abstract AccountType getAccountType(AccountTypeWithDataSet accountTypeWithDataSet); 280 281 public final AccountType getAccountType(String accountType, String dataSet) { 282 return getAccountType(AccountTypeWithDataSet.get(accountType, dataSet)); 283 } 284 285 public final AccountType getAccountTypeForAccount(AccountWithDataSet account) { 286 if (account != null) { 287 return getAccountType(account.getAccountTypeWithDataSet()); 288 } 289 return getAccountType(null, null); 290 } 291 292 /** 293 * Find the best {@link DataKind} matching the requested 294 * {@link AccountType#accountType}, {@link AccountType#dataSet}, and {@link DataKind#mimeType}. 295 * If no direct match found, we try searching {@link FallbackAccountType}. 296 */ 297 public DataKind getKindOrFallback(AccountType type, String mimeType) { 298 return type == null ? null : type.getKindForMimetype(mimeType); 299 } 300 301 /** 302 * Returns whether the specified account still exists 303 */ 304 public boolean exists(AccountWithDataSet account) { 305 final List<AccountWithDataSet> accounts = 306 AccountInfo.extractAccounts(Futures.getUnchecked(getAccountsAsync())); 307 return accounts.contains(account); 308 } 309 310 /** 311 * Returns whether the specified account is writable 312 * 313 * <p>This checks that the account still exists and that 314 * {@link AccountType#areContactsWritable()} is true</p> 315 */ 316 public boolean isWritable(AccountWithDataSet account) { 317 return exists(account) && getAccountInfoForAccount(account).getType().areContactsWritable(); 318 } 319 320 public boolean hasGoogleAccount() { 321 return getDefaultGoogleAccount() != null; 322 } 323 324 private static boolean hasRequiredPermissions(Context context) { 325 final boolean canGetAccounts = ContextCompat.checkSelfPermission(context, 326 android.Manifest.permission.GET_ACCOUNTS) == PackageManager.PERMISSION_GRANTED; 327 final boolean canReadContacts = ContextCompat.checkSelfPermission(context, 328 android.Manifest.permission.READ_CONTACTS) == PackageManager.PERMISSION_GRANTED; 329 return canGetAccounts && canReadContacts; 330 } 331 332 public static Predicate<AccountInfo> writableFilter() { 333 return AccountFilter.CONTACTS_WRITABLE; 334 } 335 336 public static Predicate<AccountInfo> drawerDisplayableFilter() { 337 return AccountFilter.DRAWER_DISPLAYABLE; 338 } 339 340 public static Predicate<AccountInfo> groupWritableFilter() { 341 return AccountFilter.GROUPS_WRITABLE; 342 } 343 } 344 345 class AccountTypeManagerImpl extends AccountTypeManager 346 implements OnAccountsUpdateListener, SyncStatusObserver { 347 348 private final Context mContext; 349 private final AccountManager mAccountManager; 350 private final DeviceLocalAccountLocator mLocalAccountLocator; 351 private final Executor mMainThreadExecutor; 352 private final ListeningExecutorService mExecutor; 353 private AccountTypeProvider mTypeProvider; 354 355 private final AccountType mFallbackAccountType; 356 357 private ListenableFuture<List<AccountWithDataSet>> mLocalAccountsFuture; 358 private ListenableFuture<List<AccountWithDataSet>> mSimAccountsFuture; 359 private ListenableFuture<AccountTypeProvider> mAccountTypesFuture; 360 361 private List<AccountWithDataSet> mLocalAccounts = new ArrayList<>(); 362 private List<AccountWithDataSet> mSimAccounts = new ArrayList<>(); 363 private List<AccountWithDataSet> mAccountManagerAccounts = new ArrayList<>(); 364 365 private final Handler mMainThreadHandler = new Handler(Looper.getMainLooper()); 366 367 private final Function<AccountTypeProvider, List<AccountWithDataSet>> mAccountsExtractor = 368 new Function<AccountTypeProvider, List<AccountWithDataSet>>() { 369 @Nullable 370 @Override 371 public List<AccountWithDataSet> apply(@Nullable AccountTypeProvider typeProvider) { 372 return getAccountsWithDataSets(mAccountManager.getAccounts(), typeProvider); 373 } 374 }; 375 376 377 private final BroadcastReceiver mBroadcastReceiver = new BroadcastReceiver() { 378 @Override 379 public void onReceive(Context context, Intent intent) { 380 // Don't use reloadAccountTypesIfNeeded when packages change in case a contacts.xml 381 // was updated. 382 reloadAccountTypes(); 383 } 384 }; 385 386 private final BroadcastReceiver mSimBroadcastReceiver = new BroadcastReceiver() { 387 @Override 388 public void onReceive(Context context, Intent intent) { 389 if (ContactsContract.SimContacts.ACTION_SIM_ACCOUNTS_CHANGED.equals( 390 intent.getAction())) { 391 reloadSimAccounts(); 392 } 393 } 394 }; 395 396 /** 397 * Internal constructor that only performs initial parsing. 398 */ 399 public AccountTypeManagerImpl(Context context) { 400 mContext = context; 401 mLocalAccountLocator = new DeviceLocalAccountLocator(context, AccountManager.get(context)); 402 mTypeProvider = new AccountTypeProvider(context); 403 mFallbackAccountType = new FallbackAccountType(context); 404 405 mAccountManager = AccountManager.get(mContext); 406 407 mExecutor = ContactsExecutors.getDefaultThreadPoolExecutor(); 408 mMainThreadExecutor = ContactsExecutors.newHandlerExecutor(mMainThreadHandler); 409 410 // Request updates when packages or accounts change 411 IntentFilter filter = new IntentFilter(Intent.ACTION_PACKAGE_ADDED); 412 filter.addAction(Intent.ACTION_PACKAGE_REMOVED); 413 filter.addAction(Intent.ACTION_PACKAGE_CHANGED); 414 filter.addDataScheme("package"); 415 mContext.registerReceiver(mBroadcastReceiver, filter); 416 IntentFilter sdFilter = new IntentFilter(); 417 sdFilter.addAction(Intent.ACTION_EXTERNAL_APPLICATIONS_AVAILABLE); 418 sdFilter.addAction(Intent.ACTION_EXTERNAL_APPLICATIONS_UNAVAILABLE); 419 mContext.registerReceiver(mBroadcastReceiver, sdFilter); 420 421 // Request updates when locale is changed so that the order of each field will 422 // be able to be changed on the locale change. 423 filter = new IntentFilter(Intent.ACTION_LOCALE_CHANGED); 424 mContext.registerReceiver(mBroadcastReceiver, filter); 425 426 IntentFilter simFilter = new IntentFilter( 427 ContactsContract.SimContacts.ACTION_SIM_ACCOUNTS_CHANGED); 428 mContext.registerReceiver(mSimBroadcastReceiver, simFilter, Context.RECEIVER_EXPORTED); 429 430 mAccountManager.addOnAccountsUpdatedListener(this, mMainThreadHandler, false); 431 432 ContentResolver.addStatusChangeListener(ContentResolver.SYNC_OBSERVER_TYPE_SETTINGS, this); 433 434 loadAccountTypes(); 435 } 436 437 @Override 438 public void onStatusChanged(int which) { 439 reloadAccountTypesIfNeeded(); 440 } 441 442 /* This notification will arrive on the UI thread */ 443 public void onAccountsUpdated(Account[] accounts) { 444 reloadLocalAccounts(); 445 maybeNotifyAccountsUpdated(mAccountManagerAccounts, 446 getAccountsWithDataSets(accounts, mTypeProvider)); 447 } 448 449 private void maybeNotifyAccountsUpdated(List<AccountWithDataSet> current, 450 List<AccountWithDataSet> update) { 451 if (Objects.equal(current, update)) { 452 return; 453 } 454 current.clear(); 455 current.addAll(update); 456 notifyAccountsChanged(); 457 } 458 459 private void notifyAccountsChanged() { 460 ContactListFilterController.getInstance(mContext).checkFilterValidity(true); 461 LocalBroadcastManager.getInstance(mContext).sendBroadcast( 462 new Intent(BROADCAST_ACCOUNTS_CHANGED)); 463 } 464 465 private synchronized void startLoadingIfNeeded() { 466 if (mTypeProvider == null && mAccountTypesFuture == null) { 467 reloadAccountTypesIfNeeded(); 468 } 469 if (mLocalAccountsFuture == null) { 470 reloadLocalAccounts(); 471 } 472 if (mSimAccountsFuture == null) { 473 reloadSimAccounts(); 474 } 475 } 476 477 private synchronized void loadAccountTypes() { 478 mTypeProvider = new AccountTypeProvider(mContext); 479 480 mAccountTypesFuture = mExecutor.submit(new Callable<AccountTypeProvider>() { 481 @Override 482 public AccountTypeProvider call() throws Exception { 483 // This will request the AccountType for each Account forcing them to be loaded 484 getAccountsWithDataSets(mAccountManager.getAccounts(), mTypeProvider); 485 return mTypeProvider; 486 } 487 }); 488 } 489 490 private FutureCallback<List<AccountWithDataSet>> newAccountsUpdatedCallback( 491 final List<AccountWithDataSet> currentAccounts) { 492 return new FutureCallback<List<AccountWithDataSet>>() { 493 @Override 494 public void onSuccess(List<AccountWithDataSet> result) { 495 maybeNotifyAccountsUpdated(currentAccounts, result); 496 } 497 498 @Override 499 public void onFailure(Throwable t) { 500 } 501 }; 502 } 503 504 private synchronized void reloadAccountTypesIfNeeded() { 505 if (mTypeProvider == null || mTypeProvider.shouldUpdate( 506 mAccountManager.getAuthenticatorTypes(), ContentResolver.getSyncAdapterTypes())) { 507 reloadAccountTypes(); 508 } 509 } 510 511 private synchronized void reloadAccountTypes() { 512 loadAccountTypes(); 513 Futures.addCallback( 514 Futures.transform(mAccountTypesFuture, mAccountsExtractor, 515 MoreExecutors.directExecutor()), 516 newAccountsUpdatedCallback(mAccountManagerAccounts), 517 mMainThreadExecutor); 518 } 519 520 private synchronized void loadLocalAccounts() { 521 mLocalAccountsFuture = mExecutor.submit(new Callable<List<AccountWithDataSet>>() { 522 @Override 523 public List<AccountWithDataSet> call() throws Exception { 524 return mLocalAccountLocator.getDeviceLocalAccounts(); 525 } 526 }); 527 } 528 529 private synchronized void reloadLocalAccounts() { 530 loadLocalAccounts(); 531 Futures.addCallback(mLocalAccountsFuture, newAccountsUpdatedCallback(mLocalAccounts), 532 mMainThreadExecutor); 533 } 534 535 private synchronized void loadSimAccounts() { 536 mSimAccountsFuture = mExecutor.submit(new Callable<List<AccountWithDataSet>>() { 537 @Override 538 public List<AccountWithDataSet> call() throws Exception { 539 List<AccountWithDataSet> simAccountWithDataSets = new ArrayList<>(); 540 List<ContactsContract.SimAccount> simAccounts = 541 ContactsContract.SimContacts.getSimAccounts(mContext.getContentResolver()); 542 for (ContactsContract.SimAccount simAccount : simAccounts) { 543 simAccountWithDataSets.add(new AccountWithDataSet(simAccount.getAccountName(), 544 simAccount.getAccountType(), null)); 545 } 546 return simAccountWithDataSets; 547 } 548 }); 549 } 550 551 private synchronized void reloadSimAccounts() { 552 loadSimAccounts(); 553 Futures.addCallback(mSimAccountsFuture, newAccountsUpdatedCallback(mSimAccounts), 554 mMainThreadExecutor); 555 } 556 557 @Override 558 public ListenableFuture<List<AccountInfo>> getAccountsAsync() { 559 return getAllAccountsAsyncInternal(); 560 } 561 562 private synchronized ListenableFuture<List<AccountInfo>> getAllAccountsAsyncInternal() { 563 startLoadingIfNeeded(); 564 final AccountTypeProvider typeProvider = mTypeProvider; 565 final ListenableFuture<List<List<AccountWithDataSet>>> all = 566 Futures.nonCancellationPropagating( 567 Futures.successfulAsList( 568 Futures.transform(mAccountTypesFuture, mAccountsExtractor, 569 MoreExecutors.directExecutor()), 570 mLocalAccountsFuture, 571 mSimAccountsFuture)); 572 573 return Futures.transform(all, new Function<List<List<AccountWithDataSet>>, 574 List<AccountInfo>>() { 575 @Nullable 576 @Override 577 public List<AccountInfo> apply(@Nullable List<List<AccountWithDataSet>> input) { 578 // input.get(0) contains accounts from AccountManager 579 // input.get(1) contains device local accounts 580 // input.get(2) contains SIM accounts 581 Preconditions.checkArgument(input.size() == 3, 582 "List should have exactly 3 elements"); 583 584 final List<AccountInfo> result = new ArrayList<>(); 585 for (AccountWithDataSet account : input.get(0)) { 586 result.add( 587 typeProvider.getTypeForAccount(account).wrapAccount(mContext, account)); 588 } 589 590 for (AccountWithDataSet account : input.get(1)) { 591 result.add( 592 typeProvider.getTypeForAccount(account).wrapAccount(mContext, account)); 593 } 594 595 for (AccountWithDataSet account : input.get(2)) { 596 result.add( 597 typeProvider.getTypeForAccount(account).wrapAccount(mContext, account)); 598 } 599 AccountInfo.sortAccounts(null, result); 600 return result; 601 } 602 }, MoreExecutors.directExecutor()); 603 } 604 605 @Override 606 public ListenableFuture<List<AccountInfo>> filterAccountsAsync( 607 final Predicate<AccountInfo> filter) { 608 return Futures.transform(getAllAccountsAsyncInternal(), new Function<List<AccountInfo>, 609 List<AccountInfo>>() { 610 @Override 611 public List<AccountInfo> apply(List<AccountInfo> input) { 612 return new ArrayList<>(Collections2.filter(input, filter)); 613 } 614 }, mExecutor); 615 } 616 617 @Override 618 public AccountInfo getAccountInfoForAccount(AccountWithDataSet account) { 619 if (account == null) { 620 return null; 621 } 622 AccountType type = mTypeProvider.getTypeForAccount(account); 623 if (type == null) { 624 type = mFallbackAccountType; 625 } 626 return type.wrapAccount(mContext, account); 627 } 628 629 private List<AccountWithDataSet> getAccountsWithDataSets(Account[] accounts, 630 AccountTypeProvider typeProvider) { 631 List<AccountWithDataSet> result = new ArrayList<>(); 632 for (Account account : accounts) { 633 final List<AccountType> types = typeProvider.getAccountTypes(account.type); 634 for (AccountType type : types) { 635 result.add(new AccountWithDataSet( 636 account.name, account.type, type.dataSet)); 637 } 638 } 639 return result; 640 } 641 642 /** 643 * Returns the default google account specified in preferences, the first google account 644 * if it is not specified in preferences or is no longer on the device, and null otherwise. 645 */ 646 @Override 647 public Account getDefaultGoogleAccount() { 648 final SharedPreferences sharedPreferences = 649 mContext.getSharedPreferences(mContext.getPackageName(), Context.MODE_PRIVATE); 650 final String defaultAccountKey = 651 mContext.getResources().getString(R.string.contact_editor_default_account_key); 652 return getDefaultGoogleAccount(mAccountManager, sharedPreferences, defaultAccountKey); 653 } 654 655 @Override 656 public List<AccountInfo> getWritableGoogleAccounts() { 657 final Account[] googleAccounts = 658 mAccountManager.getAccountsByType(GoogleAccountType.ACCOUNT_TYPE); 659 final List<AccountInfo> result = new ArrayList<>(); 660 for (Account account : googleAccounts) { 661 final AccountWithDataSet accountWithDataSet = new AccountWithDataSet( 662 account.name, account.type, null); 663 final AccountType type = mTypeProvider.getTypeForAccount(accountWithDataSet); 664 if (type != null) { 665 // Accounts with a dataSet (e.g. Google plus accounts) are not writable. 666 result.add(type.wrapAccount(mContext, accountWithDataSet)); 667 } 668 } 669 return result; 670 } 671 672 /** 673 * Returns true if there are real accounts (not "local" account) in the list of accounts. 674 * 675 * <p>This is overriden for performance since the default implementation blocks until all 676 * accounts are loaded 677 * </p> 678 */ 679 @Override 680 public boolean hasNonLocalAccount() { 681 final Account[] accounts = mAccountManager.getAccounts(); 682 if (accounts == null) { 683 return false; 684 } 685 for (Account account : accounts) { 686 if (mTypeProvider.supportsContactsSyncing(account.type)) { 687 return true; 688 } 689 } 690 return false; 691 } 692 693 /** 694 * Find the best {@link DataKind} matching the requested 695 * {@link AccountType#accountType}, {@link AccountType#dataSet}, and {@link DataKind#mimeType}. 696 * If no direct match found, we try searching {@link FallbackAccountType}. 697 */ 698 @Override 699 public DataKind getKindOrFallback(AccountType type, String mimeType) { 700 DataKind kind = null; 701 702 // Try finding account type and kind matching request 703 if (type != null) { 704 kind = type.getKindForMimetype(mimeType); 705 } 706 707 if (kind == null) { 708 // Nothing found, so try fallback as last resort 709 kind = mFallbackAccountType.getKindForMimetype(mimeType); 710 } 711 712 if (kind == null) { 713 if (Log.isLoggable(TAG, Log.DEBUG)) { 714 Log.d(TAG, "Unknown type=" + type + ", mime=" + mimeType); 715 } 716 } 717 718 return kind; 719 } 720 721 /** 722 * Returns whether the account still exists on the device 723 * 724 * <p>This is overridden for performance. The default implementation loads all accounts then 725 * searches through them for specified. This implementation will only load the types for the 726 * specified AccountType (it may still require blocking on IO in some cases but it shouldn't 727 * be as bad as blocking for all accounts). 728 * </p> 729 */ 730 @Override 731 public boolean exists(AccountWithDataSet account) { 732 final Account[] accounts = mAccountManager.getAccountsByType(account.type); 733 for (Account existingAccount : accounts) { 734 if (existingAccount.name.equals(account.name)) { 735 return mTypeProvider.getTypeForAccount(account) != null; 736 } 737 } 738 return false; 739 } 740 741 /** 742 * Return {@link AccountType} for the given account type and data set. 743 */ 744 @Override 745 public AccountType getAccountType(AccountTypeWithDataSet accountTypeWithDataSet) { 746 final AccountType type = mTypeProvider.getType( 747 accountTypeWithDataSet.accountType, accountTypeWithDataSet.dataSet); 748 return type != null ? type : mFallbackAccountType; 749 } 750 } 751