使用 tabindex 配合 focus-within 巧妙实现父选择器

简介: 使用 tabindex 配合 focus-within 巧妙实现父选择器

本文将介绍一个不太实用的小技巧,使用 tabindex 配合 :focus-within 巧妙实现父选择器。

 

CSS 中是否存在父选择器?



这是一个非常经典的问题,到目前为止,CSS 没有真正意义上被广泛实现的父选择器,这和浏览器的渲染机制有关。

如果你对 CSS 中是否存在父选择器有疑惑,可以去看看 知乎 -- CSS 中能否选取父元素?

当然,这不代表 CSS 完全无法通过子元素去控制父元素,通过 :focus-within 伪类可以近似的达到类似的目的。

 

:focus-within 伪类



首先需要复习一下 :focus-within,它是一个伪类。

它表示一个元素获得焦点,或,该元素的后代元素获得焦点。划重点,它或它的后代获得焦点。

关于 :focus-within,不算太了解的可以先看看这篇文章:《神奇的选择器 :focus-within》

利用它,我们可以实现类似这样的功能,通过元素的子元素的获焦(focus事件),触发该伪元素,从而实现一个狭义的父选择器,类似这样:

608782-20201030113104826-1624563117.gif

CodePen -- CSS focus-within INPUT

 

:focus-within 伪类实现父选择的缺陷


借助 :focus-within 实现父类选择器最大的问题是,元素必须要有 focus 事件,才能触发它或者它的父元素的 :focus-within。


所以,这就导致了在之前我认为 :focus-within 只能配合 <button>、<input > 元素一起使用。


诸如 <button>,<input>,<select>,<a> 这类可交互元素,默认是存在 focus 事件的,而类似 <div>,<span> 和 <table> 这类非交互元素,默认是不能被聚焦的。


也是因为这个原因,大大限制了它的使用场景。基于此,我们引入本文的另外一个主角 -- tabindex。

 

使用 tabindex 使元素获得 focus 事件



tabindex: HTML 标签的属性,指示其元素是否可以聚焦,以及它是否/在何处参与顺序键盘导航(通常使用Tab键,因此得名)。


也就是说,一个单纯的 div 标签,他是没有 focus 事件的,然而,我们给它加上一个 tabindex 属性,这个时候他就会获得类似 input 框一样的表现,拥有了 focus 事件,再配合 :focus-within,能够使用的场景就大大提升了。


看看伪代码:

<div class="g-father">
    <!-- 没有 focus 事件的 .g-children 元素 -->
    <div class="g-children">Click</div>
</div>


<div class="g-father">
    <!-- 拥有 focus 事件的 .g-children 元素 -->
    <div class="g-children" tabindex="-1">Click</div>
</div>

这里为什么是 tabindex="-1" 呢,tabindex 负值表示元素是可聚焦的,但是不能通过键盘导航来访问到该元素。因为我们只需要让元素能够获得 focus 事件,而不需要他真的能够被键盘导航来访问。


这样,配合 :focus-within,就能做到当点击子元素的时候,去改变父元素的样式了。

并且,我们可以在任意元素上搭配 tabindex,脱离了 <input>, <a>, <button> 等元素才有 focus 事件的束缚。


.g-father:focus-within {
    background: #fc0;
}

608782-20201030113118649-93176106.gif


一个小细节,button 的 focus 事件在 Safari 和 firefox 的上冒泡问题


由于 input 元素(或者任意元素 +tabindex) 配合 :focus-within 的方案依赖 focus 事件的冒泡。


而对于 <button> 元素,稍微有点特殊,存在这样两个问题,即:


  1. 在 MacOS 的 Safari 和 Firefox 中, **点击 <button> 元素,不会触发 <button> 的 focus 事件,也没有 focus 事件冒泡。
  2. 在 Windows 的 Safari 和 Firefox 中, 点击 <button> 元素,会触发 <button> 的 focus 事件,但在被目标元素捕捉到之后,不会继续向上冒泡。


什么意思呢?我们来验证一下,使用类似这样的结构:

<div class="g-father">
    <input type="button" value="Button">
</div>
input:focus {
    background: #00bcd4;
}
body:focus-within {
    background: blue;
}
.g-father:focus-within {
    background: red;
}


看看,在 Chrome 下的表现:


608782-20201030113128522-465362371.gif


在 Windows 的 Safari,Firefox 下的表现:


608782-20201030113135502-1872876266.gif

在 MacOS 的 Safari,Firefox 下的表现:


608782-20201030113141841-130359527.gif


在 Chrome 上的表现是正常,而在 Windows 的 Safari、Firefox 上,会触发 button 的 focus 事件,但不会触发父元素的 :focus-within 事件,也就是上面说的,focus 事件,在被目标元素捕捉到之后,不会继续向上冒泡。而在 Mac 上,则连 focus 都不会触发。


这一点,在使用的时候务必需要留意。


CodePen -- button 的 focus 事件冒泡性验证(Chorme / Safari / Firefox)

608782-20201030113154751-1055238086.png


最后



当然,本文介绍的小技巧,只能算是一个非常简陋,特定条件(点击目标元素改变父元素样式)下的父选择器,真正意义上的父选择器仍需等待未来规范的实现。

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


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


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

目录
相关文章
|
4月前
|
JavaScript
子组件获取外层组件的scrollTop,达到实时定位的效果
本文介绍了如何通过Vue的自定义事件总线(eventBus),实现子组件获取外层组件的scrollTop值,并达到实时定位的效果。
38 1
子组件获取外层组件的scrollTop,达到实时定位的效果
|
6月前
|
移动开发 JavaScript 前端开发
VUE实现一个列表清单【props 父子组件通信、slot插槽的使用、全局自定义指令的封装、$nextTick解决异步DOM更新、巧用v-model简化父子组件之间的通信、触发事件的事件源event】
VUE实现一个列表清单【props 父子组件通信、slot插槽的使用、全局自定义指令的封装、$nextTick解决异步DOM更新、巧用v-model简化父子组件之间的通信、触发事件的事件源event】
52 0
|
7月前
技术经验分享:Cascader级联选择器Element的使用和总结
技术经验分享:Cascader级联选择器Element的使用和总结
90 0
|
前端开发
前端学习笔记202304学习笔记第六天-样式冲突-动态绑定props的值
前端学习笔记202304学习笔记第六天-样式冲突-动态绑定props的值
66 0
|
前端开发 JavaScript 测试技术
CSS 解决z-index上层元素遮挡下层元素点击事件问题
CSS 解决z-index上层元素遮挡下层元素点击事件问题
605 0
|
前端开发
CSS之选择器(十一):focus-within
CSS之选择器(十一):focus-within
CSS之选择器(十一):focus-within
|
前端开发
iframe的运用---特别是获取父子页面的元素
iframe的运用---特别是获取父子页面的元素
514 0
|
前端开发
前端项目实战41-子组件嵌套memo防止重复渲染
前端项目实战41-子组件嵌套memo防止重复渲染
103 0
|
Web App开发 前端开发 JavaScript
浅谈逻辑选择器 -- 父选择器它来了!
浅谈逻辑选择器 -- 父选择器它来了!
158 0
|
前端开发
前端工作总结288-pc父组件通过props传值给子组件,如何避免子组件改变props的属性值报错问题
前端工作总结288-pc父组件通过props传值给子组件,如何避免子组件改变props的属性值报错问题
126 0