Добавление нескольких аннотаций к карте, но все они показывают одни и те же данные


Обновление 2:

Вот существующий код внутри карт, но это похоже на то, как аннотации выходят из строя с помощью булавок. Один раз булавка будет зеленой, в следующий раз, когда я ее запущу, та же булавка будет красной. Откуда берется разъединение?

-(void) viewWillAppear:(BOOL)animated {
    self.annotationArray = [[NSMutableArray alloc] init];

CLLocationCoordinate2D coord = {.latitude =  15.8700320, .longitude =  100.9925410};
MKCoordinateSpan span = {.latitudeDelta =  3, .longitudeDelta =  3};
MKCoordinateRegion region = {coord, span};
[mapViewUI setRegion:region];
    PFQuery *query = [PFQuery queryWithClassName:@"Share"];
    [query findObjectsInBackgroundWithBlock:^(NSArray *objects, NSError *error) {
        if (!error) {
                       self.mapViewData = objects;
                        for (int i=0;i<[objects count];i++)
            {
                // Question * q = [[Question alloc]init];

                PFObject * obj = [self.mapViewData objectAtIndex:i];
                NSLog(@"%@", obj);
                self.theObject = obj;

                NSString *string = obj[@"WhereAt"];


                NSArray *stringArray = [string componentsSeparatedByString: @","];

                CLLocationDegrees myLatitude = [[stringArray objectAtIndex:0] doubleValue];
                CLLocationDegrees myLongitude = [[stringArray objectAtIndex:1] doubleValue];
                CLLocationCoordinate2D coord2 = {.latitude =  myLatitude, .longitude =  myLongitude};
                NSDateFormatter *dateFormatter = [[NSDateFormatter alloc] init];
                dateFormatter.dateFormat = @"MMM dd, yyyy";

                MKPointAnnotation *annotation2 = [[MKPointAnnotation alloc] init];
                [annotation2 setCoordinate:coord2];
                [annotation2 setTitle:obj[@"FamilyName"]];
                [annotation2 setSubtitle:obj[@"Result"]];
                [mapViewUI addAnnotation:annotation2];


            }


        } else {
            // Log details of the failure
            NSLog(@"Error: %@ %@", error, [error userInfo]);
        }


    }];

}
- (MKAnnotationView *)mapView:(MKMapView *)mapView viewForAnnotation:(id <MKAnnotation>)annotation {
    if (annotation == mapViewUI.userLocation)
    {
        return nil;
    }
    else
    {
    NSUInteger index = [[mapView annotations] indexOfObject:annotation];
    PFObject *objectForCurrentAnnotation = [self.mapViewData objectAtIndex:index];
    MKPinAnnotationView *annotationView = [[MKPinAnnotationView alloc] init];

    annotationView.enabled = YES;
    annotationView.canShowCallout = YES;
    NSDateFormatter *dateFormatter = [[NSDateFormatter alloc] init];
    dateFormatter.dateFormat = @"MMM dd, yyyy";
    NSString *theResult = objectForCurrentAnnotation[@"Result"];
    NSLog(@"Result is %@", theResult);

    if ([theResult isEqualToString:@"Accepted Bible"]) {
        NSLog(@"Accepted");
        annotationView.pinTintColor = [UIColor greenColor];

    }
   else if ([theResult isEqualToString:@"Requested Different Material"]) {
       NSLog(@"Different");

        annotationView.pinTintColor = [UIColor blackColor];

    }
   else if ([theResult isEqualToString:@"Not Home"]) {
      NSLog(@"Not Home");

        annotationView.pinTintColor = [UIColor yellowColor];

    }
   else if ([theResult isEqualToString:@"Rejected Bible"]) {
       NSLog(@"Rejected");

        annotationView.pinTintColor = [UIColor redColor];

    }



    return annotationView;
    }

    return nil;
}

Обновление:

Теперь я вижу, что проблема заключается в этом "я".theObject устанавливается один раз, поэтому он просто извлекает данные из этого для всего, что не обрабатывается аннотациями. Я думаю, теперь вопрос в том, как я могу заставить его показать для каждого объекта вместо того, что было в последнем наборе?

Мое приложение использует Parse.com и объектов pfobject. Приложение запрашивает PFObject, чтобы получить все данные для каждой записи в строке, извлекает координаты для данного местоположения для этой записи вместе с различными битами данных и добавляет аннотацию для каждого из них. Когда я устанавливаю MKAnnotationView для этого, так что я могу иметь несколько цветных булавок вне того, что Apple обычно позволяет вам легко делать, я сталкиваюсь с проблемами. Изображение и некоторые данные для аннотация устанавливает каждый контакт на то, что называлось последним. Заголовок и подзаголовок хороши, но не цвет или изображение выноски. Вот мой код, что здесь происходит не так?

-(void) viewWillAppear:(BOOL)animated {
CLLocationCoordinate2D coord = {.latitude =  15.8700320, .longitude =  100.9925410};
MKCoordinateSpan span = {.latitudeDelta =  3, .longitudeDelta =  3};
MKCoordinateRegion region = {coord, span};
[mapView setRegion:region];
    PFQuery *query = [PFQuery queryWithClassName:@"Share"];
    [query findObjectsInBackgroundWithBlock:^(NSArray *objects, NSError *error) {
        if (!error) {
            NSLog(@"YAY");
            // The find succeeded. The first 100 objects are available in objects
            //  questionsList = [[NSMutableArray alloc]init];
            NSLog(@"Objects%lu", (unsigned long)[objects count]);

            for (int i=0;i<[objects count];i++)
            {
                // Question * q = [[Question alloc]init];

                PFObject * obj = [objects objectAtIndex:i];
                self.theObject = obj;

                NSString *string = obj[@"WhereAt"];


                NSArray *stringArray = [string componentsSeparatedByString: @","];

                CLLocationDegrees myLatitude = [[stringArray objectAtIndex:0] doubleValue];
                CLLocationDegrees myLongitude = [[stringArray objectAtIndex:1] doubleValue];
                CLLocationCoordinate2D coord2 = {.latitude =  myLatitude, .longitude =  myLongitude};
                NSDateFormatter *dateFormatter = [[NSDateFormatter alloc] init];
                dateFormatter.dateFormat = @"MMM dd, yyyy";
                NSString *theResult = obj[@"Result"];


                NSString *theDate = [dateFormatter stringFromDate:obj[@"DateVisited"]];
                NSString *combined = [[theResult stringByAppendingString:@" on "] stringByAppendingString:theDate];
                NSString *theTitle = [[obj[@"FamilyName"] stringByAppendingString:@" "] stringByAppendingString:obj[@"StreetAddress"]];

                MKPointAnnotation *annotation = [[MKPointAnnotation alloc] init];
                [annotation setCoordinate:coord2];
                [annotation setTitle:theTitle];
                [annotation setSubtitle:combined];

                [mapView addAnnotation:annotation];

            }


        } else {
            // Log details of the failure
            NSLog(@"Error: %@ %@", error, [error userInfo]);
        }
    }];



}
- (MKAnnotationView *)mapView:(MKMapView *)mapView viewForAnnotation:(id <MKAnnotation>)annotation {


    MKPinAnnotationView *annotationView = [[MKPinAnnotationView alloc] init];

    annotationView.enabled = YES;
    annotationView.canShowCallout = YES;
    NSDateFormatter *dateFormatter = [[NSDateFormatter alloc] init];
    dateFormatter.dateFormat = @"MMM dd, yyyy";
    NSString *theResult = self.theObject[@"Result"];
    NSLog(@"Result is %@", theResult);

    if ([theResult isEqualToString:@"Accepted Bible"]) {
      //  NSLog(@"Accepted");
        annotationView.pinTintColor = [UIColor greenColor];

    }
   else if ([theResult isEqualToString:@"Requested Different Material"]) {
       //NSLog(@"Different");

        annotationView.pinTintColor = [UIColor blackColor];

    }
   else if ([theResult isEqualToString:@"Not Home"]) {
      // NSLog(@"Not Home");

        annotationView.pinTintColor = [UIColor yellowColor];

    }
   else if ([theResult isEqualToString:@"Rejected Bible"]) {
       //NSLog(@"Rejected");

        annotationView.pinTintColor = [UIColor redColor];

    }
    //annotationView.image = [UIImage imageNamed:@"arrest.png"];//here we use a nice image instead of the default pins
    PFFile *thumbnail = self.theObject[@"HousePicture"];
    [thumbnail getDataInBackgroundWithBlock:^(NSData *imageData, NSError *error) {

        UIImage *thumbnailImage = [UIImage imageWithData:imageData];
        self.theImage = thumbnailImage;
        //annotationView.image = thumbnailImage;
        UIView *leftCAV = [[UIView alloc] initWithFrame:CGRectMake(0,0,60,60)];
        UIImageView *theImageView = [[UIImageView alloc] initWithFrame:CGRectMake(0, 0, 60, 60)];
        theImageView.image = self.theImage;
        annotationView.leftCalloutAccessoryView = leftCAV;
        [leftCAV addSubview:theImageView];
        annotationView.canShowCallout = YES;

    }];


    return annotationView;


    return nil;
}
2 8

2 ответа:

Подкласс MKPointAnnotation например MyPointAnnotation Добавьте свойство result

@interface MyPointAnnotation : MKPointAnnotation
    @property (nonatomic, strong) NSString *result;
@end

В viewWillAppear: измените эти строки

NSString *theResult = obj[@"Result"];

MyPointAnnotation *myAnnotation = [[MyPointAnnotation alloc] init];
[myAnnotation setCoordinate:coord2];
[myAnnotation setTitle:theTitle];
[myAnnotation setSubtitle:combined];
[myAnnotation setResult:theResult];

[mapView addAnnotation:myAnnotation];

В методе делегирования viewForAnnotation: получаем такой результат

MyPointAnnotation *myAnnotation = (MyPointAnnotation *)annotation;
NSString *theResult = myAnnotation.result;

Вы можете решить эту проблему несколькими различными способами.

1. MKMapView класс имеет свойство, называемое аннотациями. При вызове этого свойства он вернет вам отсортированный массив всех аннотаций на вашей карте, который связан с порядком ваших данных в исходном массиве. Таким образом, вы просто должны хранить все извлеченные объекты из синтаксического анализа. В интерфейсе вашего контроллера вида просто добавьте новое свойство:

@property (nonatomic, string) NSArray *mapViewData;

В вашем методе viewWillAppear:

    [query findObjectsInBackgroundWithBlock:^(NSArray *objects, NSError *error) {
            if (!error) {
                self.mapViewData = objects;
               // here you form your annotations
            }
    }];

Тогда просто вытяните эти данные метод делегирования

 - (nullable MKAnnotationView *)mapView:(MKMapView *)mapView viewForAnnotation:(id <MKAnnotation>)annotation {
     NSUInteger index = [[mapView annotations] indexOfObject:annotation];
     PFObject *objectForCurrentAnnotation = [self.mapViewData objectAtIndex:index];
     // do your customisation
}

2. Переопределение и подкласс объекта MKPointAnnotation.

@interface SuperAnnotation : MKPointAnnotation

@property (nonatomic, strong) PFObject *dataObject;

@end

@implementation SuperAnnotation

- (void)setDataObject:(PFObject *)dataObject {
    _dataObject = dataObject;

    self.title = // calculate your string for title
    self.subtitle = // and the same for subtitle
}

@end

А затем в методе делегата просто получить этот объект из объекта аннотации и делать то, что вы хотите

- (nullable MKAnnotationView *)mapView:(MKMapView *)mapView viewForAnnotation:(id <MKAnnotation>)annotation {
     PFObject *objectForCurrentAnnotation = ((SuperAnnotation *) annotation).dataObject;    
     // do your customisation
}

3. Вы можете создать словарь, который будет хранить пары MKAnnotation и PFObject, которые вам нужны для настройки (не рекомендуется для большого объема данных).

@property (nonatomic, string) NSMutableDictionary *dataDictionary;

В viewWillAppear:

[query findObjectsInBackgroundWithBlock:^(NSArray *objects, NSError *error) {
        if (!error) {
            self.dataDictionary = [NSMutableDictionary new];
           // here you form your annotations

            [self.dataDictionary setObject:yourPFObject forKey:yourAnnoationObject];
        }
}];

А затем еще раз вытяните данные в делегате:

 - (nullable MKAnnotationView *)mapView:(MKMapView *)mapView viewForAnnotation:(id <MKAnnotation>)annotation {
     PFObject *objectForCurrentAnnotation = [self.dataDictionary objectForKey:annotation];
     // do your customisation
}