起因
最近看了@南峰子_老驢寫的ViewStyle。利用KVC的來簡化抽取出控制元件設定style的程式碼,想達到一個css的效果。這個庫用swift編寫,使用起來大概這樣:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 |
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的套路:
1 2 |
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這個屬性。
那為什麼我們平時沒有注意到這個問題呢?
1 |
view.isHidden = true |
這麼寫程式碼是沒問題的。
其實就是Swift團隊在升級3.0的時候keypath忘記做這個功能了。。。
你可以安慰自己這個功能在swift 2.0的時候還是好好的。
如果讀者寫過OC這個原因很容易猜到。有些單獨宣告過getter的bool值直接kvc會報錯。比如isHidden這樣:
1 |
@property (nonatomic, getter=isHidden) BOOL hidden; |
UIView上確實沒有isHidden屬性,那麼怎麼解決呢?只能回到老路上了:
1 |
view.setValue(true, forKey: "Hidden") |
Enum
KVC在設定列舉的時候也會報錯。
錯誤提示的無效的引數型別。
因為setValue的引數型別是Any,所以這裡引數可以是任意型別,沒有型別檢查。但是這畢竟是一個OC的方法,所以無法直接使用swift的型別(其實String到NSString也是有轉換的,只是編譯器幫我們做了這層轉換)。
前面直接賦值的時候是正常的,但是呼叫KVC方法的時候編譯器沒有幫我們轉。所以這裡報錯了。
那就只能自己手動轉換了:
1 |
view.setValue(UIViewTintAdjustmentMode.automatic.rawValue, forKey: "tintAdjustmentMode") |
在OC中的列舉其實就是對應幾個數值,這裡直接取出rawValue的值就可以正常使用了。
最後我有一句話想對swift說:
歡迎關注我的微博:@沒故事的卓同學