RxSwift + MJRefresh 打造自動處理刷新控件狀態

本文是基於 iOS - RxSwift 項目實戰記錄 所述,若是你還未閱讀過,建議你最好還先閱讀一遍,並下載Demo熟悉一下 : )git

LXFBiliBili

前言

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 的刷新狀態

image.png

開源庫

LXFProtocolTool

推薦使用 LXFProtocolTool/Refreshable ,會不斷更新,功能更增強大

pod 'LXFProtocolTool/Refreshable'
複製代碼

案例

協議:Refreshable.swift

ViewModel:LXFLiveViewModel

Controller:LXFLiveViewController

LXFBiliBili

微信公衆號
相關文章
相關標籤/搜索