如何用純 CSS 創作一個圓環旋轉錯覺動畫
阿新 • • 發佈:2018-12-11
效果預覽
線上演示按下右側的“點選預覽”按鈕可以在當前頁面預覽,點選連結可以全屏預覽。
https://codepen.io/comehope/pen/oPWJNj/
可互動視訊
此視訊是可以互動的,你可以隨時暫停視訊,編輯視訊中的程式碼。
請用 chrome, safari, edge 開啟觀看。
https://scrimba.com/p/pEgDAM/cbvPWHM
原始碼下載
本地下載每日前端實戰系列的全部原始碼請從 github 下載:
https://github.com/comehope/front-end-daily-challenges
程式碼解讀
定義 dom,容器中包含 10 個 <div>
子元素,每個 <div>
子元素內還有一個 <span>
子元素:
<figure class="container"> <div><span></span></div> <div><span></span></div> <div><span></span></div> <div><span></span></div> <div><span></span></div> <div><span></span></div> <div><span></span></div> <div><span></span></div> <div><span></span></div> <div><span></span></div> </figure>
定義容器尺寸:
.container {
width: 17em;
height: 17em;
font-size: 16px;
}
定義子元素的尺寸,和容器相同:
.container {
position: relative;
}
.container div {
position: absolute;
width: inherit;
height: inherit;
}
在子元素的正中畫一個黃色的小方塊:
.container div { display: flex; align-items: center; justify-content: center; } .container span { position: absolute; width: 1em; height: 1em; background-color: yellow; }
增加讓小方塊左右移動的動畫效果,動畫時長還會在後面用到,所以定義成變數:
.container span {
--duration: 2s;
animation: move var(--duration) infinite;
}
@keyframes move {
0%, 100% {
left: calc(10% - 0.5em);
}
50% {
left: calc(90% - 0.5em);
}
}
用貝賽爾曲線調整動畫的時間函式,使小方塊看起來就像在左右兩側跳來跳去:
.container span {
animation: move var(--duration) cubic-bezier(0.6, -0.3, 0.7, 0) infinite;
}
增加小方塊變形的動畫,使它看起來有下蹲起跳的擬人效果:
.container span {
animation:
move var(--duration) cubic-bezier(0.6, -0.3, 0.7, 0) infinite,
morph var(--duration) ease-in-out infinite;
}
@keyframes morph {
0%, 50%, 100% {
transform: scale(0.75, 1);
}
25%, 75% {
transform: scale(1.5, 0.5);
}
}
至此,完成了 1 個方塊的動畫。接下來設定多個方塊的動畫效果。
為子元素定義 CSS 下標變數:
.container div:nth-child(1) { --n: 1; }
.container div:nth-child(2) { --n: 2; }
.container div:nth-child(3) { --n: 3; }
.container div:nth-child(4) { --n: 4; }
.container div:nth-child(5) { --n: 5; }
.container div:nth-child(6) { --n: 6; }
.container div:nth-child(7) { --n: 7; }
.container div:nth-child(8) { --n: 8; }
.container div:nth-child(9) { --n: 9; }
旋轉子元素,使小方塊分佈均勻地在容器的四周,圍合成一個圓形:
.container div {
transform: rotate(calc(var(--n) * 40deg));
}
設定動畫延時,現在看起來就像是一群小方塊貼著一個圓的內邊線在旋轉了(但實際上沒有任何元素在做旋轉運動,大腦感覺到的旋轉是一種錯覺):
.container span {
animation-delay: calc(var(--duration) / 9 * var(--n) * -1);
}
最後,為小方塊上色:
.container span {
background-color: hsl(calc(var(--n) * 80deg), 100%, 70%);
}
大功告成!
原文地址:https://segmentfault.com/a/1190000016271648