当前位置:首页 > 科技  > 软件

Vue3 实现最近很火的酷炫功能:卡片悬浮发光

来源: 责编: 时间:2024-05-22 17:14:11 286观看
导读有趣的动画效果前几天在网上看到了一个很有趣的动画效果,如下,光会跟随鼠标在卡片上进行移动,并且卡片会有视差的效果。那么在 Vue3 中应该如何去实现这个效果呢?图片基本实现思路其实实现思路很简单,无非就是分几步:首先,卡

有趣的动画效果

前几天在网上看到了一个很有趣的动画效果,如下,光会跟随鼠标在卡片上进行移动,并且卡片会有视差的效果。KcX28资讯网——每日最新资讯28at.com

那么在 Vue3 中应该如何去实现这个效果呢?KcX28资讯网——每日最新资讯28at.com

图片图片KcX28资讯网——每日最新资讯28at.com

基本实现思路

其实实现思路很简单,无非就是分几步:KcX28资讯网——每日最新资讯28at.com

  • 首先,卡片是相对定位,光是绝对定位
  • 监听卡片的鼠标移入事件mouseenter,当鼠标进入时显示光
  • 监听卡片的鼠标移动事件mouseover,鼠标移动时修改光的left、top,让光跟随鼠标移动
  • 监听卡片的鼠标移出事件mouseleave,鼠标移出时,隐藏光

我们先在 Index.vue 中准备一个卡片页面,光的CSS效果可以使用filter: blur() 来实现。KcX28资讯网——每日最新资讯28at.com

图片图片KcX28资讯网——每日最新资讯28at.com

可以看到现在的效果是这样:KcX28资讯网——每日最新资讯28at.com

图片图片KcX28资讯网——每日最新资讯28at.com

实现光源跟随鼠标

在实现之前我们需要注意几点:KcX28资讯网——每日最新资讯28at.com

1、鼠标移入时需要设置卡片 overflow: hidden,否则光会溢出,而鼠标移出时记得还原。KcX28资讯网——每日最新资讯28at.com

2、获取鼠标坐标时需要用clientX/Y而不是pageX/Y,因为前者会把页面滚动距离也算进去,比较严谨。KcX28资讯网——每日最新资讯28at.com

刚刚说到实现思路时我们说到了mouseenter、mousemove、mouseleave,其实mouseenter、mouseleave 这二者的逻辑比较简单,重点是 mouseover 这个监听函数。KcX28资讯网——每日最新资讯28at.com

而在 mouseover 这个函数中,最重要的逻辑就是:光怎么跟随鼠标移动呢?KcX28资讯网——每日最新资讯28at.com

或者也可以这么说:怎么计算光相对于卡片盒子的 left 和 top。KcX28资讯网——每日最新资讯28at.com

对此我专门画了一张图,相信大家一看就懂怎么算了。KcX28资讯网——每日最新资讯28at.com

图片图片KcX28资讯网——每日最新资讯28at.com

  • left = clientX - x - width/2
  • height = clientY - y - height/2

知道了怎么计算,那么逻辑的实现也很明了了~封装一个use-light-card.ts:KcX28资讯网——每日最新资讯28at.com

图片图片KcX28资讯网——每日最新资讯28at.com

接着在页面中去使用:KcX28资讯网——每日最新资讯28at.com

图片图片KcX28资讯网——每日最新资讯28at.com

这样就能实现基本的效果啦~KcX28资讯网——每日最新资讯28at.com

图片图片KcX28资讯网——每日最新资讯28at.com

卡片视差效果

卡片的视差效果需要用到样式中 transform 样式,主要是配置四个东西:KcX28资讯网——每日最新资讯28at.com

  • perspective:定义元素在 3D 变换时的透视效果
  • rotateX:X 轴旋转角度
  • rotateY:Y 轴旋转角度
  • scale3d:X/Y/Z 轴上的缩放比例

图片图片KcX28资讯网——每日最新资讯28at.com

现在就有了卡片视差的效果啦~KcX28资讯网——每日最新资讯28at.com

图片图片KcX28资讯网——每日最新资讯28at.com

给所有卡片添加光源

上面只是给一个卡片增加光源,接下来可以给每一个卡片都增加光源啦!!!KcX28资讯网——每日最新资讯28at.com

图片图片KcX28资讯网——每日最新资讯28at.com

图片图片KcX28资讯网——每日最新资讯28at.com

让光源变成可配置

上面的代码,总感觉这个 hooks 耦合度太高不太通用,所以我们可以让光源变成可配置化,这样每个卡片就可以展示不同大小、颜色的光源了~像下面一样。KcX28资讯网——每日最新资讯28at.com

图片图片KcX28资讯网——每日最新资讯28at.com

既然是配置化,那我们希望是这么去使用 hooks 的,我们并不需要自己在页面中去写光源的dom节点,也不需要自己去写光源的样式,而是通过配置传入 hooks 中。KcX28资讯网——每日最新资讯28at.com

图片KcX28资讯网——每日最新资讯28at.com

所以 hooks 内部要自己通过操作 DOM 的方式,去添加、删除光源,可以使用createElement、appendChild、removeChild 去做这些事~。KcX28资讯网——每日最新资讯28at.com

图片图片KcX28资讯网——每日最新资讯28at.com

完整源码

<!-- Index.vue --><template>  <div class="container">    <!-- 方块盒子 -->    <div class="item" ref="cardRef1"></div>    <!-- 方块盒子 -->    <div class="item" ref="cardRef2"></div>    <!-- 方块盒子 -->    <div class="item" ref="cardRef3"></div>  </div></template><script setup lang="ts">import { useLightCard } from './use-light-card';const { cardRef: cardRef1 } = useLightCard();const { cardRef: cardRef2 } = useLightCard({  light: {    color: '#ffffff',    width: 100,  },});const { cardRef: cardRef3 } = useLightCard({  light: {    color: 'yellow',  },});</script><style scoped lang="less">.container {  background: black;  width: 100%;  height: 100%;  padding: 200px;  display: flex;  justify-content: space-between;  .item {    position: relative;    width: 125px;    height: 125px;    background: #1c1c1f;    border: 1px solid rgba(255, 255, 255, 0.1);  }}</style>
// use-light-card.tsimport { onMounted, onUnmounted, ref } from 'vue';interface IOptions {  light?: {    width?: number; // 宽    height?: number; // 高    color?: string; // 颜色    blur?: number; // filter: blur()  };}export const useLightCard = (option: IOptions = {}) => {  // 获取卡片的dom节点  const cardRef = ref<HTMLDivElement | null>(null);  let cardOverflow = '';  // 光的dom节点  const lightRef = ref<HTMLDivElement>(document.createElement('div'));  // 设置光源的样式  const setLightStyle = () => {    const { width = 60, height = 60, color = '#ff4132', blur = 40 } = option.light ?? {};    const lightDom = lightRef.value;    lightDom.style.position = 'absolute';    lightDom.style.width = `${width}px`;    lightDom.style.height = `${height}px`;    lightDom.style.background = color;    lightDom.style.filter = `blur(${blur}px)`;  };  // 设置卡片的 overflow 为 hidden  const setCardOverflowHidden = () => {    const cardDom = cardRef.value;    if (cardDom) {      cardOverflow = cardDom.style.overflow;      cardDom.style.overflow = 'hidden';    }  };  // 还原卡片的 overflow  const restoreCardOverflow = () => {    const cardDom = cardRef.value;    if (cardDom) {      cardDom.style.overflow = cardOverflow;    }  };  // 往卡片添加光源  const addLight = () => {    const cardDom = cardRef.value;    if (cardDom) {      cardDom.appendChild(lightRef.value);    }  };  // 删除光源  const removeLight = () => {    const cardDom = cardRef.value;    if (cardDom) {      cardDom.removeChild(lightRef.value);    }  };  // 监听卡片的鼠标移入  const onMouseEnter = () => {    // 添加光源    addLight();    setCardOverflowHidden();  };  // use-light-card.ts  // 监听卡片的鼠标移动  const onMouseMove = (e: MouseEvent) => {    // 获取鼠标的坐标    const { clientX, clientY } = e;    // 让光跟随鼠标    const cardDom = cardRef.value;    const lightDom = lightRef.value;    if (cardDom) {      // 获取卡片相对于窗口的x和y坐标      const { x, y } = cardDom.getBoundingClientRect();      // 获取光的宽高      const { width, height } = lightDom.getBoundingClientRect();      lightDom.style.left = `${clientX - x - width / 2}px`;      lightDom.style.top = `${clientY - y - height / 2}px`;      //   设置动画效果      const maxXRotation = 10; // X 轴旋转角度      const maxYRotation = 10; // Y 轴旋转角度      const rangeX = 200 / 2; // X 轴旋转的范围      const rangeY = 200 / 2; // Y 轴旋转的范围      const rotateX = ((clientX - x - rangeY) / rangeY) * maxXRotation; // 根据鼠标在 Y 轴上的位置计算绕 X 轴的旋转角度      const rotateY = -1 * ((clientY - y - rangeX) / rangeX) * maxYRotation; // 根据鼠标在 X 轴上的位置计算绕 Y 轴的旋转角度      cardDom.style.transform = `perspective(1000px) rotateX(${rotateX}deg) rotateY(${rotateY}deg)`; //设置 3D 透视    }  };  // 监听卡片鼠标移出  const onMouseLeave = () => {    // 鼠标离开移出光源    removeLight();    restoreCardOverflow();  };  onMounted(() => {    // 设置光源样式    setLightStyle();    // 绑定事件    cardRef.value?.addEventListener('mouseenter', onMouseEnter);    cardRef.value?.addEventListener('mousemove', onMouseMove);    cardRef.value?.addEventListener('mouseleave', onMouseLeave);  });  onUnmounted(() => {    // 解绑事件    cardRef.value?.removeEventListener('mouseenter', onMouseEnter);    cardRef.value?.removeEventListener('mousemove', onMouseMove);    cardRef.value?.removeEventListener('mouseleave', onMouseLeave);  });  return {    cardRef,  };};

KcX28资讯网——每日最新资讯28at.com

本文链接:http://www.28at.com/showinfo-26-90040-0.htmlVue3 实现最近很火的酷炫功能:卡片悬浮发光

声明:本网页内容旨在传播知识,若有侵权等问题请及时与本网联系,我们将在第一时间删除处理。邮件:2376512515@qq.com

上一篇: Python 并发编程的 12 个实用技巧

下一篇: ViewRootImpl如何负责管理绘制视图树和刷新界面

标签:
  • 热门焦点
  • Redmi Pad评测:红米充满野心的一次尝试

    从Note系列到K系列,从蓝牙耳机到笔记本电脑,红米不知不觉之间也已经形成了自己颇有竞争力的产品体系,在中端和次旗舰市场上甚至要比小米新机的表现来得更好,正所谓“大丈夫生居
  • 5月iOS设备好评榜:iPhone 14仅排第43?

    来到新的一月,安兔兔的各个榜单又重新汇总了数据,像安卓阵营的榜单都有着比较大的变动,不过iOS由于设备的更新换代并没有那么快,所以相对来说变化并不大,特别是iOS好评榜,老款设
  • 学习JavaScript的10个理由...

    作者 | Simplilearn编译 | 王瑞平当你决心学习一门语言的时候,很难选择到底应该学习哪一门,常用的语言有Python、Java、JavaScript、C/CPP、PHP、Swift、C#、Ruby、Objective-
  • 十个简单但很有用的Python装饰器

    装饰器(Decorators)是Python中一种强大而灵活的功能,用于修改或增强函数或类的行为。装饰器本质上是一个函数,它接受另一个函数或类作为参数,并返回一个新的函数或类。它们通常用
  • 只需五步,使用start.spring.io快速入门Spring编程

    步骤1打开https://start.spring.io/,按照屏幕截图中的内容创建项目,添加 Spring Web 依赖项,并单击“生成”按钮下载 .zip 文件,为下一步做准备。请在进入步骤2之前进行解压。图
  • .NET 程序的 GDI 句柄泄露的再反思

    一、背景1. 讲故事上个月我写过一篇 如何洞察 C# 程序的 GDI 句柄泄露 文章,当时用的是 GDIView + WinDbg 把问题搞定,前者用来定位泄露资源,后者用来定位泄露代码,后面有朋友反
  • 微软邀请 Microsoft 365 商业用户,测试视频编辑器 Clipchamp

    8 月 1 日消息,微软近日宣布即将面向 Microsoft 365 商业用户,开放 Clipchamp 应用,邀请用户通过该应用来编辑视频。微软于 2021 年收购 Clipchamp,随后开始逐步整合到 Microsof
  • 华为将推出盘古数字人大模型 可帮助用户12小时完成数字人生成

    在今日举行的2023年华为云数字文娱AI创新峰会上,华为云全球Marketing与销售服务总裁石冀琳表示,华为云将在后续推出盘古数字人大模型,可帮助用户12小
  • 机构称Q2国内智能手机销量同比下滑4% vivo份额重回第1

    7月29日消息,根据市场调查机构Counterpoint Research公布的最新报告,2023年第2季度中国智能手机销量同比下降4%,创新自2014年以来第2季度销量新低。报
Top