微信小程序自制动态导航栏

写在前面

关于微信小程序导航栏的问题以及解决办法我已经在先前的文章中有提到,点击下面的链接即可跳转~
🤏微信小程序自定义的导航栏🤏

在这篇文章中我们需要做一个这样的导航栏!先上效果图
👇👇👇👇👇👇👇👇👇👇👇👇👇👇👇👇👇👇👇👇👇👇👇👇
请添加图片描述
这个导航栏是codepen上的大神写的,但是它是用前端三件套(即html\css\js)来完成的,在微信小程序原生语法中有很多地方是不支持一些特性的,比如它里面的js核心用到了gsap动画库,而微信小程序是不支持的! 除此之外还有html与wxml、css与wxss转换的问题。总之假如直接复制粘贴是完全行不通的!
(https://codepen.io/v_Bauer/pen/WNroMOq)


最终效果展示

请添加图片描述


全部代码

在这里将会分为两个部分,即codepen上的原版和微信小程序版本
注:微信小程序的引用了外部组件库Vant中的ICON、以及自制的LOADING组件()

codepen

❤️HTML❤️

<html>

<head>
  <meta charset="utf-8">
  <title></title>
  <meta name="viewport" content="width=device-width, initial-scale=1">
</head>
<body>
<div id="navbarContainer">
  <div id="navbar">
    <div id="bubbleWrapper">
      <div id="bubble1" class="bubble"><span class="icon"><i class="fas fa-home"></i></span></div>
      <div id="bubble2" class="bubble"><span class="icon"><i class="fab fa-twitter"></i></span></div>
      <div id="bubble3" class="bubble"><span class="icon"><i class="fas fa-bell"></i></span></div>
      <div id="bubble4" class="bubble"><span class="icon"><i class="fas fa-user"></i></span></div>
    </div>
    <div id="menuWrapper">
      <div id="menu1" class="menuElement" onclick="move('1', '50px', '#ffcc80')"><i class="fas fa-home"></i></div>
      <div id="menu2" class="menuElement" onclick="move('2', '150px', '#81d4fa')"><i class="fab fa-twitter"></i></div>
      <div id="menu3" class="menuElement" onclick="move('3', '250px', '#c5e1a5')"><i class="fas fa-bell"></i></div>
      <div id="menu4" class="menuElement" onclick="move('4', '350px', '#ce93d8')"><i class="fas fa-user"></i></div>
    </div>
  </div>
  <div id="bgWrapper">
    <div id="bg"></div>
    <div id="bgBubble"></div>
  </div>
</div>

<!--   <svg width="0" height="0" >
    <defs>
      <filter id="goo">
        <feGaussianBlur in="SourceGraphic" stdDeviation="20" result="blur" id="blurFilter"/>
        <feColorMatrix in="blur" mode="matrix" values="1 0 0 0 0  0 1 0 0 0  0 0 1 0 0  0 0 0 30 -15" result="goo" />
        <feComposite in="SourceGraphic" in2="goo" operator="atop"/>
      </filter>
    </defs>
  </svg> -->

</body>

</html>

❤️CSS💕


body {
  background: #37474f;
  width: 100vw;
  height: 100vh;
  display: flex;
  justify-content: center;
  align-items: center;
  margin: 0;
  overflow: hidden;
}

#navbarContainer{
  width: 400px;
  min-width: 400px;
  height: 70vh;
  background-color: #ffcc80;
  border-radius: 20px;
  display: flex;
  justify-content: flex-end;
  flex-direction: column;
  overflow: hidden;
  position: relative;
  box-shadow: 0 10px 20px rgba(0,0,0,0.19), 0 6px 6px rgba(0,0,0,0.23);
}

#navbar{
  width: 100%;
  height: 60px;
  background-color: #fff;
  position: absolute;
}

#bubbleWrapper{
  position: absolute;
  display: flex;
  justify-content: space-around;
  width: 100%;
  bottom: 25px;
}

.bubble{
  background-color: #fff;
  width: 50px;
  height: 50px;
  bottom: 85px;
  border-radius: 50%;
  z-index: 1;
  transform: translateY(120%);
  display: flex;
  justify-content: center;
  align-items: center;
}
.icon{
  opacity: 0;
}

#bubble1{
  transform: translateY(0%);
  box-shadow: 0 1px 3px rgba(0,0,0,0.12), 0 1px 2px rgba(0,0,0,0.24);
  > span{
    opacity: 0.7;
  }
}

#bgWrapper{
  filter: url(#goo);
  width: 100%;
  height: 100px;
  position: absolute;
  bottom: 60px;
}
#bg{
  background-color: #ffcc80;
  width: 120%;
  height: 100%;
  margin-left: -10%;
}
#bgBubble{
  position: absolute;
  background-color: #ffcc80;
  width: 70px;
  height: 70px;
  border-radius: 50%;
  bottom: -50px;
  left: 50px;
  transform: translateX(-50%);
}

#menuWrapper{
  position: absolute;
  width: 100%;
  display: flex;
  justify-content: space-around;
}

.menuElement{
  opacity: 0.4;
  transform: translateY(100%);
  cursor: pointer;
  &:hover{
    opacity: 0.5;
  }
}

#contentWrapper{
  position: absolute;
  top: 50%;
  width: 100%;
  transform: translateY(-50%);
  display: flex;
  justify-content: center;
  align-items: center;
  h2{
    color: #fff;
    font-family: sans-serif;
    font-weight: 400;
  }
}
.content{
  display: none;
  opacity: 0;
}

💕JS💕

 function move(id, position, color) {
    var tl = gsap.timeline();
    tl.to("#bgBubble", {duration: 0.15, bottom: "-30px", ease: "ease-out"}, 0)
      .to("#bubble1", {duration: 0.1, y: "120%", boxShadow: 'none', ease: "ease-out",}, 0)
      .to("#bubble2", {duration: 0.1, y: "120%", boxShadow: 'none', ease: "ease-out",}, 0)
      .to("#bubble3", {duration: 0.1, y: "120%", boxShadow: 'none', ease: "ease-out",}, 0)
      .to("#bubble4", {duration: 0.1, y: "120%", boxShadow: 'none', ease: "ease-out",}, 0)
      .to(".icon", {duration: 0.05, opacity: 0, ease: "ease-out",}, 0)
      .to("#bgBubble", {duration: 0.2, left: position, ease: "ease-in-out"}, 0.1)
      .to("#bgBubble", {duration: 0.15, bottom: "-50px", ease: "ease-out"}, '-=0.2')
      .to(`#bubble${id}`, {duration: 0.15, y: "0%", opacity: 1, boxShadow: '0 1px 3px rgba(0,0,0,0.12), 0 1px 2px rgba(0,0,0,0.24)', ease: "ease-out"}, '-=0.1')
      .to(`#bubble${id}> span`, {duration: 0.15, y: "0%", opacity: 0.7, ease: "ease-out"}, '-=0.1')
      .to("#navbarContainer", {duration: 0.3, backgroundColor: color, ease: "ease-in-out"}, 0)
      .to("#bg", {duration: 0.3, backgroundColor: color, ease: "ease-in-out"}, 0)
      .to("#bgBubble", {duration: 0.3, backgroundColor: color, ease: "ease-in-out"}, 0)
  }

wx_miniprograme

❤️WXML❤️

<!-- index.wxml -->
<navigation-bar title="侨韵潮绘" back="{{false}}" color="black" background="#FFF" class="nav"></navigation-bar>
<my-loading showLoading="{{isLoading}}" class="loading"></my-loading>
<!-- 导航栏 -->
<view id="navbarContainer" animation="{{navbarContainerAnimation}}">
  <view id="navbar">
    <view id="bubbleWrapper">
      <view id="bubble1" class="bubble" animation="{{bubble1Animation}}">
        <span class="icon" animation="{{icon1Animation}}">
          <van-icon name="location-o" size="25px" />
        </span>
      </view>
      <view id="bubble2" class="bubble" animation="{{bubble2Animation}}">
        <span class="icon" animation="{{icon2Animation}}">
          <van-icon name="contact-o" size="25px" />
        </span>
      </view>
      <view id="bubble3" class="bubble" animation="{{bubble3Animation}}">
        <span class="icon" animation="{{icon3Animation}}">
          <van-icon name="link-o" size="25px" />
        </span>
      </view>
      <view id="bubble4" class="bubble" animation="{{bubble4Animation}}">
        <span class="icon" animation="{{icon4Animation}}">
          <van-icon name="list-switch" size="25px" />
        </span>
      </view>
    </view>
    <view id="menuWrapper">
      <view id="menu1" class="menuElement" bindtap="move" data-id="1" data-position="95rpx" data-color="#ffcc80">
        <van-icon name="location-o" size="20px" animation="{smallIcon1Animation}" />
      </view>
      <view id="menu2" class="menuElement" bindtap="move" data-id="2" data-position="280rpx" data-color="#81d4fa">
        <van-icon name="contact-o" size="20px" animation="{smallIcon2Animation}" />
      </view>
      <view id="menu3" class="menuElement" bindtap="move" data-id="3" data-position="467rpx" data-color="#c5e1a5">
        <van-icon name="link-o" size="20px" animation="{smallIcon3Animation}" />
      </view>
      <view id="menu4" class="menuElement" bindtap="move" data-id="4" data-position="655rpx" data-color="#ce93d8">
        <van-icon name="list-switch" size="20px" animation="{smallIcon4Animation}" />
      </view>
    </view>
  </view>
  <view id="bgWrapper">
    <view id="bg" animation="{{bgAnimation}}"></view>
    <view id="bgBubble" animation="{{bgBubbleAnimation}}"></view>
  </view>
</view>                 

❤️WXSS💕

/**index.wxss**/
page {
  height: 100vh;
  display: flex;
  flex-direction: column;
}

.loading {
  position: absolute;
  z-index: 999;
}

/* NAV-BAR样式START */
.nav {
  z-index: 2;
}

/* NAV-BAR样式END */


/* 导航栏的样式  START*/
#navbarContainer {
  width: 100%;
  height: 90%;
  margin-bottom: 5rpx;
  background-color: #ffcc80;
  border-radius: 40rpx;
  display: flex;
  justify-content: flex-end;
  flex-direction: column;
  overflow: hidden;
  position: relative;
  box-shadow: 0 20rpx 20rpx rgba(0, 0, 0, 0.19), 0 12rpx 12rpx rgba(0, 0, 0, 0.23);
}


#navbar {
  width: 100%;
  height: 120rpx;
  background-color: #fff;
  position: absolute;
}

#bubbleWrapper {
  position: absolute;
  display: flex;
  justify-content: space-around;
  width: 100%;
  bottom: 50rpx;
}

.bubble {
  background-color: #fff;
  width: 100rpx;
  height: 100rpx;
  border-radius: 50%;
  z-index: 1;
  transform: translateY(120%);
  display: flex;
  justify-content: center;
  align-items: center;
}

.icon {
  opacity: 0;
}

#bubble1 {
  transform: translateY(0%);
  box-shadow: 0 1px 3px rgba(0, 0, 0, 0.12), 0 1px 2px rgba(0, 0, 0, 0.24);
}

#bubble1 span {
  opacity: 0.7;
}

#bgWrapper {
  filter: blur(3rpx);
  width: 100%;
  height: 200rpx;
  position: absolute;
  bottom: 120rpx;
}

#bg {
  background-color: #ffcc80;
  width: 120%;
  height: 100%;
  margin-left: -10%;
}

#bgBubble {
  position: absolute;
  background-color: #ffcc80;
  width: 140rpx;
  height: 140rpx;
  border-radius: 50%;
  bottom: -100rpx;
  left: 95rpx;
  transform: translateX(-50%);
}

#menuWrapper {
  position: absolute;
  width: 100%;
  display: flex;
  justify-content: space-around;
}

.menuElement {
  opacity: 0.4;
  transform: translateY(100%);
  cursor: pointer;
}


#contentWrapper {
  position: absolute;
  top: 50%;
  width: 100%;
  transform: translateY(-50%);
  display: flex;
  justify-content: center;
  align-items: center;
}

/* 导航栏的样式END */

💕JS💕

// pages/index/index.js
Page({
    data: {
        checked: false,
        isLoading: false,
        bgBubbleAnimation: {},
        index: 1,
    },
    move: function (event) {
        // 接受点击事件的参数
        var id = event.currentTarget.dataset.id;
        var position = event.currentTarget.dataset.position;
        var color = event.currentTarget.dataset.color;
        let that = this;


        // 创建背景泡泡动画-第一步
        var bgBubbleAnimation = wx.createAnimation({
            duration: 150,
            timingFunction: 'ease-out'
        });
        bgBubbleAnimation.bottom('-60rpx').step();

        // 创建背景泡泡动画-第二步
        var bgBubbleAnimation_second_step = wx.createAnimation({
            duration: 400,
            timingFunction: 'ease-in-out'
        });
        bgBubbleAnimation_second_step.left(position).step();

        // 创建背景泡泡动画-第三步
        var bgBubbleAnimation_third_step = wx.createAnimation({
            duration: 450,
            timingFunction: 'ease-out'
        });
        bgBubbleAnimation_third_step.bottom('-100rpx').step();

        // 连续执行动画
        var promise = new Promise((resolve, reject) => {
            this.setData({
                bgBubbleAnimation: bgBubbleAnimation.export(),
                // isLoading: true
            });
            setTimeout(resolve, 50); // 等待第一步动画执行完毕
        });

        var bubbleAnimations = [];
        var iconAnimations = [];

        promise.then(() => {
            return new Promise((resolve, reject) => {
                // 创建气泡和图标动画

                for (var i = 1; i <= 4; i++) {
                    var bubbleAnimation = wx.createAnimation({
                        duration: 100,
                        timingFunction: 'ease-out'
                    });
                    bubbleAnimation.translateY('120%').step();
                    bubbleAnimations.push(`bubble${i}Animation`);
                    that.setData({ [`bubble${i}Animation`]: bubbleAnimation.export() });

                    var iconAnimation = wx.createAnimation({
                        duration: 50,
                        timingFunction: 'ease-out'
                    });
                    iconAnimation.opacity(0).step();
                    iconAnimations.push(`icon${i}Animation`);
                    that.setData({ [`icon${i}Animation`]: iconAnimation.export() });
                }
                this.setData({
                    bgBubbleAnimation: bgBubbleAnimation_second_step.export(),
                });
                setTimeout(resolve, 100); // 等待第一步动画执行完毕

            });
        }).then(() => {
            this.setData({
                bgBubbleAnimation: bgBubbleAnimation_third_step.export()
            });

            var clickBubbleAnimation = wx.createAnimation({
                duration: 1000,
                timingFunction: 'ease-out'
            });
            clickBubbleAnimation.translateY('0%').opacity(1).step();

            var clickBubbleSpanAnimation = wx.createAnimation({
                duration: 1000,
                timingFunction: 'ease-out'
            });
            clickBubbleSpanAnimation.opacity(0.7).step();
            that.setData({
                [bubbleAnimations[id - 1]]: clickBubbleAnimation.export(),
                [iconAnimations[id - 1]]: clickBubbleSpanAnimation.export()
            });
            // 更新导航栏和背景颜色动画
            var navbarContainerAnimation = wx.createAnimation({
                duration: 300,
                timingFunction: 'ease-out'
            });
            navbarContainerAnimation.backgroundColor(color).step();

            var bgAnimation = wx.createAnimation({
                duration: 300,
                timingFunction: 'ease-out'
            });
            bgAnimation.backgroundColor(color).step();

            var bgBubbleAnimation_final = wx.createAnimation({
                duration: 300,
                timingFunction: 'ease-out'
            });
            bgBubbleAnimation_final.backgroundColor(color).step();

            this.setData({
                navbarContainerAnimation: navbarContainerAnimation.export(),
                bgAnimation: bgAnimation.export(),
                bgBubbleAnimation: bgBubbleAnimation_final.export(),
            });
        }).catch((err) => {
            console.log(err);
        });



    }

})

结束语

如果有疑问欢迎大家留言讨论,你如果觉得这篇文章对你有帮助可以给我一个免费的赞吗?我们之间的交流是我最大的动力!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/410924.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

HTTP/HTTPS协议

什么是HTTP协议 HTTP被称为超文本传输协议(里面不仅仅可以是字符串,还可以是图片,特殊字符等),这是一种应用非常广泛的应用层协议. HTTP协议诞生于1991年,现在是最主流使用的一种应用层协议.它从诞生到现在为止迭代了多个版本. 但目前最主流使用的还是HTTP1.1和HTTP2.0. HTTP协…

大学餐厅菜品推荐和点评系统设计与实现

**&#x1f345;点赞收藏关注 → 私信领取本源代码、数据库&#x1f345; 本人在Java毕业设计领域有多年的经验&#xff0c;陆续会更新更多优质的Java实战项目希望你能有所收获&#xff0c;少走一些弯路。&#x1f345;关注我不迷路&#x1f345;**一 、设计说明 1.1 研究背景…

Opencv(2)深浅拷贝与基本绘图(c++python

Opencv(2)深浅拷贝与基本绘图 文章目录 Opencv(2)深浅拷贝与基本绘图三、深浅拷贝四、HSV色域(1).意义(2).cvtColor()(3).inRange()(4).适应光线 三、深浅拷贝 浅拷贝是指当图像之间进行赋值时&#xff0c;图像数据并未发生复制&#xff0c;而是两个对象都指向同一块内存块。 …

Amazon Generative AI | 基于 Amazon 扩散模型原理的代码实践之采样篇

以前通过论文介绍 Amazon 生成式 AI 和大语言模型&#xff08;LLMs&#xff09;的主要原理之外&#xff0c;在代码实践环节主要还是局限于是引入预训练模型、在预训练模型基础上做微调、使用 API 等等。很多开发人员觉得还不过瘾&#xff0c;希望内容可以更加深入。因此&#x…

鲲鹏arm64架构下安装KubeSphere

鲲鹏arm64架构下安装KubeSphere 官方参考文档: https://kubesphere.io/zh/docs/quick-start/minimal-kubesphere-on-k8s/ 在Kubernetes基础上最小化安装 KubeSphere 前提条件 官方参考文档: https://kubesphere.io/zh/docs/installing-on-kubernetes/introduction/prerequi…

【教程】ONLYOFFICE 桌面应用程序 v8.0 版本更新全面解读

文章目录 &#x1f4df; 引言 官网&#xff1a;ONLYOFFICE 官方网站 近日&#xff0c;ONLYOFFICE 桌面应用程序迎来了 v8.0 版本的更新&#xff0c;为用户带来了许多新功能和改进&#xff0c;如全新的 RTL 界面、本地主题、Moodle 集成等。本文将详细介绍 ONLYOFFICE 桌面编辑…

微服务架构中的 隔离和超时控制

文章目录 隔离的应用场景隔离的措施机房隔离实例隔离分组隔离连接池隔离和线程池隔离第三方依赖隔离慢任务隔离 隔离的缺点更多思考 超时控制超时控制目标超时控制形态确定超时时间1、根据用户体验来确定2、根据响应时间来确定3、压力测试4、根据代码计算 超时中断业务更多思考…

QT信号槽实现分析

1.宏定义 qt中引入了MOC来反射&#xff0c;编译阶段变成 MOC–>预处理–>编译–>汇编–>链接 1-1、Q_OBJECT 这个宏定义了一系列代码&#xff0c;包括元对象和处理的函数 #define Q_OBJECT \public: \QT_WARNING_PUSH \Q_OBJECT_NO_OVERRIDE_WARNING \static c…

Python接口自动化测试 —— unittest批量用例管理!

我们日常项目中的接口测试案例肯定不止一个&#xff0c;当案例越来越多时我们如何管理这些批量案例&#xff1f;如何保证案例不重复&#xff1f;如果案例非常多&#xff08;成百上千&#xff0c;甚至更多&#xff09;时如何保证案例执行的效率&#xff1f;如何做&#xff08;批…

git 拉取远程分支到本地

背景&#xff1a; 我的 github 上的远程仓库上除了 main 分支外还提交了好几个别的分支&#xff0c;现在我换机器了&#xff0c;git clone 原仓库后只剩 main 分支&#xff0c;我要把其他分支拉下来到本地。 1. 查看所有远程remote分支 git branch -r 比如我这里&#xff1…

回归测试详解

&#x1f345; 视频学习&#xff1a;文末有免费的配套视频可观看 &#x1f345; 关注公众号&#xff1a;互联网杂货铺&#xff0c;回复1 &#xff0c;免费获取软件测试全套资料&#xff0c;资料在手&#xff0c;涨薪更快 什么是回归测试 回归测试&#xff08;Regression testi…

openGauss学习笔记-229 openGauss性能调优-系统调优-配置Ustore

文章目录 openGauss学习笔记-229 openGauss性能调优-系统调优-配置Ustore229.1 设计原理229.2 核心优势229.3 使用指导 openGauss学习笔记-229 openGauss性能调优-系统调优-配置Ustore Ustore存储引擎&#xff0c;又名In-place Update存储引擎&#xff08;原地更新&#xff09…

【笔记】【电子科大 离散数学】 2.命题

文章目录 数理逻辑定义 命题定义不是命题的例子 原子命题和复合命题定义约定 命题联结词否定联结词定义例子真值表 合取联结词定义例子真值表 析取联结词定义例子 蕴含联结词定义例子真值表 等价联结词定义例子真值表 命题符号化及其应用速查表格优先级复合命题符号化布尔检索演…

每日一题——LeetCode1512.好数对的数目

方法一 暴力循环 var numIdenticalPairs function(nums) {let ans 0;for (let i 0; i < nums.length; i) {for (let j i 1; j < nums.length; j) {if (nums[i] nums[j]) {ans;}}}return ans; }; 消耗时间和内存情况&#xff1a; 方法二&#xff1a;组合计数 var …

Redis高并发分布锁实战

Redis高并发分布锁实战 问题场景 场景一: 没有捕获异常 // 仅仅加锁 // 读取 stock15 Boolean ret stringRedisTemplate.opsForValue().setIfAbsent("lock_key", "1"); // jedis.setnx(k,v) // TODO 业务代码 stock-- stringRedisTemplate.delete(&quo…

2024年湖北省事业单位考试报名流程图解

⏰ 时间安排 ✔️ 注册&#xff1a;2024年2月19日至2月27日15:00 ✔️ 报名&#xff1a;2024年2月21日9:00至2月27日17:00 ✔️ 资格审查&#xff1a;2024年2月21日9:00至2月28日9:00 ✔️ 缴费确认&#xff1a;2024年2月28日9:00至3月1日24:00 ✔️ 岗位调整和改报&#…

什么是媒体发稿?发稿媒体分类及发稿流程

传媒如春雨&#xff0c;润物细无声&#xff0c;大家好&#xff0c;我是51媒体网胡老师。 媒体发稿是一种企业推广和宣传的手段&#xff0c;通过媒体渠道传递企业信息和形象。 媒体发稿的含义在于&#xff0c;当企业有新闻、事件或其他消息需要对外公布时&#xff0c;可以选择…

2024.2.25 -ElasticSearch 进阶

倒排索引 Elasticsearch的倒排索引机制是通过将文档中出现的词汇与它们所在的文档ID关联起来&#xff0c;实现快速查找包含特定词汇的文档。下面是一个具体的例子来说明倒排索引的工作原理&#xff1a; 假设我们有一个简单的文章集合&#xff0c;包含以下三篇文章&#xff1a…

【PyTorch][chapter 18][李宏毅深度学习]【无监督学习][ VAE]

前言: VAE——Variational Auto-Encoder&#xff0c;变分自编码器&#xff0c;是由 Kingma 等人于 2014 年提出的基于变分贝叶斯&#xff08;Variational Bayes&#xff0c;VB&#xff09;推断的生成式网络结构。与传统的自编码器通过数值的方式描述潜在空间不同&#xff0c;它…

通天星CMSV6 车载视频监控平台信息泄露漏洞

免责声明&#xff1a;文章来源互联网收集整理&#xff0c;请勿利用文章内的相关技术从事非法测试&#xff0c;由于传播、利用此文所提供的信息或者工具而造成的任何直接或者间接的后果及损失&#xff0c;均由使用者本人负责&#xff0c;所产生的一切不良后果与文章作者无关。该…