关于iOS 应用如何添加谷歌地图,网上其实有很多文档,但当自己实际开发时,还是会有很多问题,我这次的需求就是地图国际化,原项目中接的是高德地图,现在要替换成谷歌地图,要求功能及页面的UI显示要和之前一致(肯定不可能百分百,至少8成像)
一、项目集成Google maps
官方文档连接
推荐是使用cooapods集成,通常会使用一下两个sdk
- 'GoogleMaps',基本的定位功能,通常加载这一个sdk就可以了
- 'GooglePlaces',实现搜索功能,官方文档叫做地点自动完成,可查找周围的兴趣点,即POI
在Podfile添加
pod 'GoogleMaps', '指定版本号'
二、获取API密匙
前提是已经在GoogleMapSDK中创建好自己的应用,需要有自己的Google账号,我这边是Android开发早就申请好了,我复制了APIKey直接使用
三、配置plist文件搭建定位环境
info.pliste文件中添加定位权限相关字段
四、调用代理方法实现相关需求
- 在
AppDelegate.m
导入框架
#import <GoogleMaps/GoogleMaps.h>
- 在
application:didFinishLaunchingWithOptions
方法中添加
[GMSServices provideAPIKey: @"APIKey"];
- 在我们需要显示地图的控制器调用API方法
为了解耦代码增加可复用性,可以写一个地图的类,对地图做一些简单的配置,增加一些自定义方法,比如地图截屏方法,因为项目的需求的是在一个IM空间中,像他人发送一个地理位置,以聊天气泡的形式发送(参考微信中的“发送位置”)
简单介绍一下我用到的 GMSMapView
的属性,如果想知道全部的属性,可以在地图的sdk加载好之后,点击进入到 GMSMapView
中查看
- (GMSMapView *)mapView {
if (!_mapView) {
// 根据经纬度和缩放等级,初始化相机,显示地图内容,用于海外的,最好给个国外的经纬度
GMSCameraPosition *camera = [GMSCameraPosition cameraWithLatitude:38.8879
longitude:-77.0200
zoom:17];
_mapView = [GMSMapView mapWithFrame:CGRectZero camera:camera];
_mapView.mapType = kGMSTypeNormal;
// 设备当前位置的点和准确性圆圈
_mapView.myLocationEnabled = YES;
// 使用指南针
_mapView.settings.compassButton = YES;
// 当前位置按钮
_mapView.settings.myLocationButton = YES;
_mapView.settings.indoorPicker = NO;
}
return _mapView;
}
- 需求1:大头针和当前位置的地址始终显示在屏幕中间
这时候建议不要使用GMSMarker
,因为在实际使用过程中,每当移动地图,大头针会一跳一跳的,虽然也是一直显示在最中间,但是会有间歇性空白,视觉效果不是很好,如果强行用GMSMarker
来定义大头针,操作会很复杂,不如直接定义一个UIImageView,那对应的详细地址气泡也需要自定义,最好写个类,下面还有需求可以复用 - 需求2:地理位置反编译,本来是要显示当前位置的POI,但是
GMSMapView
没有一个代理方法可以做到这一点,只有点击某个兴趣点可以拿到那个兴趣点的placeID,再根据placeID做其他操作,但是这不符合需求,需求是任意点都能拿到POI,如果任意点不是POI那就拿到最近的POI信息,这时候只引入GoogleMaps
的sdk已经不能满足需求了,需要在 Podfile 中添加 'GooglePlaces'
pod 'GooglePlaces','指定版本号'
但是我实际使用过程中发现,还是不能满足现在的需求,他的代理方法返回的是附近一些POI点的集合,一个数组,而且测试发现,这个数组中的元素,只是你当前设备所在位置周边的POI的信息,不是随着你地图中心位置移动而变化的,达不到想要的效果(也许还有其他方法我没发现,有经验的大佬请教教我),最后退而求其次,用了 GMSAddress
,对当前经纬度的位置做了反编译,做多能拿到当前位置是那条路和邮编。
提示: GMSServices 和 GMSPlacesClient 的 APIKey是不同的,不能使用同一个,
[GMSServices provideAPIKey: @"key1"];
[GMSPlacesClient provideAPIKey: @"key2"];
否则 GooglePlaces 里面类的代理方法使用都会报错Error Domain=com.google.places.ErrorDomain Code=-3 "An internal error occurred in the Places API library. If you believe this error represents a bug, please file a report using the instructions on our community and support page ([https://developers.google.com/places/ios-sdk/support)](https://developers.google.com/places/ios-sdk/support))." UserInfo={NSLocalizedFailureReason=An internal error occurred in the Places API library. If you believe this error represents a bug, please file a report using the instructions on our community and support page ([https://developers.google.com/places/ios-sdk/support).](https://developers.google.com/places/ios-sdk/support).), NSUnderlyingError=0x171251c10 {Error Domain=com.google.places.api.server.ErrorDomain Code=-2 "This API project is not authorized to use this API." UserInfo={NSLocalizedFailureReason=This API project is not authorized to use this API.}}}
- 需求三3:别人点开我发送给他的位置,要求地图可滑动缩放,但是我发送的位置处会有大头针及地址气泡,并且位置固定在对应的经纬度,这就和需求1又不一样了,这时候最方便的就是用sdk中的大头针
GMSMarker
并且自定义大头针气泡,在下面这个代理方法中写视图
- (nullable UIView *)mapView:(GMSMapView *)mapView markerInfoWindow:(GMSMarker *)marker;
五、遇到的问题
- 当他人查看我送的位置时,一进入到该页面要求大头针气泡是始终显示的,这就需要将地图的
selectedMarker
设为当前的大头针,因为地图可缩放滑动,当点击到地图任意一点,大头针气泡都会隐藏,我这边是在didTapAtCoordinate
方法中设置地图的selectedMarker
- (void)mapView:(GMSMapView *)mapView didTapAtCoordinate:(CLLocationCoordinate2D)coordinate {
self.mapView.selectedMarker = self.defaultMarker;
}
- 打开地图,将地图中心点移到很远的地方,大概5分钟左右,地图中心点会在自动回到当前设备所在位置,
处理方法:我是将 CLLocationManager 的distanceFilter
属性设置成默认的,感觉是可以的,我自测没什么问题,具体我也不清楚,等待测试反馈结果。