CSS 美化滑动输入条 input range

简介: CSS 美化滑动输入条 input range
欢迎关注我的公众号:前端侦探


关于原生 input range 滑动输入条如何自定义样式一直都是我心里的一道坎,一般情况下,可以很轻易的美化到这个程度


image.png

为什么很容易呢?因为这些都是有对应的伪元素可以修改的

::-webkit-slider-container {
  /*可以修改容器的若干样式*/
}
::-webkit-slider-runnable-track {
  /*可以修改轨道的若干样式*/
}
::-webkit-slider-thumb {
  /*可以修改滑块的若干样式*/
}


可是,偏偏没有已滑过部分的样式,如果要定义下面这样的样式,单纯的 CSS 可能没办法实现了

image.png

注意:Firefox 有单独的伪类可以修改,本文讨论的是 Chrome 实现方案


一、我的实现思路


既然没有专门的伪元素可以修改已滑过部分的颜色,而且只有滑块是可动的,是不是可以在滑块上下手呢?


假设滑块左边有一个矩形,是跟随滑块一起的,


image.png

当这个矩形足够长时,能够完全覆盖左边的轨道,在可视范围内,是不是就可以表示左边的已滑过部分了呢?示意如下(左边半透明表示滑动条之外)


image.png

尝试过伪元素的想法,像这样

::-webkit-slider-thumb::after{
  /*本想绘制一个足够长的矩形*/
}


可惜,伪元素里并不能再次生成伪元素。


所以,如何在元素之外绘制一个矩形呢?


二、通过 border-image 在元素之外绘制图形


有哪些方式可以在元素之外绘制图形呢?想了一下,有 box-shadowoutline,但是好像并不适合这种情况,我们需要绘制的是一个尺寸可控的矩形,而这两种方式都不能很好地控制形状。那还有其他方式吗?


还真有!前两天刚看到张鑫旭老师的一篇文章:被低估的border-image属性,其中有一个特性就是在元素之外构建图像,并且不占据任何空间。赶紧试试,这里绘制一个宽度为99vw的矩形(足够覆盖滑动条就行了),代码如下

::-webkit-slider-thumb {
    -webkit-appearance: none;
    appearance: none;
    width: 20px;
    height: 20px;
    border-radius: 50%;
    background-color: #f44336;
    border: 1px solid transparent;
    margin-top: -8px;
    border-image: linear-gradient(#f44336,#f44336) 0 fill / 8 20 8 0 / 0 0 0 99vw; /*绘制元素外矩形*/
}


效果如下

image.png

注意几点:


  1. border-image 要想生效,必须指定border,这里设置的是border: 1px solid transparent;
  2. 矩形是通过线性渐变绘制的 linear-gradient(#f44336,#f44336)
  3. border-image 中8 20 8 0表示border-image-width,距离上、右、下、左的距离,由于滑块自身大小是 20 * 20,所以这个可以确定高度是 4 (20 - 8- 8),位置是滑块自身的最左边(距离右边是20)
  4. border-image 中 0 0 0 99vw表示 border-image-outset扩展大小,这里指的是向左扩展99vw的距离


接下来通过overflow:hidden隐藏外面的部分就可以了

::-webkit-slider-container {
    /*其他样式*/
    overflow: hidden;
}


image.png

完整代码可以访问:input range (codepen.io)点击预览


下面附上完整代码(最近codepen貌似不太稳定)

[type="range"] {
    -webkit-appearance: none;
    appearance: none;
    margin: 0;
    outline: 0;
    background-color: transparent;
    width: 500px;
}
[type="range"]::-webkit-slider-runnable-track {
    height: 4px;
    background: #eee;
}
[type="range" i]::-webkit-slider-container {
    height: 20px;
    overflow: hidden;
}
[type="range"]::-webkit-slider-thumb {
    -webkit-appearance: none;
    appearance: none;
    width: 20px;
    height: 20px;
    border-radius: 50%;
    background-color: #f44336;
    border: 1px solid transparent;
    margin-top: -8px;
    border-image: linear-gradient(#f44336,#f44336) 0 fill / 8 20 8 0 / 0px 0px 0 2000px;
}


三、还是有一些局限


上面的实现成本其实是很低的,相比常规的实现基础上,仅仅增加了1行用于绘制元素之外的矩形。

border-image: linear-gradient(#f44336,#f44336) 0 fill / 8 20 8 0 / 0px 0px 0 2000px;


但是,由于是通过超出隐藏的方式裁剪掉多出的部分,使得滑动条边缘是“一刀切”的,所以,如果要求滑动条带有圆角,这种实现方式就不行了


image.png

如果还有好的想法欢迎留言讨论


四、简单总结一下


关于border-image-outset 这个属性,其实之前已经在 MDN 上见识过了,但只是简单了解,还觉得很鸡肋,现在看来,这些属性不是没什么用,只是没有碰到适合应用的场景。下面简单总结一下:


  1. 滑动条有 3 个伪元素可以自定义容器、轨道、滑块
  2. 伪元素里不能再嵌套伪元素了
  3. 元素之外绘制有 box-shadow、outline、border-image 3种方法
  4. border-image 可以使用任意格式图片,包括 CSS 渐变
  5. 这个方案不能实现圆角


当然这些思路都只是“偏方”,像 Firefox 就完全支持自定义样式了,可惜桌面端还是 Chrome 的天下,只能慢慢期待一下 Chrome 后面的更新了。最后,如果觉得还不错,对你有帮助的话,欢迎点赞、收藏、转发❤❤❤


欢迎关注我的公众号:前端侦探
相关文章
|
6月前
|
前端开发
页面无法滑动-CSS: touch-action属性
页面无法滑动-CSS: touch-action属性
35 0
|
6月前
CSS3滑动轮播动画
CSS3滑动轮播动画
55 8
|
前端开发
css改input变输入框光标颜色demo效果示例(整理)
css改input变输入框光标颜色demo效果示例(整理)
|
25天前
|
前端开发 UED
CSS3:linear-gradient&input&inset&table
本文介绍了 CSS 中的 `linear-gradient` 渐变、`input` 样式、`inset` 简写属性、`border-collapse: collapse` 表格边框合并以及 `<label>` 元素的 `for` 属性。通过示例代码和解释,帮助读者理解这些 CSS 特性及其应用。
40 13
|
3月前
|
前端开发
背景滑动,动感加倍:CSS动画对角线效果全解析!
背景滑动,动感加倍:CSS动画对角线效果全解析!
|
3月前
|
前端开发
导航栏也疯狂:CSS动画让导航栏活起来,跟随鼠标滑动!
导航栏也疯狂:CSS动画让导航栏活起来,跟随鼠标滑动!
|
5月前
|
前端开发 iOS开发
通过css内修改input框placeholder样式
通过css内修改input框placeholder样式
87 1
|
6月前
|
前端开发
css去除滑动框
css去除滑动框
53 2
|
6月前
|
前端开发
css样式实现一个滑动按钮
css样式实现一个滑动按钮
53 0
|
前端开发
HTML与CSS实现网页的超链接及美化
HTML与CSS实现网页的超链接及美化
210 0
HTML与CSS实现网页的超链接及美化