react Input組件Compositionstart和Compositionend事件
前言
Input組件大家都用過(guò),是吧,但是你有沒(méi)有想過(guò)這樣一個(gè)場(chǎng)景,如下圖,我正在搜索數(shù)據(jù)
你組件上注冊(cè)了onChange事件,然后邊輸入,底下會(huì)顯示你搜索相關(guān)的內(nèi)容,

但是有一個(gè)問(wèn)題就是,輸入中文的時(shí)候,你比如打三國(guó)的三字,要先輸入san然后才出現(xiàn)三

可問(wèn)題來(lái)了,onChange事件監(jiān)聽(tīng)的是san,已經(jīng)開(kāi)始搜索了,其實(shí)我們根本不想這樣,我們想的是中文的話,等中文顯示在輸入框,也就是輸入完“三”這個(gè)字的時(shí)候,才搜索。這個(gè)咋辦
這就需要我們今天的主角Compositionstart和Compositionend事件出場(chǎng)了。
Compositionstart和Compositionend事件是啥
compositionstart 事件在用戶開(kāi)始進(jìn)行非直接輸入的時(shí)候觸發(fā),而在非直接輸入結(jié)束,也即用戶點(diǎn)選候選詞或者點(diǎn)擊「選定」按鈕之后,比如按回車鍵,會(huì)觸發(fā) compositionend 事件。
舉個(gè)例子,還是上面輸入三這個(gè)字的過(guò)程,當(dāng)你輸入s的時(shí)候,已經(jīng)打開(kāi)了中文輸入法,此時(shí)compositionstart事件觸發(fā)了,當(dāng)你輸入完三并且確認(rèn)的時(shí)候,compositionend事件觸發(fā)。
好了,所以如果你用的組件庫(kù),Input組件沒(méi)有優(yōu)化這一項(xiàng)的話,也就是onChange事件在中文輸入法下,應(yīng)該返回的是中文輸入完成,也就是compositionend事件完成的字符,而不是輸入過(guò)程的字符,那么就需要組件封裝一下這個(gè)邏輯了,如果封裝,我們看下源碼。
本來(lái)是打算解析完整的Input組件的,后來(lái)想想其實(shí)這個(gè)組件太簡(jiǎn)單了,就只講一下其中的難點(diǎn)吧。
源碼探析解決方案
首先Input組件如下
<input
value={innerValue}
onCompositionStart={handleCompositionStart}
onCompositionEnd={handleCompositionEnd}
onChange={handleChange}
/>
然后我們看下value屬性的formatDisplayValue是什么
// 用來(lái)記錄此時(shí)是否Compositionstart事件觸發(fā)了,如果觸發(fā)就置為true
// Compositionend結(jié)束就置為false
const composingRef = useRef(false);
// 暴露給
const [value, onChange] = useState<string>('');
const [composingValue, setComposingValue] = useState<string>('');
// 如果啟動(dòng)了中文輸入法,那么innerValue就是composingValue
// composingValue就是中文輸入的時(shí)候比如“三國(guó)”,你輸入從“s”到“sanguo”,此時(shí)innerValue都是composingValue
// 除了中文輸入法外,innerValue都是value
const innerValue = composingRef.current ? composingValue : value ?? '';
上面可以看到innerValue是最終渲染給input框的value,用戶一般通過(guò)onChange事件獲取值,所以 我們?cè)谥形妮斎氲臅r(shí)候,只要不觸發(fā)onChange事件是不是就好了!
關(guān)鍵??!最重要的知識(shí)點(diǎn)就是Compositionstart事件觸發(fā)了,代表正在輸入中文,那么onChange事件就不要觸發(fā),所以我們接著把事件的代碼補(bǔ)上
// 開(kāi)始輸入中文的時(shí)候把 composingRef.current 置為true
function handleCompositionStart(e: React.CompositionEvent<HTMLInputElement>) {
composingRef.current = true;
const {
currentTarget: { value },
} = e;
}
// 中文輸入完畢,把composingRef.current置為false,并把此時(shí)輸入完的值給handleChange(handleChange會(huì)觸發(fā)onChange)
function handleCompositionEnd(e: React.CompositionEvent<HTMLInputElement>) {
if (composingRef.current) {
composingRef.current = false;
handleChange(e);
}
}
function handleChange(e: React.ChangeEvent<HTMLInputElement> | React.CompositionEvent<HTMLInputElement>) {
let { value: newStr } = e.currentTarget;
// 當(dāng)中文輸入的時(shí)候,不觸發(fā)onChange事件,觸發(fā)setComposingValue只是為了讓輸入框同步你輸入的text
if (composingRef.current) {
setComposingValue(newStr);
} else {
// 完成中文輸入時(shí)同步一次 composingValue
setComposingValue(newStr);
// 中文輸入完畢,此時(shí)觸發(fā)onChange
onChange(newStr, { e });
}
}
好了,本文結(jié)束,其實(shí)里面還有兩個(gè)細(xì)節(jié),后面再寫一篇文章說(shuō),就是我們input框傳參,可以有defaultValue這個(gè)value是外部傳入的,還有props傳入的value,還有組件默認(rèn)的defalut value,這3個(gè)值怎么寫一個(gè)hooks去處理,還有就是你們知道input框有種類型是顯示password嗎,這種顯示密碼的input框,輸入文字后顯示的是...如下:

點(diǎn)擊眼睛圖標(biāo)就能看到輸入的文字了,這個(gè)是咋辦到的。
以上就是react Input組件Compositionstart和Compositionend事件的詳細(xì)內(nèi)容,更多關(guān)于react Input組件事件的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
react中將html字符串渲染到頁(yè)面的實(shí)現(xiàn)方式
這篇文章主要介紹了react中將html字符串渲染到頁(yè)面的實(shí)現(xiàn)方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-12-12
react-router 路由切換動(dòng)畫的實(shí)現(xiàn)示例
這篇文章主要介紹了react-router 路由切換動(dòng)畫的實(shí)現(xiàn)示例,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2018-12-12
Ant?Design?組件庫(kù)之步驟條實(shí)現(xiàn)
這篇文章主要為大家介紹了Ant?Design組件庫(kù)之步驟條實(shí)現(xiàn),有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-08-08
React的createElement和render手寫實(shí)現(xiàn)示例
這篇文章主要為大家介紹了React的createElement和render手寫實(shí)現(xiàn)示例,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-08-08
基于React-Dropzone開(kāi)發(fā)上傳組件功能(實(shí)例演示)
這篇文章主要介紹了基于React-Dropzone開(kāi)發(fā)上傳組件,主要講述的是在React-Flask框架上開(kāi)發(fā)上傳組件的技巧,需要的朋友可以參考下2021-08-08
React鉤子函數(shù)之useDeferredValue的基本使用示例詳解
useDeferredValue是React 18中非常有用的一個(gè)鉤子函數(shù),它可以幫助我們優(yōu)化渲染性能,并讓UI更加流暢,如果你還沒(méi)有嘗試過(guò)它,不妨在你的下一個(gè)React項(xiàng)目中試一試,這篇文章主要介紹了React鉤子函數(shù)之useDeferredValue的基本使用,需要的朋友可以參考下2023-08-08
Redux thunk中間件及執(zhí)行原理詳細(xì)分析
redux的核心概念其實(shí)很簡(jiǎn)單:將需要修改的state都存入到store里,發(fā)起一個(gè)action用來(lái)描述發(fā)生了什么,用reducers描述action如何改變state tree,這篇文章主要介紹了Redux thunk中間件及執(zhí)行原理分析2022-09-09

