這是前端面試系列的第3篇,前面的章節,你能夠在這裏找到:css
面試前端候選人的時候,我常常會問這樣一個有關CSS的問題:html
你知道僞類與僞元素麼,它們的分別是什麼? 這時,能回答上來的不多。前端
換一種問法,你知道 :hover, :active, :focus, :visited麼? 這時,基本都能回答上來,這不就是 a 標籤的四種狀態麼。面試
嗯,ok。而後繼續問,那麼 ::before 和 ::after,據說過麼? 這時,能聽到的回答是,嗯,我看到過,偶爾會用。bash
僞類與僞元素,都有一個「僞」字,那它們有什麼區別麼? 這時,迴應個人,是一片沉默。。。antd
從回答上來分析,雖然僞類和僞元素平時都有接觸,但在概念上,都比較模糊。今天,咱們就來講說僞類與僞元素的區別,以及使用場景。僞類,不是隻有 a 標籤的四種狀態。僞元素,也不是隻有 ::before 與 ::after。更多的僞類與僞元素,詳見文末附錄。佈局
從概念上來區分,大體有如下幾點:post
僞類,更多的定義的是狀態。常見的僞類有 :hover,:active,:focus,:visited,:link,:not,:first-child,:last-child等等。字體
僞元素,不存在於DOM樹中的虛擬元素,它們能夠像正常的html元素同樣定義css,但沒法使用JavaScript獲取。常見僞元素有 ::before,::after,::first-letter,::first-line等等。ui
CSS3明確規定了,僞類用一個冒號(:)來表示,而僞元素則用兩個冒號(::)來表示。但目前由於兼容性的問題,它們的寫法能夠是一致的,都用一個冒號(:)就能夠了,因此很是容易混淆。
表單的校驗中,常會用到 :required
、:valid
和 :invalid
這三個僞類。先來看看它們所表明的含義。
看下面這個例子:
<p>input中類型爲email的校驗</p>
<p>符合email校驗規則</p>
<input type="email" required placeholder="請輸入" value="24238477@qq.com" />
<br><br>
<p>不符合email校驗規則</p>
<input type="email" required placeholder="請輸入" value="lalala" />
<br><br>
<p>有required標識,但未填寫</p>
<input type="email" required placeholder="請輸入" value="" />
複製代碼
input {
&:valid {
border-color: green;
box-shadow: inset 5px 0 0 green;
}
&:invalid {
border-color: red;
box-shadow: inset 5px 0 0 red;
}
&:required {
border-color: red;
box-shadow: inset 5px 0 0 red;
}
}
複製代碼
效果以下:
過去,要實現摺疊面板的顯示或隱藏,只能用JavaScript來搞定。可是如今,能夠用僞類 :target
來實現。 :target 是文檔的內部連接,即 URL 後面跟有錨名稱 #,指向文檔內某個具體的元素。
看下面這個例子:
<div class="t-collapse">
<!-- 在url最後添加 #modal1,使得target生效 —>
<a class="collapse-target" href="#modal1">target 1</a>
<div class="collapse-body" id="modal1">
<!-- 將url的#modal1 變爲 #,使得target失效 —>
<a class="collapse-close" href="#">target 1</a>
<p>...</p>
</div>
</div>
複製代碼
.t-collapse {
>.collapse-body {
display: none;
&:target {
display: block;
}
}
}
複製代碼
當咱們要指定一系列標籤中的某個元素時,並不須要用JavaScript獲取。能夠用 :nth-child(n)
與 :nth-of-type(n)
來找到,並指定樣式。但它們有一些小區別,須要注意。
首先,它們的n能夠是大於零的數字,或者相似2n+1的表達式,再或者是 even / odd。
另外,還有2個區別:
類型
,也就是標籤。看下面這個例子,注意二者的差別:
<h1>這是標題</h1>
<p>第一個段落。</p>
<p>第二個段落。</p>
<p>第三個段落。</p>
<p>第四個段落。</p>
<p>第五個段落。</p>
複製代碼
還記得2018年聖誕節的「彩蛋事件」,在整個前端圈,轟動一時。由於按鈕上的一朵雲,致使很多前端er提早回家過年了。當時,彩蛋事件出現的第一時間,就嚇得我趕快打開工程看了一眼,果真也中招了。爲了保住飯碗,得趕忙把雲朵去掉。
查看了生成的html,發現原來是 button 下藏了一個 ::before
。因此,趕忙把樣式覆蓋掉,兼容代碼以下:
.ant-btn {
&::before {
display: none !important;
}
}
複製代碼
在網頁中,默認的劃詞效果是,原字色保持不變,劃過期的背景變爲藍底色。其實,這是能夠用 ::selection
來進行美化的。看下面這個例子:
<p>Custom text selection color</p>
複製代碼
::selection {
color: red;
background-color: yellow;
}
複製代碼
效果以下:
劃過的部分美化爲:紅色的字體,而且底色變爲了黃色。
CSS也能夠實現動態的交互,並不是只有JavaScript才能實現。
書寫的時候,要尊重規範。寫僞類的時候用 :
,而寫僞元素的時候用 ::
。
兼容性的問題,交給postcss去作。本文並未涉及兼容性的寫法,包括前綴問題,能夠交給autoprefixer去作。
:root 選擇文檔的根元素,等同於 html 元素
:empty 選擇沒有子元素的元素
:target 選取當前活動的目標元素
:not(selector) 選擇除 selector 元素意外的元素
:enabled 選擇可用的表單元素
:disabled 選擇禁用的表單元素
:checked 選擇被選中的表單元素
:nth-child(n) 匹配父元素下指定子元素,在全部子元素中排序第n
:nth-last-child(n) 匹配父元素下指定子元素,在全部子元素中排序第n,從後向前數
:nth-child(odd) 、 :nth-child(even) 、 :nth-child(3n+1) :first-child 、 :last-child 、 :only-child
:nth-of-type(n) 匹配父元素下指定子元素,在同類子元素中排序第n
:nth-last-of-type(n) 匹配父元素下指定子元素,在同類子元素中排序第n,從後向前數
:nth-of-type(odd) 、 :nth-of-type(even) 、 :nth-of-type(3n+1) :first-of-type 、 :last-of-type 、 :only-of-type
::after 已選中元素的最後一個子元素
::before 已選中元素的第一個子元素
::first-letter 選中某個款級元素的第一行的第一個字母
::first-line 匹配某個塊級元素的第一行
::selection 匹配用戶劃詞時的高亮部分
PS:歡迎關注個人公衆號 「超哥前端小棧」,交流更多的想法與技術。