我创建了一个 UIView 子类,并在其中添加了一些 imageView 和标签。我使用 Visual Format 来创建 constraints 。并且 View 布局对于不同的大小类是不同的。我使用下面的代码来识别尺寸等级。
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() 中创建的方法。
基于这个实现,我有一些问题。
- 在重写方法
updateConstraints() 中创建所有 constraint 是否正确?
- 如果不是,那是西方式根据size class创建自动布局
constraints 。
- 如何在方向更改时更新自动布局
constraints 。(因为我使用视觉格式来创建自动布局constraints )?
我知道我可以使用 UIViewController 生命周期方法解决我的问题。但我使用 @IBDesignable 在 Storyboard 上呈现 View 。那么是否有可能在 UIView 子类中实现这些。
请帮我解决这些问题。
Best Answer-推荐答案 strong>
小心,一些链接继续 Obj-C 即将出现。
第一步是学习整个 View 生命周期,this should help .下一步是了解 adaptive layout 是什么是。
关注这些覆盖:
viewDidLoad():
这是您可以设置约束的地方。确保 subview 已加载并且 translatesAutoresizingMaskIntoConstraints 对所有内容都是错误的。
viewWillLayoutSubviews():
我倾向于在这里更改方向,但这只是因为我需要知道整体边界是纵向(高度>宽度)还是横向(宽度>高度)。听起来你有一个更好的地方......
viewWillTransition(to size: with coordinator:
这是 任何 尺寸类更改将发生或不发生的地方。 iPad 也可以滑出或 Split View。请注意,基本的 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/
|