如何使用 RxSwift 检测双击

How to detect double tap with RxSwift

我正在尝试使用 RxSwift 检测双击

如果没有 RxSwift,我会这样:

private func setupFakePanView() {
    let singleTapGesture = UITapGestureRecognizer()
    let doubleTapGesture = UITapGestureRecognizer()

    singleTapGesture.numberOfTapsRequired = 1
    doubleTapGesture.numberOfTapsRequired = 2

    singleTapGesture.addTarget(self, action: #selector(self.tapped))
    doubleTapGesture.addTarget(self, action: #selector(self.doubleTapped))

    someView.addGestureRecognizer(singleTapGesture)
    someView.addGestureRecognizer(doubleTapGesture)

    singleTapGesture.require(toFail: doubleTapGesture)
}

@objc private func tapped() {
    // Do something
}

@objc private func doubleTapped() {
    // Do something else
}

有没有一种方法可以让我用 RxSwift、RxCocoa 和 RxGesture 达到同样的效果?我尝试了以下方法,但当然行不通:

someView.rx
    .tapGesture(numberOfTouchesRequired: 1, numberOfTapsRequired: 1)
    .when(.recognized)
    .subscribe(onNext: { _ in
        // Do something
    })
    .disposed(by: bag)

someView.rx
    .tapGesture(numberOfTouchesRequired: 1, numberOfTapsRequired: 2)
    .when(.recognized)
    .subscribe(onNext: { _ in
        // Do something else
    })
    .disposed(by: bag)

有没有办法让第一个 tapGesture 知道第二个必须失败?

我找到了 2 个解决方案来解决这个问题!

一个。使用自定义 UITapGestureRecognizer

let doubleTapGesture = UITapGestureRecognizer()
doubleTapGesture.numberOfTapsRequired = 2

let singleTapGesture = UITapGestureRecognizer()
singleTapGesture.numberOfTapsRequired = 1
singleTapGesture.require(toFail: doubleTapGesture)

let singleTap = someView.rx
    .gesture(singleTapGesture)
    .when(.recognized)
    .subscribe(onNext: { _ in
        // Do something
    })
    .disposed(by: bag)


let doubleTap = someView.rx
    .gesture(doubleTapGesture)
    .when(.recognized)
    .subscribe(onNext: { _ in
        // Do something else
    })
    .disposed(by: bag)

或..

乙。使用自定义 UIGestureRecognizerDelegate

谢谢Kishan for suggesting Jegnux's answer

1 - 为单击手势设置自定义委托...

someView.rx
    .tapGesture(
        numberOfTouchesRequired: 1,
        numberOfTapsRequired: 1,
        configuration: { [weak self] gesture, delegate in
            gesture.delegate = self
        }
    )
    .subscribe(onNext: { _ in
        // Do something
    })
    .disposed(by: bag)

// double tap same as before

2 - 实施 gestureRecognizer(_:shouldRequireFailureOf:)

extension MyController: UIGestureRecognizerDelegate {
    func gestureRecognizer(_ gestureRecognizer: UIGestureRecognizer, shouldRequireFailureOf otherGestureRecognizer: UIGestureRecognizer) -> Bool {
        if let gesture = otherGestureRecognizer as? UITapGestureRecognizer, gesture.numberOfTapsRequired == 2 {
             return true
        }
        return false
    }
}

两种解决方案都可以正常工作。

此代码应该有效:

tap
    .flatMapFirst {
        tap
        .takeUntil(tap.startWith(()).debounce(.milliseconds(300), scheduler: MainScheduler.instance))
        .startWith(())
        .reduce(0) { acc, _ in acc + 1 }
    }
    .map { min([=10=], 2) }

只需使用一个手势识别器(其事件称为 tap),它会在点击发生时立即发出。 map 上方的代码将此问题概括为在本例中输出特定时间段(300 毫秒)内连续点击的次数。 map只是为了保证只有一个1或者一个2出来。然后随心所欲地执行任何条件逻辑。我用 UIButton 测试成功了。