Lazy loaded image
前端
现代 CSS 解决方案:全尺寸的带圆角的渐变边框
字数 2111阅读时长 6 分钟
2024-7-17
2025-4-29
type
status
date
slug
summary
tags
category
icon
password
  • 巧妙实现带圆角的渐变边框[1]
会有这么一个话题的本质在于,在过往,想使用纯 CSS 实现纯粹的,内部透明渐变边框,是一件非常困难的事情,像是这样:
notion image
这个效果的几个核心难点:
  1. 边框带渐变色
  1. 边框支持设置 border-radius
  1. 内部支持透明
思考一下,使用 CSS,我们可以如何实现这个效果?

过往比较好的方法

之前有一个比较接近上面的诉求的方法。主要利用了 clip-path 和 border-image
clip-path[2],大家应该都非常熟悉了。它可以创建一个只有元素的部分区域可以显示的剪切区域。区域内的部分显示,区域外的隐藏。剪切区域是被引用内嵌的URL定义的路径或者外部 SVG 的路径。
简而言之,这里我们只需要在 border-image 的基础上,再利用 clip-path 裁剪出一个带圆角的矩形容器即可:
解释一下:clip-path: inset(0 round 10px) 。
  • clip-path: inset() 是矩形裁剪
  • inset() 的用法有多种,在这里 inset(0 round 10px) 可以理解为,实现一个父容器大小(完全贴合,垂直水平居中于父容器)且 border-radius: 10px 的容器,将这个元素之外的所有东西裁剪掉(即不可见)。
效果如下:
notion image
但是,可以看到上图的 border-width 的值比较大,整个边框的宽度比较粗。
如果我们想得到一条 1px 宽度的渐变边框,我们尝试将上面的边框 CSS 样式修改一下:
border: 10px solid --> border: 1px solid
得到如下效果:
notion image
由于圆角的原因,利用了 clip-path: inset(0 round 10px) 对图形进行切割后,元素的四个圆角都被切割掉了!
所以,有没有一种更好的方式,实现带圆角的渐变边框呢?

使用 mask 和 background-clip 巧妙实现带圆角的渐变边框

这里,我们介绍一种更为巧妙的方法。主要会利用 background-clip 和 mask 两个核心属性。
首先,我们利用背景 background 实现一个普通的渐变背景:
利用角向渐变 background: conic-gradient(#ff00fa, #fe3, #0f3, #ff00fa),我们得到了这么一个图形:
notion image
思考一下,如果我们有办法将图形中间部分镂空裁剪,我们不就能得到一个带圆角的渐变边框了吗?
通过一个示意图,你能很快明白到底是什么意思:
notion image
好,那剩下的问题就转换为了:
  1. 如何裁剪掉一个元素内部的区域,并且能够控制裁剪区域的大小
  1. 裁剪区域,与图形的轮廓是一致的
在 CSS 中想使用裁剪功能,首先想到的肯定是 clip-path,但是上面的例子已经证明了 clip-path 无法实现细边框的裁剪,因此,我们需要另寻解法。
而 CSS 中,另外一个与裁剪功能相关的属性就是 mask
不了解 mask 的,可以戳我的这几篇文章看看:奇妙的 CSS MASK[3]、高阶切图技巧!基于单张图片的任意颜色转换[4]
在此处,我们利用 mask,并且,最为核心的是,需要配合 mask-composite,实现图形轮廓的精确裁剪

深入理解 mask-composite

什么是 mask-composite
mask-composite[5]: 属性指定了将应用于同一元素的多个蒙版图像相互合成的方式。
通俗点来说,他的作用就是,当一个元素存在多重 mask 时,我们就可以运用 -webkit-mask-composite 进行效果叠加。
举个栗子:
我们用一个 radial-gradient 作为 mask,切割原本的矩形,得到一个新的图形。
notion image
如果再换一个方向:
notion image
如果我想得到这样一个效果:
notion image
该怎么做呢?
我们尝试合并上述两个 mask 的效果:
效果如下:
notion image
与我们想象的不太一样,这是因为,两个 mask 的图形叠加,就是上述图形的效果,所以上述效果是没有问题的。
只是,我们想得到的是两个 mask 图形的重叠部分:
notion image
这时,我们就可以使用 mask-composite
添加了 -webkit-mask-composite: source-in 后,我们就可以得到两个 mask 图形的重叠部分,再基于这个重叠部分作用到整个 mask 遮罩:
notion image
CodePen Demo -- mask-composite Demo[6]
  • webkit-mask-composite 还可以实现非常多不同的功能,包括但不限于:
看看这张图,就一目了然(图片源自 CSS mask 实现鼠标跟随镂空效果[7]
notion image

理解 background-clip

要实现最终效果,还有一个有意思的细节点需要掌握。那就是理解 backgrund-clip
一般我们用 backgrund-clip 比较多的场景是 background-clip: text,用于将背景图作用与文字之上。
但是,其实 backgrund-clip 还有几个与 box-content 类似的取值:
什么意思呢?background-clip 设置元素的背景(背景图片或颜色)是否延伸到边框下面。看看下面这张图(图片源自:CSS Background Clip[8]),就是对 background-clip 很好的一个阐述:
notion image
而本文,我们会用到 content-box,举个例子:
效果如下:
notion image
可以看到,此时,背景的填充不再是从元素的右上角开始,而是在内容区域,算上了 10px 的 padding 之后,开始绘制。
也就是说,基于 background-clip,是可以改变元素背景的绘制规则!这一点非常重要。

利用 mask 配合 mask-composite 实现图形轮廓裁剪

只有在掌握了 mask-composite 和 background-clip 的基础上,你才能理解下面整个裁剪代码个核心精髓处。
基于上述讲解,我们就可以利用上面的综合技巧,实现我们最终想要的 -- 带圆角的渐变边框
代码如下:
这样,我们就完美的得到了一个 1px 宽度的带圆角的渐变边框,并且,内部是镂空的
notion image
通过控制上述的 padding: 1px 来控制元素的边框宽度。
完整的代码,你可以戳这里查看:CodePen Demo -- 纯 CSS 实现带圆角的渐变边框![9]
参考资料
[1]
巧妙实现带圆角的渐变边框: https://github.com/chokcoco/iCSS/issues/77
[2]
clip-path: https://developer.mozilla.org/zh-CN/docs/Web/CSS/clip-path
[3]
奇妙的 CSS MASK: https://github.com/chokcoco/iCSS/issues/80
[4]
高阶切图技巧!基于单张图片的任意颜色转换: https://github.com/chokcoco/iCSS/issues/189
[5]
mask-composite: https://developer.mozilla.org/en-US/docs/Web/CSS/-webkit-mask-composite
[6]
CodePen Demo -- mask-composite Demo: https://codepen.io/Chokcoco/pen/KKQjxMP
[7]
CSS mask 实现鼠标跟随镂空效果: https://segmentfault.com/a/1190000040996523
[8]
CSS Background Clip: https://www.programiz.com/css/background-clip
[9]
CodePen Demo -- 纯 CSS 实现带圆角的渐变边框!: https://codepen.io/Chokcoco/pen/JjqJqZR
💡
文章为转载
上一篇
postMessage 实现父子 iframe 通信 (跨源/同源)
下一篇
告别漫长等待:快速删除 node_modules 的正确姿势