京东6.18大促主会场领京享红包更优惠

 找回密码
 立即注册

QQ登录

只需一步,快速开始

查看: 4012|回复: 0

CSS 实现Chrome标签栏的技巧

[复制链接]

31

主题

0

回帖

10

积分

新手上路

积分
10
发表于 2021-11-21 02:17:08 | 显示全部楼层 |阅读模式 来自 中国
这次来看一个带特殊圆角导航栏布局,如下谷歌浏览器的标签栏:; q6 F. w! g$ M/ S5 ^

0 H- `# }* s( |% i这样一个布局如何实现呢?下面介绍几种方法
! |+ [5 h$ M; ?" K1 ?一、伪元素拼接
: w+ n$ ^1 v2 O. g3 T# t) \6 P! p/ D  L) Z3 o; \
假设有这样一个 HTML 结构
  1.   [url=https://mxzdjyxk.cn/]Svelte API[/url]  [url=https://mxzdjyxk.cn/]Svelte API[/url]  [url=https://mxzdjyxk.cn/]Svelte API[/url]  [url=https://mxzdjyxk.cn/]Svelte API[/url]
复制代码
首先可以考虑的一种方式就利用两个伪元素拼接0 e2 D  R# J$ w  m( t- t

. Z# G+ [( Z7 ~& w+ }中间的圆角比较容易,左右两边的反向圆角如何实现呢?其实可以想想有哪些可以实现圆形的样式,这里想到了border-radius ,可以这样来实现5 b/ y6 M) k& H# _$ i
       
    : v+ H  O3 b3 u! m: }) @9 G
  • 画一个透明的圆    , V% H  p+ d1 u% `
  • 给圆加上足够大的边框或者投影    ' k& g' G. O* v3 S' t$ J
  • 裁剪一小部分    " q7 B7 |3 w1 K% q6 x+ F
  • 完成4 |( }: W9 Q6 {9 a
示意如下) @1 D/ m& ?5 l$ O

# l3 i' \5 ^0 m$ `0 d用代码实现就是
  1. .tab-item{  position: relative;  background-color: red;  padding: 10px 15px;  border-radius: 12px 12px 0 0;  cursor: pointer;  transition: .2s;}.tab-item::before,.tab-item::after{  position: absolute;  bottom: 0;  content: '';  width: 20px;  height: 20px;  border-radius: 100%;  box-shadow: 0 0 0 40px red;/*使用box-shadow不影响尺寸*/  transition: .2s;}.tab-item::before{  left: -20px;  clip-path: inset(50% -10px 0 50%);}.tab-item::after{  right: -20px;  clip-path: inset(50% 50% 0 -10px);}
复制代码
最终实时效果如下' ]& x1 A& V0 E$ F; ~

, A% t+ N$ A- a7 \9 b% }3 n& I这里裁剪是用 clip-path 实现的,注意左右可以朝里面多裁剪一点,以免拼接出现缝隙,完成代码可访问 chrome-tab (codepen.io)4 }5 T" I5 t+ d8 o. ^
当然这里的反向圆角还可以采用径向渐变来实现,接着往下看。$ u9 q# j( i6 t7 h
二、万能的渐变4 @$ [0 B4 ]! h% m0 Q; ]% }! \) h
CSS 渐变几乎是无所不能的,什么的图形都能绘制,这里可以拆分一下,两个矩形,两个圆形,还有两个反向圆角,也就是 2 个 线性渐变,4 个径向渐变,示意如下, y, v* R3 g8 x( _; J, h
7 Y% Y* m: Q: E3 Y$ t
用代码实现就是
  1. .tab-item{  padding: 10px 35px;  background-image:     radial-gradient(circle at 0 0, transparent 15px,blue 0),    radial-gradient(circle at 0 0, transparent 15px,blue 0),    radial-gradient(circle at 0 0, green 12px,transparent 0,    radial-gradient(circle at 12px 0, green 12px,transparent 0,    linear-gradient(red,red),    linear-gradient(red,red);  background-repeat: no-repeat;  background-position: 15px top, right 15px top 0, left bottom, right bottom, center bottom, center, bootom;  background-size: 30px 30px, 30px 30px, 12px 12px, 12px 12px, calc(100% - 30px) calc(100% - 12px), calc(100% - 54px) 100%;}
复制代码
虽然实现了,但是非常拢邢腹鄄旆⑾郑礁鲈残问强梢岳闷狡淌迪值模礁龇聪蛟步强梢钥闯墒且桓霭朐玻缓笠部梢云狡蹋疽馊缦/p>
3 W9 o/ D# p1 f& B
这样,只需要两个径向渐变就可以实现了,代码如下
  1. .tab-item{  position: relative;  padding: 10px 35px;  cursor: pointer;  background-image: radial-gradient(circle at 15px 0, transparent 15px,blue 0),    radial-gradient(circle at 27px 12px, green 12px,transparent 0),    linear-gradient(red,red),    linear-gradient(red,red);  background-size: 100% 15px,calc(100% - 54px), calc(100% - 30px) calc(100% - 12px), calc(100% - 54px) 100%;  background-position: -15px bottom, left top, center bottom, center bottom;  background-repeat: repeat-x, repeat-x, no-repeat, no-repeat;}
复制代码
最终实时效果如下(上面是原理图)0 [$ Z8 d* M7 r2 K

* N% X* k+ f. Y6 m3 V" V完成代码可访问 chrome-tab-gradient (codepen.io)
7 D  n' P4 @1 J8 {; I5 P三、自适应的svg7 o8 X7 A" H# `4 b
5 H! B+ t9 A- k2 p' }$ V
渐变虽然是万能的,但是代码量比较多,非常考验耐心。对于这个例子,svg 也是不错的方案。
0 O# C3 m1 E- I& v
% l" v$ x. p' [, l' e! Z1 c* S
中间的圆角矩形比较容易,用 rect 就行
  1.           
复制代码
两边的反向圆角可以直接使用一段 path 路径(各种图形软件都可以生成)
  1.        
复制代码
  1.        
复制代码
然后把这 3 段 svg 代码作为背景就可以了,可以用 background-size 和 background-position 进行调整和控制
  1. .tab-item{  position: relative;  padding: 10px 35px;        margin: 0 -15px;  cursor: pointer;  transition: .2s;  background-image: url("data:image/svg+xml,%3Csvg width='100' height='100' viewBox='0 0 100 100' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath fill-rule='evenodd' clip-rule='evenodd' d='M100 100C44.772 100 0 55.228 0 0v100h100z' fill='%23F8EAE7'/%3E%3C/svg%3E"),    url("data:image/svg+xml,%3Csvg width='100' height='100' viewBox='0 0 100 100' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath fill-rule='evenodd' clip-rule='evenodd' d='M0 100c55.228 0 100-44.772 100-100v100H0z' fill='%23F8EAE7'/%3E%3C/svg%3E"),    url("data:image/svg+xml,");  background-size: 12px 12px, 12px 12px, calc(100% - 24px) calc(100% + 12px);  background-position: right bottom, left bottom, center top;  background-repeat: no-repeat;}
复制代码
实时效果如下
& n0 E) E3 t& G) Q3 }. V1 d

$ M; M) B3 j7 c7 H, T7 o0 b完整代码可以访问 chrome-tab-svg (codepen.io)
0 \% f) Y) @! `另外,有人可能会奇怪,这里**为什么要用3段 svg?用 1 段 svg 里面包含 3 个路径不行吗?**答案是不行的。svg 里没法灵活使用定位,比如要实现位于右下角,svg 只能使用 100% 而不能使用 calc( 100% - 12px ),更别说 CSS 还有 right bottom 这样的定位属性了,所以必须采用 CSS多背景 实现
; i$ ^1 P( J# B  M四、图片边框+ c  C2 l" F! T4 l( S" F
- ^; y0 B  q- c; w. |/ K' b  u, J( ~
上面几种方式还是觉得太复杂了,能不能**“切图”**呢?当然也是可以的,不过也需要一定的技巧,这样才能实现自适应。这里可以采用 CSS3 border-image 来实现。关于 border-image 可以参考这篇文章:JELLY | border-image 的正确用法 (jd.com)。0 I  O) S8 J; \' A* F, {# F4 Q
准备这样一张图就可以了,svg 或者 png 都行3 {" u& g# S7 I

6 E( k  D: a& H) z) g4 csvg 如下
  1.   
复制代码
接着根据 border-image 规范进行切割就行了* o7 p5 g& A0 I& D3 {/ T5 P2 R
- \. I' Q! z0 C- G- F8 @
代码实现如下,记得要加上 border
  1. .tab-item{  position: relative;  padding: 0 8px;  margin: 0 -15px;  cursor: pointer;  border-width: 12px 27px 15px;  border-style: solid;  border-image: url("data:image/svg+xml,%3Csvg width='67' height='33' viewBox='0 0 67 33' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath fill-rule='evenodd' clip-rule='evenodd' d='M27 0c-6.627 0-12 5.373-12 12v6c0 8.284-6.716 15-15 15h67c-8.284 0-15-6.716-15-15v-6c0-6.627-5.373-12-12-12H27z' fill='%23F8EAE7'/%3E%3C/svg%3E") 12 27 15 fill;}
复制代码
实时效果如下) s* |" m7 A, x: \- x
7 V3 Q% @* S$ c3 ^9 r
完整代码可以访问 chrome-tab-border-image (codepen.io)
; {8 h7 I# S2 P7 p虽然代码实现比较简介,但是由于要加上 border,导致内容尺寸有些不好控制
7 ~1 P1 i  c: }3 G五、mask 遮罩' `9 M3 _$ r" U- v; f0 Q) [, @

0 i. @* D& b5 U1 R4 ]+ Q前面几种背景图片的方式,其实有一个问题,颜色都在背景图片中,几乎是固定的,不方便修改,那么,借助 mask 遮罩,可以很轻松的解决这个问题。
0 F7 W  v2 z6 o有了前面的背景(渐变或者svg都行),只需要把 background 批量换成 -webkit-mask 就行了,就像这样  e1 `+ T* f) U
. b+ O$ R5 }3 y" A' w. v
以 svg 为例,替换以后如下
  1. .tab-item{  position: relative;  padding: 10px 35px;  cursor: pointer;  background: #F8EAE7;  -webkit-mask-image: url("data:image/svg+xml,%3Csvg width='100' height='100' viewBox='0 0 100 100' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath fill-rule='evenodd' clip-rule='evenodd' d='M100 100C44.772 100 0 55.228 0 0v100h100z' fill='%23F8EAE7'/%3E%3C/svg%3E"),    url("data:image/svg+xml,%3Csvg width='100' height='100' viewBox='0 0 100 100' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath fill-rule='evenodd' clip-rule='evenodd' d='M0 100c55.228 0 100-44.772 100-100v100H0z' fill='%23F8EAE7'/%3E%3C/svg%3E"),    url("data:image/svg+xml,");  -webkit-mask-size: 12px 12px, 12px 12px, calc(100% - 24px) calc(100% + 12px);  -webkit-mask-position: right bottom, left bottom, center top;  -webkit-mask-repeat: no-repeat;}
复制代码
现在控制背景颜色就方便了,如果需要改变背景色,直接改变就行了
  1. .tab-item:hover{  background: #F2D0CA;}
复制代码
完整代码可以查看 chrome-tab-mask (codepen.io)5 w) f- e; g) T
另外,喜欢**“切图”**的还可以使用 mask-border 实现,和上面的 border-image 基本一致,只不过得到了遮罩的效果
3 S0 z8 o6 c. P6 K4 y0 q还是采用这张图,进行切割
* o9 \, h. W3 [6 `8 q
! D, {6 ^% F: Y, d. G. _
代码实现就是
  1. .tab-item{  /*...*/  -webkit-mask-box-image: url("data:image/svg+xml,%3Csvg width='67' height='33' viewBox='0 0 67 33' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath fill-rule='evenodd' clip-rule='evenodd' d='M27 0c-6.627 0-12 5.373-12 12v6c0 8.284-6.716 15-15 15h67c-8.284 0-15-6.716-15-15v-6c0-6.627-5.373-12-12-12H27z' fill='%23F8EAE7'/%3E%3C/svg%3E") 12 27 15;}
复制代码

3 J$ R% m2 c; a4 G7 H8 W+ W% c0 Z完整代码可以访问 chrome-tab-mask-border (codepen.io)
- |1 M' ~0 S" L  {$ v4 K! s目前仍然在草案当中,有一个替代属性-webkit-mask-box-image 可以使用/ V: ^! ~& o7 |6 D6 ~7 y
六、总结和说明
. `9 N8 q$ e1 P  X9 C2 f# n: a1 e
以上共介绍了 5 种不同的布局方式,下面总结一下实现要点:( ~( P9 ~" P+ A1 m  m1 M/ {9 x
        0 T3 _. I( {0 G
  • border-radius 配合 clip-path 可以实现内凹圆角    : n( j: u0 ^% S8 g5 ?1 v! D. u. l
  • 渐变是万能的,重复的内容尽量通过 background-repeat 来完成   
    % f( c0 r/ K* Y5 R9 N5 m' M! h
  • svg 中 rect 可以实现自适应圆角矩形,作为背景同样适用   
    % v+ l  @- f- w7 v* {9 X
  • 可以将多段 svg 作为多背景,分别控制尺寸和位置   
    ( W# H; M% M: r' V1 [* L
  • border-image 可以实现自适应效果,需要注意设置 border-width   
    . u- w! _( H: Y
  • mask 遮罩可以直接使用渐变或者svg作为遮罩层,可以更方便的修改背景色   
    / i$ @5 Y" p' \: B2 D# N$ i
  • mask-border 和 border-image 使用类似,不过目前只有 -webkit- 内核支持
    4 m) i- y3 Y6 e7 @6 O" b- T; t1 ^% z
到此这篇关于CSS 实现Chrome标签栏的技巧的文章就介绍到这了,更多相关CSS Chrome标签栏内容请搜索脚本之家以前的文章或继续浏览下面的相关文章,希望大家以后多多支持脚本之家!
$ i1 S# T" G: Q# g' q! l, P
* {8 b/ }2 w% b" i- A来源:http://www.jb51.net/css/784321.html) z2 Q+ j5 \' k3 _4 j
免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!

本帖子中包含更多资源

您需要 登录 才可以下载或查看,没有账号?立即注册

×

帖子地址: 

梦想之都-俊月星空 优酷自频道欢迎您 http://i.youku.com/zhaojun917
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

QQ|手机版|小黑屋|梦想之都-俊月星空 ( 粤ICP备18056059号 )|网站地图

GMT+8, 2025-11-10 20:24 , Processed in 0.057373 second(s), 23 queries .

Powered by Mxzdjyxk! X3.5

© 2001-2025 Discuz! Team.

快速回复 返回顶部 返回列表