欧美一级特黄大片做受成人-亚洲成人一区二区电影-激情熟女一区二区三区-日韩专区欧美专区国产专区

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

前言

我們提供的服務(wù)有:成都做網(wǎng)站、成都網(wǎng)站設(shè)計、微信公眾號開發(fā)、網(wǎng)站優(yōu)化、網(wǎng)站認(rèn)證、上街ssl等。為成百上千家企事業(yè)單位解決了網(wǎng)站和推廣的問題。提供周到的售前咨詢和貼心的售后服務(wù),是有科學(xué)管理、有技術(shù)的上街網(wǎng)站制作公司

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

場景

這里應(yīng)用一個最簡單的Tab組件來呈現(xiàn)這個場景。

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項激活。

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

需求的變更與缺陷的暴露

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

這種場景下就需要修改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的時候已經(jīng)沒辦法用 數(shù)組+map 的形式去寫了。

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

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

嘗試掩蓋active判斷重復(fù)的問題

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

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不斷重復(fù)的問題

想要onClick不重復(fù),那么就不能將其寫在TabItem上,而是應(yīng)該寫在Tab上。

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

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

并且你還得期望點擊的switchBtn的內(nèi)的結(jié)構(gòu)不那么復(fù)雜,最好是就只有一個文本。

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

想一想就覺得麻煩。

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

嵌套組件與被嵌套組件的通信: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上,以便子組件內(nèi)部通過props調(diào)用
        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>
  )
 }
}

通過這種方式,我們發(fā)現(xiàn)在使用Tab和TabItem時會變得非常簡單。

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

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

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

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

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

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

foreach就不解釋了,很容易理解是干嘛的。
count就是得到被嵌套組件的數(shù)量。
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的就相當(dāng)克隆一個組件,然后可以傳給它額外的props和children。

總結(jié)

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

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

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

當(dāng)前文章:React中嵌套組件與被嵌套組件的通信過程
網(wǎng)站路徑:http://aaarwkj.com/article32/pccjpc.html

成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供關(guān)鍵詞優(yōu)化、服務(wù)器托管定制網(wǎng)站網(wǎng)站設(shè)計公司、網(wǎng)站改版App設(shè)計

廣告

聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請盡快告知,我們將會在第一時間刪除。文章觀點不代表本網(wǎng)站立場,如需處理請聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時需注明來源: 創(chuàng)新互聯(lián)

綿陽服務(wù)器托管
亚洲精品日韩国产3区| 久久亚洲中文字幕乱码| 日本人妻中文字幕在线一区| 亚洲视频在线的视频在| 亚洲一区二区午夜福利亚洲| 午夜麻豆影网在线观看| 欧美黄片视频在线免费看| 欧美激情日韩精品久久久| 91香蕉伊人综合久久麻豆| 日本区一区二区三视频| 91国产精品视频在线| 亚洲午夜一区二区三区精品影院| 久久国产精品乱码电影| 欧美日韩一级性生活片| 欧美日韩国产成人一区| 最新亚洲av熟女播放| 国产强烈高潮粗暴对白| 日本在线人妻一区二区| 可以看黄片的在线观看| 亚洲综合中文字幕日韩| 亚洲国产一区二区三区三州| 国产伦奸在线播放免费| 日韩大片一区二区三区在线观看 | 精品视频在线观看传媒| 玩弄丰满熟妇留守妇女| 亚洲成人精品一区二区| 亚洲成人日韩在线播放| 日本欧美精品一区二区三区| 饥渴少妇高潮特殊按摩| 福利午夜福利在线观看| 欧洲女人av天堂精品| 欧美大吊视频在线观看| av天堂午夜精品蜜臀| 亚洲欧美国产日韩天堂区| 日韩欧美亚洲国产一区久久精品| 日韩精品一区二区毛片| 亚洲成av人片又粗又长| 亚洲国产欧美日韩一区| 羞涩插射视频网站在线观看| 欧美精品成人在线一区| 欧美中文字幕内射合集|