最近比較清閒,就把以前學習的過程記錄下吧,多少年後如果能在網際網路上找到自己的痕跡,想想還是一件蠻值得高興的事情
容器在開發過程中用到的地方還是蠻多的,像網易新聞的框架如果使用程式碼去實現的話,呼叫系統的API
addChildViewController(<#T##childController: UIViewController##UIViewController#>)
複製程式碼
使它成為子控制器,加入一個類似的容器裡面就可以,具體的大家在做專案過程中都用到過,這裡就不多說了。今天就分享一下SB上使用容器類管理VC的控制元件ContainerView(純程式碼裡沒有這個控制元件)
新建個專案(這裡我使用的是以前自己隨便寫的一個小demo),在Manstoryboard 的 view 上拖拽兩個ContainerView ,給他們兩個分別設定約束,寬度加起來等於螢幕寬度就可以。拖拽完後你會發現view上多出來兩根線並且關聯到兩個VC。其實這就是containerView 自帶的容器VC,它讓我們可以把程式碼和業務都分發到一個獨立的VC裡面去操作,耦合性更低。
為了後面的操作我們需要在swift檔案裡面給這兩個VC繫結兩個Class,取名為Class1 和Class2 都繼承於BaseVC(BaseVC裡面只是簡單的對導航欄裡面的操作) 。然後要將ContainerView關聯的VC跟我們swift檔案裡面的類關聯起來,並且繫結它們的storyboardId
到現在的話,基本上容器的使用算是完成了,我們想做的業務就可以分別在兩個vc裡面實現了。但是這樣的話就太不直觀了,接著說個例子吧。
現在好多App 都有雙列表,比方說京東的
這個雙列表頁面,我們就能使用ContainerView 來實現類似的效果。
首先我們在Class1 和Class2 裡面各自新增兩個tableView 並給他們附上資料,最終的效果是這樣的
這個時候如果我們想做資料的互動,就需要傳值。OC裡面的傳值方式有很多,通知,代理,block 等等,其實最簡單的就是使用NSNotificationCenter ,通知比較簡單,我就不說了,swift裡面的使用跟OC差不多。這裡說一下代理的用法和swift裡面閉包的用法
代理在這裡的用法
先說下代理,因為我們使用的是容器ContainerView 它會自帶兩個VC,又加上原來的一個VC,相當於三個VC,怎麼ContainerView裡面取到對應關聯的兩個VC,這是一個關鍵。
在承載連個ContainerView 的VC裡面重寫下面的方法
override func prepare(for segue: UIStoryboardSegue, sender: Any?) {
}
複製程式碼
只需要重寫系統的這個方法,這個方法是SB裡面的檢視切換的時候系統會呼叫的一個方法,它的引數 segue.destination
let vc = segue.destination
複製程式碼
返回的就是對應的ContainerView 自帶的VC
因為我們使用了兩個VC 所以要判斷下是具體哪個VC
var class1:Class1? = nil
var class2:Class2? = nil
override func prepare(for segue: UIStoryboardSegue, sender: Any?) {
if let vc = segue.destination as? Class1 {
class1 = vc
}
if let vc = segue.destination as? Class2 {
class2 = vc
}
}
複製程式碼
取到這兩個VC後,就好辦多了。然後要做的無非就是宣告協議,遵循代理,實現代理方法
protocol XWContainerViewResponseDelegate {
func clickWithGetIndex(index:NSIndexPath)
}
複製程式碼
宣告一個協議,引數為NSIndexPath 型別,然後讓Class1 宣告一個代理屬性
var delegate:XWContainerViewResponseDelegate?
複製程式碼
然後在Class1 tabviewVIew 的點選方法裡面呼叫代理(不需要去呼叫responds(to: <#T##Selector!#>) delegate? 就相當於OC裡面的responds)
override func tableView(_ tableView: UITableView, didSelectRowAt indexPath: IndexPath) {
delegate?.clickWithGetIndex(index: (indexPath as? NSIndexPath)!)
}
複製程式碼
然後讓Class2 去遵循協議,實現代理方法,這樣就OK了。但是不知道大家有沒有發現,OC使用的時候我們一般都會設定
XXXX.delegate = XXXX
複製程式碼
這一步我們還沒有寫,但是這Class1 和 Class2 又是獨立的,沒有關聯。所以這個時候我們就要用到承載這兩個容器所在的VC了,在上面的那個方法裡面,我們獲取到了這兩個VC,在這裡進行關聯
var class1:Class1? = nil
var class2:Class2? = nil
override func prepare(for segue: UIStoryboardSegue, sender: Any?) {
if let vc = segue.destination as? Class1 {
class1 = vc
}
if let vc = segue.destination as? Class2 {
class2 = vc
}
class1?.delegate = class2
}
複製程式碼
OK,這樣就完整了,點選Class1 的cell 就會把對應的IndexPath 傳過去,就可以做操作了
閉包在這裡的用法
在說下使用閉包傳值的做法 其實核心的地方都在承載兩個ContainerView 所在的VC裡面。因為我們所做的操作是想點選Class1 裡面的cell,把對方的Indexpath 給傳過去,那麼肯定是想把IndexPath 給回撥出去。回撥出去之後Class2 怎麼去接受呢?其實Class2 也可以有一個閉包專門去接受這個資料,但是這樣就太麻煩了。我們可以在Class2 裡面宣告一個屬性,去實現它的屬性觀察器方法,這樣就能隨時監聽這個屬性的變化,更方便的去做操作
首先在class1 裡面去宣告閉包,順便給它一個屬性去記錄所要傳遞的IndexPath
typealias CallBack = (_ index:NSIndexPath)->Void
var xwCallBack:CallBack?
var class1Index: NSIndexPath? = nil
複製程式碼
Class2 裡面的操作 宣告屬性觀察器
var class2Index:NSIndexPath? = nil {
willSet {
}
didSet {
print("得到的從1傳過來的值 \(String(describing: class2Index?.row))")
}
}
複製程式碼
然後在Class1 cell的點選方法裡面去呼叫它
override func tableView(_ tableView: UITableView, didSelectRowAt indexPath: IndexPath) {
xwCallBack!(indexPath as NSIndexPath)
}
複製程式碼
ContainerView 承載所在的VC 讓Class1 實現閉包
var class1:Class1? = nil
var class2:Class2? = nil
override func prepare(for segue: UIStoryboardSegue, sender: Any?) {
if let vc = segue.destination as? Class1 {
class1 = vc
}
if let vc = segue.destination as? Class2 {
class2 = vc
}
unowned let weakSelf = self
class1?.xwCallBack = { index in
weakSelf.class2?.class2Index = index
}
}
複製程式碼
然後把Class1傳過來的IndexPath 賦值給Class2的 帶有屬性觀察器的屬性。OK,這樣就可以了
其實swift 裡面的東西 大致跟OC用法還是一樣的,只不過增加了一些確實特別好用的東西。無論什麼方法,用到好就行。黑貓白貓,逮到老鼠就是好貓
一起學習吧。。。