如上圖所示,在許多的活動宣傳html5中會常常須要用到這樣的一個動畫效果。特別是快到年末了,也許有同窗正在爲了公司的活動頁面而忙碌,get到這樣一個小技能說不定恰好對你有幫助哦。css
在css3中,咱們使用animation與keyframes結合,能夠給元素添加各類各樣的動畫效果。具體的動畫,在keyframes中定義,在animation中使用。例如能夠定義一個從上飛入的動畫效果。html
@keyframes topIn {
from { transform: translateY(-50px) }
to { transform: translateY(0px) }
}
複製代碼
並在目標元素中經過animation來使用動畫。html5
<div class="target topIn"></div>
複製代碼
.topIn {
animation: topIn 1s ease;
}
複製代碼
這樣,當元素第一次渲染進入DOM時,就會有一個從上到下的位移動畫效果。固然,這種效果並非咱們想要的。每每咱們還在在動畫上加上一個透明度從0到1的漸變。node
@keyframes topIn {
from {
transform: translateY(-50px);
opacity: 0;
}
to {
transform: translateY(0px);
opacity: 1;
}
}
複製代碼
咱們還但願可以控制元素的顯示時機應該怎麼辦?簡單一點的辦法就是在須要動畫效果展現時,纔給目標元素添加控制動畫的class樣式。css3
btn.addEventListener('click', function() {
document.querySelector('.target').classList.add('topIn');
}, !1);
複製代碼
可是這樣作有一個問題。我相信實踐過的朋友都已經發現過的。咱們指望元素在入場以前,是處於看不見的狀態。可是僅僅只是上面的作法,動畫開始前元素是可以被看見的。那麼應該怎麼辦?app
咱們能夠很簡單的想到,給元素添加display: none
或者visibility: hidden
。可是因爲display: none
以後,元素是不佔位的。所以若是這樣的話,會致使頁面佈局出現混亂。因此咱們在開始以前,給元素添加一個新的class。ide
.aninode {
visibility: hidden;
}
複製代碼
而且添加一個新的class讓元素顯示出來。佈局
.animated .aninode {
visibility: visible;
}
複製代碼
控制動畫效果的class也在css上進行一些調整。動畫
.animated .topIn {
animation: topIn 1s ease;
}
複製代碼
這樣作的好處是,咱們只須要在class中添加一個animated
,就可以達到咱們的效果。實例demo完整代碼以下:ui
<div class="container">
<div class="target aninode leftIn"></div>
<button class="btn show">show</button>
<button class="btn hide">hide</button>
</div>
複製代碼
.container {
width: 100px;
margin: 0 auto;
}
.aninode {
visibility: hidden;
}
.animated .aninode {
visibility: visible;
}
.target {
width: 100px;
height: 100px;
background: orange;
border-radius: 4px;
margin: 20px 0;
}
.animated .topIn {
animation: topIn 1s ease;
}
.animated .leftIn {
animation: leftIn 1s ease;
}
.btn {
width: 100px;
height: 30px;
border: 1px solid #ccc;
outline: none;
transition: 0.1s;
}
.btn:active {
border: none;
background: orange;
color: #fff;
}
@keyframes topIn {
from {
transform: translateY(-50px);
opacity: 0;
}
to {
transform: translateY(0px);
opacity: 1;
}
}
@keyframes leftIn {
from {
transform: translateX(-50px);
opacity: 0;
}
to {
transform: translateX(0px);
opacity: 1;
}
}
複製代碼
var show = document.querySelector('.show');
var hide = document.querySelector('.hide');
var container = document.querySelector('.container');
show.addEventListener('click', function() {
container.classList.add('animated');
}, !1);
hide.addEventListener('click', function() {
container.classList.remove('animated');
}, !1);
複製代碼
Demo顯示以下:
See the Pen <a href='https://codepen.io/yangbo5207/pen/NXKrPg/'>NXKrPg</a> by Ormie (<a href='https://codepen.io/yangbo5207'>@yangbo5207</a>) on <a href='https://codepen.io'>CodePen</a>.可是這樣離咱們想要的效果好像還差一點點。繼續思考。首先想要後面的元素比前一個元素晚一點出現,那麼確定是要控制延遲時間,咱們就必須有許多設置延遲時間的class。
.delay200 {
animation-delay: 200ms;
animation-fill-mode: backwards!important;
}
.delay400 {
animation-delay: 400ms;
animation-fill-mode: backwards!important;
}
.delay600 {
animation-delay: 600ms;
animation-fill-mode: backwards!important;
}
.delay800 {
animation-delay: 800ms;
animation-fill-mode: backwards!important;
}
複製代碼
animation-fill-mode: backwards!important;
的目的是爲了元素在出現以前,保持透明度爲0的狀態。防止當添加animated
以後元素直接出現了。
加!important
是爲了防止在新的class中使用animation簡寫時對animation-fill-mode
的屬性進行覆蓋改寫。若是此處不寫!important
的話,那麼在topIn
這樣的動畫class中就不能使用簡寫形式。
這樣以後,咱們只須要在css中添加上上述代碼,並對html作一些改動,就可以實現咱們想要的效果了。
See the Pen <a href='https://codepen.io/yangbo5207/pen/mpbEEE/'>mpbEEE</a> by Ormie (<a href='https://codepen.io/yangbo5207'>@yangbo5207</a>) on <a href='https://codepen.io'>CodePen</a>.完整代碼以下:
<div class="container">
<div class="targets aninode">
<div class="item leftIn">春曉</div>
<div class="item leftIn delay200">春眠不覺曉</div>
<div class="item leftIn delay400">到處蚊子咬</div>
<div class="item leftIn delay600">夜來風雨聲</div>
<div class="item leftIn delay800"><此處請留下大家的才華></div>
</div>
<button class="btn show">show</button>
<button class="btn hide">hide</button>
</div>
複製代碼
.container {
width: 200px;
margin: 0 auto;
}
.aninode {
visibility: hidden;
}
.animated .aninode {
visibility: visible;
}
.targets {
margin: 20px 0;
}
.targets .item {
border: 1px solid #ccc;
margin: 10px 0;
line-height: 2;
padding: 2px 6px;
border-radius: 4px;
}
.animated .topIn {
animation: topIn 1s ease;
}
.animated .leftIn {
animation-name: leftIn;
animation-duration: 1s;
}
.btn {
width: 100px;
height: 30px;
border: 1px solid #ccc;
outline: none;
transition: 0.1s;
}
.btn:active {
border: none;
background: orange;
color: #fff;
}
@keyframes topIn {
from { transform: translateY(-50px) }
to { transform: translateY(0px) }
}
@keyframes leftIn {
from {
transform: translateX(-50px);
opacity: 0;
}
to {
transform: translateX(0px);
opacity: 1;
}
}
.delay200 {
animation-delay: 200ms;
animation-fill-mode: backwards!important;
}
.delay400 {
animation-delay: 400ms;
animation-fill-mode: backwards!important;
}
.delay600 {
animation-delay: 600ms;
animation-fill-mode: backwards!important;
}
.delay800 {
animation-delay: 800ms;
animation-fill-mode: backwards!important;
}
複製代碼
var show = document.querySelector('.show');
var hide = document.querySelector('.hide');
var container = document.querySelector('.container');
show.addEventListener('click', function() {
container.classList.add('animated');
}, !1);
hide.addEventListener('click', function() {
container.classList.remove('animated');
}, !1);
複製代碼
咱們發現js的邏輯並無發生任何改變。仍然僅僅只是在合適的位置添加/刪除animated。
在實踐中咱們還會遇到一個比較麻煩的事兒。就是延遲class的編寫。咱們可能並不知道會使用到那些時差,有多少個元素會使用到,若是都用手來寫的話,重複工做確實太過麻煩。所以咱們可使用js動態插入。代碼以下:
const styleSheet = getSheet();
var delay = 100;
while (delay < 10000) {
styleSheet.insertRule(`.animated .delay${delay}{ animation-delay: ${delay}ms; animation-fill-mode: backwards; }`, styleSheet.cssRules.length);
delay += delay < 3000 ? 100 : 1000;
}
function getSheet() {
var sheets = document.styleSheets;
var len = sheets.length;
for(var i = 0; i <= len; i++) {
var sheet = sheets.item(i);
try {
if (sheet.cssRules) {
return sheet;
}
} catch(e) {}
}
var style = document.createElement('style');
style.type = "text/css";
document.getElementsByTagName('head')[0].appendChild(style);
return style.sheet;
}
複製代碼