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

news2025/1/16 21:13:08

写在前面

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

在这篇文章中我们需要做一个这样的导航栏!先上效果图
👇👇👇👇👇👇👇👇👇👇👇👇👇👇👇👇👇👇👇👇👇👇👇👇
请添加图片描述
这个导航栏是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);
        });



    }

})

结束语

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

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

如何将实景三维倾斜模型叠加到三维地球上?

​ 通过以下方法可以将实景三维倾斜模型叠加到三维地球上。 方法/步骤 下载三维地图浏览器 http://www.geosaas.com/download/map3dbrowser.exe&#xff0c;安装完成后桌面上出现”三维地图浏览器“图标。 2、双击桌面图标打开”三维地图浏览器“ 3、点击“倾斜模型”…

物麒平台根据入耳出耳状态使能或禁止触摸按键实现方法

是否需要申请加入数字音频系统研究开发交流答疑群(课题组)?可加我微信hezkz17, 本群提供音频技术答疑服务,+群赠送语音信号处理降噪算法,蓝牙耳机音频,DSP音频项目核心开发资 料, 1 消息发送 2 消息处理 3 宏开关 4 代码 #include "app_main.h" #include &q…

Vue.js+SpringBoot开发生活废品回收系统

目录 一、摘要1.1 项目介绍1.2 项目录屏 二、研究内容三、界面展示3.1 登录注册3.2 资源类型&资源品类模块3.3 回收机构模块3.4 资源求购/出售/交易单模块3.5 客服咨询模块 四、免责说明 一、摘要 1.1 项目介绍 生活废品回收系统是可持续发展的解决方案&#xff0c;旨在鼓…

FreeRTOS学习第8篇--同步和互斥操作引子

目录 FreeRTOS学习第8篇--同步和互斥操作引子同步和互斥概念实现同步和互斥的机制PrintTask_Task任务相关代码片段CalcTask_Task任务相关代码片段实验现象本文中使用的测试工程 FreeRTOS学习第8篇–同步和互斥操作引子 本文目标&#xff1a;学习与使用FreeRTOS中的同步和互斥操…

【vue3语法】开发使用创建项目等

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 文章目录 前言一、vue3创建vue3v2函数式、v3组合式api响应式方法ref、reactive计算属性conputed监听属性wacthvue3 选项式生命周期父子通信父传子defineProps编译宏 子传父de…

linux 0.11 调试c代码

我们可以通过实验楼实验环境 来调试linux0.11的c代码。 cd ~/oslab/ tar -zxvf hit-oslab-linux-20110823.tar.gz -C ~ cd ~/oslab/linux-0.11/ make cd ~/oslab/ nohup ./dbg-c & nohup terminator & ls在新的窗口执行 ./rungdb,进入调试状态。 输入 set disassemb…

三 . 运算符和流程控制——Java基础篇

三 . 运算符和流程控制 1. 运算符 1 . 赋值运算符 1.2.1 基本语法 符号&#xff1a; 当“”两侧数据类型不一致时&#xff0c;可以使用自动类型转换或使用强制类型转换原则进行处理。支持连续赋值。 扩展赋值运算符&#xff1a; 、 -、*、 /、% 赋值运算符符号解释将符号左…

三、矩阵基本知识

目录 1. 理解 2、矩阵的直观感受 3、矩阵与向量 1. 理解 ①矩阵是一个向量组&#xff0c;由许多 行向量 和 纵向量 组成。 ②矩阵方程求解 用增广矩阵初等变换化为 E 。齐次/非齐次方程组 的解用 初等变化 化为 行最简阶梯型。 ③初步认为由多元一次方程组的系数组成&…

谈谈Redis的哈希槽和一致性哈希

一 . 前言 在 Redis 集群里面主要涉及到两种 Hash 算法 : 一种是一致性哈希 , 这种算法在 适用dis Cluster方案中并没有实现,主要在外部的代理模式 (Twemproxy)一种是 Slot 哈希槽算法 ,这种算法就是 Cluster 的核心算法所以谈到这个问题的时候,不能只讲一部分。在 Red…

SpringMVC 学习(六)之视图

目录 1 SpringMVC 视图介绍 2 JSP 视图 3 Thymeleaf 视图 4 FreeMarker 视图 5 XSLT 视图 6 请求转发与重定向 6.1 请求转发 (Forward) 6.2 重定向 (Redirect) 7 视图控制器 (view-controller) 1 SpringMVC 视图介绍 在 SpringMVC 框架中&#xff0c;视图可以是一个 J…

电机控制常见的外围器件

小型断路器&#xff1a; 这些通通都叫小型断路器&#xff0c;二十年的老电工不一定都认识&#xff0c;不信看看_哔哩哔哩_bilibili 1PIN 2PIN 3PIN 4PIN: 正常情况下火线和零线的电流是相等的&#xff0c;但是漏电的情况下&#xff0c;两线的电流差值大于30毫安&#xff0c;漏…

成功解决ModuleNotFoundError: No module named ‘tensorboard‘

成功解决ModuleNotFoundError: No module named ‘tensorboard’ &#x1f4c5;2024年02月25日 &#x1f308; 个人主页&#xff1a;高斯小哥 &#x1f525; 高质量专栏&#xff1a;Matplotlib之旅&#xff1a;零基础精通数据可视化、Python基础【高质量合集】、PyTorch零基础…

深度学习数据集——球类、运动、体育相关数据集

体育运动的动作识别分类等工作&#xff0c;一直是深度学习实践的重要领域&#xff0c;这里涉及到动作捕捉、分类、识别、计数等多种类型&#xff01;今天就给大家分享关于以上内容的深度学习数据集&#xff01;&#xff01; 1、收集22种类型的运动的图像数据集 数据说明&…

使用apt-mirror做一个本地ubuntu离线apt源

1. 安装 apt-mirror sudo apt-get install apt-mirror2. 创建文件夹 mkdir ./apt_mirror_dir3. 修改apt-mirror的配置文件 sudo gedit /etc/apt/mirror.list得到以下文件,重点对两个位置进行修改&#xff1a; ############# config ################## # ## 修改1&#xff…

鸿蒙中的九种布局概述

鸿蒙中的九种布局概述 概述 鸿蒙开发中包含就种布局&#xff0c;分别为线性布局、层叠布局、弹性布局、相对布局、栅格布局、媒体布局、列表、网格、轮播。 线性布局 线性布局通过Row和Column进行构建&#xff0c;是其他布局的基础。其中Row是水平方向排列&#xff0c;Colu…

在Linux服务器上部署一个单机项目

目录 一、jdk安装 二、tomcat安装 三、MySQL安装 四、部署项目 一、jdk安装 1. 上传jdk安装包 jdk-8u151-linux-x64.tar.gz 进入opt目录&#xff0c;将安装包拖进去 2. 解压安装包 这里需要解压到usr/local目录下&#xff0c;在这里我新建一个文件夹保存解压后的文件 [r…

babylonjs入门模

基于babylonjs封装的一些功能和插件 &#xff0c;希望有更多的小伙伴一起玩babylonjs&#xff1b; 欢迎加群&#xff1a;464146715 官方文档 中文文档 最小模版 ​ 代码如下&#xff1a; 在react中使用 import React, { FC, useCallback, useEffect, useRef, useState } f…

抖音视频提取软件怎么用|抖音数据抓取工具

针对用户获取抖音视频的需求&#xff0c;我们开发了一款功能强大的抖音视频提取软件&#xff0c;旨在帮助用户轻松获取他们感兴趣的视频内容。无需逐个复制链接&#xff0c;无需繁琐的下载步骤&#xff0c;我们的软件让抖音视频获取变得简单快捷。 使用我们的软件非常简单&…

使用免费的L53巧解Freenom域名失效问题

进入2月份以来&#xff0c;不少小伙伴纷纷收到Freenom提供的域名失效&#xff0c;状态由正常变成了Pending。 失效后&#xff0c;域名无法使用&#xff0c;免费的午餐没有了&#xff0c;而现在域名的价格也是水涨船高&#xff0c;真是XXX。很多做外贸的小伙伴表示 难 啊&#x…

谁是单身狗?——C语言刷题

创造不易&#xff0c;可以点点赞吗~ 如有错误&#xff0c;欢迎指出~ 单身狗1 题述 在一个整型数组中&#xff0c;只有一个数字出现一次&#xff0c;其他数组都是成对出现的&#xff0c;请找出那个只出现一次的数字。 例如&#xff1a; 数组中有&#xff1a;1 2 3 4 5 1 2 3 4&a…