美文网首页
类似钉钉打卡功能

类似钉钉打卡功能

作者: iOS程序媛ing | 来源:发表于2021-11-12 17:26 被阅读0次

    我们的打卡功能分为地理位置打卡和Wi-Fi打卡
    1、地理位置打卡:用户必须到达指定的地理围栏内打卡,才能正常打卡成功,否则为外勤打卡。
    2、Wi-Fi打卡:用户必须连接到指定Wi-Fi,才能打卡成功
    当然还会有外勤打卡的情况:没有在指定地理围栏内,并且没有连接指定Wi-Fi。

    一、地理位置打卡,就不得不提到地理围栏的概念

    我们使用的是腾讯定位SDK 腾讯定位SDK开发文档

    第1步,引入头文件

    在调用地理围栏功能的类中引入AMapFoundationKit.h和AMapLocationKit.h这两个头文件

    #import <AMapFoundationKit/AMapFoundationKit.h>
    
    #import <AMapLocationKit/AMapLocationKit.h>
    
    第 2 步,配置Key

    在调用定位时,需要添加Key,需要注意的是请在 SDK 任何类的初始化以及方法调用之前设置正确的 Key。

    如果您使用的是定位SDK v2.x版本需要引入基础 SDK AMapLocationKit.framework ,设置apiKey的方式如下:

    iOS 定位SDK v2.x版本设置 Key:

    [AMapServices sharedServices].apiKey =@"您的key";
    
    第 3 步:创建地理围栏

    地理围栏没有最大个数限制,您可以无限制的创建围栏。但请您根据业务需求合理的创建围栏,控制围栏个数可以有效的保证程序执行效率。定位 SDK 提供根据高德POI、行政区划,自定义圆形、自定义多边形四种方式创建地理围栏。
    1、初始化地理围栏管理manager

    self.geoFenceManager = [[AMapGeoFenceManager alloc] init];
    self.geoFenceManager.delegate = self; 
    self.geoFenceManager.activeAction = AMapGeoFenceActiveActionInside | AMapGeoFenceActiveActionOutside | AMapGeoFenceActiveActionStayed; //设置希望侦测的围栏触发行为,默认是侦测用户进入围栏的行为,即AMapGeoFenceActiveActionInside,这边设置为进入,离开,停留(在围栏内10分钟以上),都触发回调
    self.geoFenceManager.allowsBackgroundLocationUpdates = YES;  //允许后台定位
    

    2、创建高德POI地理围栏
    提供两个创建高德POI围栏的接口,一个是根据关键字创建POI围栏,另一个是根据经纬度进行周边搜索创建POI围栏。
    根据关键字创建围栏:

    - (void)addKeywordPOIRegionForMonitoringWithKeyword:(NSString *)keyword POIType:(NSString *)type city:(NSString *)city size:(NSInteger)size customID:(NSString *)customID;
    

    示例代码

    [self.geoFenceManager addKeywordPOIRegionForMonitoringWithKeyword:@"北京大学" POIType:@"高等院校" city:@"北京" size:20 customID:@"poi_1"];
    

    根据周边POI创建围栏:

    - (void)addAroundPOIRegionForMonitoringWithLocationPoint:(CLLocationCoordinate2D)locationPoint aroundRadius:(NSInteger)aroundRadius keyword:(NSString *)keyword POIType:(NSString *)type size:(NSInteger)size customID:(NSString *)customID;
    

    示例代码

    CLLocationCoordinate2D coordinate = CLLocationCoordinate2DMake(39.908692, 116.397477); //天安门
    [self.geoFenceManager addAroundPOIRegionForMonitoringWithLocationPoint:coordinate aroundRadius:10000 keyword:@"肯德基" POIType:@"050301" size:20 customID:@"poi_2"];
    

    3、创建行政区域围栏
    根据行政区域关键字创建行政区域围栏:

    - (void)addDistrictRegionForMonitoringWithDistrictName:(NSString *)districtName customID:(NSString *)customID;
    

    示例代码

    [self.geoFenceManager addDistrictRegionForMonitoringWithDistrictName:@"海淀区" customID:@"district_1"];
    

    4、创建自定义圆形围栏
    需要提供中心点和半径来创建圆形围栏,一次创建一个

    - (void)addCircleRegionForMonitoringWithCenter:(CLLocationCoordinate2D)center radius:(CLLocationDistance)radius customID:(NSString *)customID;
    

    示例代码

    CLLocationCoordinate2D coordinate = CLLocationCoordinate2DMake(39.908692, 116.397477); //天安门
    [self.geoFenceManager addCircleRegionForMonitoringWithCenter:coordinate radius:300 customID:@"circle_1"];
    

    5、创建自定义多边形围栏
    根据经纬度坐标数据添加一个闭合的多边形围栏,点与点之间按顺序尾部相连, 第一个点与最后一个点相连,一次创建一个

    - (void)addPolygonRegionForMonitoringWithCoordinates:(CLLocationCoordinate2D *)coordinates count:(NSInteger)count customID:(NSString *)customID;
    

    示例代码

    NSInteger count = 4;
    CLLocationCoordinate2D *coorArr = malloc(sizeof(CLLocationCoordinate2D) * count);
    coorArr[0] = CLLocationCoordinate2DMake(39.933921, 116.372927);     //平安里地铁站
    coorArr[1] = CLLocationCoordinate2DMake(39.907261, 116.376532);     //西单地铁站
    coorArr[2] = CLLocationCoordinate2DMake(39.900611, 116.418161);     //崇文门地铁站
    coorArr[3] = CLLocationCoordinate2DMake(39.941949, 116.435497);     //东直门地铁站
    [self.geoFenceManager addPolygonRegionForMonitoringWithCoordinates:coorArr count:count customID:@"polygon_1"];
        
    free(coorArr);
    coorArr = NULL;
    
    第 4 步,开始定位

    当围栏创建完毕,且围栏创建成功时会启动定位,这部分无需您来设置,SDK内部执行。 定位机制:通过“远离围栏时逐渐降低定位频率”来降低电量消耗,“离近围栏时逐渐提高定位频率”来保证有足够的定位精度从而完成围栏位置检测。
    需要注意,在iOS9及之后版本的系统中,如果您希望程序在后台持续检测围栏触发行为,需要保证manager的allowsBackgroundLocationUpdates为YES,设置为YES的时候必须保证 Background Modes 中的 Location updates 处于选中状态,否则会抛出异常。
    第 5 步,获取围栏创建后的回调和围栏状态改变时的回调

    创建围栏后的信息和围栏状态改变时的信息均会通过AMapGeoFenceManagerDelegate进行回调,设置内容如下:

    self.geoFenceManager.delegate = self;
    

    1、获取围栏创建后的回调----在这里可以添加定位范围是图(高德demo)
    在如下回调中知道创建的围栏是否成功,以及查看所创建围栏的具体内容。

    - (void)amapGeoFenceManager:(AMapGeoFenceManager *)manager didAddRegionForMonitoringFinished:(NSArray<AMapGeoFenceRegion *> *)regions customID:(NSString *)customID error:(NSError *)error {
        if (error) {
            NSLog(@"创建失败 %@",error);
        } else {
            NSLog(@"创建成功");
        }
    }
    

    2、围栏状态改变时的回调在这里,可以获取围栏状态(在围栏内、在围栏外、在围栏内停留超过10分钟等状态)

    在如下回调中知道围栏的状态是否发生改变,或者定位是否失败。围栏的状态表示的就是用户和围栏的关系,有未知、进入围栏、退出围栏、在围栏内停留。回调触发的条件需同时满足:1.围栏的状态从A变成B;2.B符合您在第一步设置的需要侦测的行为的范围内。当然如果self.geoFenceManager.activeAction在监听的过程中改变了,所有符合侦测范围的围栏即使状态没有改变也会再次触发回调。

    - (void)amapGeoFenceManager:(AMapGeoFenceManager *)manager didGeoFencesStatusChangedForRegion:(AMapGeoFenceRegion *)region customID:(NSString *)customID error:(NSError *)error {
        if (error) {
            NSLog(@"status changed error %@",error);
        }else{
            NSLog(@"status changed success %@",[region description]);
        }
    }
    

    3、获取用户是否开启允许定位的权限
    iOS 13以前使用这个方法获取

    /**
    *  @brief 定位权限状态改变时回调函数。注意:iOS13及之前版本回调
    *  @param manager 定位 AMapLocationManager 类。
    *  @param status 定位权限状态。
    */
    - (void)amapLocationManager:(AMapLocationManager *)manager didChangeAuthorizationStatus:(CLAuthorizationStatus)status;
    

    iOS 14以后使用这个方法获取

    /**
     *  @brief 定位权限状态改变时回调函数。注意:iOS14及之后版本回调
     *  @param manager 定位 AMapLocationManager 类。
     *  @param locationManager  定位CLLocationManager类,可通过locationManager.authorizationStatus获取定位权限,通过locationManager.accuracyAuthorization获取定位精度权限
     */
    - (void)amapLocationManager:(AMapLocationManager *)manager locationManagerDidChangeAuthorization:(CLLocationManager*)locationManager;
    
    

    用户是否允许开启定位的状态------CLAuthorizationStatus

    typedef NS_ENUM(int, CLAuthorizationStatus) {
          kCLAuthorizationStatusNotDetermined = 0,  // 用户未授权,即还未弹出OS的授权弹窗
          kCLAuthorizationStatusDenied, // 用户拒绝定位权限,包括拒绝App或者全局开关关闭
          kCLAuthorizationStatusRestricted, // 定位服务受限,该状态位用户无法通过设置页面进行改变
          kCLAuthorizationStatusAuthorizedAlways, // 始终定位,即后台定位
          kCLAuthorizationStatusAuthorizedWhenInUse, // App使用的时候,允许定位
          kCLAuthorizationStatusAuthorized, // iOS8.0之后已经被废弃
    };
    

    最后,移除围栏

    当不再需要使用围栏时,可以调用以下几个函数对已经设定的围栏进行移除操作。

    - (void)removeTheGeoFenceRegion:(AMapGeoFenceRegion *)region; //移除指定围栏
    - (void)removeGeoFenceRegionsWithCustomID:(NSString *)customID; //移除指定customID的围栏
    - (void)removeAllGeoFenceRegions;  //移除所有围栏
    

    遇到的问题

    1、使用定位功能,如果想使用后台定位功能,但是没有开启权限的情况下,会出现如下错误如下


    截屏2021-11-10 上午10.24.52.png

    解决方案:

    方案一:

    一种方法是处在拥有前台定位权限的情况下:

    设置CLLocationManager的allowsBackgroundLocationUpdates为YES
    

    然后打开Xcode -> Targets -> Capabilities 中的Background Modes并勾选其中的Location updates选项

    截屏2021-11-10 上午10.31.17.png
    原文链接:https://blog.csdn.net/feosun/article/details/77097086
    方案二:

    在.infoPlist文件中添加如下操作即可


    截屏2021-11-10 上午10.24.41.png

    二、wifi打卡

    当前连接Wi-Fi的mac地址和Wi-Fi名称要和考勤Wi-Fi一致,以下是获取Wi-Fi Mac地址和Wi-Fi名称的方法
    获取Wi-Fi名称

    + (NSString *)wifiName
    {
        NSArray *ifs = CFBridgingRelease(CNCopySupportedInterfaces());
        id info = nil;
        for (NSString *ifname in ifs) {
            info = (__bridge_transfer id)CNCopyCurrentNetworkInfo((CFStringRef) ifname);
            if (info && [info count]) {
                break;
            }
        }
        NSDictionary *dic = (NSDictionary *)info;
        NSString *ssid = [[dic objectForKey:@"SSID"] lowercaseString];
        
        return ssid;
    }
    

    获取Wi-Fi Mac地址

    + (NSString *)wifiMac
    {
        NSArray *ifs = CFBridgingRelease(CNCopySupportedInterfaces());
        id info = nil;
        for (NSString *ifname in ifs) {
            info = (__bridge_transfer id)CNCopyCurrentNetworkInfo((CFStringRef) ifname);
            if (info && [info count]) {
                break;
            }
        }
        NSDictionary *dic = (NSDictionary *)info;
        NSString *bssid = [dic objectForKey:@"BSSID"];
        return bssid;
    }
    

    只写上面两个方式是不能获取到Wi-Fi的mac地址和名称的。还需要开启Wi-Fi权限
    Xcode -> [Project Name] -> Targets -> [Target Name] -> Capabilities -> Access WiFi Information -> ON


    截屏2021-11-16 下午4.56.22.png

    遇到的问题

    1、iOS 获取到的Wi-Fi Mac地址和安卓获取到的不一样,
    假设mac地址为 12:03:45:67:89:12
    iOS. 获取到的结果回事12:3:45:67:89:12,会少一个0,因为iOS 默认情况下会省略首位为0 的情况

    解决方案:自己拼接

    + (NSString *)wifiMac
    {    
        NSString *ssid = nil;
        NSArray *ifs = (__bridge_transfer id)CNCopySupportedInterfaces();
        for (NSString *ifnam in ifs) {
            NSDictionary *info = (__bridge_transfer id)CNCopyCurrentNetworkInfo((__bridge CFStringRef)ifnam);
            if (info[@"BSSID"]) {
                ssid = info[@"BSSID"];
            }
        }
        NSArray *ssidArray = [ssid componentsSeparatedByString:@":"];
        NSMutableArray *newArray = [NSMutableArray arrayWithArray:ssidArray];
        for (NSInteger i = 0; i < ssidArray.count; i++) {
            NSString *value = ssidArray[i];
            if (value.length == 1) {
                value = [NSString stringWithFormat:@"0%@", value];
            }
            newArray[i] = value;
        }
        NSMutableString *newSSID = [NSMutableString string];
        for (NSInteger i = 0; i < newArray.count; i ++) {
            [newSSID appendString:newArray[i]];
            if (i != newArray.count - 1) {
                [newSSID appendString:@":"];
            }
            
        }
        ssid = [newSSID uppercaseString];
        return ssid;
         
    }
    

    这样,问题就解决了

    相关文章

      网友评论

          本文标题:类似钉钉打卡功能

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