[Android实例]通话接通后震动提示

 


 

今天介绍一下在Android 中实现通话接通后震动提示, 这里通话主要是针对拨出的电话. 想要在接通的时候给出震动提示那么我们就需要知道通话何时是被接通的. 这样才能在进入该状态后给出提示. 但SDK 中并没有直接获得这种状态的方式. 首先我们看一下SDK 中电话服务类为我们提供了哪些通话状态.

SDK 中的TelephonyManager 类提供了3 种电话的状态.

CALL_STATE_IDLE

CALL_STATE_OFFHOOK

CALL_STATE_RINGING 

这几个状态很容易理解, 其中摘机状态即拿起话筒( 对于座机电话而言的动作), 但这个状态可能发生在拨入电话接通时, 也可能是拨出电话时. 但是却不能说明拨出电话接通时. 通过以上3 种状态我们仅能组合出: 挂机, 来电接通. 这两个状态. 而今天我们要实现的功能却无法做到.

看来我们需要寻找其他方法来实现了,SDK 靠不住啊……

 

还好Android 在运行时会有大量的log 产生, 看看我们能不能从这上面找到突波口呢. 我们选择Android 的Radio 模块的日志来分析. 首先我们需要写一段代码来读取Radio 相关的log. 读取log 就不得不用到logcat 了.

 

Process process;
		InputStream inputstream;
		BufferedReader bufferedreader;
		try {
			process = Runtime.getRuntime().exec("logcat -v time -b radio");
			inputstream = process.getInputStream();
			InputStreamReader inputstreamreader = new InputStreamReader(
					inputstream);
			bufferedreader = new BufferedReader(inputstreamreader);
	String str = "";
while ((str = bufferedreader.readLine()) != null) {
       log.i("mLogcat",str);
}
} catch (Exception e) {
			
		}

另外,要让程序能够读取系统log需要指定权限,在AndroidManifest.xml文件中加入一下内容.


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

通过上面这段代码我们就可以将Radio 的log 输出到了, 这样我们就可以通过在DDMS 中查看这些log, 分析其中的通话过程. 具体抓到的log 就不贴出来了. 大家可以自己编写程序通过上面的代码来抓取和分析. 我只说一下我的分析结果.

通过分析log 发现了一些蛛丝马迹. 其中有几条日志很有用.

GET_CURRENT_CALLS  id=1,DIALING

GET_CURRENT_CALLS  id=1,ALERTING

GET_CURRENT_CALLS  id=1,ACTIVE

由于log 较长我只拿了每条log 的开头部分, 真实的会多很多内容. 当我们拨出电话的时候, 会输入这么几条log.

拨号-> 提醒-> 活动.

 

大致是这么个过程. 经过几次测试发现, 电话接通时会进入活动状态, 并会输出:GET_CURRENT_CALLS  id=1,ACTIVE  这条log, 至此我们已经接近成功了.

不过之后我又发现在拨号开始到电话接通这段时间内会经过多次的

拨号-> 提醒-> 活动, 这样的状态变化, 仅当话筒中嘟声响起后GET_CURRENT_CALLS 这条日志会锁定在ALERTING.

GET_CURRENT_CALLS 日志了.

可能上面的这段表述大家不是很清楚,

GET_CURRENT_CALLS ACTIVE 这样的日志, 而仅有一次是电话接通产生的. 这就给我们造成了麻烦. 不能只是单纯的抓取GET_CURRENT_CALLS ACTIVE 这样的信息来判断了.

我们只能通过一些逻辑上的判断来实现了.下面看我的代码.

class TestThread implements Runnable {
	//振动器
	Vibrator mVibrator;
	//电话服务
	TelephonyManager telManager;
	public TestThread(Vibrator mVibrator, TelephonyManager telManager) {
		this.mVibrator = mVibrator;
		this.telManager = telManager;
	}
	@Override
	public void run() {
		//获取当前话机状态
		int callState = telManager.getCallState();
		Log.i("TestService", "开始.........." + Thread.currentThread().getName());
		//记录拨号开始时间
		long threadStart = System.currentTimeMillis();
		Process process;
		InputStream inputstream;
		BufferedReader bufferedreader;
		try {
			process = Runtime.getRuntime().exec("logcat -v time -b radio");
			inputstream = process.getInputStream();
			InputStreamReader inputstreamreader = new InputStreamReader(
					inputstream);
			bufferedreader = new BufferedReader(inputstreamreader);
			String str = "";
			long dialingStart = 0;
			boolean enableVibrator = false;
			boolean isAlert = false;
			while ((str = bufferedreader.readLine()) != null) {
				//如果话机状态从摘机变为空闲,销毁线程
				if (callState == TelephonyManager.CALL_STATE_OFFHOOK
						&& telManager.getCallState() == TelephonyManager.CALL_STATE_IDLE) {
					break;
				}
				// 线程运行5分钟自动销毁
				if (System.currentTimeMillis() - threadStart > 300000) {
					break;
				}
				Log.i("TestService", Thread.currentThread().getName() + ":"
						+ str);
				// 记录GSM状态DIALING
				if (str.contains("GET_CURRENT_CALLS")
						&& str.contains("DIALING")) {
					// 当DIALING开始并且已经经过ALERTING或者首次DIALING
					if (!isAlert || dialingStart == 0) {
						//记录DIALING状态产生时间
						dialingStart = System.currentTimeMillis();
						isAlert = false;
					}
					continue;
				}
				if (str.contains("GET_CURRENT_CALLS")
						&& str.contains("ALERTING")&&!enableVibrator) {
					
					long temp = System.currentTimeMillis() - dialingStart;
					isAlert = true;
					//这个是关键,当第一次DIALING状态的时间,与当前的ALERTING间隔时间在1.5秒以上并且在20秒以内的话
					//那么认为下次的ACTIVE状态为通话接通.
					if (temp > 1500 && temp < 20000) {
						enableVibrator = true;
						Log.i("TestService", "间隔时间....." + temp + "....."
								+ Thread.currentThread().getName());
					}
					continue;
				}
				if (str.contains("GET_CURRENT_CALLS") && str.contains("ACTIVE")
						&& enableVibrator) {
					mVibrator.vibrate(100);
					enableVibrator = false;
					break;
				}
			}
			Log.i("TestService", "结束.........."
					+ Thread.currentThread().getName());
		} catch (Exception e) {
			// TODO: handle exception
		}
	}
}

 

我的这个方法比较牵强,是通过判断第一次DIALING与每一次ALERTING之间的间隔,当间隔大于1.5秒.那么认为已经进入了"嘟"声提示 的时候了,那么下一个ACTIVE将是电话接通.这个1.5秒是通过分析日志得出的.但是这种方法我始终觉得不太靠谱.如果大家有好的方法可以交流交流.

剩下的就是让这个线程在电话拨出时触发,并且常驻在电话中时候准备这就可以了.可以采用Service配合Receiver来实现.Service来实现常驻,Receiver来实现监听拨出电话.基本就可以完成,我们想要的功能了.

以上代码我都测试过99%有效,哈哈.这里面提到了一些Android的基础内容像logcat,Service,Receiver.这些如果大家不了解的话可以到网上搜一下.有很多写的很不错的文章介绍Android的基础内容.我这里就不一一介绍了.

希望我写的东西对大家有所帮助.OK,就到这里吧.