之前在這篇文章中 -- 《N種使用 CSS 實(shí)現(xiàn)三角形的技巧》,介紹了 6 種使用 CSS 實(shí)現(xiàn)三角形的方式。
但是其中漏掉了一個(gè)非常重要的場(chǎng)景,如何使用純 CSS 實(shí)現(xiàn)帶圓角的三角形呢?,像是這樣:

本文將介紹幾種實(shí)現(xiàn)帶圓角的三角形的實(shí)現(xiàn)方式。
法一. 全兼容的 SVG 大法
想要生成一個(gè)帶圓角的三角形,代碼量最少、最好的方式是使用 SVG 生成。
使用 SVG 的 多邊形標(biāo)簽 <polygon> 生成一個(gè)三邊形,使用 SVG 的 stroke-linejoin="round" 生成連接處的圓角。
代碼量非常少,核心代碼如下:
<svg width="250" height="250" viewBox="-50 -50 300 300"> <polygon class="triangle" stroke-linejoin="round" points="100,0 0,200 200,200"/> </svg>
.triangle {
fill: #0f0;
stroke: #0f0;
stroke-width: 10;
}實(shí)際圖形如下:

這里,其實(shí)是借助了 SVG 多邊形的 stroke-linejoin: round 屬性生成的圓角,stroke-linejoin 是什么?它用來(lái)控制兩條描邊線段之間,有三個(gè)可選值:
miter 是默認(rèn)值,表示用方形畫筆在連接處形成尖角
round 表示用圓角連接,實(shí)現(xiàn)平滑效果
bevel 連接處會(huì)形成一個(gè)斜接

我們實(shí)際是通過(guò)一個(gè)帶邊框,且邊框連接類型為 stroke-linejoin: round 的多邊形生成圓角三角形的。
如果,我們把底色和邊框色區(qū)分開,實(shí)際是這樣的:
.triangle {
fill: #0f0;
stroke: #000;
stroke-width: 10;
}
通過(guò) stroke-width 控制圓角大小
那么如何控制圓角大小呢?也非常簡(jiǎn)單,通過(guò)控制 stroke-width 的大小,可以改變圓角的大小。
當(dāng)然,要保持三角形大小一致,在增大/縮小 stroke-width 的同時(shí),需要縮小/增大圖形的 width/height:

完整的 DEMO 你可以戳這里:CodePen Demo -- 使用 SVG 實(shí)現(xiàn)帶圓角的三角形
法二. 圖形拼接
不過(guò),上文提到了,使用純 CSS 實(shí)現(xiàn)帶圓角的三角形,但是上述第一個(gè)方法其實(shí)是借助了 SVG。那么僅僅使用 CSS,有沒(méi)有辦法呢?
當(dāng)然,發(fā)散思維,CSS 有意思的地方正在于此處,用一個(gè)圖形,能夠有非常多種巧妙的解決方案!
我們看看,一個(gè)圓角三角形,它其實(shí)可以被拆分成幾個(gè)部分:

所以,其實(shí)我們只需要能夠畫出一個(gè)這樣的帶圓角的菱形,通過(guò) 3 個(gè)進(jìn)行旋轉(zhuǎn)疊加,就能得到圓角三角形:

繪制帶圓角的菱形
那么,接下來(lái)我們的目標(biāo)就變成了繪制一個(gè)帶圓角的菱形,方法有很多,本文給出其中一種方式:
首先將一個(gè)正方形變成一個(gè)菱形,利用 transform 有一個(gè)固定的公式:

<div></div>
div {
width: 10em;
height: 10em;
transform: rotate(-60deg) skewX(-30deg) scale(1, 0.866);
}
將其中一個(gè)角變成圓角:
div {
width: 10em;
height: 10em;
transform: rotate(-60deg) skewX(-30deg) scale(1, 0.866);
+ border-top-right-radius: 30%;
}
至此,我們就順利的得到一個(gè)帶圓角的菱形了!
拼接 3 個(gè)帶圓角的菱形
接下來(lái)就很簡(jiǎn)單了,我們只需要利用元素的另外兩個(gè)偽元素,再生成 2 個(gè)帶圓角的菱形,將一共 3 個(gè)圖形旋轉(zhuǎn)位移拼接起來(lái)即可!
完整的代碼如下:
<div></div>
div{
position: relative;
background-color: orange;
}
div:before,
div:after {
content: '';
position: absolute;
background-color: inherit;
}
div,
div:before,
div:after {
width: 10em;
height: 10em;
border-top-right-radius: 30%;
}
div {
transform: rotate(-60deg) skewX(-30deg) scale(1,.866);
}
div:before {
transform: rotate(-135deg) skewX(-45deg) scale(1.414, .707) translate(0,-50%);
}
div:after {
transform: rotate(135deg) skewY(-45deg) scale(.707, 1.414) translate(50%);
}就可以得到一個(gè)圓角三角形了!效果如下:

完整的代碼你可以戳這里:CodePen Demo -- A triangle with rounded
法三. 圖形拼接實(shí)現(xiàn)漸變色圓角三角形
完了嗎?沒(méi)有!
上述方案,雖然不算太復(fù)雜,但是有一點(diǎn)還不算太完美的。就是無(wú)法支持漸變色的圓角三角形。像是這樣:

如果需要實(shí)現(xiàn)漸變色圓角三角形,還是有點(diǎn)復(fù)雜的。但真就還有人鼓搗出來(lái)了,下述方法參考至 -- How to make 3-corner-rounded triangle in CSS。
同樣也是利用了多塊進(jìn)行拼接,但是這次我們的基礎(chǔ)圖形,會(huì)非常的復(fù)雜。
首先,我們需要實(shí)現(xiàn)這樣一個(gè)容器外框,和上述的方法比較類似,可以理解為是一個(gè)圓角菱形(畫出 border 方便理解):

<div></div>
div {
width: 200px;
height: 200px;
transform: rotate(30deg) skewY(30deg) scaleX(0.866);
border: 1px solid #000;
border-radius: 20%;
}
接著,我們同樣使用兩個(gè)偽元素,實(shí)現(xiàn)兩個(gè)稍顯怪異的圖形進(jìn)行拼接,算是對(duì) transform 的各種用法的合集:
div::before,
div::after {
content: "";
position: absolute;
width: 200px;
height: 200px;
}
div::before {
border-radius: 20% 20% 20% 55%;
transform: scaleX(1.155) skewY(-30deg) rotate(-30deg) translateY(-42.3%) skewX(30deg) scaleY(0.866) translateX(-24%);
background: red;
}
div::after {
border-radius: 20% 20% 55% 20%;
background: blue;
transform: scaleX(1.155) skewY(-30deg) rotate(-30deg) translateY(-42.3%) skewX(-30deg) scaleY(0.866) translateX(24%);
}為了方便理解,制作了一個(gè)簡(jiǎn)單的變換動(dòng)畫:

本質(zhì)就是實(shí)現(xiàn)了這樣一個(gè)圖形:

最后,給父元素添加一個(gè) overflow: hidden 并且去掉父元素的 border 即可得到一個(gè)圓角三角形:

由于這兩個(gè)元素重疊空間的特殊結(jié)構(gòu),此時(shí),給兩個(gè)偽元素添加同一個(gè)漸變色,會(huì)完美的疊加在一起:
div::before,
div::after, {
background: linear-gradient(#0f0, #03a9f4);
}最終得到一個(gè)漸變圓角三角形:

上述各個(gè)圖形的完整代碼,你可以戳這里:CodePen Demo -- A triangle with rounded and gradient background
最后
本文介紹了幾種在 CSS 中實(shí)現(xiàn)帶圓角三角形的方式,雖然部分有些繁瑣,但是也體現(xiàn)了 CSS ”有趣且折磨人“ 的一面,具體應(yīng)用的時(shí)候,還是要思考一下,對(duì)是否使用上述方式進(jìn)行取舍,有的時(shí)候,切圖也許是更好的方案。
好了,本文到此結(jié)束,希望對(duì)你有幫助 :)
轉(zhuǎn)自:https://segmentfault.com/a/1190000040344317






