更优雅地绘制阴影

作者头像
周骅 , 
• 1033个字 • 4分钟读完
hero

Box-shadow 虽然是一个 css3 的属性,但由于浏览器支持不错,且用它来营造一种立体感、层次感着实方便,这让它成为了互联网上随处可见的 css3 特效。不过我感觉想写好阴影不是一件容易的事情。至少我常常摸索半天,写出来的阴影却总让人很难受。 上周在知乎上看到了一个问答,很受启发:如何理解 Material Design 中卡片的两层阴影,于是特意去看了 Meterial Design 的设计准则(中文翻译),觉得其中的一些设计思想和细节追求很值得我们去借签。

本文标题是“更优雅地绘制阴影”,但其实我并不懂“优雅”是什么,只是觉得别人的做法比我的好,比我的优雅。那么就来看看别人是怎么理解阴影的。

Meterial Design 把 App 的部件、面板、模态框等都抽象成了卡片。对于一个个上面,除了有 x/y 轴坐标的定义,还增加了 z 轴。处于不同层级的卡片,有着相对独立的交互逻辑。可以说,卡片和层级就是一个 app 页面的交互隐喻。而层次的视觉定义就是通过阴影来完成。下面搬运两张图:

layout-principles-dimensionality-shadows-01_large_mdpi layout-principles-dimensionality-shadows-08_large_mdpi

以上的阴影定义虽然是 AI 的格式,但很容易就能转换成前端的语言:

.z-index-1 {
  box-shadow:
    0 1px 1.5px rgba(0, 0, 0, 0.12),
    0 1px 1px rgba(0, 0, 0, 0.24);
}
.z-index-2 {
  box-shadow:
    0 3px 3px 0 rgba(0, 0, 0, 0.16),
    0 3px 3px 0 rgba(0, 0, 0, 0.23);
}
.z-index-3 {
  box-shadow:
    0 10px 10px rgba(0, 0, 0, 0.19),
    0 6px 3px rgba(0, 0, 0, 0.23);
}
.z-index-4 {
  box-shadow:
    0 14px 14px rgba(0, 0, 0, 0.25),
    0 10px 5px rgba(0, 0, 0, 0.22);
}
.z-index-5 {
  box-shadow:
    0 19px 19px rgba(0, 0, 0, 0.3),
    0 15px 6px rgba(0, 0, 0, 0.22);
}

看看效果:

嗯,确实比我自己实现的感觉要优雅。

先插几句话,box-shadow 属性很有意思,它允许有若干重属性值,效果是相互叠加。张鑫旭有一篇博客谈这个问题(传送门),写得很好,大家可以参考一下。

这样的阴影效果大家自行感受,我觉得不管怎么样,好过我自己编造、瞎调出来的。但为什么要使用双重阴影?官方解释是用两个光源来模拟现实场景,一个是关键光,一个是环境光,所以会产生两个阴影。另外 @jordanfc 迟方的回答 很棒,但很多是结论性的东西,我们需要更多一点思考。


原文中大量内容不够严谨,现全部删除。