iOS蓝牙4.0,后台监听来电

作者: 平原河流 | 来源:发表于2016-11-14 15:53 被阅读1050次

    在上一篇文章《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开发的伙伴们有所帮助,有帮助的点个赞吧!哈哈!!

    相关文章

      网友评论

      • Corbin___:按着楼主做的,进入后台定时器1s执行一次,然后执行7、8次就不执行了,相当被挂起了,还是没法达到驻留后台的目标
        搞蓝牙后台,快哭了
        Corbin___:@平原河流 刚试了下,使用gcdtimer异步发送心跳数据的
        我描述下我的问题(很奇葩)
        1.在主界面会执行几次,然后就停了
        2.这时候锁屏,又会执行几次,然后又停了
        3.按home键开屏,又执行几次,又挺了
        总结:好像是执行几次就挂起了,然后需要换状态(从主界面到锁屏)来唤醒

        很奇葩,完全不知道怎么弄了
        平原河流:@陈泽槟 进入后台后,你要定时发心跳数据哦
      • Corbin___:还有就是有没有代码,我不知道后台发送蓝牙数据,监听来电这个代码写哪里
        平原河流:定时器的方法,你可以写到- (void)applicationDidEnterBackground:(UIApplication *)application;方法中。
      • Corbin___:plist文件的使用蓝牙,那个我怎么打不出来,可以写全部出来看看嘛
        平原河流:在plist中,新增row,选择Privacy - Bluetooth Peripheral Usage Description,就好。
      • zjw7sky:为什么不直接用ancs呢
      • Maginawin:楼主可以了解一下 ANCS,你这个方法也可以,就是比较耗电。
        Maginawin:@平原河流 是的,一般的蓝牙模块都支持 ANCS,这样不用启动你的应用都可以监听手机的通知,包括电话短信微信QQ之类的
        平原河流:@Maginawin 刚看了,这个需要外设硬件实现ANCS协议,对吧?
        平原河流:@Maginawin 好的,多谢!等下试试看。
      • 达若漠沙:楼主大牛,请问一下,app处于后台,通过蓝牙信号强度自动连接合适的外设需要哪些步骤?请指点一下小菜鸟吧😄~~先谢谢啦😜
        平原河流:@红天石 这样就有问题了,因为你没办法在后台长时执行搜索方法。
        达若漠沙:是app进入后台,被托托管着的时候~:blush:
        平原河流:@红天石 首先,你说的APP后台,是指程序运行时,后台线程;还是程序后台托管?

      本文标题:iOS蓝牙4.0,后台监听来电

      本文链接:https://www.haomeiwen.com/subject/fzphpttx.html