前端:鼠标点击实现高亮特效

一、实现思路

获取鼠标点击位置

通过鼠标点击位置设置高亮裁剪动画

二、效果展示

三、按钮组件代码

html 复制代码
<template>
  <button
    class="blueBut"
    @click="clickHandler"
    :style="{
      backgroundColor: clickBut ? 'rgb(31, 67, 117)' : 'rgb(128, 128, 128)',
    }"
  >
    <slot></slot>
    <!-- 光亮效果 -->
    <div
      class="lightBox"
      ref="lightBoxRef"
      :style="{
        background: 'rgba(209, 209, 209, 0.3)',
        backgroundSize: '200% 200%',
        '--clickX': `${clickPos.x}%`,
        '--clickY': `${clickPos.y}%`,
      }"
    ></div>
  </button>
</template>
<script setup lang="ts">
//获取鼠标在元素中点击位置,该函数见博客:https://blog.csdn.net/qq_45820271/article/details/139706893?spm=1001.2014.3001.5502
import { getClickPos } from "./getClickPos";
import { ref, reactive } from "vue";
const clickBut = defineModel<boolean>();
//获取光亮盒子元素,在H5中可以使用lightBoxRef = document.getElementById('lightbox')获取
const lightBoxRef = ref<HTMLElement | null>(null);
const clickPos = reactive({ x: 0, y: 0 });
const clickHandler = (e: MouseEvent) => {
  clickBut.value = !clickBut.value;
  const lightBox = lightBoxRef.value;
  if (!lightBox) return;
  const pos = getClickPos(e);
  let width = lightBox.getBoundingClientRect().width;
  let height = lightBox.getBoundingClientRect().height;
  //获取点击位置相对于元素的百分比
  clickPos.x = (pos.x / width) * 100;
  clickPos.y = (pos.y / height) * 100;
  //通过移除和添加让每次鼠标点击都触发动画
  lightBox.classList.remove("lightShow");
  setTimeout(() => {
    lightBox.classList.add("lightShow");
  }, 10);
};
</script>
<style scoped>
/* 自定义CSS属性解决无法过渡问题,方式见博客:https://blog.csdn.net/qq_45820271/article/details/139242637?spm=1001.2014.3001.5502 */
@property --time {
  syntax: "<time>";
  initial-value: 0.6s;
  inherits: false;
}
.blueBut {
  width: 200px;
  height: 50px;
  border-radius: 5px;
  border: none;
  color: #ffffff;
  box-shadow: 0 0 1px rgba(0, 0, 0, 0.4);
  cursor: pointer;
  position: relative;
  transition: all var(--time) linear;
}
.lightBox {
  width: 100%;
  height: 100%;
  pointer-events: none;
  transition: all var(--time) ease;
  position: absolute;
  top: 0;
  left: 0;
  filter: blur(3px);
}
.lightShow {
  animation: changeImg var(--time) linear forwards;
}
@keyframes changeImg {
  0% {
    opacity: 0;
    clip-path: circle(0% at var(--clickX) var(--clickY));
  }
  50% {
    opacity: 1;
  }
  100% {
    opacity: 0;
    clip-path: circle(200% at var(--clickX) var(--clickY));
  }
}
</style>

四、组件使用

html 复制代码
<template>
  <div class="page">
    <blueBut v-model="clickBut">
      <div class="ButInfos">
        <div class="icon">
          <svg
            t="1718506308447"
            class="icon"
            viewBox="0 0 1024 1024"
            version="1.1"
            xmlns="http://www.w3.org/2000/svg"
            p-id="2403"
            width="20"
            height="20"
          >
            <path :d="path" fill="#ffffff" p-id="2404"></path>
          </svg>
        </div>

        <div class="texts">
          {{ runText }}<br /><span style="font-size: 9px; font-weight: 600">{{
            numText
          }}</span>
        </div>
      </div>
    </blueBut>
  </div>
</template>

<script setup lang="ts">
import { ref, computed } from "vue";
import blueBut from "../components/blueBut.vue";
const clickBut = ref(false);
const path = computed(() => {
  return clickBut.value
    ? "M512 42.666667A469.333333 469.333333 0 0 0 42.666667 512 469.333333 469.333333 0 1 0 512 42.666667z m0 878.506666A409.173333 409.173333 0 0 1 102.826667 512a409.173333 409.173333 0 0 1 818.346666 0A409.173333 409.173333 0 0 1 512 921.173333zM810.666667 354.133333L756.906667 298.666667l-307.2 315.733333L267.093333 426.666667 213.333333 482.133333l236.373334 243.2 51.626666-53.333333z"
    : "M939.52 331.38A465.39 465.39 0 1 0 976 512a462.4 462.4 0 0 0-36.48-180.62zM512 896c-211.74 0-384-172.26-384-384a382.29 382.29 0 0 1 90.31-247.12l540.81 540.81A382.29 382.29 0 0 1 512 896z m302.65-147.92L275.92 209.35A382.1 382.1 0 0 1 512 128c211.74 0 384 172.26 384 384a382.1 382.1 0 0 1-81.35 236.08z";
});
const runText = computed(() => {
  return clickBut.value ? "运行中" : "已停止";
});
const byteNum = ref(0);
const numText = computed(() => {
  return clickBut.value ? `${byteNum.value} Bytes已转发` : "点此启动";
});
</script>
<style scoped>
.page {
  width: 100%;
  height: 100vh;
  display: flex;
  justify-content: center;
  align-items: center;
}
.ButInfos {
  display: flex;
  align-items: center;
  padding-left: 5px;
  text-align: left;
}
.icon {
  width: 40px;
  display: flex;
  justify-content: center;
  align-items: center;
}
</style>
相关推荐
abc80021170349 分钟前
前端Bug 修复手册
前端·bug
Best_Liu~12 分钟前
el-table实现固定列,及解决固定列导致部分滚动条无法拖动的问题
前端·javascript·vue.js
_斯洛伐克1 小时前
下降npm版本
前端·vue.js
苏十八2 小时前
前端进阶:Vue.js
前端·javascript·vue.js·前端框架·npm·node.js·ecmascript
st紫月3 小时前
用MySQL+node+vue做一个学生信息管理系统(四):制作增加、删除、修改的组件和对应的路由
前端·vue.js·mysql
乐容3 小时前
vue3使用pinia中的actions,需要调用接口的话
前端·javascript·vue.js
似水明俊德4 小时前
ASP.NET Core Blazor 5:Blazor表单和数据
java·前端·javascript·html·asp.net
至天5 小时前
UniApp 中 Web/H5 正确使用反向代理解决跨域问题
前端·uni-app·vue3·vue2·vite·反向代理
与墨学长5 小时前
Rust破界:前端革新与Vite重构的深度透视(中)
开发语言·前端·rust·前端框架·wasm
H-J-L5 小时前
Web基础与HTTP协议
前端·http·php