關于JavaScript限制字數(shù)的輸入框的那些事
前言
最近產(chǎn)品需要做不少輸入框,產(chǎn)品想要的交互效果是:用戶可以輸入中英文,隨著用戶輸入能實時顯示已經(jīng)輸入的字符個數(shù),當超過數(shù)量限制時輸入框邊框變紅,同時給用戶提示信息。
這交互聽起來沒啥問題,技術實現(xiàn)上似乎也沒啥難點。但是當我實現(xiàn)出來以后遇到中文輸入法就有坑了。
怎么個坑呢,且看下文~~
實時監(jiān)測輸入框內容長度所遇到的坑使用 oninput事件來監(jiān)聽
使用這個oninput事件的好處有2個:
當用戶通過右鍵復制改變輸入框內容時,可以監(jiān)聽到;
只有在輸入框內容發(fā)生變化時才會觸發(fā)此事件,比如用戶按下方向鍵、control/shift 等這些控制字符鍵時此事件是不會觸發(fā)的;
當你輸入英文字符或者數(shù)字時效果完美,甚至在你正常輸入中文時也效果完美。但當你非正常輸入中文時就出現(xiàn) bug 了。非正常輸入是怎樣的呢?看下面這張示例圖:

看到了嗎,在這種中文輸入方式下,其實用戶還沒有輸入他想輸入的中文,只是輸入了幾個拼音,但 input 事件被觸發(fā)了,而且監(jiān)聽到的輸入框value居然是d'd'd,不單單是拼音字符,還包括了分隔的點。假如輸入框內容長度被限制為不超過5,那么在截圖這種情況下,就會提示用戶字符長度超過限制!。這樣的交互效果當然不是產(chǎn)品想要的。
使用onkeydown/onkeypress/onkeyup事件來監(jiān)聽
這幾個事件的缺點是無法監(jiān)聽右鍵復制而來的輸入內容,但是否也會存在與input事件一樣的問題呢?
我做了幾個實驗,發(fā)現(xiàn)keydown、keyup都會遇到和 input一樣的問題,但keypress沒有這個問題,因為在中文輸入狀態(tài)下,keypress不會觸發(fā),不單是你輸入拼音的過程中不會觸發(fā),等你選中所要輸入的中文如“對對對”后也不會觸發(fā)。那么當輸入“對對對”后雖然超過了字符限制但無法給出字符長度超過限制!的提示。
折中解決方法
要想做到實時監(jiān)測內容長度,又想保證中文輸入法狀態(tài)下沒有 bug,我折騰了好久最后發(fā)現(xiàn) 臣妾做不到呀?。ㄒ悄奈缓澜苷业搅耍欢ㄒ嬖V我呀~~)。
所以最后犧牲了下用戶體驗,找到了一個折中的方式:輸入框失去焦點時(即blur),或者用戶輸入回車鍵時才進行內容長度的檢測。當然如果發(fā)現(xiàn)輸入框內容超過限制,要將光標停留在輸入框內,方便用戶進行修改。
哎,一說到用戶輸入回車鍵時才進行內容長度的檢測又得說說以前栽的坑了
輸入框中如何檢測輸入了回車鍵
其實這是一個很常見的交互,比如修改名稱時支持用戶輸入回車后直接保存、登錄時支持用戶輸入回車后直接登錄。但其中要小心的坑是:**中文輸入法下按回車鍵來輸入英文字符**。
中文輸入法下按回車鍵來輸入英文字符的過程舉例:
比如我要輸入賬號進行登錄,我的賬號是全英文的,我當前處于中文輸入法,但我懶得切換輸入法,于是我就直接敲了我的賬號(全英文字符),這時搜狗輸入法給我提示了一大串中文,然后我按了個回車,輸入框就輸入了我想要的英文字符。
在這種情況下,用戶雖然輸入了回車鍵,但用戶按下回車鍵只是想在中文輸入法下輸入英文字符而已,這個回車鍵并不是我們想要監(jiān)聽的回車鍵。那么怎么排除這種情況下的回車鍵呢?
一般來說監(jiān)聽回車鍵我們會用keydown事件或者keyup事件,實現(xiàn)代碼如下所示。那么是否這兩種方法都能過濾掉我們不想監(jiān)聽的回車鍵呢?
//方法一:使用 keydown 事件
input.onkeydown = function(e){
if(e.keyCode == 13){
//用戶輸入的是回車鍵
//做相關操作
}
}
//方法二:使用 keyup 事件
input.onkeyup = function(e){
if(e.keyCode == 13){
//用戶輸入的是回車鍵
//做相關操作
}
}
經(jīng)過試驗發(fā)現(xiàn):使用keydown是可以成功過濾的,但使用keyup不能。
那么我們來看看為什么?
是因為在keydown事件中:中文輸入法狀態(tài)下輸入的回車鍵,檢測的 keyCode為 229 而不是13;單純輸入一個回車時,keyCode才是13。
而在keyup事件中:中文輸入法狀態(tài)下輸入的回車鍵,檢測的 keyCode是13;單純輸入一個回車時,keyCode也是13。
下圖是我打印在 console 中的結果:(代碼示例見這里)

結語
關于輸入框涉及到的幾個事件:keydown/keyup/keypress/input/change
查看這里:http://www.dhdzp.com/article/21237.htm
- JavaScript動態(tài)提示輸入框輸入字數(shù)的方法
- JavaScript輸入框字數(shù)實時統(tǒng)計更新
- JavaScript中統(tǒng)計Textarea字數(shù)并提示還能輸入的字符
- Javascript實現(xiàn)真實字符串剩余字數(shù)提示的實例代碼
- .NET+JS對用戶輸入內容進行字數(shù)提示功能的實例代碼
- 利用Angularjs和原生JS分別實現(xiàn)動態(tài)效果的輸入框
- Angularjs中的驗證input輸入框只能輸入數(shù)字和小數(shù)點的寫法(推薦)
- AngularJS實現(xiàn)的輸入框字數(shù)限制提醒功能示例
相關文章
JavaScript 學習筆記 Black.Caffeine 09.11.28
這兩天終于寫出了一個個人認為真正意義上的作業(yè)了,不過,最大感受就是。。。IE太垃圾了。。。so,這個作業(yè)實際上也沒有完成的很完美,因為,在IE上還是有兼容性的問題存在,不錯,就此先自我總結一下吧~免得以后又忘了~2009-11-11
javascript彈出窗口 window.open使用方法以及參數(shù)說明分析篇
window.open是網(wǎng)頁中經(jīng)常遇到的彈出窗口代碼,不是網(wǎng)絡中比較反感的那類彈出代碼。用于新頁面?zhèn)髦?,回傳等?/div> 2009-12-12
Javascript全局變量var與不var的區(qū)別深入解析
這篇文章主要介紹了Javascript全局變量var與不var的區(qū)別。需要的朋友可以過來參考下,希望對大家有所幫助2013-12-12
深入理解JavaScript系列(18):面向對象編程之ECMAScript實現(xiàn)
這篇文章主要介紹了深入理解JavaScript系列(18):面向對象編程之ECMAScript實現(xiàn),本文講解了數(shù)據(jù)類型、原始值類型、Object類型、動態(tài)性、內置對象、原生對象及宿主對象等內容,需要的朋友可以參考下2015-03-03
Knockout數(shù)組(observable)使用詳解示例
本文通過示例詳細說明了Knockout數(shù)組(observable)的使用方法,如從observableArray里讀取信息、操作observableArray2013-11-11最新評論

