iOS定位工具類--block傳值

在iOS開發中,經常會碰到定位到當前城市的需求,系統自帶的定位功能就能實現,于是封裝了一個類方便使用,整理一下發出來共勉,大家多提意見,多多交流??

使用

可實現定位到當前地址的功能,調用簡單,block傳值

 [[MSLocationTool sharedMSLocationTool] getCurrentLOcation:^(CLLocation *location, CLPlacemark *pl, NSString *error) {
         if ([error length] > 0) {
             NSLog(@"定位有錯誤-->%@",error);   
         }else{
             [self.locationBtn setTitle:pl.locality forState:UIControlStateNormal];
                
             [[NSUserDefaults standardUserDefaults] setValue:pl.locality forKey:CITY_KEY];
             [[NSUserDefaults standardUserDefaults] synchronize];
                
         }
            
}];

實現

  • 第一步:請求授權

在Info.plist中配置相關key:
Privacy - Location When In Use Usage Description
Privacy - Location Always Usage Description

//后面為字符串類型(隨意填寫) 例如:請您允許,我們將會為您提供更精確的信息
Privacy - Location When In Use Usage Description   
//后面為字符串類型(隨意填寫)例如:請您允許,我們將會為您提供更精確的信息 
Privacy - Location Always Usage Description        
// 特別聲明一下,這個字段的添加要看你的需求,這個功能是指在后臺時也可以時時定位,
所以app如果不需要的話,不建議添加,否則上線審核會出現問題,
如果需要這個功能,要做一些聲明描述:GPS在后臺持續運行,可以大大降低電池的壽命。
info.plist配置.png
  • 第二步:單例宏

創建一個Header File來實現單例宏,Singleton.h內容如下:

Singleton.png
#ifndef Singleton_h
#define Singleton_h

/**
 *  在.h文件中定義的宏,arc
 *
 *  MSSingletonH(name) 這個是宏
 *  + (instancetype)shared##name;這個是被代替的方法, ##代表著shared+name 高度定制化
 * 在外邊我們使用 “MSSingletonH(gege)” 那么在.h文件中,定義了一個方法"+ (instancetype)sharedgege",所以,第一個字母要大寫
 *
 *  @return 一個搞定好的方法名
 */
#define MSSingletonH(name) + (instancetype)shared##name;


/**
 *  在.m文件中處理好的宏 arc
 *
 *  MSSingletonM(name) 這個是宏,因為是多行的東西,所以每行后面都有一個"\",最后一行除外,
 * 之所以還要傳遞一個“name”,是因為有個方法要命名"+ (instancetype)shared##name"
 *  @return 單利
 */
#define MSSingletonM(name) \
static id instance_ = nil;\
+ (instancetype)shared##name{\
static dispatch_once_t onceToken;\
dispatch_once(&onceToken, ^{\
instance_ = [[self alloc] init];\
});\
return instance_;\
}\
+ (instancetype)allocWithZone:(struct _NSZone *)zone{\
static dispatch_once_t onceToken;\
dispatch_once(&onceToken, ^{\
instance_ = [super allocWithZone:zone];\
});\
return instance_;\
}\
- (id)copyWithZone:(NSZone *)zone{\
return instance_;\
}

#endif /* Singleton_h */


  • 第三步:繼承自NSObject的工具類MSLocationTool

工具類繼承自NSObject。

MSLocationTool.h文件內容如下:
1、導入系統自帶的庫#import <CoreLocation/CoreLocation.h>;
2、單例;
3、block實現返回值

#import <Foundation/Foundation.h>
#import "Singleton.h"
#import <CoreLocation/CoreLocation.h>


typedef void(^ResultCityBlock)(CLLocation *location,CLPlacemark *pl,NSString *error);


@interface MSLocationTool : NSObject


MSSingletonH(MSLocationTool)

-(void)getCurrentLOcation:(ResultCityBlock)block;


@end

MSLocationTool.m文件內容如下:

1、導入頭文件,屬性聲明
2、方法實現

#import "MSLocationTool.h"
#import <UIKit/UIKit.h>
#import "Singleton.h"

#define isIOS(version) ([[UIDevice currentDevice].systemVersion floatValue] >= version)

@interface MSLocationTool()<CLLocationManagerDelegate>

@property(nonatomic,copy)ResultCityBlock block;

/** manager */
@property(nonatomic,strong)CLLocationManager *locationManager;

/** 地理編碼 */
@property(nonatomic,strong)CLGeocoder *geoC;


@end


@implementation MSLocationTool

//單例
MSSingletonM(MSLocationTool)

#pragma mark - 懶加載
-(CLLocationManager *)locationManager
{
    if (_locationManager == nil) {
        _locationManager = [[CLLocationManager alloc] init];
        _locationManager.delegate = self;
        
        //獲取info.plist里的鍵值對
        NSDictionary *infoDict = [NSBundle mainBundle].infoDictionary;
        
        
        if (isIOS(8.0)) {
            
            //獲取后臺定位描述
            NSString *alwaysStr = infoDict[@"NSLocationAlwaysUsageDescription"];
            NSString *whenUserStr = infoDict[@"NSLocationWhenInUseUsageDescription"];
            
            // 根據開發者的設置 請求定位授權
            if ([alwaysStr length] > 0) {
                [_locationManager requestAlwaysAuthorization];
                
            }else if ([whenUserStr length] > 0){
                [_locationManager requestWhenInUseAuthorization];
                
                NSArray *backModels = infoDict[@"UIBackgroundModes"];
                if (![backModels containsObject:@"location"]) {
                    //前臺定位授權,如果想在后臺獲取位置,需勾選后臺模式location update
                    
                }else{
                    if (isIOS(9.0)) {
                        _locationManager.allowsBackgroundLocationUpdates = YES;
                    }
                    
                }
                
                
            }
     
            
        }else{
            // 如果請求的是前臺定位授權, 如果想要在后臺獲取用戶位置, 提醒其他開發者, 勾選后臺模式location updates
            NSArray *backModes = infoDict[@"UIBackgroundModes"];
            if (![backModes containsObject:@"location"]) {
                NSLog(@"當前授權模式, 如果想要在后臺獲取位置, 需要勾選后臺模式location updates");
            }
            
            
        }
        
        
    }
    
    return _locationManager;
}

-(CLGeocoder *)geoC
{
    if (_geoC == nil) {
        _geoC = [[CLGeocoder alloc] init];
    }
    return _geoC;
}

#pragma mark - 方法
-(void)getCurrentLOcation:(ResultCityBlock)block
{
    //記錄代碼塊
    self.block = block;
    
    //獲取位置信息
    if ([CLLocationManager locationServicesEnabled]) {
        
        [self.locationManager startUpdatingLocation];
        
    }else{
        self.block(nil, nil, @"定位服務未開啟");
        
    }
}

#pragma mark - 代理方法
-(void)locationManager:(CLLocationManager *)manager didUpdateLocations:(NSArray<CLLocation *> *)locations
{
    CLLocation *location = [locations lastObject];
    
    //判斷位置是否可用
    if (location.horizontalAccuracy >= 0) {
        [self.geoC reverseGeocodeLocation:location completionHandler:^(NSArray<CLPlacemark *> * _Nullable placemarks, NSError * _Nullable error) {
           
            if (error == nil) {
                CLPlacemark *pl = [placemarks firstObject];
                self.block(location, pl, nil);
                
            }else{
                self.block(location, nil, @"反地理編碼失敗");
                
            }
            
            
        }];
    }
    
    [manager stopUpdatingLocation];
    
}

-(void)locationManager:(CLLocationManager *)manager didChangeAuthorizationStatus:(CLAuthorizationStatus)status
{
    switch (status) {
        case kCLAuthorizationStatusNotDetermined:
            NSLog(@"用戶還未決定");
            break;
        
        case kCLAuthorizationStatusRestricted:
        {
            NSLog(@"訪問受限");
            self.block(nil, nil, @"訪問受限");
            break;
        }
         
        //定位關閉 或 對此APP授權未never時調用
        case kCLAuthorizationStatusDenied:
        {
            if ([CLLocationManager locationServicesEnabled]) {
                NSLog(@"定位開啟,但被拒");
                self.block(nil, nil, @"被拒絕");
                
            }else{
                NSLog(@"定位關閉,不可用");
                self.block(nil, nil, @"定位關閉,不可用");
            }
            
            break;
        }
            
        case kCLAuthorizationStatusAuthorizedAlways:
            NSLog(@"獲取前后臺定位授權");
            break;
            
        case kCLAuthorizationStatusAuthorizedWhenInUse:
            NSLog(@"獲取前臺定位授權");
            break;
            
            
            
        default:
            break;
    }
}

@end

參考鏈接
代碼附上--GitHub鏈接

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 227,837評論 6 531
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 98,196評論 3 414
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 175,688評論 0 373
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 62,654評論 1 309
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 71,456評論 6 406
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 54,955評論 1 321
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,044評論 3 440
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 42,195評論 0 287
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 48,725評論 1 333
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 40,608評論 3 354
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 42,802評論 1 369
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,318評論 5 358
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,048評論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,422評論 0 26
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 35,673評論 1 281
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 51,424評論 3 390
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 47,762評論 2 372

推薦閱讀更多精彩內容