1 /*
2  * Copyright (C) 2016 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 package com.android.contacts;
17 
18 import android.app.Activity;
19 import android.app.Fragment;
20 import android.app.LoaderManager;
21 import android.content.Context;
22 import android.content.IntentFilter;
23 import android.content.Loader;
24 import android.os.Bundle;
25 import androidx.annotation.NonNull;
26 import androidx.annotation.Nullable;
27 import com.google.android.material.snackbar.Snackbar;
28 
29 import androidx.collection.ArrayMap;
30 import androidx.core.view.ViewCompat;
31 import androidx.core.widget.ContentLoadingProgressBar;
32 import androidx.appcompat.widget.Toolbar;
33 import android.util.SparseBooleanArray;
34 import android.view.LayoutInflater;
35 import android.view.View;
36 import android.view.ViewGroup;
37 import android.widget.AbsListView;
38 import android.widget.AdapterView;
39 import android.widget.ArrayAdapter;
40 import android.widget.ListView;
41 import android.widget.TextView;
42 
43 import com.android.contacts.compat.CompatUtils;
44 import com.android.contacts.database.SimContactDao;
45 import com.android.contacts.editor.AccountHeaderPresenter;
46 import com.android.contacts.model.AccountTypeManager;
47 import com.android.contacts.model.SimCard;
48 import com.android.contacts.model.SimContact;
49 import com.android.contacts.model.account.AccountInfo;
50 import com.android.contacts.model.account.AccountWithDataSet;
51 import com.android.contacts.preference.ContactsPreferences;
52 import com.android.contacts.util.concurrent.ContactsExecutors;
53 import com.android.contacts.util.concurrent.ListenableFutureLoader;
54 import com.google.common.base.Function;
55 import com.google.common.util.concurrent.Futures;
56 import com.google.common.util.concurrent.ListenableFuture;
57 import com.google.common.util.concurrent.MoreExecutors;
58 
59 import java.util.ArrayList;
60 import java.util.Arrays;
61 import java.util.Collections;
62 import java.util.List;
63 import java.util.Map;
64 import java.util.Set;
65 import java.util.concurrent.Callable;
66 
67 /**
68  * Dialog that presents a list of contacts from a SIM card that can be imported into a selected
69  * account
70  */
71 public class SimImportFragment extends Fragment
72         implements LoaderManager.LoaderCallbacks<SimImportFragment.LoaderResult>,
73         AdapterView.OnItemClickListener, AbsListView.OnScrollListener {
74 
75     private static final String KEY_SUFFIX_SELECTED_IDS = "_selectedIds";
76     private static final String ARG_SUBSCRIPTION_ID = "subscriptionId";
77 
78     private ContactsPreferences mPreferences;
79     private AccountTypeManager mAccountTypeManager;
80     private SimContactAdapter mAdapter;
81     private View mAccountHeaderContainer;
82     private AccountHeaderPresenter mAccountHeaderPresenter;
83     private float mAccountScrolledElevationPixels;
84     private ContentLoadingProgressBar mLoadingIndicator;
85     private Toolbar mToolbar;
86     private ListView mListView;
87     private View mImportButton;
88 
89     private Bundle mSavedInstanceState;
90 
91     private final Map<AccountWithDataSet, long[]> mPerAccountCheckedIds = new ArrayMap<>();
92 
93     private int mSubscriptionId;
94 
95     @Override
onCreate(final Bundle savedInstanceState)96     public void onCreate(final Bundle savedInstanceState) {
97         super.onCreate(savedInstanceState);
98 
99         mSavedInstanceState = savedInstanceState;
100         mPreferences = new ContactsPreferences(getContext());
101         mAccountTypeManager = AccountTypeManager.getInstance(getActivity());
102         mAdapter = new SimContactAdapter(getActivity());
103 
104         final Bundle args = getArguments();
105         mSubscriptionId = args == null ? SimCard.NO_SUBSCRIPTION_ID :
106                 args.getInt(ARG_SUBSCRIPTION_ID, SimCard.NO_SUBSCRIPTION_ID);
107     }
108 
109     @Override
onActivityCreated(Bundle savedInstanceState)110     public void onActivityCreated(Bundle savedInstanceState) {
111         super.onActivityCreated(savedInstanceState);
112         getLoaderManager().initLoader(0, null, this);
113     }
114 
115     @Nullable
116     @Override
onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState)117     public View onCreateView(LayoutInflater inflater, ViewGroup container,
118             Bundle savedInstanceState) {
119         final View view = inflater.inflate(R.layout.fragment_sim_import, container, false);
120 
121         mAccountHeaderContainer = view.findViewById(R.id.account_header_container);
122         mAccountScrolledElevationPixels = getResources()
123                 .getDimension(R.dimen.contact_list_header_elevation);
124         mAccountHeaderPresenter = new AccountHeaderPresenter(
125                 mAccountHeaderContainer);
126         if (savedInstanceState != null) {
127             mAccountHeaderPresenter.onRestoreInstanceState(savedInstanceState);
128         } else {
129             // Default may be null in which case the first account in the list will be selected
130             // after they are loaded.
131             mAccountHeaderPresenter.setCurrentAccount(mPreferences.getDefaultAccount());
132         }
133         mAccountHeaderPresenter.setObserver(new AccountHeaderPresenter.Observer() {
134             @Override
135             public void onChange(AccountHeaderPresenter sender) {
136                 rememberSelectionsForCurrentAccount();
137                 mAdapter.setAccount(sender.getCurrentAccount());
138                 showSelectionsForCurrentAccount();
139                 updateToolbarWithCurrentSelections();
140             }
141         });
142         mAdapter.setAccount(mAccountHeaderPresenter.getCurrentAccount());
143 
144         mListView = (ListView) view.findViewById(R.id.list);
145         mListView.setOnScrollListener(this);
146         mListView.setAdapter(mAdapter);
147         mListView.setChoiceMode(AbsListView.CHOICE_MODE_MULTIPLE);
148         mListView.setOnItemClickListener(this);
149         mImportButton = view.findViewById(R.id.import_button);
150         mImportButton.setOnClickListener(new View.OnClickListener() {
151             @Override
152             public void onClick(View v) {
153                 importCurrentSelections();
154                 // Do we wait for import to finish?
155                 getActivity().setResult(Activity.RESULT_OK);
156                 getActivity().finish();
157             }
158         });
159 
160         mToolbar = (Toolbar) view.findViewById(R.id.toolbar);
161         mToolbar.setNavigationOnClickListener(new View.OnClickListener() {
162             @Override
163             public void onClick(View v) {
164                 getActivity().setResult(Activity.RESULT_CANCELED);
165                 getActivity().finish();
166             }
167         });
168 
169         mLoadingIndicator = (ContentLoadingProgressBar) view.findViewById(R.id.loading_progress);
170 
171         return view;
172     }
173 
rememberSelectionsForCurrentAccount()174     private void rememberSelectionsForCurrentAccount() {
175         final AccountWithDataSet current = mAdapter.getAccount();
176         if (current == null) {
177             return;
178         }
179         final long[] ids = mListView.getCheckedItemIds();
180         Arrays.sort(ids);
181         mPerAccountCheckedIds.put(current, ids);
182     }
183 
showSelectionsForCurrentAccount()184     private void showSelectionsForCurrentAccount() {
185         final long[] ids = mPerAccountCheckedIds.get(mAdapter.getAccount());
186         if (ids == null) {
187             selectAll();
188             return;
189         }
190         for (int i = 0, len = mListView.getCount(); i < len; i++) {
191             mListView.setItemChecked(i,
192                     Arrays.binarySearch(ids, mListView.getItemIdAtPosition(i)) >= 0);
193         }
194     }
195 
selectAll()196     private void selectAll() {
197         for (int i = 0, len = mListView.getCount(); i < len; i++) {
198             mListView.setItemChecked(i, true);
199         }
200     }
201 
updateToolbarWithCurrentSelections()202     private void updateToolbarWithCurrentSelections() {
203         // The ListView keeps checked state for items that are disabled but we only want  to
204         // consider items that don't exist in the current account when updating the toolbar
205         int importableCount = 0;
206         final SparseBooleanArray checked = mListView.getCheckedItemPositions();
207         for (int i = 0; i < checked.size(); i++) {
208             if (checked.valueAt(i) && !mAdapter.existsInCurrentAccount(checked.keyAt(i))) {
209                 importableCount++;
210             }
211         }
212 
213         if (importableCount == 0) {
214             mImportButton.setVisibility(View.GONE);
215             mToolbar.setTitle(R.string.sim_import_title_none_selected);
216         } else {
217             mToolbar.setTitle(String.valueOf(importableCount));
218             mImportButton.setVisibility(View.VISIBLE);
219         }
220     }
221 
222     @Override
onStart()223     public void onStart() {
224         super.onStart();
225         if (mAdapter.isEmpty() && getLoaderManager().getLoader(0).isStarted()) {
226             mLoadingIndicator.show();
227         }
228     }
229 
230     @Override
onSaveInstanceState(Bundle outState)231     public void onSaveInstanceState(Bundle outState) {
232         rememberSelectionsForCurrentAccount();
233         // We'll restore this manually so we don't need the list to preserve it's own state.
234         mListView.clearChoices();
235         super.onSaveInstanceState(outState);
236         mAccountHeaderPresenter.onSaveInstanceState(outState);
237         saveAdapterSelectedStates(outState);
238     }
239 
240     @Override
onCreateLoader(int id, Bundle args)241     public Loader<LoaderResult> onCreateLoader(int id, Bundle args) {
242         return new SimContactLoader(getContext(), mSubscriptionId);
243     }
244 
245     @Override
onLoadFinished(Loader<LoaderResult> loader, LoaderResult data)246     public void onLoadFinished(Loader<LoaderResult> loader,
247             LoaderResult data) {
248         mLoadingIndicator.hide();
249         if (data == null) {
250             return;
251         }
252         mAccountHeaderPresenter.setAccounts(data.accounts);
253         restoreAdapterSelectedStates(data.accounts);
254         mAdapter.setData(data);
255         mListView.setEmptyView(getView().findViewById(R.id.empty_message));
256 
257         showSelectionsForCurrentAccount();
258         updateToolbarWithCurrentSelections();
259     }
260 
261     @Override
onLoaderReset(Loader<LoaderResult> loader)262     public void onLoaderReset(Loader<LoaderResult> loader) {
263     }
264 
restoreAdapterSelectedStates(List<AccountInfo> accounts)265     private void restoreAdapterSelectedStates(List<AccountInfo> accounts) {
266         if (mSavedInstanceState == null) {
267             return;
268         }
269 
270         for (AccountInfo account : accounts) {
271             final long[] selections = mSavedInstanceState.getLongArray(
272                     account.getAccount().stringify() + KEY_SUFFIX_SELECTED_IDS);
273             mPerAccountCheckedIds.put(account.getAccount(), selections);
274         }
275         mSavedInstanceState = null;
276     }
277 
saveAdapterSelectedStates(Bundle outState)278     private void saveAdapterSelectedStates(Bundle outState) {
279         if (mAdapter == null) {
280             return;
281         }
282 
283         // Make sure the selections are up-to-date
284         for (Map.Entry<AccountWithDataSet, long[]> entry : mPerAccountCheckedIds.entrySet()) {
285             outState.putLongArray(entry.getKey().stringify() + KEY_SUFFIX_SELECTED_IDS,
286                     entry.getValue());
287         }
288     }
289 
importCurrentSelections()290     private void importCurrentSelections() {
291         final SparseBooleanArray checked = mListView.getCheckedItemPositions();
292         final ArrayList<SimContact> importableContacts = new ArrayList<>(checked.size());
293         for (int i = 0; i < checked.size(); i++) {
294             // It's possible for existing contacts to be "checked" but we only want to import the
295             // ones that don't already exist
296             if (checked.valueAt(i) && !mAdapter.existsInCurrentAccount(i)) {
297                 importableContacts.add(mAdapter.getItem(checked.keyAt(i)));
298             }
299         }
300         SimImportService.startImport(getContext(), mSubscriptionId, importableContacts,
301                 mAccountHeaderPresenter.getCurrentAccount());
302     }
303 
onItemClick(AdapterView<?> parent, View view, int position, long id)304     public void onItemClick(AdapterView<?> parent, View view, int position, long id) {
305         if (mAdapter.existsInCurrentAccount(position)) {
306             Snackbar.make(getView(), R.string.sim_import_contact_exists_toast,
307                     Snackbar.LENGTH_LONG).show();
308         } else {
309             updateToolbarWithCurrentSelections();
310         }
311     }
312 
getContext()313     public Context getContext() {
314         if (CompatUtils.isMarshmallowCompatible()) {
315             return super.getContext();
316         }
317         return getActivity();
318     }
319 
320     @Override
onScrollStateChanged(AbsListView view, int scrollState)321     public void onScrollStateChanged(AbsListView view, int scrollState) { }
322 
323     @Override
onScroll(AbsListView view, int firstVisibleItem, int visibleItemCount, int totalItemCount)324     public void onScroll(AbsListView view, int firstVisibleItem, int visibleItemCount,
325             int totalItemCount) {
326         int firstCompletelyVisibleItem = firstVisibleItem;
327         if (view != null && view.getChildAt(0) != null && view.getChildAt(0).getTop() < 0) {
328             firstCompletelyVisibleItem++;
329         }
330 
331         if (firstCompletelyVisibleItem == 0) {
332             ViewCompat.setElevation(mAccountHeaderContainer, 0);
333         } else {
334             ViewCompat.setElevation(mAccountHeaderContainer, mAccountScrolledElevationPixels);
335         }
336     }
337 
338     /**
339      * Creates a fragment that will display contacts stored on the default SIM card
340      */
newInstance()341     public static SimImportFragment newInstance() {
342         return new SimImportFragment();
343     }
344 
345     /**
346      * Creates a fragment that will display the contacts stored on the SIM card that has the
347      * provided subscriptionId
348      */
newInstance(int subscriptionId)349     public static SimImportFragment newInstance(int subscriptionId) {
350         final SimImportFragment fragment = new SimImportFragment();
351         final Bundle args = new Bundle();
352         args.putInt(ARG_SUBSCRIPTION_ID, subscriptionId);
353         fragment.setArguments(args);
354         return fragment;
355     }
356 
357     private static class SimContactAdapter extends ArrayAdapter<SimContact> {
358         private Map<AccountWithDataSet, Set<SimContact>> mExistingMap;
359         private AccountWithDataSet mSelectedAccount;
360         private LayoutInflater mInflater;
361 
SimContactAdapter(Context context)362         public SimContactAdapter(Context context) {
363             super(context, 0);
364             mInflater = (LayoutInflater) context.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
365         }
366 
367         @Override
getItemId(int position)368         public long getItemId(int position) {
369             // This can be called by the framework when the adapter hasn't been initialized for
370             // checking the checked state of items. See b/33108913
371             if (position < 0 || position >= getCount()) {
372                 return View.NO_ID;
373             }
374             return getItem(position).getRecordNumber();
375         }
376 
377         @Override
hasStableIds()378         public boolean hasStableIds() {
379             return true;
380         }
381 
382         @Override
getViewTypeCount()383         public int getViewTypeCount() {
384             return 2;
385         }
386 
387         @Override
getItemViewType(int position)388         public int getItemViewType(int position) {
389             return !existsInCurrentAccount(position) ? 0 : 1;
390         }
391 
392         @NonNull
393         @Override
getView(int position, View convertView, ViewGroup parent)394         public View getView(int position, View convertView, ViewGroup parent) {
395             TextView text = (TextView) convertView;
396             if (text == null) {
397                 final int layoutRes = existsInCurrentAccount(position) ?
398                         R.layout.sim_import_list_item_disabled :
399                         R.layout.sim_import_list_item;
400                 text = (TextView) mInflater.inflate(layoutRes, parent, false);
401             }
402             text.setText(getItemLabel(getItem(position)));
403 
404             return text;
405         }
406 
setData(LoaderResult result)407         public void setData(LoaderResult result) {
408             clear();
409             addAll(result.contacts);
410             mExistingMap = result.accountsMap;
411         }
412 
setAccount(AccountWithDataSet account)413         public void setAccount(AccountWithDataSet account) {
414             mSelectedAccount = account;
415             notifyDataSetChanged();
416         }
417 
getAccount()418         public AccountWithDataSet getAccount() {
419             return mSelectedAccount;
420         }
421 
existsInCurrentAccount(int position)422         public boolean existsInCurrentAccount(int position) {
423             return existsInCurrentAccount(getItem(position));
424         }
425 
existsInCurrentAccount(SimContact contact)426         public boolean existsInCurrentAccount(SimContact contact) {
427             if (mSelectedAccount == null || !mExistingMap.containsKey(mSelectedAccount)) {
428                 return false;
429             }
430             return mExistingMap.get(mSelectedAccount).contains(contact);
431         }
432 
getItemLabel(SimContact contact)433         private String getItemLabel(SimContact contact) {
434             if (contact.hasName()) {
435                 return contact.getName();
436             } else if (contact.hasPhone()) {
437                 return contact.getPhone();
438             } else if (contact.hasEmails()) {
439                 return contact.getEmails()[0];
440             } else {
441                 // This isn't really possible because we skip empty SIM contacts during loading
442                 return "";
443             }
444         }
445     }
446 
447 
448     private static class SimContactLoader extends ListenableFutureLoader<LoaderResult> {
449         private SimContactDao mDao;
450         private AccountTypeManager mAccountTypeManager;
451         private final int mSubscriptionId;
452 
SimContactLoader(Context context, int subscriptionId)453         public SimContactLoader(Context context, int subscriptionId) {
454             super(context, new IntentFilter(AccountTypeManager.BROADCAST_ACCOUNTS_CHANGED));
455             mDao = SimContactDao.create(context);
456             mAccountTypeManager = AccountTypeManager.getInstance(getContext());
457             mSubscriptionId = subscriptionId;
458         }
459 
460         @Override
loadData()461         protected ListenableFuture<LoaderResult> loadData() {
462             final ListenableFuture<List<Object>> future = Futures.<Object>allAsList(
463                     mAccountTypeManager
464                             .filterAccountsAsync(AccountTypeManager.writableFilter()),
465                     ContactsExecutors.getSimReadExecutor().<Object>submit(
466                             new Callable<Object>() {
467                         @Override
468                         public LoaderResult call() throws Exception {
469                             return loadFromSim();
470                         }
471                     }));
472             return Futures.transform(future, new Function<List<Object>, LoaderResult>() {
473                 @Override
474                 public LoaderResult apply(List<Object> input) {
475                     final List<AccountInfo> accounts = (List<AccountInfo>) input.get(0);
476                     final LoaderResult simLoadResult = (LoaderResult) input.get(1);
477                     simLoadResult.accounts = accounts;
478                     return simLoadResult;
479                 }
480             }, MoreExecutors.directExecutor());
481         }
482 
loadFromSim()483         private LoaderResult loadFromSim() {
484             final SimCard sim = mDao.getSimBySubscriptionId(mSubscriptionId);
485             LoaderResult result = new LoaderResult();
486             if (sim == null) {
487                 result.contacts = new ArrayList<>();
488                 result.accountsMap = Collections.emptyMap();
489                 return result;
490             }
491             result.contacts = mDao.loadContactsForSim(sim);
492             result.accountsMap = mDao.findAccountsOfExistingSimContacts(result.contacts);
493             return result;
494         }
495     }
496 
497     public static class LoaderResult {
498         public List<AccountInfo> accounts;
499         public ArrayList<SimContact> contacts;
500         public Map<AccountWithDataSet, Set<SimContact>> accountsMap;
501     }
502 }
503