2018-10-04 14:37:37 +01:00
---
id: 5a24c314108439a4d4036179
title: Create a Controlled Form
challengeType: 6
2019-08-05 09:17:33 -07:00
forumTopicId: 301384
2018-10-04 14:37:37 +01:00
---
## Description
2020-09-03 01:24:22 +01:00
2018-10-04 14:37:37 +01:00
<section id='description'>
The last challenge showed that React can control the internal state for certain elements like <code>input</code> and <code>textarea</code>, which makes them controlled components. This applies to other form elements as well, including the regular HTML <code>form</code> element.
</section>
## Instructions
2020-09-03 01:24:22 +01:00
2018-10-04 14:37:37 +01:00
<section id='instructions'>
The <code>MyForm</code> component is set up with an empty <code>form</code> with a submit handler. The submit handler will be called when the form is submitted.
We've added a button which submits the form. You can see it has the <code>type</code> set to <code>submit</code> indicating it is the button controlling the form. Add the <code>input</code> element in the <code>form</code> and set its <code>value</code> and <code>onChange()</code> attributes like the last challenge. You should then complete the <code>handleSubmit</code> method so that it sets the component state property <code>submit</code> to the current input value in the local <code>state</code>.
<strong>Note:</strong> You also must call <code>event.preventDefault()</code> in the submit handler, to prevent the default form submit behavior which will refresh the web page.
Finally, create an <code>h1</code> tag after the <code>form</code> which renders the <code>submit</code> value from the component's <code>state</code>. You can then type in the form and click the button (or press enter), and you should see your input rendered to the page.
</section>
## Tests
2020-09-03 01:24:22 +01:00
2018-10-04 14:37:37 +01:00
<section id='tests'>
```yml
tests:
- text: <code>MyForm</code> should return a <code>div</code> element which contains a <code>form</code> and an <code>h1</code> tag. The form should include an <code>input</code> and a <code>button</code>.
2019-07-24 05:07:46 -07:00
testString: assert((() => { const mockedComponent = Enzyme.mount(React.createElement(MyForm)); return (mockedComponent.find('div').children().find('form').length === 1 && mockedComponent.find('div').children().find('h1').length === 1 && mockedComponent.find('form').children().find('input').length === 1 && mockedComponent.find('form').children().find('button').length === 1) })());
2018-10-20 21:02:47 +03:00
- text: The state of <code>MyForm</code> should initialize with <code>input</code> and <code>submit</code> properties, both set to empty strings.
2019-07-24 05:07:46 -07:00
testString: assert(Enzyme.mount(React.createElement(MyForm)).state('input') === '' && Enzyme.mount(React.createElement(MyForm)).state('submit') === '');
2018-10-04 14:37:37 +01:00
- text: Typing in the <code>input</code> element should update the <code>input</code> property of the component's state.
2020-09-03 01:24:22 +01:00
testString: "(() => {
const mockedComponent = Enzyme.mount(React.createElement(MyForm));
const _1 = () => {
mockedComponent.setState({ input: '' });
return mockedComponent.state('input');
};
const _2 = () => {
mockedComponent
.find('input')
.simulate('change', { target: { value: 'TestInput' } });
return {
state: mockedComponent.state('input'),
inputVal: mockedComponent.find('input').props().value,
};
};
const before = _1();
const after = _2();
assert(
before === '' &&
after.state === 'TestInput' &&
after.inputVal === 'TestInput'
);
})();
"
2018-10-04 14:37:37 +01:00
- text: Submitting the form should run <code>handleSubmit</code> which should set the <code>submit</code> property in state equal to the current input.
2020-09-03 01:24:22 +01:00
testString: "(() => {
const mockedComponent = Enzyme.mount(React.createElement(MyForm));
const _1 = () => {
mockedComponent.setState({ input: '' });
mockedComponent.setState({ submit: '' });
mockedComponent
.find('input')
.simulate('change', { target: { value: 'SubmitInput' } });
return mockedComponent.state('submit');
};
const _2 = () => {
mockedComponent.find('form').simulate('submit');
return mockedComponent.state('submit');
};
const before = _1();
const after = _2();
assert(before === '' && after === 'SubmitInput');
})();
"
2018-10-04 14:37:37 +01:00
- text: The <code>h1</code> header should render the value of the <code>submit</code> field from the component's state.
2020-09-03 01:24:22 +01:00
testString: "(() => {
const mockedComponent = Enzyme.mount(React.createElement(MyForm));
const _1 = () => {
mockedComponent.setState({ input: '' });
mockedComponent.setState({ submit: '' });
mockedComponent
.find('input')
.simulate('change', { target: { value: 'TestInput' } });
return mockedComponent.find('h1').text();
};
const _2 = () => {
mockedComponent.find('form').simulate('submit');
return mockedComponent.find('h1').text();
};
const before = _1();
const after = _2();
assert(before === '' && after === 'TestInput');
})();
"
2018-10-04 14:37:37 +01:00
```
</section>
## Challenge Seed
2020-09-03 01:24:22 +01:00
2018-10-04 14:37:37 +01:00
<section id='challengeSeed'>
<div id='jsx-seed'>
```jsx
class MyForm extends React.Component {
constructor(props) {
super(props);
this.state = {
2018-10-20 21:02:47 +03:00
input: '',
submit: ''
2018-10-04 14:37:37 +01:00
};
this.handleChange = this.handleChange.bind(this);
this.handleSubmit = this.handleSubmit.bind(this);
}
handleChange(event) {
this.setState({
input: event.target.value
});
}
handleSubmit(event) {
2020-09-15 09:53:25 -07:00
// Change code below this line
2020-09-03 01:24:22 +01:00
2020-09-15 09:53:25 -07:00
// Change code above this line
2018-10-04 14:37:37 +01:00
}
render() {
return (
<div>
<form onSubmit={this.handleSubmit}>
2020-09-15 09:53:25 -07:00
{/* Change code below this line */}
2018-10-04 14:37:37 +01:00
2020-09-15 09:53:25 -07:00
{/* Change code above this line */}
2018-10-04 14:37:37 +01:00
<button type='submit'>Submit!</button>
</form>
2020-09-15 09:53:25 -07:00
{/* Change code below this line */}
2018-10-04 14:37:37 +01:00
2020-09-15 09:53:25 -07:00
{/* Change code above this line */}
2018-10-04 14:37:37 +01:00
</div>
);
}
2020-09-03 01:24:22 +01:00
}
2018-10-04 14:37:37 +01:00
```
</div>
### After Test
2020-09-03 01:24:22 +01:00
2018-10-04 14:37:37 +01:00
<div id='jsx-teardown'>
2020-07-13 18:58:50 +02:00
```jsx
2020-09-03 01:24:22 +01:00
ReactDOM.render(<MyForm />, document.getElementById('root'));
2018-10-04 14:37:37 +01:00
```
</div>
</section>
## Solution
2020-09-03 01:24:22 +01:00
<section id='solution'>
2018-10-04 14:37:37 +01:00
2020-07-13 18:58:50 +02:00
```jsx
2018-10-04 14:37:37 +01:00
class MyForm extends React.Component {
constructor(props) {
super(props);
this.state = {
2018-10-20 21:02:47 +03:00
input: '',
submit: ''
2018-10-04 14:37:37 +01:00
};
this.handleChange = this.handleChange.bind(this);
this.handleSubmit = this.handleSubmit.bind(this);
}
handleChange(event) {
this.setState({
input: event.target.value
});
}
handleSubmit(event) {
2020-09-03 01:24:22 +01:00
event.preventDefault();
this.setState(state => ({
2020-08-11 22:42:27 +05:30
submit: state.input
}));
2018-10-04 14:37:37 +01:00
}
render() {
return (
<div>
<form onSubmit={this.handleSubmit}>
2020-09-03 01:24:22 +01:00
<input value={this.state.input} onChange={this.handleChange} />
2018-10-04 14:37:37 +01:00
<button type='submit'>Submit!</button>
</form>
<h1>{this.state.submit}</h1>
</div>
);
}
2020-09-03 01:24:22 +01:00
}
2018-10-04 14:37:37 +01:00
```
</section>