亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 編程 > JavaScript > 正文

React中嵌套組件與被嵌套組件的通信過程

2019-11-19 13:30:47
字體:
來源:轉載
供稿:網友

前言

在React項目的開發中經常會遇到這樣一個場景:嵌套組件與被嵌套組件的通信。
比如Tab組件啊,或者下拉框組件。

場景

這里應用一個最簡單的Tab組件來呈現這個場景。

import React, { Component, PropTypes } from 'react'class Tab extends Component { static propTypes = {  children: PropTypes.node } render() {  return (   <ul>    {this.props.children}   </ul>  ) }}class TabItem extends Component { static propTypes = {  name: PropTypes.string,  active: PropTypes.bool,  onClick: PropTypes.func } handleClick = () => {  this.props.onClick(this.props.name) } render() {  return (   <li onClick={this.handleClick} className={this.props.active ? 'active' : 'noActive'}>    {this.props.name}   </li>  ) }}export default class Area extends Component { state = {  activeName: '' } handleClick = (name) => {  this.setState({   activeName: name  }) } render() {  return (   <Tab>    {['武漢', '上海', '北京'].map((item) => <TabItem onClick={this.handleClick} active={this.state.activeName === item} name={item} />)}   </Tab>  ) }}

這里有Tab,TabItem和Area三個組件,其中Tab為嵌套組件,TabItem為被嵌套組件,Area為使用它們的組件。
在上述場景中,點擊哪個TabItem項時,就將這個TabItem項激活。

以上方案算是嵌套組件最常用的方案了。

需求的變更與缺陷的暴露

在上述場景下應用上述方案是沒有問題的,但是我們通常用的Tab沒有這么簡單,比如當點擊武漢這個TabItem時,武漢地區的美食也要展示出來。

這種場景下就需要修改TabItem組件為:

class TabItem extends Component { static propTypes = {  name: PropTypes.string,  active: PropTypes.bool,  onClick: PropTypes.func,  children: PropTypes.node } handleClick = () => {  this.props.onClick(this.props.name) } render() {  return (   <li onClick={this.handleClick} className={this.props.active ? 'active' : 'noActive'}>    <span className='switchBtn'>{this.props.name}</span>    <div className={this.props.active ? 'show' : 'hide'}>     {this.props.children}    </div>   </li>  ) }}

然后沿用上述方案,那么就需要改變Area組件為:

export default class Area extends Component { state = {  activeName: '' } handleClick = (name) => {  this.setState({   activeName: name  }) } render() {  return (   <Tab>    <TabItem onClick={this.handleClick} active={this.state.activeName === '武漢'} name={'武漢'} >     武漢的美食,這里有一大堆jsx代碼    </TabItem>    <TabItem onClick={this.handleClick} active={this.state.activeName === '上海'} name={'上海'} >     武漢的美食,這里有一大堆jsx代碼    </TabItem>    <TabItem onClick={this.handleClick} active={this.state.activeName === '北京'} name={'北京'} >     武漢的美食,這里有一大堆jsx代碼    </TabItem>   </Tab>  ) }}

這里的Area使用TabItem的時候已經沒辦法用 數組+map 的形式去寫了。

因為這里有大量的jsx在這里,如果那樣去寫,代碼的可讀性將會非常糟糕。

那么用上面的寫法寫的時候,就會出現一個問題,就是onClick在不斷重復,active的判斷也在不斷重復。

嘗試掩蓋active判斷重復的問題

這個比較容易,修改代碼如下:

class TabItem extends Component { static propTypes = {  name: PropTypes.string,  activeName: PropTypes.string,  onClick: PropTypes.func,  children: PropTypes.node } handleClick = () => {  this.props.onClick(this.props.name) } render() {  return (   <li onClick={this.handleClick} className={this.props.activeName === this.props.name ? 'active' : 'noActive'}>    <span className='switchBtn'>{this.props.name}</span>    <div className={this.props.active ? 'show' : 'hide'}>     {this.props.children}    </div>   </li>  ) }}export default class Area extends Component { state = {  activeName: '' } handleClick = (name) => {  this.setState({   activeName: name  }) } render() {  return (   <Tab>    <TabItem onClick={this.handleClick} activeName={this.state.activeName} name={'武漢'} >     武漢的美食,這里有一大堆jsx代碼    </TabItem>    <TabItem onClick={this.handleClick} activeName={this.state.activeName} name={'上海'} >     武漢的美食,這里有一大堆jsx代碼    </TabItem>    <TabItem onClick={this.handleClick} activeName={this.state.activeName} name={'北京'} >     武漢的美食,這里有一大堆jsx代碼    </TabItem>   </Tab>  ) }}

嘗試掩蓋onClick不斷重復的問題

想要onClick不重復,那么就不能將其寫在TabItem上,而是應該寫在Tab上。

那么這個地方就得用到事件冒泡的機制。

將onClick寫在Tab上,然后根據捕獲的事件消息,獲取target的class是否為switchBtn,然后得到target的text。
再將這個text賦值為activeName。

并且你還得期望點擊的switchBtn的內的結構不那么復雜,最好是就只有一個文本。

如果需求還要給Tab項的切換按鈕每個都加上圖標,那么你還得看這個事件的target是不是這個圖標。那么又需要做更多的處理了。

想一想就覺得麻煩。

一般在這種情況下,腦子里唯一的想法就是,就這樣吧,這個onClick重復就重復吧,沒什么大不了的。
連我自己都懶得寫這部分代碼了。

嵌套組件與被嵌套組件的通信:React.Children與React.cloneElement

實際上要解決上面的問題,只需要一個東西就好了,那就是嵌套組件能傳遞值給被嵌套組件的props,比如onClick。

那么先上一份代碼吧。

class TabItem extends Component { static propTypes = {  name: PropTypes.string,  activeName: PropTypes.string,  onClick: PropTypes.func,  children: PropTypes.node } handleClick = () => {  this.props.onClick(this.props.name) } render() {  return (   <li onClick={this.handleClick} className={this.props.activeName === this.props.name ? 'active' : 'noActive'}>    <span className='switchBtn'>{this.props.name}</span>    <div className={this.props.active ? 'show' : 'hide'}>     {this.props.children}    </div>   </li>  ) }}class Tab extends Component { static propTypes = {  children: PropTypes.node,  onClickItem: PropTypes.func,  activeName: PropTypes.string } render() {  return (   <ul>    {     React.Children.map(this.props.children,(child)=>{      if (child.type === TabItem) {       return React.cloneElement(child, {        // 把父組件的props.name賦值給每個子組件(父組件傳值給子組件)        activeName: this.props.activeName,        // 父組件的方法掛載到props.onClick上,以便子組件內部通過props調用        onClick: this.props.onClickItem       })      } else {       return child      }     })    }   </ul>  ) }}export default class Area extends Component { state = {  activeName: '' } handleClick = (name) => {  this.setState({   activeName: name  }) } render() {  return (   <Tab activeName={this.state.activeName} onClick={this.handleClick} >    <TabItem name={'武漢'} >     武漢的美食,這里有一大堆jsx代碼    </TabItem>    <TabItem name={'上海'} >     武漢的美食,這里有一大堆jsx代碼    </TabItem>    <TabItem name={'北京'} >     武漢的美食,這里有一大堆jsx代碼    </TabItem>   </Tab>  ) }}

通過這種方式,我們發現在使用Tab和TabItem時會變得非常簡單。

那么接下來讓我們介紹一下解決嵌套組件通信這個問題的關鍵:React.Children.map和React.cloneElement。
React.Children

React.Children是專門用來處理this.props.children這個東西的工具。

通常props.children可以是任何變量類型:數組、對象、文本或者其他的一些類型,但是我們這里使用

React.Children.map(this.props.children,(child)=>{ // ***})

無論this.props.children的類型是什么都不會報錯。

這里只是用了React.children的map函數,實際上它還有foreach,count以及only的玩法。

foreach就不解釋了,很容易理解是干嘛的。
count就是得到被嵌套組件的數量。
only就是返回被嵌套的組件,并且只能有一個被嵌套的組件,否則會拋異常。

React.cloneElement

先看下面這段代碼

const child= <Child value={1} />const newChild=React.cloneElement(child,{ name:'額外的props'},'123')

newChild的值為:

<Child value={1} name='額外的props' > 123</Child>

可以很明顯看到,React.cloneElement的就相當克隆一個組件,然后可以傳給它額外的props和children。

總結

對于簡單的嵌套組件用最開始的方法其實已經夠了。

但是對于復雜的嵌套組件為了更好更方便的使用,往往需要與被嵌套的組件進行通信。

而我們可以使用React.Children和React.cloneElement來解決這個問題。

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
亚洲一区美女视频在线观看免费| 91久久综合亚洲鲁鲁五月天| 中国人与牲禽动交精品| 久久天天躁狠狠躁夜夜av| 国产精品一二三视频| 68精品国产免费久久久久久婷婷| 国产精品一区二区3区| 91精品久久久久久久久久久久久| 成人国产精品一区| 91久久精品久久国产性色也91| 日韩资源在线观看| 亚洲免费一在线| 日韩视频―中文字幕| 91精品国产综合久久久久久蜜臀| 国产精品美女在线观看| 国产精品丝袜久久久久久高清| 91亚洲国产成人精品性色| 欧美华人在线视频| 欧美日韩亚洲天堂| 日本不卡视频在线播放| 久久久国产视频91| 国产精品扒开腿做爽爽爽视频| 久久777国产线看观看精品| 欧美在线日韩在线| 91在线中文字幕| 97免费中文视频在线观看| 亚洲欧美国产日韩中文字幕| 亚洲精品电影在线| 国产成人免费av| 91精品成人久久| 亚洲精品国偷自产在线99热| 粉嫩av一区二区三区免费野| 国产一区二区三区久久精品| 九九热在线精品视频| 成人免费淫片视频软件| 国产精品揄拍500视频| 欧美性色视频在线| 亚洲二区中文字幕| 狠狠做深爱婷婷久久综合一区| 国产精品久久久久久久av电影| 久久色免费在线视频| 精品国产91久久久| 国产亚洲a∨片在线观看| 欧美洲成人男女午夜视频| 久久全球大尺度高清视频| 青青a在线精品免费观看| 一本色道久久88综合亚洲精品ⅰ| 久久久精品亚洲| 亚洲欧美综合另类中字| 国产精品久久久久久网站| 亚洲电影成人av99爱色| 久久久亚洲影院| 亚洲va电影大全| 欧美成人精品在线播放| 91免费看片在线| 欧美性69xxxx肥| 日韩av高清不卡| 最近中文字幕mv在线一区二区三区四区| 26uuu另类亚洲欧美日本一| 8x海外华人永久免费日韩内陆视频| 国产精品久久二区| 亚洲人成毛片在线播放| 岛国精品视频在线播放| 中文字幕不卡在线视频极品| 精品爽片免费看久久| 91高清视频在线免费观看| 中文字幕精品一区久久久久| 国产大片精品免费永久看nba| 美女久久久久久久久久久| 97国产suv精品一区二区62| 欧美成人精品不卡视频在线观看| 欧美成人全部免费| 日韩在线视频免费观看| 日韩福利伦理影院免费| 热99精品只有里视频精品| 中文字幕在线看视频国产欧美在线看完整| 国产网站欧美日韩免费精品在线观看| 国产女人精品视频| 91久久久久久久久| 国产精品999999| 日韩欧美在线字幕| 久久久91精品国产一区不卡| 国产综合久久久久久| 精品香蕉一区二区三区| 欧美高清理论片| 亚洲欧洲第一视频| 成人黄色在线播放| 久久久精品中文字幕| 中国日韩欧美久久久久久久久| 欧美激情视频免费观看| 欧美专区国产专区| 91国产视频在线播放| 亚洲无限av看| 欧美大尺度激情区在线播放| 欧美超级免费视 在线| 中文字幕综合一区| 亚洲综合精品一区二区| 国产成人精品最新| 中文字幕精品www乱入免费视频| 亚洲视频免费一区| 91久久久久久久一区二区| 日韩激情av在线免费观看| 欧美中文字幕在线播放| 亚洲色图第一页| 性欧美激情精品| 国产日本欧美一区二区三区在线| 成年无码av片在线| 久久夜精品va视频免费观看| 97国产一区二区精品久久呦| 欧美视频裸体精品| 欧美视频一区二区三区…| 欧美日韩国产一区中文午夜| 57pao国产成人免费| 国产视频精品自拍| 欧美黑人性生活视频| 色综合老司机第九色激情| 久久av红桃一区二区小说| 久久精品国产亚洲| 亚洲精品动漫久久久久| 中文字幕在线看视频国产欧美| 日韩高清中文字幕| 欧美成人国产va精品日本一级| 伊人伊成久久人综合网站| 亚洲第一区中文字幕| 热re91久久精品国99热蜜臀| 亚洲免费人成在线视频观看| 国产成人精品在线| 三级精品视频久久久久| 久久福利网址导航| 精品亚洲一区二区三区在线播放| 国产成人精品免费视频| 久久久成人av| 国产精品日韩欧美| 国产精品一区二区三区久久久| 亚洲综合日韩在线| 国产精品海角社区在线观看| 日韩成人在线观看| 国产成+人+综合+亚洲欧洲| 国产亚洲xxx| 91免费看片在线| 国内精品小视频在线观看| 久久久免费精品| 国产精品美乳在线观看| 在线看福利67194| 欧美性理论片在线观看片免费| 欧美极品少妇xxxxⅹ喷水| 91在线免费看网站| 成人在线一区二区| 日韩欧美精品网址| 久久97久久97精品免视看| 久久99精品国产99久久6尤物| 国产精品国产三级国产专播精品人| 久久久成人的性感天堂| 久久免费成人精品视频| 日韩国产精品亚洲а∨天堂免| 欧美日韩亚洲一区二区三区| 亚洲免费福利视频| 亚洲人成五月天| 97久久超碰福利国产精品…| 欧美激情一区二区三区久久久| 国产小视频91| 日韩av电影在线免费播放| 日韩视频免费看| 中文国产成人精品|