问题
I have a mapview where the annotation's coordinates are constantly being updated but when I use setCoordinate, the annotation does not move. How do I refresh the annotations to reflect their coordinates?
- (void)updateUnits {
PFQuery *query = [PFQuery queryWithClassName:@"devices"];
[query whereKeyExists:@"location"];
[query findObjectsInBackgroundWithBlock:^(NSArray *objects, NSError *error) {
if (!error) {
for (PFObject *row in objects) {
PFGeoPoint *geoPoint = [row objectForKey:@"location"];
CLLocationCoordinate2D coord = { geoPoint.latitude, geoPoint.longitude };
for (id<MKAnnotation> ann in mapView.annotations) {
if ([ann.title isEqualToString:[row objectForKey:@"deviceid"]]) {
[ann setCoordinate:coord];
NSLog(@"latitude is: %f" , coord.latitude);
NSLog(@"Is called");
break;
}
else {
//[self.mapView removeAnnotation:ann];
}
}
}
}
else {
}
}];
}
回答1:
Updated (to reflect the solution):
Having your own custom annotations and implementing setCoordinate and/or synthesizing coordinate may cause issues.
Source: http://developer.apple.com/library/ios/#documentation/UserExperience/Conceptual/LocationAwarenessPG/AnnotatingMaps/AnnotatingMaps.html
Previous solution:
You can simply remove all of the annotations and then re-add them.
[mapView removeAnnotations:[mapView.annotations]];
[mapView addAnnotations:(NSArray *)];
or remove them and re-add them one by one:
for (id<MKAnnotation> annotation in mapView.annotations)
{
[mapView removeAnnotation:annotation];
// change coordinates etc
[mapView addAnnotation:annotation];
}
回答2:
Dispatch the add annotation to the main thread rather then attempt to modify the UI on a background thread
dispatch_async(dispatch_get_main_queue()) {
self.mapView.addAnnotation(annotation)
}
回答3:
Swift 3.0 version of Nathan's answer (thank you Nathan):
DispatchQueue.main.async {
mapView.addAnnotation(annotation)
}
Side note, Nathan's answer should have far more upvotes. I actually needed this help with removing an annotation, the same issue exists and is fixed by dispatching the update to the main queue.
回答4:
Try using [self.mapView removeAnnotations:self.mapView.annotations]
:
- (void)viewDidAppear:(BOOL)animated
{
[super viewDidAppear:animated];
[self.mapView removeAnnotations:self.mapView.annotations];
PFQuery *query = [PFQuery queryWithClassName:@"caches"];
[query findObjectsInBackgroundWithBlock:^(NSArray *comments, NSError *error)
{
for (PFObject *comment in comments)
{
NSString *tit = comment[@"titulo"];
NSString *lat = comment[@"latitud"];
NSString *lon = comment[@"longitud"];
float latFloat = [lat floatValue];
float lonFloat = [lon floatValue];
CLLocationCoordinate2D Pin;
Pin.latitude = latFloat;
Pin.longitude = lonFloat;
MKPointAnnotation *annotationPoint = [[MKPointAnnotation alloc]init];
annotationPoint.coordinate = Pin;
annotationPoint.title = tit;
[self.mapView addAnnotation:annotationPoint];
}
}];
}
回答5:
For dark mode purposes I needed to refresh the views of annotations. I had to call also the delegate method to recreate the view of the annotations.
- (void)traitCollectionDidChange:(UITraitCollection *)previousTraitCollection {
[super traitCollectionDidChange:previousTraitCollection];
if (@available(iOS 13.0, *)) {
if ([self.traitCollection hasDifferentColorAppearanceComparedToTraitCollection:previousTraitCollection]) {
for (id<MKAnnotation> annotation in self.mapView.annotations) {
[self.mapView removeAnnotation:annotation];
[self.mapView addAnnotation:annotation];
[self mapView:self.mapView viewForAnnotation:annotation];
}
}
}
}
回答6:
Just Add
mapView.reloadStyle(self)
来源:https://stackoverflow.com/questions/14131345/ios-refresh-annotations-on-mapview