• CSS魔法堂:改变单选框颜色就这么吹毛求疵!


    前言

     是否曾经被业务提出"能改改这个单选框的颜色吧!让它和主题颜色搭配一下吧!",然后苦于原生不支持换颜色,最后被迫自己手撸一个凑合使用。若抛开input[type=radio]重新开发一个,发现要模拟选中、未选中、不可用等状态很繁琐,而涉及单选框组就更烦人了,其实我们可以通过label::before:checkedtabindex,然后外加少量JavaScript脚本就能很好地模拟出一个样式更丰富的“原生”单选框。下面我们一起来尝试吧!

    单选框了解一下

     由于我们的目标是改变单选框颜色,其他外观特征和行为与原来的单选框一致,那么我们就必须先了解单选框原来的外观特征和行为主要有哪些。
    1.外观特征
    1.1.常态样式

    margin: 3px 3px 0px 5px;
    border: none 0;
    padding: 0;
    box-sizing: border-box;
    display: inline-block;
    line-height: normal;
    position: static;
    

    注意:外观上我们必须要保证布局特性和原生的一致,否则采用自定义单选框替换后很大机会会影响整体的布局,最后导致被迫调整其他元素的布局特性来达到整体的协调,从而扩大了修改范围。

    1.2.获得焦点的样式

    outline-offset: 0px;
    outline: -webkit-focu-ring-color auto 5px;
    

    注意:这里的获取焦点的样式仅通过键盘Tab键才生效,若通过鼠标点击虽然单选框已获得焦点,但上述样式并不会生效。

    1.3.设置为disabled的样式

    color: rgb(84, 84, 84);
    

    2.行为特征
     单选框的行为特征,明显就是选中与否,及选中状态的改变事件,因此我们必须保持对外提供change事件。
     另外值得注意的是,当通过键盘的Tab键让单选框获得焦点后,再按下Space键则会选中该单选框。

     有了上述的了解,我们可以开始着手撸代码了!

    少废话,撸代码

    上图中左侧就是原生单选框,右侧为我们自定义的单选框。从上到下依次为未选中选中获得焦点disabled状态的样式。

    CSS部分

    label.radio {
      /* 保证布局特性保持一致 */
      margin: 3px 3px 0px 5px;
      display: inline-block;
      box-sizing: border-box;
    
       12px;
      height: 12px;
    }
    
    .radio__appearance{
      display: block; /* 设置为block则不受vertical-align影响,从而不会意外影响到.radio的linebox高度 */
      position: relative;
      box-shadow: 0 0 0 1px tomato; /* box-shadow不像border那样会影响盒子的框高 */
      border-radius: 50%;
      height: 90%;
       90%;
      text-align: center;
    }
    label.radio [type=radio] + .radio__appearance::before{
      content: "";
      display: block;
      border-radius: 50%;
       85%;
      height: 85%;
    
      position: absolute;
      top: 50%;
      left: 50%;
      transform: translate(-50%, -50%);
    
      transition: background .3s;
    }
    label.radio [type=radio]:checked + .radio__appearance::before{
      background: tomato;
    }
    label.radio [type=radio][disabled] + .radio__appearance{
      opacity: .5;
    }
    label.radio:focus{
      outline-offset: 0px;
      outline: #999 auto 5px;
    }
    /* 通过鼠标单击获得焦点时,outline效果不生效 */
    label.radio.clicked{
      outline: none 0;
    }
    /* 自定义单选框的行为主要是基于原生单选框的,因此先将原生单选框隐藏 */
    label.radio input {
      display: none;
    }
    

    HTML部分

    <!-- 未选中状态 -->
    <label class="radio" tabindex="0">
      <input type="radio" name="a">
      <i class="radio__appearance"></i>
    </label>
    
    <br>
    
    <!-- 选中状态 -->
    <label class="radio" tabindex="0">
      <input type="radio" name="a" checked>
      <i class="radio__appearance"></i>
    </label>
    
    <br>
    
    <!-- disabled状态 -->
    <label class="radio">
      <input type="radio" name="a" disabled>
      <i class="radio__appearance"></i>
    </label>
    

    JavaScript部分

    var radios = document.querySelectorAll(".radio")
    radios.forEach(radio => {
      // 模拟鼠标点击后:focus样式无效
      radio.addEventListener("mousedown", e => {
        var tar = e.currentTarget
        tar.classList.add("clicked")
        var fp = setInterval(function(){
          if (document.activeElement != tar){
            tar.classList.remove("clicked")
            clearInterval(fp)
          }
        }, 400)
      })
      // 模拟通过键盘获得焦点后,按`Space`键执行选中操作
      radio.addEventListener("keydown", e => {
        if (e.keyCode === 32){
          e.target.click()
        }
      })
    })
    

    这个实现有3个注意点:

    1. 通过label传递鼠标点击事件到关联的input[type=radio],因此可以安心隐藏单选框又可以利用单选框自身特性。但由于label控件自身的限制,如默认不是可获得焦点元素,因此无法传递键盘按键事件到单选框,即使添加tabindex特性也需手写JS来实现;
    2. 当tabindex大于等于0时表示该元素可以获得焦点,为0时表示根据元素所在位置安排获得焦点的顺序,而大于0则表示越小越先获得焦点;
    3. 由于单选框的displayinline-block,因此单选框将影响line box高度。当自定义单选框内元素采用inline-block时,若vertical-align设置稍有不慎就会导致内部元素所在的line box被撑高,从而导致自定义单选框所在的line box高度变大。因此这里采用将内部元素的display均设置为block的做法,直接让vertical-align失效,提高可控性。

    通过opacity:0实现(2018/10/5追加)

     上面我们通过label关联display:noneinput[type=radio]从而利用input[type=radio]简化自定义单选框的实现,但依然要手写JS实现按Space键选中的行为特征,有没有另一种方式可以更省事呢?我们只是想让用户看不到原生单选框,那么直接设置为opacity:0不就可以了吗?!
    CSS部分

    .radio {
      /* 保证布局特性保持一致 */
      margin: 3px 3px 0px 5px;
      display: inline-block;
      box-sizing: border-box;
    
       13px;
      height: 13px;
    }
    /* 自定义单选框的行为主要是基于原生单选框的,因此先将原生单选框透明,且沾满整个父元素 */
    .radio input {
      opacity: 0;
      position: absolute;
      z-index: 1; /* 必须覆盖在.radio__appearance上才能响应鼠标事件 */
       100%;
      height: 100%;
    }
    .radio__container-box{
      position: relative;
       100%;
      height: 100%;
    }
    .radio__appearance{
      display: block; /* 设置为block则不受vertical-align影响,从而不会意外影响到.radio的linebox高度 */
      position: relative;
      box-shadow: 0 0 0 1px tomato; /* box-shadow不像border那样会影响盒子的框高 */
      border-radius: 50%;
      height: 90%;
       90%;
      text-align: center;
    }
    .radio [type=radio] + .radio__appearance::before{
      content: "";
      display: block;
      border-radius: 50%;
       85%;
      height: 85%;
    
      position: absolute;
      top: 50%;
      left: 50%;
      transform: translate(-50%, -50%);
    
      transition: background .3s;
    }
    .radio [type=radio]:checked + .radio__appearance::before{
      background: tomato;
    }
    .radio [type=radio][disabled] + .radio__appearance{
      opacity: .5;
    }
    .radio:focus-within .radio__appearance{
      outline-offset: 0px;
      outline: #999 auto 5px;
    }
    /* 通过鼠标单击获得焦点时,outline效果不生效 */
    .radio.clicked .radio_appearance{
      outline: none 0;
    }
    

    HTML部分

    <!-- 未选中状态 -->
    <span class="radio">
      <span class="radio__container-box">
        <input type="radio" name="a">
        <i class="radio__appearance"></i>
      </span>
    </span>
    
    <br>
    
    <!-- 选中状态 -->
    <span class="radio">
      <span class="radio__container-box">
        <input type="radio" name="a" checked>
        <i class="radio__appearance"></i>
      </span>
    </span>
    
    <br>
    
    <!-- disabled状态 -->
    <span class="radio">
      <span class="radio__container-box">
        <input type="radio" name="a" disabled>
        <i class="radio__appearance"></i>
      </span>
    </span>
    

    JavaScript部分

    var radios = document.querySelectorAll(".radio")
    radios.forEach(radio => {
      // 模拟鼠标点击后:focus样式无效
      radio.addEventListener("mousedown", e => {
        var tar = e.currentTarget
        tar.classList.add("clicked")
        var fp = setInterval(function(){
          if (!tar.contains(document.activeElement){
            tar.classList.remove("clicked")
            clearInterval(fp)
          }
        }, 400)
      })
    })
    

    总结

     对于复选框我们可以稍加修改就可以了,然后通过VUE、React等框架稍微封装一下提供更简约的API,使用起来就更方便了。
     尊重原创,转载请注明来自:https://www.cnblogs.com/fsjohnhuang/p/9741345.html _肥仔John

  • 相关阅读:
    C 运算符, 有符号数据运算,
    P1337 [JSOI2004]平衡点 / 吊打XXX 模拟退火
    [POI2011]Garbage 欧拉回路
    # bzoj2215: [Poi2011]Conspiracy 2-sat
    hdu1814 Peaceful Commission 2-sat
    2-sat相关复习
    #2718. 「NOI2018」归程 kruskal重构树
    JXOI2018守卫 区间DP
    [NOI1995]石子合并 四边形不等式优化
    3900: 交换茸角
  • 原文地址:https://www.cnblogs.com/fsjohnhuang/p/9741345.html
Copyright © 2020-2023  润新知