iOS 自定義大頭針

先了解一下大頭針視圖:

MKPinAnnotationView :不能改變大頭針的圖片 可以改變顏色
MKAnnotationView:自己隨意定義大頭針的樣式

大頭針使用的一些屬性:

《1》初始化initWithAnnotation:reuseIdentifier:
《2》image 設(shè)置大頭針圖片
《3》centerOffset 中心點(diǎn)的偏移量 x正右 y正下
《4》calloutOffset 插圖的偏移量
《5》enabled 是否激活 默認(rèn)YES
《6》highlighted 是否高亮 默認(rèn)NO
《9》selected 是否選中
《10》canShowCallout 設(shè)置是否可以顯示 插入視圖
《11》leftCalloutAccessoryView 左側(cè)插入視圖的附加視圖
《12》rightCalloutAccessoryView:右側(cè)插入視圖的附加視圖
《13》detailCalloutAccessoryView iOS9之后出現(xiàn) 插入視圖的詳細(xì)視圖
《14》draggable 是否可以拖拽
《15》dragState 拖拽的狀態(tài)

代碼主要實現(xiàn)了:

  • 地圖添加 大頭針 并且保存到本地
    1.判斷有沒有可以讀取的大頭針模型數(shù)據(jù)
    2.如果有 讀取添加到地圖上
    3.添加大頭針的時候 同時把大頭針數(shù)據(jù) 存到數(shù)組中
    4.退出頁面的時候 對數(shù)組 數(shù)據(jù)持久化
  • 長按地圖上的某個點(diǎn) 添加一個大頭針峭拘,大頭針顯示地理位置的真實信息
    1.添加長按手勢
    2.添加一個大頭針數(shù)據(jù)模型 到地圖上
    《1》視圖上的point 需要轉(zhuǎn)換成經(jīng)緯度
    《2》把數(shù)據(jù)模型添加到地圖上
    addAnnotation:添加一個大頭針數(shù)據(jù)模型
    addAnnotations:添加一組大頭針數(shù)據(jù)模型
  • 移除大頭針
    removeAnnotation :移除一個
    removeAnnotations :移除多個

代碼如下:

#import "ViewController.h"
#import <MapKit/MapKit.h>
#import <CoreLocation/CoreLocation.h>

@interface ViewController ()<MKMapViewDelegate>
{
    CLLocationManager *locationManger;
    MKMapView *myMapView;
    NSMutableArray *allAnnotationsArray;
    
}

@property (nonatomic,strong) CLGeocoder *geoCoder;
@end

@implementation ViewController

- (void)viewDidLoad {
    [super viewDidLoad];
   
    if (![CLLocationManager locationServicesEnabled]) {
      UIAlertController *alertController = [UIAlertController alertControllerWithTitle:@"溫馨提示" message:@"請在設(shè)置中打開定位功能" preferredStyle:UIAlertControllerStyleAlert];
        
        UIAlertAction *action = [UIAlertAction actionWithTitle:@"OK" style:UIAlertActionStyleDefault handler:^(UIAlertAction * _Nonnull action) {
            
        }];
        
        [alertController addAction:action];
        
        [self presentViewController:alertController animated:YES completion:nil];

    }
    locationManger = [[CLLocationManager alloc]init];
    [locationManger requestWhenInUseAuthorization];
    myMapView = [[MKMapView alloc]initWithFrame:self.view.frame];
    myMapView.mapType = MKMapTypeStandard;
    myMapView.showsUserLocation = YES;
    myMapView.delegate = self;
    [self.view addSubview:myMapView];
    
    UILongPressGestureRecognizer *addPin = [[UILongPressGestureRecognizer alloc]initWithTarget:self action:@selector(addPin:)];
    [myMapView addGestureRecognizer:addPin];    
}

地圖代理方法:

- (void)mapView:(MKMapView *)mapView didUpdateUserLocation:(MKUserLocation *)userLocation{
    
    [mapView setCenterCoordinate:userLocation.location.coordinate animated:YES];
    
    MKCoordinateSpan span = MKCoordinateSpanMake(0.1, 0.1);
    MKCoordinateRegion regin = MKCoordinateRegionMake(userLocation.location.coordinate, span);
    [mapView setRegion:regin animated:YES];
    
//    userLocation -> MKUserLocation ->大頭針數(shù)據(jù)模型(用戶)->遵守
//    MKAnnotation協(xié)議 ->經(jīng)緯度  標(biāo)題  副標(biāo)題
//    MKUserLocation 經(jīng)緯度  標(biāo)題  副標(biāo)題 航向 location
    userLocation.title = @"深圳市龍崗區(qū)"; 
}

在這里可以設(shè)置大頭針的圖片科吭、位置等:

- (nullable MKAnnotationView *)mapView:(MKMapView *)mapView viewForAnnotation:(id <MKAnnotation>)annotation{
    
    //    判斷是不是用戶的大頭針數(shù)據(jù)模型
    if ([annotation isKindOfClass:[MKUserLocation class]]) {
        
        MKAnnotationView *userView = [[MKAnnotationView alloc]initWithAnnotation:annotation reuseIdentifier:@"user"];
        
        //        image 設(shè)置大頭針視圖的圖片
        userView.image = [UIImage imageNamed:@"7"];
        
        //  是否允許顯示插入視圖
        userView.canShowCallout = YES;
        
        //設(shè)置大頭針視圖中心點(diǎn)偏移
        //x 正 -> 向右
        //Y 正 -> 向下
        userView.centerOffset = CGPointMake(100, -100);
        
        //設(shè)置大頭針插入視圖的 偏移量
        userView.calloutOffset = CGPointMake(-50, 0);
        
        //設(shè)置大頭針視圖  是否可以拖動
        userView.draggable = YES;
        
        //設(shè)置配件視圖
        
        UIImageView *leftView = [[UIImageView alloc]initWithFrame:CGRectMake(0, 0, 30, 30)];
        leftView.image = [UIImage imageNamed:@"7"];
        userView.leftCalloutAccessoryView = leftView;
        
        UIImageView *rightView = [[UIImageView alloc]initWithFrame:CGRectMake(0, 0, 30, 30)];
        rightView.image = [UIImage imageNamed:@"7"];
        userView.rightCalloutAccessoryView = rightView;
        
 UIImageView *detailView = [[UIImageView alloc]initWithFrame:CGRectMake(0, 0, 30, 30)];
        detailView.image = [UIImage imageNamed:@"7"];
        userView.detailCalloutAccessoryView = detailView;
        return userView;
    }
    
    return nil;
}
//定位成功的代理方法
- (void)locationManager:(CLLocationManager *)manager
     didUpdateLocations:(NSArray<CLLocation *> *)locations{
    NSLog(@"定位成功");
    //此處locations存儲了持續(xù)更新的位置坐標(biāo)值空执,取最后一個值為最新位置腺逛,如果不想讓其持續(xù)更新位置驶拱,則在此方法中獲取到一個值之后讓locationManager stopUpdatingLocation
    
    CLLocation *currentLocation = [locations lastObject];
    
    NSLog(@"locations===%@",[locations lastObject]);
//    [self ungeocoder:currentLocation];
    // 獲取當(dāng)前所在的城市名
    CLGeocoder *geocoder = [[CLGeocoder alloc] init];
    
    //根據(jù)經(jīng)緯度反向地理編譯出地址信息
    
    [geocoder reverseGeocodeLocation:currentLocation completionHandler:^(NSArray *array, NSError *error)
     
     {
         if (array.count > 0)
             
         {
             CLPlacemark *placemark = [array objectAtIndex:0];
            NSLog(@"%@",placemark.name);

             //獲取城市
             
             NSString *city = placemark.locality;
             
             if (!city) {
                 //四大直轄市的城市信息無法通過locality獲得伦糯,只能通過獲取省份的方法來獲得(如果city為空,則可知為直轄市)
                 city = placemark.administrativeArea;
                 
             }
             NSLog(@"%@",city);

         }
         
         else if (error == nil && [array count] == 0)
             
         {
             NSLog(@"No results were returned.");
         }
         
         else if (error != nil)
             
         {
          NSLog(@"An error occurred = %@", error);
             
         }
         
     }];
    
    //系統(tǒng)會一直更新數(shù)據(jù)靠娱,直到選擇停止更新沧烈,因為我們只需要獲得一次經(jīng)緯度即可,所以獲取之后就停止更新
    
    [manager stopUpdatingLocation];
}

//定位失敗的代理方法
- (void)locationManager:(CLLocationManager *)manager

       didFailWithError:(NSError *)error {
    
    NSLog(@"定位失敗");
    
    if (error.code == kCLErrorDenied) {
        
        // 提示用戶出錯原因像云,可按住Option鍵點(diǎn)擊 KCLErrorDenied的查看更多出錯信息锌雀,可打印error.code值查找原因所在
        
    }
}

把大頭針保存到本地,代碼如下:

- (void)viewWillAppear:(BOOL)animated{
    [super viewWillAppear:animated];
    //    作.判斷有沒有可以讀取的大頭針數(shù)據(jù)模型
    if ([[NSUserDefaults standardUserDefaults] boolForKey:@"isSave"]) {
        
        allAnnotationsArray = [[[NSUserDefaults standardUserDefaults]arrayForKey:@"allAnnotationsArray"] mutableCopy];
        
        NSMutableArray *dataList = [NSMutableArray array];
        for (NSDictionary *info in allAnnotationsArray) {
            MKPointAnnotation *an = [[MKPointAnnotation alloc]init];
            an.coordinate = CLLocationCoordinate2DMake([info[@"coordinate"][@"latitude"] doubleValue], [info[@"coordinate"][@"longitude"] doubleValue]);
            an.title = info[@"title"];
            [dataList addObject:an];
        }
        
        // 作第二步
        [myMapView addAnnotations:dataList];
        
    }else{
        allAnnotationsArray = [NSMutableArray array];
    }
}
- (void)viewDidDisappear:(BOOL)animated{
    [super viewDidDisappear:animated];
    NSUserDefaults *defaults = [NSUserDefaults standardUserDefaults]; 
    if (allAnnotationsArray.count != 0) {
        [defaults setObject:allAnnotationsArray forKey:@"allAnnotationsArray"];
        [defaults setBool:YES forKey:@"isSave"];
    }
    //    多線程同步  方式保存
    [defaults synchronize];    
}

長按屏幕迅诬,添加大頭針腋逆,代碼如下:

- (void)addPin:(UILongPressGestureRecognizer *)sender{
    
    if (sender.state != UIGestureRecognizerStateBegan) {
    return;
    }
    
    MKPointAnnotation *annotation = [[MKPointAnnotation alloc]init];
    
    
//    通過手勢  得到手指 觸摸的點(diǎn) -> point
    CGPoint point = [sender locationInView:self.view];
    
//    通過地圖類的對象  把點(diǎn)轉(zhuǎn)換成經(jīng)緯度
    CLLocationCoordinate2D coordinate = [myMapView convertPoint:point toCoordinateFromView:sender.view];
    
  [self.geoCoder reverseGeocodeLocation:[[CLLocation alloc]initWithLatitude:coordinate.latitude longitude:coordinate.longitude] completionHandler:^(NSArray<CLPlacemark *> * _Nullable placemarks, NSError * _Nullable error) {
        
        annotation.coordinate = coordinate;
        annotation.title = placemarks.firstObject.name;
       [myMapView addAnnotation:annotation];
        
//        作第三步  添加大頭針地圖的時候  同時添加到數(shù)組
        [allAnnotationsArray addObject:@{@"coordinate":@{@"latitude":@(coordinate.latitude),@"longitude":@(coordinate.longitude)},@"title":placemarks.firstObject.name}];
 
    }];    
}

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市侈贷,隨后出現(xiàn)的幾起案子惩歉,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 212,454評論 6 493
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件柬泽,死亡現(xiàn)場離奇詭異慎菲,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)锨并,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,553評論 3 385
  • 文/潘曉璐 我一進(jìn)店門露该,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人第煮,你說我怎么就攤上這事解幼。” “怎么了包警?”我有些...
    開封第一講書人閱讀 157,921評論 0 348
  • 文/不壞的土叔 我叫張陵撵摆,是天一觀的道長。 經(jīng)常有香客問我害晦,道長特铝,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 56,648評論 1 284
  • 正文 為了忘掉前任壹瘟,我火速辦了婚禮鲫剿,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘稻轨。我一直安慰自己灵莲,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,770評論 6 386
  • 文/花漫 我一把揭開白布殴俱。 她就那樣靜靜地躺著政冻,像睡著了一般。 火紅的嫁衣襯著肌膚如雪线欲。 梳的紋絲不亂的頭發(fā)上明场,一...
    開封第一講書人閱讀 49,950評論 1 291
  • 那天,我揣著相機(jī)與錄音询筏,去河邊找鬼榕堰。 笑死,一個胖子當(dāng)著我的面吹牛嫌套,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播圾旨,決...
    沈念sama閱讀 39,090評論 3 410
  • 文/蒼蘭香墨 我猛地睜開眼踱讨,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了砍的?” 一聲冷哼從身側(cè)響起痹筛,我...
    開封第一講書人閱讀 37,817評論 0 268
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后帚稠,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體谣旁,經(jīng)...
    沈念sama閱讀 44,275評論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,592評論 2 327
  • 正文 我和宋清朗相戀三年滋早,在試婚紗的時候發(fā)現(xiàn)自己被綠了榄审。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 38,724評論 1 341
  • 序言:一個原本活蹦亂跳的男人離奇死亡杆麸,死狀恐怖搁进,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情昔头,我是刑警寧澤饼问,帶...
    沈念sama閱讀 34,409評論 4 333
  • 正文 年R本政府宣布,位于F島的核電站揭斧,受9級特大地震影響莱革,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜讹开,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 40,052評論 3 316
  • 文/蒙蒙 一盅视、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧萧吠,春花似錦左冬、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,815評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至狰腌,卻和暖如春除破,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背琼腔。 一陣腳步聲響...
    開封第一講書人閱讀 32,043評論 1 266
  • 我被黑心中介騙來泰國打工瑰枫, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人丹莲。 一個月前我還...
    沈念sama閱讀 46,503評論 2 361
  • 正文 我出身青樓光坝,卻偏偏與公主長得像,于是被迫代替她去往敵國和親甥材。 傳聞我的和親對象是個殘疾皇子盯另,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,627評論 2 350

推薦閱讀更多精彩內(nèi)容