uniapp小程序自定义中间凸起样式底部tabbar

我自己写的自定义的tabbar效果图

废话少说咱们直接上代码,一步一步来

第一步:

找到根目录下的 pages.json 文件,在 tabBar 中把 custom 设置为 true,默认值是 false。list 中设置自定义的相关信息,

pagePath: 页面路径;
iconPath: 点击前显示的图片;
selectedIconPath: 点击后显示的图片;
text: 底部导航的名称。

第二步

在根目录下创建 components 文件夹用来存放自定义tabBar组件,小伙伴们可以自定义文件夹和文件的名称。记得命名要规范哦!!!

第三步

咱也不截图了,看了那么多的文章全他妈截图,咱直接上代码方便有需要的小伙伴使用!!!!

javascript 复制代码
<template>
	<view class="tabbar-container flex items-center">
		<!-- isRound是中间凸出样式的标志,用来判断当前子级是否是凸出样式的 -->
		<view :class="!item.isRound ? 'square' : 'is-square'" v-for="(item, index) in tabbarList" :key="index" @click="tabbarChange(index, item)">
			<!-- 不是凸出的子级 -->
			<template v-if="!item.isRound">
				<view class="item-top flex justify-center flex-col">
					<u-image :src="active == index ? item.selectedIconPath : item.iconPath" :width="item.width"
						:height="item.height"></u-image>
				</view>
				<view class="item-bottom" :class="{'active': active === index}">
					<text>{{ item.text }}</text>
				</view>
			</template>
			<!-- 凸出的子级 -->
			<template v-else>
				<view class="flex flex-col justify-center items-center center-round">
					<view class="flex flex-col justify-center items-center">
						<u-image :src="active == index ? item.selectedIconPath : item.iconPath" :width="item.width"
							:height="item.height"></u-image>
						<text :class="{'round-active': active === index}">{{ item.text }}</text>
					</view>
				</view>
			</template>
		</view>
	</view>
</template>
<script>
	export default {
		name: 'Tabbar',
		props: {
			tabbarIndex: {
				type: Number,
				default: 0
			}
		},
		data() {
			return {
				// tabbar列表,配置自定义用到的属性字段
				tabbarList: [{
						pagePath: "/pages/homePage/index",
						iconPath: "/static/images/tabbar/index.png",
						selectedIconPath: "/static/images/tabbar/index-selected.png",
						text: "首页",
						width: '45rpx',
						height: '41rpx',
						isRound: false
					},
					{
						pagePath: "/pages/information/index",
						iconPath: "/static/images/tabbar/data.png",
						selectedIconPath: "/static/images/tabbar/data-selected.png",
						text: "数据",
						width: '44rpx',
						height: '43rpx',
						isRound: false
					},
					{
						pagePath: "/pages/aiRecommend/index",
						iconPath: "/static/images/tabbar/ai-recommend.png",
						selectedIconPath: "/static/images/tabbar/ai-recommend.png",
						text: "智能推荐",
						width: '120rpx',
						height: '120rpx',
						isRound: true
					},
					{
						pagePath: "/pages/vip/index",
						iconPath: "/static/images/tabbar/vip.png",
						selectedIconPath: "/static/images/tabbar/vip-selected.png",
						text: "会员",
						width: '39rpx',
						height: '37rpx',
						isRound: false
					},
					{
						pagePath: "/pages/center/index",
						iconPath: "/static/images/tabbar/center.png",
						selectedIconPath: "/static/images/tabbar/center-selected.png",
						text: "我的",
						width: '40rpx',
						height: '41rpx',
						isRound: false
					},
				],
				active: 0,
				isRound: false,
			}
		},
		mounted() {
			this.active = this.tabbarIndex
		},
		methods: {
			tabbarChange(index, item) {
				this.$emit('updateTabbar', index)
				uni.switchTab({
					url: item.pagePath
				});
			}
		}
	}
</script>

<style lang="scss" scoped>
	.tabbar-container {
		width: 100%;
		height: 132rpx;
		position: fixed;
		bottom: 0;
		.square {
			width: 100%;
			height: 130rpx;
			background: #FFFFFF;
			display: flex;
			flex-direction: column;
			align-items: center;
			justify-content: center;
			.round-active {
				color: #207BDB;
			}
		}
		.square .item-top {
			width: 45rpx;
			height: 45rpx;
		}
		.is-square {
			width: 100%;
			height: 130rpx;
			background: #FFFFFF;
			display: flex;
			flex-direction: column;
			align-items: center;
			justify-content: center;
			.round-active {
				color: #207BDB;
			}
		}
		.center-round {
			width: 203rpx;
			height: 130rpx;
			background: url('');
			background-position: center top;
			background-size: 100% 37rpx;
			background-repeat: no-repeat;
			position: absolute;
			margin-top: -72rpx;

			text {
				font-weight: 500;
				font-size: 20rpx;
				color: #555B66;
				position: absolute;
				top: 117rpx;
			}
		}

		.item-bottom {
			font-weight: 500;
			font-size: 20rpx;
			color: #555B66;
			margin-top: 14rpx;
		}

		.active {
			color: #207BDB;
		}
	}
</style>

注意:小程序背景图无法使用本地图片,要么转成base64,要么把图片存到服务器上,然后调用接口把图片路径返回。推荐第二种方法。

我写的有点繁琐,中间凸起的样式应该为一整块,可是ui只给了顶部的白色图片导致写的代码有点多。不过大体的思路差不多都是这样,小伙伴们也可以参考其他博主写的文章的思路去编写自定义组件。

这是我用到的图片,小伙伴们可以下载下来配合代码进行使用。









第四步

在需要的页面中引用自定义组件

javascript 复制代码
<template>
	<Tabbar :tabbar-index="tabbarIndex" @updateTabbar="updateTabbar"></Tabbar>
</template>
<script>
	import Tabbar from "@/components/tabbar/tabbar.vue"
	components: { Tabbar },
	data() {
		return {
			tabbarIndex: 4,
		}
	},
	methods: {
		updateTabbar(e) {
			this.tabbarIndex = e
		},
	}
</script>

有需求的小伙伴们可以试试,我写的并不是很完美但是大体的思路是这样的,小伙伴们可以根据自己项目的需求进行改动。此文章可作为参考使用,希望能帮到有需求的小伙伴!!!!

如果有用的话就点击收藏起来吧!!!

相关推荐
tekin29 分钟前
Vue.js组件开发
前端·javascript·vue.js·vue·vue组件·vue.js组件开发
大雄野比32 分钟前
Vue3.5常用特性整理
前端·javascript·vue.js
m0_5287238133 分钟前
如何利用i18n实现国际化
前端·javascript·vue.js
说私域41 分钟前
开源2 + 1链动模式AI智能名片S2B2C商城小程序视角下从产品经营到会员经营的转型探究
人工智能·小程序·开源·流量运营
一个处女座的程序猿O(∩_∩)O1 小时前
React+AI 技术栈(2025 版)
前端·人工智能·react.js
恰小面包1 小时前
react使用DatePicker日期选择器
前端·react.js·前端框架
engchina1 小时前
React中key值的正确使用指南:为什么需要它以及如何选择
前端·react.js·前端框架
engchina1 小时前
React组件开发技巧:如何优雅地传递Props?
前端·javascript·react.js
偷光1 小时前
Vue3 对比 React18—不只是技术选择
前端·javascript·react.js
计算机-秋大田2 小时前
基于微信小程序的培训机构客户管理系统设计与实现(LW+源码+讲解)
java·vue.js·spring boot·后端·微信小程序·小程序·课程设计