首页 技术 正文
技术 2022年11月15日
0 收藏 648 点赞 4,756 浏览 6497 个字

一、react-navigation的初使用

createStackNavigator  ==> createSwitchNavigator  ==>  createAppContainer

createBottomTabNavigator ==> 自定义的底部Tab

NavigationActions 和 StackActions 可以获取navigation的方法名

二、this.props.navigation使用的方法和属性

详细请看: react-navigation 3.x版本的push、navigate、goback、pop、dispatch等常用方法

1. 应用中的每个页面组件都会自动提供 this.props.navigation

this.props.navigation可以获取的一些方法:

 navigate - 转到另一个页面, 计算出需要执行的操作  (常用)
goBack - 关闭活动屏幕并在堆栈中向后移动  (常用)
addListener - 订阅导航生命周期的更新
isFocused - 函数返回 true 如果屏幕焦点和 false 否则。
state - 当前状态/路由  (常用)
setParams - 对路由的参数进行更改  (常用)
getParam - 获取具有回退的特定参数  (常用)
dispatch - 向路由发送 action  (常用)
dangerouslyGetParent - 返回父级 navigator 的函数

navigation的方法

注意: this.props.navigation并不是在所有页面(组件)中都可以使用,而是必须在StackNavigator、DrawerNavigator中声明的screen组件,才可以使用this.props.navigation

也就是说,screen组件会自动获得这个props

this.props.navigation 上还有一些方法取决于当前 navigator 的附加函数(StackNavigator 和 DrawerNavigator)

2. 如果是StackNavigator,除了以上方法,this.props.navigation还提供如下的一些方法:

 push - 推一个新的路由到堆栈  (常用)
pop - 返回堆栈中的上一个页面  (常用)
popToTop - 跳转到堆栈中最顶层的页面  (常用)
replace - 用新路由替换当前路由
reset- 操作会擦除整个导航状态,并将其替换为多个操作的结果。  (常用)
dismiss - 关闭当前堆栈

StackNavigator的方法

3. 如果是DrawerNavigator,除了以上方法,this.props.navigation还提供如下的一些方法:

 openDrawer - 打开
closeDrawer - 关闭
toggleDrawer - 切换,如果是打开则关闭,反之亦然

DrawerNavigator的方法

由于笔者没有使用过DrawerNavigator,在此就不做说明;

以上内容均摘自react-navigation的官网,但是官网有点繁琐,写此以自用,有读者也是我的幸运。

三、react-navigation的特殊方法

===============================================================

==> 清空堆栈,跳转指定页面

import {NavigationActions, StackActions} from 'react-navigation'const resetAction = StackAction.reset({
  index: 0,
  actions: [NAvigationActions.navigate({ routeName: 'HomeScreen'})],});
//使用
this.props.navigation.dispatch(resetaction)

说明: 上面的方法是使用dispatch重写了reset方法,

也可以直接使用reset方法:this.props.navigation.reset()

import { NavigationActions } from 'react-navigation';
navigation.reset([NavigationActions.navigate({ routeName: 'HomeScreen' })], 0)

========================================================================

//在不在StackNavigation的堆栈中的页面,如何使用navigation

1.在主Navigator的组件中添加ref,并导出

//StackNavigator来自createStackNavigator方法
const MainNavigator = createAppContainer(StackNavigator);import {setTopLevelNavigator} from 'xxx.js'export default calss Navigator extends Component{
render() {
return(
<MainNavigator
   ref={navigatorRef => {setTopLevelNavigator(navigatorRef )}}    //setTopLevelNavigator来自需要使用的页面
       />
)
}
}

说明: 将MainNavigator 通过ref 传递给 其他页面传过来的 setTopLevelNavigator

2.在需要使用的页面,接收这个ref引用,如xxx.js

let  _navigator;export function setTopLevelNavigator(navigatorRef ) {
  _navigator = navigatorRef
}
// 定义navigate 方法
function navigate(routeName, params) {
  _navigator && _navigator.dispatch(
   NavigationActions.navigate({
    type: NavigationActions.NAVIGATE,
    routeName,
    params
  })
  )
}
// 定义dispatch方法
function dispatch(params) {
  _navigator && _navigator.dispatch(
     params
  )
}//使用,直接使用上面的方法即可
dispatch(resetAction)  //resetAction同上  navigate("Home", {title: 'profile'})

说明: 通过setTopLevelNavigator去主导航页面接收navigation相关的方法

另外: 使用withNavigation(Component) 可以使得组件携带navigation相关的方法

withNavigation是一个更高阶的组件,它将navigationprop 传递给一个包装组件。当你不能navigation直接将prop 传递给组件时,或者在深度嵌套的子级的情况下不想传递它时,它很有用。

withNavigation(Component) 返回一个Component。

import React from 'react';
import { Button } from 'react-native';
import { withNavigation } from 'react-navigation';class MyBackButton extends React.Component {
render() {
return <Button title="Back" onPress={() => { this.props.navigation.goBack() }} />; //此处可以使用navigation相关方法
}
}
// withNavigation returns a component that wraps MyBackButton and passes in the
// navigation prop
export default withNavigation(MyBackButton);

========================================================================

==》阻止多次点击重复跳转同一页面

const navigateOnce = (getStateForAction) => (action, lastState) => {
  const {type, routeName, params} = action;
   return (
    lastState &&
    (type === StackActions.PUSH) && //此处原先使用NavigationActions.NAVIGATE
    routeName === lastState.routes[lastState.routes.length - 1].routeName &&
    JSON.stringify(params) === JSON.stringify(lastState.routes[lastState.routes.length - 1].params)
  ) ? null : getStateForAction(action, lastState)
}//使用,MainNavigator是主导航页面通过createAppContainer方法的返回值
MainNavigator.router.getStateForAction = navigateOnce(MainNavigator.router.getStateForAction)

说明: MainNavigator经过 getStateForAction 检查校验,阻止多次点击跳转相同页面的情况发生

注意: 当你使用createSwitchNavigator时,就在路由数组外在套上一层switch数组,所以 lastState.routes[lastState.routes.length – 1] 可能取不到想要的值,因为还有一层数组

6月26日修改:

当使用了createSwitchNavigator时,修改上面的 navigateOnce 方法

const MainSwitch = createSwitchNavigator({
MainStack: MainStack,
... //其他
})const MainNavigator = createAppContainer(MainSwitch);const navigateOnce = (getStateForAction) => (action, lastState) => {
   const {type, routeName, params} = action;
//此处需要注意,使用了createSwitchNavigator后,lastState.routes[lastState.routes.length - 1]拿不到我们想要的那个对象
const mainStackRoutes = lastState && lastState.routes.find((item)=>item.key === "MainStack"); //拿到我们想要的那个对象
   return (
    mainStackRoutes &&
    (type === StackActions.PUSH) && //此处原先使用NavigationActions.NAVIGATE
    routeName === mainStackRoutes .routes[mainStackRoutes .routes.length - 1].routeName &&
    JSON.stringify(params) === JSON.stringify(mainStackRoutes .routes[mainStackRoutes .routes.length - 1].params)
  ) ? null : getStateForAction(action, lastState)
}//使用,MainNavigator是主导航页面通过createAppContainer方法的返回值
MainNavigator.router.getStateForAction = navigateOnce(MainNavigator.router.getStateForAction)

但是,当我们使用了createSwitchNavigator时,又使用了createDrawerNavigator,于是,再次修改上面的 navigateOnce 方法

const MainDrawer = createDrawerNavigator({
MainStack: MainStack,
... //其他
}, {
order: ['MainStack'],
initialRouteName: 'MainStack',
...
})const MainSwitch = createSwitchNavigator({
MainDrawer : MainDrawer ,
StartPresentation: StartPresentation, //APP启动页
... //其他
}, {
initialRouteName: 'StartPresentation',
...
})const MainNavigator = createAppContainer(MainSwitch);const navigateOnce = (getStateForAction) => (action, lastState) => {
  const {type, routeName, params} = action;
//使用了createDrawerNavigator后,上面的方法也拿不到我们想要的对象
const mainDrawerRoutes = lastState && lastState.routes.find((item)=>item.key === "MainDrawer "); //拿到我们想要的那个对象
//此处需要注意,使用了createSwitchNavigator后,lastState.routes[lastState.routes.length - 1]拿不到我们想要的那个对象
const mainStackRoutes = mainDrawerRoutes&& mainDrawerRoutes.routes.find((item)=>item.key === "MainStack"); //拿到我们想要的那个对象   return (
    mainStackRoutes &&
    (type === StackActions.PUSH) && //此处原先使用NavigationActions.NAVIGATE
    routeName === mainStackRoutes .routes[mainStackRoutes .routes.length - 1].routeName &&
    JSON.stringify(params) === JSON.stringify(mainStackRoutes .routes[mainStackRoutes .routes.length - 1].params)
  ) ? null : getStateForAction(action, lastState)
}//使用,MainNavigator是主导航页面通过createAppContainer方法的返回值
MainNavigator.router.getStateForAction = navigateOnce(MainNavigator.router.getStateForAction)

总而言之,当我们修改navigation代码结构时,navigateOnce 方法也要随着修改,灵活一点。。。

相关推荐
python开发_常用的python模块及安装方法
adodb:我们领导推荐的数据库连接组件bsddb3:BerkeleyDB的连接组件Cheetah-1.0:我比较喜欢这个版本的cheeta…
日期:2022-11-24 点赞:878 阅读:9,086
Educational Codeforces Round 11 C. Hard Process 二分
C. Hard Process题目连接:http://www.codeforces.com/contest/660/problem/CDes…
日期:2022-11-24 点赞:807 阅读:5,561
下载Ubuntn 17.04 内核源代码
zengkefu@server1:/usr/src$ uname -aLinux server1 4.10.0-19-generic #21…
日期:2022-11-24 点赞:569 阅读:6,410
可用Active Desktop Calendar V7.86 注册码序列号
可用Active Desktop Calendar V7.86 注册码序列号Name: www.greendown.cn Code: &nb…
日期:2022-11-24 点赞:733 阅读:6,183
Android调用系统相机、自定义相机、处理大图片
Android调用系统相机和自定义相机实例本博文主要是介绍了android上使用相机进行拍照并显示的两种方式,并且由于涉及到要把拍到的照片显…
日期:2022-11-24 点赞:512 阅读:7,820
Struts的使用
一、Struts2的获取  Struts的官方网站为:http://struts.apache.org/  下载完Struts2的jar包,…
日期:2022-11-24 点赞:671 阅读:4,903