定位-代理模式到block模式的转换

  • 主要思想就是,先记录下外界传递过来的block,
  • 然后在对应的代理方法里面执行这个block;

import UIKit
import CoreLocation

typealias RequestBlock = (location : CLLocation?, error : String?) -> ()

class LocationTool: NSObject {

    // MARK:- 懒加载
    private lazy var locationM : CLLocationManager = {
        let locationM = CLLocationManager()
        locationM.delegate = self

        // 1.获取info.plist文件判断有没有对应的key
        let infoDict = NSBundle.mainBundle().infoDictionary!
        // 在iOS8.0之后需要主动请求授权
        if #available (iOS 8.0, *) {

//            print(infoDict)
            // NSLocationWhenInUseUsageDescription
            // NSLocationAlwaysUsageDescription

            // 2.获取前后台定位授权key的值
            let alwaysStr = infoDict["NSLocationAlwaysUsageDescription"]

            // 3.获取前台定位授权key的值
            let whenInUseStr = infoDict["NSLocationWhenInUseUsageDescription"]

            // 4.判断
            // 如果两个key只有一个有值,那么请求对应授权
            // 如果两个都有值,那么请求前后台定位授权
            // 如果都没有,给开发者提示
            if alwaysStr != nil {
                // 请求前后台定位授权
                locationM.requestAlwaysAuthorization()
            } else if whenInUseStr != nil {
                // 请求前台定位授权
                locationM.requestWhenInUseAuthorization()

                // 判断当前后台模式有没有勾选
                // location
                // 获取后台模式数组
                let backMode = infoDict["UIBackgroundModes"] as! [String]

                // 判断是否开启了位置后台模式
                if backMode.contains("location") {

                    // 在iOS9.0之后,在前台定位授权情况下,想要在后台获取用户的位置信息,除了开启后台模式,还需要允许后台定位
                    if #available (iOS 9.0, *) {
                        locationM.allowsBackgroundLocationUpdates = true
                    }
                } else {
                    print("在iOS8.0之后前台定位授权情况下,如果想要在后台获取用户位置信息,需要开启后台模式 location updates")
                }


            } else {
                // 给用户提示
                print("这位开发者,难道你忘了吗?在iOS8.0之后,如果想要获取用户的位置信息,必须在info.plist文件中配置对应key NSLocationWhenInUseUsageDescription或者NSLocationAlwaysUsageDescription")
            }

        } else {
             let backMode = infoDict["UIBackgroundModes"] as! [String]
            // 判断是否开启了位置后台模式
            if !backMode.contains("location") {
                print("如果想要在后台获取用户位置信息,需要开启后台模式 location updates")
            }
        }

        return locationM
    }()

    // MARK:- 用于记录block
    private var requestBlock : RequestBlock?

    // MARK:- 单例
    static let shareInstance = LocationTool()

    // 如果想要创建一个工具类分以下几步
    // 1.确定接口的类型:是对象方法还是类方法,如果无法确定,先使用类方法
    // 2.确定方法名:见名知意
    // 3.确定参数:如果不知道,先传空
    // 4.确定返回值:如果不知道,什么都不返回

    func getCurrentLocation(requestBlock :RequestBlock) {

        // 1.记录block
        self.requestBlock = requestBlock

        // 2.开始定位
        locationM.startUpdatingLocation()
    }
}

// MARK:- CLLocationManagerDelegate
extension LocationTool : CLLocationManagerDelegate {
    /**
     当获取到用户的位置时,就会调用该方法

     - parameter manager:   位置管理者
     - parameter locations: 位置数组
     */
    func locationManager(manager: CLLocationManager, didUpdateLocations locations: [CLLocation]) {

        // 1.获取位置对象
        guard let location = locations.last else {
            self.requestBlock!(location: nil, error: "未获取用户的位置")
            return
        }

        // 2.判断位置是否有效
        if location.horizontalAccuracy < 0 {return}

        // 3.返回当前位置
        self.requestBlock!(location: location, error: nil)

    }


}

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 123,307评论 18 134
  • iOS网络架构讨论梳理整理中。。。 其实如果没有APIManager这一层是没法使用delegate的,毕竟多个单...
    yhtang阅读 4,448评论 1 23
  • 前言 Blocks是C语言的扩充功能,而Apple 在OS X Snow Leopard 和 iOS 4中引入了这...
    小人不才阅读 3,264评论 0 23
  • 1985年 01.放課後 台湾:放學後(臉譜.2009/林白出版社.1991) 大陆:放学后(南海出版社.2010...
    搜盘君阅读 1,252评论 0 3
  • 今天爷爷住院了,我不心疼,只是很生气,为什么很生气,因为那是他自找的,家里人没有一个敢去强烈的阻止他吃什么?不能吃...
    葫芦娃一个藤上七个瓜阅读 61评论 0 0