欢迎转载,但请保留作者链接:http://www.jianshu.com/p/ca4ab4e9817f
作为Dialer Owner,作一下基于M版本的总结吧。
在线源码阅读:http://androidxref.com

总体轮廓

手机之所以被称为手机,是因为它是一个通讯工具,而完成这一核心功能的软件模块,即为Telephony。
Telephony包含的范围非常广泛,单拿上层来说,大致可以划分成五大部分:Telephony应用(DialerContactsMms),service Telephony和service Telecomm,framework Telephony和framework Telecomm。
现在这一架构的主要变化是从L版本开始的,相较旧版的主要变迁可以参考:Android 4.4 Kitkat Phone工作流程浅析(十二)__4.4小结与5.0概览


图片资料

本文只关注Dialer,那么先看几张Nexus 6p的实机截图来个感性的认识:

Dialer_示例1

Dialer_示例2

Dialer_示例3


架构分析

Dialer主要涉及的包有:
1)/packages/apps下
DialerInCallUIContactsCommonPhoneCommonVoiceDialer

凭借makefile,分包可以非常的自由随意,看如下片断:

6incallui_dir := ../InCallUI
7contacts_common_dir := ../ContactsCommon
8phone_common_dir := ../PhoneCommon
9
10src_dirs := src \
11    $(incallui_dir)/src \
12    $(contacts_common_dir)/src \
13    $(phone_common_dir)/src

DialerInCallUIPhoneCommonContactsCommon全都在src_dirs路径下了,于是最终的Dialer.apk由这四个包下的代码编译生成。

VoiceDialer提供语音相关功能,入口看下图:

但是,此功能侵略性过强,在天朝是基本残废的,在海外多数运营商也不喜欢表示要去除,所以不予关注。

2)/packages/services下
MmsTelephonyTelecomm,生成MmsService.apk,Telecom.apk与TeleService.apk,对Dialer来说是提供通话菜单功能的。
应该说不管从逻辑还是物理上,切分出来都是大有好处,这样才能让Android能够良好支持第三方通讯类应用。

3)/packages/providers下
TelephonyProviderContactsProvider,数据创建及查询,当然也是要切分的部分。

4)frameworks/opt和frameworks/base下
telephony等和上面类似的眼熟名字,具体关系到各种功能点如MmiCode,Clear Code,Number match,Number format,DTMF,FDN等等等等。


具体分析

看完整体架构之后,单单一个Dialer包的定位也变得很清晰了:它就只是一个拨号器而已。

1.层次结构

Dialer的UI是否美观是个见仁见智的问题,我个人还是挺喜欢的。

这里我们只谈其实现原理。
这是Dialer的主layout dialtacts_activity.xml:

16<FrameLayout
17    xmlns:android="http://schemas.android.com/apk/res/android"
18    android:id="@+id/dialtacts_mainlayout"
19    android:layout_width="match_parent"
20    android:layout_height="match_parent"
21    android:orientation="vertical"
22    android:focusable="true"
23    android:focusableInTouchMode="true"
24    android:clipChildren="false"
25    android:background="@color/background_dialer_light">
26
27    <FrameLayout
28        android:id="@+id/dialtacts_container"
29        android:layout_width="match_parent"
30        android:layout_height="match_parent"
31        android:clipChildren="false">
32        <!-- The main contacts grid -->
33        <FrameLayout
34            android:layout_height="match_parent"
35            android:layout_width="match_parent"
36            android:id="@+id/dialtacts_frame"
37            android:clipChildren="false" />
38    </FrameLayout>
39
40    <FrameLayout
41        android:id="@+id/floating_action_button_container"
42        android:background="@drawable/fab_blue"
43        android:layout_width="@dimen/floating_action_button_width"
44        android:layout_height="@dimen/floating_action_button_height"
45        android:layout_marginBottom="@dimen/floating_action_button_margin_bottom"
46        android:layout_gravity="center_horizontal|bottom">
47
48        <ImageButton
49            android:id="@+id/floating_action_button"
50            android:background="@drawable/floating_action_button"
51            android:layout_width="match_parent"
52            android:layout_height="match_parent"
53            android:contentDescription="@string/action_menu_dialpad_button"
54            android:src="@drawable/fab_ic_dial"/>
55
56    </FrameLayout>
57
58    <!-- Host container for the contact tile drag shadow -->
59    <FrameLayout
60        android:id="@+id/activity_overlay"
61        android:layout_height="match_parent"
62        android:layout_width="match_parent">
63        <ImageView
64            android:id="@+id/contact_tile_drag_shadow_overlay"
65            android:layout_width="wrap_content"
66            android:layout_height="wrap_content"
67            android:visibility="gone"
68            android:importantForAccessibility="no" />
69    </FrameLayout>
70
71</FrameLayout>

虽然不同情况下显示的部分不同,但总得来说以z轴从大到小做一个侧向剖面图排列的话,主要元素是这样的:

Dialer UI剖面图

  • FAB button(FloatingActionButton)
    layout中R.id.floating_action_button_container位置即是,并不是真正的FloatingActionButton,Google工程师手搓了一个看上去有着类似效果的控件而已。用来控制Dialpad的展开和收起。

  • Dialpad(Fragment)
    实现类为xref: /packages/apps/Dialer/src/com/android/dialer/dialpad/DialpadFragment.java,填充进R.id.dialtacts_container中,自带号码输入条,按下拨号钮后会构造相应Intent然后启动service Telecomm中的不可见Activity如UserCallActivity(L中对应为CallActivity,M中为实现AFW,Android for Work模式引入)开始拨号处理流程。

  • SearchUI(Fragment)
    填充在R.id.dialtacts_frame中,注意展开与收起Dialpad时,虽然肉眼感知不到,但却会使用不同的Fragment来提供搜索结果页面。一个是SmartDialSearchFragment,另一个是RegularSearchFragment。
    这是因为,其在设计上还支持更强大的搜索功能,能使用输入法进行输入:

  • 国内的一般都把这功能直接做掉了-_-||。

  • Content pages(Fragment)
    实现类为xref: /packages/apps/Dialer/src/com/android/dialer/list/ListsFragment.java,填充进R.id.dialtacts_frame,因为时间上肯定比SearchUI要早,所以在其下面。ListsFragment中使用ViewPager又组织着三个Fragment作为之前图示中的三个Tab页(当满足情况时,第四个页面会出现)。这样的嵌套是否是一个好的设计值得商榷。


2.拨号盘

实现代码为/packages/apps/Dialer/src/com/android/dialer/dialpad/DialpadFragment.java
拨号盘分三个,全都使用了PhoneCommon包中的同一套资源:

  • Dialer中一个
  • InCallUI中一个
  • KeyGuard中有个“紧急呼叫”按钮,会调用到service Telephony中的一个弱化版拨号盘

关于UI:Google原生设计上这三处使用了一致的UI,所以直接复用即可。但是,其他手机设计有很多都是不一样的,所以这里是一个客制化比较麻烦的点。

关于双卡拨号:Google对于双卡的支持就是:选择默认卡->Dialer中按下拨号->使用默认卡拨号。国内很多厂商的做法却都是在Dialpad上提供两个按钮,如卡一“中国移动”卡二“中国联通”,需要按哪个键就用哪张卡进行拨号。其实在拨号流程中的service Telecomm中的一环有使用一个关键值PhoneAccountHandle来进行判定使用哪张SIM卡,所以实现的方法也就只是很简单地Intent传值、取值、处理即可。L中拨号流程为CallActivity#processOutgoingCallIntent->CallReceiver#processOutgoingCallIntent->CallsManager#startOutgoingCall,M中略有变动,开始的变为了UserCallActivity,往下找即可。

关于长按数字键快速拨号

如设置2键拨号119,则长按2能够立即进行拨号。这个功能很长一段时间以来由MTK的MTK Plugin提供,然而在L版本中Google提供了SpeedDial,即Dialer_示例1中的Tab页面,对联系人收藏之后就变成了这个样子,多了一个小卡片可以直接点击呼叫:

SpeedDial

虽然本身完全不是同一个东西,但MTK表示此功能将废弃。
如果要自行实现的话,实现原理大略是这样:遵循Fragment寄宿于Activity的思路写一个类似组件,关键回调中调用同名方法。建立一个数据库,用户设置相应键位的快速拨号时,如果是设置号码,那就简单保存号码;如果设置的是联系人,那么置标志位,保存联系人条目的主键,每次拨号或显示时之前,使用该主键查询数据库获取所需信息。


3.设置

设置页的UI非常糟糕,毫无设计可言,而且层次多得不像话,以下演示的是如何进行通话帐户设置:

示例5跟6都只是简单罗列多行单调的文字,而且风格还不统一,一个有分隔线另一个没有(示例5在Dialer包中,示例6则在services/Telephony中,可以推断不是同一拨人做的,而且没有沟通好,于是出现了如此明显的差异),只有示例7看出了点儿Material Design的影子,但是只有一个分类的话Category又变得没有意义了。
我期待的设置页是层次合理,有分类带说明文字的,下图来自于我的开源练习之作PureNote:

示例5的页面由Dialer中的DialerSettingsActivity.java提供,并且,会根据单双卡而添加不同的Header,而往后的通话设置则主要由service Telephony提供支持,参见如下代码:

59            // Show "Call Settings" if there is one SIM and "Phone Accounts" if there are more.
60            if (telephonyManager.getPhoneCount() <= 1) {
61                Header callSettingsHeader = new Header();
62                Intent callSettingsIntent = new Intent(TelecomManager.ACTION_SHOW_CALL_SETTINGS);
63                callSettingsIntent.setFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP);
64
65                callSettingsHeader.titleRes = R.string.call_settings_label;
66                callSettingsHeader.intent = callSettingsIntent;
67                target.add(callSettingsHeader);
68            } else {
69                Header phoneAccountSettingsHeader = new Header();
70                Intent phoneAccountSettingsIntent =
71                        new Intent(TelecomManager.ACTION_CHANGE_PHONE_ACCOUNTS);
72                phoneAccountSettingsIntent.setFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP);
73
74                phoneAccountSettingsHeader.titleRes = R.string.phone_account_settings_label;
75                phoneAccountSettingsHeader.intent = phoneAccountSettingsIntent;
76                target.add(phoneAccountSettingsHeader);
77            }

单卡为CallFeaturesSetting,双卡为PhoneAccountSettingsActivityPhoneAccountSettingsActivity只是显示两行卡名,如行一中国移动行二中国联通,然后点击后再复用CallFeaturesSetting


4.数据获取

Dialer特色的自然就是CallLog部分还有SearchUI部分了。
关于SearchUI,可以参考这篇文章Android拨号搜索机制源码分析(原)。
对于CallLog,其数据查询与更新采用的是AsyncQueryHandler+ContentObserver,Google应该考虑用Loader来取代它们。
参考Android4.4 Telephony流程分析——拨号应用(Dialer)的通话记录加载过程、Handler官方范例AsyncQueryHandler源码解析

我只想说:读CallLog的代码(这里指得是数据获取+内容显示),不啻于去地狱走一遭,一坨一坨的极为恐怖。


代码细节

Dialer中的一些代码细节。

组合模式

设置监听器分两种情况:setOnXXXListeneraddOnXXXListener,通常来讲,后者要优于前者,所以许多类都增加了add方法而废弃了set方法。可假如说你使用的这个类现在只有set方法可得怎么办呢?只需要使用组合模式即可达成效果,具体使用场景可以参考ListsFragment。以下代码为一个示例:

public class ViewPagerListenersUtil implements ViewPager.OnPageChangeListener {private ArrayList<OnPageChangeListener> mOnPageChangeListeners = new rrayList<OnPageChangeListener>();@Overridepublic void onPageScrolled(int position, float positionOffset, int positionOffsetPixels) {final int count = mOnPageChangeListeners.size();for (int i = 0; i < count; i++) {mOnPageChangeListeners.get(i).onPageScrolled(position, positionOffset,positionOffsetPixels);}}@Overridepublic void onPageSelected(int position) {final int count = mOnPageChangeListeners.size();for (int i = 0; i < count; i++) {mOnPageChangeListeners.get(i).onPageSelected(position);}}@Overridepublic void onPageScrollStateChanged(int state) {final int count = mOnPageChangeListeners.size();for (int i = 0; i < count; i++) {mOnPageChangeListeners.get(i).onPageScrollStateChanged(state);}}public void addOnPageChangeListener(OnPageChangeListener onPageChangeListener) {if (!mOnPageChangeListeners.contains(onPageChangeListener)) {mOnPageChangeListeners.add(onPageChangeListener);}}
}

异步设置TextWatcher

DialpadFragment中为了实现i18n号码处理,需要给号码输入条添加一个TextWatcher,Google工程师是这样做的,其中mDigits为TextView

370        PhoneNumberFormatter.setPhoneNumberFormattingTextWatcher(getActivity(), mDigits);

跟踪代码:

26public final class PhoneNumberFormatter {
27    private PhoneNumberFormatter() {}
28
29    /**
30     * Load {@link TextWatcherLoadAsyncTask} in a worker thread and set it to a {@link TextView}.
31     */
32    private static class TextWatcherLoadAsyncTask extends
33            AsyncTask<Void, Void, PhoneNumberFormattingTextWatcher> {
34        private final String mCountryCode;
35        private final TextView mTextView;
36
37        public TextWatcherLoadAsyncTask(String countryCode, TextView textView) {
38            mCountryCode = countryCode;
39            mTextView = textView;
40        }
41
42        @Override
43        protected PhoneNumberFormattingTextWatcher doInBackground(Void... params) {
44            return new PhoneNumberFormattingTextWatcher(mCountryCode);
45        }
46
47        @Override
48        protected void onPostExecute(PhoneNumberFormattingTextWatcher watcher) {
49            if (watcher == null || isCancelled()) {
50                return; // May happen if we cancel the task.
51            }
52            // Setting a text changed listener is safe even after the view is detached.
53            mTextView.addTextChangedListener(watcher);
54
55            // Note changes the user made before onPostExecute() will not be formatted, but
56            // once they type the next letter we format the entire text, so it's not a big deal.
57            // (And loading PhoneNumberFormattingTextWatcher is usually fast enough.)
58            // We could use watcher.afterTextChanged(mTextView.getEditableText()) to force format
59            // the existing content here, but that could cause unwanted results.
60            // (e.g. the contact editor thinks the user changed the content, and would save
61            // when closed even when the user didn't make other changes.)
62        }
63    }
64
65    /**
66     * Delay-set {@link PhoneNumberFormattingTextWatcher} to a {@link TextView}.
67     */
68    public static final void setPhoneNumberFormattingTextWatcher(Context context,
69            TextView textView) {
70        new TextWatcherLoadAsyncTask(GeoUtil.getCurrentCountryIso(context), textView)
71                .executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR, (Void[]) null);
72    }
73}

合一

Contacts_示例1

Contacts_示例2

王自如评价三星手机内置应用时说“不同的产品经理也许从来都没有交流过,所以才会做出风格这么不统一的产品”,这话放在Android M的Dialer与Contacts身上也是十分贴切啊,肉眼可辨的坑爹啊,强迫症能忍么?

从功能上来讲,你会发现与Dialer相比较,Contacts的存在感简直弱爆了,它能做到的事情,Dialer全都能做。而且因为Contacts只能操作联系人数据,几乎让人没有想点它的兴趣。

实际上,在4.2版本中Contacts与Dialer就是同一个应用Contacts.apk,只不过分出了两个应用入口来而已。当然,虽然实际上是同一个应用,但在用户感受上则是两个。

而国内UI如MIUI还有锤子Rom都很明智地对这两个应用进行了代码与用户感受上的“合一”,只不过小米是保留了两个入口,但启动的都是同一个应用;而锤子是只提供Dialer应用,给你两个应用的功能。华为最残暴,大手一挥把联系不太大的Mms都合了,号称“三合一”(不过,最新版本的EMUI又只有二合一了):

三合一

如何达到这一效果?不复用原生代码的话,自己刷刷刷开写可以解决问题,但这实在是费力。复用原生代码的同时达到这一效果,主要就三点,一是前面提到过的makefile的修改,删除两个makefile,修改最后一个makefile让它们编译出一个应用来;二是修改manifest,善用alias让应用能正常被使用;最后则是应用重构想办法让它一个顶仨了。


推荐阅读

  • http://blog.csdn.net/yihongyuelan?viewmode=contents
  • http://blog.csdn.net/xiashaohua?viewmode=contents

作者:郭非文
链接:https://www.jianshu.com/p/ca4ab4e9817f
来源:简书
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。

Android6 M Dialer完全总结相关推荐

  1. android 6.0 拨号界面,【Dialer】android6.0拨号界面分析一

    题记 工作需要,最近对拨号模块进行研究,下文主要介绍Android6.0下拨号应用主界面. Dialer代码目录 Dialer上层代码主要有以下部分组成: /packages/apps/Dialer/ ...

  2. Android6.0 源码添加黑名单拦截电话和短信记录

    [目标] 在上一篇 Android6.0 源码增加黑名单功能 的基础上增加黑名单和短信拦截记录 [实现] 黑名单数据库和拦截记录数据库上一篇已经增加完成,这就需要我们在电话和短信分发的地方去判断号码是 ...

  3. android 蓝牙找不到电脑,Android6.0 蓝牙搜索不到设备原因

    原因: 为提供更高的数据保护 Android6.0版本上增加了关于Wifi和蓝牙的权限,以下是官方文档说明: 图1 修改方法: 在AndroidManifest 中添加权限 或者 注意 如果targe ...

  4. 编译可在Nexus5上运行的CyanogenMod13.0 ROM(基于Android6.0)

    编译可在Nexus5上运行的CyanogenMod13.0 ROM (基于Android6.0) 作者:寻禹@阿里聚安全 前言 下文中无特殊说明时CM代表CyanogenMod的缩写. 下文中说的&q ...

  5. android6.0麦克风权限,android 6.0权限检测以及6.0以下,麦克风、相机权限判断

    android 6.0以上权限 android 6.0以上权限,我是通过PermissionsDispatcher进行申请,操作的,具体使用方法,见PermissionsDispatcher,Andr ...

  6. nexus5 刷 Android6.0+Xposed

    不得不说现在的刷机工具都做的方便快捷,只需要简单的几天命令就解决了.以前用windows刷的时候还需要装一堆驱动软件啥的. 原文链接: nexus5 刷 Android6.0+Xposed 0x00 ...

  7. Android6.0执行时权限解析,RxPermissions的使用,自己封装一套权限框架

    Android6.0执行时权限解析,RxPermissions的使用.自己封装一套权限框架 在Android6.0中,新添加了一个执行时的权限,我相信非常多人都已经知道了.预计也知道怎么用了,这篇博客 ...

  8. Android6.0------权限申请管理(单个权限和多个权限申请)

    2019独角兽企业重金招聘Python工程师标准>>> Android开发时,到6.0系统上之后,有的权限就得申请才能用了. Android将权限分为正常权限 和 危险权限 Andr ...

  9. Android6.0以上的权限问题

    Android6.0以上的权限问题 可以在使用权限的页面oncreate方法中进行如下的动态的申请 if (ContextCompat.checkSelfPermission(this, Manife ...

最新文章

  1. 搭建Eclipse+MyEclipse开发环境
  2. 机器学习中防止过拟合的处理方法
  3. [NET] 如何从 Winform 移植到 Webform [自己搞定HTTP协议]
  4. IOS7新增120*120Icon图标
  5. 用MOS管防止电源反接的原理
  6. mysql level用法_MYSQL使用方法
  7. 工业以太网交换机品牌排行榜,国产工业交换机哪个品牌好?
  8. javascript消除字符串两边空格的两种方式,面向对象和函数式编程。python oop在调用时候的优点...
  9. 广电总局:坚决抵制含有暴力血腥等不良情节动画片上网播出
  10. 34. 丑数(C++版本)
  11. dataframe groupby_python pandas获取groupby之后的数据
  12. 【NLP】第6章 使用 Transformer 进行机器翻译
  13. 已经无限接近于真实!EA下一代寒霜引擎展示令人惊叹的毛发效果
  14. 你知道大量群发邮件用什么邮箱好吗?
  15. hotmail邮箱设置
  16. Tryhackme -Skynet(考点:smb 密码爆破 cuppa RFI cronjob - tar提权)
  17. [艺术创作]摄影构图的传统法则
  18. 使用Iframe嵌套其他系统页面遇到的跨域问题
  19. ubuntu五笔死机
  20. heic格式转化jpg,详细转换步骤

热门文章

  1. MyBatis简介与配置
  2. 苹果微信多开共存软件免费发布
  3. WIN32开发之Unicode字符集下cstring转换为char[]相互转换
  4. 重装WIN7系统后连不上网络怎么办
  5. 【PCL】LINK2001 “protected:virtual void_cdel pcl::XXX报错
  6. [VP] 投影空间坐标和线的齐次变换
  7. Hive特殊分隔符处理
  8. 个人微信开发API协议
  9. 兰卡斯特大学研究人员开发电信量子技术专利
  10. IGMP协议软件开发实验