React組件設(shè)計(jì)模式之組合組件應(yīng)用實(shí)例分析
本文實(shí)例講述了React組件設(shè)計(jì)模式之組合組件應(yīng)用。分享給大家供大家參考,具體如下:
這種模式本質(zhì)上解決的是組件之間傳值的問(wèn)題。但是它對(duì)于傳值以及一些內(nèi)部操控的邏輯封裝得更嚴(yán)密。
場(chǎng)景:希望減少上下級(jí)組件之間props的傳遞,簡(jiǎn)單來(lái)說(shuō)就是不用傳做顯式地傳值,來(lái)達(dá)到組件之間相互通信的目的
舉例來(lái)說(shuō),某些界面中應(yīng)該有Tabs這樣的組件,由Tab和TabItem組成,點(diǎn)擊每個(gè)TabItem,該TabItem會(huì)高亮,
那么Tab和TabItem自然要進(jìn)行溝通。很自然的寫法是像下面這樣
<TabItem active={true} onClick={this.onClick}>One</TabItem>
<TabItem active={false} onClick={this.onClick}>Two</TabItem>
<TabItem active={false} onClick={this.onClick}>Three</TabItem>
這樣的缺點(diǎn)很明顯:
- 每次使用 TabItem 都要傳遞一堆 props
- 每增加一個(gè)新的 TabItem,都要增加對(duì)應(yīng)的 props
- 如果要增加 TabItem,就要去修改 Tabs 的 JSX 代碼
但是,組件之間的交互我們又不希望通過(guò)props或者context來(lái)實(shí)現(xiàn)。希望用法如下面一樣簡(jiǎn)潔。
<Tabs> <TabItem>第一</TabItem> <TabItem>第二</TabItem> <TabItem>第三</TabItem> </Tabs>
組件之間通過(guò)隱秘的方式進(jìn)行通信,但這里的隱秘實(shí)際上是對(duì)props的操作在一個(gè)地方進(jìn)行管理。
實(shí)現(xiàn)
明白了要實(shí)現(xiàn)的交互,和代碼層面要實(shí)現(xiàn)的效果,就可以開始動(dòng)手了。
TabItem組件有兩個(gè)關(guān)鍵的props: active(表明當(dāng)前是否應(yīng)高亮),onTabClick(自己被點(diǎn)擊時(shí)調(diào)用的回調(diào)函數(shù)),
TabItem由于是每個(gè)Tab頁(yè)面的容器,它只負(fù)責(zé)把props.children渲染出來(lái),所以用函數(shù)式組件即可。
export const TabItem = props => {
const { active, onTabClick, children } = props
const style = {
color: active ? 'red' : 'green',
cursor: 'pointer'
}
return <>
<h1 style={style} onClick={onTabClick}>
{children}
</h1>
</>
}
我們?cè)賮?lái)回顧一下想到達(dá)到的效果:
<Tabs> <TabItem>第一</TabItem> <TabItem>第二</TabItem> <TabItem>第三</TabItem> </Tabs>
使用組件時(shí)要避免傳遞props的缺點(diǎn),那么在哪里傳遞呢?自然是是Tabs組件。但上面并沒(méi)有傳入props啊。
Tabs 雖然可以訪問(wèn)到props里邊的children,但是到手的children已經(jīng)是現(xiàn)成的如果直接改它的話,會(huì)出問(wèn)題。
不可以直接改children的話,我們就把children復(fù)制一份,然后改這個(gè)復(fù)制過(guò)來(lái)的children,再渲染出去,就ok啦!
下面來(lái)看Tabs的實(shí)現(xiàn):
class Tabs extends React.Component {
state={
activeIndex: 0
}
render() {
const { activeIndex } = this.state
const newChildren = React.Children.map(this.props.children, (child, index) => {
if (child.type) {
// 復(fù)制并修改children
return React.cloneElement(child, {
active: activeIndex === index,
onTabClick: () => this.setState({activeIndex: index})
})
} else {
return child
}
})
return <div className="tabs">
{newChildren}
</div>
}
}
這里需要用到React不常用的api:
- React.Children.map
- React.cloneElement
使用React.Children.map來(lái)對(duì)props.children進(jìn)行遍歷。
而React.cloneElement可以復(fù)制某個(gè)元素,第一個(gè)參數(shù)是被復(fù)制的元素,第二個(gè)參數(shù)我們可以把想傳入的props加進(jìn)去,也就是這個(gè)時(shí)機(jī),
我們將active和onTabClick傳入。實(shí)現(xiàn)最終效果。
總結(jié)
這種模式比較好的把復(fù)雜邏輯完全封裝起來(lái)了,抽象程度更好,比較適合開發(fā)組件開發(fā)者。針對(duì)props的擴(kuò)展性也比較好,對(duì)于使用組件的開發(fā)者來(lái)說(shuō),也比較友好。
希望本文所述對(duì)大家react程序設(shè)計(jì)有所幫助。
相關(guān)文章
ahooks封裝cookie?localStorage?sessionStorage方法
這篇文章主要為大家介紹了ahooks封裝cookie?localStorage?sessionStorage的方法示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-07-07
使用Ant Design Anchor組件的一個(gè)坑及解決
這篇文章主要介紹了使用Ant Design Anchor組件的一個(gè)坑及解決,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2024-04-04
React報(bào)錯(cuò)之Object?is?possibly?null的問(wèn)題及解決方法
這篇文章主要介紹了React報(bào)錯(cuò)之Object?is?possibly?null的問(wèn)題,造成 "Object is possibly null"的錯(cuò)誤是因?yàn)閡seRef()鉤子可以傳遞一個(gè)初始值作為參數(shù),而我們傳遞null作為初始值,本文給大家分享詳細(xì)解決方法,需要的朋友可以參考下2022-07-07
React Native如何消除啟動(dòng)時(shí)白屏的方法
本篇文章主要介紹了React Native如何消除啟動(dòng)時(shí)白屏的方法,詳細(xì)的介紹了解決的方法,具有一定的參考價(jià)值,有興趣的可以了解一下2017-08-08
react-dnd實(shí)現(xiàn)任意拖動(dòng)與互換位置
這篇文章主要為大家詳細(xì)介紹了react-dnd實(shí)現(xiàn)任意拖動(dòng)與互換位置,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-08-08

