iOS蓝牙4.0,后台监听来电

在上一篇文章《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文件中申请相应的权限。如下图:

iOS蓝牙4.0,后台监听来电_第1张图片
申请蓝牙权限.png

二、怎样在后台时,监听来电呢?

如果你在定时执行的commTimer方法中,执行callEventHandler监听方法,会发现依然监听不到来电。
仔细看苹果提供的CTCallCenter类,会发现还有一个属性变量currentCalls

/*
 *  CTCallCenter.h
 *  CFTelephony
 *
 *  Copyright 2010 Apple, Inc. All rights reserved.
 *
 */

#import 
#import 

@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 *currentCalls __OSX_AVAILABLE_BUT_DEPRECATED_MSG(__MAC_NA, __MAC_NA, __IPHONE_4_0, __IPHONE_10_0, "Replaced by ");

/*
 * 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 ");

@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开发的伙伴们有所帮助,有帮助的点个赞吧!哈哈!!

你可能感兴趣的:(iOS蓝牙4.0,后台监听来电)