源码分析之Android通过Dialer实现暗码启动

2024-09-06 09:08

本文主要是介绍源码分析之Android通过Dialer实现暗码启动,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

目前接触比较多的就是通过dialer应用来启动/触发暗码。

也有通过Calculator来实现的。

本文以Dialer为例,

1.经过调试定位,发现拨号盘接对应的Activity为DialtactsActivity。

2.DialtactsActivity中有个showDialpadFragment方法,用来加载显示拨号盘,因为有可能此时拨号盘正处于收缩/隐藏状态。

    /*** Initiates a fragment transaction to show the dialpad fragment. Animations and other visual* updates are handled by a callback which is invoked after the dialpad fragment is shown.* @see #onDialpadShown*/private void showDialpadFragment(boolean animate) {if (mIsDialpadShown || mStateSaved) {return;}mIsDialpadShown = true;mListsFragment.setUserVisibleHint(false);final FragmentTransaction ft = getFragmentManager().beginTransaction();if (mDialpadFragment == null) {mDialpadFragment = new DialpadFragment();ft.add(R.id.dialtacts_container, mDialpadFragment, TAG_DIALPAD_FRAGMENT);} else {ft.show(mDialpadFragment);}//mDialpadFragment.setAnimate(animate);AnalyticsUtil.sendScreenView(mDialpadFragment);ft.commit();maybeEnterSearchUi();if (animate) {mFloatingActionButtonController.scaleOut();} else {mFloatingActionButtonController.setVisible(false);}mActionBarController.onDialpadUp();mListsFragment.getView().animate().alpha(0).withLayer();}

3.接下来重点处理实现就在DialpapFragment中,首先来看类的声明/继承。

import com.android.dialer.dialpad.DialpadFragment;

/*** Fragment that displays a twelve-key phone dialpad.*/
public class DialpadFragment extends Fragmentimplements View.OnClickListener,View.OnLongClickListener, View.OnKeyListener,AdapterView.OnItemClickListener, TextWatcher,PopupMenu.OnMenuItemClickListener,DialpadKeyButton.OnPressedListener,/// M: add for plug-in @{DialpadExtensionAction {

从以上类实现/继承中可以发现,其继承了TextWatcher类,也正是这个类使之能够监听实现输入变化。

TextWatcher有3个重要方法,分别为:beforeTextChanged,onTextChanged和afterTextChanged。分别看下面那份源码。


onTextChanged


其中最重点的是afterTextChanged方法,其调用了SpecialCharSequenceMgr辅助工具类的handleChars方法。


4.handleChars方法中,会对各种特殊的secret code进行匹配处理。

    public static boolean handleChars(Context context, String input, EditText textField) {//get rid of the separators so that the string gets parsed correctlyString dialString = PhoneNumberUtils.stripSeparators(input);if (handleDeviceIdDisplay(context, dialString) //*#06#|| handleRegulatoryInfoDisplay(context, dialString)|| handlePinEntry(context, dialString)|| handleAdnEntry(context, dialString, textField)|| handleSecretCode(context, dialString) //for the form of *#*#<code>#*#*./// @}/// M: for plug-in @{|| ExtensionManager.getInstance().getDialPadExtension().handleChars(context,dialString)/// @}) {return true;}return false;}
5.接下来分开两种讲,一种是直接弹出对话框的那种,类如*#06#,另一种则是调起别的应用等方式。
5.1 )*#*#<code>#*#*

    /*** Handles secret codes to launch arbitrary activities in the form of *#*#<code>#*#*.* If a secret code is encountered an Intent is started with the android_secret_code://<code>* URI.** @param context the context to use* @param input the text to check for a secret code in* @return true if a secret code was encountered*/static boolean handleSecretCode(Context context, String input) {// Secret codes are in the form *#*#<code>#*#*/// M: for plug-in @{input = ExtensionManager.getInstance().getDialPadExtension().handleSecretCode(input);/// @}int len = input.length();if (len > 8 && input.startsWith("*#*#") && input.endsWith("#*#*")) {final Intent intent = new Intent(SECRET_CODE_ACTION,Uri.parse("android_secret_code://" + input.substring(4, len - 4)));///android_secret_code://287context.sendBroadcast(intent);return true;}return false;}
有以上代码可知,最终是通过broadcast发送出去的,并往intent里面加载了2种数据:Uri和Action。

Action:

private static final String SECRET_CODE_ACTION = "android.provider.Telephony.SECRET_CODE";
接受端的注册方式,Action 和 data必须和发送的broadcast相匹配才行:

/vendor/mediatek/proprietary/packages/apps/EngineerMode/AndroidManifest.xml

        <receiverandroid:name=".EngineerModeReceiver"android:exported="true" ><intent-filter><action android:name="android.provider.Telephony.SECRET_CODE" /><dataandroid:host="3646633"android:scheme="android_secret_code" /></intent-filter></receiver>

/vendor/mediatek/proprietary/packages/apps/EngineerMode/src/com/mediatek/engineermode/EngineerModeReceiver.java


由上面代码可知,这就对应上了,在Receiver接受到广播后,启动对应的应用/Activity来处理接下来的工作。


5.2 )*#06#  直接在Context中弹出对话框,显示IMEI信息

/packages/apps/Dialer/src/com/android/dialer/SpecialCharSequenceMgr.java

    // TODO: Use TelephonyCapabilities.getDeviceIdLabel() to get the device id label instead of a// hard-coded string.static boolean handleDeviceIdDisplay(Context context, String input) {TelephonyManager telephonyManager =(TelephonyManager) context.getSystemService(Context.TELEPHONY_SERVICE);if (telephonyManager != null && input.equals(MMI_IMEI_DISPLAY)) {int labelResId = (telephonyManager.getPhoneType() == TelephonyManager.PHONE_TYPE_GSM) ?R.string.imei : R.string.meid;List<String> deviceIds = new ArrayList<String>();if (TelephonyManagerCompat.getPhoneCount(telephonyManager) > 1 &&CompatUtils.isMethodAvailable(TelephonyManagerCompat.TELEPHONY_MANAGER_CLASS,"getDeviceId", Integer.TYPE)) {for (int slot = 0; slot < telephonyManager.getPhoneCount(); slot++) {String deviceId = telephonyManager.getDeviceId(slot);if (!TextUtils.isEmpty(deviceId)) {deviceIds.add(deviceId);}}} else {deviceIds.add(telephonyManager.getDeviceId());}AlertDialog alert = new AlertDialog.Builder(context).setTitle(labelResId).setItems(deviceIds.toArray(new String[deviceIds.size()]), null).setPositiveButton(android.R.string.ok, null).setCancelable(false).show();///直接在Context中弹出对话框,显示IMEI信息return true;}return false;}

*#07# 直接通过隐式intent启动相关应用

    private static boolean handleRegulatoryInfoDisplay(Context context, String input) {if (input.equals(MMI_REGULATORY_INFO_DISPLAY)) {Log.d(TAG, "handleRegulatoryInfoDisplay() sending intent to settings app");Intent showRegInfoIntent = new Intent(Settings.ACTION_SHOW_REGULATORY_INFO);try {context.startActivity(showRegInfoIntent);} catch (ActivityNotFoundException e) {Log.e(TAG, "startActivity() failed: " + e);}return true;}return false;}

以上就是Android通过Dialer方式启动暗码的大致源码流程分析。


这篇关于源码分析之Android通过Dialer实现暗码启动的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1141592

相关文章

SpringBoot中SM2公钥加密、私钥解密的实现示例详解

《SpringBoot中SM2公钥加密、私钥解密的实现示例详解》本文介绍了如何在SpringBoot项目中实现SM2公钥加密和私钥解密的功能,通过使用Hutool库和BouncyCastle依赖,简化... 目录一、前言1、加密信息(示例)2、加密结果(示例)二、实现代码1、yml文件配置2、创建SM2工具

Mysql实现范围分区表(新增、删除、重组、查看)

《Mysql实现范围分区表(新增、删除、重组、查看)》MySQL分区表的四种类型(范围、哈希、列表、键值),主要介绍了范围分区的创建、查询、添加、删除及重组织操作,具有一定的参考价值,感兴趣的可以了解... 目录一、mysql分区表分类二、范围分区(Range Partitioning1、新建分区表:2、分

MySQL 定时新增分区的实现示例

《MySQL定时新增分区的实现示例》本文主要介绍了通过存储过程和定时任务实现MySQL分区的自动创建,解决大数据量下手动维护的繁琐问题,具有一定的参考价值,感兴趣的可以了解一下... mysql创建好分区之后,有时候会需要自动创建分区。比如,一些表数据量非常大,有些数据是热点数据,按照日期分区MululbU

MySQL中查找重复值的实现

《MySQL中查找重复值的实现》查找重复值是一项常见需求,比如在数据清理、数据分析、数据质量检查等场景下,我们常常需要找出表中某列或多列的重复值,具有一定的参考价值,感兴趣的可以了解一下... 目录技术背景实现步骤方法一:使用GROUP BY和HAVING子句方法二:仅返回重复值方法三:返回完整记录方法四:

IDEA中新建/切换Git分支的实现步骤

《IDEA中新建/切换Git分支的实现步骤》本文主要介绍了IDEA中新建/切换Git分支的实现步骤,通过菜单创建新分支并选择是否切换,创建后在Git详情或右键Checkout中切换分支,感兴趣的可以了... 前提:项目已被Git托管1、点击上方栏Git->NewBrancjsh...2、输入新的分支的

怎样通过分析GC日志来定位Java进程的内存问题

《怎样通过分析GC日志来定位Java进程的内存问题》:本文主要介绍怎样通过分析GC日志来定位Java进程的内存问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、GC 日志基础配置1. 启用详细 GC 日志2. 不同收集器的日志格式二、关键指标与分析维度1.

Python实现对阿里云OSS对象存储的操作详解

《Python实现对阿里云OSS对象存储的操作详解》这篇文章主要为大家详细介绍了Python实现对阿里云OSS对象存储的操作相关知识,包括连接,上传,下载,列举等功能,感兴趣的小伙伴可以了解下... 目录一、直接使用代码二、详细使用1. 环境准备2. 初始化配置3. bucket配置创建4. 文件上传到os

关于集合与数组转换实现方法

《关于集合与数组转换实现方法》:本文主要介绍关于集合与数组转换实现方法,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录1、Arrays.asList()1.1、方法作用1.2、内部实现1.3、修改元素的影响1.4、注意事项2、list.toArray()2.1、方

使用Python实现可恢复式多线程下载器

《使用Python实现可恢复式多线程下载器》在数字时代,大文件下载已成为日常操作,本文将手把手教你用Python打造专业级下载器,实现断点续传,多线程加速,速度限制等功能,感兴趣的小伙伴可以了解下... 目录一、智能续传:从崩溃边缘抢救进度二、多线程加速:榨干网络带宽三、速度控制:做网络的好邻居四、终端交互

java实现docker镜像上传到harbor仓库的方式

《java实现docker镜像上传到harbor仓库的方式》:本文主要介绍java实现docker镜像上传到harbor仓库的方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地... 目录1. 前 言2. 编写工具类2.1 引入依赖包2.2 使用当前服务器的docker环境推送镜像2.2