uni_app 구성 요소는 탭 슬라이딩 전환을 구현하고 기본 선택 vue3+ts+setup을 지원합니다.

효과 스크린샷:

머리말:

최근 작은 프로그램을 개발할 때 탭 컴포넌트를 사용해야 하는데, 탭이 많을 때에는 좌우로 슬라이드하면 됩니다. 플러그인 라이브러리에서 uView UI를 찾았는데 이 라이브러리의 플러그인 크기가 그리 작지 않은데, 미니 프로그램은 패키지 크기를 최대한 줄여야 해서 직접 패키징할 예정입니다.

주요 기능 : 선택한 탭이 자동으로 컴포넌트의 중간 위치로 이동합니다. 기본적으로 선택한 탭을 지원합니다. 스와이프와 연동하여 데이터를 얻을 수 있습니다. 스와이프와 연동하여 사용을 지원합니다.

주요 메소드: 스크롤뷰

scroll-x:true //允许横向滚动

scroll-left //设置横向滚动条位置

scroll-with-animation  //在设置滚动条位置时使用动画过渡

탭이 허용하는 props 매개변수, tablist는 상위 컴포넌트가 전달하는 탭 배열, defaultsSelectIndex는 기본적으로 선택되는 탭 수입니다.

<tabs :tablist='list' :defaultSelectIndex='defaultSelectIndex' @select='change'></tabs>

각 탭의 너비 계산

onMounted(() => {
		const query = uni.createSelectorQuery().in(getCurrentInstance());
		 
		query.selectAll('.tabs-scroll_item').boundingClientRect((data:Array<UniApp.NodeInfo>) => {
			let dataLen = data.length;
			for (let i = 0; i < dataLen; i++) {
				//  scroll-view 子元素组件距离左边栏的距离
				itemList.value[i].left = data[i].left;
				//  scroll-view 子元素组件宽度
				itemList.value[i].width = data[i].width
			}
		}).exec()


	})

선택한 탭을 상위 구성 요소에 노출하기 위해 select에서 사용하는 방법입니다.

const emits=defineEmits<{
		(e : 'select',value : ItabItem) : void
	}>()

슬라이더 숨기기

/* 隐藏滚动条*/
	::v-deep.uni-scroll-view::-webkit-scrollbar {
		display: none
	}

구성 요소 구현 코드 tabs.vue

<template>
	<view class="nav">
		<scroll-view class="tabs" scroll-x="true" scroll-with-animation :scroll-left="scrollLeft">
			<view class="tabs-scroll">
				<view class="tabs-scroll_item" v-for=" (item,index) in itemList" :key="index"
					:class="{'active':selectIndex==index}" @click="chenked(index)">
					{
   
   {item.title}}
				</view>
			</view>
		</scroll-view>
	</view>
</template>

<script setup lang="ts">
	import { getCurrentInstance, onMounted, ref, watch } from "vue";
	const scrollLeft = ref<number>(0)
	const selectIndex = ref<number>(0)
 	const itemList = ref<Array<ItabItem>>()
	interface ItabItem {
		title : string
		width ?: number
		left ?: number
		id : number
	}
	interface Iprops {
		tablist : Array<ItabItem>
		defaultSelectIndex ?: number
	}
    
	const props = withDefaults(defineProps<Iprops>(), {
		tablist: () => [],
		defaultSelectIndex: 6
	})
	const emits=defineEmits<{
		(e : 'select',value : ItabItem) : void
	}>()
	const chenked = (index:number) => {
		selectIndex.value = index
		scrollLeft.value = 0
		for (let i = 0; i < index - 2; i++) {
			scrollLeft.value = scrollLeft.value + itemList.value[i]?.width
		}
		emits('select',props.tablist[index])
	}
	onMounted(() => {
		const query = uni.createSelectorQuery().in(getCurrentInstance());
		 
		query.selectAll('.tabs-scroll_item').boundingClientRect((data:Array<UniApp.NodeInfo>) => {
			let dataLen = data.length;
			for (let i = 0; i < dataLen; i++) {
				//  scroll-view 子元素组件距离左边栏的距离
				itemList.value[i].left = data[i].left;
				//  scroll-view 子元素组件宽度
				itemList.value[i].width = data[i].width
			}
		}).exec()


	})
	watch(() => [props.tablist, props.defaultSelectIndex], () => {
		itemList.value = props.tablist
		selectIndex.value = props.defaultSelectIndex
		if(props.defaultSelectIndex!==0){
			setTimeout(()=>{
				chenked(props.defaultSelectIndex)
			},50)
		}
	}, {
		immediate: true
	})
</script>
<style lang="scss" scoped>
	.nav {

		background-color: rgba(0, 0, 0, .1);
		position: fixed;
		z-index: 99;
		width: 100%;
		align-items: center;
		height: 100rpx;

		.tabs {
			flex: 1;
			overflow: hidden;
			box-sizing: border-box;
			padding-left: 30rpx;
			padding-right: 30rpx;

			.tabs-scroll {
				display: flex;
				align-items: center;
				flex-wrap: nowrap;
				box-sizing: border-box;

				.tabs-scroll_item {
					line-height: 60rpx;
					margin-right: 35rpx;
					flex-shrink: 0;
					padding-bottom: 10px;
					display: flex;
					justify-content: center;
					font-size: 16px;
					padding-top: 10px;
				}
			}
		}
	}


	.active {
		position: relative;
		color: #333;
		font-weight: 800;
	}

	.active::after {
		content: "";
		position: absolute;
		width: 40rpx;
		height: 8rpx;
		border-radius: 8rpx;
		background-color: #333;
		left: 0px;
		right: 0px;
		bottom: 0px;
		margin: auto;
	}

	/* 隐藏滚动条,但依旧具备可以滚动的功能 */
	::v-deep.uni-scroll-view::-webkit-scrollbar {
		display: none
	}
</style>

사용법 index.vue

<template>
	<view>
		<text style="text-align: center;width: 100%;display: block;"> 标题</text>
		<tabs :tablist='list' :defaultSelectIndex='defaultSelectIndex' @select='change'></tabs>
	</view>
	<button @click="add" style="margin-top: 100rpx;">父组件调用切换</button>
	
</template>

<script lang="ts" setup>
	import { ref } from "vue";
	const defaultSelectIndex=ref(0)
	const list = ref([
		
		{
			id: 1,
			title: '直播直播',
		},
		{
			id: 2,
			title: '热门推荐',
		},
		{
			id: 3,
			title: '音乐',
		},
		{
			id: 4,
			title: '经典小说',
		},
		{
			id: 5,
			title: '看书',
		},
		{
			id: 6,
			title: '短剧',
		},
		{
			id: 7,
			title: '相声评述',
		},
		{
			id: 8,
			title: '找书广场',
		},
	])
    const change=(value)=>{
		uni.showToast({
			title: value.title,
			duration: 2000
		});

	}
	const add=( )=>{
		defaultSelectIndex.value++
	}
</script>

<style>

</style>

이 글이 여러분에게 도움이 되길 바라며, 필요에 따라 캡슐화도 했습니다. 필요에 따라 코드를 최적화할 수도 있습니다.

추천

출처blog.csdn.net/u012941592/article/details/133153866