iOS14 - 查看层次结构更改以防止 UIView 扩展工作

iOS14 - ViewHierarchy changes do prevent UIView extension to work

使用 iOS14.0.1、Xcode12.0.1 和 Swift5.3,

使用 iOS14,以下 UIView 扩展不再正常工作:


@discardableResult
func anchor(top: NSLayoutYAxisAnchor?, leading: NSLayoutXAxisAnchor?, bottom: NSLayoutYAxisAnchor?, trailing: NSLayoutXAxisAnchor?, padding: UIEdgeInsets = .zero, size: CGSize = .zero) -> AnchoredConstraints {
    
    translatesAutoresizingMaskIntoConstraints = false
    var anchoredConstraints = AnchoredConstraints()
    
    if let top = top {
        anchoredConstraints.top = topAnchor.constraint(equalTo: top, constant: padding.top)
    }
    
    if let leading = leading {
        anchoredConstraints.leading = leadingAnchor.constraint(equalTo: leading, constant: padding.left)
    }
    
    if let bottom = bottom {
        anchoredConstraints.bottom = bottomAnchor.constraint(equalTo: bottom, constant: -padding.bottom)
    }
    
    if let trailing = trailing {
        anchoredConstraints.trailing = trailingAnchor.constraint(equalTo: trailing, constant: -padding.right)
    }
    
    if size.width != 0 {
        anchoredConstraints.width = widthAnchor.constraint(equalToConstant: size.width)
    }
    
    if size.height != 0 {
        anchoredConstraints.height = heightAnchor.constraint(equalToConstant: size.height)
    }
    
    [anchoredConstraints.top, anchoredConstraints.leading, anchoredConstraints.bottom, anchoredConstraints.trailing, anchoredConstraints.width, anchoredConstraints.height].forEach{ [=11=]?.isActive = true }
    
    return anchoredConstraints
}

我只在 iOS14 发布后才观察到这个问题。之前,使用 iOS12 和 iOS13 我从来没有遇到过问题,我确实经常使用这个扩展。

但现在 iOS14 会导致 UIControls 不再工作的问题。

因此,如果我使用扩展并执行以下操作 - 那么我的 ViewHierarchy 中的任何 UIControl 都不起作用(即没有目标操作、switch-didChanges 等将不再启动)

SettingsStackView.anchor(top: safeAreaLayoutGuide.topAnchor, leading: contentView.leadingAnchor, bottom: contentView.bottomAnchor, trailing: contentView.trailingAnchor)

但是,如果我在没有 UIView 扩展的情况下执行此操作,则一切正常:

SettingsStackView.translatesAutoresizingMaskIntoConstraints = false
SettingsStackView.topAnchor.constraint(equalTo: contentView.safeAreaLayoutGuide.topAnchor).isActive = true
SettingsStackView.leadingAnchor.constraint(equalTo: contentView.leadingAnchor).isActive = true
SettingsStackView.bottomAnchor.constraint(equalTo: contentView.bottomAnchor).isActive = true
SettingsStackView.trailingAnchor.constraint(equalTo: contentView.trailingAnchor).isActive = true

参见 my other example 我用更多代码描述问题的地方。

谁能告诉我为什么上面的 UIView 扩展是 UIControls 在代码中用于约束布局约束时不再对触摸事件做出反应的根本原因?

除非您在显示的代码中有拼写错误...

您使用 .anchor 函数的代码行是:

SettingsStackView.anchor(top: safeAreaLayoutGuide.topAnchor, leading: contentView.leadingAnchor, bottom: contentView.bottomAnchor, trailing: contentView.trailingAnchor)

但是没有该函数的代码是:

SettingsStackView.topAnchor.constraint(equalTo: contentView.safeAreaLayoutGuide.topAnchor).isActive = true

注意第一行使用的是:

safeAreaLayoutGuide.topAnchor

第二行使用:

contentView.safeAreaLayoutGuide.topAnchor