react-native/React/Views/RCTMap.m

200 lines
5.7 KiB
Objective-C

/**
* Copyright (c) 2015-present, Facebook, Inc.
* All rights reserved.
*
* This source code is licensed under the BSD-style license found in the
* LICENSE file in the root directory of this source tree. An additional grant
* of patent rights can be found in the PATENTS file in the same directory.
*/
#import "RCTMap.h"
#import "RCTEventDispatcher.h"
#import "RCTLog.h"
#import "RCTMapAnnotation.h"
#import "RCTMapOverlay.h"
#import "RCTUtils.h"
const CLLocationDegrees RCTMapDefaultSpan = 0.005;
const NSTimeInterval RCTMapRegionChangeObserveInterval = 0.1;
const CGFloat RCTMapZoomBoundBuffer = 0.01;
@implementation RCTMap
{
UIView *_legalLabel;
CLLocationManager *_locationManager;
}
- (instancetype)init
{
if ((self = [super init])) {
_hasStartedRendering = NO;
// Find Apple link label
for (UIView *subview in self.subviews) {
if ([NSStringFromClass(subview.class) isEqualToString:@"MKAttributionLabel"]) {
// This check is super hacky, but the whole premise of moving around
// Apple's internal subviews is super hacky
_legalLabel = subview;
break;
}
}
}
return self;
}
- (void)dealloc
{
[_regionChangeObserveTimer invalidate];
}
- (void)layoutSubviews
{
[super layoutSubviews];
if (_legalLabel) {
dispatch_async(dispatch_get_main_queue(), ^{
CGRect frame = _legalLabel.frame;
if (_legalLabelInsets.left) {
frame.origin.x = _legalLabelInsets.left;
} else if (_legalLabelInsets.right) {
frame.origin.x = self.frame.size.width - _legalLabelInsets.right - frame.size.width;
}
if (_legalLabelInsets.top) {
frame.origin.y = _legalLabelInsets.top;
} else if (_legalLabelInsets.bottom) {
frame.origin.y = self.frame.size.height - _legalLabelInsets.bottom - frame.size.height;
}
_legalLabel.frame = frame;
});
}
}
#pragma mark Accessors
- (void)setShowsUserLocation:(BOOL)showsUserLocation
{
if (self.showsUserLocation != showsUserLocation) {
if (showsUserLocation && !_locationManager) {
_locationManager = [CLLocationManager new];
if ([_locationManager respondsToSelector:@selector(requestWhenInUseAuthorization)]) {
[_locationManager requestWhenInUseAuthorization];
}
}
super.showsUserLocation = showsUserLocation;
// If it needs to show user location, force map view centered
// on user's current location on user location updates
_followUserLocation = showsUserLocation;
}
}
- (void)setRegion:(MKCoordinateRegion)region animated:(BOOL)animated
{
// If location is invalid, abort
if (!CLLocationCoordinate2DIsValid(region.center)) {
return;
}
// If new span values are nil, use old values instead
if (!region.span.latitudeDelta) {
region.span.latitudeDelta = self.region.span.latitudeDelta;
}
if (!region.span.longitudeDelta) {
region.span.longitudeDelta = self.region.span.longitudeDelta;
}
// Animate to new position
[super setRegion:region animated:animated];
}
// TODO: this doesn't preserve order. Should it? If so we should change the
// algorithm. If not, it would be more efficient to use an NSSet
- (void)setAnnotations:(RCTMapAnnotationArray *)annotations
{
NSMutableArray<NSString *> *newAnnotationIDs = [NSMutableArray new];
NSMutableArray<RCTMapAnnotation *> *annotationsToDelete = [NSMutableArray new];
NSMutableArray<RCTMapAnnotation *> *annotationsToAdd = [NSMutableArray new];
for (RCTMapAnnotation *annotation in annotations) {
if (![annotation isKindOfClass:[RCTMapAnnotation class]]) {
continue;
}
[newAnnotationIDs addObject:annotation.identifier];
// If the current set does not contain the new annotation, mark it to add
if (![_annotationIDs containsObject:annotation.identifier]) {
[annotationsToAdd addObject:annotation];
}
}
for (RCTMapAnnotation *annotation in self.annotations) {
if (![annotation isKindOfClass:[RCTMapAnnotation class]]) {
continue;
}
// If the new set does not contain an existing annotation, mark it to delete
if (![newAnnotationIDs containsObject:annotation.identifier]) {
[annotationsToDelete addObject:annotation];
}
}
if (annotationsToDelete.count) {
[self removeAnnotations:(NSArray<id<MKAnnotation>> *)annotationsToDelete];
}
if (annotationsToAdd.count) {
[self addAnnotations:(NSArray<id<MKAnnotation>> *)annotationsToAdd];
}
self.annotationIDs = newAnnotationIDs;
}
// TODO: this doesn't preserve order. Should it? If so we should change the
// algorithm. If not, it would be more efficient to use an NSSet
- (void)setOverlays:(RCTMapOverlayArray *)overlays
{
NSMutableArray *newOverlayIDs = [NSMutableArray new];
NSMutableArray *overlaysToDelete = [NSMutableArray new];
NSMutableArray *overlaysToAdd = [NSMutableArray new];
for (RCTMapOverlay *overlay in overlays) {
if (![overlay isKindOfClass:[RCTMapOverlay class]]) {
continue;
}
[newOverlayIDs addObject:overlay.identifier];
// If the current set does not contain the new annotation, mark it to add
if (![_annotationIDs containsObject:overlay.identifier]) {
[overlaysToAdd addObject:overlay];
}
}
for (RCTMapOverlay *overlay in self.overlays) {
if (![overlay isKindOfClass:[RCTMapOverlay class]]) {
continue;
}
// If the new set does not contain an existing annotation, mark it to delete
if (![newOverlayIDs containsObject:overlay.identifier]) {
[overlaysToDelete addObject:overlay];
}
}
if (overlaysToDelete.count) {
[self removeOverlays:(NSArray<id<MKOverlay>> *)overlaysToDelete];
}
if (overlaysToAdd.count) {
[self addOverlays:(NSArray<id<MKOverlay>> *)overlaysToAdd
level:MKOverlayLevelAboveRoads];
}
self.overlayIDs = newOverlayIDs;
}
@end