Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

How do I edit multiple input controlled components in React?

I have a component that stores a contact object as state - {firstName: "John", lastName: "Doe", phone: "1234567890} I want to create a form to edit this object but if I want the inputs to hold the value of the original contact parameter, I need to make each input a controlled component. However, I don't know how to create a handleChange function that will adjust to each parameter because my state only holds {contact: {...}}. Below is what I currently have -

  getInitialState: function () {     return ({contact: {}});   },   handleChange: function (event) {     this.setState({contact: event.target.value });   },   render: function () {     return (         <div>           <input type="text" onChange={this.handleChange} value={this.state.contact.firstName}/>           <input type="text" onChange={this.handleChange} value={this.state.contact.lastName}/>           <input type="text" onChange={this.handleChange} value={this.state.contact.lastName}/>         </div>       );     } 

I wish in my handleChange I can do something like

  handleChange: function (event) {     this.setState({contact.firstName: event.target.value });   } 
like image 403
akantoword Avatar asked Mar 13 '16 01:03

akantoword


People also ask

How do you make an input editable in React?

You need to add an onChange event and then set the shop_profile_data.NAME to a different value. Then the value of the input will change. If you only want to set the initial value of the input , use defaultValue property (docs).


2 Answers

There's a "simple" way to do this, and a "smart" way. If you ask me, doing things the smart way is not always the best, because I may be harder to work with later. In this case, both are quite understandable.

Side note: One thing I'd ask you to think about, is do you need to update the contact object, or could you just keep firstName etc. directly on state? Maybe you have a lot of data in the state of the component? If that is the case, it's probably a good idea to separate it into smaller components with narrower responsibilities.

The "simple" way

  changeFirstName: function (event) {     const contact = this.state.contact;     contact.firstName = event.target.value;     this.setState({ contact: contact });   },   changeLastName: function (event) {     const contact = this.state.contact;     contact.lastName = event.target.value;     this.setState({ contact: contact });   },   changePhone: function (event) {     const contact = this.state.contact;     contact.phone = event.target.value;     this.setState({ contact: contact });   },   render: function () {     return (       <div>         <input type="text" onChange={this.changeFirstName.bind(this)} value={this.state.contact.firstName}/>         <input type="text" onChange={this.changeLastName.bind(this)} value={this.state.contact.lastName}/>         <input type="text" onChange={this.changePhone.bind(this)} value={this.state.contact.phone}/>       </div>     );   } 

The "smart" way

  handleChange: function (propertyName, event) {     const contact = this.state.contact;     contact[propertyName] = event.target.value;     this.setState({ contact: contact });   },   render: function () {     return (       <div>         <input type="text" onChange={this.handleChange.bind(this, 'firstName')} value={this.state.contact.firstName}/>         <input type="text" onChange={this.handleChange.bind(this, 'lastName')} value={this.state.contact.lastName}/>         <input type="text" onChange={this.handleChange.bind(this, 'phone')} value={this.state.contact.lastName}/>       </div>     );   } 



Update: Same examples using ES2015+

This section contains the same examples as shown above, but using features from ES2015+.

To support the following features across browsers you need to transpile your code with Babel using e.g. the presets es2015 and react, and the plugin stage-0.

Below are updated examples, using object destructuring to get the contact from the state, spread operator to create an updated contact object instead of mutating the existing one, creating components as Classes by extending React.Component, and using arrow funtions to create callbacks so we don't have to bind(this).

The "simple" way, ES2015+

class ContactEdit extends React.Component {    changeFirstName = (event) => {     const { contact } = this.state;     const newContact = {       ...contact,       firstName: event.target.value     };     this.setState({ contact: newContact });   }   changeLastName = (event) => {     const { contact } = this.state;     const newContact = {       ...contact,       lastName: event.target.value     };     this.setState({ contact: newContact });   }   changePhone = (event) => {     const { contact } = this.state;     const newContact = {       ...contact,       phone: event.target.value     };     this.setState({ contact: newContact });   }    render() {     return (       <div>         <input type="text" onChange={this.changeFirstName} value={this.state.contact.firstName}/>         <input type="text" onChange={this.changeLastName} value={this.state.contact.lastName}/>         <input type="text" onChange={this.changePhone} value={this.state.contact.phone}/>       </div>     );   } } 

The "smart" way, ES2015+

Note that handleChangeFor is a curried function: Calling it with a propertyName creates a callback function which, when called, updates [propertyName] of the (new) contact object in the state.

class ContactEdit extends React.Component {    handleChangeFor = (propertyName) => (event) => {     const { contact } = this.state;     const newContact = {       ...contact,       [propertyName]: event.target.value     };     this.setState({ contact: newContact });   }    render() {     return (       <div>         <input type="text" onChange={this.handleChangeFor('firstName')} value={this.state.contact.firstName}/>         <input type="text" onChange={this.handleChangeFor('lastName')} value={this.state.contact.lastName}/>         <input type="text" onChange={this.handleChangeFor('phone')} value={this.state.contact.lastName}/>       </div>     );   } } 
like image 125
ArneHugo Avatar answered Oct 02 '22 18:10

ArneHugo


ES6 one liner approach

<input type="text"         value={this.state.username}        onChange={(e) => this.setState({ username: e.target.value })}        id="username"/> 
like image 39
Stu P. Avatar answered Oct 02 '22 18:10

Stu P.