web-dev-qa-db-ja.com

MKMapViewでタップジェスチャーをキャプチャする方法

MKMapViewでタップイベントをキャプチャしようとしています。これにより、ユーザーがタップした場所にMKPinAnnotationをドロップできます。基本的に私はMKOverlayViewsでオーバーレイされた地図(建物を示すオーバーレイ)を持っています。ユーザーがMKPinAnnotaionをドロップして詳細情報を表示することで、そのオーバーレイに関する詳細情報をユーザーに提供したいと思います吹き出し内。ありがとうございました。

31
sbkb

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];
}
59
user467105

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]);
    }   
}
5
o0oKodako0o

マップビューでシングルクリック/タップを使用する場合は、次のコードスニペットを使用します。 (ココアとスウィフト)

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
}

また、マップをさまざまな「状態」にしたい場合は、他のデリゲートメソッドでジェスチャーをフィルター処理することもできます。

2
bshirley

何らかの理由で、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)
}
0
William Tong