动手写个数字输入框3:痛点——输入法是个魔鬼

简介:

前言

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

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

IE的先进性

 辛辛苦苦终于控制只能输入数字了,但只要用户启用了输入法就轻松突破我们的重重包围:-<心碎得一地都是。这是我们会想到底有没有一个API可以禁用输入法呢?答案是有的,但出人意料的是只有IE才支持。

<style>
    .disabled-ime-mode{
      /*ime-mode为CSS3规则
         *取值
         *auto: 不影响IME的状态,默认值
         *normal: 正常的IME状态
         *active: 激活本地语言输入法
         *inactive: 激活非本地语言输入法
         *disabled: 禁用IME
         */
        ime-mode: disabled;
    }
</style>

 而其他浏览器就呵呵了。。。

别无他法只能补救~

 由于chrome、firefox等无法通过样式ime-mode来处理,因此想到依葫芦画瓢,同样在keydown事件中对特定的keyCode进行拦截过滤就好了,谁知道在输入法中按下字符键时keydown事件的keyCode永远是229。其规律为:

  1. 按字符键时,keydown中keyCode恒为229,且key为Undefined;而keyup中才会得到正确的keyCode,且key为正确的字符。
  2. entershift时仅触发keydown不会触发keyup,而keyCode为229。
    因此我们能做的是
  3. 通过keyup事件作事后补救措施;
  4. 在keydown中拦截输入法中输入的entershift按键事件,然后自行出发keyup事件执行补救措施。
    废话少讲,上代码!
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)
        , isHome = eq(36)
        , isEnd = eq(35)
const isValidStr = precision =>
                                     a => RegExp("^[+-]?[0-9]*"+ (precision ? "(\\.[0-9]{0," + precision + "})?" : "") + "$").test(a)

// 获取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, (o, v) => o.value = v)
        , lensDataValue = lens(a => a && a.getAttribute('data-value'), (a, v) => a && a.setAttribute('data-value', v))

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

const isIME = eq(229)
const isValidChar = c => /[-+0-9.]/.test(c)
const invalid2Empty = c => isValidChar(c) ? c : ''
const recoverValue = v => flatMap(CharSequence(v), invalid2Empty)

// 是否激活IME
const isInIME = comp(isIME, keyCode)
// 是否为功能键
        , isFnKey = comp(anyPass(isArrowLeft, isArrowRight, isArrowUp, isArrowDown, isBackspace, isDelete, isHome, isEnd), keyCode)

$('input[type=text]').addEventListener('keydown', e => {
    var el = view(lensTarget)(e)
        , val = view(lensTargetValue)(e)
    // 暂存value值,keyup时发现问题可以恢复出厂设置
    set(lensDataValue)(el)(val)

    if (isInIME(e)){
        fireKeyup(el)
    }
})
$('input[type=text]').addEventListener('keyup', e => {
    if (isFnKey(e)) return

    var el = view(lensTarget)(e)
        , v = view(lensValue)(el)
        , p = view(lensTargetPrecision)(e)
        , isValid = isValidStr(p)
        , max = view(lensMax)(el)
        , min = view(lensMin)(el)

    var val = recoverValue(v)
    var setVal = set(lensValue)(el)
    if (isValid(val)){
        if (val !== v){
            setVal(val)
        }
        else{
            var n = Number(v)
            if (!gte(max)(n)){
                setVal(max)
            }
            if (!lte(min)(n)){
                setVal(min)
            }
        }
    }
    else{
        setVal(attr(el, 'data-value'))
    }
})

附录:工具函数

// 工具函数,请无视我吧: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) => lenses
const view = lenses => a => {
    if (!~Object.prototype.toString.call(lenses).indexOf('Array')){
        lenses = [lenses]
    }
    return lenses.reduce((accu, lens) => accu && lens.getter(accu), a)
}
const set = lenses => a => v => {
    if (!~Object.prototype.toString.call(lenses).indexOf('Array')){
        lenses = [lenses]
    }
    var setLens = lenses.pop()
    var o = view(lenses)(a)
    if (o){
        setLens.setter(o, v)
    }
}

const $ = invoker(1, document, "querySelector")
const attr = (o, a) => invoker(1, o, 'getAttribute')(a)
const flatMap = (functor, f) => {
    return functor.flatMap(f)
}
function CharSequence(v){
    if (this instanceof CharSequence);else return new CharSequence(v)
    this.v = v
}
CharSequence.prototype.flatMap = function(f){
    return this.v.split('').map(f).join('')
}

const fireKeyup = (el) => {
    if (KeyboardEvent){
        // DOM3
        var e = new KeyboardEvent('keyup')
        el.dispatchEvent(e)
    }
    else{
        // DOM2
        var e = document.createEvent('KeyboardEvent')
        e.initEvent('keyup', true, true)
        el.dispatchEvent(e)
    }
}

未完待续

 到这里我们已经成功地控制了IME下的输入,虽然事后补救导致用户输入出现闪烁的现象:D那是不是就over了呢?当然不是啦。
用户输入时,光标位置是随机的,于是遗留以下问题:

  1. 在keydow中预判断值合法性时,是假定光标位置处于行尾,将导致预判失误;
  2. 在keyup中对value重新赋值时会导致光标移动到行尾,严重中断了用户的输入流程;
  3. type=text会导致在移动端无法自动显示数字键盘。

总结

 后面我们会针对上述问题继续探讨,敬请留意!
尊重原创,转载请注明来自:http://www.cnblogs.com/fsjohnhuang/p/6953984.html ^_^肥仔John

如果您觉得本文的内容有趣就扫一下吧!捐赠互勉!

分类: HTML5, JavaScript
0
0
« 上一篇: 动手写个数字输入框2:起手式——拦截非法字符
» 下一篇: (cljs/run-at (->JSVM :browser) "语言基础")
posted @ 2017-06-07 09:25 ^_^肥仔John 阅读( 503) 评论( 0) 编辑 收藏
 
相关文章
|
1天前
|
云安全 数据采集 人工智能
古茗联名引爆全网,阿里云三层防护助力对抗黑产
阿里云三层校验+风险识别,为古茗每一杯奶茶保驾护航!
古茗联名引爆全网,阿里云三层防护助力对抗黑产
|
5天前
|
Kubernetes 算法 Go
Kubeflow-Katib-架构学习指南
本指南带你深入 Kubeflow 核心组件 Katib,一个 Kubernetes 原生的自动化机器学习系统。从架构解析、代码结构到技能清单与学习路径,助你由浅入深掌握超参数调优与神经架构搜索,实现从使用到贡献的进阶之旅。
277 139
|
5天前
|
人工智能 中间件 API
AutoGen for .NET - 架构学习指南
《AutoGen for .NET 架构学习指南》系统解析微软多智能体框架,涵盖新旧双架构、核心设计、技术栈与实战路径,助你从入门到精通,构建分布式AI协同系统。
294 142
|
16天前
|
存储 关系型数据库 分布式数据库
PostgreSQL 18 发布,快来 PolarDB 尝鲜!
PostgreSQL 18 发布,PolarDB for PostgreSQL 全面兼容。新版本支持异步I/O、UUIDv7、虚拟生成列、逻辑复制增强及OAuth认证,显著提升性能与安全。PolarDB-PG 18 支持存算分离架构,融合海量弹性存储与极致计算性能,搭配丰富插件生态,为企业提供高效、稳定、灵活的云数据库解决方案,助力企业数字化转型如虎添翼!
|
11天前
|
缓存 并行计算 PyTorch
144_推理时延优化:Profiling与瓶颈分析 - 使用PyTorch Profiler诊断推理延迟,优化矩阵运算的独特瓶颈
在2025年的大模型时代,推理时延优化已经成为部署LLM服务的关键挑战之一。随着模型规模的不断扩大(从数亿参数到数千亿甚至万亿参数),即使在最先进的硬件上,推理延迟也常常成为用户体验和系统吞吐量的主要瓶颈。
357 147
|
5天前
|
人工智能 移动开发 自然语言处理
阿里云百炼产品月刊【2025年9月】
本月通义千问模型大升级,新增多模态、语音、视频生成等高性能模型,支持图文理解、端到端视频生成。官网改版上线全新体验中心,推出高代码应用与智能体多模态知识融合,RAG能力增强,助力企业高效部署AI应用。
294 1
|
11天前
|
机器学习/深度学习 存储 缓存
92_自我反思提示:输出迭代优化
在大型语言模型(LLM)应用日益普及的今天,如何持续提升模型输出质量成为了业界关注的核心问题。传统的提示工程方法往往依赖一次性输入输出,难以应对复杂任务中的多轮优化需求。2025年,自我反思提示技术(Self-Reflection Prompting)作为提示工程的前沿方向,正在改变我们与LLM交互的方式。这项技术通过模拟人类的自我反思认知过程,让模型能够对自身输出进行评估、反馈和优化,从而实现输出质量的持续提升。
429 136
|
15天前
|
存储 人工智能 搜索推荐
终身学习型智能体
当前人工智能前沿研究的一个重要方向:构建能够自主学习、调用工具、积累经验的小型智能体(Agent)。 我们可以称这种系统为“终身学习型智能体”或“自适应认知代理”。它的设计理念就是: 不靠庞大的内置知识取胜,而是依靠高效的推理能力 + 动态获取知识的能力 + 经验积累机制。
414 135
|
14天前
|
存储 人工智能 Java
AI 超级智能体全栈项目阶段二:Prompt 优化技巧与学术分析 AI 应用开发实现上下文联系多轮对话
本文讲解 Prompt 基本概念与 10 个优化技巧,结合学术分析 AI 应用的需求分析、设计方案,介绍 Spring AI 中 ChatClient 及 Advisors 的使用。
543 133
AI 超级智能体全栈项目阶段二:Prompt 优化技巧与学术分析 AI 应用开发实现上下文联系多轮对话