一、Contacts应用的主界面和联系人详情界面增加顶部菜单添加退出按钮
通过Hierarchy View 工具可以发现
主界面对应的类为 PeopleActivity
联系人详情界面对应的类为 QuickContactActivity
左上角的退出按钮其实很简单,系统actionBar已经帮我们实现了这一功能,只是没有显示出来而已。在onCreate()方法中,在setContentView()方法之后,添加如下代码即可显示返回的箭头
ActionBar mActionBar = getActionBar();
if (mActionBar != null) {
Log.i(TAG, "getSupportActionBar != null....");
mActionBar.setDisplayHomeAsUpEnabled(true);
mActionBar.setHomeButtonEnabled(true);
}
接下来在onOptionsItemSelected()中监听返回按钮的事件即可
@Override
public boolean onOptionsItemSelected(MenuItem item) {
switch (item.getItemId()) {
case android.R.id.home: {
finish();
}
return true;
}
....
}
图1 左上角返回退出功能
二、第三方app拉起主界面时直接显示模糊查询对应的联系人列表
简单分析一下,模糊查询需要对应的查询联系人名称,可以通过intent传递参数,这里定义为String类型,当传递参数不为null时,模拟手动点击搜索框对应的逻辑。如下在 PeopleActivity 的 onCreate()方法中增加获取参数的代码
final String queryString = getIntent().getStringExtra("queryString");
if (!TextUtils.isEmpty(queryString)) {
new Handler().postDelayed(new Runnable() {
@Override
public void run() {
showQueryTextFragment(queryString);
}
}, 100);//让搜索逻辑延迟100ms执行
}
通过测试发现,不加延迟触发搜索框对应的逻辑并不会显示模糊查询结果界面。接下来我们分析点击搜索框对应的逻辑代码,找到搜索框对应的控件id,menu_search, 回到刚刚的菜单监听方法 onOptionsItemSelected()中
@Override
public boolean onOptionsItemSelected(MenuItem item) {
...
switch (item.getItemId()) {
case R.id.menu_search: {
onSearchRequested();
return true;
}
}
...
}
@Override
public boolean onSearchRequested() { // Search key pressed.
Log.d(TAG, "[onSearchRequested]");
//不在搜索模式下,也就是没有点击过搜索框
if (!mActionBarAdapter.isSelectionMode()) {
//获取焦点,弹出键盘
mActionBarAdapter.setSearchMode(true);
}
return true;
}
从上面不难看出最终调用 mActionBarAdapter 的方法,我们接着跟进去
源码位置 packages/apps/Contacts/src/com/android/contacts/activities/ActionBarAdapter.java
public void setSearchMode(boolean flag) {
if (mSearchMode != flag) {
mSearchMode = flag;
update(false /* skipAnimation */);
if (mSearchView == null) {
return;
}
if (mSearchMode) {
mSearchView.setEnabled(true);
setFocusOnSearchView();
} else {
// Disable search view, so that it doesn't keep the IME visible.
mSearchView.setEnabled(false);
}
setQueryString(null);
} else if (flag) {
// Everything is already set up. Still make sure the keyboard is up
//需要注释此处,不然多次调用并退出再次拉起容易出现键盘弹出的情况
//if (mSearchView != null) setFocusOnSearchView();
}
}
public void setFocusOnSearchView() {
//mSearchView获取焦点(先获取焦点才能弹出键盘)
mSearchView.requestFocus();
//弹出键盘
showInputMethod(mSearchView); // Workaround for the "IME not popping up" issue.
}
private void showInputMethod(View view) {
final InputMethodManager imm = (InputMethodManager) mActivity.getSystemService(
Context.INPUT_METHOD_SERVICE);
if (imm != null) {
imm.showSoftInput(view, 0);
}
}
看到这里我们可以猜想到 mSearchView 肯定设置了文字改变监听,继续查找 addTextChangedListener
...
mSearchView.setInputType(EditorInfo.TYPE_CLASS_TEXT
| EditorInfo.TYPE_TEXT_VARIATION_EMAIL_ADDRESS);
mSearchView.addTextChangedListener(new SearchTextWatcher());
...
private class SearchTextWatcher implements TextWatcher {
@Override
public void onTextChanged(CharSequence queryString, int start, int before, int count) {
if (queryString.equals(mQueryString)) {
return;
}
//当前输入的模糊查询的名称
mQueryString = queryString.toString();
if (!mSearchMode) {
if (!