深入理解box-shadow的渲染

box-shadow这个属性,目的为框盒子添加阴影,使用起来也不是十分复杂。但是最近在开展单位的前端cop项目时,发现此属性不简单,还可以这样用,而且会发现一些看似复杂好玩的东西竟然是用box-shadow实现的。
网上介绍这个属性的大部分文章都是介绍怎么使用,各个属性值怎么设置会有如何的效果之类的,少有深入探讨,感觉读完之后知其然,却不知所以然,无法清晰融会。于是下来好好研究了一下,总结如下。
使用语法:

    box-shadow: x-shadow y-shadow blur-radius spread-radius color type;

各属性值含义:

  • x-shadow:水平阴影的偏移,当值为正时,阴影往x轴正向偏移,即水平向右;反之,值为负时,阴影往x轴反向偏移,即水平向左
  • y-shadow:垂直阴影的偏移,当值为正时,阴影往y轴正向偏移,即垂直向下;反之,值为负时,阴影往y轴反向偏移,即垂直向上
  • blur-radius:模糊距离,不能为负值;为0表示不模糊,值越大,阴影的边缘就越大,也就越模糊
  • spread-radius:阴影的尺寸,参数可选,不设置为0;正值表示阴影扩展,负值表示阴影反向缩小,可抵消偏移和模糊距离的尺寸
  • color:阴影的颜色,参数可选,不设置便使用浏览器的默认色,因为各浏览器的默认色不同,推荐还是设置一下
  • type:阴影类型,参数可选,不设置默认outset(外部阴影), 还有inset(内部阴影)

1、外部阴影

当type不设置或设置为outset时,是为外部阴影,例如:

box-shadow: 5px 5px 5px 5px #ccc;
box-shadow: 5px 5px 5px 5px #ccc outset;

外部阴影在浏览器渲染时一般是如下几步实现的,

  1. 根据color克隆一个和原始元素相同尺寸的元素“覆盖”其上
  2. 根据spread-radius向四周增加对应颜色的阴影,类似于“边框”
  3. 然后根据指定的x-shadow 和 x-shadow 将克隆出来的元素进行偏移
  4. 根据指定的blur-radius设置模糊半径,一般是依据高斯算法进行模糊处理,本质上是在阴影边缘将阴影色往纯透明色之间进行颜色过渡,所以看到是模糊是逐渐变淡的;而且据了解blur是沿边缘线两边各一半距离,并从里向外扩散
  5. 最后,将克隆元素与原始原属的交集“剪切”去,剩余部分便是最终阴影效果

各步大概的图示如下:

box-shadow-outset.png

为了更加方便的观察这个原理,图中特意设置了透明度,可进入外部阴影观察动态效果
最终的阴影尺寸为:

  • top阴影: spread-radius - y-shadow + blur-radius/2
  • left阴影: spread-radius - x-shadow + blur-radius/2
  • bottom阴影: spread-radius + y-shadow + blur-radius/2
  • right阴影: spread-radius + x-shadow + blur-radius/2

当模糊距离为0,只有spread-radius时,效果相当于border,但这并不是真正的border,盒子模型计算时宽高不会被计算在内

2、内部阴影

当type设置为inset时,是为内部阴影,例如:

box-shadow: 5px 5px 5px 5px #ccc inset;

个人理解,内部阴影在浏览器渲染时一般是如下几步实现的,

  1. 根据color克隆一个比原始元素相同尺寸大的元素“覆盖”其上
  2. 根据spread-radius向四周沿着border向内切割掉部分克隆的元素,留下对应尺寸的spread
  3. 然后根据指定的x-shadow 和 x-shadow 将克隆出来的元素进行偏移
  4. 根据指定的blur-radius设置模糊半径,一般是依据高斯算法进行模糊处理,本质上是在阴影边缘将阴影色往纯透明色之间进行颜色过渡,所以看到是模糊是逐渐变淡并向外扩散的
  5. 最后,将克隆元素在原始原始边框外面的部分“剪切”去,剩余部分便是最终阴影效果

各步大概的图示如下:


box-shadow-inset.png

为了更加方便的观察这个原理,图中特意设置了透明度,可进入内部阴影观察动态效果

最终的阴影尺寸为:

  • top阴影: spread-radius + y-shadow + blur-radius/2
  • left阴影: spread-radius + x-shadow + blur-radius/2
  • bottom阴影: spread-radius - y-shadow + blur-radius/2
  • right阴影: spread-radius - x-shadow + blur-radius/2

3、多个阴影及层级关系

1) 多个阴影
当多个阴影重叠时,声明在前面会覆盖后面的,比如:

box-shadow: 0px 0px 5px 10px blue, 0px 0px 5px 20px red;

其中前面的蓝色阴影会覆盖后面的红色阴影

2) 层级关系
有了框阴影,便有了内外阴影,元素边框,背景和内容等的呈现层级关系,一般为如下层级关系:
border > 内阴影 > background-image > background-color > 外阴影
可以点击此处试一试: 内部阴影

附:几个box-shadow的demo

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,098评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,213评论 2 380
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 149,960评论 0 336
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,519评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,512评论 5 364
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,533评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,914评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,574评论 0 256
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,804评论 1 296
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,563评论 2 319
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,644评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,350评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,933评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,908评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,146评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,847评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,361评论 2 342

推荐阅读更多精彩内容