Android6.0 源码修改之 Contacts应用
一、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 (!TextUtils.isEmpty(queryString)) {
                setSearchMode(true);
            }
        } else if (mListener != null) {
			//回调通知 PeopleActivity 改变界面
            mListener.onAction(Action.CHANGE_SEARCH_QUERY);
        }
        mClearSearchView.setVisibility(
                TextUtils.isEmpty(queryString) ? View.GONE : View.VISIBLE);
    }
    @Override
    public void afterTextChanged(Editable s) {}
    @Override
    public void beforeTextChanged(CharSequence s, int start, int count, int after) {}
}
回到 PeopleActivity 中找到监听 Action.CHANGE_SEARCH_QUERY 的代码如下
 @Override
public void onAction(int action) {
    Log.d(TAG,"[onAction]action = " + action);
    /// M: [vcs] @{
    if (mVcsController != null) {
        mVcsController.onActionVcs(action);
    }
    /// @}
    switch (action) {
	...
	case ActionBarAdapter.Listener.Action.CHANGE_SEARCH_QUERY:
			//获取当前输入的模糊查询姓名
            final String queryString = mActionBarAdapter.getQueryString();
			//显示对应的fragment
            setQueryTextToFragment(queryString);
            updateDebugOptionsVisibility(
                    ENABLE_DEBUG_OPTIONS_HIDDEN_CODE.equals(queryString));
            break;
        default:
            throw new IllegalStateException("Unkonwn ActionBarAdapter action: " + action);
    }
}
到此,搜索框模糊查询对应的逻辑就分析完了,那么我们就模拟调用对应的逻辑就ok了,再来把整体流程捋一遍,
点击搜索框->获取焦点->弹出键盘->输入姓名->收到文字内容改变的监听->将输入的内容回调给 PeopleActivity->收到回调显示对应的结果Fragment
好了,通过调用EditText.setText()方法也能触发文字内容改变的监听,前提是要先获取焦点,那么我们的 showQueryTextFragment() 实现如下
private void showQueryTextFragment(String queryString){
    Log.d(TAG, "[showQueryTextFragment]");
    if (!mActionBarAdapter.isSelectionMode()) {
        Log.e(TAG, "[queryString==]"+queryString);
        mActionBarAdapter.setSearchMode(true);
        mActionBarAdapter.setQueryString(queryString);
    }
}

图2 第三方app拉起主界面显示对应的联系人
三、第三方app拉起联系人详情界面只滑动到一半显示的问题
-
图3 拉起只显示一半

图4 拉起完全显示
首先从系统的联系人列表界面点击进入详情界面是能完整显示的,所以猜想应该是传递的参数不太一样。所以还是从onCreate()方法看下来
public class QuickContactActivity extends ContactsActivity {
		/**
     * QuickContacts immediately takes up the full screen. All possible information is shown.
     * This value for {@link android.provider.ContactsContract.QuickContact#EXTRA_MODE}
     * should only be used by the Contacts app.
     */
    public static final int MODE_FULLY_EXPANDED = 4;
	//看上面的注释就知道了肯定是跟这个变量有关系, 立刻显示全屏,应当只用于 联系人 app 使用
	 @Override
    protected void onCreate(Bundle savedInstanceState) {
        Trace.beginSection("onCreate()");
        super.onCreate(savedInstanceState);
        if (RequestPermissionsActivity.startPermissionActivity(this)) {
            return;
        }
        getWindow().setStatusBarColor(Color.TRANSPARENT);
		//处理Intent传递的参数
        processIntent(getIntent());
 		.....
		//Scroller初始化,传递滚动模式
		mScroller.initialize(mMultiShrinkScrollerListener, mExtraMode == MODE_FULLY_EXPANDED);
        // mScroller needs to perform asynchronous measurements after initalize(), therefore
        // we can't mark this as GONE.
        mScroller.setVisibility(View.INVISIBLE);
		...
	}
	private void processIntent(Intent intent) {
        ...
		//获取传递的EXTRA_MODE,不传默认为large,查看api对应的int值为3,MODE_FULLY_EXPANDED为4, 所以不传递参数或者参数对应值不为4就只显示半屏
        mExtraMode = getIntent().getIntExtra(QuickContact.EXTRA_MODE, QuickContact.MODE_LARGE);
		...
    }
}
通过上面的分析 intent需要传递 QuickContact.EXTRA_MODE 参数, 当你点进去 QuickContact中发现并没有对应4的变量(猜想应该是留了一手不让第三方app直接全屏显示)
正确的打开姿势
private void gotoContact(){
    Uri personUri = ContentUris.withAppendedId(Contacts.People.CONTENT_URI, 1);
    Intent intent = new Intent();
    intent.setAction(Intent.ACTION_VIEW);
    intent.setData(personUri);
	//这句比较关键
    intent.putExtra(ContactsContract.QuickContact.EXTRA_MODE, 4);
    intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
    startActivity(intent);
}
总结
1、话说当把navigationBar去掉以后,给每个activity添加返回按钮是个很麻烦的工作,可以借鉴一下苹果的思路,直接在屏幕(Window)中添加一个悬浮的按钮处理返回点击事件。具体实现可以看这篇Android6.0 源码修改之 仿IOS添加全屏可拖拽浮窗返回按钮
2、源码没那么可怕,干起来。
Android6.0 源码修改之 Contacts应用的更多相关文章
- Android6.0 源码修改之 仿IOS添加全屏可拖拽浮窗返回按钮
		
前言 之前写过屏蔽系统导航栏功能的文章,具体可看Android6.0 源码修改之屏蔽导航栏虚拟按键(Home和RecentAPP)/动态显示和隐藏NavigationBar 在某些特殊定制的版本中要求 ...
 - Android6.0 源码修改之屏蔽系统短信功能和来电功能
		
一.屏蔽系统短信功能 1.屏蔽所有短信 android 4.2 短信发送流程分析可参考这篇 戳这 源码位置 vendor\mediatek\proprietary\packages\apps\Mms\ ...
 - Android6.0 源码修改之Settings音量调节界面增加通话音量调节
		
前言 今天客户提了个需求,因为我们的设备在正常情况下无法调节通话音量,只有在打电话过程中,按物理音量加减键才能出现调节通话音量seekBar,很不方便,于是乎需求就来了.需要优化两个地方 1.在正常情 ...
 - Android6.0 源码修改之屏蔽导航栏虚拟按键(Home和RecentAPP)/动态显示和隐藏NavigationBar
		
场景分析, 为了完全实现沉浸式效果,在进入特定的app后可以将导航栏移除,当退出app后再次将导航栏恢复.(下面将采用发送广播的方式来移除和恢复导航栏) ps:不修改源码的情况下,简单的沉浸式效果实现 ...
 - Android6.0 源码修改之Setting列表配置项动态添加和静态添加
		
写在前面 最近客户有个需求,要求增加操作Setting列表配置项的功能,是不是一脸懵,没关系,一图胜千言,接下来就上图.诺,就是这么个意思. 原来的列表配置项 增加了单个配置项 增 ...
 - 在Ubuntu Server14.04上编译Android6.0源码
		
此前编译过Android4.4的源码,但是现在Android都到了7.0的版本,不禁让我感叹Google的步伐真心难跟上,趁这周周末时间比较充裕,于是在过去的24小时里,毅然花了9个小时编译了一把An ...
 - Android6.0源码下载编译刷入真机
		
编译环境是Ubuntu12.04.手机nexus 5,编译安卓6.0.1源码并烧录到真机. 源码用的是科大的镜像:http://mirrors.ustc.edu.cn/aosp-monthly/,下载 ...
 - Ubuntu16.04下编译android6.0源码
		
http://blog.csdn.net/cnliwy/article/details/52189349 作为一名合格的android开发人员,怎么能不会编译android源码呢!一定要来一次说编译就 ...
 - Android6.0源码分析之录音功能(一)【转】
		
本文转载自:http://blog.csdn.net/zrf1335348191/article/details/54949549 从现在开始一周时间研究录音,下周出来一个完整的博客,监督,激励!!! ...
 
随机推荐
- maven仓库添加jar架包
			
推荐几个好的 Maven 常用仓库网址:http://mvnrepository.com/http://search.maven.org/http://repository.sonatype.org/ ...
 - Navicat永久激活步骤,激活工具,解决注册码无效的问题
			
Navicat for MySQL是一套管理和开发MySQL或MariaDB的理想解决方案,支持单一程序,可同时连接到MySQL和MariaDB.这个功能齐备的前端软件为数据库管理.开发和维护提供了直 ...
 - lvs+keepalive实现主从效果,以及RS健康监测和tcp,udp实现非web的负载均衡
			
前面文章讲到了tcp和udp负载均衡,但是没有健康监测,这几天我优化了一下上次的操作.当然,我也是用的跨网段的通讯,因为线上业务主要是海外业务,所以做了iptables流量转发 IP: lvs-mas ...
 - 用户注册登录系统 V2.0
			
# 准备空列表 users = [] # 准备当前在线用户 online_user = {} while True: # 打印系统提示 print("欢迎使用 用户注册登录系统V2.0&qu ...
 - JS入门熟知
			
JS是面向对象的语言 封装 继承 多态 聚集(对象中具有引用其他对象的能力) JS使用中绝大多数情况不需要进行面向对象的设计,很多情况是使用已经设计好,准备好的对象,基于对象的语言. JS的使用(引入 ...
 - VS2010+OpenCV3.4.1+zbar 64位
			
1. OpenCV3.4.1和zbar文件夹放到指定的路径下,我把它们放在了"D:\二维码\环境"中. zbar:链接:https://pan.baidu.com/s/11eCDV ...
 - PAT1091:Acute Stroke
			
1091. Acute Stroke (30) 时间限制 400 ms 内存限制 65536 kB 代码长度限制 16000 B 判题程序 Standard 作者 CHEN, Yue One impo ...
 - mysql中float类型使用总结
			
对于单精度浮点数Float: 当数据范围在±131072(65536×2)以内的时候,float数据精度是正确的,但是超出这个范围的数据就不稳定,没有发现有相关的参数设置建议:将float改成dou ...
 - java对象深复制、浅复制(深拷贝、浅拷贝)的理解
			
先看一个例子 User user1 = new User(); user1.setId("111"); Map<String, User> map1 = new Has ...
 - SQL Server 2008更改数据库保存路径
			
本文由荒原之梦原创,原文链接:http://zhaokaifeng.com/?p=641 操作环境: WindowsXP 数据库: Microsoft SQL Server 2008 操作步骤: 选中 ...