【react-native】滑動(dòng)吸頂效果 + 下拉刷新

現(xiàn)象:最近開發(fā)RN項(xiàng)目蛛枚,需要開發(fā)一個(gè)頁面滑動(dòng)過程中的tab吸頂?shù)男Ч?/p>

解決方案: 使用Animated創(chuàng)建動(dòng)畫

具體代碼如下:

  1. index.tsx
import React, { Component } from 'react'

import { View, Text, ViewStyle, StyleSheet, ImageBackground, Animated, RefreshControl } from 'react-native'
import StickyHeader from "./StickyHeader"
import { images, dimensions } from '../../../../res'

interface IState {
  refreshing: boolean,
  scrollHeight: number,
  scrollY: Animated.Value
}

export class PrivateCollectScreen extends Component<any, IState> {

  readonly state: IState = {
    scrollY: new Animated.Value(0),
    scrollHeight: -1,
    refreshing: false
  }



  render() {
    return (
      <View style={styles.container}>
        <Animated.ScrollView
          style={{ flex: 1 }}
          onScroll={
            Animated.event( // 映射動(dòng)畫值
              [{
                // 垂直滾動(dòng)時(shí)扭吁,將 event.nativeEvent.contentOffset.y映射到this.state.scrollY,以此記錄滑動(dòng)距離
                nativeEvent: { contentOffset: { y: this.state.scrollY } }
              }],
              { useNativeDriver: true }) // 啟用原生動(dòng)畫驅(qū)動(dòng)盲镶。默認(rèn)不啟用(false)
          }
          scrollEventThrottle={1} // 滾動(dòng)條距離視圖邊緣的坐標(biāo)
          refreshControl={ // 下拉刷新功能
            <RefreshControl
              style={{ backgroundColor: 'transparent' }}
              tintColor={'white'}
              refreshing={this.state.refreshing}
              onRefresh={() => { // 再刷新時(shí)調(diào)用
                this.setState({ refreshing: true })
              }} />
          }
        >


          <View onLayout={(e) => {
            this.setState({ scrollHeight: e.nativeEvent.layout.height }) // 獲取頭部的高度
          }}>
            <ImageBackground
              style={{ width: dimensions.screenWidth, height: 190 }}
              source={images.icReact}
            >
              {/* 頭部內(nèi)容 */}
            </ImageBackground>
          </View>


          <StickyHeader
            stickyHeaderY={this.state.scrollHeight} // 將頭部高度傳入組件
            stickyScrollY={this.state.scrollY}  // 將滑動(dòng)距離傳入組件
          >
            <View style={{ height: 50, backgroundColor: '#3356d9' }}>
              <Text style={{ fontSize: 20, textAlign: 'center', color: '#fff', lineHeight: 50 }}>固定欄</Text>
            </View>
          </StickyHeader>


          {/* 底部內(nèi)容 */}
          {
            [0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11].map((item, index) => {
              return (<View style={{ height: 100 }}><Text>底部內(nèi)容{index}</Text></View>)
            })
          }
        </Animated.ScrollView>
      </View>
    )
  }
}

interface Styles {
  container: ViewStyle
}

const styles = StyleSheet.create<Styles>({
  container: {
    flex: 1,
    backgroundColor: '#fff'
  }
})
  1. StickyHeader.tsx
import * as React from 'react'
import { StyleSheet, Animated } from 'react-native'

interface IState {
  stickyLayoutY: number,
  stickyHeaderY: number,
  stickyScrollY: Animated.Value
}

/**
 * 滑動(dòng)吸頂效果組件
 * @export
 * @class StickyHeader
 */
export default class StickyHeader extends React.Component<any, IState> {


  readonly state: IState = {
    stickyHeaderY: -1,
    stickyScrollY: new Animated.Value(0),
    stickyLayoutY: 0
  }
  // 兼容代碼,防止沒有傳頭部高度
  _onLayout = (event) => {
    this.setState({
      stickyLayoutY: event.nativeEvent.layout.y,
    })
  }

  render() {
    const { stickyHeaderY, stickyScrollY, children, style } = this.props
    const { stickyLayoutY } = this.state
    let y = stickyHeaderY !== -1 ? stickyHeaderY : stickyLayoutY
    const translateY = stickyScrollY.interpolate({
      inputRange: [-1, 0, y, y + 1],
      outputRange: [0, 0, 0, 1],
    })

    return (
      <Animated.View
        onLayout={this._onLayout}
        style={
          [
            style,
            styles.container,
            { transform: [{ translateY }] }
          ]}
      >
        {children}
      </Animated.View>
    )
  }
}

const styles = StyleSheet.create({
  container: {
    zIndex: 100
  }
})
  1. 效果圖


    效果圖
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末溉贿,一起剝皮案震驚了整個(gè)濱河市枫吧,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌宇色,老刑警劉巖九杂,帶你破解...
    沈念sama閱讀 210,978評(píng)論 6 490
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異宣蠕,居然都是意外死亡例隆,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 89,954評(píng)論 2 384
  • 文/潘曉璐 我一進(jìn)店門抢蚀,熙熙樓的掌柜王于貴愁眉苦臉地迎上來镀层,“玉大人,你說我怎么就攤上這事皿曲÷瓜欤” “怎么了羡微?”我有些...
    開封第一講書人閱讀 156,623評(píng)論 0 345
  • 文/不壞的土叔 我叫張陵,是天一觀的道長惶我。 經(jīng)常有香客問我妈倔,道長,這世上最難降的妖魔是什么绸贡? 我笑而不...
    開封第一講書人閱讀 56,324評(píng)論 1 282
  • 正文 為了忘掉前任盯蝴,我火速辦了婚禮,結(jié)果婚禮上听怕,老公的妹妹穿的比我還像新娘捧挺。我一直安慰自己,他們只是感情好尿瞭,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,390評(píng)論 5 384
  • 文/花漫 我一把揭開白布闽烙。 她就那樣靜靜地躺著,像睡著了一般声搁。 火紅的嫁衣襯著肌膚如雪黑竞。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 49,741評(píng)論 1 289
  • 那天疏旨,我揣著相機(jī)與錄音很魂,去河邊找鬼。 笑死檐涝,一個(gè)胖子當(dāng)著我的面吹牛遏匆,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播谁榜,決...
    沈念sama閱讀 38,892評(píng)論 3 405
  • 文/蒼蘭香墨 我猛地睜開眼幅聘,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了窃植?” 一聲冷哼從身側(cè)響起帝蒿,我...
    開封第一講書人閱讀 37,655評(píng)論 0 266
  • 序言:老撾萬榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎撕瞧,沒想到半個(gè)月后陵叽,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體狞尔,經(jīng)...
    沈念sama閱讀 44,104評(píng)論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡丛版,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,451評(píng)論 2 325
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了偏序。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片页畦。...
    茶點(diǎn)故事閱讀 38,569評(píng)論 1 340
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖研儒,靈堂內(nèi)的尸體忽然破棺而出豫缨,到底是詐尸還是另有隱情独令,我是刑警寧澤,帶...
    沈念sama閱讀 34,254評(píng)論 4 328
  • 正文 年R本政府宣布好芭,位于F島的核電站燃箭,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏舍败。R本人自食惡果不足惜招狸,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,834評(píng)論 3 312
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望邻薯。 院中可真熱鬧裙戏,春花似錦、人聲如沸厕诡。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,725評(píng)論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽灵嫌。三九已至壹罚,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間醒第,已是汗流浹背渔嚷。 一陣腳步聲響...
    開封第一講書人閱讀 31,950評(píng)論 1 264
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留稠曼,地道東北人形病。 一個(gè)月前我還...
    沈念sama閱讀 46,260評(píng)論 2 360
  • 正文 我出身青樓,卻偏偏與公主長得像霞幅,于是被迫代替她去往敵國和親漠吻。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,446評(píng)論 2 348