react-native 实现tabBar上弹抽屉

不知道大家有没有遇到这样尴尬的问题:使用react-navigation createBottomTabNavigator实现底部tabBar,在tabBar的一个界面上弹出一个抽屉,发现抽屉没有覆盖tabBar?或者实现了抽屉覆盖tabBar,结果在tabBar界面上不能跳转到新的界面?

一、问题描述

大致需求如下:用户打开app底部有tabBar,tabBar有三个界面:Home,Release,Order;在Home界面左上角有一个按钮A,用户点击按钮A打开抽屉,显示界面Mine,点击空白处关闭抽屉;home界面有另一个按钮B,点击按钮B跳转新界面Login。

二、实现思路

来说说我的实现思路:

  • react-navigation createBottomTabNavigator实现底部tabBar;
  • createStackNavigator 中直接添加TabBar为Main;
  • react-native-drawer实现Drawer,弹出Mine;
  • this.props.navigation.push('Login')实现Home跳转Login界面;
Home.js
import React, {PureComponent} from 'react';
import {connect} from 'react-redux';
import {
    View,
    DeviceEventEmitter,
    Dimensions,
    StyleSheet,
} from 'react-native';
import * as Color from '../../constants/colors';
import Text from '../../components/common/scalingText';
import NavigationBar from '../../components/navigationBar/navigationBar';
import styles from '../../../assets/css/home';
import ControlPanel from '../../containers/mine/mine';
import Drawer from 'react-native-drawer'

const { width, height } = Dimensions.get('window');

class Home extends PureComponent {
    constructor(props) {
      super(props);
        this.state = {
        }
    }
    
    componentDidMount() {
        DeviceEventEmitter.addListener('clickDrawer',(events) =>{
            if(this._drawer && this._drawer.open()){
                this.closeControlPanel();
            }else{
                this.openControlPanel();
            }
        });

    }
    closeControlPanel = () => {
         this._drawer.close()
    };
    openControlPanel = () => {
         this._drawer.open()
    };
    render() {
        return (
            <Drawer
              ref={(ref) => this._drawer = ref}
              type="displace"
              content={<ControlPanel navigation={this.props.navigation}/>}
              tapToClose={true}
              openDrawerOffset={0.2} // 20% gap on the right side of drawer
              panCloseMask={0.2}
              closedDrawerOffset={-3}
              styles={drawerStyles}
              tweenHandler={(ratio) => ({
                main: { opacity:(2-ratio)/2 }
              })}
              >
                <NavigationBar
                    title={'首页'}
                    router={this.props.navigation}
                    hiddenBackIcon={false}
                    backIconFont='&#xe66e;'
                    backViewClick={()=>{
                        DeviceEventEmitter.emit('clickDrawer');
                        console.log('this.props.navigation',this.props.navigation);
                    }}
                />
                    <Text onPress={()=>{
                        this.props.navigation.push('PhoneCodeLogin');
                    }}
                    >登录</Text>
            </Drawer>
        );
    }
}

const drawerStyles = {
    drawer: { shadowColor: '#000000', shadowOpacity: 0.8, shadowRadius: 3},
    main: {paddingLeft: 3},
  }

function mapStateToProps(state){
    return {
        nav: state.nav
    };
}

function mapDispatchToProps (dispatch){
    return {};
}

export default connect(mapStateToProps, mapDispatchToProps)(Home);

router.js
...
import TabBar from './tabBar';
const StackRouteConfigs = {
    Login: {
        screen: Login,
        navigationOptions: {
            header: null
        }
    },
    Main: {
        screen: TabBar,
        navigationOptions: {
            header: null
        }
    },
}
...

上图效果图:

QQ20180926-101241.gif

发现问题:这种实现思路,Home可以正常跳转Login界面,但是Drawer弹出Mine并没有将TabBar覆盖。

改进1
  • react-navigation createBottomTabNavigator实现底部tabBar;
  • 新建Main文件,将TabBar包在其中,createStackNavigator 中添加Main为Main;
  • react-native-drawer实现Drawer,弹出Mine;
  • this.props.navigation.push('Login')实现Home跳转Login界面;
Main.js
import React, {PureComponent} from 'react';
import {connect} from 'react-redux';
import {
    View,
    Text,
    StyleSheet,
    DeviceEventEmitter
} from 'react-native';
import TabBar from '../../routes/tabBar';
import Drawer from 'react-native-drawer'
import ControlPanel from '../../containers/mine/mine';
class Main extends PureComponent {
    constructor(props) {
      super(props);
    }
    componentDidMount() {
        DeviceEventEmitter.addListener('clickDrawer',(events) =>{
            if(this._drawer && this._drawer.open()){
                this.closeControlPanel();
            }else{
                this.openControlPanel();
            }
        });

    }
    closeControlPanel = () => {
         this._drawer.close()
    };
    openControlPanel = () => {
         this._drawer.open()
    };
    render() {
        return (
            <Drawer
              ref={(ref) => this._drawer = ref}
              type="displace"
              content={<ControlPanel navigation={this.props.navigation}/>}
              tapToClose={true}
              openDrawerOffset={0.2} // 20% gap on the right side of drawer
              panCloseMask={0.2}
              closedDrawerOffset={-3}
              styles={drawerStyles}
              tweenHandler={(ratio) => ({
                main: { opacity:(2-ratio)/2 }
              })}
              >
                <View
                    style={styles.container}
                >
                    <TabBar/>
                </View>
            </Drawer>
        );
    }
}
const drawerStyles = {
  drawer: { shadowColor: '#000000', shadowOpacity: 0.8, shadowRadius: 3},
  main: {paddingLeft: 3},
}
const styles =StyleSheet.create({
    container: {
        flex: 1,
        backgroundColor: '#f5f5f5'
    },
});

function mapStateToProps(state){
    return {};
}

function mapDispatchToProps (dispatch){
    return {};
}

export default connect(mapStateToProps, mapDispatchToProps)(Main);

router.js
...
import Main from '../containers/app/main';
const StackRouteConfigs = {
    Login: {
        screen: Login,
        navigationOptions: {
            header: null
        }
    },
    Main: {
        screen: Main,
        navigationOptions: {
            header: null
        }
    },
}
...

上图效果图:

sybil052-20180926-092459.gif

发现问题:这种实现思路,Drawer弹出Mine覆盖了TabBar,但Home不可以跳转Login界面,报错。
原因:通过打印this.props.navigation发现根本没有push方法,通过main文件包装tabBar导致其不是一个Stack,不具有Stack中push、pop等方法。

sybil052-20180926-093043.png
改进2
  • react-navigation createBottomTabNavigator实现底部tabBar;
  • react-navigation createDrawerNavigator实现Drawer;
  • 新建drawer文件,将Drawer作为TabBar的父导航,添加到createStackNavigator中;
  • this.props.navigation.push('Login')实现Home跳转Login界面;

点击Home界面按钮A打开抽屉方法
this.props.navigation.toggleDrawer(); // 如果抽屉状态为打开,操作为关闭;如果抽屉状态为关闭,操作为打开
this.props.navigation.openDrawer(); // 打开抽屉
this.props.navigation.closeDrawer(); // 关闭抽屉

最后发现,改进之后的可以满足需求,下面直接上代码~

三、具体代码

root.js
import React from 'react';
import {AppRegistry} from 'react-native';
import {name as appName} from '../app.json';
import {Provider} from 'react-redux';
import configureStore from './store/store';
import {AppNavigator} from './routes/routers';

const store = configureStore();

export default class Root extends React.Component{
    render(){
        return(
            <Provider store={store}>
              <AppNavigator/>
            </Provider>
        )
    }
}

AppRegistry.registerComponent(appName, () => Root);
router.js
import React from 'react';
import PropTypes from 'prop-types';
import {connect} from 'react-redux';
import {
    createStackNavigator,
} from 'react-navigation';
import {
    reduxifyNavigator,
    createReactNavigationReduxMiddleware,
} from 'react-navigation-redux-helpers';

import Login from '../containers/login/login';
import Drawer from './drawer';

const navMiddleware = createReactNavigationReduxMiddleware(
    'root',
    state => state.nav
);

const StackNavigatorConfigs = {
    initialRouteName: 'Main', // 初始化哪个界面为根界面
    mode: 'card', // 跳转方式:默认的card,在iOS上是从右到左跳转,在Android上是从下到上,都是使用原生系统的默认跳转方式。
    headerMode:'screen', // 导航条动画效果:float表示会渐变,类似于iOS的原生效果,screen表示没有渐变。none表示隐藏导航条
    navigationOptions: {
      gesturesEnabled: false, //是否可以使用手势关闭此屏幕。在iOS上默认为true,在Android上为false
    },
};

const StackRouteConfigs = {
    Login:{
      screen: Login,
      navigationOptions: {
          header: null
      }
    },
    Main: {
        screen: Drawer,
        navigationOptions: {
            header: null
        }
    },
   
};

const RootNavigator = createStackNavigator(StackRouteConfigs, StackNavigatorConfigs);

const AppWithNavigationState = reduxifyNavigator(RootNavigator, 'root');

const mapStateToProps = state => ({
    state: state.nav,
});

const AppNavigator = connect(mapStateToProps)(AppWithNavigationState);

export {RootNavigator, AppNavigator, navMiddleware};
drawer.js
import React, {Component} from 'react';
import {
    Dimensions,
} from 'react-native';
import {
    createDrawerNavigator,
} from "react-navigation";

import Mine from '../containers/mine/mine';
import TabBar from './tabBar';

const {width,height} = Dimensions.get('window');

export default Drawer = createDrawerNavigator({
    TabBar: {
        screen: TabBar,
        navigationOptions: {
            header: null
        }
    },
},
{
    drawerPosition:'left',
    drawerWidth: width * 0.8,
    drawerLockMode:'locked-closed',
    useNativeAnimations:true,
    contentComponent: props => {
        return <Mine {...props}/>;
    }
})
tabBar.js
import React, {Component} from 'react';
import {
    StyleSheet,
    Dimensions,
    Platform,
    Text,
    TouchableOpacity,
    Image,
    View
} from 'react-native';
import {
    createBottomTabNavigator,
} from "react-navigation";
import TabBarIcon from '../../assets/imgs/tabBar/tabbar.png';
import Home from '../containers/home/home';
import Release from '../containers/release/release';
import Order from '../containers/order/order';

import * as Color from '../constants/colors';

const {width,height} = Dimensions.get('window');

const styles = StyleSheet.create({
    pressedIcon: {
        fontFamily: 'iconfont',
        fontSize: 23,
        color: Color.BLUE_17A9DF,
        marginTop: 7,
    },
    renderIcon: {
        fontFamily: 'iconfont',
        fontSize: 23,
        color: Color.GRAY_B4B4B4,
        marginTop: 7,
    }
});

export default Tab = createBottomTabNavigator({
    Home: {
        screen: Home,
        navigationOptions: {
            tabBarPosition: 'bottom',
            tabBarLabel: '首页',
            showLabel: false,
            tabBarIcon: ({tintColor, focused}) => (
                focused ? <Text style={styles.pressedIcon}>&#xe664;</Text> : <Text style={styles.renderIcon}>&#xe663;</Text>
            ),
        }
    },
    Release: {
        screen: Release,
        navigationOptions: {
            tabBarVisible: false, // tabBar导航栏显示/隐藏 false:隐藏 true:显示
            tabBarPosition: 'bottom',
            tabBarLabel: '发布',
            showLabel: false,
            tabBarIcon: ({tintColor, focused}) => (
                <Image source={TabBarIcon} style={{marginBottom:25}}></Image>
            ),
        }
    },
    Order: {
        screen: Order,
        navigationOptions: {
            tabBarPosition: 'bottom',
            tabBarLabel: '订单',
            showLabel: false,
            tabBarIcon: ({tintColor, focused}) => (
                focused ? <Text style={styles.pressedIcon}>&#xe666;</Text> : <Text style={styles.renderIcon}>&#xe65c;</Text>
            ),
        }
    },
}, {
    animationEnabled: true,
    swipeEnabled: false,
    //是否可以滑动切换
    swipeEnabled: false,
    //切换是否有动画
    animationEnabled: true,
    //进入App的首页面
    initialRouteName: 'Home',
    //对于导航的设置
    tabBarOptions: {
        activeTintColor: Color.BLUE_17A9DF,
        inactiveTintColor: Color.GRAY_B4B4B4,
        //android特有下划线的颜色1
        indicatorStyle: {height: 0},
        //文字的样式
        labelStyle: {
            fontSize: 10,
            paddingBottom: 4
        },
        //对于导航的stytles
        style :{
            borderTopColor: 'rgba(0,0,0,0.2)',
            borderTopWidth: 1,
            backgroundColor: Color.WHITE_FFFFFF,
            // height: Dimensions.get('window').height*0.08,
        }
    }
});
Home.js
import React, {PureComponent} from 'react';
import {connect} from 'react-redux';
import {
    View,
    Dimensions,
} from 'react-native';
import * as Color from '../../constants/colors';
import Text from '../../components/common/scalingText';
import NavigationBar from '../../components/navigationBar/navigationBar';
import styles from '../../../assets/css/home';

const { width, height } = Dimensions.get('window');

class Home extends PureComponent {
    constructor(props) {
      super(props);
        this.state = {
        }
    }
    componentDidMount() {
    }

    render() {
        return (
                <View style={styles.container}>
                    <NavigationBar
                        title={'首页'}
                        router={this.props.navigation}
                        hiddenBackIcon={false}
                        backIconFont='&#xe66e;'
                        backViewClick={()=>{
                            this.props.navigation.toggleDrawer();
                        }}
                        />
                    <Text  
                        onPress={()=>{
                            this.props.navigation.push('Login');
                        }
                      >登录</Text>
                </View>
        );
    }
}

function mapStateToProps(state){
    return {
        nav: state.nav
    };
}

function mapDispatchToProps (dispatch){
    return {};
}

export default connect(mapStateToProps, mapDispatchToProps)(Home);

四、效果图

sybil052-20180925-181128.gif

完毕~~~你学会了吗?

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,099评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,828评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,540评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,848评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,971评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,132评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,193评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,934评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,376评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,687评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,846评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,537评论 4 335
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,175评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,887评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,134评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,674评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,741评论 2 351

推荐阅读更多精彩内容