我创建了一个UIView子类,并在其中添加了一些imageViews和标签。我使用Visual Format创建constraints。对于不同大小的类,视图布局也不同。我使用下面的代码来标识大小类。

let rule = UITraitCollection(horizontalSizeClass: .regular)
let isHorizontalRegular = self.traitCollection.containsTraits(in: rule)
if isHorizontalRegular {
// constraints for  Horizontal Regular size class
} else {
// constraints for  other size class
}

由于我没有在init方法中获取大小类信息,因此我创建了所有自动布局constraints都是在updateConstraints()方法中创建的。

基于此实现,我有一些问题。
  • 用覆盖的方法constraint创建所有updateConstraints()是正确的方法吗?
  • 如果不是,这是基于size类创建自动布局constraints的西方方法。
  • 如何在方向更改时更新自动布局constraints。(因为我使用视觉格式创建自动布局constraints)?

  • 我知道我可以使用UIViewController生命周期方法解决问题。但是我正在使用@IBDesignable在故事板上渲染视图。因此有可能在UIView子类中实现这些目标。

    请帮助我解决这些问题。

    最佳答案

    小心,一些链接继续出现Obj-C。

    第一步是学习整个视图生命周期this should help。下一步是了解adaptive layout是什么。

    关注这些替代:

    viewDidLoad():
    在这里可以设置约束。确保已加载子视图,并且所有内容的translationsAutoresizingMaskIntoConstraints均为false。

    viewWillLayoutSubviews():
    我倾向于在这里改变方向,但这只是因为我需要知道整体边界是纵向(高度>宽度)还是横向(宽度>高度)。您听起来好像有个更好的地方。

    viewWillTransition(大小:与协调人:):
    这是更改任何大小级别的地方-或不发生。 iPad也可以滑出或拆分视图。请注意,基本的iPad方向更改并不意味着表示尺寸类别更改,因为它们是常规/常规的。不过,此功能将被ping通。

    这是我通常做什么的模板:

    var p = [NSLayoutConstraint]()
    var l = [NSLayoutConstraint]()
    var initialOrientation = true
    var isInPortrait = false
    
    override func viewDidLoad() {
        super.viewDidLoad()
        setUpConstraints()
    }
    
    override func viewWillLayoutSubviews() {
        super.viewDidLayoutSubviews()
        if initialOrientation {
            initialOrientation = false
            if view.frame.width > view.frame.height {
                isInPortrait = false
            } else {
                isInPortrait = true
            }
            orientationChanged()
        } else {
            if view.orientationHasChanged(&isInPortrait) {
                orientationChanged()
            }
        }
    }
    func setUpConstraints() {
        view.turnOffAutoResizing()
        // append your portrait (p) and landscape (l) constraint arrays here
    }
    
    extension UIView {
    
        public func turnOffAutoResizing() {
            self.translatesAutoresizingMaskIntoConstraints = false
            for view in self.subviews as [UIView] {
                view.translatesAutoresizingMaskIntoConstraints = false
            }
        }
        public func orientationHasChanged(_ isInPortrait:inout Bool) -> Bool {
            if self.frame.width > self.frame.height {
                if isInPortrait {
                    isInPortrait = false
                    return true
                }
            } else {
                if !isInPortrait {
                    isInPortrait = true
                    return true
                }
            }
            return false
        }
        public func setOrientation(_ p:[NSLayoutConstraint], _ l:[NSLayoutConstraint]) {
            NSLayoutConstraint.deactivate(l)
            NSLayoutConstraint.deactivate(p)
            if self.bounds.width > self.bounds.height {
                NSLayoutConstraint.activate(l)
            } else {
                NSLayoutConstraint.activate(p)
            }
        }
    }
    

    这可能有点满足您的需求,但是您应该能够掌握要点。

    关于ios - 如何在方向更改时更新UIView的子类中的自动布局约束,我们在Stack Overflow上找到一个类似的问题:https://stackoverflow.com/questions/40825824/

    10-12 03:45