React Native 系列(九) -- Tab标签组件

前言

本系列是基于React Native版本号0.44.3写的。很多的App都使用了Tab标签组件,例如QQ微信等等,就是切换不同的选项,显示不同的内容。那么这篇文章将介绍RN中的Tab标签组件。

Tab标签

什么是Tab标签?(ps:我是这样叫的),就拿微信来说吧,底部有4个选项卡,点击不同的按钮切换不同的内容。
RN中有两个组件负责实现这样的效果,它们是:

  • TabBarIOS
  • TabNavigator

TabBarIOS

NavigatorIOS相似,看名字就知道该组件只适用于iOS,不能用于android

TabBarIOS 常用属性

barTintColor string:标签栏的背景颜色。 
style:样式 
tintColor string: 当前被选中的标签图标的颜色。 unselectedItemTintColor string: 当前没有被选中的标签图标的颜色。仅在iOS 10及以上版本有效 
translucent bool: 一个布尔值,决定标签栏是否需要半透明化。

TabBarIOS.Item

  • 注意:TabBarIOS.Item必须包装一个View,作为点击选项卡,切换的view

TabBarIOS.Item 常用属性

badge string,number :在图标右上角显示一个红色的气泡。
 
icon Image.propTypes.source :给当前标签指定一个自定义的图标。如果定义了systemIcon属性, 这个属性会被忽略。 

onPress function :当此标签被选中时调用。你应该修改组件的状态来使得selected={true}。 

selected bool :这个属性决定了子视图是否可见。如果你看到一个空白的页面,很可能是没有选中任何一个标签。 

selectedIcon Image.propTypes.source :当标签被选中的时候显示的自定义图标。如果定义了systemIcon属性,这个属性会被忽略。如果定义了icon而没定义这个属性,在选中的时候图标会染上蓝色。 

systemIcon enum('bookmarks','contacts','downloads','favorites','featured','history','more','most-recent','most-viewed','recents','search','top-rated') :一些预定义的系统图标。注意如果你使用了此属性,标题和自定义图标都会被覆盖为系统定义的值。 

title string :在图标下面显示的标题文字。如果定义了systemIcon属性,这个属性会被忽略

使用步骤

  1. 创建TabBar标签: 使用下面行代码,底部就会有一个条

    <TabBarIOS>
    </TabBarIOS>
  2. 添加选项卡

    <TabBarIOS.Item
            title='首页'
            icon={{uri: 'tabbar_home_select',scale: 3}}
    >
        <View style={{flex: 1,justifyContent: 'center',alignItems: 'center',backgroundColor: 'red'}}>
            <Text>首页</Text>
        </View>
    </TabBarIOS.Item>
  3. 监听按钮点击,切换界面

    • 只要设置对应的tabBarItemselectedtrue,就会自动跳转到对应界面
    • 注意:tabBarItemselected属性不能写死,可以定义个flag来标记当前选中的item
    • 监听tabBarItem的点击,修改selected属性

实战演练

先看效果图:

我们在index.ios.js文件做修改,直接创建一个TabBarIOS组件:

render() {
    return (
        <TabBarIOS
            tintColor='orange'
            barTintColor='black'
        >
        </TabBarIOS>
    );
}

再创建三个选项卡(ps:3个选项卡创建方式类似,所以这里只贴出创建一个的代码,),icon是图片,关于图片方面的知识,本文最后做一个总结吧:

<TabBarIOS.Item
        title='首页'
        icon={{uri: 'tabbar_home_select',scale: 3}}
        selected={true}
    >
        <View style={{flex: 1,backgroundColor: 'red'}}>
            <Text>首页</Text>
        </View>

</TabBarIOS.Item>

监听按钮点击,然后实现界面切换,我们需要定义一个flag记录当前选中的,

constructor(props) {
    super(props);
    // 初始状态
    this.state = {
        selectIndex:0,};
}

TabBarIOS.Item添加点击事件:

<TabBarIOS.Item
    title='首页'
    icon={{uri: 'tabbar_home_select',scale: 3}}
    onPress={()=>{
        this.setState({
            selectIndex:0
        })
    }}
    selected={0==this.state.selectIndex}
>

我们会发现,创建3个TabBarIOS.Item的代码是一样的,避免写重复代码,我们可以抽取出来:

_renderTabBarItem(title,iconName,selected,bgColor,badgeNumber){
    return (
        <TabBarIOS.Item
            title={title}
            icon={{uri: iconName,scale: 3}}
            onPress={()=>{
                this.setState({
                    selectIndex: selected
                })
            }}
            badge={badgeNumber}
            selected={this.state.selectIndex === selected}
        >
            <View style={{flex: 1,backgroundColor: bgColor}}>
                <Text>{title}</Text>
            </View>
        </TabBarIOS.Item>
    );
}

然后我们整个创建标签Tab的代码就应该是这样:

render() {
    return (
        <TabBarIOS
            tintColor='orange'
            barTintColor='black'
        >
            {this._renderTabBarItem("首页",'tabbar_my_select','red',null)}
            {this._renderTabBarItem("消息",1,'yellow',2)}
            {this._renderTabBarItem("我的",2,'cyan',null)}

        </TabBarIOS>
    );
}

TabNavigator

本系列上篇文章中,介绍到React Navigation组件中包含了TabNavigator。所以接下来的讲解是在引入了React Navigation的基础之上的。

常用属性

screen:和导航的功能是一样的,对应界面名称,可以在其他页面通过这个screen传值和跳转。  
      
navigationOptions:配置TabNavigator的一些属性  
{  
    title:标题,会同时设置导航条和标签栏的title  
    tabBarVisible:是否隐藏标签栏。默认不隐藏(true)  
    tabBarIcon:设置标签栏的图标。需要给每个都设置  
    tabBarLabel:设置标签栏的title。推荐  
}


tabBarPosition:设置tabbar的位置,iOS默认在底部,安卓默认在顶部。(属性值:'top','bottom')  
      
swipeEnabled:是否允许在标签之间进行滑动  
      
animationEnabled:是否在更改标签时显示动画  
      
lazy:是否根据需要懒惰呈现标签,而不是提前,意思是在app打开的时候将底部标签栏全部加载,默认false,推荐为true  
      
trueinitialRouteName: 设置默认的页面组件  
      
backBehavior:按 back 键是否跳转到第一个Tab(首页), none 为不跳转  
      
tabBarOptions:配置标签栏的一些属性iOS属性  
      
activeTintColor:label和icon的前景色 活跃状态下  
      
activeBackgroundColor:label和icon的背景色 活跃状态下  
      
inactiveTintColor:label和icon的前景色 不活跃状态下  
      
inactiveBackgroundColor:label和icon的背景色 不活跃状态下  
      
showLabel:是否显示label,默认开启 style:tabbar的样式  
      
labelStyle:label的样式安卓属性  
      
activeTintColor:label和icon的前景色 活跃状态下  
      
inactiveTintColor:label和icon的前景色 不活跃状态下  
      
showIcon:是否显示图标,默认关闭  
      
showLabel:是否显示label,默认开启 style:tabbar的样式  
      
labelStyle:label的样式 upperCaseLabel:是否使标签大写,默认为true  
      
pressColor:material涟漪效果的颜色(安卓版本需要大于5.0)  
      
pressOpacity:按压标签的透明度变化(安卓版本需要小于5.0)  
      
scrollEnabled:是否启用可滚动选项卡 tabStyle:tab的样式  
      
indicatorStyle:标签指示器的样式对象(选项卡底部的行)。安卓底部会多出一条线,可以将height设置为0来暂时解决这个问题  
      
labelStyle:label的样式  
      
iconStyle:图标样式

实战演练

我们创建App.js,然后在index.ios.js文件引用该组件:

import App from './App'
export default class RNDemoTwo extends Component {
    render() {
        return (
            <App/>
        );
    }
}

现在来配置下App.js文件:

import {TabNavigator} from 'react-navigation'
import HelloViewCompnent from './HelloViewComponent'
import DetailComponent from './DetailComponent'
import ThreeComponent from './Three'
import CustTabBarItem from "./TabBarItem"


const SimpleApp = TabNavigator({
    Home: {
        screen: HelloViewCompnent,navigationOptions: ({navigation}) => ({
            tabBarVisible: true,tabBarLabel: '首页',tabBarIcon: ({focused,tintColor})=>(
                <CustTabBarItem
                    tintColor={tintColor}
                    focused={focused}
                    selectImage={{uri: 'tabbar_home_select'}}
                    normalImage={{uri: 'tabbar_home'}}
                />
            ),})
    },Detail: {
        screen: DetailComponent,navigationOptions: ({navigation}) => ({
            tabBarLabel: '消息',tintColor})=>(
                <CustTabBarItem
                    tintColor={tintColor}
                    focused={focused}
                    selectImage={{uri: 'tabbar_licai_select'}}
                    normalImage={{uri: 'tabbar_licai'}}
                />
            ),Three: {
        screen: ThreeComponent,navigationOptions: ({navigation}) => ({
            tabBarLabel: '我的',tintColor})=>(
                <CustTabBarItem
                    tintColor={tintColor}
                    focused={focused}
                    selectImage={{uri: 'tabbar_my_select'}}
                    normalImage={{uri: 'tabbar_my'}}
                />
            ),})
    }
},{
    tabBarPosition:'bottom',swipeEnabled:false,animationEnabled:false,lazy:true,tabBarOptions:{
        activeTintColor:'red',inactiveTintColor:'black',style:{backgroundColor:'#fff',},labelStyle: {
            fontSize: 16,// 文字大小
        },}
})

export default SimpleApp

可以看到我们导入了三个页面组件,一个CustTabBarItem
三个页面组件很简单,都类似,由于篇幅原因,只贴出一个就行了:

export default class HelloViewCompnent extends Component {

    render() {
        return (
            <View style={styles.container}>
                <Text style={styles.welcome}>
                    首页
                </Text>
            </View>
        );
    }
}

const styles = StyleSheet.create({
    container: {
        flex: 1,backgroundColor: 'white',welcome: {
        fontSize: 20,textAlign: 'center',margin: 10,});

最后,就是我们的TabBarItem.js文件了:

export default class TabBarItem extends Component {
    render(){
        return (
            <Image source={this.props.focused ? this.props.selectImage : this.props.normalImage}
            style={{tintColor: this.props.tintColor,width: 25,height: 25}}/>
        );
    }
}

保存代码,运行项目,不出意外,应该和使用TabBarIOS是一样的效果。

关于Image组件

由于这里我们都使用到<Image/>组件,这里就稍微讲一下。

  • Image:用于加载图片,可加载网络图片,也可以加载本地图片
  • Image常用属性

    source {uri: string},number : 设置Image图片资源 
    blurRadius number:让图片模糊 
    defaultSource {uri: string,width: number,height:number,scale: number},number 占位图片,在读取图片时默认显示的加载提示图片 
    
    resizeMode enum('cover','contain','stretch','repeat','center') 决定图片尺寸大小。 
    
    cover: 在保持图片宽高比的前提下缩放图片,直到宽度和高度都大于等于容器视图的尺寸(如果容器有padding内衬的话,则相应减去)。译注:这样图片完全覆盖甚至超出容器,容器中不留任何空白。 
    
    contain: 在保持图片宽高比的前提下缩放图片,直到宽度和高度都小于等于容器视图的尺寸(如果容器有padding内衬的话,则相应减去)。译注:这样图片完全被包裹在容器中,容器中可能留有空白 
    
    stretch: 拉伸图片且不维持宽高比,直到宽高都刚好填满容器。
    repeat: 重复平铺图片直到填满容器。图片会维持原始尺寸。仅iOS可用。 
    center: 居中不拉伸。
  • 本地图片存放位置

      1. 直接放在RN项目中
      1. 可以放在ios项目中,放到images.xcassets文件中
      1. 可以放在android项目中(安卓中图片文字不能以数字开头,也不能有大写字母)
  • 如何加载本地图片

    • RN中加载资源:require(文件路径),用于加载RN中的资源,不管是图片,还是json都是一样的
    • uri:指定一个资源路径,就会自动加载
    • uri加载注意:通过uri加载资源,必须设置图片尺寸,否则不显示
    • 如果网络加载http图片,iOS默认不支持,需要开启App Transport

致谢

如果发现有错误的地方,欢迎各位指出,谢谢!

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐


react 中的高阶组件主要是对于 hooks 之前的类组件来说的,如果组件之中有复用的代码,需要重新创建一个父类,父类中存储公共代码,返回子类,同时把公用属性...
我们上一节了解了组件的更新机制,但是只是停留在表层上,例如我们的 setState 函数式同步执行的,我们的事件处理直接绑定在了 dom 元素上,这些都跟 re...
我们上一节了解了 react 的虚拟 dom 的格式,如何把虚拟 dom 转为真实 dom 进行挂载。其实函数是组件和类组件也是在这个基础上包裹了一层,一个是调...
react 本身提供了克隆组件的方法,但是平时开发中可能很少使用,可能是不了解。我公司的项目就没有使用,但是在很多三方库中都有使用。本小节我们来学习下如果使用该...
mobx 是一个简单可扩展的状态管理库,中文官网链接。小编在接触 react 就一直使用 mobx 库,上手简单不复杂。
我们在平常的开发中不可避免的会有很多列表渲染逻辑,在 pc 端可以使用分页进行渲染数限制,在移动端可以使用下拉加载更多。但是对于大量的列表渲染,特别像有实时数据...
本小节开始前,我们先答复下一个同学的问题。上一小节发布后,有小伙伴后台来信问到:‘小编你只讲了类组件中怎么使用 ref,那在函数式组件中怎么使用呢?’。确实我们...
上一小节我们了解了固定高度的滚动列表实现,因为是固定高度所以容器总高度和每个元素的 size、offset 很容易得到,这种场景也适合我们常见的大部分场景,例如...
上一小节我们处理了 setState 的批量更新机制,但是我们有两个遗漏点,一个是源码中的 setState 可以传入函数,同时 setState 可以传入第二...
我们知道 react 进行页面渲染或者刷新的时候,会从根节点到子节点全部执行一遍,即使子组件中没有状态的改变,也会执行。这就造成了性能不必要的浪费。之前我们了解...
在平时工作中的某些场景下,你可能想在整个组件树中传递数据,但却不想手动地通过 props 属性在每一层传递属性,contextAPI 应用而生。
楼主最近入职新单位了,恰好新单位使用的技术栈是 react,因为之前一直进行的是 vue2/vue3 和小程序开发,对于这些技术栈实现机制也有一些了解,最少面试...
我们上一节了了解了函数式组件和类组件的处理方式,本质就是处理基于 babel 处理后的 type 类型,最后还是要处理虚拟 dom。本小节我们学习下组件的更新机...
前面几节我们学习了解了 react 的渲染机制和生命周期,本节我们正式进入基本面试必考的核心地带 -- diff 算法,了解如何优化和复用 dom 操作的,还有...
我们在之前已经学习过 react 生命周期,但是在 16 版本中 will 类的生命周期进行了废除,虽然依然可以用,但是需要加上 UNSAFE 开头,表示是不安...
上一小节我们学习了 react 中类组件的优化方式,对于 hooks 为主流的函数式编程,react 也提供了优化方式 memo 方法,本小节我们来了解下它的用...
开源不易,感谢你的支持,❤ star me if you like concent ^_^
hel-micro,模块联邦sdk化,免构建、热更新、工具链无关的微模块方案 ,欢迎关注与了解
本文主题围绕concent的setup和react的五把钩子来展开,既然提到了setup就离不开composition api这个关键词,准确的说setup是由...
ReactsetState的执行是异步还是同步官方文档是这么说的setState()doesnotalwaysimmediatelyupdatethecomponent.Itmaybatchordefertheupdateuntillater.Thismakesreadingthis.staterightaftercallingsetState()apotentialpitfall.Instead,usecom