android之来电自动拒接并自动回复短信_上课模式app「建议收藏」

上课的时候老师说总是错过电话,对方打来没人接还一遍遍的打,觉得可以有个app在上课期间自动拒接电话,并自动回复短信过去.当然了,需要权限的.尝试做了个雏形出来.界面如下:主要代码如下:package jason.teacher;import java.lang.reflect.Method;import java.util.HashMap;import ja

大家好,又见面了,我是全栈君。

上课的时候老师说总是错过电话,对方打来没人接还一遍遍的打,觉得可以有个app在上课期间自动拒接电话,并自动回复短信过去.

当然了,需要权限的.

尝试做了个雏形出来.

界面如下:

android之来电自动拒接并自动回复短信_上课模式app「建议收藏」

主要代码如下:

package jason.teacher;

import java.lang.reflect.Method;
import java.util.HashMap;
import java.util.List;

import android.app.Activity;
import android.content.Context;
import android.content.SharedPreferences;
import android.content.SharedPreferences.Editor;
import android.os.Bundle;
import android.telephony.PhoneStateListener;
import android.telephony.SmsManager;
import android.telephony.TelephonyManager;
import android.util.Log;
import android.view.View;
import android.view.View.OnClickListener;
import android.widget.Button;
import android.widget.EditText;
import android.widget.TextView;

import com.android.internal.telephony.ITelephony;

public class MainActivity extends Activity {
	
    public final static String TAG = "jason.com";
    TelephonyManager tpm;
    String num;//存储来电号码
    Button end;//退出按钮
    EditText sms;//回复短信的内容编辑框
    SharedPreferences sp;
    int count = 0;//来电总数
    int peo = 0;//来电的号码个数,跟来电总数有区别,这个不包括重复来电,
    TextView counttext;//拦截数量通知的显示
    HashMap<String, String> numMap;//用来存储来电号码
    @Override
    public void onCreate(Bundle savedInstanceState) {  
        super.onCreate(savedInstanceState);  
        setContentView(R.layout.activity_main);
        sp = this.getSharedPreferences("SP", MODE_PRIVATE);
        counttext = (TextView) findViewById(R.id.count);
        sms = (EditText) findViewById(R.id.sms);
        end = (Button) findViewById(R.id.start);
        numMap = new HashMap<String, String>();
        if(sp.getString("sms", null) != null){
        	sms.setText(sp.getString("sms", "我现在正在上课,一会儿下课了联系你"));
        }
        tpm = (TelephonyManager) this.getSystemService(Context.TELEPHONY_SERVICE);//获取电话通讯服务
         tpm.listen(new MyPhoneStateListener(),PhoneStateListener.LISTEN_CALL_STATE);//给电话服务增加状态监听器,监听来电,通话,挂断等状态
        end.setOnClickListener(new OnClickListener() {
			@Override
			public void onClick(View v) {
				Editor editor = sp.edit();
				editor.putString("sms", sms.getText().toString());
				editor.commit();//这里是默认自动保存用户编辑过的回复短信内容的,
				finish();
			}
		});
    }
    class MyPhoneStateListener extends PhoneStateListener {
    	@Override
        public void onCallStateChanged(int state, String incomingNumber) {
    		num = incomingNumber;
            switch(state) {
            case TelephonyManager.CALL_STATE_IDLE: //空闲
                break;
            case TelephonyManager.CALL_STATE_RINGING: //来电
            	endCall();//自动挂断
            	if(!numMap.containsKey(num)){//如果之前没有来电,则把这个号码加入已经来电过的列表
            		sendMes();
            		numMap.put(num, null);
            		peo ++;
            		updateUi();//更新来电数目
            	}
            	break;
            case TelephonyManager.CALL_STATE_OFFHOOK: //摘机(正在通话中)
                break;
            }
        }
    	private void updateUi(){
    		if(count > 0){
    			counttext.setVisibility(View.VISIBLE);
    		}
    		counttext.setText("已拒接" + count + "个来电,共" + peo +"个人联系过您,请到通话记录查看");
    	}
    	private void endCall()//估计这里是唯一有点难度的,用到了java的反射
        {  
            Class<TelephonyManager> c = TelephonyManager.class;           
            try
            {  
                Method getITelephonyMethod = c.getDeclaredMethod("getITelephony", (Class[]) null);  
                getITelephonyMethod.setAccessible(true);  
                ITelephony iTelephony = null;
                iTelephony = (ITelephony) getITelephonyMethod.invoke(tpm, (Object[]) null);  
                iTelephony.endCall();
                count ++;
                updateUi();
            }  
            catch (Exception e)  
            {  
                Log.e(TAG, "Fail to answer ring call.", e);  
            }          
        }  
    	
    	private void sendMes(){
    		//直接调用短信接口发短信
    		SmsManager smsManager = SmsManager.getDefault();
    		List<String> divideContents = smsManager.divideMessage(sms.getText().toString()); 
    		for (String text : divideContents) {    
    			smsManager.sendTextMessage(num, null, text, null, null);  
    		}
    	}
    }
}

这里解释一下,在android在1.1版本后就已经把Phone类的相关API给隐藏起来了,想要用代码实现挂断电话的功能,就必须通过AIDL才行,然后利用反射来使用其方法。

第一步:在程序中新建一个包,包名必须为:com.android.internal.telephony,因为要使用aidl。

第二步:在这个包里面新建一个名为ITelephony.aidl的文件,然后在文件里面写入代码:

package com.android.internal.telephony;
	interface ITelephony{
		boolean endCall();
		void answerRingingCall();
	}

这样就可以通过如下代码进行进一步获取拒接电话的api的操作了.

 

private void endCall()
        {  
            Class<TelephonyManager> c = TelephonyManager.class;           
            try
            {  
                Method getITelephonyMethod = c.getDeclaredMethod("getITelephony", (Class[]) null);  
                getITelephonyMethod.setAccessible(true);  
                ITelephony iTelephony = null;
                iTelephony = (ITelephony) getITelephonyMethod.invoke(tpm, (Object[]) null);  
                iTelephony.endCall();
                count ++;
                updateUi();
            }  
            catch (Exception e)  
            {  
                Log.e(TAG, "Fail to answer ring call.", e);  
            }          
        }  

还有就是别忘了加权限,否则是一点效果都没有的.

<uses-permission android:name="android.permission.READ_PHONE_STATE" />  
	<uses-permission android:name="android.permission.CALL_PHONE"/>
	<uses-permission android:name="android.permission.SEND_SMS"/>

不过有一点不解的是在这里没有用到打电话的权限,但是如果不加

<uses-permission android:name="android.permission.CALL_PHONE"/>

的话,来电话的时候竟然不会进行挂断操作,尝试了好几次都是如此,最后只能再把这个权限加上,虽然在我看来完全没用,安装的时候还会让人觉得你这个应用会不会偷偷打电话啊.

当然了,这个完全可以做成后台服务的形式,不过老师就是上课才用,也没那么多要求,如果继续丰富一下,还是比较实用的.

作者:jason0539

博客:http://blog.csdn.net/jason0539(转载请说明出处)

扫码关注我微信公众号

android之来电自动拒接并自动回复短信_上课模式app「建议收藏」

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/121400.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • 如何引用jstl标签

    如何引用jstl标签第一个,直接用下面这个,一个就够了,不需要别的什么standard,不过比较旧,06年的<!–https://mvnrepository.com/artifact/jstl/jstl–><dependency><groupId>jstl</groupId><artifactId>jstl</artifactId><version>1.2</version></de

    2025年6月24日
    5
  • Uniapp中onShow()的应用

    Uniapp中onShow()的应用遇到问题:在使用一些变量进行判断时,用完一次开始下一次判断时,结果会跟前一次一样,比如门禁中第一个房子打开后,切换到另外的房子,结果返回结果跟第一个房子一样。原因分析:用于使用了相同的变量进行判断,然而在第一次执行完后,并没有把变量重新初始化,导致页面在第二次加载的时候,显示的结果跟第一次一样。解决方法:在onShow()里面,通过一些参数,判断在切换页面的时候,有没有切换房屋,如果房屋没变的话,切换页面时就不用重新初始化变量。如果房屋改变了,就把变量重新初始化。总结:.

    2022年6月21日
    105
  • IDEA 2018 2020 2021 2022 各版本对Maven版本兼容问题汇总「建议收藏」

    IDEA 2018 2020 2021 2022 各版本对Maven版本兼容问题汇总「建议收藏」IDEA2022兼容maven3.8.1及之前的所用版本IDEA2021兼容maven3.8IDEA2018202020212022各版本对Maven版本兼容问题汇总Maven3.6.3版本兼容问题错误信息如下:1.IDEA2021兼容maven3.8.1及之前的所用版本2.IDEA2020兼容Maven3.6.3及之前所有版本3.IDEA2018兼容Maven3.6.1及之前所有版本IDEA2022兼容maven3.8.1及之前的所用版本

    2022年8月22日
    81
  • tensorflow pycharm教程_tensorflow支持python3.8吗

    tensorflow pycharm教程_tensorflow支持python3.8吗pycharm使用tensorflow流程最近在学人工智能与大数据管理,环境是python+tensorflow。但配置有些麻烦,记录一下。其实主要分为两个部分,配置tnsorflow和在pycharm中使用tensorflow。首次尝试平常安装python包都是去pycharm的setting里面,在设置ProjectInterpreter中点小加号去装的,但这次却报了错。想来应该是有些…

    2022年8月25日
    5
  • js后退按钮事件触发

    js后退按钮事件触发if(window.history&&window.history.pushState){$(window).on(‘popstate’,function(){varhashLocation=location.hash;varhashSplit=hashLocation.split(“#!/”);…

    2022年7月25日
    15
  • cs模式与bs的区别_BS架构是CS架构的替代品

    cs模式与bs的区别_BS架构是CS架构的替代品C/S:又称Client/Server或客户/服务器模式。客户端需要安装专用的客户端软件。 能充分发挥客户端PC的处理能力,,很多工作可以在客户端处理后再提交给服务器。C/S的主要特点是交互性强、具有安全的存取模式、网络通信量低、响应速度快、利于处理大量数据。但是该结构的程序是针对性开发,变更不够灵活,维护和管理的难度较大。通常只局限于小型局域网,不利于扩展。B/S是Browe

    2025年10月11日
    7

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号