1. 程式人生 > >如何用純 CSS 和 D3 創作一艘遨遊太空的宇宙飛船

如何用純 CSS 和 D3 創作一艘遨遊太空的宇宙飛船

在這裡插入圖片描述

效果預覽

線上演示

按下右側的“點選預覽”按鈕可以在當前頁面預覽,點選連結可以全屏預覽。

https://codepen.io/comehope/pen/oMqNmv

可互動視訊

此視訊是可以互動的,你可以隨時暫停視訊,編輯視訊中的程式碼。

請用 chrome, safari, edge 開啟觀看。

https://scrimba.com/p/pEgDAM/cm48rta

原始碼下載

本地下載

每日前端實戰系列的全部原始碼請從 github 下載:

https://github.com/comehope/front-end-daily-challenges

程式碼解讀

定義 dom,spacecraft 表示飛船,容器中包含 1 個表示尾冀的元素 fins

<div class="spacecraft">
    <div class="fins"></div>
</div>

居中顯示:

body {
    margin: 0;
    height: 100vh;
    display: flex;
    align-items: center;
    justify-content: center;
    background: linear-gradient(black, midnightblue);
}
```

畫出飛船的船艙:

.spacecraft {
    width: 7em;
    height: 11em;
    font-size: 16px;
    background: 
        linear-gradient(whitesmoke, darkgray);
    border-radius: 50% / 70% 70% 5% 5%;
}
```

用偽元素畫出飛船尾部的火焰:

.spacecraft::before {
    content: '';
    position: absolute;
    width: 6em;
    height: 2em;
    background-color: #444;
    border-radius: 20%;
    top: 10em;
    left: 0.5em;
    z-index: -1;
}

.spacecraft::after {
content: ‘’;
position: absolute;
box-sizing: border-box;
width: 4em;
height: 4em;
background: gold;
top: 10em;
left: 1.5em;
border-radius: 80% 0 50% 45% / 50% 0 80% 45%;
transform: rotate(135deg);
border: 0.5em solid orange;
z-index: -2;
}


<p>畫出飛船兩側的尾冀:</p>
<pre class="brush:css">.fins::before,
.fins::after {
    content: '';
    position: absolute;
    width: 2em;
    height: 6em;
    background: linear-gradient(tomato, darkred);
    top: 7em;
}

.fins::before {
    left: -2em;
    border-radius: 3em 0 50% 100%;
}

.fins::after {
    right: -2em;
    border-radius: 0 3em 100% 50%;
}

用徑向漸變畫出飛船的舷窗:

.spacecraft {
    background: 
        radial-gradient(
            circle at 3.5em 5em,
            transparent 1.5em,
            lightslategray 1.5em, lightslategray 2em,
            transparent 2em
        ),
        radial-gradient(
            circle at 3.3em 5.2em,
            deepskyblue 1.4em,
            transparent 1.6em
        ),
        radial-gradient(
            circle at 3.5em 5em,
            white 1.5em,
            transparent 1.5em
        ),
        linear-gradient(whitesmoke, darkgray);
}
```

增加飛船火焰噴射的動畫效果:

.spacecraft::after {
    animation: flame-spout 0.3s infinite;
}

@keyframes flame-spout {
0%, 100% {
filter: opacity(0.1);
}

50% {
    filter: opacity(1);
}

}


<p>接下來畫星空。<br>在 dom 中增加 <code>stars</code> 容器,其中包含表示星星的 4 個子元素:</p>

<div class=“stars”>
<span></span>
<span></span>
<span></span>
<span></span>
</div>
<div class=“rocket”>
<div class=“fins”></div>
</div>


<p>定義星星的樣式:</p>
<pre class="brush:css">.stars span {
    position: absolute;
    width: 2px;
    height: 8px;
    border-radius: 50%;
    background-color: white;
    top: calc(50% - 7em);
}

用變數使星星分佈在水平方向的不同位置:

.stars span {
    left: calc(var(--left) * 1vw);
}

.stars span:nth-child(1) {
–left: 20;
}

.stars span:nth-child(2) {
–left: 40;
}

.stars span:nth-child(3) {
–left: 60;
}

.stars span:nth-child(4) {
–left: 80;
}


<p>用變數設定星星的尺寸和不透明度,使每顆星星看起來稍有差異:</p>
<pre class="brush:css">.stars span {
    width: calc(var(--size) * 1px);
    height: calc(var(--size) * 4px);
    filter: opacity(var(--opacity));
}

.stars span:nth-child(1) {
    --size: 0.8;
    --opacity: 0.5;
}

.stars span:nth-child(2) {
    --size: 1.25;
    --opacity: 0.6;
}

.stars span:nth-child(3) {
    --size: 1.5;
    --opacity: 0.7;
}

.stars span:nth-child(4) {
    --size: 2;
    --opacity: 0.8;
}

定義星星從太空中飄過的動畫效果:

.stars span {
    top: -5vh;
    animation: star-move linear infinite;
}

@keyframes star-move {
to {
top: 100vh;
}
}


<p>用變數設定動畫的時長和延時時間:</p>
<pre class="brush:css">.stars span {
    animation-duration: calc(var(--duration) * 1s);
    animation-delay: calc(var(--delay) * 1s);
}

.stars span:nth-child(1) {
    --duration: 1;
    --delay: -0.05;
}

.stars span:nth-child(2) {
    --duration: 1.5;
    --delay: -0.1;
}

.stars span:nth-child(3) {
    --duration: 2;
    --delay: -0.15;
}

.stars span:nth-child(4) {
    --duration: 2.5;
    --delay: -0.2;
}

隱藏螢幕外的內容:

body {
    overflow: hidden;
}
```

接下來用 d3 批量處理表示星星的 dom 元素和 css 變數。
引入 d3 庫:

&lt;script src="https://d3js.org/d3.v5.min.js"&gt;&lt;/script&gt;

用 d3 建立表示星星的 dom 元素:

const COUNT_OF_STARS = 4;

d3.select('.stars')
    .selectAll('span')
    .data(d3.range(COUNT_OF_STARS))
    .enter()
    .append('span');

用 d3 為 css 變數 --left, --size, --opacity 賦值,--left 的取值範圍是 1 到 100,--size 的取值範圍是 1 到 2.5,'--opacity' 的取值範圍是 0.5 到 0.8:

d3.select('.stars')
    .selectAll('span')
    .data(d3.range(COUNT_OF_STARS))
    .enter()
    .append('span')
    .style('--left', () =&gt; Math.ceil(Math.random() * 100))
    .style('--size', () =&gt; Math.random() * 1.5 + 1)
    .style('--opacity', () =&gt; Math.random() * 0.3 + 0.5);

用 d3 為 css 變數 --duration--delay 賦值,--duration 的取值範圍是 1 到 3,--delay 的取值是依次減少 0.05:

d3.select('.stars')
    .selectAll('span')
    .data(d3.range(COUNT_OF_STARS))
    .enter()
    .append('span')
    .style('--left', () =&gt; Math.ceil(Math.random() * 100))
    .style('--size', () =&gt; Math.random() * 1.5 + 1)
    .style('--opacity', () =&gt; Math.random() * 0.3 + 0.5)
    .style('--duration', () =&gt; Math.random() * 2 + 1)
    .style('--delay', (d) =&gt; d * -0.05);

刪除掉 html 檔案中相關的 dom 宣告和 css 檔案中的變數宣告。

最後,把星星的數量增加到 30 顆:

const COUNT_OF_STARS = 30;

大功告成!

原文地址:https://segmentfault.com/a/1190000015853738