今天我在做自适应高度的UITextView时,无意中碰到了一个非常奇怪的问题:
我使用纯代码布局+snapKit建立约束的时候,在代理方法里面使用代码:
func textViewDidChange(_ textView: UITextView) {
let frame = textView.frame
let constraintSize = CGSize(width: frame.width, height: CGFloat(MAXFLOAT))
let size = textView.sizeThatFits(constraintSize)
if size.height <= maxRowHeight {
textView.snap.updateConstraint({
$0.height.equalToSuperView().offset(size.height).priority(999)
})
NotificationCenter.default.post(name: IBTextView.RowChangeNotification, object: nil)
}
}
这时没有任何问题。文本框可以自动换行!!!
However
当我使用snapKit+storyboard开发时想要使用同样的代码更新从storyboard中建立的约束时,就没有效果了。后面通过检查,发现snapKit框架自身有把系统约束给封装一层:
public class LayoutConstraint : NSLayoutConstraint {
这个就导致了snapKit在执行自己的updateConstraint方法时由于不认识NSLayoutConstraint,直接把原生的约束给忽略了......
最后
给出我想到的解决办法:
func textViewDidChange(_ textView: UITextView) {
let frame = textView.frame
let constraintSize = CGSize(width: frame.width, height: CGFloat(MAXFLOAT))
let size = textView.sizeThatFits(constraintSize)
if size.height <= maxRowHeight {
let constraints = textView.constraints.filter({ $0.firstAttribute == NSLayoutConstraint.Attribute.height })
if constraints.count > 0 {
textView.removeConstraints(constraints)
}
let newConstraints = NSLayoutConstraint(item: textView, attribute: .height, relatedBy: .equal, toItem: nil, attribute: .notAnAttribute, multiplier: 1.0, constant: size.height)
newConstraints.priority = UILayoutPriority(rawValue: 999)
textView.addConstraint(newConstraints)
NotificationCenter.default.post(name: IBTextView.RowChangeNotification, object: nil)
}
}
还有一个疑惑?
我这里使用filter是被逼无奈啊!!!本来我可以只用一个first查询就可以了的,但是我发现苹果输入法的键盘在点击提示中文信息的时候这个textViewDidChange方法被执行了两次。点键盘上面的字母是正常调用一次的?。难道是因为一个中文字符所占字节位数更长?所以才?烦请知道缘由的看客老爷在评论区给出答案,我们讨论一下。