Browse Source

[docs] Use controlled components for tutorial

main
Tay Yang Shun 9 years ago
parent
commit
5ab2016f32
  1. 132
      docs/tutorial.md

132
docs/tutorial.md

@ -492,28 +492,91 @@ var CommentForm = React.createClass({
}); });
``` ```
Let's make the form interactive. When the user submits the form, we should clear it, submit a request to the server, and refresh the list of comments. To start, let's listen for the form's submit event and clear it. #### Controlled components
With the traditional DOM, `input` elements are rendered and the browser manages the state (its rendered value). As a result, the state of the actual DOM will differ from that of the component. This is not ideal as the state of the view will differ from that of the component. In React, components should always represent the state of the view and not only at the point of initialization.
Hence, we will be using `this.state` to save the user's input as it is entered. We define an initial `state` with two properties `author` and `text` and set them to be empty strings. In our `<input>` elements, we set the `value` prop to reflect the `state` of the component and attach `onChange` handlers to them. These `<input>` elements with a `value` set are called controlled components. Read more about controlled components on the [Forms article](/react/docs/forms.html#controlled-components).
```javascript{3-14,16-19} ```javascript{3-11,15-26}
// tutorial16.js // tutorial16.js
var CommentForm = React.createClass({ var CommentForm = React.createClass({
getInitialState: function() {
return {author: '', text: ''};
},
handleAuthorChange: function(e) {
this.setState({author: e.target.value});
},
handleTextChange: function(e) {
this.setState({text: e.target.value});
},
render: function() {
return (
<form className="commentForm">
<input
type="text"
placeholder="Your name"
value={this.state.author}
onChange={this.handleAuthorChange}
/>
<input
type="text"
placeholder="Say something..."
value={this.state.text}
onChange={this.handleTextChange}
/>
<input type="submit" value="Post" />
</form>
);
}
});
```
#### Events
React attaches event handlers to components using a camelCase naming convention. We attach `onChange` handlers to the two `<input>` elements. Now, as the user enters text into the `<input>` fields, the attached `onChange` callbacks are fired and the `state` of the component is modified. Subsequently, the rendered value of the `input` element will be updated to reflect the current component `state`.
#### Submitting the form
Let's make the form interactive. When the user submits the form, we should clear it, submit a request to the server, and refresh the list of comments. To start, let's listen for the form's submit event and clear it.
```javascript{12-21,24}
// tutorial17.js
var CommentForm = React.createClass({
getInitialState: function() {
return {author: '', text: ''};
},
handleAuthorChange: function(e) {
this.setState({author: e.target.value});
},
handleTextChange: function(e) {
this.setState({text: e.target.value});
},
handleSubmit: function(e) { handleSubmit: function(e) {
e.preventDefault(); e.preventDefault();
var author = this.refs.author.value.trim(); var author = this.state.author.trim();
var text = this.refs.text.value.trim(); var text = this.state.text.trim();
if (!text || !author) { if (!text || !author) {
return; return;
} }
// TODO: send request to the server // TODO: send request to the server
this.refs.author.value = ''; this.setState({author: '', text: ''});
this.refs.text.value = '';
return;
}, },
render: function() { render: function() {
return ( return (
<form className="commentForm" onSubmit={this.handleSubmit}> <form className="commentForm" onSubmit={this.handleSubmit}>
<input type="text" placeholder="Your name" ref="author" /> <input
<input type="text" placeholder="Say something..." ref="text" /> type="text"
placeholder="Your name"
value={this.state.author}
onChange={this.handleAuthorChange}
/>
<input
type="text"
placeholder="Say something..."
value={this.state.text}
onChange={this.handleTextChange}
/>
<input type="submit" value="Post" /> <input type="submit" value="Post" />
</form> </form>
); );
@ -521,24 +584,18 @@ var CommentForm = React.createClass({
}); });
``` ```
##### Events We attach an `onSubmit` handler to the form that clears the form fields when the form is submitted with valid input.
React attaches event handlers to components using a camelCase naming convention. We attach an `onSubmit` handler to the form that clears the form fields when the form is submitted with valid input.
Call `preventDefault()` on the event to prevent the browser's default action of submitting the form. Call `preventDefault()` on the event to prevent the browser's default action of submitting the form.
##### Refs #### Callbacks as props
We use the `ref` attribute to assign a name to a child component and `this.refs` to reference the DOM node.
##### Callbacks as props
When a user submits a comment, we will need to refresh the list of comments to include the new one. It makes sense to do all of this logic in `CommentBox` since `CommentBox` owns the state that represents the list of comments. When a user submits a comment, we will need to refresh the list of comments to include the new one. It makes sense to do all of this logic in `CommentBox` since `CommentBox` owns the state that represents the list of comments.
We need to pass data from the child component back up to its parent. We do this in our parent's `render` method by passing a new callback (`handleCommentSubmit`) into the child, binding it to the child's `onCommentSubmit` event. Whenever the event is triggered, the callback will be invoked: We need to pass data from the child component back up to its parent. We do this in our parent's `render` method by passing a new callback (`handleCommentSubmit`) into the child, binding it to the child's `onCommentSubmit` event. Whenever the event is triggered, the callback will be invoked:
```javascript{16-18,31} ```javascript{16-18,31}
// tutorial17.js // tutorial18.js
var CommentBox = React.createClass({ var CommentBox = React.createClass({
loadCommentsFromServer: function() { loadCommentsFromServer: function() {
$.ajax({ $.ajax({
@ -577,26 +634,43 @@ var CommentBox = React.createClass({
Let's call the callback from the `CommentForm` when the user submits the form: Let's call the callback from the `CommentForm` when the user submits the form:
```javascript{10} ```javascript{19}
// tutorial18.js // tutorial19.js
var CommentForm = React.createClass({ var CommentForm = React.createClass({
getInitialState: function() {
return {author: '', text: ''};
},
handleAuthorChange: function(e) {
this.setState({author: e.target.value});
},
handleTextChange: function(e) {
this.setState({text: e.target.value});
},
handleSubmit: function(e) { handleSubmit: function(e) {
e.preventDefault(); e.preventDefault();
var author = this.refs.author.value.trim(); var author = this.state.author.trim();
var text = this.refs.text.value.trim(); var text = this.state.text.trim();
if (!text || !author) { if (!text || !author) {
return; return;
} }
this.props.onCommentSubmit({author: author, text: text}); this.props.onCommentSubmit({author: author, text: text});
this.refs.author.value = ''; this.setState({author: '', text: ''});
this.refs.text.value = '';
return;
}, },
render: function() { render: function() {
return ( return (
<form className="commentForm" onSubmit={this.handleSubmit}> <form className="commentForm" onSubmit={this.handleSubmit}>
<input type="text" placeholder="Your name" ref="author" /> <input
<input type="text" placeholder="Say something..." ref="text" /> type="text"
placeholder="Your name"
value={this.state.author}
onChange={this.handleAuthorChange}
/>
<input
type="text"
placeholder="Say something..."
value={this.state.text}
onChange={this.handleTextChange}
/>
<input type="submit" value="Post" /> <input type="submit" value="Post" />
</form> </form>
); );
@ -607,7 +681,7 @@ var CommentForm = React.createClass({
Now that the callbacks are in place, all we have to do is submit to the server and refresh the list: Now that the callbacks are in place, all we have to do is submit to the server and refresh the list:
```javascript{17-28} ```javascript{17-28}
// tutorial19.js // tutorial20.js
var CommentBox = React.createClass({ var CommentBox = React.createClass({
loadCommentsFromServer: function() { loadCommentsFromServer: function() {
$.ajax({ $.ajax({
@ -660,7 +734,7 @@ var CommentBox = React.createClass({
Our application is now feature complete but it feels slow to have to wait for the request to complete before your comment appears in the list. We can optimistically add this comment to the list to make the app feel faster. Our application is now feature complete but it feels slow to have to wait for the request to complete before your comment appears in the list. We can optimistically add this comment to the list to make the app feel faster.
```javascript{17-23} ```javascript{17-23}
// tutorial20.js // tutorial21.js
var CommentBox = React.createClass({ var CommentBox = React.createClass({
loadCommentsFromServer: function() { loadCommentsFromServer: function() {
$.ajax({ $.ajax({

Loading…
Cancel
Save