/** * 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 "RCTMapManager.h" #import "RCTBridge.h" #import "RCTConvert+CoreLocation.h" #import "RCTConvert+MapKit.h" #import "RCTEventDispatcher.h" #import "RCTMap.h" #import "RCTUtils.h" #import "UIView+React.h" #import "RCTMapAnnotation.h" #import "RCTMapOverlay.h" #import static NSString *const RCTMapViewKey = @"MapView"; #if __IPHONE_OS_VERSION_MIN_REQUIRED < __IPHONE_9_0 static NSString *const RCTMapPinRed = @"#ff3b30"; static NSString *const RCTMapPinGreen = @"#4cd964"; static NSString *const RCTMapPinPurple = @"#c969e0"; @implementation RCTConvert (MKPinAnnotationColor) RCT_ENUM_CONVERTER(MKPinAnnotationColor, (@{ RCTMapPinRed: @(MKPinAnnotationColorRed), RCTMapPinGreen: @(MKPinAnnotationColorGreen), RCTMapPinPurple: @(MKPinAnnotationColorPurple) }), MKPinAnnotationColorRed, unsignedIntegerValue) @end #endif @interface RCTMapAnnotationView : MKAnnotationView @property (nonatomic, strong) UIView *contentView; @end @implementation RCTMapAnnotationView - (void)setContentView:(UIView *)contentView { [_contentView removeFromSuperview]; _contentView = contentView; [self addSubview:_contentView]; } - (void)layoutSubviews { [super layoutSubviews]; self.bounds = (CGRect){ CGPointZero, _contentView.frame.size, }; } @end @interface RCTMapManager() @end @implementation RCTMapManager RCT_EXPORT_MODULE() - (UIView *)view { RCTMap *map = [RCTMap new]; map.delegate = self; return map; } RCT_EXPORT_VIEW_PROPERTY(showsUserLocation, BOOL) RCT_EXPORT_VIEW_PROPERTY(showsPointsOfInterest, BOOL) RCT_EXPORT_VIEW_PROPERTY(showsCompass, BOOL) RCT_EXPORT_VIEW_PROPERTY(zoomEnabled, BOOL) RCT_EXPORT_VIEW_PROPERTY(rotateEnabled, BOOL) RCT_EXPORT_VIEW_PROPERTY(pitchEnabled, BOOL) RCT_EXPORT_VIEW_PROPERTY(scrollEnabled, BOOL) RCT_EXPORT_VIEW_PROPERTY(maxDelta, CGFloat) RCT_EXPORT_VIEW_PROPERTY(minDelta, CGFloat) RCT_EXPORT_VIEW_PROPERTY(legalLabelInsets, UIEdgeInsets) RCT_EXPORT_VIEW_PROPERTY(mapType, MKMapType) RCT_EXPORT_VIEW_PROPERTY(annotations, NSArray) RCT_EXPORT_VIEW_PROPERTY(overlays, NSArray) RCT_EXPORT_VIEW_PROPERTY(onChange, RCTBubblingEventBlock) RCT_EXPORT_VIEW_PROPERTY(onPress, RCTBubblingEventBlock) RCT_CUSTOM_VIEW_PROPERTY(region, MKCoordinateRegion, RCTMap) { [view setRegion:json ? [RCTConvert MKCoordinateRegion:json] : defaultView.region animated:YES]; } - (NSDictionary *)constantsToExport { NSString *red, *green, *purple; #if __IPHONE_OS_VERSION_MIN_REQUIRED < __IPHONE_9_0 if (![MKPinAnnotationView respondsToSelector:@selector(redPinColor)]) { red = RCTMapPinRed; green = RCTMapPinGreen; purple = RCTMapPinPurple; } else #endif { red = RCTColorToHexString([MKPinAnnotationView redPinColor].CGColor); green = RCTColorToHexString([MKPinAnnotationView greenPinColor].CGColor); purple = RCTColorToHexString([MKPinAnnotationView purplePinColor].CGColor); } return @{ @"PinColors": @{ @"RED": red, @"GREEN": green, @"PURPLE": purple, } }; } #pragma mark MKMapViewDelegate - (void)mapView:(RCTMap *)mapView didSelectAnnotationView:(MKAnnotationView *)view { if (mapView.onPress && [view.annotation isKindOfClass:[RCTMapAnnotation class]]) { RCTMapAnnotation *annotation = (RCTMapAnnotation *)view.annotation; mapView.onPress(@{ @"action": @"annotation-click", @"annotation": @{ @"id": annotation.identifier, @"title": annotation.title ?: @"", @"subtitle": annotation.subtitle ?: @"", @"latitude": @(annotation.coordinate.latitude), @"longitude": @(annotation.coordinate.longitude) } }); } } - (MKAnnotationView *)mapView:(RCTMap *)mapView viewForAnnotation:(RCTMapAnnotation *)annotation { if (![annotation isKindOfClass:[RCTMapAnnotation class]]) { return nil; } MKAnnotationView *annotationView; annotationView.clipsToBounds = YES; if (annotation.viewIndex != NSNotFound) { NSString *reuseIdentifier = NSStringFromClass([RCTMapAnnotationView class]); annotationView = [mapView dequeueReusableAnnotationViewWithIdentifier:reuseIdentifier]; if (!annotationView) { annotationView = [[RCTMapAnnotationView alloc] initWithAnnotation:annotation reuseIdentifier:reuseIdentifier]; } UIView *reactView = mapView.reactSubviews[annotation.viewIndex]; ((RCTMapAnnotationView *)annotationView).contentView = reactView; } else if (annotation.image) { NSString *reuseIdentifier = NSStringFromClass([MKAnnotationView class]); annotationView = [mapView dequeueReusableAnnotationViewWithIdentifier:reuseIdentifier] ?: [[MKAnnotationView alloc] initWithAnnotation:annotation reuseIdentifier:reuseIdentifier]; annotationView.image = annotation.image; } else { NSString *reuseIdentifier = NSStringFromClass([MKPinAnnotationView class]); annotationView = [mapView dequeueReusableAnnotationViewWithIdentifier:reuseIdentifier] ?: [[MKPinAnnotationView alloc] initWithAnnotation:annotation reuseIdentifier:reuseIdentifier]; ((MKPinAnnotationView *)annotationView).animatesDrop = annotation.animateDrop; #if __IPHONE_OS_VERSION_MIN_REQUIRED < __IPHONE_9_0 if (![annotationView respondsToSelector:@selector(pinTintColor)]) { NSString *hexColor = annotation.tintColor ? RCTColorToHexString(annotation.tintColor.CGColor) : RCTMapPinRed; ((MKPinAnnotationView *)annotationView).pinColor = [RCTConvert MKPinAnnotationColor:hexColor]; } else #endif { ((MKPinAnnotationView *)annotationView).pinTintColor = annotation.tintColor ?: [MKPinAnnotationView redPinColor]; } } annotationView.canShowCallout = true; if (annotation.leftCalloutViewIndex != NSNotFound) { annotationView.leftCalloutAccessoryView = mapView.reactSubviews[annotation.leftCalloutViewIndex]; } else if (annotation.hasLeftCallout) { annotationView.leftCalloutAccessoryView = [UIButton buttonWithType:UIButtonTypeDetailDisclosure]; } else { annotationView.leftCalloutAccessoryView = nil; } if (annotation.rightCalloutViewIndex != NSNotFound) { annotationView.rightCalloutAccessoryView = mapView.reactSubviews[annotation.rightCalloutViewIndex]; } else if (annotation.hasRightCallout) { annotationView.rightCalloutAccessoryView = [UIButton buttonWithType:UIButtonTypeDetailDisclosure]; } else { annotationView.rightCalloutAccessoryView = nil; } //http://stackoverflow.com/questions/32581049/mapkit-ios-9-detailcalloutaccessoryview-usage if ([annotationView respondsToSelector:@selector(detailCalloutAccessoryView)]) { if (annotation.detailCalloutViewIndex != NSNotFound) { UIView *calloutView = mapView.reactSubviews[annotation.detailCalloutViewIndex]; NSLayoutConstraint *widthConstraint = [NSLayoutConstraint constraintWithItem:calloutView attribute:NSLayoutAttributeWidth relatedBy:NSLayoutRelationEqual toItem:nil attribute:NSLayoutAttributeNotAnAttribute multiplier:1 constant:calloutView.frame.size.width]; [calloutView addConstraint:widthConstraint]; NSLayoutConstraint *heightConstraint = [NSLayoutConstraint constraintWithItem:calloutView attribute:NSLayoutAttributeHeight relatedBy:NSLayoutRelationEqual toItem:nil attribute:NSLayoutAttributeNotAnAttribute multiplier:1 constant:calloutView.frame.size.height]; [calloutView addConstraint:heightConstraint]; annotationView.detailCalloutAccessoryView = calloutView; } else { annotationView.detailCalloutAccessoryView = nil; } } return annotationView; } - (MKOverlayRenderer *)mapView:(__unused MKMapView *)mapView rendererForOverlay:(RCTMapOverlay *)overlay { if ([overlay isKindOfClass:[RCTMapOverlay class]]) { MKPolylineRenderer *polylineRenderer = [[MKPolylineRenderer alloc] initWithPolyline:overlay]; polylineRenderer.strokeColor = overlay.strokeColor; polylineRenderer.lineWidth = overlay.lineWidth; return polylineRenderer; } else { return nil; } } - (void)mapView:(RCTMap *)mapView annotationView:(MKAnnotationView *)view calloutAccessoryControlTapped:(UIControl *)control { if (mapView.onPress) { // Pass to JS RCTMapAnnotation *annotation = (RCTMapAnnotation *)view.annotation; mapView.onPress(@{ @"side": (control == view.leftCalloutAccessoryView) ? @"left" : @"right", @"action": @"callout-click", @"annotationId": annotation.identifier }); } } - (void)mapView:(RCTMap *)mapView didUpdateUserLocation:(MKUserLocation *)location { if (mapView.followUserLocation) { MKCoordinateRegion region; region.span.latitudeDelta = RCTMapDefaultSpan; region.span.longitudeDelta = RCTMapDefaultSpan; region.center = location.coordinate; [mapView setRegion:region animated:YES]; // Move to user location only for the first time it loads up. mapView.followUserLocation = NO; } } - (void)mapView:(RCTMap *)mapView regionWillChangeAnimated:(__unused BOOL)animated { [self _regionChanged:mapView]; mapView.regionChangeObserveTimer = [NSTimer timerWithTimeInterval:RCTMapRegionChangeObserveInterval target:self selector:@selector(_onTick:) userInfo:@{ RCTMapViewKey: mapView } repeats:YES]; [[NSRunLoop mainRunLoop] addTimer:mapView.regionChangeObserveTimer forMode:NSRunLoopCommonModes]; } - (void)mapView:(RCTMap *)mapView regionDidChangeAnimated:(__unused BOOL)animated { [mapView.regionChangeObserveTimer invalidate]; mapView.regionChangeObserveTimer = nil; [self _regionChanged:mapView]; // Don't send region did change events until map has // started rendering, as these won't represent the final location if (mapView.hasStartedRendering) { [self _emitRegionChangeEvent:mapView continuous:NO]; }; } - (void)mapViewWillStartRenderingMap:(RCTMap *)mapView { mapView.hasStartedRendering = YES; [self _emitRegionChangeEvent:mapView continuous:NO]; } #pragma mark Private - (void)_onTick:(NSTimer *)timer { [self _regionChanged:timer.userInfo[RCTMapViewKey]]; } - (void)_regionChanged:(RCTMap *)mapView { BOOL needZoom = NO; CGFloat newLongitudeDelta = 0.0f; MKCoordinateRegion region = mapView.region; // On iOS 7, it's possible that we observe invalid locations during // initialization of the map. Filter those out. if (!CLLocationCoordinate2DIsValid(region.center)) { return; } // Calculation on float is not 100% accurate. If user zoom to max/min and then // move, it's likely the map will auto zoom to max/min from time to time. // So let's try to make map zoom back to 99% max or 101% min so that there is // some buffer, and moving the map won't constantly hit the max/min bound. if (mapView.maxDelta > FLT_EPSILON && region.span.longitudeDelta > mapView.maxDelta) { needZoom = YES; newLongitudeDelta = mapView.maxDelta * (1 - RCTMapZoomBoundBuffer); } else if (mapView.minDelta > FLT_EPSILON && region.span.longitudeDelta < mapView.minDelta) { needZoom = YES; newLongitudeDelta = mapView.minDelta * (1 + RCTMapZoomBoundBuffer); } if (needZoom) { region.span.latitudeDelta = region.span.latitudeDelta / region.span.longitudeDelta * newLongitudeDelta; region.span.longitudeDelta = newLongitudeDelta; mapView.region = region; } // Continously observe region changes [self _emitRegionChangeEvent:mapView continuous:YES]; } - (void)_emitRegionChangeEvent:(RCTMap *)mapView continuous:(BOOL)continuous { if (mapView.onChange) { MKCoordinateRegion region = mapView.region; if (!CLLocationCoordinate2DIsValid(region.center)) { return; } mapView.onChange(@{ @"continuous": @(continuous), @"region": @{ @"latitude": @(RCTZeroIfNaN(region.center.latitude)), @"longitude": @(RCTZeroIfNaN(region.center.longitude)), @"latitudeDelta": @(RCTZeroIfNaN(region.span.latitudeDelta)), @"longitudeDelta": @(RCTZeroIfNaN(region.span.longitudeDelta)), } }); } } @end