如何在触摸时向MKMapView(IOS)添加推针?

59

我需要获取用户在MKMapView上触摸的点的坐标。 我没有使用Interface Builder。 你能给我一个例子吗?

2个回答

196

您可以使用 UILongPressGestureRecognizer 来实现此功能。在创建或初始化地图视图的任何位置,首先将识别器附加到它上面:

UILongPressGestureRecognizer *lpgr = [[UILongPressGestureRecognizer alloc] 
    initWithTarget:self action:@selector(handleLongPress:)];
lpgr.minimumPressDuration = 2.0; //user needs to press for 2 seconds
[self.mapView addGestureRecognizer:lpgr];
[lpgr release];

然后在手势处理程序中:

- (void)handleLongPress:(UIGestureRecognizer *)gestureRecognizer
{
    if (gestureRecognizer.state != UIGestureRecognizerStateBegan)
        return;

    CGPoint touchPoint = [gestureRecognizer locationInView:self.mapView];   
    CLLocationCoordinate2D touchMapCoordinate = 
        [self.mapView convertPoint:touchPoint toCoordinateFromView:self.mapView];

    YourMKAnnotationClass *annot = [[YourMKAnnotationClass alloc] init];
    annot.coordinate = touchMapCoordinate;
    [self.mapView addAnnotation:annot];
    [annot release];
}

YourMKAnnotationClass是你定义的一个类,遵循MKAnnotation协议。如果你的应用程序只在iOS 4.0或更高版本上运行,可以使用预定义的MKPointAnnotation类。

关于如何创建自己的MKAnnotation类,可以参考示例应用程序MapCallouts


7
非常好的回答,谢谢。个人而言,我将if语句改为==,这样它返回的就是不是UIGestureRecognizerStateBegan。这么做将在指定时间后放置图钉,即使用户仍然持有地图,这对我来说是理想的(也是官方Maps应用程序的操作方式)。 - William Denniss
我只想说,我已经将你的答案实现到我的项目中了,它非常完美地工作了。非常感谢你提供如此优秀的答案。 - DoubleDunk
这在模拟器中完美运行,但在我的实体手机上没有回调。有什么想法吗?我正在使用带ARC的iOS5。 - rjgonzo
@rjgonzo:应该可以在iOS5、ARC和设备上正常工作。尝试从设备中删除应用程序并进行清理、重建和重新安装。在设备上运行时,添加断点或NSLog以确保在添加lpgr时mapView不为空。 - user467105
这个能做成动画吗? - Oliver Ni
释放...那些回忆! - Nicolas Miari

34

感谢Anna提供如此好的答案!如果有人感兴趣,这里有Swift版本(答案已更新为Swift 4.1语法)。

创建UILongPressGestureRecognizer:

let longPressRecogniser = UILongPressGestureRecognizer(target: self, action: #selector(MapViewController.handleLongPress(_:)))
longPressRecogniser.minimumPressDuration = 1.0
mapView.addGestureRecognizer(longPressRecogniser)

处理手势:

@objc func handleLongPress(_ gestureRecognizer : UIGestureRecognizer){
    if gestureRecognizer.state != .began { return }

    let touchPoint = gestureRecognizer.location(in: mapView)
    let touchMapCoordinate = mapView.convert(touchPoint, toCoordinateFrom: mapView)

    let album = Album(coordinate: touchMapCoordinate, context: sharedContext)

    mapView.addAnnotation(album)
}

1
哇...我没注意到这一点,浪费了很长时间来转换它。 - Oliver Ni
2
可以使用 _let longPressRecogniser = UILongPressGestureRecognizer(target: self, action: "handleLongPress:")_。 - Dx_
@Dx_ 是的,你可以这样做,因为识别器并没有被修改。被修改的是识别器中的属性。 - 3366784
我在Swift 3中遇到了错误信息。错误是:“使用未解决的标识符'gestureRecogniser'”。有人有解决方案吗? - PhilipS
嗨@PhilipS,我已经更新了我的答案到Swift 3.0语法。希望它能有所帮助。 - Allan Spreys
我在Swift 4.2中收到错误消息,错误信息为:“使用了未解决的标识符'Album'”。 - PPPPPPPPP

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接