动手写个数字输入框2:起手式——拦截非法字符

简介: 最近在用Polymer封装纯数字的输入框,开发过程中发现不是坑,也有不少值得研究的地方。本系列打算分4篇来叙述这段可歌可泣的踩坑经历: 1. [《动手写个数字输入框1:input[type=number]的遗憾》](http://www.

前言

 最近在用Polymer封装纯数字的输入框,开发过程中发现不是坑,也有不少值得研究的地方。本系列打算分4篇来叙述这段可歌可泣的踩坑经历:

  1. [《动手写个数字输入框1:input[type=number]的遗憾》](http://www.cnblogs.com/fsjohnhuang/p/6918305.html)
  2. 《动手写个数字输入框2:起手式——拦截非法字符》
  3. 《动手写个数字输入框3:痛点——输入法是个魔鬼》
  4. 《动手写个数字输入框4:魔鬼在细节——打磨光标位置》

从源头抓起——拦截非法字符

 从《动手写个数字输入框1:input[type=number]的遗憾》中我们了解到input[type=number]基本不能满足我们的需求,为了简单化我们就直接在input[type=text]上加工出自己的数字输入框吧。
 首先很明确的一点是最终数值可以包含以下字符[+-0-9.],而可输入的功能键为Backspace,Delete,Arrow-Left,Arrow-Right,Arrow-Up,Arrow-DownTab
于是我们可以设置如下规则了

// 断言库
const keyCode = anyPass(prop('keyCode'), prop('which'))
const isBackspace = eq(8)
        , isDelete = eq(46)
        , isArrowLeft = eq(37)
        , isArrowRight = eq(38)
        , isArrowUp = eq(39)
        , isArrowDown = eq(40)
        , isTab = eq(9)
        , isMinus = anyPass(eq(109), eq(189))
        , isDot = anyPass(eq(110), eq(190))
        , isDigit = anyPass(
                                    allPass(lte(49), gte(57))
                                    , allPass(lte(96), gte(105)))
        , isPlus = anyPass(
                                comp(eq(107), keyCode)
                                , allPass(
                                        prop('shiftKey')
                                        , comp(eq(187), keyCode)))

const isValid  = anyPass(
                                    comp(
                                        anyPass(isBackspace, isDelete, isArrowLeft
                                            , isArrowLeft, isArrowUp, isArrowDown
                                            , isTab, isMinus, isDot, isDigit)
                                        , keyCode)
                                    , isPlus)

$('input[type=text]').addEventListener('keydown', e => {
    if (!isValid(e)){
        e.preventDefault()
    }
})

扩大非法字符集

 还记得min,max,precision吗?

  1. 当min大于等于0时,负号应该被纳入非法字符;
  2. 当max小于0时,正号应该被纳入非法字符;
  3. 当precision为0时,小数点应该被纳入非法字符。于是我们添加如下规则,并修改一下isValid就好了
// 获取min,max,precision值
const lensTarget = lens(a => a.target || a.srcElement)
        , lensMin = lens(a => Number(a.min) || Number(attr(a, 'min')) || Number.MIN_SAFE_INTEGER)
        , lensMax = lens(a => Number(a.max) || Number(attr(a, 'max')) || Number.MAX_SAFE_INTEGER)
        , lensPrecision = lens(a => Number(a.precision) || Number(attr(a, 'precision')) || 0)
        , lensValue = lens(a => a.value)

const lensTargetMin = lcomp(lensTarget, lensMin)
        , lensTargetMax = lcomp(lensTarget, lensMax)
        , lensTargetPrecision = lcomp(lensTarget, lensPrecision)

const isValid  = anyPass(
                                    comp(
                                        anyPass(isBackspace, isDelete, isArrowLeft
                                            , isArrowLeft, isArrowUp, isArrowDown
                                            , isTab, isDigit)
                                        , keyCode)
                                    , allPass(
                                            comp(gt(0), view(lensTargetMin))
                                            , comp(isMinus, keyCode))
                                    , allPass(
                                            comp(lte(0), view(lensTargetMax))
                                            , isPlus)
                                    , allPass(
                                            comp(lt(0), view(lensTargetPrecision))
                                            , comp(isDot, keyCode)))

预判断

 到这里为止我们已经成功地拦截了各种非法字符,也就是最终值必须之含[+-0-9.],但含这些字符跟整体符合数值格式就是两回事了。因此我们要继续补充下面两步,并且由于keydown事件触发时value值还没被修改,于是我们需要将value值和当前输入值做组合来做预判,进一步扩大非法字符集。

  1. 通过正则检查最终值是否符合格式要求(是否存在多个小数点也会在这一步处理掉);
  2. 检查最终值是否在minmax范围内。
const isValidStr = precision =>
                                     a => RegExp("^[+-]?[0-9]*"+ (precision ? "(\\.[0-9]{0," + precision + "})?" : "") + "$").test(a)
const lensValue = lens(a => a.value)
      , lensTargetValue = lcomp(lensTarget, lensValue)

$('input[type=text]').addEventListener('keydown', e => {
    var prevented = true
    // 拦截非法字符
    if (isValid(e)){
        prevented = false

        // 预判断
        if (anyPass(comp(anyPass(isMinus, isDigit, isDot), keyCode), isPlus)(e)){
            var str = view(lensTargetValue)(e) + prop('key')(e)
            // 预判断格式
            prevented = !isValidStr(view(lensTargetPrecision)(e))(str)

            // 预判断值范围
            if (!prevented){
                if (str == '-') str = '-0'
                if (str == '+') str = '0'
                if (str == '.') str = '0'

                prevented = !allPass(
                                            gte(view(lensTargetMax)(e))
                                            , lte(view(lensTargetMin)(e)))(Number(str))
            }
        }
    }

    if (prevented){
        e.preventDefault()
    }
})

附录:工具函数

// 工具函数,请无视我吧: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 = Number(args[0]) || 1
                        , ret = []
                    while (times--){
                        var tmpRet
                        try{
                            tmpRet = o[m].apply(o, args4m)
                        }
                        catch(e){
                            tmpRet = void 0
                        }
                        ret.push(tmpRet)
                    }
                    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 and = (...args) => args.reduce((accu, x) => accu && x, true)
const or = (...args) => args.reduce((accu, x) => accu || x, false)
const allPass = (...fns) => v => fns.reduce((accu, x) => accu && x(v), true)
const anyPass = (...fns) => v => fns.reduce((accu, x) => accu || x(v), false)
const eq = a => b => a === b
const gt = a => b => a > b
const gte = a => anyPass(eq(a), gt(a))
const lt = a => b => a < b
const lte = a => anyPass(eq(a), lt(a))
const prop = k => o => o[k]
const lens = (g, s) => ({getter: g, setter: s})
const lensPath = (...args) => ({ getter: a => args.reduce((accu, x) => accu && accu[x], a) })
const lcomp = (...lenses) => ({ getter: a => lenses.reduce((accu, lens) => accu && lens.getter(accu), a)})
const view = l => a => l.getter(a)

const $ = invoker(1, document, "querySelector")
const attr = (o, a) => invoker(1, o, 'getAttribute')(a)

总结

 现在可以终于可以牢牢控制住用户输入了,直到用户切换到IME为止:-<当使用IME输入时会发现上述措施一点用也没有,不用皱眉了,后面我们会一起把IME KO掉!
尊重原创,转载请注明来自:http://www.cnblogs.com/fsjohnhuang/p/6929167.html ^_^肥仔John

目录
相关文章
|
12月前
|
存储 Java
Java扫描某个文件夹且要保证不重复扫描,如何实现?
【10月更文挑战第18天】Java扫描某个文件夹且要保证不重复扫描,如何实现?
225 3
|
10月前
|
XML JSON API
淘宝/天猫获得淘宝商品详情 API 返回值说明
API(Application Programming Interface),即应用程序编程接口,是一组用于构建软件应用程序的协议、例程和工具。它定义了不同软件组件之间如何进行交互,就像是软件世界中的 “语言翻译官” 或者 “沟通桥梁”。 简单来说,当你使用一个软件应用(比如手机上的天气应用)去获取天气数据时,这个应用就是通过 API 接口向提供天气数据的服务器发送请求,服务器收到请求后,通过 API 接口返回天气数据给应用,然后应用才能把天气信息展示给你。
119 2
|
存储 网络安全 网络架构
网络技术基础(5)——VRP和telnet
VRP(Versatile Routing Platform)是其数据通信产品的操作系统,支持路由器和交换机的高效运行,提供统一用户界面和控制平面功能。VRP通过组件化结构实现功能裁剪和扩展。设备初始化时,BootROM先启动,然后运行系统软件,从默认路径读取配置文件。管理设备可通过命令行或Web网管,命令行包括用户界面和级别控制,提供权限管理。文件系统管理涉及存储设备如SDRAM、Flash、NVRAM等。用户可使用 PuTTY 工具通过Console口本地登录,或通过SSH远程登录。VRP命令行具有编辑和在线帮助功能,提供undo命令恢复默认设置。
|
开发工具 git
成功解决git rebase问题:First, rewinding head to replay your work on top of it...
成功解决git rebase问题:First, rewinding head to replay your work on top of it...
|
前端开发 BI
前端基础(十)_标签分类(行级标签、块级标签、行块标签)
本文阐述了HTML标签的分类,包括行级标签、块级标签和行块标签,并展示了如何使用CSS的display属性实现标签类型之间的转换。
454 3
|
存储 缓存 算法
大文件 MD5 SHA 校验时间优化之路
【8月更文挑战第12天】处理大文件的MD5与SHA校验时,可通过选择高效算法实现、分块读取处理文件、利用多线程并行处理、采用硬件加速及缓存校验结果等方式优化校验时间。例如,使用性能良好的加密库如`pycryptodome`替代Python的标准`hashlib`库;分块读取文件并逐块计算哈希值,减少内存占用;利用多线程处理不同文件块;若条件允许,使用硬件加速如Intel AES-NI指令集;以及缓存重复校验的文件哈希值避免重算。这些策略可显著提高校验速度和系统效率。
931 1
ARM64技术 —— MMU处于关闭状态时,内存访问是怎样的?
ARM64技术 —— MMU处于关闭状态时,内存访问是怎样的?
|
存储 Shell Linux
【Shell 命令集合 系统设置 】Linux 将参数作为命令行输入 eval命令 使用指南
【Shell 命令集合 系统设置 】Linux 将参数作为命令行输入 eval命令 使用指南
279 0
|
Web App开发 存储 数据可视化
LaTeX基础使用【系列五】
LaTeX基础使用【系列五】
|
应用服务中间件 Linux Apache
阿里云服务器Linux一键安装web环境全攻略
阿里云服务器Linux一键安装web环境全攻略