android中暂停服务,Android 装逼技术之暗码启动应用

前言

喜欢的小伙伴欢迎关注,我会定期分享Android知识点及解析,还会不断更新的BATJ面试专题,欢迎大家前来探讨交流,如有好的文章也欢迎投稿。

什么是暗码?

在拨号盘中输入*#*##*#*后,APP 可以监控到这些输入,然后做相应的动作,比如启动应用,是不是有点骚。

下面看下这个骚操作是如何实现的。

效果预览

8e8485f37eb894b9eaac874853f632ac.gif

源码

DialtactsActivity#showDialpadFragment

DialtactsActivity 中有个 showDialpadFragment 方法,用来加载显示拨号盘,因此入口就从 showDialpadFragment 看起,基于 Android P 分析。private void showDialpadFragment(boolean animate) {

//……

final FragmentTransaction ft = getFragmentManager().beginTransaction();

if (dialpadFragment == null) {

dialpadFragment = new DialpadFragment();

ft.add(R.id.dialtacts_container, dialpadFragment, TAG_DIALPAD_FRAGMENT);

} else {

ft.show(dialpadFragment);

}

//……

}

具体实现在 DialpapFragment 中,看到 DialpapFragment 实现了 TextWatcher,TextWatcher 有 3 个重要方法,分别为:beforeTextChanged,onTextChanged 和 afterTextChanged,重点看 afterTextChanged 方法。

DialpadFragment#afterTextChangedpublic class DialpadFragment extends Fragment

implements View.OnClickListener,

View.OnLongClickListener,

View.OnKeyListener,

AdapterView.OnItemClickListener,

TextWatcher,

PopupMenu.OnMenuItemClickListener,

DialpadKeyButton.OnPressedListener {

//……

@Override

public void afterTextChanged(Editable input) {

// When DTMF dialpad buttons are being pressed, we delay SpecialCharSequenceMgr sequence,

// since some of SpecialCharSequenceMgr's behavior is too abrupt for the "touch-down"

// behavior.

if (!digitsFilledByIntent

&& SpecialCharSequenceMgr.handleChars(getActivity(), input.toString(), digits)) {

// A special sequence was entered, clear the digits

digits.getText().clear();

}

if (isDigitsEmpty()) {

digitsFilledByIntent = false;

digits.setCursorVisible(false);

}

if (dialpadQueryListener != null) {

dialpadQueryListener.onDialpadQueryChanged(digits.getText().toString());

}

updateDeleteButtonEnabledState();

}

//……

}

这里调用了 SpecialCharSequenceMgr 辅助工具类的 handleChars 方法,看这个方法。

SpecialCharSequenceMgr#handleCharspublic static boolean handleChars(Context context, String input, EditText textField) {

// get rid of the separators so that the string gets parsed correctly

String dialString = PhoneNumberUtils.stripSeparators(input);

if (handleDeviceIdDisplay(context, dialString)

|| handleRegulatoryInfoDisplay(context, dialString)

|| handlePinEntry(context, dialString)

|| handleAdnEntry(context, dialString, textField)

|| handleSecretCode(context, dialString)) {

return true;

}

if (MotorolaUtils.handleSpecialCharSequence(context, input)) {

return true;

}

return false;

}

handleChars 方法中,会对各种特殊的 secret code 进行匹配处理,这里我们看 handleSecretCode。

SpecialCharSequenceMgr#handleSecretCodestatic boolean handleSecretCode(Context context, String input) {

// Secret code specific to OEMs should be handled first.

if (TranssionUtils.isTranssionSecretCode(input)) {

TranssionUtils.handleTranssionSecretCode(context, input);

return true;

}

// Secret codes are accessed by dialing *#*##*#* or "*##"

if (input.length() > 8 && input.startsWith("*#*#") && input.endsWith("#*#*")) {

String secretCode = input.substring(4, input.length() - 4);

TelephonyManagerCompat.handleSecretCode(context, secretCode);

return true;

}

return false;

}

再看下 TelephonyManagerCompat.handleSecretCode 方法。

TelephonyManagerCompat#handleSecretCodepublic static void handleSecretCode(Context context, String secretCode) {

// Must use system service on O+ to avoid using broadcasts, which are not allowed on O+.

if (BuildCompat.isAtLeastO()) {

if (!TelecomUtil.isDefaultDialer(context)) {

LogUtil.e(

"TelephonyManagerCompat.handleSecretCode",

"not default dialer, cannot send special code");

return;

}

context.getSystemService(TelephonyManager.class).sendDialerSpecialCode(secretCode);

} else {

// System service call is not supported pre-O, so must use a broadcast for N-.

Intent intent =

new Intent(SECRET_CODE_ACTION, Uri.parse("android_secret_code://" + secretCode));

context.sendBroadcast(intent);

}

}

可以看到在拨号中接收到*#*##*#* 这样的指令时,程序会对外发送广播,这就意味着我们能够接收这个广播然后可以做我们想做的事情。

接下来我们看看这个接受广播代码是怎么写。

应用

首先在 AndroidManifest 文件中注册广播接收器。

android:name=".SecretCodeReceiver">

接收广播,启动应用。public class SecretCodeReceiver extends BroadcastReceiver {

@Override

public void onReceive(Context context, Intent intent) {

if (intent != null && SECRET_CODE_ACTION.equals(intent.getAction())){

Intent i = new Intent(Intent.ACTION_MAIN);

i.setClass(context, MainActivity.class);

i.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);

context.startActivity(i);

}

}

}

这样只要在拨号中输入*#*#1010#*#*就能启动相应的应用程序,OK,收功。

觉得文章不错的小伙伴帮忙点点赞加关注哦 ,有什么问题的话也欢迎大家前来探讨交流。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值