summaryrefslogtreecommitdiff
path: root/platform/ios/Integration Tests/Annotation Tests/MGLAnnotationViewIntegrationTests.m
diff options
context:
space:
mode:
Diffstat (limited to 'platform/ios/Integration Tests/Annotation Tests/MGLAnnotationViewIntegrationTests.m')
-rw-r--r--platform/ios/Integration Tests/Annotation Tests/MGLAnnotationViewIntegrationTests.m409
1 files changed, 407 insertions, 2 deletions
diff --git a/platform/ios/Integration Tests/Annotation Tests/MGLAnnotationViewIntegrationTests.m b/platform/ios/Integration Tests/Annotation Tests/MGLAnnotationViewIntegrationTests.m
index fefb938773..7ec45de072 100644
--- a/platform/ios/Integration Tests/Annotation Tests/MGLAnnotationViewIntegrationTests.m
+++ b/platform/ios/Integration Tests/Annotation Tests/MGLAnnotationViewIntegrationTests.m
@@ -2,9 +2,26 @@
#import "MGLTestUtility.h"
#import "MGLMapAccessibilityElement.h"
#import "MGLTestLocationManager.h"
+#import "MGLCompactCalloutView.h"
+
+@interface MGLTestCalloutView : MGLCompactCalloutView
+@property (nonatomic) BOOL implementsMarginHints;
+@end
+
+@implementation MGLTestCalloutView
+- (BOOL)respondsToSelector:(SEL)aSelector {
+ if (!self.implementsMarginHints &&
+ (aSelector == @selector(marginInsetsHintForPresentationFromRect:))) {
+ return NO;
+ }
+ return [super respondsToSelector:aSelector];
+}
+@end
+
@interface MGLMapView (Tests)
- (MGLAnnotationTag)annotationTagAtPoint:(CGPoint)point persistingResults:(BOOL)persist;
+@property (nonatomic) UIView<MGLCalloutView> *calloutViewForSelectedAnnotation;
@end
@interface MGLAnnotationViewIntegrationTests : MGLMapViewIntegrationTest
@@ -12,6 +29,368 @@
@implementation MGLAnnotationViewIntegrationTests
+#pragma mark - Offscreen/panning selection tests
+
+typedef struct PanTestData {
+ CGPoint relativeCoord;
+ BOOL showsCallout;
+ BOOL implementsMargins;
+ BOOL moveIntoView;
+ BOOL expectMapToHavePanned;
+ BOOL calloutOnScreen;
+} PanTestData;
+
+#define PAN_TEST_TERMINATOR {{FLT_MAX, FLT_MAX}, NO, NO, NO, NO, NO}
+static const CGFloat kAnnotationScale = 0.125f;
+
+- (void)internalTestOffscreenSelectionTitle:(NSString*)title withTestData:(PanTestData)test animateSelection:(BOOL)animateSelection {
+
+ CGPoint relativeCoordinate = test.relativeCoord;
+ BOOL showsCallout = test.showsCallout;
+ BOOL calloutImplementsMarginHints = test.implementsMargins;
+ BOOL moveIntoView = test.moveIntoView;
+ BOOL expectMapToHavePanned = test.expectMapToHavePanned;
+ BOOL expectCalloutToBeFullyOnscreen = test.calloutOnScreen;
+
+ // Reset the map to a consistent state - want the map to be zoomed in, so that
+ // it's free to be panned without hitting boundaries.
+ [self.mapView setCenterCoordinate:CLLocationCoordinate2DMake(0, 0) zoomLevel:14 animated:NO];
+ [self waitForMapViewToBeRenderedWithTimeout:1.0];
+
+ XCTAssert(self.mapView.annotations.count == 0);
+
+ NSString * const MGLTestAnnotationReuseIdentifer = @"MGLTestAnnotationReuseIdentifer";
+ CGSize size = self.mapView.bounds.size;
+ CGSize annotationSize = CGSizeMake(floor(size.width*kAnnotationScale), floor(size.height*kAnnotationScale));
+
+ self.viewForAnnotation = ^MGLAnnotationView*(MGLMapView *view, id<MGLAnnotation> annotation) {
+
+ if (![annotation isKindOfClass:[MGLPointAnnotation class]]) {
+ return nil;
+ }
+
+ // No dequeue
+ MGLAnnotationView *annotationView = [[MGLAnnotationView alloc] initWithAnnotation:annotation reuseIdentifier:MGLTestAnnotationReuseIdentifer];
+ annotationView.bounds = (CGRect){ .origin = CGPointZero, .size = annotationSize };
+ annotationView.backgroundColor = UIColor.redColor;
+ annotationView.enabled = YES;
+
+ return annotationView;
+ };
+
+ // Coordinate for annotation screen coordinate
+ CGPoint annotationPoint = CGPointMake(relativeCoordinate.x * size.width, relativeCoordinate.y * size.height);
+ CLLocationCoordinate2D coordinate = [self.mapView convertPoint:annotationPoint toCoordinateFromView:self.mapView];
+
+ MGLPointAnnotation *point = [[MGLPointAnnotation alloc] init];
+ point.title = title;
+ point.coordinate = coordinate;
+
+ self.mapViewAnnotationCanShowCalloutForAnnotation = ^BOOL(MGLMapView *mapView, id<MGLAnnotation> annotation) {
+ return showsCallout;
+ };
+
+ self.mapViewCalloutViewForAnnotation = ^id<MGLCalloutView>(MGLMapView *mapView, id<MGLAnnotation> annotation) {
+ if (!showsCallout)
+ return nil;
+
+ MGLTestCalloutView *calloutView = [[MGLTestCalloutView alloc] init];
+ calloutView.representedObject = annotation;
+ calloutView.implementsMarginHints = calloutImplementsMarginHints;
+ return calloutView;
+ };
+
+ [self.mapView addAnnotation:point];
+
+ // Check assumptions before selection
+ UIView *annotationViewBeforeSelection = [self.mapView viewForAnnotation:point];
+ XCTAssertNotNil(annotationViewBeforeSelection);
+
+ CLLocationCoordinate2D mapCenterBeforeSelection = self.mapView.centerCoordinate;
+ XCTAssert(CLLocationCoordinate2DIsValid(mapCenterBeforeSelection));
+
+ // Also note, that at this point, the internal mechanism that determines if
+ // an annotation view is offscreen and should be put back in the reuse queue
+ // will have run, and `viewForAnnotation` may return nil
+
+ [self.mapView selectAnnotation:point moveIntoView:moveIntoView animateSelection:animateSelection];
+
+ // Animated selection takes MGLAnimationDuration (0.3 seconds), so wait a little
+ // longer. We don't need to wait as long if we're not animated (but we do
+ // want the runloop to tick over)
+ [self waitFor:animateSelection ? 0.4: 0.05];
+
+ UIView *annotationViewAfterSelection = [self.mapView viewForAnnotation:point];
+ CLLocationCoordinate2D mapCenterAfterSelection = self.mapView.centerCoordinate;
+ XCTAssert(CLLocationCoordinate2DIsValid(mapCenterAfterSelection));
+
+ // If the annotation is still "offscreen" at this point, then the above annotation view
+ // may be nil, which is expected.
+ BOOL (^CGRectContainsRectWithAccuracy)(CGRect, CGRect, CGFloat) = ^(CGRect rect1, CGRect rect2, CGFloat accuracy) {
+ CGRect expandedRect1 = CGRectInset(rect1, -accuracy, -accuracy);
+ return CGRectContainsRect(expandedRect1, rect2);
+ };
+
+ CGFloat epsilon = 0.00001;
+ if (expectMapToHavePanned) {
+ CLLocationDegrees latitudeDelta = fabs(mapCenterAfterSelection.latitude - mapCenterBeforeSelection.latitude);
+ CLLocationDegrees longitudeDelta = fabs(mapCenterAfterSelection.longitude - mapCenterBeforeSelection.longitude);
+
+ XCTAssert( (latitudeDelta > epsilon) || (longitudeDelta > epsilon), @"Deltas: lat=%f, long=%f", latitudeDelta, longitudeDelta); // One of them should have moved
+
+ // If the map panned - the intention is that the annotation is on-screen,
+ // and so should have an annotation view and that it is fully on screen
+ CGRect annotationFrameAfterSelection = annotationViewAfterSelection.frame;
+
+ XCTAssertNotNil(annotationViewAfterSelection);
+
+ XCTAssert(CGRectContainsRectWithAccuracy(self.mapView.bounds, annotationFrameAfterSelection, 0.25), @"Mapview:%@ frame:%@", NSStringFromCGRect(self.mapView.bounds), NSStringFromCGRect(annotationFrameAfterSelection));
+
+ // Check the callout
+ if (showsCallout) {
+ UIView *calloutView = self.mapView.calloutViewForSelectedAnnotation;
+ XCTAssertNotNil(calloutView);
+
+ // If kAnnotationScale == 0.25, then the following assert can fail.
+ // This is really a warning (see https://github.com/mapbox/mapbox-gl-native/issues/13744 )
+ // If you need this NOT to fail the tests, consider replacing with MGLTestWarning
+ XCTAssert(expectCalloutToBeFullyOnscreen == CGRectContainsRectWithAccuracy(self.mapView.bounds, calloutView.frame, 0.25),
+ @"Expect contains:%d, Mapview:%@ annotation:%@ callout:%@",
+ expectCalloutToBeFullyOnscreen,
+ NSStringFromCGRect(self.mapView.bounds),
+ NSStringFromCGRect(annotationFrameAfterSelection),
+ NSStringFromCGRect(calloutView.frame));
+ }
+ }
+ else {
+ // The map shouldn't have moved, so use equality (rather than an error check)
+ XCTAssertEqual(mapCenterBeforeSelection.latitude, mapCenterAfterSelection.latitude);
+ XCTAssertEqual(mapCenterBeforeSelection.longitude, mapCenterAfterSelection.longitude);
+
+ // Annotation shouldn't have moved
+ CGPoint annotationPoint2 = [self.mapView convertCoordinate:point.coordinate toPointToView:self.mapView];
+ CGFloat xDelta = fabs(annotationPoint2.x - annotationPoint.x);
+ CGFloat yDelta = fabs(annotationPoint2.y - annotationPoint.y);
+
+ XCTAssert((xDelta < epsilon) && (yDelta < epsilon));
+
+ if (showsCallout) {
+ UIView *calloutView = self.mapView.calloutViewForSelectedAnnotation;
+
+ if (annotationViewAfterSelection) {
+ XCTAssertNotNil(calloutView);
+
+ // If kAnnotationScale == 0.25, then the following assert can fail.
+ // This is really a warning (see https://github.com/mapbox/mapbox-gl-native/issues/13744 )
+ // If you need this NOT to fail the tests, consider replacing with MGLTestWarning
+ XCTAssert((expectCalloutToBeFullyOnscreen == CGRectContainsRectWithAccuracy(self.mapView.bounds, calloutView.frame, 0.25)),
+ @"Mapview:%@ annotation:%@ callout:%@",
+ NSStringFromCGRect(self.mapView.bounds),
+ NSStringFromCGRect(annotationViewAfterSelection.frame),
+ NSStringFromCGRect(calloutView.frame));
+ }
+ else {
+ // If there's no annotation view, should we expect a callout?
+ XCTAssertNil(calloutView);
+ XCTAssertFalse(expectCalloutToBeFullyOnscreen);
+ }
+ }
+ }
+
+ // Remove the annotation
+ [self.mapView removeAnnotation:point];
+
+ XCTAssert(self.mapView.annotations.count == 0);
+}
+
+// See https://github.com/mapbox/mapbox-gl-native/pull/13727#issuecomment-454028698
+// What follows are tests based on this table.
+// This is not a full-set of possible combinations, just the most important/likely
+// ones
+- (void)internalRunTests:(PanTestData*)testData
+{
+ // Test both animated and not-animated.
+ for (int i = 0; i<2; i++) {
+ int row = 0;
+ PanTestData *test = testData;
+ while (test->relativeCoord.x != FLT_MAX) {
+ NSString *activityTitle = [NSString stringWithFormat:@"Row %d/%d", row, i];
+ [XCTContext runActivityNamed:activityTitle
+ block:^(id<XCTActivity> _Nonnull activity) {
+ [self internalTestOffscreenSelectionTitle:activityTitle
+ withTestData:*test
+ animateSelection:!i];
+ }];
+ ++test;
+ ++row;
+ }
+ }
+}
+
+- (void)testBasicSelection {
+ // Tests moveIntoView:NO
+ // WITHOUT a callout
+
+ PanTestData tests[] = {
+ // Coord showsCallout impl margins? moveIntoView expectMapToPan calloutOnScreen
+ // Offscreen
+ { {-1.0f, 0.5f}, NO, NO, NO, NO, NO },
+ { { 2.0f, 0.5f}, NO, NO, NO, NO, NO },
+ { { 0.5f,-1.0f}, NO, NO, NO, NO, NO },
+ { { 0.5f, 2.0f}, NO, NO, NO, NO, NO },
+
+ // Partial
+ { { 0.0f, 0.5f}, NO, NO, NO, NO, NO },
+ { { 1.0f, 0.5f}, NO, NO, NO, NO, NO },
+ { { 0.5f, 0.0f}, NO, NO, NO, NO, NO },
+ { { 0.5f, 1.0f}, NO, NO, NO, NO, NO },
+
+ // Onscreen
+ { { 0.5f, 0.5f}, NO, NO, NO, NO, NO },
+
+ PAN_TEST_TERMINATOR
+ };
+
+ [self internalRunTests:tests];
+}
+
+- (void)testBasicSelectionWithCallout {
+ // Tests moveIntoView:NO
+ // WITH the default callout (implements marginshint)
+
+ PanTestData tests[] = {
+ // Coord showsCallout impl margins? moveIntoView expectMapToPan calloutOnScreen
+ { {-1.0f, 0.5f}, YES, YES, NO, NO, NO },
+ { { 0.0f, 0.5f}, YES, YES, NO, NO, NO },
+ { { 0.5f, 1.0f}, YES, YES, NO, NO, YES }, // Because annotation was off the bottom of screen, and callout is above annotation
+ { { 0.5f, 0.5f}, YES, YES, NO, NO, YES },
+
+ PAN_TEST_TERMINATOR
+ };
+
+ [self internalRunTests:tests];
+}
+
+- (void)testSelectionMoveIntoView {
+ // Tests moveIntoView:YES
+ // without a callout
+
+ // From https://github.com/mapbox/mapbox-gl-native/pull/13727#issuecomment-454028698
+ //
+ // | Annotation position | Has callout? | Callout implements `marginInsets...`? | Map pans when selected with moveIntoView=YES? |
+ // |---------------------|--------------|---------------------------------------|-----------------------------------------------|
+ // | Offscreen | No | n/a | Yes (no margins) |
+ // | Partially | No | n/a | No |
+ // | Onscreen | No | n/a | No |
+ //
+
+ PanTestData tests[] = {
+ // Coord showsCallout impl margins? moveIntoView expectMapToPan calloutOnScreen
+ // Offscreen
+ { {-1.0f, 0.5f}, NO, NO, YES, YES, NO },
+ { { 2.0f, 0.5f}, NO, NO, YES, YES, NO },
+ { { 0.5f,-1.0f}, NO, NO, YES, YES, NO },
+ { { 0.5f, 2.0f}, NO, NO, YES, YES, NO },
+
+ // Partial
+ { { 0.0f, 0.5f}, NO, NO, YES, NO, NO },
+ { { 1.0f, 0.5f}, NO, NO, YES, NO, NO },
+ { { 0.5f, 0.0f}, NO, NO, YES, NO, NO },
+ { { 0.5f, 1.0f}, NO, NO, YES, NO, NO },
+
+ // Onscreen
+ { { 0.5f, 0.5f}, NO, NO, YES, NO, NO },
+
+ PAN_TEST_TERMINATOR
+ };
+
+ [self internalRunTests:tests];
+}
+
+- (void)testSelectionMoveIntoViewWithCallout {
+ // Tests moveIntoView:YES
+ // WITH the default callout (implements marginshint)
+
+ // From https://github.com/mapbox/mapbox-gl-native/pull/13727#issuecomment-454028698
+ //
+ // | Annotation position | Has callout? | Callout implements `marginInsets...`? | Map pans when selected with moveIntoView=YES? |
+ // |---------------------|--------------|---------------------------------------|-----------------------------------------------|
+ // | Offscreen | Yes | Yes | Yes to ensure callout is fully visible |
+ // | Partially | Yes | Yes | Yes to ensure callout is fully visible |
+ // | Onscreen | Yes | Yes | Yes, but *only* to ensure callout is fully visible |
+ //
+
+ PanTestData tests[] = {
+ // Coord showsCallout impl margins? moveIntoView expectMapToPan calloutOnScreen
+ // Offscreen
+ { {-1.0f, 0.5f}, YES, YES, YES, YES, YES },
+ { { 2.0f, 0.5f}, YES, YES, YES, YES, YES },
+ { { 0.5f,-1.0f}, YES, YES, YES, YES, YES },
+ { { 0.5f, 2.0f}, YES, YES, YES, YES, YES },
+
+ // Partial
+ { { 0.0f, 0.5f}, YES, YES, YES, YES, YES },
+ { { 1.0f, 0.5f}, YES, YES, YES, YES, YES },
+ { { 0.5f, 0.0f}, YES, YES, YES, YES, YES },
+ { { 0.5f, 1.0f}, YES, YES, YES, YES, YES },
+
+ // Onscreen
+ { { 0.5f, 0.5f}, YES, YES, YES, NO, YES },
+
+ // Just at the edge of the screen.
+ // Expects to move, because although onscreen, callout would not be.
+ // However, if the scale is 0.25, then expectToPan should be NO, because
+ // of the width of the annotation
+ //
+ // Coord showsCallout impl margins? moveIntoView expectMapToPan calloutOnScreen
+ { {kAnnotationScale, 0.5f}, YES, YES, YES, (kAnnotationScale == 0.125f), YES },
+
+ PAN_TEST_TERMINATOR
+ };
+
+ [self internalRunTests:tests];
+}
+
+- (void)testSelectionMoveIntoViewWithBasicCallout {
+ // Tests moveIntoView:YES
+ // WITH a callout that DOES NOT implement marginshint
+
+ // From https://github.com/mapbox/mapbox-gl-native/pull/13727#issuecomment-454028698
+ //
+ // | Annotation position | Has callout? | Callout implements `marginInsets...`? | Map pans when selected with moveIntoView=YES? |
+ // |---------------------|--------------|---------------------------------------|-----------------------------------------------|
+ // | Offscreen | Yes | No | Yes, but only to show annotation (not callout) with no margins |
+ // | Partially | Yes | No | No |
+ // | Onscreen | Yes | No | No |
+ //
+
+ PanTestData tests[] = {
+ // Coord showsCallout impl margins? moveIntoView expectMapToPan calloutOnScreen
+ // Offscreen
+ { {-1.0f, 0.5f}, YES, NO, YES, YES, NO },
+ { { 2.0f, 0.5f}, YES, NO, YES, YES, NO },
+ { { 0.5f,-1.0f}, YES, NO, YES, YES, NO },
+ { { 0.5f, 2.0f}, YES, NO, YES, YES, YES }, // Because annotation was off the bottom of screen, and callout is above annotation
+ { { 2.0f, 2.0f}, YES, NO, YES, YES, NO },
+
+ // Partial
+ { { 0.0f, 0.5f}, YES, NO, YES, NO, NO },
+ { { 1.0f, 0.5f}, YES, NO, YES, NO, NO },
+ { { 0.5f, 0.0f}, YES, NO, YES, NO, NO },
+ { { 0.5f, 1.0f}, YES, NO, YES, NO, YES }, // Because annotation was off the bottom of screen, and callout is above annotation
+ { { 1.0f, 1.0f}, YES, NO, YES, NO, NO },
+
+ // Onscreen
+ { { 0.5f, 0.5f}, YES, NO, YES, NO, YES },
+
+ PAN_TEST_TERMINATOR
+ };
+
+ [self internalRunTests:tests];
+}
+
+#pragma mark - Selection with an offset
+
- (void)testSelectingAnnotationWithCenterOffset {
for (CGFloat dx = -100.0; dx <= 100.0; dx += 100.0 ) {
@@ -66,8 +445,8 @@
// Check that the annotation is in the center of the view
CGPoint annotationPoint = [self.mapView convertCoordinate:point.coordinate toPointToView:self.mapView];
- XCTAssertEqualWithAccuracy(annotationPoint.x, size.width/2.0, epsilon);
- XCTAssertEqualWithAccuracy(annotationPoint.y, size.height/2.0, epsilon);
+ XCTAssertEqualWithAccuracy(annotationPoint.x, size.width/2.0, 0.25);
+ XCTAssertEqualWithAccuracy(annotationPoint.y, size.height/2.0, 0.25);
// Now test taps around the annotation
CGPoint tapPoint = CGPointMake(annotationPoint.x + offset.dx, annotationPoint.y + offset.dy);
@@ -117,6 +496,30 @@
XCTAssertEqual(originalFrame.origin.y + offset.y, offsetFrame.origin.y);
}
+#pragma mark - Utilities
+
+- (void)runRunLoop {
+ [[NSRunLoop mainRunLoop] runMode:NSDefaultRunLoopMode beforeDate:[NSDate distantFuture]];
+}
+
+- (void)waitFor:(NSTimeInterval)seconds {
+ XCTestExpectation *timerExpired = [self expectationWithDescription:@"Timer expires"];
+
+ NSTimer *timer = [NSTimer scheduledTimerWithTimeInterval:0.1
+ target:self
+ selector:@selector(runRunLoop)
+ userInfo:nil
+ repeats:YES];
+
+ double duration = seconds * (double)NSEC_PER_SEC;
+ dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)duration), dispatch_get_main_queue(), ^{
+ [timerExpired fulfill];
+ });
+
+ [self waitForExpectations:@[timerExpired] timeout:seconds + 1.0];
+ [timer invalidate];
+}
+
- (void)waitForCollisionDetectionToRun {
XCTAssertNil(self.renderFinishedExpectation, @"Incorrect test setup");
@@ -129,6 +532,8 @@
});
[self waitForExpectations:@[timerExpired, self.renderFinishedExpectation] timeout:5];
+
+ self.renderFinishedExpectation = nil;
}
@end