loading动效实现

news2025/1/16 1:52:20

在站上闲逛发现一个非常有意思的loading效果,跟着大佬仿写了一下Vue版本的。

https://blog.csdn.net/tianjian4592/article/details/44538605

在这里插入图片描述
直接放源码

<script setup>
import {ref, defineProps, watch} from "vue";

const props = defineProps({
    isShow: {
        type: Boolean,
        default: false
    },
    leafNum: {
        type: Number,
        default: 5
    }  // 叶子数量
})

const isLoading = ref(false);
const progressNum = ref(0);
let progress;
let leafIntervalArr;  // 保存leaf的定时器

watch(() => props.isShow, () => {
    if (!props.isShow) {
        // 触发关闭loading的时候延迟1.4s关闭,确保动画完整
        progressNum.value = 100;
        clearInterval(progress);
        setTimeout(() => {
            isLoading.value = false;
            progressNum.value = 0;
            // 结束动画后清空定时器
            for (let i = 0; i < leafIntervalArr.length; i++) {
                clearInterval(leafIntervalArr[i]);
            }
        }, 1400)
    } else {
        isLoading.value = true;
        setInterval(() => {
            if (progressNum.value <= 90) {
                progressNum.value += 1;
            }
        }, 300);
        setTimeout(() => {
            randomLeaf();
        }, 1000)
    }
})

function randomLeaf() {
    leafIntervalArr = new Array(props.leafNum);
    for (let i = 0; i < props.leafNum; i++) {
        let leaf = document.createElement('img');
        leaf.className = 'leaf';
        leaf.src = '/src/assets/images/loading/leaf.png';
        leaf.alt = '叶子';
        console.log(document.querySelector('.loading-progress'));
        document.querySelector('.loading-progress').appendChild(leaf);
        // 随机时间 1~5s
        setTimeout(() => {
            let activeLeaf = document.querySelectorAll('.leaf')[i];
            let x = 260, y = 0;
            activeLeaf.style.left = x + "px"; //初始坐标x
            activeLeaf.style.top = y + "px"; //初始坐标y

            leafIntervalArr[i] = setInterval(function () {
                x = x - 5; //运动速度
                y = Math.floor(Math.sin(x / 260 * 2 * Math.PI) * Math.floor(Math.random() * 10)) + Math.floor(Math.random() * 10); //运动的高度
                if (x < 0) x = 260;
                activeLeaf.style.left = x + "px";
                activeLeaf.style.top = y + "px";
                activeLeaf.style.transform = "rotate(" + Math.floor(Math.random() * 360) + "deg)";
            }, 100);
        }, Math.floor(Math.random() * 5000) + 1);
    }
}
</script>

<template>
    <transition>
        <div class="loading-container" v-if="isLoading">
            <div class="loading-text">loading...</div>
            <div class="loading-wrapper">
                <div class="loading-progress">
                    <div class="loading-progress-content" :style="{width: progressNum + '%'}"></div>
                    <!-- 叶子区域 -->
                </div>
                <div class="loading-fan fan-container">
                    <img class="fan" :class="{'fan-close':progressNum === 100}"
                         src="../../assets/images/loading/fan.png" alt="风扇">
                    <img class="max" v-if="progressNum === 100" src="../../assets/images/loading/100.png"
                         alt="100%">
                </div>
            </div>
        </div>
    </transition>
</template>

<style scoped>
.loading-container {
    width: 100vw;
    height: 100vh;
    background-color: #E3B12FD0;
    display: flex;
    justify-content: center;
    align-content: center;
    align-items: center;
    flex-wrap: wrap;
    position: fixed;
    top: 0;
    right: 0;
}

.loading-text {
    width: 100%;
    text-align: center;
    font-size: 20px;
    font-weight: bold;
    color: rgb(255, 168, 0);
    margin-bottom: 20px;
}

.loading-wrapper {
    width: 300px;
    height: 48px;
    background-color: #EED385D2;
    border-radius: 24px;
    position: relative;
    z-index: 9;
}

.loading-progress {
    width: 260px;
    height: 36px;
    background-color: #EED385D2;
    border-top-left-radius: 16px;
    border-bottom-left-radius: 16px;
    margin: 6px 0 6px 6px;
    overflow: hidden;
    position: absolute;
    top: 0;
    left: 0;
}

.loading-progress-content {
    width: 0;
    height: 36px;
    background-color: rgb(255, 168, 0);
    transition: all 0.5s;
    position: relative;
    z-index: 9;
}

.loading-fan {
    width: 48px;
    height: 48px;
    border-radius: 50%;
    background-color: rgb(251, 205, 81);
    border: 3px solid rgba(255, 255, 255, 0.6);
    box-sizing: border-box;
    position: absolute;
    top: 0;
    right: 0;
    z-index: 9;
}

.fan {
    width: 36px;
    height: 36px;
    position: absolute;
    top: 3px;
    left: 4px;
    animation: spin 2s infinite linear;
}

.fan-close {
    animation: fanClose 1.3s 1 linear;
    animation-fill-mode: forwards;
}

.max {
    width: 36px;
    height: 36px;
    position: absolute;
    top: 3px;
    left: 3px;
    animation: maxFont 1.3s 1 linear;
    animation-fill-mode: forwards;
}

@keyframes spin {
    from {
        transform: rotate(360deg);
    }
    to {
        transform: rotate(0deg);
    }
}

@keyframes fanClose {
    from {
        transform: rotate(360deg) scale(1);
    }
    to {
        transform: rotate(0deg) scale(0);
    }
}

@keyframes maxFont {
    from {
        transform: scale(0);
    }
    to {
        transform: scale(1);
    }
}
</style>
<style>
.leaf {
    width: 15px;
    height: 15px;
    position: absolute;
    top: 0;
    left: 260px;
    z-index: 5;
    transition: all 0.01s;
}
</style>

核心思想:

  1. css动画旋转风扇,根据时间慢慢增加进度条的宽度。
  2. 生成最大数量的叶子,随机事件、随机振幅、随机旋转角度从右向左沿正弦曲线运动

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

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

相关文章

三菱FX3U系列-定位指令

目录 一、简介 二、指令形式 1、相对定位[DRVI、DDRVI] 2、绝对定位[DRVA、DDRVA] 三、总结 一、简介 定位指令用于控制伺服电机或步进电机的位置移动。可以通过改变脉冲频率和脉冲数量来控制电机的移动速度和移动距离&#xff0c;同时还可以指定移动的方向。 二、指令形…

YOLOv5检测界面-PyQt5实现

1.将detect.py运用到界面 要将 YOLOv5 的检测结果与 PyQt 界面结合&#xff0c;你需要进行一些额外的步骤。以下是一个简单的示例代码&#xff0c;展示如何使用 YOLOv5 进行目标检测并在 PyQt 界面中显示结果。 首先&#xff0c;确保你已经安装了必要的库&#xff1a; pip …

C++初阶(九)内存管理

&#x1f4d8;北尘_&#xff1a;个人主页 &#x1f30e;个人专栏:《Linux操作系统》《经典算法试题 》《C》 《数据结构与算法》 ☀️走在路上&#xff0c;不忘来时的初心 文章目录 一、C/C内存分布1、选择题2、填空题3、sizeof 和 strlen 区别&#xff1f;4、总结 二、 C语言…

『Nacos』 入门教程

前言 本文为 Nacos 平台快速入门教程&#xff0c;本文将会使用通俗易懂的语言手把手带您了解、使用 Nacos 平台&#xff0c;适合未接触过 Nacos 的初学者 官方手册&#xff1a;Nacos | Nacos 官方仓库&#xff1a;alibaba/nacos 版本&#xff1a;2.X 本文示例代码仓库&#xf…

Power Automate-时间戳转化为时区时间

点击编辑 在两个步骤之间再插入一个新步骤&#xff0c;添加操作 在内置里点击日期时间 点击转换时区 下面再修改为已转换的时间

「Verilog学习笔记」4位数值比较器电路

专栏前言 本专栏的内容主要是记录本人学习Verilog过程中的一些知识点&#xff0c;刷题网站用的是牛客网 分析 这里要注意题目的“门级描述方式”&#xff0c;所以我们只能使用基本门电路&#xff1a;&,|,!,^,^~。 具体实现思路&#xff1a;通过真值表得出Y0 Y1 Y2的逻辑表达…

【腾讯云 HAI域探秘】探索AI绘画之路:利用腾讯云HAI服务打造智能画家

目录 前言1 使用HAI服务作画的步骤1.1 注册腾讯云账户1.2 创建算力服务器1.3 进入模型管理界面1.4 汉化界面1.5 探索AI绘画 2 模型参数的含义和调整建议2.1 模型参数的含义和示例2.2 模型参数的调整建议 3 调整参数作画的实践和效果3.1 实践说明3.2 实践效果13.3 实践效果23.4 …

电梯用电量-第10届蓝桥杯国赛Python真题精选

[导读]&#xff1a;超平老师的Scratch蓝桥杯真题解读系列在推出之后&#xff0c;受到了广大老师和家长的好评&#xff0c;非常感谢各位的认可和厚爱。作为回馈&#xff0c;超平老师计划推出《Python蓝桥杯真题解析100讲》&#xff0c;这是解读系列的第8讲。 电梯用电量&#x…

闪站侠洗衣洗鞋管理系统app小程序开发;

闪站侠洗护软件系统为您提供全面的洗衣洗鞋解决方案&#xff0c;系统多门店&#xff0c;多网点。为您开通公中号小程序&#xff0c;并与顺丰、天猫、抖音、美团点评等第三方平台紧密连接。 我们解决洗衣工厂/门店的五大问题&#xff1a; 一、效率 从门店收衣到工厂出库&#xf…

解决GitHub不能访问

1&#xff1a;ping github.com,得到可以解析的地址 2&#xff1a;在host文件里面添加地址&#xff08;记事本打开&#xff09;。 3&#xff1a;可以添加稳定的地址&#xff08;记得地址前加“#”&#xff09; #140.82.121.3 # GitHub #140.82.114.4 # GitHub #140.82.112.4 …

如何构建新一代实时湖仓?袋鼠云基于数据湖的探索升级之路

在之前的实时湖仓系列文章中&#xff0c;我们已经介绍了实时湖仓对于当前企业数字化转型的重要性&#xff0c;实时湖仓的功能架构设计&#xff0c;以及实时计算和数据湖结合的应用场景。 在本篇文章中&#xff0c;将介绍袋鼠云数栈在构建实时湖仓系统上的探索与落地实践&#…

html实现竖直步骤条

1、问题描述 最近碰到一个需求&#xff0c;要把审批流程改为竖直步骤条的形式。本来想直接抄网上的&#xff0c;但是网上给的要么是水平步骤条&#xff0c;要么是集成在框架里的&#xff0c;要么就是人家写的太复杂了&#xff0c;js&#xff0c;css一大堆。 2、我的代码 代码下…

【沁恒 CH32V208 开发板免费试用】+ U盘/ SD NAND读写与多功能数码相框

CH32V208继承了沁恆产品一贯的传统&#xff0c;即U盘的读写功能。这使得尽管CH32V208的闪存要比CH32V307的小一倍&#xff0c;但有了U盘读写功能的支持就可有效地缓解用户对存储空间的需求。它除了支持U盘的读取&#xff0c;还支持对CS SD NAND (贴片式TF卡/SD卡) 这类器件的使…

小黑子—springMVC:第一章 请求处理与响应数据

springMVC入门1.0 1、小黑子的springMVC基础1.1 SpringMVC概述1.2 SpringMVC快速入门1.3 Controller中直接注入spring中维护的Bean1.4 SpringMVC关键组件浅析 2、SpringMVC的请求处理2.1 请求映射路径配置2.2 请求数据的接收2.2.1 键值对方式接收数据2.2.1 - I RquestParam属性…

Linux Hadoop平台伪分布式安装(Hive on Spark)

&#x1f4d4;Linux Hadoop 伪分布式安装(Hive on Spark) 安装目录 1. JDK2. Hadoop3. MysqlHive3.1 Mysql8安装3.2 Hive安装 4. Spark4.1 Maven安装4.2 Scala安装4.3 Spark编译并安装 5. Zookeeper6. HBase 版本概要&#xff1a; jdk&#xff1a; jdk-8u391-linux-x64.tar.gz…

浅析SR隧道路径批量构造方法

为什么要仿真PCE LSP下发隧道路径&#xff1f; 在大型的多区域网络中&#xff0c;路径计算非常复杂。在某些场景下&#xff0c;为了完成路径计算&#xff0c;需要在控制器上部署特殊的计算组件&#xff0c;并需要不同区域中的节点之间协作。这使得网元在进行路径计算时效率低&…

局部路由守卫path守卫

局部路由守卫path守卫 path守卫&#xff08;beforeEnter&#xff09; 代码位置&#xff1a;path守卫代码写在route对象中&#xff08;index.js文件&#xff09;beforeEnter是一个函数&#xff0c;参数上没有回调函数beforeEnter有三个参数&#xff1a; to参数&#xff1a;to是…

CocosCreator | 2.3.3及后续版本浏览器无法断点和控制台不显示错误代码路径的解决方案(cocos代码报错无法定位的问题)

在2.3.3正式版的官方介绍中有这么一项&#xff1a; 提升网页预览时的加载速度 为了进一步提升开发效率&#xff0c;我们优化了网页预览时的脚本加载速度。不论是对引擎还是项目中的代码&#xff0c;载入速度都获得了提升。特别是在开启自定义引擎&#xff0c;或者使用手机扫码…

实时时钟和日历电路MS85163/MS85163M

主要特点 ◼ 基于 32.768kHz 晶振提供年、月、日、 周工作日、小时、分钟和秒 ◼ 具有世纪标记&#xff0c;可工作于 2000-2199 年 ◼ 工作电压&#xff1a; 1.8V-5.5V ◼ 低功耗 ◼ 最高频率达 400kHz 的 I 2 C 接口 ◼ 可编程的时钟输出 (32.768kHz, 1.024kHz…

开发人员请注意:在 PyPI 上的 Python 包中发现 BlazeStealer 恶意软件

1、开发人员请注意&#xff1a;在 PyPI 上的 Python 包中发现 BlazeStealer 恶意软件 一组新的恶意 Python 包已经滑入 Python 包索引 &#xff08;PyPI&#xff09; 存储库&#xff0c;其最终目的是从受感染的开发人员系统中窃取敏感信息。这些软件包伪装成看似无害的混淆工具…