现代 CSS 解决方案:全尺寸的带圆角的渐变边框

7,626次阅读
没有评论

共计 4552 个字符,预计需要花费 12 分钟才能阅读完成。

在之前,我们有一篇介绍带圆角的渐变边框的纯 CSS 实现的文章:

会有这么一个话题的本质在于,在过往,想使用纯 CSS 实现纯粹的,内部透明渐变边框,是一件非常困难的事情,像是这样:

现代 CSS 解决方案:全尺寸的带圆角的渐变边框

这个效果的几个核心难点:

  1. 边框带渐变色
  2. 边框支持设置 border-radius
  3. 内部支持透明

思考一下,使用 CSS,我们可以如何实现这个效果?

过往比较好的方法

之前有一个比较接近上面的诉求的方法。主要利用了 clip-pathborder-image

clip-path,大家应该都非常熟悉了。它可以创建一个只有元素的部分区域可以显示的剪切区域。区域内的部分显示,区域外的隐藏。剪切区域是被引用内嵌的 URL 定义的路径或者外部 SVG 的路径。

简而言之,这里我们只需要在 border-image 的基础上,再利用 clip-path 裁剪出一个带圆角的矩形容器即可:

div class="border-image-clip-path">div>
.border-image-clip-path {
    position: relative;
    width: 200px;
    height: 100px;
    border: 10px solid;
    border-image: linear-gradient(45deg, gold, deeppink) 1;
    clip-path: inset(0 round 10px);
}

解释一下:clip-path: inset(0 round 10px)

  • clip-path: inset() 是矩形裁剪
  • inset() 的用法有多种,在这里 inset(0 round 10px) 可以理解为,实现一个父容器大小(完全贴合,垂直水平居中于父容器)且 border-radius: 10px 的容器,将这个元素之外的所有东西裁剪掉(即不可见)。

效果如下:

现代 CSS 解决方案:全尺寸的带圆角的渐变边框

但是,可以看到上图的 border-radius 的值比较大,整个边框的宽度比较粗。

如果我们想得到一条 1px 宽度的渐变边框,我们尝试将上面的边框 CSS 样式修改一下:

border: 10px solid –> border: 1px solid

得到如下效果:

现代 CSS 解决方案:全尺寸的带圆角的渐变边框

由于圆角的原因,利用了 clip-path: inset(0 round 10px) 对图形进行切割后,元素的四个圆角都被切割掉了!

所以,有没有一种更好的方式,实现带圆角的渐变边框呢?

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

这里,我们介绍一种更为巧妙的方法。主要会利用 background-clipmask 两个核心属性。

首先,我们利用背景 background 实现一个普通的渐变背景:

div>div>
div {
    position: relative;
    width: 140px;
    height: 80px;
    border-radius: 100px; 
    background: conic-gradient(#ff00fa, #fe3, #0f3, #ff00fa);
}

利用角向渐变 background: conic-gradient(#ff00fa, #fe3, #0f3, #ff00fa),我们得到了这么一个图形:

现代 CSS 解决方案:全尺寸的带圆角的渐变边框

思考一下,如果我们有办法将图形中间部分镂空裁剪,我们不就能得到一个带圆角的渐变边框了吗?

通过一个示意图,你能很快明白到底是什么意思:

现代 CSS 解决方案:全尺寸的带圆角的渐变边框

好,那剩下的问题就转换为了:

  1. 如何裁剪掉一个元素内部的区域,并且能够控制裁剪区域的大小
  2. 裁剪区域,与图形的轮廓是一致的

在 CSS 中想使用裁剪功能,首先想到的肯定是 clip-path,但是上面的例子已经证明了 clip-path 无法实现细边框的裁剪,因此,我们需要另寻解法。

而 CSS 中,另外一个与裁剪功能相关的属性就是 mask

不了解 mask 的,可以戳我的这几篇文章看看:奇妙的 CSS MASK高阶切图技巧!基于单张图片的任意颜色转换

在此处,我们利用 mask,并且,最为核心的是,需要配合 mask-composite,实现 图形轮廓的精确裁剪

深入理解 mask-composite

什么是 mask-composite

mask-composite: 属性指定了将应用于同一元素的多个蒙版图像相互合成的方式。

通俗点来说,他的作用就是,当一个元素存在多重 mask 时,我们就可以运用 -webkit-mask-composite 进行效果叠加。

举个栗子:

div class="original">div>
.original {
    background: #000;
    mask: radial-gradient(circle at 0 0, #000, #000 200px, transparent 200px);
}

我们用一个 radial-gradient 作为 mask,切割原本的矩形,得到一个新的图形。

现代 CSS 解决方案:全尺寸的带圆角的渐变边框

如果再换一个方向:

div class="original">div>
.original {
    background: #000;
    mask: radial-gradient(circle at 100% 0, #000, #000 200px, transparent 200px);
}

现代 CSS 解决方案:全尺寸的带圆角的渐变边框

如果我想得到这样一个效果:

现代 CSS 解决方案:全尺寸的带圆角的渐变边框

该怎么做呢?

我们尝试合并上述两个 mask 的效果:

.mask {
    background: #000;
    mask: radial-gradient(circle at 100% 0, #000, #000 200px, transparent 200px),
        radial-gradient(circle at 0 0, #000, #000 200px, transparent 200px);
}

效果如下:

现代 CSS 解决方案:全尺寸的带圆角的渐变边框

与我们想象的不太一样,这是因为,两个 mask 的图形叠加,就是上述图形的效果,所以上述效果是没有问题的。

只是,我们想得到的是两个 mask 图形的重叠部分:

现代 CSS 解决方案:全尺寸的带圆角的渐变边框

这时,我们就可以使用 mask-composite

.mask {
    background: #000;
    mask: radial-gradient(circle at 100% 0, #000, #000 200px, transparent 200px),
        radial-gradient(circle at 0 0, #000, #000 200px, transparent 200px);
    -webkit-mask-composite: source-in;
}

添加了 -webkit-mask-composite: source-in 后,我们就可以得到两个 mask 图形的重叠部分,再基于这个重叠部分作用到整个 mask 遮罩:

现代 CSS 解决方案:全尺寸的带圆角的渐变边框

CodePen Demo — mask-composite Demo

-webkit-mask-composite 还可以实现非常多不同的功能,包括但不限于:

-webkit-mask-composite: clear; /* 清除,不显示任何遮罩 */
-webkit-mask-composite: copy; /* 只显示上方遮罩,不显示下方遮罩 */
-webkit-mask-composite: source-over; 
-webkit-mask-composite: source-in; /* 只显示重合的地方 */
-webkit-mask-composite: source-out; /* 只显示上方遮罩,重合的地方不显示 */
-webkit-mask-composite: source-atop;
-webkit-mask-composite: destination-over;
-webkit-mask-composite: destination-in; /* 只显示重合的地方 */
-webkit-mask-composite: destination-out;/* 只显示下方遮罩,重合的地方不显示 */
-webkit-mask-composite: destination-atop;
-webkit-mask-composite: xor; /* 只显示不重合的地方 */

看看这张图,就一目了然(图片源自 CSS mask 实现鼠标跟随镂空效果

现代 CSS 解决方案:全尺寸的带圆角的渐变边框

理解 background-clip

要实现最终效果,还有一个有意思的细节点需要掌握。那就是理解 backgrund-clip

一般我们用 backgrund-clip 比较多的场景是 background-clip: text,用于将背景图作用与文字之上。

但是,其实 backgrund-clip 还有几个与 box-content 类似的取值:

{
  background-clip: border-box;  // 背景延伸到边框外沿(但是在边框之下)
  background-clip: padding-box; // 边框下面没有背景,即背景延伸到内边距外沿。
  background-clip: content-box; // 背景裁剪到内容区 (content-box) 外沿。
}

什么意思呢?background-clip 设置元素的背景(背景图片或颜色)是否延伸到边框下面。看看下面这张图(图片源自:CSS Background Clip),就是对 background-clip 很好的一个阐述:

现代 CSS 解决方案:全尺寸的带圆角的渐变边框

而本文,我们会用到 content-box,举个例子:

div>div>
div {
  width: 140px;
  height: 80px;
  border-radius: 100px; 
  border: 1px dashed #000;
  background: conic-gradient(#ff00fa, #fe3, #0f3, #ff00fa);
  background-clip: content-box;
  padding: 10px; 
}

效果如下:

现代 CSS 解决方案:全尺寸的带圆角的渐变边框

可以看到,此时,背景的填充不再是从元素的右上角开始,而是在内容区域,算上了 10px 的 padding 之后,开始绘制。

也就是说,基于 background-clip,是 可以改变元素背景的绘制规则!这一点非常重要。

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

只有在掌握了 mask-compositebackground-clip 的基础上,你才能理解下面整个裁剪代码个核心精髓处。

基于上述讲解,我们就可以利用上面的综合技巧,实现我们最终想要的 — 带圆角的渐变边框

代码如下:

div>div>
div {
position: relative;
    width: 140px;
    height: 80px;
    border-radius: 100px; 
    background: conic-gradient(#ff00fa, #fe3, #0f3, #ff00fa);
    padding: 1px;
    -webkit-mask: 
        linear-gradient(#fff 0 100%) content-box,
        linear-gradient(#fff 0 100%);
    -webkit-mask-composite: xor;
}

这样,我们就完美的得到了 一个 1px 宽度的带圆角的渐变边框,并且,内部是镂空的

现代 CSS 解决方案:全尺寸的带圆角的渐变边框

通过控制上述的 padding: 1px 来控制元素的边框宽度。

完整的代码,你可以戳这里查看:CodePen Demo — 纯 CSS 实现带圆角的渐变边框!

最后

好了,本文到此结束,希望本文对你有所帮助 :)

想 Get 到最有意思的 CSS 资讯,千万不要错过我的公众号 — iCSS 前端趣闻 😄

更多精彩 CSS 技术文章汇总在我的 Github — iCSS,持续更新,欢迎点个 star 订阅收藏。

如果还有什么疑问或者建议,可以多多交流,原创文章,文笔有限,才疏学浅,文中若有不正之处,万望告知。

    正文完
     0
    Yojack
    版权声明:本篇文章由 Yojack 于2024-09-18发表,共计4552字。
    转载说明:
    1 本网站名称:优杰开发笔记
    2 本站永久网址:https://yojack.cn
    3 本网站的文章部分内容可能来源于网络,仅供大家学习与参考,如有侵权,请联系站长进行删除处理。
    4 本站一切资源不代表本站立场,并不代表本站赞同其观点和对其真实性负责。
    5 本站所有内容均可转载及分享, 但请注明出处
    6 我们始终尊重原创作者的版权,所有文章在发布时,均尽可能注明出处与作者。
    7 站长邮箱:laylwenl@gmail.com
    评论(没有评论)