I'm working with ReactJS with ES6, but I have some problems to communicate child > parent through props. Example of my approach:
class SearchBar extends React.Component {
handler(e){
this.props.filterUser(e.target.value);
}
render () {
return <div>
<input type='text' className='from-control search-bar' placeholder='Search' onChange={this.handler} />
</div>
}
}
export default class User extends React.Component {
constructor(props) {
super(props);
this.state = {name: '', age: '', filter: ''};
}
filterUser(filterValue){
this.setState({
filter: filterValue
});
}
render() {
return <div>
<SearchBar filterUser={this.filterUser} />
<span>Value: {this.state.filter}</span>
</div>
}
}
This returns Uncaught TypeError: this.props.filterUser is not a function
.
Any idea? Binding maybe?
[EDIT] Solution (Thanks @knowbody & @Felipe Skinner):
I was missing binding in my constructor. Binding in the SearchBar constructor works perfectly.
Using React.createClass()
(ES5), it automatically does bindings to this
for your functions. In ES6 you need bind this
manually. More info https://facebook.github.io/react/docs/reusable-components.html#es6-classes
For passing the data from the child component to the parent component, we have to create a callback function in the parent component and then pass the callback function to the child component as a prop. This callback function will retrieve the data from the child component.
The React. js "Uncaught TypeError: X is not a function" occurs when we try to call a value that is not a function as a function, e.g. calling the props object instead of a function. To solve the error, console. log the value you are calling and make sure it is a function.
You are missing binding in your constructor, also you don't need to pass props
if you are not using them in the constructor. Also you need to import { PropTypes } from 'react'
class SearchBar extends React.Component {
constructor() {
super();
this.handler = this.handler.bind(this);
}
handler(e){
this.props.filterUser(e.target.value);
}
render () {
return (
<div>
<input type='text' className='from-control search-bar' placeholder='Search' onChange={this.handler} />
</div>
);
}
}
export default class User extends React.Component {
constructor() {
super();
this.filterUser = this.filterUser.bind(this);
this.state = { name: '', age: '', filter: '' };
}
filterUser(filterValue){
this.setState({
filter: filterValue
});
}
render() {
return (
<div>
<SearchBar filterUser={this.filterUser} />
<span>Value: {this.state.filter}</span>
</div>
);
}
}
When ur using React.createClass(), it automatically does bindings to this for your functions.
Since you're using the ES6 class syntax, you need to do those bindings by yourself. Here's two options:
render() {
return <div>
<SearchBar filterUser={this.filterUser.bind(this)} />
<span>Value: {this.state.filter}</span>
</div>
}
Or you could bind it on your constructor like this:
constructor(props) {
super(props);
this.state = {name: '', age: '', filter: ''};
this.filterUser = this.filterUser.bind(this);
}
You can read about this on the docs: https://facebook.github.io/react/docs/reusable-components.html#es6-classes
Note that those two options are mutually exclusive.
In my case, I was importing the component the wrong way. I have the components "HomeAdmin" and "Register".
I had this in HomeAdmin.js:
import { Register } from "/path/to/register"
Changed to this and worked:
import Register from "/path/to/register"
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With