最近利用 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
当然,上述成果能够通过:
- 管制
<feTurbulence>
的baseFrequency
属性调节 - 管制
<feTurbulence>
的numOctaves
属性调节 - 管制
<feDisplacementMap>
的scale
属性调节
将 <feTurbulence>
的 numOctaves
属性由 30 改成 70,根本就看不到文字的轮廓了,文字整个雾化。咱们能够制作相似这样的 hover 成果:
上述残缺代码,你能够猛击这里:CodePen CSS + SVG Text Smoke Hover Effect
这样,基于 filter: blur()
配合 <feturbulence>
滤镜,咱们能够失去十分真切的烟雾成果,基于上述的演示,咱们还能够再开掘十分多有意思的成果,本文就不再赘述。
最初
好了,本文到此结束,心愿本文对你有所帮忙 :)
想 Get 到最有意思的 CSS 资讯,千万不要错过我的公众号 — iCSS 前端趣闻 😄
更多精彩 CSS 技术文章汇总在我的 Github — iCSS,继续更新,欢送点个 star 订阅珍藏。
如果还有什么疑难或者倡议,能够多多交换,原创文章,文笔无限,满腹经纶,文中若有不正之处,万望告知。