随笔:使用SnapKit的updateConstraint方法无法更新storyboard中建立的约束问题

今天我在做自适应高度的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,直接把原生的约束给忽略了......

这里我就是想diss它一下,看!你的bug😁

最后

给出我想到的解决办法:

    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方法被执行了两次。点键盘上面的字母是正常调用一次的😢。难道是因为一个中文字符所占字节位数更长?所以才?烦请知道缘由的看客老爷在评论区给出答案,我们讨论一下。

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章