在上一篇文章《iOS蓝牙4.0收发数据设计》,有简单的介绍收发数据设计。
在iOS蓝牙4.0开发过程中,肯定还有许多伙伴,会遇到监听后台来电的需求。
那么iOS下,如果监听后台来电呢?
其实苹果官方CoreTelephony.framework中已经提供了相应监听来电方法。
来看看系统提供的方法:
CTCallCenter *center = [[CTCallCenter alloc] init];
center.callEventHandler = ^(CTCall *call) {
if ([call.callState isEqualToString:CTCallStateIncoming]) {
//来电
}else if ([call.callState isEqualToString:CTCallStateConnected]) {
//接通
}else if ([call.callState isEqualToString:CTCallStateDisconnected]) {
//挂断
}else if ([call.callState isEqualToString:CTCallStateDialing]) {
//正在通话
}else {
//其他
}
};
如果你照搬上面的代码,会发现不能监听来电,因为在ARC模式下,center创建后,就要被释放了。所以,如果要监听来电,必须把center对象定义为全局或者被某个对象引用,不要被释放掉,或者再不需要监听来电后,再释放掉。
好,即使你采用了我上面的建议,最后还是发现有些不太完美。
上面的方法的确能够监听来电了,但是只能在APP运行时,才能够正常监听;如果APP进入后台,就再也监听不到了。尴尬!!!
重点来了,我怎么才能让后台监听来电呢?
我们先把问题分解,上面的需求可以分解为两个。一个是后台运行;另一个是监听来电。
一、怎么才能后台运行呢?
苹果官方给出以下可以申请后台运行:
1.Audio and AirPlay(音频播放)
2.Location updates(位置更新)
3.Voice over IP(voip)
4.Newsstand downloads
5.External accessory communication
6.Uses Bluetooth LE accessories
7.Acts as a Bluetooth LE accessory
8.Background fetch
9.Remote notifications
以上9种,都可以申请后台运行。很好,因为我们的蓝牙4.0开发,也可以申请后台运行。
那么怎么才能做到我们蓝牙4.0相关的APP能够后台运行呢?我给出的做法,是使用定时器。
1、定义一个属性变量定时器。定义成属性变量,方便我们在不使用时,进行销毁。比如APP进入前台时。
@property (nonatomic, strong) NSTimer *phoneTimer;
2、定时器
if (self.phoneTimer == nil) {
self.phoneTimer = [NSTimer scheduledTimerWithTimeInterval:3.0 target:self selector:@selector(commTimer:) userInfo:nil repeats:YES];
}
3、发送蓝牙数据。(如果实际当中,我们并不需要后台发送数据时,可以发送一些无意义的数据,只是达到后台运行的目的)
- (void)commTimer:(NSTimer *)timer {
//发送蓝牙数据
}
好啦,上面三步就可以完成蓝牙4.0后台运行。当然不要忘记了在plist文件中申请相应的权限。如下图:
申请蓝牙权限.png二、怎样在后台时,监听来电呢?
如果你在定时执行的commTimer
方法中,执行callEventHandler
监听方法,会发现依然监听不到来电。
仔细看苹果提供的CTCallCenter
类,会发现还有一个属性变量currentCalls
。
/*
* CTCallCenter.h
* CFTelephony
*
* Copyright 2010 Apple, Inc. All rights reserved.
*
*/
#import <Foundation/Foundation.h>
#import <CoreTelephony/CoreTelephonyDefines.h>
@class CTCall;
NS_ASSUME_NONNULL_BEGIN
CORETELEPHONY_CLASS_AVAILABLE(4_0)
@interface CTCallCenter : NSObject
/*
* currentCalls
*
* Discussion:
* An array containing CTCall objects for all calls that are currently
* in progress. If no calls are active, this will be nil.
*
*/
@property(readonly, retain, nullable) NSSet<CTCall*> *currentCalls __OSX_AVAILABLE_BUT_DEPRECATED_MSG(__MAC_NA, __MAC_NA, __IPHONE_4_0, __IPHONE_10_0, "Replaced by <CallKit/CXCallObserver.h>");
/*
* callEventHandler
*
* Discussion:
* A block that will be dispatched on the default priority global dispatch
* queue when a new call event occurs. Set this property to a block
* that is defined in your application to handle call events.
*/
@property(nonatomic, copy, nullable) void (^callEventHandler)(CTCall*) __OSX_AVAILABLE_BUT_DEPRECATED_MSG(__MAC_NA, __MAC_NA, __IPHONE_4_0, __IPHONE_10_0, "Replaced by <CallKit/CXCallObserver.h>");
@end
NS_ASSUME_NONNULL_END
对,我们就从这个变量入手。来重写commTimer
方法。代码示意如下:
/**
后台发送蓝牙数据、监听来电
*/
- (void)commTimer:(NSTimer *)timer {
//发送蓝牙数据
if ([[BTManager shareManager] isConnect]) {
[[BTManager shareManager] sendBackgroudData];
//监听来电
CTCallCenter *center = [[CTCallCenter alloc] init];
if (center.currentCalls) {
CTCall *call = [center.currentCalls anyObject];
if ([call.callState isEqual:CTCallStateIncoming]) {
isPhone = YES;
[[BTManager shareManager] phoneCallIn];
}else if ([call.callState isEqual:CTCallStateDisconnected]) {
if (isPhone) {
isPhone = NO;
[[BTManager shareManager] phoneCallEnd];
}
}else {
}
}
}
}
BTManager
是我设计的一个蓝牙管理类,用来管理收发蓝牙数据的。
isConnect
方法是判断当前是否有蓝牙连接,当然你也可以设计中,当蓝牙断开连接时,就停止定时器。
sendBackgroudData
发送是数据,保持蓝牙后台长连接。
以上就是我在iOS蓝牙4.0后台监听来电的解决方案。希望对蓝牙4.0开发的伙伴们有所帮助,有帮助的点个赞吧!哈哈!!
网友评论
搞蓝牙后台,快哭了
我描述下我的问题(很奇葩)
1.在主界面会执行几次,然后就停了
2.这时候锁屏,又会执行几次,然后又停了
3.按home键开屏,又执行几次,又挺了
总结:好像是执行几次就挂起了,然后需要换状态(从主界面到锁屏)来唤醒
很奇葩,完全不知道怎么弄了