android 监听电话去电接通瞬间[非来电]

时间:2022-07-23 16:44:52

首先感叹一下android强大,它可以把同一个apk里面得不同Actvity分别运行在不同的进程中,比如我想让自己的应用中Activity运行在Phone进程中,那么我需要做三个事情。

本文包含三个知识点:

1.监听去电接通     2.apk获取root权限 3. 运行时将apk push 到system/app


4.shareUid设置为phone   5.Android.mk设置为platform签名  6.android:process="com.android.phone"  6.重启手机生效

(1)

    <activity  
   android:process="com.android.phone"
                android:label="@string/app_name"  
                android:name=".AutoCallActivity" >  
                  
            </activity>  
(2)

    <manifest xmlns:android="http://schemas.android.com/apk/res/android"  
        package="com.spreadst.drag"  
        coreApp="true"  
android:sharedUserId="android.uid.system"
        android:versionCode="1"  

(3)
    LOCAL_PACKAGE_NAME := autocall  
    #LOCAL_CERTIFICATE := shared  
LOCAL_CERTIFICATE := platform 
为什么会有这个需求呢?TelephonyManager 中状态 只有  IDLE 空闲状态, 来电接通状态 ,拨打状态,没有去电接通状态,参考phone中得实现如下:
Phone  phone = PhoneFactory.getDefaultPhone();
这个调用是不能运行在phone以外的进程中的,不信你可以试一下,请记住一个进程的主线程的looper是唯一的

原因如下:

if (sLooper != Looper.myLooper()) {
            throw new RuntimeException(
                "PhoneFactory.getDefaultPhone must be called from Looper thread");
        }
去电接通若干秒之后挂断代码如下,由于使用了hide class,所以必须在android源代码底下编译,当然你也可以通过反射、AIDL、class.jar等方式解决这个问题。

CallManager mCM = CallManager.getInstance();  
   Phone  phone = PhoneFactory.getDefaultPhone();  
  
mCM.registerPhone(phone);  
       mCM.registerForPreciseCallStateChanged(mHandler, PHONE_STATE_CHANGED, null); 

    private Handler mHandler=new Handler(){  
      
            public void handleMessage(android.os.Message msg) {  
                switch (msg.what) {  
                case PHONE_STATE_CHANGED:  
                    updatePhoneSateChange();  
                    break;  
                default:  
                    break;  
                }  
            };  
              
        };  

    private void updatePhoneSateChange(){  
            Call fgCall = mCM.getActiveFgCall();  
            if (mCM.hasActiveRingingCall()) {  
                    fgCall = mCM.getFirstActiveRingingCall();  
              }  
      
            final Call.State state = fgCall.getState();  
            switch (state) {  
                case IDLE:  
                    break;  
      
                case ACTIVE://去电接通  
                    Log.d("yzy","ACTVIE");  
                    final Timer timer = new Timer();  
                    if(mode == Mode.mode2){  
                        timer.schedule(new TimerTask() {  
                            @Override  
                            public void run() {  
                                try {  
                                    Log.d("yzy", "endcall()");  
      
                                    mITelephony.endCall();  
                                    timer.cancel();  
                                } catch (RemoteException e) {  
                                    e.printStackTrace();  
                                }  
                            }  
                        }, holdonDuration * 1000, holdonDuration * 1000);  
                    }  
                    //mITelephony.endCall();  
                    break;  
                default:  
                    break;  
            }  
              
        }  

如果你将这个apk 简单的install 会报告如下错误:不存在该共享用户或签名不匹配,这是因为
android:sharedUserId="android.uid.system"

必须放到system/app目录下才能工作。这就由牵扯出了两个问题,临时获取root权限与运行时拷贝.

其他关键代码:

private ITelephony mITelephony;
mITelephony = ITelephony.Stub.asInterface(ServiceManager
                .getService(Context.TELEPHONY_SERVICE));
        mPhoneStateListener = getPhoneStateListener();
        ((TelephonyManager) getSystemService(Context.TELEPHONY_SERVICE)).listen(
                mPhoneStateListener, PhoneStateListener.LISTEN_CALL_STATE);
private PhoneStateListener getPhoneStateListener() {

        return new PhoneStateListener() {
            @Override
            public void onCallStateChanged(int state, String incomingNumber) {
                try {
                     Log.d("yzy"," state: "
                     + mITelephony.getCallState());

                } catch (RemoteException e) {
                    e.printStackTrace();
                }

            }
        };

    }