禁用手势以下拉表单/页面表单模式演示

Jor*_*n H 39 uigesturerecognizer ios ios13

在iOS 13中,可以使用平移手势关闭使用表单和页面样式的模式演示。在我的一个表单中,这是有问题的,因为用户会拖入该框,从而干扰手势。它将屏幕下拉而不是画一条垂直线。

如何在显示为图纸的模态视图控制器中禁用垂直滑动以消除手势?

设置isModalInPresentation = true仍然允许将工作表拉下,但不会关闭。

Jor*_*n H 30

通常,对于未来的互联网旅行者来说,不应禁用滑动以关闭功能。但是,如果您的手势或触摸处理受到此功能的影响,我确实从Apple工程师那里收到了一些有关如何完成此操作的建议。

如果可以阻止系统的平移手势识别器启动,则可以防止手势解雇。一些方法可以做到这一点:

  1. 如果您的画布绘制是使用手势识别器(例如您自己的UIGestureRecognizer子类)完成的,请began在图纸的关闭手势之前输入阶段。如果您尽快识别UIPanGestureRecognizer,您将获胜,并且工作表的关闭手势将被颠覆。

  2. 如果您的画布绘制是使用手势识别器完成的,请使用-shouldBeRequiredToFailByGestureRecognizer:(或相关的委托方法)设置动态失败要求,NO如果传入的手势识别器是,则在此处返回UIPanGestureRecognizer

  3. 如果您的画布绘制是通过手动触摸处理完成的(例如touchesBegan:),请-gestureRecognizerShouldBegin在触摸处理视图上进行覆盖,NO如果传入的手势识别器为,则返回UIPanGestureRecognizer

在我的设置3中,事实证明效果很好。这样一来,用户可以向下滑动绘图画布之外的任何位置以关闭它(例如导航栏),同时允许用户在不移动图纸的情况下进行绘图,就像人们期望的那样。

我不建议尝试找到禁用该手势的手势,因为它似乎是动态的,并且例如在不同大小的类之间切换时可以重新启用自身,并且在将来的版本中可能会更改。

  • #3 对于我的应用程序来说效果很好,它允许用户在视图中绘制。 (4认同)
  • 做得好!很有意思。#3 非常适合我。 (4认同)
  • @Jordan H 你能用一些代码更好地解释#3 吗? (2认同)

Zol*_*ler 23

在呈现的 ViewController viewDidLoad 中使用它:

if #available(iOS 13.0, *) {
    self.isModalInPresentation = true
}
Run Code Online (Sandbox Code Playgroud)

  • 正如问题中所指出的,“isModalInPresentation = true”仍然允许下拉工作表,它只是不会关闭,这可能正是您所需要的,或者根据您的用例,它可能会出现问题,就像我的绘图一样帆布。 (7认同)
  • 你是对的。我认为最简单的解决方案是使用旧的全屏模式样式:`self.modalPresentationStyle = .fullScreen` (3认同)

Ene*_*nso 16

就我而言,我有一个带有视图的模式屏幕,可以接收触摸以捕获客户签名。

禁用导航控制器中的手势识别器解决了这个问题,完全阻止了模态交互关闭被触发。

以下方法在我们的模态视图控制器中实现,并通过我们自定义签名视图的委托调用。

来自touchesBegan

private func disableDismissalRecognizers() {
    navigationController?.presentationController?.presentedView?.gestureRecognizers?.forEach {
        $0.isEnabled = false
    }
}
Run Code Online (Sandbox Code Playgroud)

来自touchesEnded

private func enableDismissalRecognizers() {
    navigationController?.presentationController?.presentedView?.gestureRecognizers?.forEach {
        $0.isEnabled = true
    }
}
Run Code Online (Sandbox Code Playgroud)

这是显示行为的 GIF: 在此处输入图片说明

这个被标记为重复的问题更好地描述了我遇到的问题:从主视图拖动时禁用 iOS 13 上呈现的视图控制器的交互式关闭

  • 它不适用于 iOS 14;`navigationController?.presentationController?.presentedView` 始终为 `nil`。我尝试使用 `navigationController?.presentationController?.presentedViewController.view` 代替,但在那里找不到 `_UISheetInteractionBackgroundDismissRecognizer` ...... (2认同)

val*_*ine 16

无需重新发明轮子。就像UIAdaptivePresentationControllerDelegatedestinationViewController上采用协议然后实现相关方法一样简单:

func presentationControllerShouldDismiss(_ presentationController: UIPresentationController) -> Bool {
    return false
}
Run Code Online (Sandbox Code Playgroud)

例如,假设您的destinationViewController已准备好进行segue,如下所示:

override func prepare(for segue: UIStoryboardSegue, sender: Any?) {
    if segue.identifier == "yourIdentifier",
       let destinationVC = segue.destination as? DetailViewController
    {
        //do other stuff

        destinationVC.presentationController?.delegate = destinationVC

    }
}
Run Code Online (Sandbox Code Playgroud)

然后在destinationVC(应采用上述协议)上,您可以实现所描述的方法func presentationControllerShouldDismiss(_ presentationController:) -> Bool或任何其他方法,以便正确处理您的自定义行为。

  • 您可以在“viewDidLoad”中使用“presentationController?.delegate = self”,而不是在上一个视图控制器的“prepare(for:sender:)”中设置委托。这样所有的代码都在一个视图控制器中。 (2认同)

Mo *_*ani 13

可以在模态视图控制器的presentedView属性中找到此手势。正如我调试的那样,gestureRecognizers此属性的数组只有一项,并且打印它会导致如下所示:

UIPanGestureRecognizer:0x7fd3b8401aa0(_UISheetInteractionBackgroundDismissRecognizer);

因此,要禁用此手势,您可以执行以下操作:

let vc = UIViewController()

self.present(vc, animated: true, completion: {
  vc.presentationController?.presentedView?.gestureRecognizers?[0].isEnabled = false
})
Run Code Online (Sandbox Code Playgroud)

要重新启用它,只需将其设置isEnabledtrue

vc.presentationController?.presentedView?.gestureRecognizers?[0].isEnabled = true
Run Code Online (Sandbox Code Playgroud)

请注意,iOS 13仍处于测试阶段,因此可能会在即将发布的版本中添加更简单的方法。

尽管此解决方案目前似乎可以使用,但我不建议您这样做,因为它在某些情况下可能不起作用,或者在将来的iOS版本中可能会更改,并可能影响您的应用。

  • UITableView也会创建_UISwipeDismissalGestureRecognizer。此外,如果您使用根视图控制器创建导航控制器,将堆栈以页面/表单的形式呈现出来,然后将另一个视图控制器推入其顶部,则从下滑到关闭的手势将通过手势识别器在“ UIView”层次结构的较高位置创建。如果没有苹果公司明确支持禁用向下滑动到关闭触摸事件处理功能,唯一可靠的解决方案(从Xcode 11 beta 3开始)是使用UIModalPresentationFullScreen的UIModalPresentationStyle。 (5认同)
  • 如果您想更安全地了解要禁用的手势,您甚至可以按名称或类型进行搜索。```对于 guestures 中的手势,其中gesture.name == "_UISheetInteractionBackgroundDismissRecognizer" {gesture.isEnabled = false }``` (5认同)

小智 12

您可以更改演示文稿样式,如果它在全屏模式下,下拉关闭将被禁用

navigationCont.modalPresentationStyle = .fullScreen
Run Code Online (Sandbox Code Playgroud)

  • 我认为这是正确的答案。实际上,我将它与 isModalInPresentation 一起使用只是为了确定,它工作得很好。对我来说关键是将这些设置在父级中。当我尝试在呈现的控制器中的 viewDidLoad 中设置时,它不起作用。 (2认同)

小智 5

您可以首先在 viewDidAppear() 方法中获取对处理页面工作表解除的 UIPanGestureRecognizer 的引用。请注意,该引用在 viewWillAppear() 或 viewDidLoad() 中为 nil。然后你只需禁用它即可。

override func viewDidAppear(_ animated: Bool) {
    super.viewDidAppear(animated)
    presentationController?.presentedView?.gestureRecognizers?.first.isEnabled = false
}
Run Code Online (Sandbox Code Playgroud)

如果您想要更多自定义而不是完全禁用它,例如,在页面表中使用导航栏时,请将该 UIPanGestureRecognizer 的委托设置为您自己的视图控制器。这样,您可以在 contentView 中专门禁用手势识别器,同时通过实现在 navBar 区域中保持其活动状态

func gestureRecognizer(_ gestureRecognizer: UIGestureRecognizer, shouldReceive touch: UITouch) -> Bool {}
Run Code Online (Sandbox Code Playgroud)