#import "MGLPolyline.h" #import "MGLMultiPoint_Private.h" #import "MGLGeometry_Private.h" #import "MGLPolyline+MGLAdditions.h" #import @implementation MGLPolyline @dynamic overlayBounds; + (instancetype)polylineWithCoordinates:(const CLLocationCoordinate2D *)coords count:(NSUInteger)count { return [[self alloc] initWithCoordinates:coords count:count]; } - (mbgl::LineString)lineString { NSUInteger count = self.pointCount; CLLocationCoordinate2D *coordinates = self.coordinates; mbgl::LineString geometry; geometry.reserve(self.pointCount); for (NSUInteger i = 0; i < count; i++) { geometry.push_back(mbgl::Point(coordinates[i].longitude, coordinates[i].latitude)); } return geometry; } - (mbgl::Annotation)annotationObjectWithDelegate:(id )delegate { mbgl::LineAnnotation annotation { [self lineString] }; annotation.opacity = { static_cast([delegate alphaForShapeAnnotation:self]) }; annotation.color = { [delegate strokeColorForShapeAnnotation:self] }; annotation.width = { static_cast([delegate lineWidthForPolylineAnnotation:self]) }; return annotation; } - (mbgl::Geometry)geometryObject { return [self lineString]; } - (NSDictionary *)geoJSONDictionary { return @{@"type": @"LineString", @"coordinates": self.mgl_coordinates}; } - (BOOL)isEqual:(id)other { return self == other || ([other isKindOfClass:[MGLPolyline class]] && [super isEqual:other]); } @end @interface MGLMultiPolyline () @property (nonatomic, copy, readwrite) NS_ARRAY_OF(MGLPolyline *) *polylines; @end @implementation MGLMultiPolyline { MGLCoordinateBounds _overlayBounds; } @synthesize overlayBounds = _overlayBounds; + (instancetype)multiPolylineWithPolylines:(NS_ARRAY_OF(MGLPolyline *) *)polylines { return [[self alloc] initWithPolylines:polylines]; } - (instancetype)initWithPolylines:(NS_ARRAY_OF(MGLPolyline *) *)polylines { if (self = [super init]) { _polylines = polylines; mbgl::LatLngBounds bounds = mbgl::LatLngBounds::empty(); for (MGLPolyline *polyline in _polylines) { bounds.extend(MGLLatLngBoundsFromCoordinateBounds(polyline.overlayBounds)); } _overlayBounds = MGLCoordinateBoundsFromLatLngBounds(bounds); } return self; } - (instancetype)initWithCoder:(NSCoder *)decoder { if (self = [super initWithCoder:decoder]) { _polylines = [decoder decodeObjectOfClass:[NSArray class] forKey:@"polylines"]; } return self; } - (void)encodeWithCoder:(NSCoder *)coder { [super encodeWithCoder:coder]; [coder encodeObject:_polylines forKey:@"polylines"]; } - (BOOL)isEqual:(id)other { if (self == other) return YES; if (![other isKindOfClass:[MGLMultiPolyline class]]) return NO; MGLMultiPolyline *otherMultipoline = other; return ([super isEqual:otherMultipoline] && [self.polylines isEqualToArray:otherMultipoline.polylines]); } - (NSUInteger)hash { NSUInteger hash = [super hash]; for (MGLPolyline *polyline in self.polylines) { hash += [polyline hash]; } return hash; } - (BOOL)intersectsOverlayBounds:(MGLCoordinateBounds)overlayBounds { return MGLCoordinateBoundsIntersectsCoordinateBounds(_overlayBounds, overlayBounds); } - (mbgl::Geometry)geometryObject { mbgl::MultiLineString multiLineString; multiLineString.reserve(self.polylines.count); for (MGLPolyline *polyline in self.polylines) { multiLineString.push_back([polyline lineString]); } return multiLineString; } - (NSDictionary *)geoJSONDictionary { NSMutableArray *coordinates = [NSMutableArray array]; for (MGLPolylineFeature *feature in self.polylines) { [coordinates addObject: feature.mgl_coordinates]; } return @{@"type": @"MultiLineString", @"coordinates": coordinates}; } @end