Как передать данные от дочернего компонента его родителю в ReactJS?

213

Я пытаюсь отправить данные от дочернего компонента к его родителю следующим образом:

const ParentComponent = React.createClass({
    getInitialState() {
        return {
            language: '',
        };
    },
    handleLanguageCode: function(langValue) {
        this.setState({language: langValue});
    },

    render() {
         return (
                <div className="col-sm-9" >
                    <SelectLanguage onSelectLanguage={this.handleLanguage}/> 
                </div>
        );
});

а вот дочерний компонент:

export const SelectLanguage = React.createClass({
    getInitialState: function(){
        return{
            selectedCode: '',
            selectedLanguage: '',
        };
    },

    handleLangChange: function (e) {
        var lang = this.state.selectedLanguage;
        var code = this.state.selectedCode;
        this.props.onSelectLanguage({selectedLanguage: lang});   
        this.props.onSelectLanguage({selectedCode: code});           
    },

    render() {
        var json = require("json!../languages.json");
        var jsonArray = json.languages;
        return (
            <div >
                <DropdownList ref='dropdown'
                    data={jsonArray} 
                    value={this.state.selectedLanguage}
                    caseSensitive={false} 
                    minLength={3}
                    filter='contains'
                    onChange={this.handleLangChange} />
            </div>            
        );
    }
});

Что мне нужно, это получить выбранное значение пользователем в родительском компоненте. Я получаю эту ошибку:

Uncaught TypeError: this.props.onSelectLanguage is not a function

Может кто-нибудь помочь мне найти проблему?

PS Дочерний компонент создает раскрывающийся список из файла json, и мне нужен раскрывающийся список, чтобы показать оба элемента массива json рядом друг с другом (например, «aaa, english» в качестве первого выбора!)

{  
   "languages":[  
      [  
         "aaa",
         "english"
      ],
      [  
         "aab",
         "swedish"
      ],
}
Birish
источник
3
<SelectLanguage onSelectLanguage={this.handleLanguage*Code*}/> опечатка.
Юрий Тарабанко
@YuryTarabanko Спасибо, но все равно получаю ту же ошибку
Birish
@DavinTryon Как мне его добавить? Я пытался так: handleLanguageCode: function(langValue) { this.setState({ language: langValue }).bind(this); },Но он возвращает ошибку:ncaught TypeError: Cannot read property 'bind' of undefined
Бирюш
2
@DavinTryon автоматически createClassсвязывает не реагирующие методы.
Юрий Тарабанко
@OP Не могли бы вы создать скрипку, демонстрирующую проблему?
Юрий Тарабанко

Ответы:

259

Это должно работать. Отправляя реквизит обратно, вы отправляете его как объект, а скорее отправляете его как значение или, в качестве альтернативы, используете его как объект в родительском компоненте. Во-вторых, вам нужно отформатировать ваш объект json, чтобы он содержал пары «имя-значение», а также использование valueFieldи textFieldатрибутDropdownList

Короткий ответ

родитель:

<div className="col-sm-9">
     <SelectLanguage onSelectLanguage={this.handleLanguage} /> 
</div>

Ребенок:

handleLangChange = () => {
    var lang = this.dropdown.value;
    this.props.onSelectLanguage(lang);            
}

Детальнее:

РЕДАКТИРОВАТЬ:

Учитывая, что React.createClass устарел с версии v16.0 и выше, лучше пойти дальше и создать компонент React путем расширения React.Component. Передача данных от дочернего к родительскому компоненту с этим синтаксисом будет выглядеть так

родитель

class ParentComponent extends React.Component {

    state = { language: '' }

    handleLanguage = (langValue) => {
        this.setState({language: langValue});
    }

    render() {
         return (
                <div className="col-sm-9">
                    <SelectLanguage onSelectLanguage={this.handleLanguage} /> 
                </div>
        )
     }
}

ребенок

var json = require("json!../languages.json");
var jsonArray = json.languages;

export class SelectLanguage extends React.Component {
    state = {
            selectedCode: '',
            selectedLanguage: jsonArray[0],
        }

    handleLangChange = () => {
        var lang = this.dropdown.value;
        this.props.onSelectLanguage(lang);            
    }

    render() {
        return (
            <div>
                <DropdownList ref={(ref) => this.dropdown = ref}
                    data={jsonArray} 
                    valueField='lang' textField='lang'
                    caseSensitive={false} 
                    minLength={3}
                    filter='contains'
                    onChange={this.handleLangChange} />
            </div>            
        );
    }
}

Используя createClassсинтаксис, который ОП использовал в своем ответе Parent

const ParentComponent = React.createClass({
    getInitialState() {
        return {
            language: '',
        };
    },

    handleLanguage: function(langValue) {
        this.setState({language: langValue});
    },

    render() {
         return (
                <div className="col-sm-9">
                    <SelectLanguage onSelectLanguage={this.handleLanguage} /> 
                </div>
        );
});

ребенок

var json = require("json!../languages.json");
var jsonArray = json.languages;

export const SelectLanguage = React.createClass({
    getInitialState: function() {
        return {
            selectedCode: '',
            selectedLanguage: jsonArray[0],
        };
    },

    handleLangChange: function () {
        var lang = this.refs.dropdown.value;
        this.props.onSelectLanguage(lang);            
    },

    render() {

        return (
            <div>
                <DropdownList ref='dropdown'
                    data={jsonArray} 
                    valueField='lang' textField='lang'
                    caseSensitive={false} 
                    minLength={3}
                    filter='contains'
                    onChange={this.handleLangChange} />
            </div>            
        );
    }
});

JSON:

{ 
"languages":[ 

    { 
    "code": "aaa", 
    "lang": "english" 
    }, 
    { 
    "code": "aab", 
    "lang": "Swedish" 
    }, 
  ] 
}
Шубхам Хатри
источник
Спасибо Shubham, ваш ответ устранил проблему, но не показывает выбранный элемент в выпадающем списке. После этого пусто: /
Бирюш
@Sarah При создании dropDownList вы присвоили его значение this.state.selectedLanguage, которое не инициализируется никаким значением. Это может быть проблемой. Какой компонент DropdownList вы используете. Если я знаю, может быть, я могу работать над этим, чтобы решить проблему.
Шубхам Хатри
Ты прав. this.state.selectedLanguage, Кажется, всегда нуль: / Я использую DropDownList изreact-widgets
Birish
@Sarah Я сделал небольшое изменение в коде, включая привязку значения к функции onChange и настройку начального значения состояния. Попробуй это и скажи мне, работает ли он на тебя
Шубхам Хатри
Возвращает ошибку: Uncaught ReferenceError: value is not defined. Я внес это изменение: onChange={this.handleLangChange.bind(this, this.value)}ошибка не возвращается, но все равно не отображается выбранное значение :(
Birish
108

Чтобы передать данные из дочернего компонента в родительский компонент

В родительском компоненте:

getData(val){
    // do not forget to bind getData in constructor
    console.log(val);
}
render(){
 return(<Child sendData={this.getData}/>);
}

В дочернем компоненте:

demoMethod(){
   this.props.sendData(value);
 }
Маниша Тан
источник
15
Не забудьте this.getData = this.getData.bind(this);в конструкторе, так как вы можете получить this.setState не является функцией, если вы хотите манипулировать состоянием в getData.
Миро Дж.
12

Я нашел подход, как получить данные от дочернего компонента у родителей, когда мне это нужно.

родитель:

class ParentComponent extends Component{
  onSubmit(data) {
    let mapPoint = this.getMapPoint();
  }

  render(){
    return (
      <form onSubmit={this.onSubmit.bind(this)}>
        <ChildComponent getCurrentPoint={getMapPoint => {this.getMapPoint = getMapPoint}} />
        <input type="submit" value="Submit" />
      </form>
    )
  }
}

Ребенок:

class ChildComponent extends Component{
  constructor(props){
    super(props);

    if (props.getCurrentPoint){
      props.getCurrentPoint(this.getMapPoint.bind(this));
    }
  }

  getMapPoint(){
    return this.Point;
  }
}

В этом примере показано, как передать функцию из дочернего компонента в родительский и использовать эту функцию для получения данных от дочернего компонента.

MrDuDuDu
источник
10

Учитывая, что компоненты React Function и использование хуков становятся все более популярными в наши дни, я приведу простой пример передачи данных от дочернего к родительскому компоненту.

в компоненте родительской функции мы будем иметь:

import React, { useState, useEffect } from "react";

затем

const [childData, setChildData] = useState("");

и передача setChildData (которая выполняет работу, аналогичную this.setState в компонентах класса) дочернему элементу

return( <ChildComponent passChildData={setChildData} /> )

в дочернем компоненте сначала мы получаем реквизит

function ChildComponent(props){ return (...) }

тогда вы можете передавать данные в любом случае, как с помощью функции обработчика

const functionHandler = (data) => {

props.passChildData(data);

}
Шаян Могадам
источник
3

Метод React.createClass устарел в новой версии React, вы можете сделать это очень просто следующим образом: создать один функциональный компонент и другой компонент класса для поддержания состояния:

родитель:

const ParentComp = () => {
  
  getLanguage = (language) => {
    console.log('Language in Parent Component: ', language);
  }
  
  <ChildComp onGetLanguage={getLanguage}
};

Ребенок:

class ChildComp extends React.Component {
    state = {
      selectedLanguage: ''
    }
    
    handleLangChange = e => {
        const language = e.target.value;
        thi.setState({
          selectedLanguage = language;
        });
        this.props.onGetLanguage({language}); 
    }

    render() {
        const json = require("json!../languages.json");
        const jsonArray = json.languages;
        const selectedLanguage = this.state;
        return (
            <div >
                <DropdownList ref='dropdown'
                    data={jsonArray} 
                    value={tselectedLanguage}
                    caseSensitive={false} 
                    minLength={3}
                    filter='contains'
                    onChange={this.handleLangChange} />
            </div>            
        );
    }
};

Саня Хушбахт Джамиль
источник
3
Просто примечание ... вы должны ссылаться на конкретную версию React вместо «новой версии», чтобы подтвердить этот ответ в будущем.
steve-o
3

от дочернего компонента к родительскому компоненту, как показано ниже

родительский компонент

class Parent extends React.Component {
   state = { message: "parent message" }
   callbackFunction = (childData) => {
       this.setState({message: childData})
   },
   render() {
        return (
            <div>
                 <Child parentCallback = {this.callbackFunction}/>
                 <p> {this.state.message} </p>
            </div>
        );
   }
}

дочерний компонент

class Child extends React.Component{
    sendBackData = () => {
         this.props.parentCallback("child message");
    },
    render() { 
       <button onClick={sendBackData}>click me to send back</button>
    }
};

Надеюсь эта работа

Маной Рана
источник
2

в React v16.8+компоненте функции вы можете использовать useState()для создания состояния функции, которое позволит вам обновить родительское состояние, затем передать его дочернему элементу в качестве атрибута props, а затем внутри дочернего компонента вы можете вызвать функцию родительского состояния, ниже приведен рабочий фрагмент :

const { useState , useEffect } = React;

function Timer({ setParentCounter }) {
  const [counter, setCounter] = React.useState(0);

  useEffect(() => {
    let countersystem;
    countersystem = setTimeout(() => setCounter(counter + 1), 1000);

    return () => {
      clearTimeout(countersystem);
    };
  }, [counter]);

  return (
    <div className="App">
      <button
        onClick={() => {
          setParentCounter(counter);
        }}
      >
        Set parent counter value
      </button>
      <hr />
      <div>Child Counter: {counter}</div>
    </div>
  );
}

function App() {
  const [parentCounter, setParentCounter] = useState(0);

  return (
    <div className="App">
      Parent Counter: {parentCounter}
      <hr />
      <Timer setParentCounter={setParentCounter} />
    </div>
  );
}

ReactDOM.render(<App />, document.getElementById('react-root'));
<script src="https://cdnjs.cloudflare.com/ajax/libs/react/16.8.4/umd/react.production.min.js"></script>
<script src="https://cdnjs.cloudflare.com/ajax/libs/react-dom/16.8.4/umd/react-dom.production.min.js"></script>
<div id="react-root"></div>

ROOT
источник
1

Вы даже можете избежать функции у родителя, обновляющего состояние напрямую

В родительском компоненте:

render(){
 return(<Child sendData={ v => this.setState({item: v}) } />);
}

В дочернем компоненте:

demoMethod(){
   this.props.sendData(value);
}
Banzy
источник
0

Идея состоит в том, чтобы отправить обратный вызов ребенку, который будет вызван, чтобы вернуть данные

Полный и минимальный пример с использованием функций:

Приложение создаст дочерний элемент, который вычислит случайное число и отправит его обратно родителю, что приведет console.logк

const Child = ({ handleRandom }) => {
  handleRandom(Math.random())

  return <span>child</span>
}
const App = () => <Child handleRandom={(num) => console.log(num)}/>
Фабьен Са
источник
0

Родительский компонент: -TimeModal

  handleTimeValue = (timeValue) => {
      this.setState({pouringDiff: timeValue});
  }

  <TimeSelection 
        prePourPreHours={prePourPreHours}
        setPourTime={this.setPourTime}
        isPrePour={isPrePour}
        isResident={isResident}
        isMilitaryFormatTime={isMilitaryFormatTime}
        communityDateTime={moment(communityDT).format("MM/DD/YYYY hh:mm A")}
        onSelectPouringTimeDiff={this.handleTimeValue}
     />

Примечание : - onSelectPouringTimeDiff = {this.handleTimeValue}

В дочернем компоненте вызов реквизита при необходимости

 componentDidMount():void{
      // Todo use this as per your scenrio
       this.props.onSelectPouringTimeDiff(pouringDiff);  
  }
Кешав Гера
источник