Странное поведение в MKMapView - исчезают контакты
В моем приложении у нас есть экран с MKMapView.
На этой карте показаны булавки для группы местоположений (местоположение - это модель, определенная в приложении).
При запуске в симуляторе это работает нормально. Однако при запуске этого на устройстве контакты, кажется, исчезают и появляются снова почти каждый раз, когда карта панорамируется. Даже легкие прикосновения и движения могут заставить карту показать и скрыть булавки.
Карта не особенно занята, даже если она показывает одну булавку, она покажет и скроет ее.
Кто-нибудь знает, почему это может происходить? Я вставил свой код ниже... Это приложение Rubymotion, поэтому код написан на Ruby.
- ОБНОВИТЬ -
Я добавил Objective C, эквивалентный этому Ruby-коду ниже. Извинения, если есть несколько опечаток или идиоматических ошибок, прошло много времени с тех пор, как я написал какой-либо OC.
- ОБНОВЛЕНИЕ 2 -
Наблюдая за регистратором, я вижу, что mapView:viewForAnnotation
вызывается каждый раз, когда булавка карты исчезает / появляется снова.
Также каждый раз mapView:regionDidChangeAnimated
Я вижу, что идентификаторы объектов для аннотаций одинаковы - поэтому я не думаю, что они удаляются (что не должно быть)
# ====================
# = MKMapKitDelegate =
# ====================
# Don't react if the user has moved less than three meters
USER_MOVE_THRESHOLD = 3
# The user location has changed
def mapView(mapView, didUpdateUserLocation: newLocation)
NSLog("mapView:didUpdateUserLocation")
return unless userLocation
coord = newLocation.coordinate
newLocationAsCL = CLLocation.alloc.initWithCoordinate(coord, altitude: 1, horizontalAccuracy:1, verticalAccuracy: -1, timestamp: nil)
meters = newLocationAsCL.distanceFromLocation(@lastUserCLLocation)
# If user has moved less than 3m, return
if meters > 0 and meters < USER_MOVE_THRESHOLD
log "Distance was less than #{USER_MOVE_THRESHOLD} meters (#{meters}) - returning ***"
return
end
# If the coord is the same as the previous user location
if userLocation.coordinate.latitude == coord.latitude && userLocation.coordinate.longitude == coord.longitude
log "User hasn't moved - returning ***"
else
log 'User has moved'
end
log "Did update user location: #{coord.latitude},#{coord.longitude}"
if coord.latitude.to_f == 0.0 and coord.longitude.to_f == 0.0
log 'Invalid coordinate received - returning ***'
else
fetchLocationsFromAPI
end
end
def mapView(mapView, regionDidChangeAnimated: animated)
NSLog("mapView:regionDidChangeAnimated:#{animated}")
# do nothing here yet...
end
# create map pins...
def mapView(mapView, viewForAnnotation: annotation)
log "mapView:viewForAnnotation: #{annotation.inspect}"
if annotation.is_a?(Location)
# If there's already an annotation we can use, use it! Otherwise create a new one
annotationView = mapView.dequeueReusableAnnotationViewWithIdentifier(annotation.class.to_s) || begin
annotationView = MKPinAnnotationView.alloc.initWithAnnotation(annotation, reuseIdentifier: annotation.class.to_s)
annotationView.enabled = true
annotationView.canShowCallout = true
annotationView.animatesDrop = false
annotationView.pinColor = MKPinAnnotationColorRed
rightButton = UIButton.buttonWithType(UIButtonTypeDetailDisclosure)
rightButton.addTarget(self, action: 'showLocationScreen:', forControlEvents: UIControlEventTouchUpInside)
annotationView.rightCalloutAccessoryView = rightButton
annotationView
end
annotationView.annotation = annotation
annotationView.rightCalloutAccessoryView.tag = @mapLocations.index(annotation)
return annotationView
end
end
def mapView(mapViewm, didAddAnnotationViews: views)
NSLog("mapView:didAddAnnotationViews - #{views}")
# do nothing here yet...
end
Цель С
#define kUserMoveThreshold 1
-(void)mapView:(MKMapView *)mapView didUpdateUserLocation:(MKUserLocation *)newLocation
{
NSLog(@"mapView:didUpdateUserLocation")
if (!userLocation) {
return;
}
CLLocationCoordinate2D coord = newLocation.coordinate;
CLLocation newLocationAsCL = [[CLLocation alloc] initWithCoordinate: coord altitude: 1 horizontalAccuracy: 1 verticalAccuracy: -1 timestamp: NULL];
CLLocationDistance meters = [newLocationAsCL distanceFromLocation: lastUserCLLocation];
// If user has moved less than 3m, return
if (meters > 0 && meters < kUserMoveThreshold){
NSLog(@"Distance was less than %d meters (%d) - returning ***", kUserMoveThreshold, meters);
return;
}
// If the coord is the same as the previous user location
if (userLocation.coordinate.latitude == coord.latitude && userLocation.coordinate.longitude == coord.longitude){
NSLog(@"User hasn't moved - returning ***");
return;
} else {
NSLog(@"User has moved");
}
NSLog(@"Did update user location: %f,%f", coord.latitude, coord.longitude);
if (coord.latitude == 0.0 && coord.longitude == 0.0){
NSLog(@"Invalid coordinate received - returning ***");
} else {
[self fetchLocationsFromAPI];
}
}
-(void)mapView:(MKMapView *)mapView regionDidChangeAnimated:(BOOL)animated
{
NSLog(@"mapView:regionDidChangeAnimated: %s", animated ? @"TRUE" : @"FALSE");
}
-(MKAnnotationView *)mapView:(MKMapView *)mapView viewForAnnotation:(id < MKAnnotation >)annotation
{
NSLog(@"mapView:viewForAnnotation %s", annotation.description);
if ([annotation isKindOfClass: [Location class]]){
// If there's already an annotation we can use, use it! Otherwise create a new one
MKAnnotationView *annotationView = [mapView dequeueReusableAnnotationViewWithIdentifier: [annotation className]];
if (!annotationView){
annotationView = [[MKPinAnnotationView alloc] initWithAnnotation: annotation reuseIdentifier: [annotation className]];
[annotationView setEnabled: YES];
[annotationView setCanShowCallout: YES];
[annotationView setAnimatesDrop: NO];
[annotationView setPinColor: MKPinAnnotationColorRed];
UIButton *rightButton = [UIButton buttonWithType: UIButtonTypeDetailDisclosure];
[rightButton addTarget: self action: @selector(showLocationScreen:) forControlEvents: UIControlEventTouchUpInside];
[annotationView setRightCalloutAccessoryView: rightButton];
}
[annotationView annotation: annotation];
[[annotationView rightCalloutAccessoryView] setTag: [mapLocations indexOfObject: annotation]];
return annotationView
}
}
-(void)mapView:(MKMapView *)mapView didAddAnnotationViews:(NSArray *)views
{
NSLog(@"mapView:didAddAnnotationViews %@", views)
}
2 ответа
Это была ошибка в Rubymotion, которая, похоже, была исправлена в версии 1.30
= RubyMotion 1.30 =
...
- Исправлена ошибка, из-за которой булавки аннотаций MapKit исчезали, потому что некоторые объекты Ruby (в данном случае MKAnnotations) использовали значение Bignum в качестве возвращаемого значения метода `hash', который не работал бы должным образом при использовании в MapKit.
...
Когда вы создаете аннотации выводов, добавляете ли вы их в массив аннотаций mapView?
CLLocationCoordinate2D locationForSelectedFloorOffice;
locationForSelectedFloorOffice.longitude = [[myGeoCode objectAtIndex:0] floatValue] * 1.0;
locationForSelectedFloorOffice.latitude = [[myGeoCode objectAtIndex:1] floatValue] * 1.0;
MyLocation *annotation = [[MyLocation alloc] initWithName:dStore.selectedFloorRoomName address:@" " coordinate:locationForSelectedFloorOffice];
[self.mapView addAnnotation:annotation];