乐趣区

关于前端:AmazingCSS-也能实现烟雾效果

最近利用 CSS 实现了一些看似超出 CSS 能力的成果:

  • 巧用突变实现高级感拉满的背景光动画
  • Amazing!!CSS 也能实现极光?

本文持续此系列,本文次要想探讨一下,应用 CSS 是否比拟好的实现一些 烟雾成果。像是这样:

仔细观察烟雾成果,有两个比拟重要的特点:

  • 含糊成果
  • 颗粒感

首先看含糊成果,想到含糊,大部分同学首先都会想到应用 filter: blur()

当然没错,不过在 CSS 中,除了滤镜,咱们还能应用一类其余伎俩去模仿 含糊 的成果。

纯 CSS 实现烟雾动画

咱们首先来看这样一个成果:

假如,咱们有这样一个字符:

<span>C</span>

咱们仅仅是通过 text-shadow + opacity 的变动,就能模仿烟雾的成果:

span {
  text-shadow: 0 0 0 whitesmoke;
  animation: smoky 5s;
}

@keyframes smoky {
  to {
    text-shadow: 0 0 20px whitesmoke;
    opacity: 0;
  }
}

看看成果:

在上述的根底上,咱们能够加上位移、旋转、缩放,略微革新一下上述代码,增加一些 transform 变换:

span {
  text-shadow: 0 0 0 whitesmoke;
  animation: smoky 5s;
}

@keyframes smoky {
  to {
    transform:
      translate3d(200px, -80px, 0)
      rotate(-40deg)
      skewX(70deg)
      scale(1.5);
    text-shadow: 0 0 20px whitesmoke;
    opacity: 0;
  }
}

就能够失去如下成果:

叠加了 transform 之后,就很有一个字被吹跑,变成烟雾的感觉。在此基础之上,咱们只须要将多个字放在一起,利用 animation-delay 顺序控制每个字触发动画成果,即可失去上述的残缺烟雾成果。

伪代码如下:

<div>
    <span>C</span>
    <span>S</span>
    <span>S</span>
    // ...
</div>
// ... 上述所有 CSS 代码

@for $item from 1 through 21 {span:nth-of-type(#{$item}){animation-delay: #{(($item/10))}s; 
  }
} 

就能够失去这样一个被风吹跑的字,幻化成烟雾的成果:

上述成果并非我原创,最早见于这位作者 — CodePen Demo — Smoky Text By Bennett Feely

借助 SVG feturbulence 滤镜实现烟雾成果

上述的烟雾动画的烟雾还是比拟毛糙的。次要是短少了一点颗粒感?短少了一些烟雾的质感。

想要实现更为粗劣的烟雾成果,咱们还得借助 SVG 的 <feturbulence> 滤镜,对这个滤镜还不太理解的,能够看看我的这几篇文章:

  • 有意思!弱小的 SVG 滤镜
  • 震惊!巧用 SVG 滤镜还能制作表情包?
  • 实现一个会动的鸿蒙 LOGO

接下来会应用 filter: blur() 配合 <feturbulence> 滤镜,失去更为真切的烟雾成果。

举个简略的例子,假如有这样几个字:

<div">SMOKE</div>

简略的 CSS:

div {background: linear-gradient(#fff, #999, #ddd, #888);
    background-clip: text;
}

失去这样几个带渐变色字:

咱们利用 <feturbulence> 滤镜简略解决一下:

<div>SMOKE</div>

<svg width="0">
  <filter id="filter">
    <feTurbulence id="turbulence" type="fractalNoise" baseFrequency=".03" numOctaves="20" />
    <feDisplacementMap in="SourceGraphic" scale="30" />
  </filter>
</svg>

CSS 的中利用 filter: url() 引入该滤镜,这里为了成果更好,我间接在 <body> 上引入了该滤镜:

body {filter: url('#filter');
}
div {background: linear-gradient(#fff, #999, #ddd, #888);
    background-clip: text;
}

咱们的字体就被 <feturbulence> 滤镜 赋予了一种流体的感觉:

这个成果能够说和烟雾成果根本没什么关系,不过只须要再增加一个含糊滤镜,神奇的事件就产生了:

body {filter: url('#filter');
}
div {background: linear-gradient(#fff, #999, #ddd, #888);
    background-clip: text;
    filter: blur(5px);
}

整个成果就霎时 烟雾化 了很多:

好,给它增加上循环的动画成果,简略的借助 JavaScript 解决一下:

const filter = document.querySelector("#turbulence");
let frames = 1;
let rad = Math.PI / 180;
let bfx, bfy;

function freqAnimation() {
    frames += .2

    bfx = 0.03;
    bfy = 0.03;

    bfx += 0.005 * Math.cos(frames * rad);
    bfy += 0.005 * Math.sin(frames * rad);

    bf = bfx.toString() + " " + bfy.toString();
    filter.setAttributeNS(null, "baseFrequency", bf);

    window.requestAnimationFrame(freqAnimation);
}

window.requestAnimationFrame(freqAnimation);

看看成果:

上述残缺代码,你能够猛击这里:CodePen CSS + SVG Text Smoke Effect

当然,上述成果能够通过:

  1. 管制 <feTurbulence>baseFrequency 属性调节
  2. 管制 <feTurbulence>numOctaves 属性调节
  3. 管制 <feDisplacementMap>scale 属性调节

<feTurbulence>numOctaves 属性由 30 改成 70,根本就看不到文字的轮廓了,文字整个雾化。咱们能够制作相似这样的 hover 成果:

上述残缺代码,你能够猛击这里:CodePen CSS + SVG Text Smoke Hover Effect

这样,基于 filter: blur() 配合 <feturbulence> 滤镜,咱们能够失去十分真切的烟雾成果,基于上述的演示,咱们还能够再开掘十分多有意思的成果,本文就不再赘述。

最初

好了,本文到此结束,心愿本文对你有所帮忙 :)

想 Get 到最有意思的 CSS 资讯,千万不要错过我的公众号 — iCSS 前端趣闻 😄

更多精彩 CSS 技术文章汇总在我的 Github — iCSS,继续更新,欢送点个 star 订阅珍藏。

如果还有什么疑难或者倡议,能够多多交换,原创文章,文笔无限,满腹经纶,文中若有不正之处,万望告知。

退出移动版