做一个好看的首屏图片展示动画

前些日逛掘金,逛到了coco大佬的很炫酷的图片动画:现代 CSS 之高阶图片渐隐消失术 - 掘金 (juejin.cn)

利用 mask@property 做出图片像素化渐隐的效果实在是炫酷,正好现在没项目在摸鱼,脑子一抽开始用 nuxt3 写个人博客,正好首屏展示用得上 (又菜又爱玩)

定义网格

开始的想法是这样的,在大佬的代码基础上修改,通过代码获取到窗口宽高,然后通过 css变量 来动态修改行数列数,试了老半天也不太对,老是报错:

var(--w) is not a number

Undefined operation "calc(var(--w) * var(--h)) + 1"

也许一开始想用 js 去修改预处理器就有问题了🙃

正好又看油管一个大佬的图像碎片化,so,干脆用这个差不多的方法来代替 mask,做一个丐版图片渐隐,不需要 @propertyPaint 这种,兼容性还是蛮稳,也就一点点小卡

首先定义一下网格宽高,主要就是一层网格铺在图片前面,颜色跟背景色一致,后面再触发隐藏网格

js 复制代码
const centerBg = ref(null); // ref 获取到标签
const mask = ref(null); // ref 获取到标签
const row = ref(0); // 行数
const col = ref(0); // 列数
const len = ref(0); // 每个网格的大小
const initImg = () => {
  const imageWidth = window.innerWidth;
  const imageHeight = window.innerHeight;
  const maxGridCount =
    window.innerWidth > 1920
      ? 1000
      : window.innerWidth > 1080
      ? 800
      : window.innerWidth > 768
      ? 600
      : 400;
  // len 取宽高中较小的值,保证铺满
  len.value = Math.min(imageWidth, imageHeight) / Math.sqrt(maxGridCount);
  col.value = Math.floor(imageWidth / len.value);
  row.value = Math.floor(imageHeight / len.value);
};
const getCellStyle = (rowIndex, colIndex) => {
  return {
    width: `${len.value}px`,
    transition: `opacity ${100 + Math.random() * 500}ms ${
      ((colIndex + rowIndex / 2) / 20) * Math.random() * 500
    }ms`,
  };
};
html 复制代码
<template>
  <div
    class="centerBg relative w-screen h-screen overflow-hidden flex flex-col justify-center items-center gap-6"
    ref="centerBg"
  >
    <!-- mask -->
    <div class="mask absolute inset-0" ref="mask">
      <div v-for="rowIndex in row" :key="rowIndex" class="row flex">
        <div
          v-for="colIndex in col"
          :key="colIndex"
          class="col flex-1 aspect-square"
          :style="getCellStyle(rowIndex, colIndex)"
        ></div>
      </div>
    </div>
    <!-- title -->
    ...
    <!-- 简介 -->
    ...
    <!-- waves -->
    ...
  </div>
</template>

这段代码也没什么好解释的,len 是取的宽高较小的值为了保证能铺满,然后看下标签那部分,getCellStyle 里面的 transition 随便改,现在是从左往右带一些倾斜

修改 css

先把 css 的部分给写了,后面修改 --o 让网格隐藏就完工~

scss 复制代码
.centerBg {
  --o: 1;
  --img: url("assets/images/0.png");
  background: var(--img) center center/cover;
  .col {
    background: var(--bg); // 背景色
    opacity: var(--o);
  }
}
.dark .centerBg {
  --img: url("assets/images/_0.png");
}

触发动画

在完成定义网格和修改样式后,可以用个延时,或者正好可以等图片加载完在触发,用 Image 对象的 decode() 等,不然直接触发是没有效果的

js 复制代码
onMounted(() => {
  initImg();
  setTimeout(()=>{
    centerBg.value.style.setProperty("--o", 0)
  });
});

几百个 div 的动画,卡不卡就另说了 🤔

相关推荐
苦夏木禾15 分钟前
使用css制作一个环形进度展示圈
前端·css
软件技术NINI42 分钟前
html css js网页制作成品——成毅html+css+js 5页附源码
javascript·css·html
苏打水com1 小时前
第四篇:Day10-12 JS事件进阶+CSS动画——实现“复杂交互+视觉动效”(对标职场“用户体验优化”需求)
javascript·css·交互
苏打水com2 小时前
第六篇:Day16-18 AJAX进阶+接口对接——实现“前后端数据交互”(对标职场“接口开发”核心需求)
css·okhttp·html·js
m0_376137942 小时前
DevUI主题系统进阶:CSS-in-JS与暗黑模式无缝切换架构
javascript·css·架构·devui
哈哈~haha13 小时前
Step 14: Custom CSS and Theme Colors 自定义CSS类
前端·css·ui5
高桥留18 小时前
可编辑的span
前端·javascript·css
名字越长技术越强18 小时前
CSS之选择器|弹性盒子模型
前端·css
华仔啊20 小时前
CSS的clamp()函数:一行代码让网页自适应如此简单
前端·css
小书包酱21 小时前
告别在 vue 中使用公共 less 文件没有提示信息的烦恼
css·vue.js·less