ReactNative的Navigator組件使用方式

歡迎Follow我的GitHub, 關(guān)注我的簡(jiǎn)書.

React Native的編程思想類似于iOS, 導(dǎo)航欄也使用Navigator作為標(biāo)識(shí), 類似于AndroidActionBar. 導(dǎo)航欄作為最重要的應(yīng)用組件之一, 除了處理頁面導(dǎo)航功能以外, 還會(huì)提供頁面棧的管理, 管理頁面的跳入和跳出. 本文介紹 Navigator 組件的使用方式.

React Native

本文源碼的GitHub下載地址

關(guān)于React Native項(xiàng)目的啟動(dòng), 參考1, 參考2.


簡(jiǎn)單使用

Navigator

添加 Navigator 的組件<Navigator/>. 設(shè)置方法: 初始化路由(initialRoute), 配置場(chǎng)景動(dòng)畫(configureScene), 渲染場(chǎng)景(renderScene). 初始化路由(initialRoute), 使用FirstPage頁面作為首頁.

// 主模塊
class SimpleView extends Component {
  // ...
  render() {
    return (
      <Navigator
        style={{flex:1}}
        initialRoute={{component: FirstPage}}
        configureScene={this.configureScene}
        renderScene={this.renderScene}/>
    );
  }
}

配置場(chǎng)景動(dòng)畫(configureScene): 根據(jù)路由的type屬性, 判斷使用的動(dòng)畫樣式, 底部彈出或右側(cè)彈出.

  /**
   * 配置場(chǎng)景動(dòng)畫
   * @param route 路由
   * @param routeStack 路由棧
   * @returns {*} 動(dòng)畫
   */
  configureScene(route, routeStack) {
    if (route.type == 'Bottom') {
      return Navigator.SceneConfigs.FloatFromBottom; // 底部彈出
    }
    return Navigator.SceneConfigs.PushFromRight; // 右側(cè)彈出
  }

渲染場(chǎng)景(renderScene): 使用動(dòng)態(tài)加載組件的方式. 設(shè)置加載頁面的navigator參數(shù), 其余使用route.passProps屬性傳遞其他參數(shù).

  /**
   * 使用動(dòng)態(tài)頁面加載
   * @param route 路由
   * @param navigator 導(dǎo)航器
   * @returns {XML} 頁面
   */
  renderScene(route, navigator) {
    return <route.component navigator={navigator}  {...route.passProps} />;
  }

也可以使用靜態(tài)加載組件, 需要預(yù)定義組件, 沒有動(dòng)態(tài)加載靈活.

  /**
   * 渲染場(chǎng)景, 通過不同參數(shù), 設(shè)置不同頁面
   * @param route 路由, 場(chǎng)景信息
   * @param navigator 導(dǎo)航器
   * @returns {XML} 頁面
   */
  renderScene(route, navigator) {
    if (route.name == 'FirstPage') {
      return <FirstPage navigator={navigator} {...route.passProps}/>
    } else if (route.name == 'SecondPage') {
      return <SecondPage navigator={navigator} {...route.passProps}/>
    }
  }

第一頁

FirstPage組件: 包含導(dǎo)航欄標(biāo)題和兩個(gè)跳轉(zhuǎn)按鈕. 提供兩種跳轉(zhuǎn)動(dòng)畫, 右出和底部. 點(diǎn)擊按鈕調(diào)用_navigate()方法, 跳轉(zhuǎn)到第二頁.

// 第一頁. 使用Component可以自動(dòng)生成注釋, 符合標(biāo)準(zhǔn)
class FirstPage extends Component {

  // ...
  
  render() {
    return (
      <View style={styles.container}>
        <View style={styles.heading}>
          <Text style={styles.headText}>
            {'第一頁'}
          </Text>
        </View>
        <TouchableOpacity
          style={styles.button}
          onPress={()=>this._navigate('你好! (來源第一頁:右出)')}>
          <Text style={styles.buttonText}>
            {'跳轉(zhuǎn)至第二頁(右出)'}
          </Text>
        </TouchableOpacity>
        <TouchableOpacity
          style={styles.button}
          onPress={()=>this._navigate('你好! (來源第一頁:底出)', 'Bottom')}>
          <Text style={styles.buttonText}>
            {'跳轉(zhuǎn)至第二頁(底部)'}
          </Text>
        </TouchableOpacity>
      </View>
    );
  }
}

也可以使用var FirstPage = React.createClass()創(chuàng)建組件, 但沒有使用繼承Component方式規(guī)范, 不能自動(dòng)生成注釋.

第一頁

_navigate()方法: 導(dǎo)航跳轉(zhuǎn), 調(diào)用navigator.push()方法. 傳遞參數(shù)passPropsname屬性, type動(dòng)畫類型, component跳轉(zhuǎn)組件.

  /**
   * 給Navigator傳遞參數(shù).
   * @param name 參數(shù)
   * @private
   */
  _navigate(name, type = 'Normal') {
    this.props.navigator.push({
      component: SecondPage,
      passProps: {
        name: name
      },
      type: type
    })
  }

下劃線表示私有方法, 類似Java的private限定符.

第二頁

SecondPage組件: 第二頁, 跳出返回第一頁. 調(diào)用navigator.pop()方法, 使用當(dāng)前頁面出棧, 顯示上一個(gè)棧內(nèi)頁面.

// 第二頁, 點(diǎn)擊跳出返回第一頁
class SecondPage extends Component {
  render() {
    return (
      <View style={styles.container}>
        <View style={styles.heading}>
          <Text style={styles.headText}>
            第二頁: {this.props.name}
          </Text>
        </View>
        <TouchableOpacity
          style={styles.button}
          onPress={()=>this.props.navigator.pop()}>
          <Text style={styles.buttonText}>
            返回上一頁
          </Text>
        </TouchableOpacity>
      </View>
    );
  }
}
第二頁

Navigator的主要功能, 是管理頁面棧, 控制頁面的跳入跳出.


統(tǒng)一導(dǎo)航欄

對(duì)于應(yīng)用而言, 需要統(tǒng)一的導(dǎo)航欄, Navigator 組件也提供導(dǎo)航欄的定制.

Navigator

與上文類似, 額外添加navigationBar的屬性, 自定義設(shè)置導(dǎo)航欄, 保持所有頁面的導(dǎo)航欄一致. 屬性添加<NavigationBar/>標(biāo)簽, 通過routeMapper控制導(dǎo)航欄的功能和樣式.

// 主模塊
class UniformView extends Component {
  //...

  render() {
    return (
      <Navigator
        style={{flex:1}}
        initialRoute={{name: 'FirstPage', component: FirstPage}}
        configureScene={this.configureScene}
        renderScene={this.renderScene}
        navigationBar={
          <Navigator.NavigationBar
            style={styles.navContainer}
            routeMapper={NavigationBarRouteMapper}/>}
        />
    );
  }
}

NavigationBarRouteMapper

NavigationBarRouteMapper: 導(dǎo)航欄路由映射器, 設(shè)置左鍵LeftButton, 右鍵RightButton, 標(biāo)題Title.

// 導(dǎo)航欄的Mapper
var NavigationBarRouteMapper = {
  // 左鍵
  LeftButton(route, navigator, index, navState) {
    // ...
  },
  // 右鍵
  RightButton(route, navigator, index, navState) {
    // ...
  },
  // 標(biāo)題
  Title(route, navigator, index, navState) {
    return (
      <View style={styles.navContainer}>
        <Text style={styles.title}>
          應(yīng)用標(biāo)題
        </Text>
      </View>
    );
  }
};

左鍵LeftButton: index屬性表示當(dāng)前頁面的索引, 通過判斷index屬性, 獲知棧內(nèi)是否有其他頁面, 判斷后退按鈕是否顯示. 點(diǎn)擊調(diào)用navigator.pop()出棧.

  // 左鍵
  LeftButton(route, navigator, index, navState) {
    if (index > 0) {
      return (
        <View style={styles.navContainer}>
          <TouchableOpacity
            underlayColor='transparent'
            onPress={() => {if (index > 0) {navigator.pop()}}}>
            <Text style={styles.leftNavButtonText}>
              后退
            </Text>
          </TouchableOpacity>
        </View>
      );
    } else {
      return null;
    }
  },

右鍵RightButton: 點(diǎn)擊調(diào)用路由(route)onPress()方法, 提示信息. 根據(jù)路由的rightText屬性添加顯示文字.

  // 右鍵
  RightButton(route, navigator, index, navState) {
    if (route.onPress)
      return (
        <View style={styles.navContainer}>
          <TouchableOpacity
            onPress={() => route.onPress()}>
            <Text style={styles.rightNavButtonText}>
              {route.rightText || '右鍵'}
            </Text>
          </TouchableOpacity>
        </View>
      );
  },
導(dǎo)航欄

第一頁/第二頁

第一頁與第二頁與上文類似, 當(dāng)?shù)谝豁撎D(zhuǎn)時(shí), 傳遞的路由信息有些變化, 控制第二頁與導(dǎo)航欄的顯示信息.

  // 填出提示框
  onPress() {
    alert("我是Spike!");
  }

  /**
   * 跳轉(zhuǎn)頁面至SecondPage
   * @param name 傳遞參數(shù)
   * @param type 動(dòng)畫類型
   */
  gotoNext(name, type = 'Normal') {
    this.props.navigator.push({
      component: SecondPage,
      passProps: {
        id: name
      },
      onPress: this.onPress,
      rightText: 'ALERT!',
      type: type
    })
  }

React Native 路由的基本功能就是這些, 控制頁面的切換, 控制導(dǎo)航欄的功能. 導(dǎo)航欄作為應(yīng)用最重要的組件之一, 一定要熟練掌握.

OK, that's all ! Enjoy it!

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末肥败,一起剝皮案震驚了整個(gè)濱河市暗挑,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌军拟,老刑警劉巖补鼻,帶你破解...
    沈念sama閱讀 216,919評(píng)論 6 502
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件哄啄,死亡現(xiàn)場(chǎng)離奇詭異雅任,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)咨跌,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,567評(píng)論 3 392
  • 文/潘曉璐 我一進(jìn)店門沪么,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人锌半,你說我怎么就攤上這事禽车。” “怎么了刊殉?”我有些...
    開封第一講書人閱讀 163,316評(píng)論 0 353
  • 文/不壞的土叔 我叫張陵殉摔,是天一觀的道長(zhǎng)。 經(jīng)常有香客問我记焊,道長(zhǎng)逸月,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,294評(píng)論 1 292
  • 正文 為了忘掉前任遍膜,我火速辦了婚禮碗硬,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘捌归。我一直安慰自己肛响,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,318評(píng)論 6 390
  • 文/花漫 我一把揭開白布惜索。 她就那樣靜靜地躺著特笋,像睡著了一般。 火紅的嫁衣襯著肌膚如雪巾兆。 梳的紋絲不亂的頭發(fā)上猎物,一...
    開封第一講書人閱讀 51,245評(píng)論 1 299
  • 那天,我揣著相機(jī)與錄音角塑,去河邊找鬼蔫磨。 笑死,一個(gè)胖子當(dāng)著我的面吹牛圃伶,可吹牛的內(nèi)容都是我干的堤如。 我是一名探鬼主播,決...
    沈念sama閱讀 40,120評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼窒朋,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼搀罢!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起侥猩,我...
    開封第一講書人閱讀 38,964評(píng)論 0 275
  • 序言:老撾萬榮一對(duì)情侶失蹤榔至,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后欺劳,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體唧取,經(jīng)...
    沈念sama閱讀 45,376評(píng)論 1 313
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡铅鲤,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,592評(píng)論 2 333
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了枫弟。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片邢享。...
    茶點(diǎn)故事閱讀 39,764評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖媒区,靈堂內(nèi)的尸體忽然破棺而出驼仪,到底是詐尸還是另有隱情,我是刑警寧澤袜漩,帶...
    沈念sama閱讀 35,460評(píng)論 5 344
  • 正文 年R本政府宣布,位于F島的核電站湾碎,受9級(jí)特大地震影響宙攻,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜介褥,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,070評(píng)論 3 327
  • 文/蒙蒙 一座掘、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧柔滔,春花似錦溢陪、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,697評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至超全,卻和暖如春咆霜,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背嘶朱。 一陣腳步聲響...
    開封第一講書人閱讀 32,846評(píng)論 1 269
  • 我被黑心中介騙來泰國(guó)打工蛾坯, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人疏遏。 一個(gè)月前我還...
    沈念sama閱讀 47,819評(píng)論 2 370
  • 正文 我出身青樓脉课,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親财异。 傳聞我的和親對(duì)象是個(gè)殘疾皇子倘零,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,665評(píng)論 2 354

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