【发布时间】:2017-08-02 13:28:55
【问题描述】:
我想创建一些可重用的自定义 UI 组件以供我使用,例如以下包含两个标签的 UIView。根据内容的不同,标签可能是多行的,我为上下边距提供了一些限制。这些视图主要使用 Interface Builder 或以编程方式添加到我的布局中 UIStackViews 内。这里的问题是,视图的高度在运行时计算不正确,会在每个视图的底部截掉一部分,尤其是当有多行时。
显然存在一些我还没有弄清楚的概念性问题,可能正确地处理这个双标签示例将有助于我更好地理解。
我注释掉了我认为必要的整体高度限制,但没有注释我只看到第二个标签的顶行。
import UIKit
@IBDesignable class TwoLabelView: UIView {
var topMargin: CGFloat = 11.0
var verticalSpacing: CGFloat = 3.0
var bottomMargin: CGFloat = 8.0
@IBInspectable var firstLabelText: String = "" { didSet { updateView() } }
@IBInspectable var secondLabelText: String = "" { didSet { updateView() } }
var viewHeight: CGFloat = 0.0
var firstLabel: UILabel!
var secondLabel: UILabel!
override init(frame: CGRect) {
super.init(frame: frame)
setUpView()
}
required public init?(coder: NSCoder) {
super.init(coder:coder)
setUpView()
}
func setUpView() {
firstLabel = UILabel()
firstLabel.font = UIFont.systemFont(ofSize: 18.0, weight: UIFontWeightBold)
firstLabel.numberOfLines = 3
firstLabel.lineBreakMode = NSLineBreakMode.byWordWrapping
secondLabel = UILabel()
secondLabel.font = UIFont.systemFont(ofSize: 13.0, weight: UIFontWeightRegular)
secondLabel.numberOfLines = 20
secondLabel.lineBreakMode = NSLineBreakMode.byWordWrapping
addSubview(firstLabel)
addSubview(secondLabel)
updateView()
}
func updateView() {
firstLabel.text = firstLabelText
secondLabel.text = secondLabelText
firstLabel.sizeToFit()
secondLabel.sizeToFit()
viewHeight = getHeight()
setNeedsUpdateConstraints()
}
override func updateConstraints() {
translatesAutoresizingMaskIntoConstraints = false
firstLabel .translatesAutoresizingMaskIntoConstraints = false
secondLabel.translatesAutoresizingMaskIntoConstraints = false
removeConstraints(constraints)
if self.isHidden == false {
self.addConstraint(NSLayoutConstraint(item: firstLabel, attribute: .top, relatedBy: .equal, toItem: self, attribute: .top, multiplier: 1, constant: topMargin))
self.addConstraint(NSLayoutConstraint(item: firstLabel, attribute: .left, relatedBy: .equal, toItem: self, attribute: .left, multiplier: 1, constant: 0.0))
self.addConstraint(NSLayoutConstraint(item: firstLabel, attribute: .right, relatedBy: .equal, toItem: self, attribute: .right, multiplier: 1, constant: 0.0))
self.addConstraint(NSLayoutConstraint(item: secondLabel, attribute: .top, relatedBy: .equal, toItem: firstLabel, attribute: .bottom, multiplier: 1, constant: verticalSpacing))
self.addConstraint(NSLayoutConstraint(item: secondLabel, attribute: .left, relatedBy: .equal, toItem: self, attribute: .left, multiplier: 1, constant: 0.0))
self.addConstraint(NSLayoutConstraint(item: secondLabel, attribute: .right, relatedBy: .equal, toItem: self, attribute: .right, multiplier: 1, constant: 0.0))
self.addConstraint(NSLayoutConstraint(item: secondLabel, attribute: .bottom, relatedBy: .equal, toItem: self, attribute: .bottom, multiplier: 1, constant: bottomMargin))
//self.addConstraint(NSLayoutConstraint(item: self, attribute: .height, relatedBy: .equal, toItem: nil, attribute: .notAnAttribute, multiplier: 1, constant: viewHeight))
}
super.updateConstraints()
}
func getHeight() -> CGFloat {
return topMargin
+ firstLabel.frame.height
+ verticalSpacing
+ secondLabel.frame.height
+ bottomMargin
}
override open var intrinsicContentSize : CGSize {
return CGSize(width: UIViewNoIntrinsicMetric, height: getHeight())
}
}
【问题讨论】:
-
如果你把
setupViews和约束代码放在override func layoutSubviews()里面可以吗? -
似乎
updateView()在layoutSubviews()之前被调用,导致尝试将文本值分配给标签时出现“意外发现nil”错误。
标签: ios swift autolayout uilabel custom-controls