动手写个数字输入框1:input[type=number]的遗憾
时间:2022-04-23
本文章向大家介绍动手写个数字输入框1:input[type=number]的遗憾,主要内容包括前言、HTML5带来的福利-input[type=number]、遗憾了我的哥、隐藏右侧微调按钮不完全解决方法、总结、基本概念、基础应用、原理机制和需要注意的事项等,并结合实例形式分析了其使用技巧,希望通过本文能帮助到大家理解应用这部分内容。
前言
最近在用Polymer封装纯数字的输入框,开发过程中发现不少坑,也有很多值得研究的地方。本系列打算分4篇来叙述这段可歌可泣的踩坑经历:
- 《动手写个数字输入框1:input[type=number]的遗憾》
- 《动手写个数字输入框2:起手式——拦截非法字符》
- 《动手写个数字输入框3:痛点——输入法是个魔鬼》
- 《动手写个数字输入框4:魔鬼在细节——打磨光标位置》
HTML5带来的福利-input[type=number]
<input
id="age" name="age"
type="number" step="1" min="0" max="120">
<input
id="inc"
type="button" value="增加">
<input
id="dec"
type="button" value="减少">
<script>
/* 工具函数...无视我吧:D */
const comp =
(...fns) =>
(...args) => {
let len = fns.length
while (len--){
args = [fns[len].apply(null, args)]
}
return args.length > 1 ? args : args[0]
}
const isSome = x => 'undefined' !== typeof x && x !== null
const invokerImpl =
n =>
o =>
m =>
(...args) => {
let args4m = args.splice(0, n)
, times = parseInt(args[0]) || 1
, ret = []
while (times--){
ret.push(o[m].apply(o, args4m))
}
return ret.length > 1 ? ret : ret[0]
}
const curry2Partial =
fn =>
(...args) => {
let c = true
, i = 0
, l = args.length
, f = fn
for (;c && i < l; ++i){
c = isSome(args[i])
if (c){
f = f(args[i])
}
}
return f
}
const invoker = curry2Partial(invokerImpl)
const invoker0 = invoker(0)
const $ = invoker(1, document, "querySelectorAll")
const invoker0AtEl = comp(invoker0, $)
/* 继续无视我吧:D */
const invoker0AtAge = invoker0AtEl('#age')
// input[type=number]提供stepUp和stepDown两个方法来增加和减少数字
const incAge = invoker0AtAge('stepUp')
, decAge = invoker0AtAge('stepDown')
$('#inc').addEventListener('click', incAge)
$('#dec').addEventListener('click', decAge)
</script>
input[type=number]
为我们提供了如下特性:
- 限制只能输入
[+-0-9.]
这几个字符 - 输入法(IME)也无法输入非
[+-0-9.]
的字符 - 自动的表单验证
-
min
和max
来限制数值的下限和上限; - 提供stepUp和stepDown两个方法实现以编程方式控制数值的增加和减少;
- 移动设备上当它获得焦点时,会出现数字键盘;
-
step
设置点击右侧微调按钮的步长(默认为1),可设置为小数、整数或any
。step
的值除了影响微调按钮的步长外,还影响表单验证信息。
<!-- step为整数时 -->
<input name="age1" type="number"
step="1" value="1">
<input name="age1" type="number"
step="1" value="1.1">
<!-- step为小数时 -->
<input name="age2" type="number"
step="0.1" value="1">
<input name="age2" type="number"
step="0.1" value="1.1">
<input name="age2" type="number"
step="0.1" value="1.11">
<!-- step为any时 -->
<input name="age3" type="number"
step="any" value="1">
<input name="age3" type="number"
step="any" value="1.1">
<input name="age3" type="number"
step="any" value="1.11">
<script>
// 显示 true false
$('[name=age1]').forEach(el => console.log(el.validity.valid))
// 显示 true true false
$('[name=age2]').forEach(el => console.log(el.validity.valid))
// 显示 true true true
$('[name=age3]').forEach(el => console.log(el.validity.valid))
</script>
另外,设置为any是让表单验证不受精度限制而已,实际上步长依然为1。
遗憾了我的哥
到这里我想大家都会发现怎么少了个精度设置呢?确实,input[type=number]
并没有为我们提供设置精度的属性或方法。但遗憾的何止是这个呢?
- 木有精度精度设置;
- 不想要右侧的微调按钮还不行了...
- 点击微调按钮和调用
stepUp
和stepDown
设置数值确实被限制在min
和max
区间,但直接输入却不受限制... - 可以输入多个小数点,如
2012.12.12
; - 设置
step=any
后,chrome on android的数字键盘居然没了小数点按键。 - 设置
step=any
后,点击微调按钮步长为1,但调用stepUp
和stepDown
则报
Uncaught DOMException: Failed to execute 'stepUp' on 'HTMLInputElement': This form element does not have an allowed value step.
隐藏右侧微调按钮不完全解决方法
Webkit和Gecko下可通过以下的CSS来隐藏右侧微调按钮
/* chrome */
input[type=number]::-webkit-outer-spin-button,
input[type=number]::-webkit-inner-spin-button{
-webkit-appearance: none!important;
margin: 0;
}
/* Firefox */
input[type=number]{
-moz-appearance: textfield;
}
IE就没辙了:-(
总结
也许你会问既然HTML5愿意为我们新增一个全新的input[type=number]
,为什么偏偏提供一个缺胳膊少腿的呢?只能说得哥情时失嫂意,既然它不满足,那就自己写写看咯:)
- 使用NumPy介绍期望值,方差和协方差
- Cleaver快速制作网页PPT
- 【学术】马尔可夫链的详细介绍及其工作原理
- 想把自拍背景改成马尔代夫?手把手教你用深度学习分分钟做到
- 还记得谷歌之前发现的两颗行星吗?今天谷歌对此披露了重要技术细节
- Golang调用动态库so
- 30多条mysql数据库优化方法,千万级数据库记录查询轻松解决
- 【学术】谷歌AI课程附带的机器学习术语整理(超详细!)
- 真疯了!Java 9 还没会用,Java 10 就要来了!
- 关于 Go 中 Map 类型和 Slice 类型的传递
- Go语言与面向对象编程
- 【Golang语言社区】四川麻将随机初始化牌型结构
- 所历前端“姿势”更替记(其一)
- Windows下效率必备软件
- JavaScript 教程
- JavaScript 编辑工具
- JavaScript 与HTML
- JavaScript 与Java
- JavaScript 数据结构
- JavaScript 基本数据类型
- JavaScript 特殊数据类型
- JavaScript 运算符
- JavaScript typeof 运算符
- JavaScript 表达式
- JavaScript 类型转换
- JavaScript 基本语法
- JavaScript 注释
- Javascript 基本处理流程
- Javascript 选择结构
- Javascript if 语句
- Javascript if 语句的嵌套
- Javascript switch 语句
- Javascript 循环结构
- Javascript 循环结构实例
- Javascript 跳转语句
- Javascript 控制语句总结
- Javascript 函数介绍
- Javascript 函数的定义
- Javascript 函数调用
- Javascript 几种特殊的函数
- JavaScript 内置函数简介
- Javascript eval() 函数
- Javascript isFinite() 函数
- Javascript isNaN() 函数
- parseInt() 与 parseFloat()
- escape() 与 unescape()
- Javascript 字符串介绍
- Javascript length属性
- javascript 字符串函数
- Javascript 日期对象简介
- Javascript 日期对象用途
- Date 对象属性和方法
- Javascript 数组是什么
- Javascript 创建数组
- Javascript 数组赋值与取值
- Javascript 数组属性和方法
- PHP实现小程序批量通知推送
- Thinkphp5.0 框架Model模型简单用法分析
- php设计模式之单例模式用法经典示例分析
- PHP实现统计代码行数小工具
- redis+php实现微博(三)微博列表功能详解
- php设计模式之工厂模式用法经典实例分析
- laravel 关联关系遍历数组的例子
- PHP开启目录引索+fancyindex漂亮目录浏览带搜索功能
- 解决Laravel 使用insert插入数据,字段created_at为0000的问题
- 关于php unset对json_encode的影响详解
- php进行md5加密简单实例方法
- Laravel timestamps 设置为unix时间戳的方法
- php实现每日签到功能
- php+redis实现消息队列功能示例
- php使用curl模拟浏览器表单上传文件或者图片的办法