MKMapView
でタップイベントをキャプチャしようとしています。これにより、ユーザーがタップした場所にMKPinAnnotation
をドロップできます。基本的に私はMKOverlayViews
でオーバーレイされた地図(建物を示すオーバーレイ)を持っています。ユーザーがMKPinAnnotaion
をドロップして詳細情報を表示することで、そのオーバーレイに関する詳細情報をユーザーに提供したいと思います吹き出し内。ありがとうございました。
UIGestureRecognizer
を使用して、マップビューでのタッチを検出できます。
ただし、シングルタップの代わりに、ダブルタップ(UITapGestureRecognizer
)または長押し(UILongPressGestureRecognizer
)を探すことをお勧めします。シングルタップは、ピンまたはコールアウト自体をシングルタップしようとするユーザーを妨害する可能性があります。
マップビューをセットアップする場所(viewDidLoad
など)で、ジェスチャレコグナイザをマップビューに接続します。
UITapGestureRecognizer *tgr = [[UITapGestureRecognizer alloc]
initWithTarget:self action:@selector(handleGesture:)];
tgr.numberOfTapsRequired = 2;
tgr.numberOfTouchesRequired = 1;
[mapView addGestureRecognizer:tgr];
[tgr release];
または長押しを使用するには:
UILongPressGestureRecognizer *lpgr = [[UILongPressGestureRecognizer alloc]
initWithTarget:self action:@selector(handleGesture:)];
lpgr.minimumPressDuration = 2.0; //user must press for 2 seconds
[mapView addGestureRecognizer:lpgr];
[lpgr release];
の中に handleGesture:
方法:
- (void)handleGesture:(UIGestureRecognizer *)gestureRecognizer
{
if (gestureRecognizer.state != UIGestureRecognizerStateEnded)
return;
CGPoint touchPoint = [gestureRecognizer locationInView:mapView];
CLLocationCoordinate2D touchMapCoordinate =
[mapView convertPoint:touchPoint toCoordinateFromView:mapView];
MKPointAnnotation *pa = [[MKPointAnnotation alloc] init];
pa.coordinate = touchMapCoordinate;
pa.title = @"Hello";
[mapView addAnnotation:pa];
[pa release];
}
viewDidLoad:
で長押し(UILongPressGestureRecognizer
)を設定しましたが、最初から1回だけタッチを検出します。
すべてのタッチを検出するために長押しはどこで設定できますか?(ユーザーが画面にタッチしてピンを押すのを待つたびにマップの準備ができていることを意味します)
viewDidLoad:
メソッド!
- (void)viewDidLoad {
[super viewDidLoad];mapView.mapType = MKMapTypeStandard;
UILongPressGestureRecognizer *longPressGesture = [[UILongPressGestureRecognizer alloc] initWithTarget:self action:@selector(handleLongPressGesture:)];
[self.mapView addGestureRecognizer:longPressGesture];
[longPressGesture release];
mapAnnotations = [[NSMutableArray alloc] init];
MyLocation *location = [[MyLocation alloc] init];
[mapAnnotations addObject:location];
[self gotoLocation];
[self.mapView addAnnotations:self.mapAnnotations];
}
およびhandleLongPressGesture
メソッド:
-(void)handleLongPressGesture:(UIGestureRecognizer*)sender {
// This is important if you only want to receive one tap and hold event
if (sender.state == UIGestureRecognizerStateEnded)
{NSLog(@"Released!");
[self.mapView removeGestureRecognizer:sender];
}
else
{
// Here we get the CGPoint for the touch and convert it to latitude and longitude coordinates to display on the map
CGPoint point = [sender locationInView:self.mapView];
CLLocationCoordinate2D locCoord = [self.mapView convertPoint:point toCoordinateFromView:self.mapView];
// Then all you have to do is create the annotation and add it to the map
MyLocation *dropPin = [[MyLocation alloc] init];
dropPin.latitude = [NSNumber numberWithDouble:locCoord.latitude];
dropPin.longitude = [NSNumber numberWithDouble:locCoord.longitude];
// [self.mapView addAnnotation:dropPin];
[mapAnnotations addObject:dropPin];
[dropPin release];
NSLog(@"Hold!!");
NSLog(@"Count: %d", [mapAnnotations count]);
}
}
マップビューでシングルクリック/タップを使用する場合は、次のコードスニペットを使用します。 (ココアとスウィフト)
let gr = NSClickGestureRecognizer(target: self, action: "createPoint:")
gr.numberOfClicksRequired = 1
gr.delaysPrimaryMouseButtonEvents = false // allows +/- button press
gr.delegate = self
map.addGestureRecognizer(gr)
ジェスチャーデリゲートメソッドでは、ダブルタップジェスチャーを優先する簡単なテスト…
func gestureRecognizer(gestureRecognizer: NSGestureRecognizer, shouldRequireFailureOfGestureRecognizer otherGestureRecognizer: NSGestureRecognizer) -> Bool {
let other = otherGestureRecognizer as? NSClickGestureRecognizer
if (other?.numberOfClicksRequired > 1) {
return true; // allows double click
}
return false
}
また、マップをさまざまな「状態」にしたい場合は、他のデリゲートメソッドでジェスチャーをフィルター処理することもできます。
何らかの理由で、UIGestureRecognizerはSwiftでは機能しませんでした。 UIGestureRecognizerの方法を使用する場合。 touchesEndedメソッドを使用すると、MKNewAnnotationContainerViewが返されます。このMKNewAnnotationContainerViewが私のMKMapViewをブロックしているようです。幸いなことに、これはMKMapViewのサブビューです。そのため、MKNewAnnotationContainerViewのスーパービューをself.viewまでループして、MKMapViewを取得しました。そして、タップしてmapViewを固定できました。
Swift 4.1
override func touchesEnded(_ touches: Set<UITouch>, with event: UIEvent?) {
let t = touches.first
print(t?.location(in: self.view) as Any)
print(t?.view?.superview?.superview.self as Any)
print(mapView.self as Any)
var tempView = t?.view
while tempView != self.view {
if tempView != mapView {
tempView = tempView?.superview!
}else if tempView == mapView{
break
}
}
let convertedCoor = mapView.convert((t?.location(in: mapView))!, toCoordinateFrom: mapView)
let pin = MKPointAnnotation()
pin.coordinate = convertedCoor
mapView.addAnnotation(pin)
}