本文是基於 iOS - RxSwift 項目實戰記錄 所述,若是你還未閱讀過,建議你最好還先閱讀一遍,並下載Demo熟悉一下 : )git
MVVM的模式中,多出了ViewModel這個角色,將邏輯處理、網絡請求等繁雜操做中ViewController中抽離出來,ViewController得以瘦身。 結合RxSwift架構,咱們通常就會在ViewModel中定義一個input收集繁雜操做所需的信息,經過一個transform方法將input做爲參數傳入,進而獲得一個output供controller使用。github
在使用RxSwift開發時會大量的使用到這種形式,其中就包括咱們的網絡請求。 結合 iOS - RxSwift 項目實戰記錄 中所述的「MJRefresh在RxSwift中的使用」,在output中定義了一個變量編程
let refreshStatus = Variable<LXFRefreshStatus>(.none)
複製代碼
controller經過output將其進行監聽,從而當值發生變化時,controller就能實時獲取當前應所處的刷新狀態swift
vmOutput.refreshStatus.asObservable().subscribe(onNext: {[weak self] status in
switch status {
case .beingHeaderRefresh:
self?.tableView.mj_header.beginRefreshing()
case .endHeaderRefresh:
self?.tableView.mj_header.endRefreshing()
case .beingFooterRefresh:
self?.tableView.mj_footer.beginRefreshing()
case .endFooterRefresh:
self?.tableView.mj_footer.endRefreshing()
case .noMoreData:
self?.tableView.mj_footer.endRefreshingWithNoMoreData()
default:
break
}
}).addDisposableTo(rx_disposeBag)
複製代碼
若是在一個項目多處使用到了這種方式,咱們就能夠看到弊端——重複代碼,過於冗餘。api
難道咱們每次都要在controller中進行如此操做嗎?ruby
關於協議的內容能夠看下我以前的這兩篇文章 iOS - Swift 面向協議編程(一) iOS - Swift 面向協議編程(二)bash
總結協議的兩大做用:一、規範 二、定製能力微信
定義協議 Refreshable網絡
/* ============================ Refreshable ================================ */
// 須要使用 MJExtension 的控制器使用
protocol Refreshable {
}
extension Refreshable where Self : UIViewController {
func initRefreshHeader(_ scrollView: UIScrollView, _ action: @escaping () -> Void) -> MJRefreshHeader {
scrollView.mj_header = MJRefreshNormalHeader(refreshingBlock: { action() })
return scrollView.mj_header
}
func initRefreshFooter(_ scrollView: UIScrollView, _ action: @escaping () -> Void) -> MJRefreshFooter {
scrollView.mj_footer = MJRefreshAutoNormalFooter(refreshingBlock: { action() })
return scrollView.mj_footer
}
}
複製代碼
在controller中遵循 Refreshable 協議,經過initRefreshHeader方法或者initRefreshFooter方法給tableView或者collectionView賦予頭部或尾部刷新的能力,而且書寫下拉刷新時須要執行的代碼架構
// 如下拉刷新爲例
let refreshHeader = initRefreshHeader(liveCollectionView) { [weak self] in
// 下拉後須要執行的操做
self?.vmOutput?.requestCommand.onNext(())
}
複製代碼
接下來再講講output,只要有網絡請求的地方,就會須要須要監聽請求狀態,既然這樣,那麼能夠爲output定義一個協議OutputRefreshProtocol,專門用來規範必需聲明的屬性
/* ============================ OutputRefreshProtocol ================================ */
// viewModel 中 output使用
protocol OutputRefreshProtocol {
// 告訴外界的tableView當前的刷新狀態
var refreshStatus : Variable<LXFRefreshStatus> {get}
}
複製代碼
接着讓output去遵循該協議,並進行初始化刷新狀態的值爲.none
struct LXFLiveOutput: OutputRefreshProtocol {
var refreshStatus: Variable<LXFRefreshStatus>
let sections: Driver<[LXFLiveSection]>
init(sections: Driver<[LXFLiveSection]>) {
self.sections = sections
refreshStatus = Variable<LXFRefreshStatus>(.none)
}
}
複製代碼
到此爲止,其實跟以前沒啥兩樣,只是使controller更方便初始化刷新控件而已。接下來纔是本文的重點。
刷新的狀態無非也就那麼幾種,下拉重載數據,上拉加載更多,請求完成時結束下拉或上拉等等。。。那咱們何須要在每一個controller中再去管理這等雜事?? 而至此,刷新控件的狀態是由變量 refreshStatus 來決定,此時 refreshStatus 又聲明在 OutputRefreshProtocol 協議中,咱們何再也不定義一個方法,將刷新控件的狀態交給refreshStatus本身來幫咱們處理呢~
extension OutputRefreshProtocol {
func autoSetRefreshHeaderStatus(header: MJRefreshHeader?, footer: MJRefreshFooter?) -> Disposable {
return refreshStatus.asObservable().subscribe(onNext: { (status) in
switch status {
case .beingHeaderRefresh:
header?.beginRefreshing()
case .endHeaderRefresh:
header?.endRefreshing()
case .beingFooterRefresh:
footer?.beginRefreshing()
case .endFooterRefresh:
footer?.endRefreshing()
case .noMoreData:
footer?.endRefreshingWithNoMoreData()
default:
break
}
})
}
}
複製代碼
這時須要咱們將刷新控件的對象 header / footer 傳入到方法中,實現自動控制刷新控件狀態。
1、output中遵照協議 OutputRefreshProtocol, 並初始化 refreshStatus 的值爲 none
struct LXFLiveOutput: OutputRefreshProtocol {
var refreshStatus: Variable<LXFRefreshStatus>
let sections: Driver<[LXFLiveSection]>
init(sections: Driver<[LXFLiveSection]>) {
self.sections = sections
refreshStatus = Variable<LXFRefreshStatus>(.none)
}
}
複製代碼
2、controller 遵照協議 Refreshable,經過協議中的方法初始化刷新控件及對應的操做,並將刷新控件對象做爲參數傳入到自動處理狀態方法中
extension LXFLiveViewController: Refreshable 複製代碼
let refreshHeader = initRefreshHeader(liveCollectionView) { [weak self] in
self?.vmOutput?.requestCommand.onNext(())
}
vmOutput?.autoSetRefreshHeaderStatus(header: refreshHeader, footer: nil).disposed(by: rx.disposeBag)
複製代碼
3、viewModel中根據實際狀況實時更新 refreshStatus 的刷新狀態
推薦使用 LXFProtocolTool/Refreshable
,會不斷更新,功能更增強大
pod 'LXFProtocolTool/Refreshable'
複製代碼
ViewModel:LXFLiveViewModel
Controller:LXFLiveViewController