摘要:受控組件與非受控組件在官網(wǎng)與國(guó)內(nèi)網(wǎng)上的資料都不多,有些人覺(jué)得它可有可不有,也不在意。受控組件與非受控組件是處理表單的入口。認(rèn)為不能多帶帶存在,需要與等控制的屬性或事件一起使用。它們共同構(gòu)成受控組件,受控是受的控制。
受控組件與非受控組件在官網(wǎng)與國(guó)內(nèi)網(wǎng)上的資料都不多,有些人覺(jué)得它可有可不有,也不在意。這恰恰顯示React的威力,滿足不同規(guī)模大小的工程需求。譬如你只是做ListView這樣簡(jiǎn)單的數(shù)據(jù)顯示,將數(shù)據(jù)拍出來(lái),那么for循壞與{}就足夠了,但后臺(tái)系統(tǒng)存在大量報(bào)表,不同的表單聯(lián)動(dòng),缺了受控組件真的不行。
受控組件與非受控組件是React處理表單的入口。從React的思路來(lái)講,作者肯定讓數(shù)據(jù)控制一切,或者簡(jiǎn)單的理解為,頁(yè)面的生成與更新得忠實(shí)地執(zhí)行JSX的指令。
但是表單元素有其特殊之處,用戶可以通過(guò)鍵盤(pán)輸入與鼠標(biāo)選擇,改變界面的顯示。界面的改變也意味著有一些數(shù)據(jù)被改動(dòng),比較明顯的是input的value,textarea的innerHTML,radio/checkbox的checked,不太明顯的是option的selected與selectedIndex,這兩個(gè)是被動(dòng)修改的。
當(dāng)input.value是由組件的state.value拍出來(lái)的,當(dāng)用戶進(jìn)行輸入修改后,然后JSX再次重刷視圖,這時(shí)input.value是采取用戶的新值還是state的新值?基于這個(gè)分歧,React給出一個(gè)折衷的方案,兩者都支持,于是就產(chǎn)生了今天的主題了。
React認(rèn)為value/checked不能多帶帶存在,需要與onInput/onChange/disabed/readOnly等控制value/checked的屬性或事件一起使用。 它們共同構(gòu)成受控組件,受控是受JSX的控制。如果用戶沒(méi)有寫(xiě)這些額外的屬性與事件,那么框架內(nèi)部會(huì)給它添加一些事件,如onClick, onInput, onChange,阻止你進(jìn)行輸入或選擇,讓你無(wú)法修改它的值。在框架內(nèi)部,有一個(gè)頑固的變量,我稱(chēng)之為 persistValue,它一直保持JSX上次賦給它的值,只能讓內(nèi)部事件修改它。
因此我們可以斷言,受控組件是可通過(guò)事件完成的對(duì)value的控制。
在受控組件中,persistValue總能被刷新。
我們?cè)倏捶鞘芸亟M件,既然value/checked已經(jīng)被占用了,React啟用了HTML中另一組被忽略的屬性defaultValue/defaultChecked。一般認(rèn)為它們是與value/checked相通的,即,value不存在的情況下,defaultValue的值就當(dāng)作是value。
上面我們已經(jīng)說(shuō)過(guò),表單元素的顯示情況是由內(nèi)部的 persistValue 控制的,因此defaultXXX也會(huì)同步persistValue,然后再由persistValue同步DOM。但非受控組件的出發(fā)點(diǎn)是忠實(shí)于用戶操作,如果用戶在代碼中
input.value = "xxxx"
以后
就再不生效,一直是xxxx。
它怎么做到這一點(diǎn),怎么辨識(shí)這個(gè)修改是來(lái)自框架內(nèi)部或外部呢?我翻看了一下React的源碼,原來(lái)它有一個(gè)叫valueTracker的東西跟蹤用戶的輸入
var tracker = { getValue: function () { return currentValue; }, setValue: function (value) { currentValue = "" + value; }, stopTracking: function () { detachTracker(node); delete node[valueField]; } }; return tracker; }
這個(gè)東西又是通過(guò)Object.defineProperty打進(jìn)元素的value/checked的內(nèi)部,因此就知曉用戶對(duì)它的取值賦值操作。
但value/checked還是兩個(gè)很核心的屬性,涉及到太多內(nèi)部機(jī)制(比如說(shuō)value與oninput, onchange, 輸入法事件oncompositionstart,
compositionchange, oncompositionend, onpaste, oncut),為了平緩地修改value/checked,
還要用到Object.getOwnPropertyDescriptor。如果我要兼容IE8,沒(méi)有這么高級(jí)的玩藝兒。我采取另一種更安全的方式,
只用Object.defineProperty修改defaultValue/defaultChecked。
首先我為元素添加一個(gè)_uncontrolled的屬性,用來(lái)表示我已經(jīng)劫持過(guò)defaultXXX。 然后描述對(duì)象 (Object.defineProperty的第三個(gè)參數(shù))的set方法里面再添加一個(gè)開(kāi)關(guān),_observing。在框架內(nèi)部更新視圖,此值為false,更新完,它置為true。
這樣就知曉 input.defaultValue = "xxx"時(shí),這是由用戶還是框架修改的。
f (!dom._uncontrolled) { dom._uncontrolled = true; inputMonitor.observe(dom, name); //重寫(xiě)defaultXXX的setter/getter } dom._observing = false;//此時(shí)是框架在修改視圖,因此需要關(guān)閉開(kāi)關(guān) dom[name] = val; dom._observing = true;//打開(kāi)開(kāi)關(guān),來(lái)監(jiān)聽(tīng)用戶的修改行為
inputMonitor的實(shí)現(xiàn)如下
export var inputMonitor = {}; var rcheck = /checked|radio/; var describe = { set: function(value) { var controllProp = rcheck.test(this.type) ? "checked" : "value"; if (this.type === "textarea") { this.innerHTML = value; } if (!this._observing) { if (!this._setValue) { //defaultXXX只會(huì)同步一次_persistValue var parsedValue = (this[controllProp] = value); this._persistValue = Array.isArray(value) ? value : parsedValue; this._setValue = true; } } else { //如果用戶私下改變defaultValue,那么_setValue會(huì)被抺掉 this._setValue = value == null ? false : true; } this._defaultValue = value; }, get: function() { return this._defaultValue; }, configurable: true }; inputMonitor.observe = function(dom, name) { try { if ("_persistValue" in dom) { dom._setValue = true; } Object.defineProperty(dom, name, describe); } catch (e) {} };
又不小心貼了這么燒腦的代碼,這是碼農(nóng)的壞毛病。不過(guò),到這步,大家都明白,無(wú)論是官方react還是anu/qreact都是通過(guò)Object.defineProperty來(lái)控制用戶的輸入的。
于是我們可以理解以下的代碼的行為了
var a = ReactDOM.render(, container); ReactDOM.render(, container); ReactDOM.render(, container); expect(a.defaultValue).toBe("noise"); expect(a.value).toBe("foo"); expect(a.textContent).toBe("noise"); expect(a.innerHTML).toBe("noise");
由于用戶一直沒(méi)有手動(dòng)修改 defaultValue,dom._setValue 一直為false/undefined,因此 _persistValue 一直能修改。
另一個(gè)例子:
var renderTextarea = function(component, container) { if (!container) { container = document.createElement("div"); } const node = ReactDOM.render(component, container); node.defaultValue = node.innerHTML.replace(/^ /, ""); return node; }; const container = document.createElement("div"); //注意這個(gè)方法,用戶在renderTextarea中手動(dòng)改變了defaultValue,_setValue就變成true const node = renderTextarea(, container); expect(node.value).toBe("giraffe"); // _setValue后,gorilla就不能同步到_persistValue,因此還是giraffe renderTextarea(, container); // expect(node.value).toEqual("giraffe"); node.value = "cat"; // 這個(gè)又是什么回事了呢,因此非監(jiān)控屬性是在diffProps中批量處理的,在監(jiān)控屬性,則是在更后的方法中處理 // 檢測(cè)到node.value !== _persistValue,于是重寫(xiě) _persistValue = node.value,于是輸出cat renderTextarea(, container); expect(node.value).toEqual("cat");
當(dāng)然表單元素也分許多種,每種表單元素也有其默認(rèn)行為。
純文本類(lèi):text, textarea, JSX的值,總是往字符串轉(zhuǎn)換
type="number"的控制,值總是為數(shù)字,不填或?yàn)椤啊眲t轉(zhuǎn)換為“0”
radio有聯(lián)動(dòng)效果,同一父節(jié)點(diǎn)下的相同name的radio控制只能選擇一個(gè)。
select的value/defaultValue支持?jǐn)?shù)組,不做轉(zhuǎn)換,但用戶對(duì)底下的option元素做增刪操作,selected會(huì)跟著變動(dòng)。
此外select還有模糊匹配與精確匹配之分。
//精確匹配 var dom = ReactDOM.render( , container ); expect(dom.options[2].selected).toBe(true);//選中第三個(gè)
//模糊匹配 var dom = ReactDOM.render( , container ); expect(dom.options[2].selected).toBe(true);//選中第二個(gè)
凡此種種,React/anu都是做了大量工作,迷你如preact/react-lite之流則可能遇坑。
文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請(qǐng)注明本文地址:http://systransis.cn/yun/92203.html
摘要:首次發(fā)表在個(gè)人博客受控組件或都要綁定一個(gè)事件每當(dāng)表單的狀態(tài)發(fā)生變化都會(huì)被寫(xiě)入組件的中這種組件在中被稱(chēng)為受控組件在受控組件中組件渲染出的狀態(tài)與它的或者向?qū)?yīng)通過(guò)這種方式消除了組件的局部狀態(tài)是的應(yīng)用的整個(gè)狀態(tài)可控官方同樣推薦使用受控表單組件總結(jié) 首次發(fā)表在個(gè)人博客 受控組件 { this.setState({ value: e.target.val...
摘要:假如我們從后臺(tái)拉取一個(gè)數(shù)據(jù)要填入輸入框,那么必須得使用受控組件,因?yàn)榉鞘芸亟M件只能被用戶輸入。不影響正常輸入填充該輸入框的默認(rèn)值,此時(shí)不顯示內(nèi)容。 網(wǎng)頁(yè)中使用的form表單大家肯定都再熟悉不過(guò)了,它主要作用是用來(lái)收集和提交信息。React中的表單組件與我們普通的Html中的表單及其表現(xiàn)形式?jīng)]有什么不同,所以如何使用表單我覺(jué)得再拿出來(lái)說(shuō)可能是畫(huà)蛇添足、毫無(wú)意義。不過(guò)再怎么樣也不能辜負(fù)大家...
摘要:盤(pán)點(diǎn)一下,模式反應(yīng)了典型的控制權(quán)問(wèn)題。異步狀態(tài)管理與控制權(quán)提到控制權(quán)話題,怎能少得了這樣的狀態(tài)管理工具。狀態(tài)管理中的控制主義和極簡(jiǎn)主義了解了異步狀態(tài)中的控制權(quán)問(wèn)題,我們?cè)購(gòu)娜纸嵌冗M(jìn)行分析。 控制權(quán)——這個(gè)概念在編程中至關(guān)重要。比如,輪子封裝層與業(yè)務(wù)消費(fèi)層對(duì)于控制權(quán)的爭(zhēng)奪,就是一個(gè)很有意思的話題。這在 React 世界里也不例外。表面上看,我們當(dāng)然希望輪子掌控的事情越多越好:因?yàn)槌橄髮?..
摘要:盤(pán)點(diǎn)一下,模式反應(yīng)了典型的控制權(quán)問(wèn)題。異步狀態(tài)管理與控制權(quán)提到控制權(quán)話題,怎能少得了這樣的狀態(tài)管理工具。狀態(tài)管理中的控制主義和極簡(jiǎn)主義了解了異步狀態(tài)中的控制權(quán)問(wèn)題,我們?cè)購(gòu)娜纸嵌冗M(jìn)行分析。 控制權(quán)——這個(gè)概念在編程中至關(guān)重要。比如,輪子封裝層與業(yè)務(wù)消費(fèi)層對(duì)于控制權(quán)的爭(zhēng)奪,就是一個(gè)很有意思的話題。這在 React 世界里也不例外。表面上看,我們當(dāng)然希望輪子掌控的事情越多越好:因?yàn)槌橄髮?..
閱讀 2286·2021-11-23 09:51
閱讀 5681·2021-09-22 15:39
閱讀 3355·2021-09-02 15:15
閱讀 3506·2019-08-30 15:54
閱讀 2364·2019-08-30 15:53
閱讀 1404·2019-08-30 14:04
閱讀 2459·2019-08-29 18:33
閱讀 2377·2019-08-29 13:08