Vue中实现过渡动画

news2025/1/29 13:55:12

文章目录

    • Vue的transition动画
      • Transition动画的使用
      • Transition组件的原理
      • Transition动画的class
    • Vue的animation动画
      • Animation动画的使用
      • 同时设置两种动画(了解)
    • 过渡的模式mode
    • 列表过渡
      • 列表过渡的介绍
      • 列表过渡的使用

Vue的transition动画

Transition动画的使用

在开发中,我们想要给一个组件的显示和消失添加某种过渡动画,可以很好的增加用户体验

React框架本身并没有提供任何动画相关的API,所以在React中使用过渡动画我们需要使用一个第三方库 react-transition-group;

Vue中为我们提供一些内置组件和对应的API来完成动画,利用它们我们可以方便的实现过渡动画效果;

我们来看一个案例

Hello World的显示和隐藏;

通过下面的代码实现,显式和隐藏Hello World是不会有任何动画效果的;

<template>
  <div class="app">
    <button @click="btnClick">Toogle</button>
    <h2 v-show="isShow">Hello World</h2>
  </div>
</template>

<script setup>
import { ref } from 'vue';

const isShow = ref(true)
const btnClick = () => {
  isShow.value = !isShow.value
}
</script>

没有动画的情况下,整个内容的显示和隐藏会非常的生硬

如果我们希望给单元素或者组件实现过渡动画,可以使用Vue提供的 transition 内置组件来完成动画;

Vue 提供了 transition 的封装组件,在下列情形中,可以给任何元素和组件添加进入/离开过渡

条件渲染 (使用 v-if)条件展示 (使用 v-show)

动态组件

组件根节点

例如我们给刚刚的Hello World添加动画, 首先定义类(类的名字暂时是固定的)

/* 从什么状态进入 */
.v-enter-from {
  opacity: 0;
}

/* 进入之后打算变成的状态 */
.v-enter-to {
  opacity: 1;
}

/* 进入的过程中做什么 */
.v-enter-active {
  transition: opacity 2s linear;
}

定义好类之后, 我们可以使用Vue的内置组件 transition 将动画元素包裹起来, transition 组件会自动在合适的时机, 将合适的类添加到它所包裹的元素上

<template>
  <div class="app">
    <button @click="btnClick">Toogle</button>
    <!-- 使用内置组件包裹h2 -->
    <transition>
      <h2 v-show="isShow">Hello World</h2>
    </transition>
  </div>
</template>

此时显式Hello World 就已经有动画效果了, 但是隐藏时并没有效果, 是因为我们刚刚定义的类是进入的类, 当离开时我们也应该定义对应的类

/* 离开后的状态 */
.v-leave-to,
/* 从什么状态进入 */
.v-enter-from {
  opacity: 0;
}

/* 开始离开的状态 */
.v-leave-from,
/* 进入之后打算变成的状态 */
.v-enter-to {
  opacity: 1;
}

/* 离开过程中做什么 */
.v-leave-active,
/* 进入的过程中做什么 */
.v-enter-active {
  transition: opacity 1s linear;
}

Transition组件的原理

我们会发现,上面演示代码中Vue自动给h2元素添加了动画,这是什么原因呢?

当插入或删除包含在 transition 组件中的元素时, Vue 将会做以下处理

1.自动嗅探目标元素是否应用了CSS过渡或者动画,如果有,那么在恰当的时机添加/删除 CSS类名;

2.如果 transition 组件提供了JavaScript钩子函数,这些钩子函数将在恰当的时机被调用;

3.如果没有找到JavaScript钩子并且也没有检测到CSS过渡/动画, DOM插入、删除操作将会立即执行;

在上面我们简单使用了Vue会自动添加/删除的CSS类名, 接下来我们详细学习一下都有哪些类吧


Transition动画的class

我们会发现上面提到了很多个class,事实上Vue就是帮助我们在这些class之间来回切换完成的动画

v-enter-from:定义进入过渡的开始状态

在元素被插入之前生效,在元素被插入之后的下一帧移除。

v-enter-active:定义进入过渡生效时的状态

在整个进入过渡的阶段中应用,在元素被插入之前生效,在过渡/动画完成之后移除。这个类可以被用来定义进入过渡的过程时间,延迟和曲线 函数。

v-enter-to:定义进入过渡的结束状态

在元素被插入之后下一帧生效 (与此同时 v-enter-from 被移除),在过渡/动画完成之后移除。

v-leave-from:定义离开过渡的开始状态

在离开过渡被触发时立刻生效,下一帧被移除。

v-leave-active:定义离开过渡生效时的状态

在整个离开过渡的阶段中应用,在离开过渡被触发时立刻生效,在过渡/动画完成之后移除。这个类可以被用来定义离开过渡的过程时间,延迟 和曲线函数。

v-leave-to:离开过渡的结束状态

在离开过渡被触发之后下一帧生效 (与此同时 v-leave-from 被删除),在过渡/动画完成之后移除。

在这里插入图片描述

class的name命名规则如下

如果我们使用的是一个没有name属性的transition,那么所有的class是以 v- 作为默认前缀

如果我们添加了一个name属性,比如<transtion name= “abc” >,那么所有的class会以 abc- 开头

Vue的animation动画

Animation动画的使用

例如有一段文本, 我们对文本的显式隐藏添加上一些动画

先定义类, 用于实现动画

/* 定义进入的动画 */
.v-enter-active {
  animation: myAnim 2s linear;
}

/* 定义离开的动画 */
.v-leave-active {
  animation: myAnim 2s linear reverse;
}

/* 定义序列帧动画 */
@keyframes myAnim {
  0% {
    transform: scale(0);
    opacity: 0;
  }

  50% {
    transform: scale(1.25);
    opacity: 0.5;
  }

  100% {
    transform: scale(1);
    opacity: 1;
  }
}

同transition动画, 使用transition组件包裹

<template>
  <div class="app">
    <button @click="btnClick">Toogle</button>
    <!-- 使用内置组件包裹h2 -->
    <transition>
      <h2 v-show="isShow">
        今天2022年8月6日星期六,利好消息来袭!因为经济数据仍显疲软,
        市场对全球经济增速放缓可能抑制需求的担忧加剧,导致成品油调价
        挂靠的国际油价继续下挫,同时综合原油品种变化率也已经完成正值
        到负值的转变。数据显示:截至国内成品油调价周期的第8个工作日
        结束,原油综合变化率为-0.83%,详细成品油调价窗口开启时间定在
        8月9日24时开启。
      </h2>
    </transition>
  </div>
</template>

同时设置两种动画(了解)

Vue为了知道过渡的完成,内部是在监听 transitionend 或 animationend,到底使用哪一个取决于元素应用的CSS规则

如果我们只是使用了其中的一个,那么Vue能自动识别类型并设置监听;

但是如果我们同时使用了transition动画和animation动画呢?

在开发中, 如果设置了两个动画, 尽量要保持两个动画的时间一致;

/* 定义transition动画 */
.v-enter-from,
.v-leave-to {
  opacity: 0;
}

.v-enter-to,
.v-leave-from {
  opacity: 1;
}


.v-enter-active {
  /* 设置两个不同的动画, 最好保持动画时间一致 */
  animation: myAnim 2s linear;
  transition: opacity 2s linear;
}

.v-leave-active {
  /* 设置两个不同的动画, 最好保持动画时间一致 */
  animation: myAnim 2s linear reverse;
  transition: opacity 2s linear;
}

/* 定义序列帧动画 */
@keyframes myAnim {
  0% {
    transform: scale(0);
    opacity: 0;
  }

  50% {
    transform: scale(1.25);
    opacity: 0.5;
  }

  100% {
    transform: scale(1);
    opacity: 1;
  }
}

如果使用两个动画, 动画的时间又不一致, 并且在这个情况下可能某一个动画执行结束时,另外一个动画还没有结

在这种情况下,我们可以设置 type 属性为 animation 或者 transition 来明确的告知Vue监听的类型, 以哪一个动画为标准(一般不设置);

<!-- 告知以哪一个动画为标准 -->
<transition type="transition">
  <h2 v-show="isShow">
    今天2022年8月6日星期六,利好消息来袭!因为经济数据仍显疲软,
    市场对全球经济增速放缓可能抑制需求的担忧加剧,导致成品油调价
    挂靠的国际油价继续下挫,同时综合原油品种变化率也已经完成正值
    到负值的转变。数据显示:截至国内成品油调价周期的第8个工作日
    结束,原油综合变化率为-0.83%,详细成品油调价窗口开启时间定在
    8月9日24时开启。
  </h2>
</transition>

过渡的模式mode

我们来看当前的动画在两个元素之间显式隐藏切换的时候添加动画存在的问题

在这里插入图片描述

我们会发现 Hello World 和 你好啊,李银河是同时存在的

这是因为默认情况下进入和离开动画是同时发生的;

如果确实我们希望达到这个的效果,那么是没有问题;

但是如果我们不希望同时执行进入和离开动画,那么我们需要设置transition的过渡模式

in-out: 新元素先进行过渡,完成之后当前元素过渡离开;

out-in: 当前元素先进行过渡,完成之后新元素过渡进入;

<transition mode="in-out">
  <h2>Hello Worl</h2>
  <h2>你好啊, 李银河</h2>
</transition>

上面的示例同样适用于我们的动态组件

<div class="app">
  <div>
    <button @click="isShow = !isShow">切换</button>
  </div>

  <transition name="why" mode="out-in">
    <component :is=" isShow ? 'home': 'about'"></component>
  </transition>
</div>

默认情况下,首次渲染的时候是没有动画的,如果我们希望给他添加上去动画,那么就可以增加另外一个属性appear

<div class="app">
  <div>
    <button @click="isShow = !isShow">切换</button>
  </div>

  <transition name="why" mode="out-in" appear="">
    <component :is=" isShow ? 'home': 'about'"></component>
  </transition>
</div>

列表过渡

列表过渡的介绍

目前为止,过渡动画我们只要是针对单个元素或者组件的

要么是单个节点;

要么是同一时间渲染多个节点中的一个;

那么如果希望渲染的是一个列表,并且该列表中添加删除数据也希望有动画执行呢?

这个时候我们要使用<transition-group>组件来完成;

使用<transition-group>有如下的特点

默认情况下,它不会渲染成任何一个元素,但是你可以指定一个元素并以 tag attribute 让他进行渲染;

过渡模式不可用,因为我们不再相互切换特有的元素;

内部元素总是需要提供唯一的 key attribute 值;

CSS 过渡的类将会应用在内部的元素中,而不是这个组/容器本身;

列表过渡的使用

我们来做一个案例

案例是一列数字,可以继续添加或者删除数字;

在添加和删除数字的过程中,对添加的或者移除的数字添加动画;

示例代码如下

<template>
  <div class="app">
    <button @click="addNumber">添加数字</button>
    <button @click="removeNumber">删除数字</button>
    
    <template v-for="item in nums" :key="item">
      <span>{{ item }}</span>
    </template>
  </div>
</template>

<script setup>
import { ref } from 'vue';

const nums = ref([0, 1, 2, 3, 4, 5, 6, 7, 8, 9])

const addNumber = () => {
  nums.value.splice(randomIndex(), 0, nums.value.length)
}

const removeNumber = () => {
  nums.value.splice(randomIndex(), 1)
}

const randomIndex = () => {
  return Math.floor(Math.random() * nums.value.length)
}

</script>

添加动画, 此时添加上动画, 这个组中的内容都会有动画效果

span {
  margin-right: 20px;
  display: inline-block;
}

.v-enter-from,
.v-leave-to {
  opacity: 0;
  transform: translateY(20px);
}

.v-enter-to,
.v-leave-from {
  opacity: 1;
  transform: translateY(0);
}

.v-enter-active,
.v-leave-active {
  transition: all 2s linear;
}
<template>
  <div class="app">
    <button @click="addNumber">添加数字</button>
    <button @click="removeNumber">删除数字</button>
    
    <!-- tag属性, 告知渲染成div -->
    <transition-group tag="div">
      <template v-for="item in nums" :key="item">
        <span>{{ item }}</span>
      </template>
    </transition-group>
  </div>
</template>

在上面的案例中虽然新增的或者删除的节点是有动画的,但是对于哪些其他需要移动的节点是没有动画的

我们可以通过使用一个新增的 v-move 的class来完成动画;

它会在元素改变位置的过程中应用;

像之前的名字一样,我们也可以通过name来自定义前缀;

/* 定位解决删除占用空间, 无法执行动画问题 */
.v-leave-active {
  position: absolute;
}
  /* 针对其他节点移动需要的动画 */
.v-move {
  transition: all 2s linear;
}

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

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

相关文章

vite配置cdn优化打包体积

文章目录前言一、版本确认二、使用步骤1.rollup-plugin-visualizer打包体积可视化面板2.配置cdn方法第一种方法&#xff1a; vite-plugin-cdn-import第二种方法&#xff1a; rollup-plugin-external-globals总结前言 大家都知道前端性能优化的方法&#xff0c;cdn外部引入的方…

【微信小程序】小程序知识补充篇

&#x1f381;写在前面&#xff1a; 观众老爷们好呀&#xff0c;这里是前端小刘不怕牛牛频道&#xff0c;小程序系列又更新了呀。 还有就是中秋节就快来啦&#xff0c;程序员过中秋&#xff0c;当然是要好好放松一下啦&#xff0c;那么中秋前我们就不能偷懒了&#xff0c;赶紧学…

Controller层接收前端传参的几种方法。@RequestParam、@RequestBody、@PathVariable。及参数校验。

一、RequestParam 主要用于将请求参数区域的数据映射到控制层方法的参数上 // http://localhost:8080/wh/user/edit?Id9452659856325148452&name天天向上// RequestParam源码Target({ElementType.PARAMETER}) // 只能作用于参数上 Retention(RetentionPolicy.RUNTIME) D…

vue watch监听数据解决新旧值一样的问题(newValue, oldValue)

watch是监听 Vue 实例变化的一个表达式或方法。回调函数得到的参数为新值和旧值。 基础用法 <!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><meta http-equiv"X-UA-Compatible" content"IEedge…

异步函数async

什么是同步异步 在最新的ES7&#xff08;ES2017&#xff09;中提出的前端异步特性&#xff1a;async、await。 在了解async和await之前得先明白什么是同步函数&#xff0c;什么是异步函数。 同步函数&#xff1a;当一个函数是同步执行时&#xff0c;那么当该函数被调用时不会…

前端CryptoJS和Java后端数据互相加解密(AES)

目录一、序言二、关于前端CryptoJS1、CryptoJS简单介绍2、加密和填充模式选择3、前端AES加解密示例(1) cryptoutils工具类(2) 测试用例(3) 加解密后输出内容说明三、Java后端AES加解密1、Java中支持的加密模式和填充说明2、工具类CryptoUtils3、测试用例一、序言 最近刚好在做…

nodejs——解决跨域问题

目录 1什么是跨域 2解决 1 jsonp(缺点&#xff1a;不能请求post请求&#xff09; 1 index.html页 2 proxy.js页面 搭建一个服务器&#xff08;写好代码后&#xff0c;在cmd上启动&#xff09; 3效果 2服务端代理 &#xff08;由于后端请求不受浏览器同源策略影响&…

Bootstrap、栅格系统布局

一、Bootstrap Bootstrap是一个基于HTML、CSS和JavaScript语言编写的框架&#xff0c;具有简单、灵活的特性&#xff0c;拥有样式库、组件和插件。 Bootstrap常用来开发响应式布局和移动设备优先的Web项目&#xff0c;能够帮助开发者快速搭建前端页面。Bootstrap官方网站:Boot…

VUE3构建Cesium项目

目录 1.Cesium开发参考资料 2.VUE中使用Cesium 2.1 使用VUE创建项目 1.创建test项目 2.项目中引入Cesium 3.修改App.vue如下 4.将cesium静态文件复制至public下 5.运行效果 1.Cesium开发参考资料 Cesium官方网站&#xff1a;Cesium: The Platform for 3D Geospatial …

前端接收 type: “application/octet-stream“ 格式的数据并下载,解决后端返回不唯一

前端接收 type: “application/octet-stream“ 格式的数据并下载&#xff0c;还有后端既返回octet-stream还返回JSON数据时的处理方法 今天些项目的时候&#xff0c;后端改了一下文件下载的方式&#xff0c;打算用接口返回 type: “application/octet-stream“格式的数据&…

【微信小程序】全局配置

目录 全局配置文件及常用的配置项 全局配置 - window 1. 小程序窗口的组成部分 2. 了解 window 节点常用的配置项 3. 设置导航栏的标题 4. 设置导航栏的背景色 5. 设置导航栏的标题颜色 6. 全局开启下拉刷新功能 7. 设置下拉刷新时窗口的背景色 8. 设置下拉刷新时 lo…

echarts 柱状图滚动

实现效果&#xff1a;柱形图展示水平滚动条&#xff0c;并且鼠标滚动支持让滚动条平移 echarts文档里&#xff0c;图形的滚动条分两种 内置型 &#xff08;效果是&#xff1a; 鼠标在图中点击拖动平移&#xff0c;在图中滚动缩放&#xff09;滚动条型 &#xff08;效果是&…

高德地图自定义图标的点标记Marker--初体验(二)

点标记Marker创建一个默认图标的点标记:创建一个自定义图标的点标记:new AMap.Marker({}) 参数说明本文以Marker为主&#xff0c;其他点标记方法大差不差 通过上两篇文章我们已经了解到如何引入高德地图并进行初始化了&#xff0c;本文主要讲解普通点标记Marker,Marker 类型推荐…

框架获取当前登录用户以及用户信息

CSDN话题挑战赛第2期 参赛话题&#xff1a;学习笔记 前言 &#x1f4eb; 作者简介&#xff1a;「六月暴雪飞梨花」&#xff0c;专注于研究Java&#xff0c;就职于科技型公司后端中级工程师 &#x1f525; 三连支持&#xff1a;如果此文还不错的话&#xff0c;还请 ❤️关注、&…

设置浏览器显示小于12px以下字体的三种方法

使用场景&#xff1a; 以往设计图给的字号一般最小就是12px&#xff0c; 开发人员一般是使用谷歌浏览器来进行调试运行。 谷歌浏览器上显示字体最小为12px&#xff0c;css设置font-size&#xff1a;10px&#xff0c;运行代码显示结果仍然是12px大小&#xff0c;但是挡不住甲方…

TypeError The view function did not return a valid response. The function either returned None 的解决

使用flask框架制作登录、注册的页面时&#xff0c;app.py运行成功&#xff0c;数据库有用户&#xff0c;1234&#xff0c;密码也是1234 点击登录之后&#xff0c; 报如下错误。 TypeError TypeError: The view function did not return a valid response. The function either …

Vue3中的父传子和子传父如何实现

大家都知道Vue2中父传子是通过父组件绑定一个属性&#xff0c;子组件再用props进行接收&#xff0c;子传父是通过this.$emit那么Vue3中有什么不同呢&#xff1f;以下为您解答谜团 #Vue3的父传子 一.现在父组件调用子组件的时候,通过动态属性把数据传递过去 二.在子组件通过prop…

XSS漏洞及其原理(详解)

文章目录前言一、XSS漏洞原理1.概述2.利用方式3.执行方式4.攻击对象5.XSS危害&#xff08;1&#xff09;窃取cookie&#xff08;2&#xff09;未授权操作&#xff08;3&#xff09;传播蠕虫病毒6.简单代码7.XSS验证8.二、XSS漏洞分类1.反射型XSS原理特点举个栗子&#xff1a;2.…

EasyExcel使用与步骤

一、导入依赖&#xff08;3.1.0版本不需要poi依赖&#xff09; <!-- easyExcel--><dependency><groupId>com.alibaba</groupId><artifactId>easyexcel</artifactId><version>3.1.1</version></dependency>二、写数据…

nvm下node安装;node环境变量配置

1、nvm安装 1、双击安装文件 nvm-setup.exe 2、选择nvm安装路径 3、选择nodejs路径 4、确认安装即可 5、安装完确认 打开CMD&#xff0c;输入命令 nvm &#xff0c;安装成功则如下显示。可以看到里面列出了各种命令&#xff0c;本节最后会列出这些命令的中文示意。 6、…