I have a simple react component with the form which I believe to have one controlled input:
import React from 'react';
export default class MyForm extends React.Component {
constructor(props) {
super(props);
this.state = {}
}
render() {
return (
<form className="add-support-staff-form">
<input name="name" type="text" value={this.state.name} onChange={this.onFieldChange('name').bind(this)}/>
</form>
)
}
onFieldChange(fieldName) {
return function (event) {
this.setState({[fieldName]: event.target.value});
}
}
}
export default MyForm;
When I run my application I get the following warning:
Warning: MyForm is changing an uncontrolled input of type text to be controlled. Input elements should not switch from uncontrolled to controlled (or vice versa). Decide between using a controlled or uncontrolled input element for the lifetime of the component
I believe my input is controlled since it has a value. I am wondering what am I doing wrong?
I am using React 15.1.0
If the props on your component was passed as a state, put a default value for your input tags
When you first render your component,
this.state.name
isn't set, so it evaluates toundefined
, and you end up passingvalue={undefined}
to yourinput
.When ReactDOM checks to see if a field is controlled, it checks to see if
value != null
(note that it's!=
, not!==
), and sinceundefined == null
in JavaScript, it decides that it's uncontrolled.So, when
onFieldChange()
is called,this.state.name
is set to a string value, your input goes from being uncontrolled to being controlled.If you do
this.state = {name: ''}
in your constructor, because'' != null
, your input will have a value the whole time, and that message will go away.I know others have answered this already. But one of the very important factor here that may help other people experiencing similar issue:
You must need to have
onChange
handler added in your input field (e.g. textField, checkbox, radio, etc). And always handle activity through theonChange
handler, like:and when you are working with checkbox then you may need to handle its
checked
state with!!
like:When you use
onChange={this.onFieldChange('name').bind(this)}
in your input you must declare your state empty string as a value of property field.incorrect way:
correct way:
For an input to be controlled, its value must correspond to that of a state variable.
That condition is not initially met in your example because
this.state.name
is not initially set. Therefore, the input is initially uncontrolled. Once theonChange
handler is triggered for the first time,this.state.name
gets set. At that point, the above condition is satisfied and the input is considered to be controlled. This transition from uncontrolled to controlled produces the error seen above.By initializing
this.state.name
in the constructor:e.g.
the input will be controlled from the start, fixing the issue. See React Controlled Components for more examples.
Unrelated to this error, you should only have one default export. Your code above has two.
Set a value to 'name' property in initial state.