UniApp中实现类似锚点定位滚动效果

 更新时间:2023年10月21日 15:56:49   作者:江城开朗的豌豆  
一个uniapp小程序的项目,我们需要实现一个非常实用的功能——类似于锚点定位的交互效果,即在首页中有多个tab(分类标签),每个tab对应着不同的模块,当用户点击某个分类的tab时,需要流畅地滚动到对应的内容位置,提供更好的用户体验,

一个uniapp小程序的项目,我们需要实现一个非常实用的功能——类似于锚点定位的交互效果,即在首页中有多个tab(分类标签),每个tab对应着不同的模块。当用户点击某个分类的tab时,需要流畅地滚动到对应的内容位置,提供更好的用户体验。

思路

为了实现这个功能,我们可以分为以下几个步骤:

实时监听滚动并选中对应 tab

点击 tab 跳转至当前模块

功能实现

1.点击 tab 跳转至当前模块

当用户点击某个 tab 时,我们根据 tab 的 currentTab值 ,然后获取到要滚动到的距离

由于 boundingClientRect 获取的 top 值是相对于视口的,所以实际上页面需要滚动的距离为相对于视口的距离加上当前页面的滚动距离,即 res.top + scrollTop

 html代码

<u-tabs :list="quickList" :current='currentTab' @click="whoBtn">
					<view
							slot="right"
							style="padding-left: 4px;"
							@tap="show = true"
					>
						<u-icon
								name="list"
								size="21"
								bold
						></u-icon>
					</view> 
</u-tabs>
 
<!-- 底部弹窗-全部应用 -->
		<u-popup :show="show" :round="10" mode="bottom" closeable @close="close">
			<view>
				<view class="popTit">全部应用</view>
				<view class="popCon">
					<view class="item" v-for="(item,index) in quickList" :key="index" @click="popClick(index)">{{ item.name }}</view>
				</view>
			</view>
		</u-popup>

js代码

methods: {
			close() {
				this.show = false
			},
			topBack(){
				uni.pageScrollTo({
				  scrollTop:0,   // 滚动到页面的目标位置  这个是滚动到顶部, 0 
				   duration:300  // 滚动动画的时长
				})
		   },
		   // 点击底部弹窗喵点定位
		   popClick(index){
			   this.meow(index)
		   },
			whoBtn(part){
				this.meow(part.index)
			},
			// 喵点定位
			meow(index){
				uni.createSelectorQuery().select(".data"+index).boundingClientRect((res)=>{//定位到你要的class的位置
					uni.pageScrollTo({
						scrollTop:res.top,
						duration: 300
					});
				}).exec()
			}
 
		}

2.实时监听滚动并选中对应tab

      在滚动时实时通过 boundingClientRect 获取模块相对于窗口的值,

        通过uiapp提供的页面级别的方法onPageScroll 去计算滚动的高度

js代码

// 监听页面滚动
		onPageScroll (event) {
			if(event){
				// 返回顶部按钮
				const { scrollTop } = event;
				scrollTop > 400 ? this.isShow = true : this.isShow = false
				 //记录当前页面的滚动距离
				this.scrollTop = scrollTop;
				if (this.isScrollByTab) return;
				const query = uni.createSelectorQuery().in(this);
				query.selectAll(".dataBoardBox").boundingClientRect((res)=>{//定位到你要的class的位置
					this.$nextTick(() => {
						const index = findLastIndex(res, (rect) => rect.top < 80);
						if (index > -1) {
							this.currentTab = index;
						}
						
					});
					
				}).exec()
			}
	  
		
		},

通过滚动时实时获取,避免了获取 top 值不准确的问题,之后判断距离当前窗口的距离,小于临界值则选中对应的tab 

全部代码

index.vue

<template>
	<view>
		<!-- 搜索栏 -->
		<view class="searchView">
			<u-search  shape="square" :showAction="false" placeholder="搜索" v-model="keyword"></u-search>
			<view class="addedBox">
				<view class="addedBox-top">
					<view class="text">已添加的应用(4)</view>
				</view>
				<view class="addedBox-bom">
					<view class="item" v-for="(item,index) in 2" :key="index">
						<image src="/static/group-icon.png" mode="widthFix"></image>
						<text>上报</text>
						<image class="delete" src="/static/quick/delete.png" mode="widthFix" @tap="$u.toast(`删除被点击${index+1}`)"></image>
					</view>
				</view>
			</view>
		</view>
		<!-- 全部应用 -->
		<view class="whole">
			<view class="title">全部应用</view>
			<u-sticky bgColor="#fff">
				<u-tabs :list="quickList" :current='currentTab' @click="whoBtn">
					<view
							slot="right"
							style="padding-left: 4px;"
							@tap="show = true"
					>
						<u-icon
								name="list"
								size="21"
								bold
						></u-icon>
					</view> 
				</u-tabs>
			</u-sticky>
			
			<!-- 数据看板 -->
			<view class="dataBoardBox data0">
				<view class="title">数据看板</view>
				<view class="dataBoard-item" v-for="(item,index) in 5" :key="index">
					<image src="/static/group-icon.png" mode="widthFix"></image>
					<view class="right">
						<view class="right-1">统计分析</view>
						<view class="right-2">添加</view>
					</view>
				</view>
			</view>
			<!-- 事件工作 -->
			<view class="dataBoardBox data1">
				<view class="title">事件工作</view>
				<view class="dataBoard-item" v-for="(item,index) in 5" :key="index">
					<image src="/static/group-icon.png" mode="widthFix"></image>
					<view class="right">
						<view class="right-1">工作日志</view>
						<view class="right-2">已添加</view>
					</view>
				</view>
			</view>
			<!-- 代办事项 -->
			<view class="dataBoardBox data2">
				<view class="title">代办事项</view>
				<view class="dataBoard-item" v-for="(item,index) in 5" :key="index">
					<image src="/static/group-icon.png" mode="widthFix"></image>
					<view class="right">
						<view class="right-1">我的代办</view>
						<view class="right-2">已添加</view>
					</view>
				</view>
			</view>
			<!-- 代办事项 -->
			<view class="dataBoardBox data3">
				<view class="title">代办事项2</view>
				<view class="dataBoard-item" v-for="(item,index) in 5" :key="index">
					<image src="/static/group-icon.png" mode="widthFix"></image>
					<view class="right">
						<view class="right-1">我的代办</view>
						<view class="right-2">已添加</view>
					</view>
				</view>
			</view>
		</view>
		<!-- 底部弹窗-全部应用 -->
		<u-popup :show="show" :round="10" mode="bottom" closeable @close="close">
			<view>
				<view class="popTit">全部应用</view>
				<view class="popCon">
					<view class="item" v-for="(item,index) in quickList" :key="index" @click="popClick(index)">{{ item.name }}</view>
				</view>
			</view>
		</u-popup>
		<!-- 回到顶部 -->
		<image v-show="isShow" class="toTop" @click="topBack" src="/static//quick/toTop.png" alt=""></image>
	</view>
</template>
 
<script>
	import { findLastIndex } from 'lodash-es';
	export default {
		data() {
			return {
				isShow:false,
				show:false,
				zanIndex:null,
				//记录滚动的距离,用在切换tab页面滚动逻辑中
				scrollTop: 0,
				currentTab:0,
				 //当前的页面滚动是否由切换tab触发
				isScrollByTab: false,
				keyword: '', // 搜索值
				quickList: [{
					name: '数据看板',
				}, {
					name: '事件工作',
				}, {
					name: '代办事项',
				}, {
					name: '代办事项2',
				}]
			}
		},
		// 监听页面滚动
		onPageScroll (event) {
			if(event){
				// 返回顶部按钮
				const { scrollTop } = event;
				scrollTop > 400 ? this.isShow = true : this.isShow = false
				 //记录当前页面的滚动距离
				this.scrollTop = scrollTop;
				if (this.isScrollByTab) return;
				const query = uni.createSelectorQuery().in(this);
				query.selectAll(".dataBoardBox").boundingClientRect((res)=>{//定位到你要的class的位置
					this.$nextTick(() => {
						const index = findLastIndex(res, (rect) => rect.top < 80);
						if (index > -1) {
							this.currentTab = index;
						}
						
					});
					
				}).exec()
			}
	  
		
		},
		methods: {
			close() {
				this.show = false
			},
			topBack(){
				uni.pageScrollTo({
				  scrollTop:0,   // 滚动到页面的目标位置  这个是滚动到顶部, 0 
				   duration:300  // 滚动动画的时长
				})
		   },
		   // 点击底部弹窗喵点定位
		   popClick(index){
			   this.meow(index)
		   },
			whoBtn(part){
				this.meow(part.index)
			},
			// 喵点定位
			meow(index){
				uni.createSelectorQuery().select(".data"+index).boundingClientRect((res)=>{//定位到你要的class的位置
					uni.pageScrollTo({
						scrollTop:res.top,
						duration: 300
					});
				}).exec()
			}
 
		}
	} 
</script>
 
<style lang="scss">
	@import 'quickPage.scss'
</style>

index.scss 

.searchView{
	width: 750rpx;
	background-color: #ffffff;
	padding: 30rpx 25rpx 0 25rpx;
	box-sizing: border-box;
	.addedBox{
		width: 100%;
		margin-top: 40rpx;
		.addedBox-top{
			width: 100%;
			height: 50rpx;
			.text{
				font-weight: bold;
				font-size: 32rpx;
			}
		}
		.addedBox-bom{
			margin-top: 40rpx;
			width: 100%;
			display: flex;
			flex-wrap: wrap;
			.item{
				display: flex;
				flex-direction: column;
				align-items: center;
				position: relative;
				width: 140rpx;
				margin-bottom: 30rpx;
				image{
					width: 80rpx;
					height: 80rpx;
					border-radius: 20rpx;
				}
				text{
					width: 100%;
					margin-top: 10rpx;
					text-align: center;
					font-size: 28rpx;
					color: #909090;
				}
				.delete{
					position: absolute;
					top: -20rpx;
					right: 8rpx;
					width: 40rpx;
					height: 40rpx;
					z-index: 10;
					background-color: #ffffff;
				}
			}
		}
	}
}
// 全部应用
.whole{
	margin-top: 25rpx;
	width: 750rpx;
	background-color: #F5F5F5;
	.title{
		padding: 30rpx 25rpx 0 25rpx;
		width: 100%;
		background-color: #ffffff;
		box-sizing: border-box;
		font-weight: bold;
		font-size: 32rpx;
	}
	.dataBoardBox{
		background-color: #ffffff;
		padding-bottom: 30rpx;
		.title{
			padding: 30rpx 25rpx 0 25rpx;
			width: 100%;
			background-color: #ffffff;
			box-sizing: border-box;
			font-weight: bold;
			font-size: 32rpx;
		}
		.dataBoard-item{
			padding: 30rpx 25rpx 0 25rpx;
			box-sizing: border-box;
			display: flex;
			image{
				width: 80rpx;
				height: 80rpx;
				border-radius: 20rpx;
			}
			.right{
				margin-left: 30rpx;
				flex: 1;
				border-bottom: 3rpx solid rgba(200, 200, 200, .4);
				display: flex;
				align-items: center;
				justify-content: space-between;
				padding-right: 30rpx;
				box-sizing: border-box;
				.right-1{
					font-size: 28rpx;
				}
				.right-2{
					font-size: 24rpx;
					border: 2rpx solid rgba(200, 200, 200, .4);
					width: 100rpx;
					height: 40rpx;
					line-height: 40rpx;
					text-align: center;
					border-radius: 10rpx;
				}
			}
		}
	}
}
// 底部弹窗-全部应用 
.popTit{
	width: 100%;
	height: 80rpx;
	border-bottom: 2rpx solid rgba(200, 200, 200, .4);
	line-height: 80rpx;
	text-align: center;
	font-weight: bold;
}
.popCon{
	width: 100%;
	padding: 20rpx 30rpx 50rpx 30rpx;
	box-sizing: border-box;
	display: flex;
	flex-wrap: wrap;
	justify-content: space-between;
	.item{
		width: 215rpx;
		height: 60rpx;
		background-color: #f6f6f6;
		margin-bottom: 40rpx;
		text-align: center;
		line-height: 60rpx;
		border-radius: 10rpx;
		font-size: 26rpx;
		border: 2rpx solid rgba(200, 200, 200, .5);
	}
}
// 回到顶部
.toTop{
    position: fixed;
    z-index: 2;
    right: 40rpx;
    bottom: 10vw;
    width: 70rpx;
    height:70rpx;
	background-color: #ffffff;
}

总结

在这篇文章中,我们介绍了如何在uniapp小程序项目中实现点击tab跳转到对应模块并滚动的功能。这个功能对于许多开发者来说非常有用,因为它可以提高用户体验,并使页面导航更加流畅。

到此这篇关于UniApp中实现类似锚点定位滚动效果的文章就介绍到这了,更多相关UniApp中锚点定位滚动内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • vuex actions异步修改状态的实例详解

    vuex actions异步修改状态的实例详解

    今天小编就为大家分享一篇vuex actions异步修改状态的实例详解,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2019-11-11
  • vue react中的excel导入和导出功能

    vue react中的excel导入和导出功能

    当我们把信息化系统给用户使用时,用户经常需要把以前在excel里录入的数据导入的信息化系统里,这样为用户提供了很大的方便,这篇文章主要介绍了vue中或者react中的excel导入和导出,需要的朋友可以参考下
    2023-09-09
  • vue2.0的contextmenu右键弹出菜单的实例代码

    vue2.0的contextmenu右键弹出菜单的实例代码

    本篇文章主要介绍了vue2.0的contextmenu右键弹出菜单的实例代码,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2017-07-07
  • vue自定义tap指令及tap事件的实现

    vue自定义tap指令及tap事件的实现

    Vue提供自定义实现指令的功能, 和组件类似,可以是全局指令和局部指令,这篇文章主要介绍了vue自定义tap指令及tap事件的实现 ,需要的朋友可以参考下
    2018-09-09
  • vue3动态路由刷新出现空白页的原因与最优解

    vue3动态路由刷新出现空白页的原因与最优解

    页面刷新白屏其实是因为vuex引起的,由于刷新页面vuex数据会丢失,这篇文章主要给大家介绍了关于vue3动态路由刷新出现空白页的原因与最优解的相关资料,需要的朋友可以参考下
    2023-11-11
  • Vue拖拽组件开发实例详解

    Vue拖拽组件开发实例详解

    本文重点给大家介绍Vue拖拽组件开发实例,拖拽的原理是手指在移动的过程中,实时改变元素的位置即top和left值,使元素随着手指的移动而移动。对实例代码感兴趣的朋友跟随脚本之家小编一起学习吧
    2018-05-05
  • Vue父子之间值传递的实例教程

    Vue父子之间值传递的实例教程

    这篇文章主要给大家介绍了关于Vue父子之间值传递的相关资料,文中通过示例代码介绍的非常详细,对大家的学习或者使用Vue具有一定的参考学习价值,需要的朋友们下面来一起学习学习吧
    2020-07-07
  • vue如何使用formData传递文件类型的数据

    vue如何使用formData传递文件类型的数据

    这篇文章主要介绍了vue如何使用formData传递文件类型的数据问题,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2023-05-05
  • Vue3中使用defineCustomElement 定义组件详解

    Vue3中使用defineCustomElement 定义组件详解

    这篇文章主要为大家介绍了Vue3中使用defineCustomElement 定义组件详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2022-10-10
  • 如何写一个 Vue3 的自定义指令

    如何写一个 Vue3 的自定义指令

    这篇文章主要介绍了如何写一个 Vue3 的自定义指令,如果我们想在 Vue.js 的项目中实现图片懒加载,那么用自定义指令就再合适不过了,那么接下来就让我手把手带你用 Vue3 去实现一个图片懒加载的自定义指令 v-lazy,需要的朋友可以参考一下
    2022-01-01

最新评论