起因
最近看了@南峰子_老驴写的ViewStyle。利用KVC的来简化抽取出控件设置style的代码,想达到一个css的效果。这个库用swift编写,使用起来大概这样:
class ViewController: UIViewController {
var tableView: UITableView?
override func viewDidLoad() {
super.viewDidLoad()
tableView = UITableView(frame: self.view.bounds)
tableView!.mi_styles = self.tableViewStyle
self.view.addSubview(tableView!)
}
}
// MARK: - Table View Style
extension ViewController {
var tableViewStyle: [Property: Any] {
return [
.rowHeight: 60.0,
.separatorStyle: UITableViewCellSeparatorStyle.singleLine,
.separatorColor: UIColor.lightGray,
.backgroundView: UIView(),
.separatorInset: UIEdgeInsets(top: 10.0, left: 5.0, bottom: 3.0, right: 10.0),
.cellLayoutMarginsFollowReadableWidth: true
]
}
}
阅读源码后发现有些属性在swift是不能直接KVC的,和OC有些区别。
Bool值:isHidden
先来看下正常的在swift中kvc的套路:
let view = UIView(frame: CGRect(x: 0, y: 0, width: 100, height: 100))
view.setValue(UIColor.blue, forKey: #keyPath(UIView.backgroundColor))
在swift 3中提供了#keypath()来省去开发者直接输入字符串的尴尬。
然而,如果这个属性是isHidden的话运行起来就会报错。
控制台会输出错误:
‘[ setValue:forUndefinedKey:]: this class is not key value coding-compliant for the key isHidden.’
提示我们没有isHidden这个属性。
那为什么我们平时没有注意到这个问题呢?
view.isHidden = true
这么写代码是没问题的。
其实就是Swift团队在升级3.0的时候keypath忘记做这个功能了。。。
你可以安慰自己这个功能在swift 2.0的时候还是好好的。
如果读者写过OC这个原因很容易猜到。有些单独声明过getter的bool值直接kvc会报错。比如isHidden这样:
@property (nonatomic, getter=isHidden) BOOL hidden;
UIView上确实没有isHidden属性,那么怎么解决呢?只能回到老路上了:
view.setValue(true, forKey: "Hidden")
Enum
KVC在设置枚举的时候也会报错。
错误提示的无效的参数类型。
因为setValue的参数类型是Any,所以这里参数可以是任意类型,没有类型检查。但是这毕竟是一个OC的方法,所以无法直接使用swift的类型(其实String到NSString也是有转换的,只是编译器帮我们做了这层转换)。
前面直接赋值的时候是正常的,但是调用KVC方法的时候编译器没有帮我们转。所以这里报错了。
那就只能自己手动转换了:
view.setValue(UIViewTintAdjustmentMode.automatic.rawValue, forKey: "tintAdjustmentMode")
在OC中的枚举其实就是对应几个数值,这里直接取出rawValue的值就可以正常使用了。
***我有一句话想对swift说: