React Native (五):上下拉刷新加載

React Native (一):基礎(chǔ)
React Native (二):StatusBar 、 NavigationBar 與 TabBar
React Native (三):自定義視圖
React Native (四):加載新聞列表
React Native (五):上下拉刷新加載
React Native (六):加載所有分類與詳情頁(yè)

1.下拉刷新

下拉刷新我們使用 React Native 提供的組件 RefreshControl空执,去 NewsList.jsListView 添加:

<ListView
    style={{flex:1, backgroundColor:'white'}}
    dataSource={this.state.dataSource} //設(shè)置數(shù)據(jù)源
    renderRow={this.renderRow} //設(shè)置cell
    removeClippedSubviews={false}
    refreshControl={
          <RefreshControl
            refreshing={this.state.isRefreshing}
            onRefresh={() => this._onRefresh(1)}
            tintColor="#999999"
            title="加載中..."
            titleColor="#999999"
            colors={['#ff0000', '#00ff00', '#0000ff']}
            progressBackgroundColor="#ffff00"
          />
    }
/>

ListView 新加了一個(gè)屬性 removeClippedSubviews贸营。

原因

我們需要給 state 添加一個(gè) keyisRefreshing: false 供汛,然后在網(wǎng)絡(luò)請(qǐng)求時(shí)對(duì)它進(jìn)行處理纸镊,(我這里省略了一些代碼被去,要不然太長(zhǎng))

_begainRefresh() {
    this.setState({
        isRefreshing: true
    })
}
_endRefresh() {
    this.setState({
        isRefreshing: false
    })
}

_onRefresh(page) {
    if (this.props.dic) {
        this._begainRefresh()

        if (page == 1) {
            this.setState({
                page
            })
        }
        let url = ''

        fetch(url, {
            method: 'GET',
            headers: {
                'Accept': 'application/json',
                'Content-Type': 'application/json',
            },
        })
            .then((res) => {

                this._endRefresh()

                res.json()
                    .then((json) => {
                    ..... 數(shù)據(jù)的處理
                    })
                    .catch((e) => {
                    })
            })
            .catch((error) => {
                this._endRefresh()
            })
    }
}

現(xiàn)在運(yùn)行程序就可以盡情地下拉刷新了梯皿。

2.上拉加載

上拉加載我們利用 ListViewonEndReached 方法來(lái)進(jìn)行加載新數(shù)據(jù)仇箱,使用 renderFooter 來(lái)進(jìn)行顯示狀態(tài)。

這樣嚴(yán)格的來(lái)說(shuō)并不算上拉加載东羹,只是滑動(dòng)到底部自動(dòng)進(jìn)行加載剂桥。

首先在 ListView 加入:

onEndReached={ this._toEnd }
onEndReachedThreshold={10}
renderFooter={ this._renderFooter }

記得進(jìn)行綁定

this._toEnd = this._toEnd.bind(this)
this._renderFooter = this._renderFooter.bind(this)

實(shí)現(xiàn)

_toEnd() {
    if (this.state.isRefreshing) return  
    this._onRefresh()
}

_renderFooter() {
    return (
        <View style={{width, height: 40, backgroundColor: '#FFFFFF', alignItems:'center', justifyContent:'center'}}>
            <Text>正在加載更多...</Text>
        </View>
    )
}

現(xiàn)在我們需要對(duì)數(shù)據(jù)進(jìn)行處理,保存請(qǐng)求到的數(shù)據(jù)百姓,再下一頁(yè)數(shù)據(jù)請(qǐng)求到后加入數(shù)組渊额,我們給 state 加入一個(gè) data: [],然后對(duì)請(qǐng)求到的數(shù)據(jù)進(jìn)行處理:

let list = json.NewsList

let swipers = []
let news = []

if (page == 1) {
    for (let index in list) {
        let dic = list[index]
        index < 4 ? swipers.push(dic) : news.push(dic)
    }
    news.splice(0, 0, swipers)
} else {
    news = list
}

let newData = this.state.data.concat(news)

this.setState({
    dataSource: this.state.dataSource.cloneWithRows(newData),
    data: newData,
    page: this.state.page + (list.length == maxCount ? 1 : 0)
})

這里的 maxCount 是為了方便管理的垒拢,定義為:

const maxCount = 20

請(qǐng)求的 url 改為 :

let url = 'http://api.iapple123.com/newspush/list/index.html?clientid=1114283782&v=1.1&type='
        + this.props.dic.NameEN
        + '&startkey=3001_9223370543834829200_537d522d125e32ae&newkey=&index='
        + page
        + '&size='
        + maxCount
        + '&ime=6271F554-7B2F-45DE-887E-4A336F64DEE6&apptypeid=ZJZYIOS1114283782'

現(xiàn)在可以運(yùn)行看看效果了旬迹。

我們會(huì)發(fā)現(xiàn)一開(kāi)始在加載第一頁(yè)數(shù)據(jù)的時(shí)候 Footer 也顯示了出來(lái),我們需要控制它的顯示與隱藏求类,給 state 加入 showFooter: false 奔垦,在第一頁(yè)數(shù)據(jù)加載完成并且返回的數(shù)組元素個(gè)數(shù)等于 maxCount 則賦值為 true

this.setState({
    dataSource: this.state.dataSource.cloneWithRows(newData),
    data: newData,
    page: this.state.page + (list.length == maxCount ? 1 : 0),
    showFooter: this.state.showFooter ? true :  (list.length == maxCount ? true : false)
})

_renderFooter() {

    if (!this.state.showFooter) {
        return null
    }
    return (
        <View style={{width, height: 40, backgroundColor: '#FFFFFF', alignItems:'center', justifyContent:'center'}}>
            <Text>正在加載更多</Text>
        </View>
    )
}

現(xiàn)在 Footer 可以正確的顯示隱藏了,但是我們還需要狀態(tài)來(lái)改變 Footer 顯示的文字尸疆,如果還有更多數(shù)據(jù)椿猎,那我們看見(jiàn) Footer 的時(shí)候它的狀態(tài)顯然是正在加載更多惶岭,如果沒(méi)有更多數(shù)據(jù)了,那我們就顯示 已加載全部 犯眠。

state 加入 hasMore: true 按灶,我們先假設(shè)它還有更多

然后在請(qǐng)求到數(shù)據(jù)進(jìn)行處理:

let hasMore = list.length == maxCount ? true : false

this.setState({
    dataSource: this.state.dataSource.cloneWithRows(newData),
    data: newData,
    page: this.state.page + (hasMore ? 1 : 0),
    showFooter: this.state.showFooter ? true :  (hasMore ? true : false),
    hasMore,
})

然后處理 renderFooter:

_renderFooter() {

    if (!this.state.showFooter) {
        return null
    }
    return (
        <View style={{width, height: 40, backgroundColor: '#FFFFFF', alignItems:'center', justifyContent:'center'}}>
            <Text>{this.state.hasMore ? '正在加載更多...' : '已加載全部'}</Text>
        </View>
    )
}

我們還需要再 toEnd 的判斷條件加入 hasMore 來(lái)避免顯示沒(méi)有更多數(shù)據(jù)的時(shí)候拉倒底部還會(huì)進(jìn)行請(qǐng)求數(shù)據(jù):

_toEnd() {
    if (this.state.isRefreshing || !this.state.hasMore) return
    this._onRefresh()
}

到現(xiàn)在上下拉刷新已經(jīng)完成

這個(gè)上拉刷新比較簡(jiǎn)陋,你也可以放 gif圖 或者使用 動(dòng)畫(huà) 來(lái)讓界面變好看點(diǎn)筐咧。

項(xiàng)目地址

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末鸯旁,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子量蕊,更是在濱河造成了極大的恐慌铺罢,老刑警劉巖,帶你破解...
    沈念sama閱讀 221,198評(píng)論 6 514
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件残炮,死亡現(xiàn)場(chǎng)離奇詭異韭赘,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī)势就,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 94,334評(píng)論 3 398
  • 文/潘曉璐 我一進(jìn)店門泉瞻,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人蛋勺,你說(shuō)我怎么就攤上這事瓦灶○荆” “怎么了抱完?”我有些...
    開(kāi)封第一講書(shū)人閱讀 167,643評(píng)論 0 360
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)刃泡。 經(jīng)常有香客問(wèn)我巧娱,道長(zhǎng),這世上最難降的妖魔是什么烘贴? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 59,495評(píng)論 1 296
  • 正文 為了忘掉前任禁添,我火速辦了婚禮,結(jié)果婚禮上桨踪,老公的妹妹穿的比我還像新娘老翘。我一直安慰自己,他們只是感情好锻离,可當(dāng)我...
    茶點(diǎn)故事閱讀 68,502評(píng)論 6 397
  • 文/花漫 我一把揭開(kāi)白布铺峭。 她就那樣靜靜地躺著,像睡著了一般汽纠。 火紅的嫁衣襯著肌膚如雪卫键。 梳的紋絲不亂的頭發(fā)上,一...
    開(kāi)封第一講書(shū)人閱讀 52,156評(píng)論 1 308
  • 那天虱朵,我揣著相機(jī)與錄音莉炉,去河邊找鬼钓账。 笑死,一個(gè)胖子當(dāng)著我的面吹牛絮宁,可吹牛的內(nèi)容都是我干的梆暮。 我是一名探鬼主播,決...
    沈念sama閱讀 40,743評(píng)論 3 421
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼绍昂,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼惕蹄!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起治专,我...
    開(kāi)封第一講書(shū)人閱讀 39,659評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤卖陵,失蹤者是張志新(化名)和其女友劉穎,沒(méi)想到半個(gè)月后张峰,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體泪蔫,經(jīng)...
    沈念sama閱讀 46,200評(píng)論 1 319
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 38,282評(píng)論 3 340
  • 正文 我和宋清朗相戀三年喘批,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了撩荣。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 40,424評(píng)論 1 352
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡饶深,死狀恐怖餐曹,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情敌厘,我是刑警寧澤台猴,帶...
    沈念sama閱讀 36,107評(píng)論 5 349
  • 正文 年R本政府宣布,位于F島的核電站俱两,受9級(jí)特大地震影響饱狂,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜宪彩,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,789評(píng)論 3 333
  • 文/蒙蒙 一休讳、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧尿孔,春花似錦俊柔、人聲如沸。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 32,264評(píng)論 0 23
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至芜辕,卻和暖如春尚骄,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背侵续。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 33,390評(píng)論 1 271
  • 我被黑心中介騙來(lái)泰國(guó)打工倔丈, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留憨闰,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 48,798評(píng)論 3 376
  • 正文 我出身青樓需五,卻偏偏與公主長(zhǎng)得像鹉动,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子宏邮,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,435評(píng)論 2 359

推薦閱讀更多精彩內(nèi)容