taro 滚到页面顶部(Taro小程序自定义顶部导航栏)

微信自带的顶部导航栏是无法支持自定义icon和增加元素的,在开发小程序的时候自带的根本满足不了需求,分享一个封装好的组件,支持自定义icon、扩展dom,适配安卓、ios、h5,全面屏。

我用的是京东的Taro多端编译框架写的小程序,原生的也可以适用,用到的微信/taro的api做调整就行,实现效果如下。

taro 滚到页面顶部(Taro小程序自定义顶部导航栏)(1)

taro 滚到页面顶部(Taro小程序自定义顶部导航栏)(2)

1、NavBar.js

import Taro from '@tarojs/taro'; import React, { Component } from 'react' import { View } from '@tarojs/components'; import { isFunction } from '../../utils/index' //判断是否为函数,可以用loadsh的_isFunction,也可以自己封装。 import './NavBar.less'; function getSystemInfo () { if (Taro.globalSystemInfo && !Taro.globalSystemInfo.ios) { return Taro.globalSystemInfo; } else { // h5环境下忽略navbar if (!isFunction(Taro.getSystemInfoSync)) { return null; } let systemInfo = Taro.getSystemInfoSync() || { model: '', system: '' }; let ios = !!(systemInfo.system.toLowerCase().search('ios') 1); let rect; try { rect = Taro.getMenuButtonBoundingClientRect ? Taro.getMenuButtonBoundingClientRect() : null; if (rect === null) { throw 'getMenuButtonBoundingClientRect error'; } //取值为0的情况 有可能width不为0 top为0的情况 if (!rect.width || !rect.top || !rect.left || !rect.height) { throw 'getMenuButtonBoundingClientRect error'; } } catch (error) { let gap; //胶囊按钮上下间距 使导航内容居中 let width = 96; //胶囊的宽度 if (systemInfo.platform === 'android') { gap = 8; width = 96; } else if (systemInfo.platform === 'devtools') { if (ios) { gap = 5.5; //开发工具中ios手机 } else { gap = 7.5; //开发工具中android和其他手机 } } else { gap = 4; width = 88; } if (!systemInfo.statusBarHeight) { //开启wifi的情况下修复statusBarHeight值获取不到 systemInfo.statusBarHeight = systemInfo.screenHeight - systemInfo.windowHeight - 20; } rect = { //获取不到胶囊信息就自定义重置一个 bottom: systemInfo.statusBarHeight gap 32, height: 32, left: systemInfo.windowWidth - width - 10, right: systemInfo.windowWidth - 10, top: systemInfo.statusBarHeight gap, width: width }; console.log('error', error); console.log('rect', rect); } let navBarHeight = ''; if (!systemInfo.statusBarHeight) { //开启wifi和打电话下 systemInfo.statusBarHeight = systemInfo.screenHeight - systemInfo.windowHeight - 20; navBarHeight = (function () { let gap = rect.top - systemInfo.statusBarHeight; return 2 * gap rect.height; })(); systemInfo.statusBarHeight = 0; systemInfo.navBarExtendHeight = 0; //下方扩展4像素高度 防止下方边距太小 } else { navBarHeight = (function () { let gap = rect.top - systemInfo.statusBarHeight; return systemInfo.statusBarHeight 2 * gap rect.height; })(); if (ios) { systemInfo.navBarExtendHeight = 4; //下方扩展4像素高度 防止下方边距太小 } else { systemInfo.navBarExtendHeight = 0; } } systemInfo.navBarHeight = navBarHeight; //导航栏高度不包括statusBarHeight systemInfo.capsulePosition = rect; //右上角胶囊按钮信息bottom: 58 height: 32 left: 317 right: 404 top: 26 width: 87 目前发现在大多机型都是固定值 为防止不一样所以会使用动态值来计算nav元素大小 systemInfo.ios = ios; //是否ios Taro.globalSystemInfo = systemInfo; //将信息保存到全局变量中,后边再用就不用重新异步获取了 //console.log('systemInfo', systemInfo); return systemInfo; } } let globalSystemInfo = getSystemInfo(); class NavBar extends Component { constructor(props) { super(props); this.state = { configStyle: this.setStyle(globalSystemInfo) }; } static options = { multipleSlots: true, addGlobalClass: true }; UNSAFE_componentWillMount () { //获取高度 // let query = Taro.createSelectorQuery().in(this.$scope) // query.select('.lxy-nav-bar').boundingClientRect(rect=>{ // const navHeight = rect.height // this.props.personalHomeMod.changeState('navHeight',navHeight) // // console.log('navHeight',toJS(this.props.personalHomeMod.state)) // }).exec() } componentDidShow () { if (globalSystemInfo.ios) { globalSystemInfo = getSystemInfo(); this.setState({ configStyle: this.setStyle(globalSystemInfo) }); } } handleBackClick () { if (isFunction(this.props.onBack)) { this.props.onBack(); } else { const pages = Taro.getCurrentPages(); if (pages.length >= 2) { Taro.navigateBack({ delta: this.props.delta }); } } } handleGoHomeClick () { if (isFunction(this.props.onHome)) { this.props.onHome(); } } handleSearchClick () { if (isFunction(this.props.onSearch)) { this.props.onSearch(); } } static defaultProps = { extClass: '', background: 'rgba(255,255,255,1)', //导航栏背景 color: '#000000', title: '', searchText: '点我搜索', searchBar: false, back: false, home: false, iconTheme: 'black', delta: 1 }; setStyle (systemInfo) { const { statusBarHeight, navBarHeight, capsulePosition, navBarExtendHeight, ios, windowWidth } = systemInfo; const { back, home, title, color } = this.props; let rightDistance = windowWidth - capsulePosition.right; //胶囊按钮右侧到屏幕右侧的边距 let leftWidth = windowWidth - capsulePosition.left; //胶囊按钮左侧到屏幕右侧的边距 let navigationbarinnerStyle = [ `color:${color}`, //`background:${background}`, `height:${navBarHeight navBarExtendHeight}px`, `padding-top:${statusBarHeight}px`, `padding-right:${leftWidth}px`, `padding-bottom:${navBarExtendHeight}px` ].join(';'); let navBarLeft; if ((back && !home) || (!back && home)) { navBarLeft = [ `width:${capsulePosition.width}px`, `height:${capsulePosition.height}px`, `margin-left:0px`, `margin-right:${rightDistance}px` ].join(';'); } else if ((back && home) || title) { navBarLeft = [ `width:${capsulePosition.width}px`, `height:${capsulePosition.height}px`, `margin-left:${rightDistance}px` ].join(';'); } else { navBarLeft = [`width:auto`, `margin-left:0px`].join(';'); } return { navigationbarinnerStyle, navBarLeft, navBarHeight, capsulePosition, navBarExtendHeight, ios, rightDistance }; } render () { const { navigationbarinnerStyle, navBarLeft, navBarHeight, capsulePosition, navBarExtendHeight, ios, rightDistance } = this.state.configStyle; const { title, background, backgroundColorTop, back, home, searchBar, searchText, iconTheme, extClass } = this.props; let nav_bar__center; if (title) { nav_bar__center = <text>{title}</text>; } else if (searchBar) { nav_bar__center = ( <View className='lxy-nav-bar-search' style={`height:${capsulePosition.height}px;`} onClick={this.handleSearchClick.bind(this)} > <View className='lxy-nav-bar-search__icon' /> <View className='lxy-nav-bar-search__input'>{searchText}</View> </View> ); } else { /* eslint-disable */ nav_bar__center = this.props.renderCenter; /* eslint-enable */ } return ( <View className={`lxy-nav-bar ${ios ? 'ios' : 'android'} ${extClass}`} style={`background: ${backgroundColorTop ? backgroundColorTop : background};height:${navBarHeight navBarExtendHeight}px;`} > <View className={`lxy-nav-bar__placeholder ${ios ? 'ios' : 'android'}`} style={`padding-top: ${navBarHeight navBarExtendHeight}px;`} /> <View className={`lxy-nav-bar__inner ${ios ? 'ios' : 'android'}`} style={`background:${background};${navigationbarinnerStyle};`} > <View className='lxy-nav-bar__left' style={navBarLeft}> {back && !home && ( <View onClick={this.handleBackClick.bind(this)} className={`lxy-nav-bar__button lxy-nav-bar__btn_goback ${iconTheme}`} /> )} {!back && home && ( <View onClick={this.handleGoHomeClick.bind(this)} className={`lxy-nav-bar__button lxy-nav-bar__btn_gohome ${iconTheme}`} /> )} {back && home && ( <View className={`lxy-nav-bar__buttons ${ios ? 'ios' : 'android'}`}> <View onClick={this.handleBackClick.bind(this)} className={`lxy-nav-bar__button lxy-nav-bar__btn_goback ${iconTheme}`} /> <View onClick={this.handleGoHomeClick.bind(this)} className={`lxy-nav-bar__button lxy-nav-bar__btn_gohome ${iconTheme}}`} /> </View> )} {!back && !home && this.props.renderLeft} </View> <View className='lxy-nav-bar__center' style={`padding-left: ${rightDistance}px`}> {nav_bar__center} </View> <View className='lxy-nav-bar__right' style={`margin-right: ${rightDistance}px`}> {this.props.renderRight} </View> </View> </View> ); } } export default NavBar;

2、NavBar.less

view, text, scroll-view, input, button, image, cover-view { box-sizing: border-box; } page { /* prettier-ignore */ --height: 44PX; /* 4*2 32 */ /* prettier-ignore */ --right: 97PX; /* 10 87 */ /* prettier-ignore */ --navBarExtendHeight: 4PX; /* prettier-ignore */ --navBarHeight: 68PX; box-sizing: border-box; } .lxy-nav-bar .ios { /* prettier-ignore */ --height: 44PX; /* 4*2 32 */ /* prettier-ignore */ --right: 97PX; /* 10 87 */ /* prettier-ignore */ --navBarExtendHeight: 4PX; box-sizing: border-box; } .lxy-nav-bar .android { /* prettier-ignore */ --height: 48PX; /* 8*2 32 */ /* prettier-ignore */ --right: 96PX; /* 10 87 */ /* prettier-ignore */ --navBarExtendHeight: 4PX; box-sizing: border-box; } .lxy-nav-bar .devtools { /* prettier-ignore */ --height: 42PX; /* 5*2 32 */ /* prettier-ignore */ --right: 88PX; /* 10 87 */ /* prettier-ignore */ --navBarExtendHeight: 4PX; box-sizing: border-box; } .lxy-nav-bar__inner { position: fixed; top: 0; left: 0; z-index: 5001; /* prettier-ignore */ height: var(--navBarHeight); display: flex; align-items: center; padding-right: var(--right); width: 100%; /* prettier-ignore */ padding-top: 20PX; /* prettier-ignore */ padding-bottom:4PX; .placeholder { position: absolute; top: 0; left: 0; width: 100%; } } .lxy-nav-bar__inner .lxy-nav-bar__left { position: relative; width: var(--right); /* prettier-ignore */ height: 32PX; /* padding-left: 10PX; */ /* prettier-ignore */ margin-left:10PX; display: flex; align-items: center; } .lxy-nav-bar__buttons { height: 100%; width: 100%; display: flex; align-items: center; /* prettier-ignore */ border-radius: 16PX; border: 1px solid rgba(204, 204, 204, 0.6); position: relative; } .lxy-nav-bar__buttons.android { border: 1px solid rgba(234, 234, 234, 0.6); } .lxy-nav-bar__buttons::after { position: absolute; content: ''; width: 1px; /* prettier-ignore */ height: 18.4PX; background: rgba(204, 204, 204, 0.6); left: 50%; top: 50%; transform: translate(-50%, -50%); } .lxy-nav-bar__buttons.android::after { background: rgba(234, 234, 234, 0.6); } .lxy-nav-bar__button { width: 50%; height: 100%; display: flex; /* prettier-ignore */ font-size: 12PX; background-repeat: no-repeat; background-position: center center; background-size: 1em 2em; } .lxy-nav-bar__inner .lxy-nav-bar__left .lxy-nav-bar__btn_goback:active, .lxy-nav-bar__inner .lxy-nav-bar__left .lxy-nav-bar__btn_gohome:active { opacity: 0.5; } .lxy-nav-bar__inner .lxy-nav-bar__center { /* prettier-ignore */ font-size: 17PX; /* prettier-ignore */ line-height: 17PX; text-align: center; position: relative; flex: 1; display: -webkit-box; display: -webkit-flex; display: flex; align-items: center; justify-content: center; /* prettier-ignore */ padding-left: 10PX; text { margin-top: -2px; font-size:34px; font-weight:550; line-height:44px; } } .lxy-nav-bar__inner .lxy-nav-bar__loading { font-size: 0; } .lxy-nav-bar__inner .lxy-nav-bar__loading .lxy-loading { margin-left: 0; } .lxy-nav-bar__inner .lxy-nav-bar__right { /* prettier-ignore */ margin-right: 10PX; } .lxy-nav-bar__placeholder { height: var(--navBarHeight); background: #f8f8f8; position: relative; z-index: 50; visibility: hidden; } .lxy-nav-bar-search { width: 100%; height: 100%; display: flex; justify-content: center; align-items: center; width: 100%; /* prettier-ignore */ height: 32PX; /* prettier-ignore */ border-radius: 16PX; position: relative; background: #f6f6f6; } .lxy-nav-bar-search__input { height: 100%; display: flex; align-items: center; color: #999; /* prettier-ignore */ font-size: 15PX; /* prettier-ignore */ line-height: 15PX; } .lxy-nav-bar__inner .lxy-nav-bar__left .lxy-nav-bar__btn_goback { background-image: url("data:image/svg xml;charset=utf8,

"); } .lxy-nav-bar__inner .lxy-nav-bar__left .lxy-nav-bar__btn_goback.white { background-image: url("data:image/svg xml;charset=utf8,

"); } .lxy-nav-bar__inner .lxy-nav-bar__left .lxy-nav-bar__btn_gohome { background-image: url("data:image/svg xml,

"); /* prettier-ignore */ background-size: 17PX 34PX; margin-top: 10px; } .lxy-nav-bar__inner .lxy-nav-bar__left .lxy-nav-bar__btn_gohome.white { background-image: url("data:image/svg xml,

"); /* prettier-ignore */ background-size: 17PX 34PX; margin-top: 10px; } .lxy-nav-bar-search__icon { /* prettier-ignore */ width: 22PX; /* prettier-ignore */ height: 22PX; display: flex; align-items: center; justify-content: center; background-image: url("data:image/svg xml,

"); background-repeat: no-repeat; background-size: cover; } input{ padding:0 20px !important; text-align: start !important; }

属性:

taro 滚到页面顶部(Taro小程序自定义顶部导航栏)(3)

slot:

taro 滚到页面顶部(Taro小程序自定义顶部导航栏)(4)

taro 滚到页面顶部(Taro小程序自定义顶部导航栏)(5)

,

免责声明:本文仅代表文章作者的个人观点,与本站无关。其原创性、真实性以及文中陈述文字和内容未经本站证实,对本文以及其中全部或者部分内容文字的真实性、完整性和原创性本站不作任何保证或承诺,请读者仅作参考,并自行核实相关内容。文章投诉邮箱:anhduc.ph@yahoo.com

    分享
    投诉
    首页