Вопрос по ios, iphone – Перетащите булавку аннотации на карту

13

Пин аннотации перетаскивается, но я не могу его уронить в место назначения. Проблема в том, как я могу получить координату места назначения, куда я уронил булавку аннотации? Какой-нибудь метод существует?

Edit 1:

Центр булавки аннотации, кажется, никогда не менялся, где бы я ни уронил булавку.

- (void)mapView:(MKMapView *)mapView annotationView:(MKAnnotationView *)annotationView didChangeDragState:(MKAnnotationViewDragState)newState 
fromOldState:(MKAnnotationViewDragState)oldState 
{
if (newState == MKAnnotationViewDragStateEnding)
{
    NSLog(@"x is %f", annotationView.center.x);
    NSLog(@"y is %f", annotationView.center.y);

    CGPoint dropPoint = CGPointMake(annotationView.center.x, annotationView.center.y);
    CLLocationCoordinate2D newCoordinate = [self.mapView convertPoint:dropPoint toCoordinateFromView:annotationView.superview];
    [annotationView.annotation setCoordinate:newCoordinate];

}
}

Edit 2:

Annotation.h

@property (nonatomic,readwrite,assign) CLLocationCoordinate2D coordinate;

Annotation.m

- (void)setCoordinate:(CLLocationCoordinate2D)newCoordinate {
coordinate = newCoordinate; 
}

Edit 3: где бы я ни перетаскивал булавку, выход NSLog для dropAt всегда одинаков.

- (void)mapView:(MKMapView *)mapView annotationView:(MKAnnotationView *)annotationView didChangeDragState:(MKAnnotationViewDragState)newState 
fromOldState:(MKAnnotationViewDragState)oldState 
{
if (newState == MKAnnotationViewDragStateEnding)
{
    CLLocationCoordinate2D droppedAt = annotationView.annotation.coordinate;
    NSLog(@"Pin dropped at %f,%f", droppedAt.latitude, droppedAt.longitude);
}
}

Edit 4:

Annotaion.m @dynamic startAddr;

- (CLLocationCoordinate2D)coordinate
{
coordinate.latitude = [self.startAddr.latitude doubleValue];
coordinate.longitude = [self.startAddr.longitude doubleValue];    
return coordinate;
}

- (void)setCoordinate:(CLLocationCoordinate2D)newCoordinate {
//I need to update the self.startAddr.latitude and longitude here. Problem solved.
self.startAddr.latitude = [NSNumber numberWithDouble:newCoordinate.latitude];
self.startAddr.longitude = [NSNumber numberWithDouble:newCoordinate.longitude];

coordinate = newCoordinate; 
}
@rckoenes спасибо за ваше уведомление. lu yuan

Ваш Ответ

6   ответов
39

Сначала вы должны создать пользовательскую аннотацию, например:

MyAnnotation.h

#import <MapKit/MapKit.h>

@interface MyAnnotation : NSObject <MKAnnotation>{

    CLLocationCoordinate2D coordinate;

}
- (id)initWithCoordinate:(CLLocationCoordinate2D)coord;
- (void)setCoordinate:(CLLocationCoordinate2D)newCoordinate;
@end

MyAnnotation.m

#import "MyAnnotation.h"

@implementation MyAnnotation
@synthesize coordinate;

- (NSString *)subtitle{
    return nil;
}

- (NSString *)title{
    return nil;
}

-(id)initWithCoordinate:(CLLocationCoordinate2D)coord {
    coordinate=coord;
    return self;
}

-(CLLocationCoordinate2D)coord
{
    return coordinate;
}

- (void)setCoordinate:(CLLocationCoordinate2D)newCoordinate {
    coordinate = newCoordinate; 
}

@end

После этого вам просто нужно использовать свою аннотацию так:

// Assuming you have imported MyAnnotation.h and you have a self.map property pointing to a MKMapView
MyAnnotation *myPin = [[MyAnnotation alloc] initWithCoordinate:self.map.centerCoordinate]; // Or whatever coordinates...
[self.map addAnnotation:myPin];

Кроме того, вы должны вернуть представление для вашей аннотации. Вы можете сделать это так:

- (MKAnnotationView *) mapView: (MKMapView *) mapView viewForAnnotation: (id<MKAnnotation>) annotation {
    MKPinAnnotationView *pin = (MKPinAnnotationView *) [self.map dequeueReusableAnnotationViewWithIdentifier: @"myPin"];
    if (pin == nil) {
        pin = [[[MKPinAnnotationView alloc] initWithAnnotation: annotation reuseIdentifier: @"myPin"] autorelease]; // If you use ARC, take out 'autorelease'
    } else {
        pin.annotation = annotation;
    }
    pin.animatesDrop = YES;
    pin.draggable = YES;

    return pin;
}

Затем, как только вы принялиMKMapViewDelegate Протокол в вашем контроллере, вы берете координаты булавки после перетаскивания его так:

- (void)mapView:(MKMapView *)mapView 
 annotationView:(MKAnnotationView *)annotationView 
didChangeDragState:(MKAnnotationViewDragState)newState 
   fromOldState:(MKAnnotationViewDragState)oldState 
{
    if (newState == MKAnnotationViewDragStateEnding)
    {
        CLLocationCoordinate2D droppedAt = annotationView.annotation.coordinate;
        NSLog(@"Pin dropped at %f,%f", droppedAt.latitude, droppedAt.longitude);
    }
}
Вот почему вы реализуетеsetCoordinate метод. Из документации Apple - & gt;Annotations that support dragging should implement this method to update the position of the annotation.
Я обновляю координаты, как в моем посте (просто отредактируйте его). Но, похоже, не работает. lu yuan
Чтобы этот код работал, вы должны установить себя в качестве делегата карт в viewDidLoad. Также добавьте & quot; if ([annotation isKindOfClass: [MKUserLocation class]]) return nil; & quot; в верхней части viewForAnnotation
где вы установили новую координату? lu yuan
Аннотация автоматически устанавливает свои координаты, когда вы заканчиваете его перетаскивать, это и есть точка ... Вам не нужно делать это самостоятельно, вы просто хотите получить эти новые координаты. Я надеюсь, что это имеет смысл.
0

В Свифте моей проблемой было то, чтоcoordinate атрибут былletв моем подклассе, а неvar как это вMKAnnotation.

class MyAnnotation  : NSObject, MKAnnotation {

    let title       : String?
    let subtitle    : String?
    var coordinate  : CLLocationCoordinate2D

    init(title: String, subtitle: String, coordinate: CLLocationCoordinate2D) {
        self.title = title
        self.subtitle = subtitle
        self.coordinate = coordinate
        super.init()
    }
}
1
- (MKAnnotationView *) mapView:(MKMapView *)mapView viewForAnnotation:(id <MKAnnotation>) annotation
{
    MKAnnotationView *a = [ [ MKAnnotationView alloc ] initWithAnnotation:annotation reuseIdentifier:@"currentloc"];
    //  a.title = @"test";
    if ( a == nil )
        a = [ [ MKAnnotationView alloc ] initWithAnnotation:annotation reuseIdentifier: @"currentloc" ];

    NSLog(@"%f",a.annotation.coordinate.latitude);
    NSLog(@"%f",a.annotation.coordinate.longitude);

    CLLocation* currentLocationMap = [[[CLLocation alloc] initWithLatitude:a.annotation.coordinate.latitude longitude:a.annotation.coordinate.longitude] autorelease];
    [self coordinatesToDMS:currentLocationMap];


    MKPinAnnotationView *annView=[[MKPinAnnotationView alloc] initWithAnnotation:a reuseIdentifier:@"currentloc"];
    if(a.annotation.coordinate.longitude == mapView.userLocation.coordinate.longitude ||  a.annotation.coordinate.latitude == mapView.userLocation.coordinate.latitude  )
    {
        if ([annotation isKindOfClass:MKUserLocation.class])
        {
            //user location view is being requested,
            //return nil so it uses the default which is a blue dot...
            return nil;
        }       
        //a.image = [UIImage imageNamed:@"userlocation.png"];
        //a.pinColor=[UIColor redColor];
    }
    else 
    {

        // annView.image =[UIImage imageNamed:@"map-pin.png"];
        //PinFirst=FALSE;
        //annView.pinColor = [UIColor redColor];
        // annView.annotation=annotation;
    }

    annView.animatesDrop = YES;
    annView.draggable = YES;

    annView.canShowCallout = YES;

    return annView;
}

- (void)mapView:(MKMapView *)mapView regionDidChangeAnimated:(BOOL)animated
{
    NSLog(@"map Drag");
}

- (void)mapView:(MKMapView *)mapView annotationView:(MKAnnotationView *)view didChangeDragState:(MKAnnotationViewDragState)newState
fromOldState:(MKAnnotationViewDragState)oldState;
{
    NSLog(@"pin Drag");

    if (newState == MKAnnotationViewDragStateEnding)
    {
        CLLocationCoordinate2D droppedAt = view.annotation.coordinate;
        NSLog(@"Pin dropped at %f,%f", droppedAt.latitude, droppedAt.longitude);

        CLLocation* draglocation = [[[CLLocation alloc] initWithLatitude:droppedAt.latitude longitude:droppedAt.longitude] autorelease];



    }
}
Спасибо за полное решение
1

Внезапно реализуйте метод setCoordinates вашей аннотации и сделайте callBAck, когда ваши координаты аннотации изменились

edit: остерегайтесь свойства dragState вашей аннотации во время его перемещения.

Спасибо за ваш ответ. Пожалуйста, проверьте обновление моего поста. lu yuan
1

Не хотите это сделать:

Установка аннотации делегата, затем

- (void)setCoordinate:(CLLocationCoordinate2D)newCoordinate {
   coordinate = newCoordinate; 
   if (self.delegate respondsToSelector:@selector(mapAnnotationCoordinateHaveChanged))
    [self.delegate performSelector(@selector(mapAnnotationCoordinateHaveChanged))];
}

В вашем делегате:

- (void) mapAnnotationCoordinateHaveChanged{
 //some coordinates update code
}
0

Пример для Swift 2.2, Xcode 7.3.1:

  • Create custom class adopting MKAnnotation protocol

    (note: Swift doesn't have automatic notification for KVO-compliant properties, so I have added my own - this is not required):

    import MapKit
    
    class MyAnnotation: NSObject, MKAnnotation {
    
        // MARK: - Required KVO-compliant Property
        var coordinate: CLLocationCoordinate2D {
            willSet(newCoordinate) {
                let notification = NSNotification(name: "VTAnnotationWillSet", object: nil)
                NSNotificationCenter.defaultCenter().postNotification(notification)
            }
    
            didSet {
                let notification = NSNotification(name: "VTAnnotationDidSet", object: nil)
                NSNotificationCenter.defaultCenter().postNotification(notification)
            }
        }
    
    
        // MARK: - Required Initializer
        init(coordinate: CLLocationCoordinate2D) {
            self.coordinate = coordinate
        }
    }
    
  • Declare pin view as draggable:

    // MARK: - MKMapViewDelegate Actions
    func mapView(mapView: MKMapView, viewForAnnotation annotation: MKAnnotation) -> MKAnnotationView? {
    
        let pinView = MKPinAnnotationView(annotation: annotation, reuseIdentifier: pinViewReuseIdentifier)
        pinView.draggable = true
    
        return pinView
    }
    

Теперь, когда вывод перетаскивается, он вызывает метод делегата:

func mapView(mapView: MKMapView, didSelectAnnotationView view: MKAnnotationView)

Здесь вы можете получить новые конечные координаты.


В качестве альтернативы, вы можете создатьuserInfo словарь для пользовательских аннотацийcoordinate уведомления собственности. Затем наблюдатели могут получить значения при получении уведомления.

Похожие вопросы