最近利用 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 订阅珍藏。

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