效果预览

在线演示

按下右侧的“点击预览”按钮可以在当前页面预览,点击链接可以全屏预览。

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

如何用纯 CSS 和 D3 创作一艘遨游太空的宇宙飞船的更多相关文章

  1. 前端每日实战:96# 视频演示如何用纯 CSS 和 D3 创作一艘遨游太空的宇宙飞船

    效果预览 按下右侧的"点击预览"按钮可以在当前页面预览,点击链接可以全屏预览. https://codepen.io/comehope/pen/oMqNmv 可交互视频 此视频是可 ...

  2. 如何用纯 CSS 和 D3 创作一只扭动的蠕虫

    效果预览 在线演示 按下右侧的"点击预览"按钮可以在当前页面预览,点击链接可以全屏预览. https://codepen.io/comehope/pen/QBQJMg 可交互视频 ...

  3. 如何用纯 CSS 为母亲节创作一颗像素画风格的爱心

    效果预览 在线演示 按下右侧的"点击预览"按钮可以在当前页面预览,点击链接可以全屏预览. https://codepen.io/comehope/pen/LmrZVX 可交互视频教 ...

  4. 前端每日实战:20# 视频演示如何用纯 CSS 为母亲节创作一颗像素画风格的爱心

    效果预览 按下右侧的"点击预览"按钮可以在当前页面预览,点击链接可以全屏预览. https://codepen.io/comehope/pen/LmrZVX 可交互视频教程 此视频 ...

  5. 如何用纯CSS布局两列,一列固定宽度,另一列自适应?

    大家都知道好多网站都是左右布局的,很多公司在笔试和面试环节也常常问这个问题.一个去网易的师兄说14年腾讯面试的时候问过这个问题,网易在笔试和面试时候也问过这个问题,还有很多互联网公司也都涉及到这个问题 ...

  6. 如何用 CSS 和 D3 创作一个无尽的六边形空间

    效果预览 按下右侧的"点击预览"按钮可以在当前页面预览,点击链接可以全屏预览. https://codepen.io/comehope/pen/NBvrWL 可交互视频 此视频是可 ...

  7. 如何用 CSS 和 D3 创作旋臂粒子动画

    效果预览 在线演示 按下右侧的"点击预览"按钮可以在当前页面预览,点击链接可以全屏预览. https://codepen.io/comehope/pen/xJrOqd 可交互视频 ...

  8. 如何用 CSS 和 D3 创作火焰动画

    效果预览 在线演示 按下右侧的"点击预览"按钮可以在当前页面预览,点击链接可以全屏预览. https://codepen.io/comehope/pen/xJdVxx 可交互视频 ...

  9. 前端每日实战:89# 视频演示如何用 CSS 和 D3 创作旋臂粒子动画

    效果预览 按下右侧的"点击预览"按钮可以在当前页面预览,点击链接可以全屏预览. https://codepen.io/comehope/pen/xJrOqd 可交互视频 此视频是可 ...

随机推荐

  1. Jmeter集成Jira提交缺陷

    笔者曾在文章<Jmeter排忧解难—生成excel结果文件>聊到了一种提高接口测试效率的方法.今天,咱们接着对“提高接口测试效率”这个话题做更深入的探讨.作为一名接口测试人员,我们是否一直 ...

  2. vim如何删除行首、行位空格、空格行

    删除空格行: 非编辑状态下输入:g/^$/d 删除行首空格: 非编辑状态下输入:%s/^\s*//g 删除行尾空格: 非编辑状态下输入:%s/\s*$//g

  3. Serilog 是 ASP.NET Core 的一个插件,可以简化日志记录

    [翻译] ASP.NET Core 利用 Docker.ElasticSearch.Kibana 来记录日志 原文: Logging with ElasticSearch, Kibana, ASP.N ...

  4. Net Core2.0 基于QuartzNet任务管理系统

    Net Core2.0 基于QuartzNet任务管理系统 Quartz.NET官网地址:https://www.quartz-scheduler.net/ Quartz.NET文档地址:https: ...

  5. shell 发送Post请求,并获取状态码

    #!/bin/bash aa=$ result=$(curl -H "Content-type: application/json" -X POST -o /dev/null -s ...

  6. 在Unity3d中解析Lua脚本的方法

    由于近期项目中提出了热更新的需求,因此本周末在Lua的陪伴下度过.对Lua与Unity3d的搭配使用,仅仅达到了一个初窥门径的程度,记录一二于此.水平有限,欢迎批评指正. 网络上关于Lua脚本和Uni ...

  7. C# List的使用

    1.所需引入的命名空间: using System.Collections.Generic; 2.初始化 [1]空: List<int> list = new List<int> ...

  8. Failed to crunch file

    Failed to crunch file 编译时,出现以上错误,经过多次排除验证,原因尽然是因为路径字符太长了... 编译路径不能超过240个字符

  9. JFinal免费公开课更新中

    价值千元的课程,免费报名学习,JFinal学院-小木 录制JFinal视频教程,JFinal核心已经周边涉及到微信小程序开发.数据库.前端实战等.

  10. node执行cmd命令方法

    var cmd='tasklist';//获取 子进程模块的exec方法,用于执行cmd命令var exec = require('child_process').exec; //运行 定义的cmd命 ...