当前位置: 移动技术网 > IT编程>开发语言>JavaScript > 微信小程序tab切换可滑动切换导航栏跟随滚动实现代码

微信小程序tab切换可滑动切换导航栏跟随滚动实现代码

2019年09月06日  | 移动技术网IT编程  | 我要评论
简介 看到今日头条小程序页面可以滑动切换,而且tab导航条也会跟着滚动,点击tab导航,页面滑动,切导航栏也会跟着滚动,就想着要怎么实现这个功能 像商城类商品类目如果做

简介

看到今日头条小程序页面可以滑动切换,而且tab导航条也会跟着滚动,点击tab导航,页面滑动,切导航栏也会跟着滚动,就想着要怎么实现这个功能

像商城类商品类目如果做成左右滑动切换类目用户体验应该会好很多,我这里只是一个小demo,没有怎么去考虑数据的问题,主要是想着去实现这么个功能,有可能后期引入数据后会出现问题,欢迎提出互相讨论

解决过程

1.在想要实现这个问题的时候找了不少别人的博客看,主体页面布局方面是比较统一的,tab导航栏使用<scroll-view>标签,内容使用<swiper>,其中的使用方法和参数希望自行参考api文档,不过多解释

布局代码如下:

wxml

<view class="container">
 <!-- tab导航栏 -->
 <!-- scroll-left属性可以控制滚动条位置 -->
 <!-- scroll-with-animation滚动添加动画过渡 -->
 <scroll-view scroll-x="true" class="nav" scroll-left="{{navscrollleft}}" scroll-with-animation="{{true}}">
  <block wx:for="{{navdata}}" wx:for-index="idx" wx:for-item="navitem" wx:key="idx">
   <view class="nav-item {{currenttab == idx ?'active':''}}" data-current="{{idx}}" bindtap="switchnav">{{navitem.text}}</view>
  </block>  
 </scroll-view>
 <!-- 页面内容 -->
 <swiper class="tab-box" current="{{currenttab}}" duration="300" bindchange="switchtab">  
  <swiper-item wx:for="{{[0,1,2,3,4,5,6,7,8]}}" wx:for-item="tabitem" wx:for-index="idx" wx:key="idx" class="tab-content">
   {{tabitem}}
  </swiper-item>
 </swiper>
</view>

wxss

/**index.wxss**/
page{
 width: 100%;
 height: 100%;
}
.container{
 width: 100%;
 height: 100%;
}
.nav {
 height: 80rpx;
 width: 100%;
 box-sizing: border-box;
 overflow: hidden;
 line-height: 80rpx;
 background: #f7f7f7;
 font-size: 16px;
 white-space: nowrap;
 position: fixed;
 top: 0;
 left: 0;
 z-index: 99;
}
.nav-item {
 width: 20%;
 display: inline-block;
 text-align: center;
}
.nav-item.active{
 color: red;
}
.tab-box{
 background: greenyellow;
 padding-top: 80rpx;
 height: 100%;
 box-sizing: border-box;
}
.tab-content{
 overflow-y: scroll;
}

js

//index.js
//获取应用实例
const app = getapp()

page({
 data: {
  motto: 'hello world',
  userinfo: {},
  hasuserinfo: false,
  caniuse: wx.caniuse('button.open-type.getuserinfo'),
  navdata:[
   {
    text: '首页'
   },
   {
    text: '健康'
   },
   {
    text: '情感'
   },
   {
    text: '职场'
   },
   {
    text: '育儿'
   },
   {
    text: '纠纷'
   },
   {
    text: '青葱'
   },
   {
    text: '上课'
   },
   {
    text: '下课'
   }
  ],
  currenttab: 0,
  navscrollleft: 0
 },
 //事件处理函数
 onload: function () {
  if (app.globaldata.userinfo) {
   this.setdata({
    userinfo: app.globaldata.userinfo,
    hasuserinfo: true
   })
  } else if (this.data.caniuse) {
   // 由于 getuserinfo 是网络请求,可能会在 page.onload 之后才返回
   // 所以此处加入 callback 以防止这种情况
   app.userinforeadycallback = res => {
    this.setdata({
     userinfo: res.userinfo,
     hasuserinfo: true
    })
   }
  } else {
   // 在没有 open-type=getuserinfo 版本的兼容处理
   wx.getuserinfo({
    success: res => {
     app.globaldata.userinfo = res.userinfo
     this.setdata({
      userinfo: res.userinfo,
      hasuserinfo: true
     })
    }
   })
  }


  wx.getsysteminfo({
   success: (res) => {
    this.setdata({
     pixelratio: res.pixelratio,
     windowheight: res.windowheight,
     windowwidth: res.windowwidth
    })
   },
  })  
 },
 switchnav(event){
  var cur = event.currenttarget.dataset.current; 
  //每个tab选项宽度占1/5
  var singlenavwidth = this.data.windowwidth / 5;
  //tab选项居中       
  this.setdata({
   navscrollleft: (cur - 2) * singlenavwidth
  })  
  if (this.data.currenttab == cur) {
   return false;
  } else {
   this.setdata({
    currenttab: cur
   })
  }
 },
 switchtab(event){
  var cur = event.detail.current;
  var singlenavwidth = this.data.windowwidth / 5;
  this.setdata({
   currenttab: cur,
   navscrollleft: (cur - 2) * singlenavwidth
  });
 }
})

页面代码如上面三部分,可以直接新建一项目进行测试

效果图如下

注意事项

在处理顶部tab导航跟随底部页面滑动的时候遇到一个问题,就是在给<scroll-view>中的scrll-left赋值的时候遇到的问题

逻辑上讲初始时tab导航下标小于2时导航栏不滚动,当大于2时向左滑动,以使被选中的导航栏居中,但是当最左侧的选项因为左滑看不到后,我又点击左侧选项希望导航往右滑动,能够看到左边的导航,按上面的js代码计算scroll-left会产生负值,但是scroll-left即使为负值,但是滚动条不会向左缩进去,所以即使为负值,相当于为0,当时做的时候一直在思考这个怎么用逻辑解决,想着要写各种判断,计算距离,结果到最后一句代码直接赋值就搞定了,也是很无语。

总结

以上所述是小编给大家介绍的微信小程序tab切换可滑动切换导航栏跟随滚动实现代码,希望对大家有所帮助

如您对本文有疑问或者有任何想说的,请 点击进行留言回复,万千网友为您解惑!

相关文章:

验证码:
移动技术网