真实的国产乱ⅩXXX66竹夫人,五月香六月婷婷激情综合,亚洲日本VA一区二区三区,亚洲精品一区二区三区麻豆

成都創(chuàng)新互聯(lián)網(wǎng)站制作重慶分公司

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

前言

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

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

場(chǎng)景

這里應(yīng)用一個(gè)最簡(jiǎn)單的Tab組件來(lái)呈現(xiàn)這個(gè)場(chǎng)景。

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

    這里有Tab,TabItem和Area三個(gè)組件,其中Tab為嵌套組件,TabItem為被嵌套組件,Area為使用它們的組件。
    在上述場(chǎng)景中,點(diǎn)擊哪個(gè)TabItem項(xiàng)時(shí),就將這個(gè)TabItem項(xiàng)激活。

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

    需求的變更與缺陷的暴露

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

    這種場(chǎng)景下就需要修改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 (
       
  • {this.props.name}
    {this.props.children}
  • ) } }

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

    export default class Area extends Component {
     state = {
      activeName: ''
     }
    
     handleClick = (name) => {
      this.setState({
       activeName: name
      })
     }
    
     render() {
      return (
       
        
         武漢的美食,這里有一大堆jsx代碼
        
        
         武漢的美食,這里有一大堆jsx代碼
        
        
         武漢的美食,這里有一大堆jsx代碼
        
       
      )
     }
    }

    這里的Area使用TabItem的時(shí)候已經(jīng)沒辦法用 數(shù)組+map 的形式去寫了。

    因?yàn)檫@里有大量的jsx在這里,如果那樣去寫,代碼的可讀性將會(huì)非常糟糕。

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

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

    這個(gè)比較容易,修改代碼如下:

    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 (
       
  • {this.props.name}
    {this.props.children}
  • ) } } export default class Area extends Component { state = { activeName: '' } handleClick = (name) => { this.setState({ activeName: name }) } render() { return ( 武漢的美食,這里有一大堆jsx代碼 武漢的美食,這里有一大堆jsx代碼 武漢的美食,這里有一大堆jsx代碼 ) } }

    嘗試掩蓋onClick不斷重復(fù)的問題

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

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

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

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

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

    想一想就覺得麻煩。

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

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

    實(shí)際上要解決上面的問題,只需要一個(gè)東西就好了,那就是嵌套組件能傳遞值給被嵌套組件的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 (
       
  • {this.props.name}
    {this.props.children}
  • ) } } class Tab extends Component { static propTypes = { children: PropTypes.node, onClickItem: PropTypes.func, activeName: PropTypes.string } render() { return (
      { React.Children.map(this.props.children,(child)=>{ if (child.type === TabItem) { return React.cloneElement(child, { // 把父組件的props.name賦值給每個(gè)子組件(父組件傳值給子組件) activeName: this.props.activeName, // 父組件的方法掛載到props.onClick上,以便子組件內(nèi)部通過props調(diào)用 onClick: this.props.onClickItem }) } else { return child } }) }
    ) } } export default class Area extends Component { state = { activeName: '' } handleClick = (name) => { this.setState({ activeName: name }) } render() { return ( 武漢的美食,這里有一大堆jsx代碼 武漢的美食,這里有一大堆jsx代碼 武漢的美食,這里有一大堆jsx代碼 ) } }

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

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

    React.Children是專門用來(lái)處理this.props.children這個(gè)東西的工具。

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

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

    無(wú)論this.props.children的類型是什么都不會(huì)報(bào)錯(cuò)。

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

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

    React.cloneElement

    先看下面這段代碼

    const child= 
    const newChild=React.cloneElement(child,{
     name:'額外的props'
    },'123')

    newChild的值為:

    
     123
    

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

    總結(jié)

    對(duì)于簡(jiǎn)單的嵌套組件用最開始的方法其實(shí)已經(jīng)夠了。

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

    而我們可以使用React.Children和React.cloneElement來(lái)解決這個(gè)問題。


    本文題目:React中嵌套組件與被嵌套組件的通信過程
    轉(zhuǎn)載注明:http://weahome.cn/article/pccjpc.html

    其他資訊

    在線咨詢

    微信咨詢

    電話咨詢

    028-86922220(工作日)

    18980820575(7×24)

    提交需求

    返回頂部