详细了解CSS :focus-within伪类及其交互应用

这篇文章发布于 2018年01月21日,星期日,23:56,归类于 CSS相关。 阅读 39923 次, 今日 16 次 16 条评论

 

一、了解CSS :focus-within伪类

CSS :focus-within伪类选择器和IE8就开始支持的:focus可以说是近亲,区别在于:focus表示当前元素处于focus状态时候干嘛干嘛,而:focus-within表示当前元素或者当前元素的子元素处于focus状态时候干嘛干嘛。

举个例子:

form:focus {
  outline: solid;
}

表示仅仅当<form>处于focus状态时候,<form>元素outline出现。

form:focus-within {
  outline: solid;
}

表示<form>元素自身,或者<form>内部的子元素处于focus状态时候,<form>元素outline出现。换句话说,子元素focus,可以让任意父级元素样式发生变化。

这个新特性的支持,未来势必会带来更加丰富的交互形式,以及特定场景简化我们实现。

兼容性

兼容性如下截图:

:focus-within兼容性

还是挺喜人的。

二、CSS :focus-within伪类实际应用举例

1. 表单输入勿扰模式

当我们表单输入,选择或者进行交互时候,页面上表单以为其他内容全部都不可见。

CSS代码如下:

form {
    outline: 2000px solid hsla(0,0%,100%,0);
    transition: outline .2s;
    position: relative;
    z-index: 1;
}
form:focus-within {
    outline: 2000px solid hsla(0,0%,100%,1);
}

效果如下截屏gif:

form表单focus勿扰模式截图

大屏浏览器体验更有感觉,您可以狠狠的点击这里::focus-within伪类选择器与表单输入勿扰模式demo

本案例是真实的勿扰模式效果可以放心大胆在实际项目中使用,因为这是一个体验增强的交互,就算浏览器不支持,对原本功能也不会有任何影响。

2. 带计数文本域的focus高亮

对于带计数的组件化的多行文本域,计数的数值通常是设计在文本域的右下角,这样能够适应各种复杂的场景。如下截图:

定位在右下角的计数效果

然而这种设计对我们的布局实现带来的挑战。

我们通常想到的方法是,计数元素浮在下面的textarea元素上。然而这种时间存在有致命的不足,那就是输入框的内容有可能和我们的计数重叠,以及出现的滚动条和技术文本重叠,如下截图所示:

计数和滚动条重叠

您可以狠狠地点击这里:带计数文本域自身:focus高亮重叠问题demo


因此,我们通常做法就是:边框使用父级<div>元素模拟,文本域元素和技术元素上下依次排列(非重叠),文本域原本的边框去除。

假设HTML如下:

<div class="textarea-x">
    <textarea></textarea>
    <label class="textarea-count">0/250</label>
</div>

则核心CSS这样:

/* 父级div元素模拟边框 */
.textarea-x {
    border: 1px solid #d0d0d5;
    border-radius: 4px;
    background-color: #fff;
}
/* 文本域原本的边框去除 */
.textarea-x > textarea {
    border: 0;
    background: none;
}

然而上面的实现有个非常严重的不足,那就是<textarea>元素:focus时候,边框无法高亮,因为CSS中没有父选择器

因此,实际开发的时候,我们会使用相邻兄弟选择器,以及新建一个兄弟元素来模拟边框。

HTML结构如下:

<div class="textarea-x">
    <textarea></textarea>
    <div class="textarea"></div>
    <label class="textarea-count">0/250</label>
</div>

原理如下图示意:

新建一个div元素模拟边框

对应的核心CSS代码如下:

.textarea-x {
    position: relative;
    z-index: 0;
}
.textarea-x > textarea {
    border: 0;
    background: none;
    outline: 0;
    resize: none;
    position: relative;
}
.textarea-x > .textarea {
    position: absolute;
    border: 1px solid #d0d0d5;
    border-radius: 4px;
    background-color: #fff;
    top: 0; bottom: 0; left: 0; right: 0;  
    z-index: -1;
}
.textarea-x > :focus + .textarea {
    border-color: #00a5e0;    
}

由于.textarea元素和原生的<textarea>元素是相邻兄弟关系,因此我们可以借助相邻兄弟选择器,让<textarea>元素focus时候后面用来模拟边框的.textarea元素高亮。也就是这么一段CSS代码:

.textarea-x > :focus + .textarea {
    border-color: #00a5e0;    
}

这种实现兼容IE7+浏览器。下图为实现后的效果截图:

相邻兄弟元素focus后高亮效果截图

当然涉及到具体代码,还是有很多细节需要注意的,您可以狠狠地点击这里:利用相邻兄弟选择器模拟带计数文本域:focus效果demo


但是,现在有了CSS :focus-within伪类选择器,我们的事情就简单多了。

还是开始提到的父级<div>元素模拟的做法,然后,配合这么一句CSS就可以了:

.textarea-x:focus-within {
    border-color: #00a5e0;    
}

就是这么简单。

您可以狠狠地点击这里::focus-within伪类与带计数文本域高亮demo

focus时候效果如下截图:

:focus-within父级边框高亮截图

三、更多思路、更棒案例和结束语

:focus-within伪类原本设计的作用是原生表单元素focus时候,祖先<form>元素可以也有状态变化。

但是在我看来,:focus-within功能之强悍,远远不是仅仅和祖先<form>元素玩过家家这么简单。

理论上,只要页面上任意一个元素focus,通过:focus-within就能对页面上任意的元素进行样式控制。

例如:

html:focus-within xxx {
    /*xxx跑得了吗?跑不了*/
}

我想到的有:

  1. 相邻选择器,加号,或者弯弯都只能选择后面的元素,但是,有了:focus-within,我们就可以对前面元素进行控制,虽然只能在:focus行为场景下。例如请重复输入密码时候,让上面的输入密码框也高亮。
  2. 基于纯CSS和focus行为实现下拉列表的交互成为了可能。只要把按钮和下拉元素放在一个容器中,就可以了。

    您可以狠狠地点击这里::focus-within伪类与纯CSS下拉列表demo

    点击“我的消息”,就会出现下拉列表;此时点击下拉列表里面的条目,下拉元素也一直显示(因为列表是可focus的<a>元素)。只有点击页面空白元素或其他位置,下拉列表才会隐藏。完全符合实际项目开发的下拉列表交互需求。

    交互效果如下GIF截屏所示:

    :focus-within与纯CSS实现下拉

    核心代码如下:

    <div class="details">
        <a href="javascript:" class="summary">我的消息</a> 
        <div class="box">
            <a href="javascript:">我的回答<sup>12</sup></a>
            <a href="javascript:">我的私信</a>
            <a href="javascript:">未评价订单<sup>2</sup></a>
            <a href="javascript:">我的关注</a>
        </div>
    </div>
    .box {
        display: none;
    }
    .details:focus-within .summary {
        background-color: #fff;
    }
    .details:focus-within .box {
        display: block;
    }

抛砖引玉,集思广益。

补充于2018-06-27
IE Edge目前最新版还没支持:focus-within,而且,测试表明,当IE支持选择器和:focus-within一起使用的时候,会自动忽略整条语句,例如:

/* IE浏览器忽略所有伪类,CSS语句无效 */
.example:hover,
.example:active,
.example:focus-within {
    color: red;
}

需要这样书写:

/* IE浏览器可识别 */
.example:hover,
.example:active {
    color: red;
}
/* IE不可识别 */
.example:focus-within {
    color: red;
}

这样现象也会我们区分IE浏览器和Chrome/Firefox提供了思路,例如,我们希望某一个元素IE浏览器下是红色,其他浏览器下是蓝色,可以这样:

.browser {
    color: red;
}
.browser,
.xxxx:focus-within {
    color: blue;
}

补充于2019-06-26

caniuse上显示Safari浏览器是支持:focus-within伪类的,但是我的demo测试发现却没有效果,怎么回事?后来一番研究知道了原因,原来在Safari浏览器下,<a>元素居然不能被Tab focus。怎么办呢,很简单,给<a>元素设置tabindex="0"就可以了。生效原因可参见这篇文章:“HTML tabindex属性与web网页键盘无障碍访问”。


感谢阅读,欢迎交流!

(本篇完) 是不是学到了很多?可以分享到微信
有话要说?点击这里



发表评论(目前16条评论)

  1. yyyao说道:

    下拉列表如果我用max-height作为聚焦前后的变化属性,当我为max-height加了transition动画,当下拉列表的高大于里面DOM元素本身的高度,滚动条就会闪一下,请问是什么原因呢
    .fold_body{
    overflow: auto;
    transition: max-height 0.2s;
    max-height: 0px;
    }
    .fold:focus-within .fold_body{
    max-height: 50vh;
    }

    第一条数据
    第二条数据
    第三条数据

  2. Mocuishle说道:

    图片都挂掉了

  3. 666说道:

    “文本域元素和技术元素上下依次排列”,应该是 计数元素 。

  4. zzz说道:

    又学到一点东西!

  5. 长江长江我是黄河说道:

    form:focus {
    outline: solid;
    }
    这句怎么理解,单独的空的form有focus事件吗,还是需要加“tabindex”? 求解释!

  6. 6K说道:

    鑫哥V5

  7. 刘东奇说道:

    6666果然神器,之前确实被不能方便的控制父元素样式困扰过,现在有了部分解决办法

  8. qq说道:

    CSS世界 出个kindle 版本吧。

  9. 崔某说道:

    鑫哥的这篇文章给了我很多启发,我发现:hover,:active 等选择器也有同样的效果,
    具体效果见demo:http://47.97.201.13/testTen.html,纯css实现三级联动菜单效果
    域名备案太难,只好用公网地址裸奔QAQ

  10. 奔跑的小蜗牛说道:

    鑫哥 最后面的那个实例 我的消息 貌似触发不了啊

  11. 依韵说道:

    文本域元素和技术元素上下依次排列。 应该是“计数”

  12. ferrinweb说道:

    二-1节错字:“页面上表单以为其他内容全部都不可见” → “页面上表单以外其他内容全部都不可见”。