苹果原生地图
一、定位
要实现地图、导航功能,往往需要先熟悉定位功能,在iOS中通过Core Location框架进行定位操作。Core Location自身可以单独使用,和地图开发框架MapKit完全是独立的,但是往往地图开发要配合定位框架使用。在Core Location中主要包含了定位、地理编码和反地理编码功能。
定位是一个很常用的功能,如一些地图软件打开之后如果用户允许软件定位的话,那么打开软件后就会自动锁定到当前位置,如果用户手机移动那么当前位置也会跟随着变化。要实现这个功能需要使用Core Loaction中CLLocationManager类
在iOS8之后,必须通过配置NSLocationAlwaysUsageDescription或者NSLocationWhenInUseUsageDescription来告诉用户使用定位服务的目的,并且注意这个配置是必须的,如果不进行配置则默认情况下应用无法使用定位服务,打开应用不会给出打开定位服务的提示,除非安装后自己设置此应用的定位服务。同时,在应用程序中需要根据配置对requestAlwaysAuthorization或locationServicesEnabled方法进行请求。
注意:
1.定位频率和定位精度并不应当越精确越好,需要视实际情况而定,因为越精确越耗性能,也就越费电。
2.定位成功后会根据设置情况频繁调用-(void)locationManager:(CLLocationManager *)manager didUpdateLocations:(NSArray *)locations方法,这个方法返回一组地理位置对象数组,每个元素一个CLLocation代表地理位置信息(包含经度、纬度、海报、行走速度等信息),之所以返回数组是因为有些时候一个位置点可能包含多个位置。
3.使用完定位服务后如果不需要实时监控应该立即关闭定位服务以节省资源。
4.除了提供定位功能,CLLocationManager还可以调用startMonitoringForRegion:方法对指定区域进行监控。
会出现的问题:
在iOS8之后,苹果地图得定位方法修改了,以前得不能用了
报错说明:Trying to start MapKit location updates without prompting for location authorization. Must call -[CLLocationManager requestWhenInUseAuthorization] or -[CLLocationManager requestAlwaysAuthorization] first.
解决办法:
在Supporting Files下得XXX.info.plis里面添加NSLocationAlwaysUsageDescription 或 NSLocationWhenInUseDescription 两个String字段,value可以为空,也可以设置YES,不过我得问题还是不能解决,最终还是找到得了问题所在,就是info.plist中还需要包含Supported interface orientations 这个Array字段。然后运行就解决了.
//示例代码
#pragma mark - 定位
-(void)createLocation
{
//初始化定位管理器
_locationManager = [[CLLocationManager alloc]init];
//判断定位服务是否打开
if (![CLLocationManager locationServicesEnabled]) {
NSLog(@"定位服务目前尚未打开,请设置打开");
return;
}
//如果没有授权则当使用定位的时候请求用户授权
if ([CLLocationManager authorizationStatus] == kCLAuthorizationStatusNotDetermined) {
[_locationManager requestWhenInUseAuthorization];
}
//如果已经授权
else if ([CLLocationManager authorizationStatus] == kCLAuthorizationStatusAuthorizedWhenInUse)
{
//设置代理
_locationManager.delegate = self;
//设置精度
_locationManager.desiredAccuracy = kCLLocationAccuracyBest;
//定位频率,也就是每隔多少米定位一次
_locationManager.distanceFilter = 10.0;//每隔10米定位一次
//启动跟踪定位
[_locationManager startUpdatingLocation];
}
}
#pragma mark - 定位的代理方法
//跟踪定位的代理方法,一旦位置发生变化则会调用这个方法
-(void)locationManager:(CLLocationManager *)manager didUpdateLocations:(NSArray *)locations
{
CLLocation * location = [locations firstObject];//取出第一个位置
//获得位置坐标
CLLocationCoordinate2D coord = location.coordinate;
//如果不使用定位服务,使用完成之后及时关闭定位
[manager stopUpdatingLocation];
//获取详细的地理位置信息
NSLog(@"经度%f~~纬度%f~~海拔%f~~航向%f~~行走速度%f",coord.longitude,coord.latitude,location.altitude,location.course,location.speed);
}
二、地理编码
除了提供位置跟踪功能之外,在定位服务中还包含CLGeocoder类用于处理地理编码和逆地理编码(又叫反地理编码)功能。
地理编码:根据给定的位置(通常是地名)确定地理坐标(经、纬度)。
逆地理编码:可以根据地理坐标(经、纬度)确定位置信息(街道、门牌等)。39.54,116.28
详细地址信息:name,thoroughfare,subThoroughfare,locality,subLocality,administrativeArea,subAdministrativeArea,postalCode,ISOcountryCode,country,inlandWate,ocean,areasOfInterest
//示例代码
#pragma mark - 地理编码
-(void)createGeocoder
{
//实例化
_geocoder = [[CLGeocoder alloc]init];
//正向地理编码,通过位置获得经纬度
[self getCoordinateByAddress:@"北京"];
//反向地理编码或逆地理编码,根据经纬度获取地理位置
[self getAddressByLatitude:39.54 longitude:116.28];
}
//正向地理编码
-(void)getCoordinateByAddress:(NSString *)adderss
{
[_geocoder geocodeAddressString:adderss completionHandler:^(NSArray * _Nullable placemarks, NSError * _Nullable error) {
//取出第一个地标,一个地名可能会搜索出多个地址
CLPlacemark * placemark = [placemarks firstObject];
//位置
CLLocation * location = placemark.location;
//区域
CLRegion * region = placemark.region;
//详细地址信息
NSDictionary * addressDic = placemark.addressDictionary;
// NSString * name = placemark.name;//地名
// NSString * thorough = placemark.thoroughfare;//街道
// NSString * subThorough = placemark.subThoroughfare;//详细的接到信息,比如门牌号
// NSString *locality=placemark.locality; // 城市
// NSString *subLocality=placemark.subLocality; // 城市相关信息,例如标志性建筑
// NSString *administrativeArea=placemark.administrativeArea; // 州
// NSString *subAdministrativeArea=placemark.subAdministrativeArea; //其他行政区域信息
// NSString *postalCode=placemark.postalCode; //邮编
// NSString *ISOcountryCode=placemark.ISOcountryCode; //国家编码
// NSString *country=placemark.country; //国家
// NSString *inlandWater=placemark.inlandWater; //水源、湖泊
// NSString *ocean=placemark.ocean; // 海洋
// NSArray *areasOfInterest=placemark.areasOfInterest; //关联的或利益相关的地标
NSLog(@"正向地理编码:位置~~~%@区域~~~%@详细信息~~~%@",location,region,addressDic);
}];
}
//逆向地理编码
-(void)getAddressByLatitude:(CLLocationDegrees)latitude longitude:(CLLocationDegrees)longitude
{
//根据经纬度获取位置
CLLocation * location = [[CLLocation alloc]initWithLatitude:latitude longitude:longitude];
[_geocoder reverseGeocodeLocation:location completionHandler:^(NSArray * _Nullable placemarks, NSError * _Nullable error) {
CLPlacemark * placemark = [placemarks firstObject];
NSLog(@"逆向地理编码:详细信息~~~%@",placemark.addressDictionary);
}];
}
三、地图
北京的经纬度:39.896304,116.410103
iOS从6.0开始地图数据不再由谷歌驱动,而是改用自家地图,当然在国内它的数据是由高德地图提供的,在iOS中进行地图开发主要有两种方式,一种是直接利用MapKit框架进行地图开发,利用这种方式可以对地图进行精准的控制;另一种方式是直接调用苹果官方自带的地图应用,主要用于一些简单的地图应用(例如:进行导航覆盖物填充等),无法进行精确的控制
使用大头针需要注意的问题
a.这个代理方法的调用时机:每当有大头针显示到系统可视界面中时就会调用此方法返回一个大头针视图放到界面中,同时当前系统位置标注(也就是地图中蓝色的位置点)也是一个大头针,也会调用此方法,因此处理大头针视图时需要区别对待。
b.类似于UITableView的代理方法,此方法调用频繁,开发过程中需要重复利用MapKit的缓存池将大头针视图缓存起来重复利用。
c.自定义大头针默认情况下不允许交互,如果交互需要设置canShowCallout=true
d.如果代理方法返回nil则会使用默认大头针视图,需要根据情况设置。
//示例代码
#pragma mark - 创建地图
-(void)createMapView
{
//创建mapView
_mapView = [[MKMapView alloc]initWithFrame:self.view.frame];
//设置代理
_mapView.delegate = self;
//设置显示的位置
CLLocationCoordinate2D coord = CLLocationCoordinate2DMake(39.896304, 116.410103);
//设置放大的比例,数值越小,放大的比例越大
MKCoordinateSpan span = MKCoordinateSpanMake(0.1, 0.1);
[_mapView setRegion:MKCoordinateRegionMake(coord, span)];
//开启显示自己的位置
_mapView.showsUserLocation = YES;
//用户位置追踪
_mapView.userTrackingMode = MKUserTrackingModeFollow;
//设置地图类型
_mapView.mapType = MKMapTypeStandard;
//显示交通路线
_mapView.showsTraffic = YES;
[self.view addSubview:_mapView];
//添加大头针
[self addAnnotation:coord];
//在地图上添加长按手势
UILongPressGestureRecognizer * gesture = [[UILongPressGestureRecognizer alloc]initWithTarget:self action:@selector(longPressGesture:)];
[_mapView addGestureRecognizer:gesture];
}
-(void)addAnnotation:(CLLocationCoordinate2D)coord
{
MKPointAnnotation * annotation = [[MKPointAnnotation alloc]init];
//赋值坐标
annotation.coordinate = coord;
//将大头针添加到地图上
[_mapView addAnnotation:annotation];
}
#pragma mark - 长按手势
//当长按的时候在地图上添加一个大头针
-(void)longPressGesture:(UILongPressGestureRecognizer *)geature
{
//获取长按手势的中心点
CGPoint point = [geature locationInView:_mapView];
//根据中心点的位置折算成坐标
CLLocationCoordinate2D coord = [_mapView convertPoint:point toCoordinateFromView:_mapView];
//添加大头针到地图上
MKPointAnnotation * annotation = [[MKPointAnnotation alloc]init];
//赋值坐标
annotation.coordinate = coord;
//设置标题
annotation.title = @"主标题";
//设置副标题
annotation.subtitle = @"副标题";
//将大头针添加到地图上
[_mapView addAnnotation:annotation];
}
#pragma mark -地图的代理方法
//设置大头针的属性
-(MKAnnotationView *)mapView:(MKMapView *)mapView viewForAnnotation:(id)annotation
{
//系统自带的大头针样式
//创建大头针,类似UITableViewCell的复用机制
MKPinAnnotationView * pinAnnotationView = (MKPinAnnotationView *)[mapView dequeueReusableAnnotationViewWithIdentifier:@"ID"];
if (!pinAnnotationView) {
pinAnnotationView = [[MKPinAnnotationView alloc]initWithAnnotation:annotation reuseIdentifier:@"ID"];
}
//设置大头针出现的动画
pinAnnotationView.animatesDrop = YES;
//设置大头针的颜色
pinAnnotationView.pinTintColor = [UIColor blackColor];
//设置气泡出现,默认无法出现
pinAnnotationView.canShowCallout = YES;
//添加左图标
UIImageView * leftImageView = [[UIImageView alloc]initWithFrame:CGRectMake(0, 0, 30, 30)];
leftImageView.image = [UIImage imageNamed:@"2.jpg"];
pinAnnotationView.leftCalloutAccessoryView = leftImageView;
//添加右侧按钮
UIButton * rightButton = [UIButton buttonWithType:UIButtonTypeContactAdd];
rightButton.frame = CGRectMake(0, 0, 30, 30);
pinAnnotationView.rightCalloutAccessoryView = rightButton;
return pinAnnotationView;
//自定义大头针
MKAnnotationView * annotationView = [mapView dequeueReusableAnnotationViewWithIdentifier:@"view"];
if (!annotationView) {
annotationView = [[MKAnnotationView alloc]initWithAnnotation:annotation reuseIdentifier:@"view"];
}
//设置大头针坐标
//annotationView.frame = CGRectMake(100, 100, 30, 30);
//设置图片
annotationView.image = [UIImage imageNamed:@"2.jpg"];
//设置气泡出现
annotationView.canShowCallout = YES;
[UIView animateWithDuration:2 animations:^{
annotationView.frame = CGRectMake(0, 0, 30, 30);
}];
//return annotationView;
}
//点击辅助按钮会调用下面的方法
-(void)mapView:(MKMapView *)mapView annotationView:(MKAnnotationView *)view calloutAccessoryControlTapped:(UIControl *)control
{
NSLog(@"点击了辅助按钮,在这个按钮的地方可以写拨打电话功能");
}