常常能看到有关 CSS 绘图的文章,譬如使用纯 HTML + CSS 绘制一幅哆啦 A 梦图画。实现的方式就是经过堆叠 div,一步一步实现图画中的一块一块。这种技巧自己没有什么问题,可是就是少了一些难度,只须要有耐心,不少图形仍是可以被慢慢实现出来的。html
基于 CSS 绘图的这个需求,逐渐又有了新的一个流派,单标签实现图形,也就是说,一个复杂的图形只借由一个标签完成,这个相对于可以无限使用标签,不断堆叠 div 来讲,无疑难度上升了不少,也要求对 CSS 有着更深入的理解。前端
譬以下面这个图形,就是由一个 div 元素完成,源自于 A Single Div:git
本文就将介绍一些使用单标签绘图的技巧,而且使用这些技巧,借用单个标签去实现一些复杂图形~😅github
虽说是一个标签,可是几乎全部打着单标签实现图形标题的例子,其中都使用了 3 个元素。这就是单标签实现图形上最为核心的一部分:动画
咱们除了元素自己的样式可以控制以外,还有元素的两个伪元素 -- ::before
,::after
,实际上一共是 3 个元素。网站
好,譬以下面这个心形图形,只能使用一个 div 实现它,该怎么作呢:spa
这种不规则的图形自己使用纯 CSS 是比较复杂的,一般会借助 SVG,固然在 CSS 中就是使用 clip-path
。不过仔细观察图形,咱们不须要 clip-path
,尝试将图片分红 3 部分:code
Wow,其实这里,咱们只须要元素自己实现正方形,元素的两个伪元素利用绝对定位实现两个圆形,叠加在一块儿便可!完整的代码也很是简单:orm
div { position: relative; transform: rotate(45deg); background: rgba(255, 20, 147, 0.85); width: 140px; height: 140px; } div::before, div::after { content: ""; position: absolute; top: 0; left: -70px; width: 140px; height: 140px; border-radius: 50%; background: rgb(255, 20, 147); } div::before { top: -70px; left: 0; }
完整的示例代码,你能够戳这里 CodePen Demo -- A Signle Div heartShapehtm
绝不夸张的说,渐变是在单标签实现图形中,使用的最多的一个 CSS 属性。
缘由就在于咱们渐变是能够多重渐变的!渐变不只仅只能是单个的 linear-gradient 或者单个的 radial-gradient,对于 background 而言,它是支持多重渐变的叠加的,一点很是重要。
好,咱们来看看这个太极图:
其实太极图就是由多个不一样颜色的圆组成,这里堆叠多个不一样的 div,而且把他们组合在一块儿确定是 OK 的。可是咱们的目标是使用单个标签完成。
当图形全是圆或者线条,就应该考虑使用多重线性(径向)渐变了,咱们能够将上图拆解一下。
它实际上是由 1 个线性渐变加上 4 个径向渐变生成的圆组成:
因此,一个太极图完整的代码只须要一个 div 便可,甚至都不须要伪元素的辅助:
div { width: 200px; height: 200px; border-radius: 50%; background-image: radial-gradient(#000 12.5px, transparent 12.5px), radial-gradient(#fff 12.5px, transparent 12.5px), radial-gradient(#fff 50px, transparent 50px), radial-gradient(#000 50px, transparent 50px), linear-gradient(90deg, #000 100px, #fff 100px); background-position: center 50px, center -50px, center 50px, center -50px, 0 0; }
完整的示例代码,你能够戳这里 CodePen Demo -- A Single Div PURE CSS Tai Chi
与渐变很是相似的一个属性就是阴影 box-shadow
,box-shadow
属性它的一个特色也是能够叠加多层的,能够内置多条阴影规则,它简直就是单标签绘图的终极大杀器!
咱们尝试使用一个 div 实现以下图形:
乍一看,这个图形其实仍是很复杂的,云朵、雨滴都不像是仅仅用一个标签或者一个伪元素可以实现的。
实则否则,首先咱们看看这个云朵,虽然带有不规则的轮廓,可是实际上就是一个一个的圆。很是适合使用多重径向渐变或者是多重阴影!
其实就是一个实现圆,而后利用阴影实现多个圆的叠加,示例动画,一看就懂:
代码量其实也很是少,实现一个云朵的代码:
div{ width:100px; height:100px; background:#fff; border-radius:50%; box-shadow: 120px 0px 0 -10px #fff, 95px 20px 0 0px #fff, 30px 30px 0 -10px #fff, 90px -20px 0 0px #fff, 40px -40px 0 0px #fff; }
CodePen Demo -- A Single Div Cloudy
与云朵的示例代码相似,雨滴其实也是借助了多重阴影实现:
div { position: absolute; width: 3px; height: 6px; border-radius: 50%; animation: rainy_rain 0.7s infinite linear; box-shadow: rgba(0, 0, 0, 0) -10px 30px, rgba(0, 0, 0, 0) 40px 40px, rgba(0, 0, 0, 0.3) -50px 75px, rgba(0, 0, 0, 0.3) 55px 50px, rgba(0, 0, 0, 0.3) -18px 100px, rgba(0, 0, 0, 0.3) 12px 95px, rgba(0, 0, 0, 0.3) -31px 45px, rgba(0, 0, 0, 0.3) 30px 35px; } @keyframes rainy_rain { 0% { box-shadow: rgba(0, 0, 0, 0) -10px 30px, rgba(0, 0, 0, 0) 40px 40px, rgba(0, 0, 0, 0.3) -50px 75px, rgba(0, 0, 0, 0.3) 55px 50px, rgba(0, 0, 0, 0.3) -18px 100px, rgba(0, 0, 0, 0.3) 12px 95px, rgba(0, 0, 0, 0.3) -31px 45px, rgba(0, 0, 0, 0.3) 30px 35px; } // 省略部分阴影位移帧动画代码 ... 100% { box-shadow: rgba(0, 0, 0, 0) -10px 120px, rgba(0, 0, 0, 0) 40px 120px, rgba(0, 0, 0, 0.3) -50px 75px, rgba(0, 0, 0, 0.3) 55px 50px, rgba(0, 0, 0, 0.3) -18px 100px, rgba(0, 0, 0, 0.3) 12px 95px, rgba(0, 0, 0, 0.3) -31px 45px, rgba(0, 0, 0, 0.3) 30px 35px; } }
刚刚已经使用了元素自己和元素的一个伪元素,剩余一个伪元素实现底部的阴影圆便可,完整的 Demo 代码你能够戳这里:A Signle Div Rainy
到这里,能够简单总结一下,单标签实现图形,尤为是复杂图形,很大程度上都是借助了上述的 3 个技巧,也就是:
::before
和 ::after
咱们练习一下,使用单个 div 实现下面这个美队盾牌:
有了上面的铺垫,其实多重的圆形使用多重径向渐变和多重阴影都是都是能够的,而中间的星星,使用字符或者 clip-path
也能很是轻松的实现:
div { position: absolute; width: 200px; height: 200px; background: radial-gradient( at center, #0033b0 20%, #ce0021 20%, #ce0021 35%, #eee 35%, #eee 55%, #ce0021 55% ); border-radius: 50%; } div::before { content: "★"; position: absolute; top: 50%; left: 50%; transform: translate(-50%, -50%); line-height: 47px; font-size: 55px; }
咱们会获得这样一个图形:
感受图形少了一些光泽,咱们能够往 div 上继续叠加一些 linear-gradient
,给盾牌表面添加一些高光:
div { position: absolute; width: 200px; height: 200px; background: linear-gradient(45deg, rgba(255, 255, 255, 0) 35%, rgba(255, 255, 255, 0.4) 50%, rgba(255, 255, 255, 0) 65%), linear-gradient(-45deg, rgba(255, 255, 255, 0) 35%, rgba(255, 255, 255, 0.4) 50%, rgba(255, 255, 255, 0) 65%), linear-gradient(to right, rgba(0, 0, 0, 0) 35%, rgba(0, 0, 0, 0.2) 50%, rgba(0, 0, 0, 0) 65%), linear-gradient(to bottom, rgba(0, 0, 0, 0) 35%, rgba(0, 0, 0, 0.2) 50%, rgba(0, 0, 0, 0) 65%), radial-gradient( ellipse at center, #0033b0 20%, #ce0021 20%, #ce0021 35%, #eee 35%, #eee 55%, #ce0021 55% ); border-radius: 50%; } div::before { content: "★"; position: absolute; top: 50%; left: 50%; transform: translate(-50%, -50%); line-height: 47px; font-size: 55px; }
OK,便能完美的实现:
完整的代码你能够戳这里:A Signle Div Shield
咱们再看看这个图形,一个磁带图形:
看着很复杂,其实都是圆和各类线条,其实也是适合使用单个标签实现的,就是很是的花时间,须要精细的控制 background-image
里面的每一个渐变的 background-size
、background-position
:
首先,借由多重渐变,实现整个背景结构:
div { width: 180px; height: 120px; border-radius: 5px; background-image: linear-gradient(to right, #444 10px, transparent 10px), linear-gradient(to left, #444 10px, transparent 10px), linear-gradient(135deg, #444 20px, transparent 20px), linear-gradient(-135deg, #444 20px, transparent 20px), linear-gradient( to bottom, transparent 35px, #be0974 35px, #be0974 43px, #da6a57 43px, #da6a57 51px, #eebc31 51px, #eebc31 59px, #92a25b 59px, #92a25b 67px, #46a7c0 67px, #46a7c0 75px, transparent 75px ), linear-gradient( to bottom, transparent 10px, #f7f7f7 10px, #f7f7f7 85px, transparent 85px ), linear-gradient(to top, transparent 26px, #444 26px), linear-gradient( 105deg, #444 70px, #333 70px, #333 73px, transparent 73px ), linear-gradient( -105deg, #444 70px, #333 70px, #333 73px, transparent 73px ), linear-gradient(to top, #444 24px, #777 24px, #777 26px, #444 26px); box-shadow: -4px -4px 2px rgb(0 0 0 / 20%); }
获得以下图形:
经过其中一个伪元素,利用 box-shadow
实现磁带上的各个圆圈点:
div:after { position: absolute; content: ""; width: 5px; height: 5px; background: #999; border-radius: 50%; box-shadow: 165px 0 0 #999, 0 104px 0 #999, 165px 104px 0 #999, 55px 101px 0 1px #222, 68px 98px 0 1px #222, 98px 98px 0 1px #222, 110px 101px 0 1px #222, 51px 38px 0 #444, 114px 38px 0 #444, 44px 46px 0 #444, 58px 46px 0 #444, 107px 46px 0 #444, 121px 46px 0 #444, 51px 53px 0 #444, 114px 53px 0 #444, 51px 46px 0 6px #ccc, 114px 46px 0 6px #ccc; left: 5px; top: 5px; }
最后剩下的一个伪元素,实现磁带中间的部分样式便可:
div:before { position: absolute; content: ""; width: 90px; height: 26px; margin-left: -45px; left: 50%; top: 41px; background-color: #ccc; background-image: linear-gradient(to bottom, #444 5px, transparent 5px), linear-gradient(to top, #444 5px, transparent 5px), linear-gradient(to right, #444 30px, transparent 30px), linear-gradient(to left, #444 30px, transparent 30px), radial-gradient(circle at 10px 12px, #a0522d 32px, transparent 32px); border-radius: 30px; }
这样,就顺利使用单个标签实现啦,该 Demo 取自 A Single Div,完整的代码你能够戳这里:CodePen Demo -- A single Div Disk。
固然,单标签能实现的远不止如此,看看下面这些,都是一个 div 可以实现的:
固然,上述的做图都仍是比较常规的,借助伪元素,使用 background
、使用 box-shadow
。咱们还能够尝试在一个 div 内增长混合模式 mix-blend-mode
、滤镜 filter
以及 遮罩 mask
等,实现一些更为有意思的效果。
譬以下述这个效果,使用了一个 div 实现的幽灵效果:
在用一个 div 实现基本效果之余,还加上了利用了 filter
滤镜实现了一些融合效果。
完整的代码你能够戳这里:CodePen Demo -- A Single Div Ghost
只使用 CSS 进行单 div 绘图仍是很是有意思的,也能够比较好的锻炼 CSS,虽然业务中不必定会用上 :)
这里再推荐几个单标签绘图的网站,你能够看看再模仿模仿:
好了,本文到此结束,但愿对你有帮助 :)
想 Get 到最有意思的 CSS 资讯,千万不要错过个人公众号 -- iCSS前端趣闻 😄
更多精彩 CSS 技术文章汇总在个人 Github -- iCSS ,持续更新,欢迎点个 star 订阅收藏。
若是还有什么疑问或者建议,能够多多交流,原创文章,文笔有限,才疏学浅,文中如有不正之处,万望告知。