• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright (C) 2008 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.dialer;
18 
19 import android.app.ActionBar;
20 import android.app.ActionBar.LayoutParams;
21 import android.app.ActionBar.Tab;
22 import android.app.ActionBar.TabListener;
23 import android.app.Activity;
24 import android.app.Fragment;
25 import android.app.FragmentManager;
26 import android.app.FragmentTransaction;
27 import android.content.ActivityNotFoundException;
28 import android.content.Context;
29 import android.content.Intent;
30 import android.content.SharedPreferences;
31 import android.net.Uri;
32 import android.os.Bundle;
33 import android.os.RemoteException;
34 import android.os.ServiceManager;
35 import android.preference.PreferenceManager;
36 import android.provider.CallLog.Calls;
37 import android.provider.ContactsContract.Contacts;
38 import android.provider.ContactsContract.Intents.UI;
39 import android.support.v13.app.FragmentPagerAdapter;
40 import android.support.v4.view.ViewPager;
41 import android.support.v4.view.ViewPager.OnPageChangeListener;
42 import android.text.TextUtils;
43 import android.util.DisplayMetrics;
44 import android.util.Log;
45 import android.view.Menu;
46 import android.view.MenuInflater;
47 import android.view.MenuItem;
48 import android.view.MenuItem.OnMenuItemClickListener;
49 import android.view.View;
50 import android.view.View.OnClickListener;
51 import android.view.View.OnFocusChangeListener;
52 import android.view.ViewConfiguration;
53 import android.view.ViewGroup;
54 import android.view.inputmethod.InputMethodManager;
55 import android.widget.PopupMenu;
56 import android.widget.SearchView;
57 import android.widget.SearchView.OnCloseListener;
58 import android.widget.SearchView.OnQueryTextListener;
59 import android.widget.Toast;
60 
61 import com.android.contacts.common.CallUtil;
62 import com.android.contacts.common.activity.TransactionSafeActivity;
63 import com.android.contacts.common.list.ContactListFilterController;
64 import com.android.contacts.common.list.ContactListFilterController.ContactListFilterListener;
65 import com.android.contacts.common.list.ContactListItemView;
66 import com.android.contacts.common.list.OnPhoneNumberPickerActionListener;
67 import com.android.contacts.common.list.PhoneNumberPickerFragment;
68 import com.android.contacts.common.util.AccountFilterUtil;
69 import com.android.dialer.calllog.CallLogFragment;
70 import com.android.dialer.dialpad.DialpadFragment;
71 import com.android.dialer.interactions.PhoneNumberInteraction;
72 import com.android.dialer.list.PhoneFavoriteFragment;
73 import com.android.dialer.util.OrientationUtil;
74 import com.android.internal.telephony.ITelephony;
75 
76 /**
77  * The dialer activity that has one tab with the virtual 12key
78  * dialer, a tab with recent calls in it, a tab with the contacts and
79  * a tab with the favorite. This is the container and the tabs are
80  * embedded using intents.
81  * The dialer tab's title is 'phone', a more common name (see strings.xml).
82  */
83 public class DialtactsActivity extends TransactionSafeActivity implements View.OnClickListener {
84     private static final String TAG = "DialtactsActivity";
85 
86     public static final boolean DEBUG = false;
87 
88     /** Used to open Call Setting */
89     private static final String PHONE_PACKAGE = "com.android.phone";
90     private static final String CALL_SETTINGS_CLASS_NAME =
91             "com.android.phone.CallFeaturesSetting";
92 
93     /** @see #getCallOrigin() */
94     private static final String CALL_ORIGIN_DIALTACTS =
95             "com.android.dialer.DialtactsActivity";
96 
97     /**
98      * Just for backward compatibility. Should behave as same as {@link Intent#ACTION_DIAL}.
99      */
100     private static final String ACTION_TOUCH_DIALER = "com.android.phone.action.TOUCH_DIALER";
101 
102     /** Used both by {@link ActionBar} and {@link ViewPagerAdapter} */
103     private static final int TAB_INDEX_DIALER = 0;
104     private static final int TAB_INDEX_CALL_LOG = 1;
105     private static final int TAB_INDEX_FAVORITES = 2;
106 
107     private static final int TAB_INDEX_COUNT = 3;
108 
109     private SharedPreferences mPrefs;
110 
111     /** Last manually selected tab index */
112     private static final String PREF_LAST_MANUALLY_SELECTED_TAB =
113             "DialtactsActivity_last_manually_selected_tab";
114     private static final int PREF_LAST_MANUALLY_SELECTED_TAB_DEFAULT = TAB_INDEX_DIALER;
115 
116     private static final int SUBACTIVITY_ACCOUNT_FILTER = 1;
117 
118     public class ViewPagerAdapter extends FragmentPagerAdapter {
ViewPagerAdapter(FragmentManager fm)119         public ViewPagerAdapter(FragmentManager fm) {
120             super(fm);
121         }
122 
123         @Override
getItem(int position)124         public Fragment getItem(int position) {
125             switch (position) {
126                 case TAB_INDEX_DIALER:
127                     return new DialpadFragment();
128                 case TAB_INDEX_CALL_LOG:
129                     return new CallLogFragment();
130                 case TAB_INDEX_FAVORITES:
131                     return new PhoneFavoriteFragment();
132             }
133             throw new IllegalStateException("No fragment at position " + position);
134         }
135 
136         @Override
setPrimaryItem(ViewGroup container, int position, Object object)137         public void setPrimaryItem(ViewGroup container, int position, Object object) {
138             // The parent's setPrimaryItem() also calls setMenuVisibility(), so we want to know
139             // when it happens.
140             if (DEBUG) {
141                 Log.d(TAG, "FragmentPagerAdapter#setPrimaryItem(), position: " + position);
142             }
143             super.setPrimaryItem(container, position, object);
144         }
145 
146         @Override
getCount()147         public int getCount() {
148             return TAB_INDEX_COUNT;
149         }
150     }
151 
152     /**
153      * True when the app detects user's drag event. This variable should not become true when
154      * mUserTabClick is true.
155      *
156      * During user's drag or tab click, we shouldn't show fake buttons but just show real
157      * ActionBar at the bottom of the screen, for transition animation.
158      */
159     boolean mDuringSwipe = false;
160     /**
161      * True when the app detects user's tab click (at the top of the screen). This variable should
162      * not become true when mDuringSwipe is true.
163      *
164      * During user's drag or tab click, we shouldn't show fake buttons but just show real
165      * ActionBar at the bottom of the screen, for transition animation.
166      */
167     boolean mUserTabClick = false;
168 
169     private class PageChangeListener implements OnPageChangeListener {
170         private int mCurrentPosition = -1;
171         /**
172          * Used during page migration, to remember the next position {@link #onPageSelected(int)}
173          * specified.
174          */
175         private int mNextPosition = -1;
176 
177         @Override
onPageScrolled( int position, float positionOffset, int positionOffsetPixels)178         public void onPageScrolled(
179                 int position, float positionOffset, int positionOffsetPixels) {
180         }
181 
182         @Override
onPageSelected(int position)183         public void onPageSelected(int position) {
184             if (DEBUG) Log.d(TAG, "onPageSelected: position: " + position);
185             final ActionBar actionBar = getActionBar();
186             if (mDialpadFragment != null) {
187                 if (mDuringSwipe && position == TAB_INDEX_DIALER) {
188                     // TODO: Figure out if we want this or not. Right now
189                     // - with this call, both fake buttons and real action bar overlap
190                     // - without this call, there's tiny flicker happening to search/menu buttons.
191                     // If we can reduce the flicker without this call, it would be much better.
192                     // updateFakeMenuButtonsVisibility(true);
193                 }
194             }
195 
196             if (mCurrentPosition == position) {
197                 Log.w(TAG, "Previous position and next position became same (" + position + ")");
198             }
199 
200             actionBar.selectTab(actionBar.getTabAt(position));
201             mNextPosition = position;
202         }
203 
setCurrentPosition(int position)204         public void setCurrentPosition(int position) {
205             mCurrentPosition = position;
206         }
207 
getCurrentPosition()208         public int getCurrentPosition() {
209             return mCurrentPosition;
210         }
211 
212         @Override
onPageScrollStateChanged(int state)213         public void onPageScrollStateChanged(int state) {
214             switch (state) {
215                 case ViewPager.SCROLL_STATE_IDLE: {
216                     if (mNextPosition == -1) {
217                         // This happens when the user drags the screen just after launching the
218                         // application, and settle down the same screen without actually swiping it.
219                         // At that moment mNextPosition is apparently -1 yet, and we expect it
220                         // being updated by onPageSelected(), which is *not* called if the user
221                         // settle down the exact same tab after the dragging.
222                         if (DEBUG) {
223                             Log.d(TAG, "Next position is not specified correctly. Use current tab ("
224                                     + mViewPager.getCurrentItem() + ")");
225                         }
226                         mNextPosition = mViewPager.getCurrentItem();
227                     }
228                     if (DEBUG) {
229                         Log.d(TAG, "onPageScrollStateChanged() with SCROLL_STATE_IDLE. "
230                                 + "mCurrentPosition: " + mCurrentPosition
231                                 + ", mNextPosition: " + mNextPosition);
232                     }
233                     // Interpret IDLE as the end of migration (both swipe and tab click)
234                     mDuringSwipe = false;
235                     mUserTabClick = false;
236 
237                     updateFakeMenuButtonsVisibility(mNextPosition == TAB_INDEX_DIALER);
238                     sendFragmentVisibilityChange(mCurrentPosition, false);
239                     sendFragmentVisibilityChange(mNextPosition, true);
240 
241                     invalidateOptionsMenu();
242 
243                     mCurrentPosition = mNextPosition;
244                     break;
245                 }
246                 case ViewPager.SCROLL_STATE_DRAGGING: {
247                     if (DEBUG) Log.d(TAG, "onPageScrollStateChanged() with SCROLL_STATE_DRAGGING");
248                     mDuringSwipe = true;
249                     mUserTabClick = false;
250                     break;
251                 }
252                 case ViewPager.SCROLL_STATE_SETTLING: {
253                     if (DEBUG) Log.d(TAG, "onPageScrollStateChanged() with SCROLL_STATE_SETTLING");
254                     mDuringSwipe = true;
255                     mUserTabClick = false;
256                     break;
257                 }
258                 default:
259                     break;
260             }
261         }
262     }
263 
264     private String mFilterText;
265 
266     /** Enables horizontal swipe between Fragments. */
267     private ViewPager mViewPager;
268     private final PageChangeListener mPageChangeListener = new PageChangeListener();
269     private DialpadFragment mDialpadFragment;
270     private CallLogFragment mCallLogFragment;
271     private PhoneFavoriteFragment mPhoneFavoriteFragment;
272 
273     private View mSearchButton;
274     private View mMenuButton;
275 
276     private final ContactListFilterListener mContactListFilterListener =
277             new ContactListFilterListener() {
278         @Override
279         public void onContactListFilterChanged() {
280             boolean doInvalidateOptionsMenu = false;
281 
282             if (mPhoneFavoriteFragment != null && mPhoneFavoriteFragment.isAdded()) {
283                 mPhoneFavoriteFragment.setFilter(mContactListFilterController.getFilter());
284                 doInvalidateOptionsMenu = true;
285             }
286 
287             if (mSearchFragment != null && mSearchFragment.isAdded()) {
288                 mSearchFragment.setFilter(mContactListFilterController.getFilter());
289                 doInvalidateOptionsMenu = true;
290             } else {
291                 Log.w(TAG, "Search Fragment isn't available when ContactListFilter is changed");
292             }
293 
294             if (doInvalidateOptionsMenu) {
295                 invalidateOptionsMenu();
296             }
297         }
298     };
299 
300     private final TabListener mTabListener = new TabListener() {
301         @Override
302         public void onTabUnselected(Tab tab, FragmentTransaction ft) {
303             if (DEBUG) Log.d(TAG, "onTabUnselected(). tab: " + tab);
304         }
305 
306         @Override
307         public void onTabSelected(Tab tab, FragmentTransaction ft) {
308             if (DEBUG) {
309                 Log.d(TAG, "onTabSelected(). tab: " + tab + ", mDuringSwipe: " + mDuringSwipe);
310             }
311             // When the user swipes the screen horizontally, this method will be called after
312             // ViewPager.SCROLL_STATE_DRAGGING and ViewPager.SCROLL_STATE_SETTLING events, while
313             // when the user clicks a tab at the ActionBar at the top, this will be called before
314             // them. This logic interprets the order difference as a difference of the user action.
315             if (!mDuringSwipe) {
316                 if (DEBUG) {
317                     Log.d(TAG, "Tab select. from: " + mPageChangeListener.getCurrentPosition()
318                             + ", to: " + tab.getPosition());
319                 }
320                 if (mDialpadFragment != null) {
321                     updateFakeMenuButtonsVisibility(tab.getPosition() == TAB_INDEX_DIALER);
322                 }
323                 mUserTabClick = true;
324             }
325 
326             if (mViewPager.getCurrentItem() != tab.getPosition()) {
327                 mViewPager.setCurrentItem(tab.getPosition(), true);
328             }
329 
330             // During the call, we don't remember the tab position.
331             if (!DialpadFragment.phoneIsInUse()) {
332                 // Remember this tab index. This function is also called, if the tab is set
333                 // automatically in which case the setter (setCurrentTab) has to set this to its old
334                 // value afterwards
335                 mLastManuallySelectedFragment = tab.getPosition();
336             }
337         }
338 
339         @Override
340         public void onTabReselected(Tab tab, FragmentTransaction ft) {
341             if (DEBUG) Log.d(TAG, "onTabReselected");
342         }
343     };
344 
345     /**
346      * Fragment for searching phone numbers. Unlike the other Fragments, this doesn't correspond
347      * to tab but is shown by a search action.
348      */
349     private PhoneNumberPickerFragment mSearchFragment;
350     /**
351      * True when this Activity is in its search UI (with a {@link SearchView} and
352      * {@link PhoneNumberPickerFragment}).
353      */
354     private boolean mInSearchUi;
355     private SearchView mSearchView;
356 
357     private final OnClickListener mFilterOptionClickListener = new OnClickListener() {
358         @Override
359         public void onClick(View view) {
360             final PopupMenu popupMenu = new PopupMenu(DialtactsActivity.this, view);
361             final Menu menu = popupMenu.getMenu();
362             popupMenu.inflate(R.menu.dialtacts_search_options);
363             final MenuItem filterOptionMenuItem = menu.findItem(R.id.filter_option);
364             filterOptionMenuItem.setOnMenuItemClickListener(mFilterOptionsMenuItemClickListener);
365             final MenuItem addContactOptionMenuItem = menu.findItem(R.id.add_contact);
366             addContactOptionMenuItem.setIntent(
367                     new Intent(Intent.ACTION_INSERT, Contacts.CONTENT_URI));
368             popupMenu.show();
369         }
370     };
371 
372     /**
373      * The index of the Fragment (or, the tab) that has last been manually selected.
374      * This value does not keep track of programmatically set Tabs (e.g. Call Log after a Call)
375      */
376     private int mLastManuallySelectedFragment;
377 
378     private ContactListFilterController mContactListFilterController;
379     private OnMenuItemClickListener mFilterOptionsMenuItemClickListener =
380             new OnMenuItemClickListener() {
381         @Override
382         public boolean onMenuItemClick(MenuItem item) {
383             AccountFilterUtil.startAccountFilterActivityForResult(
384                     DialtactsActivity.this, SUBACTIVITY_ACCOUNT_FILTER,
385                     mContactListFilterController.getFilter());
386             return true;
387         }
388     };
389 
390     private OnMenuItemClickListener mSearchMenuItemClickListener =
391             new OnMenuItemClickListener() {
392         @Override
393         public boolean onMenuItemClick(MenuItem item) {
394             enterSearchUi();
395             return true;
396         }
397     };
398 
399     /**
400      * Listener used when one of phone numbers in search UI is selected. This will initiate a
401      * phone call using the phone number.
402      */
403     private final OnPhoneNumberPickerActionListener mPhoneNumberPickerActionListener =
404             new OnPhoneNumberPickerActionListener() {
405                 @Override
406                 public void onPickPhoneNumberAction(Uri dataUri) {
407                     // Specify call-origin so that users will see the previous tab instead of
408                     // CallLog screen (search UI will be automatically exited).
409                     PhoneNumberInteraction.startInteractionForPhoneCall(
410                             DialtactsActivity.this, dataUri, getCallOrigin());
411                 }
412 
413                 @Override
414                 public void onShortcutIntentCreated(Intent intent) {
415                     Log.w(TAG, "Unsupported intent has come (" + intent + "). Ignoring.");
416                 }
417 
418                 @Override
419                 public void onHomeInActionBarSelected() {
420                     exitSearchUi();
421                 }
422     };
423 
424     /**
425      * Listener used to send search queries to the phone search fragment.
426      */
427     private final OnQueryTextListener mPhoneSearchQueryTextListener =
428             new OnQueryTextListener() {
429                 @Override
430                 public boolean onQueryTextSubmit(String query) {
431                     View view = getCurrentFocus();
432                     if (view != null) {
433                         hideInputMethod(view);
434                         view.clearFocus();
435                     }
436                     return true;
437                 }
438 
439                 @Override
440                 public boolean onQueryTextChange(String newText) {
441                     // Show search result with non-empty text. Show a bare list otherwise.
442                     if (mSearchFragment != null) {
443                         mSearchFragment.setQueryString(newText, true);
444                     }
445                     return true;
446                 }
447     };
448 
449     /**
450      * Listener used to handle the "close" button on the right side of {@link SearchView}.
451      * If some text is in the search view, this will clean it up. Otherwise this will exit
452      * the search UI and let users go back to usual Phone UI.
453      *
454      * This does _not_ handle back button.
455      */
456     private final OnCloseListener mPhoneSearchCloseListener =
457             new OnCloseListener() {
458                 @Override
459                 public boolean onClose() {
460                     if (!TextUtils.isEmpty(mSearchView.getQuery())) {
461                         mSearchView.setQuery(null, true);
462                     }
463                     return true;
464                 }
465     };
466 
467     private final View.OnLayoutChangeListener mFirstLayoutListener
468             = new View.OnLayoutChangeListener() {
469         @Override
470         public void onLayoutChange(View v, int left, int top, int right, int bottom, int oldLeft,
471                 int oldTop, int oldRight, int oldBottom) {
472             v.removeOnLayoutChangeListener(this); // Unregister self.
473             addSearchFragment();
474         }
475     };
476 
477     @Override
onCreate(Bundle icicle)478     protected void onCreate(Bundle icicle) {
479         super.onCreate(icicle);
480 
481         final Intent intent = getIntent();
482         fixIntent(intent);
483 
484         setContentView(R.layout.dialtacts_activity);
485 
486         mContactListFilterController = ContactListFilterController.getInstance(this);
487         mContactListFilterController.addListener(mContactListFilterListener);
488 
489         findViewById(R.id.dialtacts_frame).addOnLayoutChangeListener(mFirstLayoutListener);
490 
491         mViewPager = (ViewPager) findViewById(R.id.pager);
492         mViewPager.setAdapter(new ViewPagerAdapter(getFragmentManager()));
493         mViewPager.setOnPageChangeListener(mPageChangeListener);
494         mViewPager.setOffscreenPageLimit(2);
495 
496         // Do same width calculation as ActionBar does
497         DisplayMetrics dm = getResources().getDisplayMetrics();
498         int minCellSize = getResources().getDimensionPixelSize(R.dimen.fake_menu_button_min_width);
499         int cellCount = dm.widthPixels / minCellSize;
500         int fakeMenuItemWidth = dm.widthPixels / cellCount;
501         if (DEBUG) Log.d(TAG, "The size of fake menu buttons (in pixel): " + fakeMenuItemWidth);
502 
503         // Soft menu button should appear only when there's no hardware menu button.
504         mMenuButton = findViewById(R.id.overflow_menu);
505         if (mMenuButton != null) {
506             mMenuButton.setMinimumWidth(fakeMenuItemWidth);
507             if (ViewConfiguration.get(this).hasPermanentMenuKey()) {
508                 // This is required for dialpad button's layout, so must not use GONE here.
509                 mMenuButton.setVisibility(View.INVISIBLE);
510             } else {
511                 mMenuButton.setOnClickListener(this);
512             }
513         }
514         mSearchButton = findViewById(R.id.searchButton);
515         if (mSearchButton != null) {
516             mSearchButton.setMinimumWidth(fakeMenuItemWidth);
517             mSearchButton.setOnClickListener(this);
518         }
519 
520         // Setup the ActionBar tabs (the order matches the tab-index contants TAB_INDEX_*)
521         setupDialer();
522         setupCallLog();
523         setupFavorites();
524         getActionBar().setNavigationMode(ActionBar.NAVIGATION_MODE_TABS);
525         getActionBar().setDisplayShowTitleEnabled(false);
526         getActionBar().setDisplayShowHomeEnabled(false);
527 
528         // Load the last manually loaded tab
529         mPrefs = PreferenceManager.getDefaultSharedPreferences(this);
530         mLastManuallySelectedFragment = mPrefs.getInt(PREF_LAST_MANUALLY_SELECTED_TAB,
531                 PREF_LAST_MANUALLY_SELECTED_TAB_DEFAULT);
532         if (mLastManuallySelectedFragment >= TAB_INDEX_COUNT) {
533             // Stored value may have exceeded the number of current tabs. Reset it.
534             mLastManuallySelectedFragment = PREF_LAST_MANUALLY_SELECTED_TAB_DEFAULT;
535         }
536 
537         setCurrentTab(intent);
538 
539         if (UI.FILTER_CONTACTS_ACTION.equals(intent.getAction())
540                 && icicle == null) {
541             setupFilterText(intent);
542         }
543     }
544 
545     @Override
onStart()546     public void onStart() {
547         super.onStart();
548         if (mPhoneFavoriteFragment != null) {
549             mPhoneFavoriteFragment.setFilter(mContactListFilterController.getFilter());
550         }
551         if (mSearchFragment != null) {
552             mSearchFragment.setFilter(mContactListFilterController.getFilter());
553         }
554 
555         if (mDuringSwipe || mUserTabClick) {
556             if (DEBUG) Log.d(TAG, "reset buggy flag state..");
557             mDuringSwipe = false;
558             mUserTabClick = false;
559         }
560 
561         final int currentPosition = mPageChangeListener.getCurrentPosition();
562         if (DEBUG) {
563             Log.d(TAG, "onStart(). current position: " + mPageChangeListener.getCurrentPosition()
564                     + ". Reset all menu visibility state.");
565         }
566         updateFakeMenuButtonsVisibility(currentPosition == TAB_INDEX_DIALER && !mInSearchUi);
567         for (int i = 0; i < TAB_INDEX_COUNT; i++) {
568             sendFragmentVisibilityChange(i, i == currentPosition);
569         }
570     }
571 
572     @Override
onDestroy()573     public void onDestroy() {
574         super.onDestroy();
575         mContactListFilterController.removeListener(mContactListFilterListener);
576     }
577 
578     @Override
onClick(View view)579     public void onClick(View view) {
580         switch (view.getId()) {
581             case R.id.searchButton: {
582                 enterSearchUi();
583                 break;
584             }
585             case R.id.overflow_menu: {
586                 if (mDialpadFragment != null) {
587                     PopupMenu popup = mDialpadFragment.constructPopupMenu(view);
588                     if (popup != null) {
589                         popup.show();
590                     }
591                 } else {
592                     Log.w(TAG, "DialpadFragment is null during onClick() event for " + view);
593                 }
594                 break;
595             }
596             default: {
597                 Log.wtf(TAG, "Unexpected onClick event from " + view);
598                 break;
599             }
600         }
601     }
602 
603     /**
604      * Add search fragment.  Note this is called during onLayout, so there's some restrictions,
605      * such as executePendingTransaction can't be used in it.
606      */
addSearchFragment()607     private void addSearchFragment() {
608         // In order to take full advantage of "fragment deferred start", we need to create the
609         // search fragment after all other fragments are created.
610         // The other fragments are created by the ViewPager on the first onMeasure().
611         // We use the first onLayout call, which is after onMeasure().
612 
613         // Just return if the fragment is already created, which happens after configuration
614         // changes.
615         if (mSearchFragment != null) return;
616 
617         final FragmentTransaction ft = getFragmentManager().beginTransaction();
618         final Fragment searchFragment = new PhoneNumberPickerFragment();
619 
620         searchFragment.setUserVisibleHint(false);
621         ft.add(R.id.dialtacts_frame, searchFragment);
622         ft.hide(searchFragment);
623         ft.commitAllowingStateLoss();
624     }
625 
prepareSearchView()626     private void prepareSearchView() {
627         final View searchViewLayout =
628                 getLayoutInflater().inflate(R.layout.dialtacts_custom_action_bar, null);
629         mSearchView = (SearchView) searchViewLayout.findViewById(R.id.search_view);
630         mSearchView.setOnQueryTextListener(mPhoneSearchQueryTextListener);
631         mSearchView.setOnCloseListener(mPhoneSearchCloseListener);
632         // Since we're using a custom layout for showing SearchView instead of letting the
633         // search menu icon do that job, we need to manually configure the View so it looks
634         // "shown via search menu".
635         // - it should be iconified by default
636         // - it should not be iconified at this time
637         // See also comments for onActionViewExpanded()/onActionViewCollapsed()
638         mSearchView.setIconifiedByDefault(true);
639         mSearchView.setQueryHint(getString(R.string.hint_findContacts));
640         mSearchView.setIconified(false);
641         mSearchView.setOnQueryTextFocusChangeListener(new OnFocusChangeListener() {
642             @Override
643             public void onFocusChange(View view, boolean hasFocus) {
644                 if (hasFocus) {
645                     showInputMethod(view.findFocus());
646                 }
647             }
648         });
649 
650         if (!ViewConfiguration.get(this).hasPermanentMenuKey()) {
651             // Filter option menu should be shown on the right side of SearchView.
652             final View filterOptionView = searchViewLayout.findViewById(R.id.search_option);
653             filterOptionView.setVisibility(View.VISIBLE);
654             filterOptionView.setOnClickListener(mFilterOptionClickListener);
655         }
656 
657         getActionBar().setCustomView(searchViewLayout,
658                 new LayoutParams(LayoutParams.MATCH_PARENT, LayoutParams.WRAP_CONTENT));
659     }
660 
661     @Override
onAttachFragment(Fragment fragment)662     public void onAttachFragment(Fragment fragment) {
663         // This method can be called before onCreate(), at which point we cannot rely on ViewPager.
664         // In that case, we will setup the "current position" soon after the ViewPager is ready.
665         final int currentPosition = mViewPager != null ? mViewPager.getCurrentItem() : -1;
666 
667         if (fragment instanceof DialpadFragment) {
668             mDialpadFragment = (DialpadFragment) fragment;
669         } else if (fragment instanceof CallLogFragment) {
670             mCallLogFragment = (CallLogFragment) fragment;
671         } else if (fragment instanceof PhoneFavoriteFragment) {
672             mPhoneFavoriteFragment = (PhoneFavoriteFragment) fragment;
673             mPhoneFavoriteFragment.setListener(mPhoneFavoriteListener);
674             if (mContactListFilterController != null
675                     && mContactListFilterController.getFilter() != null) {
676                 mPhoneFavoriteFragment.setFilter(mContactListFilterController.getFilter());
677             }
678         } else if (fragment instanceof PhoneNumberPickerFragment) {
679             mSearchFragment = (PhoneNumberPickerFragment) fragment;
680             mSearchFragment.setOnPhoneNumberPickerActionListener(mPhoneNumberPickerActionListener);
681             mSearchFragment.setQuickContactEnabled(true);
682             mSearchFragment.setDarkTheme(true);
683             mSearchFragment.setPhotoPosition(ContactListItemView.getDefaultPhotoPosition(
684                     true /* opposite */));
685             mSearchFragment.setUseCallableUri(true);
686             if (mContactListFilterController != null
687                     && mContactListFilterController.getFilter() != null) {
688                 mSearchFragment.setFilter(mContactListFilterController.getFilter());
689             }
690             // Here we assume that we're not on the search mode, so let's hide the fragment.
691             //
692             // We get here either when the fragment is created (normal case), or after configuration
693             // changes.  In the former case, we're not in search mode because we can only
694             // enter search mode if the fragment is created.  (see enterSearchUi())
695             // In the latter case we're not in search mode either because we don't retain
696             // mInSearchUi -- ideally we should but at this point it's not supported.
697             mSearchFragment.setUserVisibleHint(false);
698             // After configuration changes fragments will forget their "hidden" state, so make
699             // sure to hide it.
700             if (!mSearchFragment.isHidden()) {
701                 final FragmentTransaction transaction = getFragmentManager().beginTransaction();
702                 transaction.hide(mSearchFragment);
703                 transaction.commitAllowingStateLoss();
704             }
705         }
706     }
707 
708     @Override
onPause()709     protected void onPause() {
710         super.onPause();
711 
712         mPrefs.edit().putInt(PREF_LAST_MANUALLY_SELECTED_TAB, mLastManuallySelectedFragment)
713                 .apply();
714     }
715 
fixIntent(Intent intent)716     private void fixIntent(Intent intent) {
717         // This should be cleaned up: the call key used to send an Intent
718         // that just said to go to the recent calls list.  It now sends this
719         // abstract action, but this class hasn't been rewritten to deal with it.
720         if (Intent.ACTION_CALL_BUTTON.equals(intent.getAction())) {
721             intent.setDataAndType(Calls.CONTENT_URI, Calls.CONTENT_TYPE);
722             intent.putExtra("call_key", true);
723             setIntent(intent);
724         }
725     }
726 
setupDialer()727     private void setupDialer() {
728         final Tab tab = getActionBar().newTab();
729         tab.setContentDescription(R.string.dialerIconLabel);
730         tab.setTabListener(mTabListener);
731         tab.setIcon(R.drawable.ic_tab_dialer);
732         getActionBar().addTab(tab);
733     }
734 
setupCallLog()735     private void setupCallLog() {
736         final Tab tab = getActionBar().newTab();
737         tab.setContentDescription(R.string.recentCallsIconLabel);
738         tab.setIcon(R.drawable.ic_tab_recent);
739         tab.setTabListener(mTabListener);
740         getActionBar().addTab(tab);
741     }
742 
setupFavorites()743     private void setupFavorites() {
744         final Tab tab = getActionBar().newTab();
745         tab.setContentDescription(R.string.contactsFavoritesLabel);
746         tab.setIcon(R.drawable.ic_tab_all);
747         tab.setTabListener(mTabListener);
748         getActionBar().addTab(tab);
749     }
750 
751     /**
752      * Returns true if the intent is due to hitting the green send key (hardware call button:
753      * KEYCODE_CALL) while in a call.
754      *
755      * @param intent the intent that launched this activity
756      * @param recentCallsRequest true if the intent is requesting to view recent calls
757      * @return true if the intent is due to hitting the green send key while in a call
758      */
isSendKeyWhileInCall(final Intent intent, final boolean recentCallsRequest)759     private boolean isSendKeyWhileInCall(final Intent intent,
760             final boolean recentCallsRequest) {
761         // If there is a call in progress go to the call screen
762         if (recentCallsRequest) {
763             final boolean callKey = intent.getBooleanExtra("call_key", false);
764 
765             try {
766                 ITelephony phone = ITelephony.Stub.asInterface(ServiceManager.checkService("phone"));
767                 if (callKey && phone != null && phone.showCallScreen()) {
768                     return true;
769                 }
770             } catch (RemoteException e) {
771                 Log.e(TAG, "Failed to handle send while in call", e);
772             }
773         }
774 
775         return false;
776     }
777 
778     /**
779      * Sets the current tab based on the intent's request type
780      *
781      * @param intent Intent that contains information about which tab should be selected
782      */
setCurrentTab(Intent intent)783     private void setCurrentTab(Intent intent) {
784         // If we got here by hitting send and we're in call forward along to the in-call activity
785         boolean recentCallsRequest = Calls.CONTENT_TYPE.equals(intent.resolveType(
786             getContentResolver()));
787         if (isSendKeyWhileInCall(intent, recentCallsRequest)) {
788             finish();
789             return;
790         }
791 
792         // Remember the old manually selected tab index so that it can be restored if it is
793         // overwritten by one of the programmatic tab selections
794         final int savedTabIndex = mLastManuallySelectedFragment;
795 
796         final int tabIndex;
797         if (DialpadFragment.phoneIsInUse() || isDialIntent(intent)) {
798             tabIndex = TAB_INDEX_DIALER;
799         } else if (recentCallsRequest) {
800             tabIndex = TAB_INDEX_CALL_LOG;
801         } else {
802             tabIndex = mLastManuallySelectedFragment;
803         }
804 
805         final int previousItemIndex = mViewPager.getCurrentItem();
806         mViewPager.setCurrentItem(tabIndex, false /* smoothScroll */);
807         if (previousItemIndex != tabIndex) {
808             sendFragmentVisibilityChange(previousItemIndex, false /* not visible */ );
809         }
810         mPageChangeListener.setCurrentPosition(tabIndex);
811         sendFragmentVisibilityChange(tabIndex, true /* visible */ );
812 
813         // Restore to the previous manual selection
814         mLastManuallySelectedFragment = savedTabIndex;
815         mDuringSwipe = false;
816         mUserTabClick = false;
817     }
818 
819     @Override
onNewIntent(Intent newIntent)820     public void onNewIntent(Intent newIntent) {
821         setIntent(newIntent);
822         fixIntent(newIntent);
823         setCurrentTab(newIntent);
824         final String action = newIntent.getAction();
825         if (UI.FILTER_CONTACTS_ACTION.equals(action)) {
826             setupFilterText(newIntent);
827         }
828         if (mInSearchUi || (mSearchFragment != null && mSearchFragment.isVisible())) {
829             exitSearchUi();
830         }
831 
832         if (mViewPager.getCurrentItem() == TAB_INDEX_DIALER) {
833             if (mDialpadFragment != null) {
834                 mDialpadFragment.setStartedFromNewIntent(true);
835             } else {
836                 Log.e(TAG, "DialpadFragment isn't ready yet when the tab is already selected.");
837             }
838         } else if (mViewPager.getCurrentItem() == TAB_INDEX_CALL_LOG) {
839             if (mCallLogFragment != null) {
840                 mCallLogFragment.configureScreenFromIntent(newIntent);
841             } else {
842                 Log.e(TAG, "CallLogFragment isn't ready yet when the tab is already selected.");
843             }
844         }
845         invalidateOptionsMenu();
846     }
847 
848     /** Returns true if the given intent contains a phone number to populate the dialer with */
isDialIntent(Intent intent)849     private boolean isDialIntent(Intent intent) {
850         final String action = intent.getAction();
851         if (Intent.ACTION_DIAL.equals(action) || ACTION_TOUCH_DIALER.equals(action)) {
852             return true;
853         }
854         if (Intent.ACTION_VIEW.equals(action)) {
855             final Uri data = intent.getData();
856             if (data != null && CallUtil.SCHEME_TEL.equals(data.getScheme())) {
857                 return true;
858             }
859         }
860         return false;
861     }
862 
863     /**
864      * Returns an appropriate call origin for this Activity. May return null when no call origin
865      * should be used (e.g. when some 3rd party application launched the screen. Call origin is
866      * for remembering the tab in which the user made a phone call, so the external app's DIAL
867      * request should not be counted.)
868      */
getCallOrigin()869     public String getCallOrigin() {
870         return !isDialIntent(getIntent()) ? CALL_ORIGIN_DIALTACTS : null;
871     }
872 
873     /**
874      * Retrieves the filter text stored in {@link #setupFilterText(Intent)}.
875      * This text originally came from a FILTER_CONTACTS_ACTION intent received
876      * by this activity. The stored text will then be cleared after after this
877      * method returns.
878      *
879      * @return The stored filter text
880      */
getAndClearFilterText()881     public String getAndClearFilterText() {
882         String filterText = mFilterText;
883         mFilterText = null;
884         return filterText;
885     }
886 
887     /**
888      * Stores the filter text associated with a FILTER_CONTACTS_ACTION intent.
889      * This is so child activities can check if they are supposed to display a filter.
890      *
891      * @param intent The intent received in {@link #onNewIntent(Intent)}
892      */
setupFilterText(Intent intent)893     private void setupFilterText(Intent intent) {
894         // If the intent was relaunched from history, don't apply the filter text.
895         if ((intent.getFlags() & Intent.FLAG_ACTIVITY_LAUNCHED_FROM_HISTORY) != 0) {
896             return;
897         }
898         String filter = intent.getStringExtra(UI.FILTER_TEXT_EXTRA_KEY);
899         if (filter != null && filter.length() > 0) {
900             mFilterText = filter;
901         }
902     }
903 
904     @Override
onBackPressed()905     public void onBackPressed() {
906         if (mInSearchUi) {
907             // We should let the user go back to usual screens with tabs.
908             exitSearchUi();
909         } else if (isTaskRoot()) {
910             // Instead of stopping, simply push this to the back of the stack.
911             // This is only done when running at the top of the stack;
912             // otherwise, we have been launched by someone else so need to
913             // allow the user to go back to the caller.
914             moveTaskToBack(false);
915         } else {
916             super.onBackPressed();
917         }
918     }
919 
920     private final PhoneFavoriteFragment.Listener mPhoneFavoriteListener =
921             new PhoneFavoriteFragment.Listener() {
922         @Override
923         public void onContactSelected(Uri contactUri) {
924             PhoneNumberInteraction.startInteractionForPhoneCall(
925                     DialtactsActivity.this, contactUri, getCallOrigin());
926         }
927 
928         @Override
929         public void onCallNumberDirectly(String phoneNumber) {
930             Intent intent = CallUtil.getCallIntent(phoneNumber, getCallOrigin());
931             startActivity(intent);
932         }
933     };
934 
935     @Override
onCreateOptionsMenu(Menu menu)936     public boolean onCreateOptionsMenu(Menu menu) {
937         MenuInflater inflater = getMenuInflater();
938         inflater.inflate(R.menu.dialtacts_options, menu);
939 
940         // set up intents and onClick listeners
941         final MenuItem callSettingsMenuItem = menu.findItem(R.id.menu_call_settings);
942         final MenuItem searchMenuItem = menu.findItem(R.id.search_on_action_bar);
943         final MenuItem filterOptionMenuItem = menu.findItem(R.id.filter_option);
944 
945         callSettingsMenuItem.setIntent(DialtactsActivity.getCallSettingsIntent());
946         searchMenuItem.setOnMenuItemClickListener(mSearchMenuItemClickListener);
947         filterOptionMenuItem.setOnMenuItemClickListener(mFilterOptionsMenuItemClickListener);
948 
949         return true;
950     }
951 
952     @Override
onPrepareOptionsMenu(Menu menu)953     public boolean onPrepareOptionsMenu(Menu menu) {
954         if (mInSearchUi) {
955             prepareOptionsMenuInSearchMode(menu);
956         } else {
957             // get reference to the currently selected tab
958             final Tab tab = getActionBar().getSelectedTab();
959             if (tab != null) {
960                 switch(tab.getPosition()) {
961                     case TAB_INDEX_DIALER:
962                         prepareOptionsMenuForDialerTab(menu);
963                         break;
964                     case TAB_INDEX_CALL_LOG:
965                         prepareOptionsMenuForCallLogTab(menu);
966                         break;
967                     case TAB_INDEX_FAVORITES:
968                         prepareOptionsMenuForFavoritesTab(menu);
969                         break;
970                 }
971             }
972         }
973         return true;
974     }
975 
976     @Override
onOptionsItemSelected(MenuItem item)977     public boolean onOptionsItemSelected(MenuItem item) {
978         switch (item.getItemId()) {
979             case R.id.add_contact:
980                 try {
981                     startActivity(new Intent(Intent.ACTION_INSERT, Contacts.CONTENT_URI));
982                 } catch (ActivityNotFoundException e) {
983                     Toast toast = Toast.makeText(this, R.string.add_contact_not_available,
984                             Toast.LENGTH_SHORT);
985                     toast.show();
986                 }
987                 return true;
988         }
989         return super.onOptionsItemSelected(item);
990     }
991 
prepareOptionsMenuInSearchMode(Menu menu)992     private void prepareOptionsMenuInSearchMode(Menu menu) {
993         // get references to menu items
994         final MenuItem searchMenuItem = menu.findItem(R.id.search_on_action_bar);
995         final MenuItem filterOptionMenuItem = menu.findItem(R.id.filter_option);
996         final MenuItem addContactOptionMenuItem = menu.findItem(R.id.add_contact);
997         final MenuItem callSettingsMenuItem = menu.findItem(R.id.menu_call_settings);
998         final MenuItem emptyRightMenuItem = menu.findItem(R.id.empty_right_menu_item);
999 
1000         // prepare the menu items
1001         searchMenuItem.setVisible(false);
1002         filterOptionMenuItem.setVisible(ViewConfiguration.get(this).hasPermanentMenuKey());
1003         addContactOptionMenuItem.setVisible(false);
1004         callSettingsMenuItem.setVisible(false);
1005         emptyRightMenuItem.setVisible(false);
1006     }
1007 
prepareOptionsMenuForDialerTab(Menu menu)1008     private void prepareOptionsMenuForDialerTab(Menu menu) {
1009         if (DEBUG) {
1010             Log.d(TAG, "onPrepareOptionsMenu(dialer). swipe: " + mDuringSwipe
1011                     + ", user tab click: " + mUserTabClick);
1012         }
1013 
1014         // get references to menu items
1015         final MenuItem searchMenuItem = menu.findItem(R.id.search_on_action_bar);
1016         final MenuItem filterOptionMenuItem = menu.findItem(R.id.filter_option);
1017         final MenuItem addContactOptionMenuItem = menu.findItem(R.id.add_contact);
1018         final MenuItem callSettingsMenuItem = menu.findItem(R.id.menu_call_settings);
1019         final MenuItem emptyRightMenuItem = menu.findItem(R.id.empty_right_menu_item);
1020 
1021         // prepare the menu items
1022         filterOptionMenuItem.setVisible(false);
1023         addContactOptionMenuItem.setVisible(false);
1024         if (mDuringSwipe || mUserTabClick) {
1025             // During horizontal movement, the real ActionBar menu items are shown
1026             searchMenuItem.setVisible(true);
1027             callSettingsMenuItem.setVisible(true);
1028             // When there is a permanent menu key, there is no overflow icon on the right of
1029             // the action bar which would force the search menu item (if it is visible) to the
1030             // left.  This is the purpose of showing the emptyRightMenuItem.
1031             emptyRightMenuItem.setVisible(ViewConfiguration.get(this).hasPermanentMenuKey());
1032         } else {
1033             // This is when the user is looking at the dialer pad.  In this case, the real
1034             // ActionBar is hidden and fake menu items are shown.
1035             // Except in landscape, in which case the real search menu item is shown.
1036             searchMenuItem.setVisible(OrientationUtil.isLandscape(this));
1037             // If a permanent menu key is available, then we need to show the call settings item
1038             // so that the call settings item can be invoked by the permanent menu key.
1039             callSettingsMenuItem.setVisible(ViewConfiguration.get(this).hasPermanentMenuKey());
1040             emptyRightMenuItem.setVisible(false);
1041         }
1042     }
1043 
prepareOptionsMenuForCallLogTab(Menu menu)1044     private void prepareOptionsMenuForCallLogTab(Menu menu) {
1045         // get references to menu items
1046         final MenuItem searchMenuItem = menu.findItem(R.id.search_on_action_bar);
1047         final MenuItem filterOptionMenuItem = menu.findItem(R.id.filter_option);
1048         final MenuItem addContactOptionMenuItem = menu.findItem(R.id.add_contact);
1049         final MenuItem callSettingsMenuItem = menu.findItem(R.id.menu_call_settings);
1050         final MenuItem emptyRightMenuItem = menu.findItem(R.id.empty_right_menu_item);
1051 
1052         // prepare the menu items
1053         searchMenuItem.setVisible(true);
1054         filterOptionMenuItem.setVisible(false);
1055         addContactOptionMenuItem.setVisible(false);
1056         callSettingsMenuItem.setVisible(true);
1057         emptyRightMenuItem.setVisible(ViewConfiguration.get(this).hasPermanentMenuKey());
1058     }
1059 
prepareOptionsMenuForFavoritesTab(Menu menu)1060     private void prepareOptionsMenuForFavoritesTab(Menu menu) {
1061         // get references to menu items
1062         final MenuItem searchMenuItem = menu.findItem(R.id.search_on_action_bar);
1063         final MenuItem filterOptionMenuItem = menu.findItem(R.id.filter_option);
1064         final MenuItem addContactOptionMenuItem = menu.findItem(R.id.add_contact);
1065         final MenuItem callSettingsMenuItem = menu.findItem(R.id.menu_call_settings);
1066         final MenuItem emptyRightMenuItem = menu.findItem(R.id.empty_right_menu_item);
1067 
1068         // prepare the menu items
1069         searchMenuItem.setVisible(true);
1070         filterOptionMenuItem.setVisible(true);
1071         addContactOptionMenuItem.setVisible(true);
1072         callSettingsMenuItem.setVisible(true);
1073         emptyRightMenuItem.setVisible(false);
1074     }
1075 
1076     @Override
startSearch(String initialQuery, boolean selectInitialQuery, Bundle appSearchData, boolean globalSearch)1077     public void startSearch(String initialQuery, boolean selectInitialQuery,
1078             Bundle appSearchData, boolean globalSearch) {
1079         if (mSearchFragment != null && mSearchFragment.isAdded() && !globalSearch) {
1080             if (mInSearchUi) {
1081                 if (mSearchView.hasFocus()) {
1082                     showInputMethod(mSearchView.findFocus());
1083                 } else {
1084                     mSearchView.requestFocus();
1085                 }
1086             } else {
1087                 enterSearchUi();
1088             }
1089         } else {
1090             super.startSearch(initialQuery, selectInitialQuery, appSearchData, globalSearch);
1091         }
1092     }
1093 
1094     /**
1095      * Hides every tab and shows search UI for phone lookup.
1096      */
enterSearchUi()1097     private void enterSearchUi() {
1098         if (mSearchFragment == null) {
1099             // We add the search fragment dynamically in the first onLayoutChange() and
1100             // mSearchFragment is set sometime later when the fragment transaction is actually
1101             // executed, which means there's a window when users are able to hit the (physical)
1102             // search key but mSearchFragment is still null.
1103             // It's quite hard to handle this case right, so let's just ignore the search key
1104             // in this case.  Users can just hit it again and it will work this time.
1105             return;
1106         }
1107         if (mSearchView == null) {
1108             prepareSearchView();
1109         }
1110 
1111         final ActionBar actionBar = getActionBar();
1112 
1113         final Tab tab = actionBar.getSelectedTab();
1114 
1115         // User can search during the call, but we don't want to remember the status.
1116         if (tab != null && !DialpadFragment.phoneIsInUse()) {
1117             mLastManuallySelectedFragment = tab.getPosition();
1118         }
1119 
1120         mSearchView.setQuery(null, true);
1121 
1122         actionBar.setDisplayShowCustomEnabled(true);
1123         actionBar.setNavigationMode(ActionBar.NAVIGATION_MODE_STANDARD);
1124         actionBar.setDisplayShowHomeEnabled(true);
1125         actionBar.setDisplayHomeAsUpEnabled(true);
1126 
1127         updateFakeMenuButtonsVisibility(false);
1128 
1129         for (int i = 0; i < TAB_INDEX_COUNT; i++) {
1130             sendFragmentVisibilityChange(i, false /* not visible */ );
1131         }
1132 
1133         // Show the search fragment and hide everything else.
1134         mSearchFragment.setUserVisibleHint(true);
1135         final FragmentTransaction transaction = getFragmentManager().beginTransaction();
1136         transaction.show(mSearchFragment);
1137         transaction.commitAllowingStateLoss();
1138         mViewPager.setVisibility(View.GONE);
1139 
1140         // We need to call this and onActionViewCollapsed() manually, since we are using a custom
1141         // layout instead of asking the search menu item to take care of SearchView.
1142         mSearchView.onActionViewExpanded();
1143         mInSearchUi = true;
1144     }
1145 
showInputMethod(View view)1146     private void showInputMethod(View view) {
1147         InputMethodManager imm = (InputMethodManager)getSystemService(Context.INPUT_METHOD_SERVICE);
1148         if (imm != null) {
1149             if (!imm.showSoftInput(view, 0)) {
1150                 Log.w(TAG, "Failed to show soft input method.");
1151             }
1152         }
1153     }
1154 
hideInputMethod(View view)1155     private void hideInputMethod(View view) {
1156         InputMethodManager imm = (InputMethodManager)getSystemService(Context.INPUT_METHOD_SERVICE);
1157         if (imm != null && view != null) {
1158             imm.hideSoftInputFromWindow(view.getWindowToken(), 0);
1159         }
1160     }
1161 
1162     /**
1163      * Goes back to usual Phone UI with tags. Previously selected Tag and associated Fragment
1164      * should be automatically focused again.
1165      */
exitSearchUi()1166     private void exitSearchUi() {
1167         final ActionBar actionBar = getActionBar();
1168 
1169         // Hide the search fragment, if exists.
1170         if (mSearchFragment != null) {
1171             mSearchFragment.setUserVisibleHint(false);
1172 
1173             final FragmentTransaction transaction = getFragmentManager().beginTransaction();
1174             transaction.hide(mSearchFragment);
1175             transaction.commitAllowingStateLoss();
1176         }
1177 
1178         // We want to hide SearchView and show Tabs. Also focus on previously selected one.
1179         actionBar.setDisplayShowCustomEnabled(false);
1180         actionBar.setDisplayShowHomeEnabled(false);
1181         actionBar.setNavigationMode(ActionBar.NAVIGATION_MODE_TABS);
1182 
1183         for (int i = 0; i < TAB_INDEX_COUNT; i++) {
1184             sendFragmentVisibilityChange(i, i == mViewPager.getCurrentItem());
1185         }
1186 
1187         // Before exiting the search screen, reset swipe state.
1188         mDuringSwipe = false;
1189         mUserTabClick = false;
1190 
1191         mViewPager.setVisibility(View.VISIBLE);
1192 
1193         hideInputMethod(getCurrentFocus());
1194 
1195         // Request to update option menu.
1196         invalidateOptionsMenu();
1197 
1198         // See comments in onActionViewExpanded()
1199         mSearchView.onActionViewCollapsed();
1200         mInSearchUi = false;
1201     }
1202 
getFragmentAt(int position)1203     private Fragment getFragmentAt(int position) {
1204         switch (position) {
1205             case TAB_INDEX_DIALER:
1206                 return mDialpadFragment;
1207             case TAB_INDEX_CALL_LOG:
1208                 return mCallLogFragment;
1209             case TAB_INDEX_FAVORITES:
1210                 return mPhoneFavoriteFragment;
1211             default:
1212                 throw new IllegalStateException("Unknown fragment index: " + position);
1213         }
1214     }
1215 
sendFragmentVisibilityChange(int position, boolean visibility)1216     private void sendFragmentVisibilityChange(int position, boolean visibility) {
1217         if (DEBUG) {
1218             Log.d(TAG, "sendFragmentVisibiltyChange(). position: " + position
1219                     + ", visibility: " + visibility);
1220         }
1221         // Position can be -1 initially. See PageChangeListener.
1222         if (position >= 0) {
1223             final Fragment fragment = getFragmentAt(position);
1224             if (fragment != null) {
1225                 fragment.setMenuVisibility(visibility);
1226                 fragment.setUserVisibleHint(visibility);
1227             }
1228         }
1229     }
1230 
1231     /**
1232      * Update visibility of the search button and menu button at the bottom.
1233      * They should be invisible when bottom ActionBar's real items are available, and be visible
1234      * otherwise.
1235      *
1236      * @param visible True when visible.
1237      */
updateFakeMenuButtonsVisibility(boolean visible)1238     private void updateFakeMenuButtonsVisibility(boolean visible) {
1239         // Note: Landscape mode does not have the fake menu and search buttons.
1240         if (DEBUG) {
1241             Log.d(TAG, "updateFakeMenuButtonVisibility(" + visible + ")");
1242         }
1243 
1244         if (mSearchButton != null) {
1245             if (visible) {
1246                 mSearchButton.setVisibility(View.VISIBLE);
1247             } else {
1248                 mSearchButton.setVisibility(View.INVISIBLE);
1249             }
1250         }
1251         if (mMenuButton != null) {
1252             if (visible && !ViewConfiguration.get(this).hasPermanentMenuKey()) {
1253                 mMenuButton.setVisibility(View.VISIBLE);
1254             } else {
1255                 mMenuButton.setVisibility(View.INVISIBLE);
1256             }
1257         }
1258     }
1259 
1260     /** Returns an Intent to launch Call Settings screen */
getCallSettingsIntent()1261     public static Intent getCallSettingsIntent() {
1262         final Intent intent = new Intent(Intent.ACTION_MAIN);
1263         intent.setClassName(PHONE_PACKAGE, CALL_SETTINGS_CLASS_NAME);
1264         intent.setFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP);
1265         return intent;
1266     }
1267 
1268     @Override
onActivityResult(int requestCode, int resultCode, Intent data)1269     protected void onActivityResult(int requestCode, int resultCode, Intent data) {
1270         if (resultCode != Activity.RESULT_OK) {
1271             return;
1272         }
1273         switch (requestCode) {
1274             case SUBACTIVITY_ACCOUNT_FILTER: {
1275                 AccountFilterUtil.handleAccountFilterResult(
1276                         mContactListFilterController, resultCode, data);
1277             }
1278             break;
1279         }
1280     }
1281 }
1282