Skip to content

Latest commit

 

History

History
 
 

05_Refactor

Folders and files

NameName
Last commit message
Last commit date

parent directory

..
 
 
 
 
 
 
 
 
 
 
 
 
 
 

05 Refactor

In the previous example we set an initial username value. What happens if we expect this value to come from e.g. an AJAX request or if it could change in time? The current approach doesn't work.

We can think about two possible solutions:

  • The first idea that could come into our mind is to implement a mix: we receive via props the current name value, then we hold an state with the current editingvalue... what drawbacks could we encounter? We have to listen on the getDerivedStateFromProps (componentWillRecieveProps has been deprecated) for any change on the parent user name control and replace our state. With this approach We end up with a mixed governance.

More info about getDerivedStateFromProps: https://medium.com/@baphemot/whats-new-in-react-16-3-d2c9b7b6193b

With this solution, the code looks like this (using the new static method getDerivedStateFromProps):

Props and interface:

./src/nameEdit.tsx

interface Props {
  initialUserName: string;
  onNameUpdated: (newName: string) => any;
}

interface State {
+  initialUserName : string,
  editingName: string;
}

Constructor update:

  constructor(props: Props) {
    super(props);
    // Watch out what would happen if we get this user name via an AJAX callback
    // you will find a different implementation on 05 sample
-    this.state = { initialUserName: this.props.initialUserName ,
- editingName: this.props.initialUserName };

+    this.state = { initialUserName: this.props.initialUserName ,
+                   editingName: this.props.initialUserName 
+     };
  }

Inside the class component

  static getDerivedStateFromProps(nextProps : Props, prevState : State) : Partial<State> {
    if(nextProps.initialUserName && 
        nextProps.initialUserName != prevState.initialUserName) {
      return {editingName: nextProps.initialUserName}  
    } else {
      return null;
    }
  }
  • The second idea is to setup two properties, the parent control will hold userName and editingUsername_. Whenever the user clicks the button to replace the name, it will notify the parent control and it will replace the content of userName with the content from editingUsername. If userName gets updated by any other third party (e.g. AJAX callback) it will update as well editingUsername.

We take as a starting point sample 04 Callback:

Summary steps:

  • Update nameEdit.tsx in order to request the new editingUsername, and remove it from the state.
  • Update app.tsx to hold the new editing property in the state, pass it to the children, control and perform the proper update on the callback event from the child control.

Prerequisites

Install Node.js and npm if they are not already installed on your computer.

Verify that you are running at least node v6.x.x and npm 3.x.x by running node -v and npm -v in a terminal/console window. Older versions may produce errors.

Steps to build it

  • Copy the content from 04 Callback and execute npm install.

  • Update nameEdit.tsx in order to request the new editingUsername, and remove it from the state.

nameEdit.tsx

import * as React from 'react';

interface Props {
-  initialUserName: string;
-  onNameUpdated: (newName: string) => any;  
+  editingUserName : string;
+  onEditingNameUpdated : (newEditingName : string) => void;
+  onNameUpdateRequest : () => void;  
}

-interface State {
-  editingName: string;
-}
  
-export class NameEditComponent extends React.Component<Props, State> {
+ export class NameEditComponent extends React.Component<Props, {}> {
  constructor(props: Props) {
    super(props);

-    this.state = {editingName: this.props.initialUserName}
  }

-  onChange = (event) => {
-    this.setState({editingName: event.target.value});
-  }

-  onNameSubmit = (event) => {
-    this.props.onNameUpdated(this.state.editingName);
-  }

+  onChange = (e: React.ChangeEvent<HTMLInputElement>) => {
+    this.props.onEditingNameUpdated((e.target as HTMLInputElement).value);
+  }


  public render() {
    return (
      <div>
          <label>Update Name:</label>
-          <input value={this.state.editingName} 
-                 onChange={this.onChange}
-           />
-          <button className="btn btn-default" 
-                  onClick={this.onNameSubmit}
-           >
-             Change
-           </button>
+          <input value={this.props.editingUserName}
+                 onChange={this.onChange} 
+           />
+          <button className="btn btn-default" 
+                  onClick={this.props.onNameUpdateRequest}
+           >
+             Change
+           </button>
      </div>
    )
  }
}
  • Update app.tsx to hold the new editing property in the state, pass it to the children controls and perform the proper update on the callback event from the child control.

./src/app.tsx

import * as React from 'react';
import { HelloComponent } from './hello';
import { NameEditComponent } from './nameEdit'

interface Props {

}

interface State {
  userName : string;
+ editingUserName : string;  
}

export class App extends React.Component<Props, State> {
  constructor(props : Props) {
    super(props);

-    this.state = {userName: 'defaultUserName'};
+      const defaultUserName = 'defaultUserName';
+      this.state = {userName: defaultUserName, editingUserName: defaultUserName};
  }

-  setUsernameState = (newName: string) => {
+  setUsernameState = () => {  
-    this.setState({userName: newName});
+    this.setState({userName: this.state.editingUserName});
  }

+ updateEditingName = (editingName : string) : void => {
+   this.setState({editingUserName: editingName});
+ }

  public render() {
    return (
      <>
        <HelloComponent userName={this.state.userName}/>
-        <NameEditComponent 
-           initialUserName={this.state.userName} 
-           onNameUpdated={this.setUsernameState}
-         />
+          <NameEditComponent
+            editingUserName={this.state.editingUserName}
+            onEditingNameUpdated={this.updateEditingName}
+            onNameUpdateRequest={this.setUsernameState}
+           />
      </>
    );
  }
}

Finally we can check the example is working as in 04 Callback by executing from the command line npm start and opening http://localhost:8080.