React

React Google Login

Introduction

In the last tutorial, you built a User login/logout feature to your App using the Parse.User class. Now you will learn how to use Google Sign-in to retrieve user data from Google and log in, sign up or link existent users with it. You will also install and configure react-google-login lib to achieve that.

The Parse.User.linkWith method is responsible for signing up and logging in users using any third-party authentication method, as long as you pass the right parameters requested by each different provider. After linking the user data to a new or existent Parse.User, Parse will store a valid user session on your device. Future calls to methods like current will successfully retrieve your User data, just like with regular logins.

Prerequisites

To complete this tutorial, you will need:

Goal

To build a User LogIn feature using Google Sign-in on Parse for a React App.

Step 1 - Installing dependencies

The most popular way to enable Google Sign-in on React is using react-google-login to handle it. Set it up following the official docs.

Make sure to create an OAuth clientId in your Google Credentials page, adding your local development address to the authorized URLs, which generally is http://localhost:3000.

Step 2 - Usign Google Sign-in with Parse

Let’s now create a new method inside the UserLogIn component that will handle the response when calling a Google Sign-in authentication modal. If the user signs in with Google, this call will retrieve the user data from Google and you need to store the id, idToken, and Google email. Then, the function will try to log in on Parse using the Parse.User.linkWith method and these credentials. Note that if your user had already signed up using this Google authentication, linkWith will log him in using the existent account.

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
const handleGoogleLoginLoginResponse = async function(response) {
  // Check if response has an error
  if (response.error !== undefined) {
    console.log(`Error: ${response.error}`);
    return false;
  } else {
    try {
      // Gather Google user info
      const userGoogleId = response.googleId;
      const userTokenId = response.tokenId;
      const userEmail = response.profileObj.email;
      // Try to login on Parse using linkWith and these credentials
      // Create a new Parse.User object
      const userToLogin = new Parse.User();
      // Set username and email to match google email
      userToLogin.set('username', userEmail);
      userToLogin.set('email', userEmail);
      try {
        let loggedInUser = await userToLogin
        .linkWith('google', {
          authData: {id: userGoogleId, id_token: userTokenId},
        });
        // logIn returns the corresponding ParseUser object
        alert(
          `Success! User ${loggedInUser.get('username')} has successfully signed in!`,
        );
        // Update state variable holding current user
        getCurrentUser();
        return true;
      } catch (error) {
        // Error can be caused by wrong parameters or lack of Internet connection
        alert(`Error! ${error.message}`);
        return false;
      }
    } catch (error) {
      console.log("Error gathering Google user info, please try again!")
      return false;
    }
  }
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
const handleGoogleLoginLoginResponse = async function(response: any): Promise<boolean> {
  // Check if response has an error
  if (response.error !== undefined) {
    console.log(`Error: ${response.error}`);
    return false;
  } else {
    try {
      // Gather Google user info
      const userGoogleId: string = response.googleId;
      const userTokenId: string = response.tokenId;
      const userEmail: string = response.profileObj.email;
      // Try to login on Parse using linkWith and these credentials
      // Create a new Parse.User object
      const userToLogin: Parse.User = new Parse.User();
      // Set username and email to match google email
      userToLogin.set('username', userEmail);
      userToLogin.set('email', userEmail);
      try {
        let loggedInUser: Parse.User = await userToLogin
        .linkWith('google', {
          authData: {id: userGoogleId, id_token: userTokenId},
        });
        // logIn returns the corresponding ParseUser object
        alert(
          `Success! User ${loggedInUser.get('username')} has successfully signed in!`,
        );
        // Update state variable holding current user
        getCurrentUser();
        return true;
      } catch (error: any) {
        // Error can be caused by wrong parameters or lack of Internet connection
        alert(`Error! ${error.message}`);
        return false;
      }
    } catch (error: any) {
      console.log("Error gathering Google user info, please try again!")
      return false;
    }
  }
}

After that, you need to use the react-google-login GoogleLogin component to call the Google SignIn modal, adding it to your JSX code. You can use Google’s default styling or create a custom one, which is the way followed by this guide. Here is the full UserLogin component code, note the react-google-login button and how it is tied to the modal response method created before.

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
import React, { useState } from 'react';
import Parse from 'parse/dist/parse.min.js';
import GoogleLogin from 'react-google-login';
import './App.css';
import { Button, Divider, Input } from 'antd';

export const UserLogin = () => {
  // State variables
  const [username, setUsername] = useState('');
  const [password, setPassword] = useState('');
  const [currentUser, setCurrentUser] = useState(null);

  const doUserLogIn = async function () {
    // Note that these values come from state variables that we've declared before
    const usernameValue = username;
    const passwordValue = password;
    try {
      const loggedInUser = await Parse.User.logIn(usernameValue, passwordValue);
      // logIn returns the corresponding ParseUser object
      alert(
        `Success! User ${loggedInUser.get(
          'username'
        )} has successfully signed in!`
      );
      // To verify that this is in fact the current user, `current` can be used
      const currentUser = await Parse.User.current();
      console.log(loggedInUser === currentUser);
      // Clear input fields
      setUsername('');
      setPassword('');
      // Update state variable holding current user
      getCurrentUser();
      return true;
    } catch (error) {
      // Error can be caused by wrong parameters or lack of Internet connection
      alert(`Error! ${error.message}`);
      return false;
    }
  };

  const doUserLogOut = async function () {
    try {
      await Parse.User.logOut();
      // To verify that current user is now empty, currentAsync can be used
      const currentUser = await Parse.User.current();
      if (currentUser === null) {
        alert('Success! No user is logged in anymore!');
      }
      // Update state variable holding current user
      getCurrentUser();
      return true;
    } catch (error) {
      alert(`Error! ${error.message}`);
      return false;
    }
  };

  // Function that will return current user and also update current username
  const getCurrentUser = async function () {
    const currentUser = await Parse.User.current();
    // Update state variable holding current user
    setCurrentUser(currentUser);
    return currentUser;
  };

  const handleGoogleLoginLoginResponse = async function (response) {
    // Check if response has an error
    if (response.error !== undefined) {
      console.log(`Error: ${response.error}`);
      return false;
    } else {
      try {
        // Gather Google user info
        const userGoogleId = response.googleId;
        const userTokenId = response.tokenId;
        const userEmail = response.profileObj.email;
        // Try to login on Parse using linkWith and these credentials
        // Create a new Parse.User object
        const userToLogin = new Parse.User();
        // Set username and email to match google email
        userToLogin.set('username', userEmail);
        userToLogin.set('email', userEmail);
        try {
          let loggedInUser = await userToLogin.linkWith('google', {
            authData: { id: userGoogleId, id_token: userTokenId },
          });
          // logIn returns the corresponding ParseUser object
          alert(
            `Success! User ${loggedInUser.get(
              'username'
            )} has successfully signed in!`
          );
          // Update state variable holding current user
          getCurrentUser();
          return true;
        } catch (error) {
          // Error can be caused by wrong parameters or lack of Internet connection
          alert(`Error! ${error.message}`);
          return false;
        }
      } catch (error) {
        console.log('Error gathering Google user info, please try again!');
        return false;
      }
    }
  };

  return (
    <div>
      <div className="header">
        <img
          className="header_logo"
          alt="Back4App Logo"
          src={
            'https://blog.back4app.com/wp-content/uploads/2019/05/back4app-white-logo-500px.png'
          }
        />
        <p className="header_text_bold">{'React on Back4App'}</p>
        <p className="header_text">{'User Login'}</p>
      </div>
      {currentUser === null && (
        <div className="container">
          <h2 className="heading">{'User Login'}</h2>
          <Divider />
          <div className="form_wrapper">
            <Input
              value={username}
              onChange={(event) => setUsername(event.target.value)}
              placeholder="Username"
              size="large"
              className="form_input"
            />
            <Input
              value={password}
              onChange={(event) => setPassword(event.target.value)}
              placeholder="Password"
              size="large"
              type="password"
              className="form_input"
            />
          </div>
          <div className="form_buttons">
            <Button
              onClick={() => doUserLogIn()}
              type="primary"
              className="form_button"
              color={'#208AEC'}
              size="large"
              block
            >
              Log In
            </Button>
          </div>
          <Divider />
          <div className="login-social">
            <div className="login-social-item login-social-item--facebook">
              <img className="login-social-item__image" src={'https://findicons.com/files/icons/2830/clean_social_icons/250/facebook.png'} alt=""/>
            </div>
            <GoogleLogin
              clientId="108490793456-0flm4qh8ek4cb4krt7e06980o4sjvado.apps.googleusercontent.com"
              render={renderProps => (
                  <div className="login-social-item">
                    <img onClick={renderProps.onClick} className="login-social-item__image" src={''} alt=""/>
                  </div>
              )}
              buttonText="Login"
              onSuccess={handleGoogleLoginLoginResponse}
              onFailure={handleGoogleLoginLoginResponse}
              cookiePolicy={'single_host_origin'}
            />
            <div className="login-social-item">
              <img className="login-social-item__image" src={''} alt=""/>
            </div>
          </div>
          <p className="form__hint">Don't have an account? <a className="form__link" href="#">Sign up</a></p>
        </div>
      )}
      {currentUser !== null && (
        <div className="container">
          <h2 className="heading">{'User Screen'}</h2>
          <Divider />
          <h2 className="heading">{`Hello ${currentUser.get('username')}!`}</h2>
          <div className="form_buttons">
            <Button
              onClick={() => doUserLogOut()}
              type="primary"
              className="form_button"
              color={'#208AEC'}
              size="large"
              block
            >
              Log Out
            </Button>
          </div>
        </div>
      )}
    </div>
  );
};
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
import React, { useState, FC, ReactElement } from 'react';
import './App.css';
import { Button, Divider, Input } from 'antd';
import GoogleLogin from 'react-google-login';
const Parse = require('parse/dist/parse.min.js');

export const UserLogin: FC<{}> = (): ReactElement => {
  // State variables
  const [username, setUsername] = useState('');
  const [password, setPassword] = useState('');
  const [currentUser, setCurrentUser] = useState<Parse.Object | null>(null);

  const doUserLogIn = async function (): Promise<boolean> {
    // Note that these values come from state variables that we've declared before
    const usernameValue: string = username;
    const passwordValue: string = password;
    try {
      const loggedInUser: Parse.User = await Parse.User.logIn(usernameValue, passwordValue);
      // logIn returns the corresponding ParseUser object
      alert(
        `Success! User ${loggedInUser.get('username')} has successfully signed in!`,
      );
      // To verify that this is in fact the current user, `current` can be used
      const currentUser: Parse.User = await Parse.User.current();
      console.log(loggedInUser === currentUser);
      // Clear input fields
      setUsername('');
      setPassword('');
      // Update state variable holding current user
      getCurrentUser();
      return true;
    } catch (error: any) {
      // Error can be caused by wrong parameters or lack of Internet connection
      alert(`Error! ${error.message}`);
      return false;
    }
  };

  const doUserLogOut = async function (): Promise<boolean> {
    try {
      await Parse.User.logOut();
      // To verify that current user is now empty, currentAsync can be used
      const currentUser: Parse.User = await Parse.User.current();
      if (currentUser === null) {
        alert('Success! No user is logged in anymore!');
      }
      // Update state variable holding current user
      getCurrentUser();
      return true;
    } catch (error: any) {
      alert(`Error! ${error.message}`);
      return false;
    }
  };

  // Function that will return current user and also update current username
  const getCurrentUser = async function (): Promise<Parse.User | null> {
    const currentUser: (Parse.User | null) = await Parse.User.current();
    // Update state variable holding current user
    setCurrentUser(currentUser);
    return currentUser;
  }

  const handleGoogleLoginLoginResponse = async function(response: any): Promise<boolean> {
    // Check if response has an error
    if (response.error !== undefined) {
      console.log(`Error: ${response.error}`);
      return false;
    } else {
      try {
        // Gather Google user info
        const userGoogleId: string = response.googleId;
        const userTokenId: string = response.tokenId;
        const userEmail: string = response.profileObj.email;
        // Try to login on Parse using linkWith and these credentials
        // Create a new Parse.User object
        const userToLogin: Parse.User = new Parse.User();
        // Set username and email to match google email
        userToLogin.set('username', userEmail);
        userToLogin.set('email', userEmail);
        try {
          let loggedInUser: Parse.User = await userToLogin
          .linkWith('google', {
            authData: {id: userGoogleId, id_token: userTokenId},
          });
          // logIn returns the corresponding ParseUser object
          alert(
            `Success! User ${loggedInUser.get('username')} has successfully signed in!`,
          );
          // Update state variable holding current user
          getCurrentUser();
          return true;
        } catch (error: any) {
          // Error can be caused by wrong parameters or lack of Internet connection
          alert(`Error! ${error.message}`);
          return false;
        }
      } catch (error: any) {
        console.log("Error gathering Google user info, please try again!")
        return false;
      }
    }
  }

  return (
    <div>
      <div className="header">
        <img
          className="header_logo"
          alt="Back4App Logo"
          src={
            'https://blog.back4app.com/wp-content/uploads/2019/05/back4app-white-logo-500px.png'
          }
        />
        <p className="header_text_bold">{'React on Back4App'}</p>
        <p className="header_text">{'User Login'}</p>
      </div>
      {currentUser === null && (
        <div className="container">
          <h2 className="heading">{'User Login'}</h2>
          <Divider />
          <div className="form_wrapper">
            <Input
              value={username}
              onChange={(event) => setUsername(event.target.value)}
              placeholder="Username"
              size="large"
              className="form_input"
            />
            <Input
              value={password}
              onChange={(event) => setPassword(event.target.value)}
              placeholder="Password"
              size="large"
              type="password"
              className="form_input"
            />
          </div>
          <div className="form_buttons">
            <Button
              onClick={() => doUserLogIn()}
              type="primary"
              className="form_button"
              color={'#208AEC'}
              size="large"
              block
            >
              Log In
            </Button>
          </div>
          <Divider />
          <div className="login-social">
            <div className="login-social-item login-social-item--facebook">
              <img className="login-social-item__image" src={'https://findicons.com/files/icons/2830/clean_social_icons/250/facebook.png'} alt=""/>
            </div>
            <GoogleLogin
              clientId="108490793456-0flm4qh8ek4cb4krt7e06980o4sjvado.apps.googleusercontent.com"
              render={renderProps => (
                  <div className="login-social-item">
                    <img onClick={renderProps.onClick} className="login-social-item__image" src={''} alt=""/>
                  </div>
              )}
              buttonText="Login"
              onSuccess={handleGoogleLoginLoginResponse}
              onFailure={handleGoogleLoginLoginResponse}
              cookiePolicy={'single_host_origin'}
            />
            <div className="login-social-item">
              <img className="login-social-item__image" src={''} alt=""/>
            </div>
          </div>
          <p className="form__hint">Don't have an account? <a className="form__link" href="#">Sign up</a></p>
        </div>
      )}
      {currentUser !== null && (
        <div className="container">
          <h2 className="heading">{'User Screen'}</h2>
          <Divider />
          <h2 className="heading">{`Hello ${currentUser.get('username')}!`}</h2>
          <div className="form_buttons">
            <Button
              onClick={() => doUserLogOut()}
              type="primary"
              className="form_button"
              color={'#208AEC'}
              size="large"
              block
            >
              Log Out
            </Button>
          </div>
        </div>
      )}
    </div>
  );
};

Add these classes to your App.css file if you want to fully render this component’s layout.

App.css

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
@import '~antd/dist/antd.css';

.App {
  text-align: center;
}

html {
  box-sizing: border-box;
  outline: none;
  overflow: auto;
}

*,
*:before,
*:after {
  margin: 0;
  padding: 0;
  box-sizing: inherit;
}

h1,
h2,
h3,
h4,
h5,
h6 {
  margin: 0;
  font-weight: bold;
}

p {
  margin: 0;
}

body {
  margin: 0;
  background-color: #fff;
}

.container {
  width: 100%;
  max-width: 900px;
  margin: auto;
  padding: 20px 0;
  text-align: left;
}

.header {
  align-items: center;
  padding: 25px 0;
  background-color: #208AEC;
}

.header_logo {
  height: 55px;
  margin-bottom: 20px;
  object-fit: contain;
}

.header_text_bold {
  margin-bottom: 3px;
  color: rgba(255, 255, 255, 0.9);
  font-size: 16px;
  font-weight: bold;
}

.header_text {
  color: rgba(255, 255, 255, 0.9);
  font-size: 15px;
}

.heading {
  font-size: 22px;
}

.flex {
  display: flex;
}

.flex_between {
  display: flex;
  justify-content: space-between;
}

.flex_child {
  flex: 0 0 45%;
}

.heading_button {
  margin-left: 12px;
}

.list_item {
  padding-bottom: 15px;
  margin-bottom: 15px;
  border-bottom: 1px solid rgba(0, 0, 0, 0.06);
  text-align: left;
}

.list_item_title {
  color: rgba(0, 0, 0, 0.87);
  font-size: 17px;
}

.list_item_description {
  color: rgba(0, 0, 0, 0.5);
  font-size: 15px;
}

.form_input {
  margin-bottom: 20px;
}

.login-social {
  display: flex;
  justify-content: center;
  margin-bottom: 30px;
}

.login-social-item {
  width: 54px;
  height: 54px;
  border-radius: 54px;
  padding: 12px;
  margin: 0 12px;
  border: 1px solid #e6e6e6;
  box-shadow: 0 2px 4px #d6d6d6;
}

.login-social-item--facebook {
  padding: 4px;
  background-color: #3C5B9B;
}

.login-social-item__image {
  width: 100%;
}

.form__hint {
  color: rgba(0, 0, 0, 0.5);
  font-size: 16px;
  text-align: center;
}

Go ahead and test your new function. If you were able to sign in to Google and the Parse linkWith call was successful, you should see a success message like this.

React Back4App

Step 3 - Verifying user sign in and session creation

To make sure that the Google sign-in worked, you can look at your Parse dashboard and see your new User (if your Google authentication data didn’t belong to another user), containing the Google authData parameters.

React Back4App

You can also verify that a valid session was created in the dashboard, containing a pointer to that User object.

React Back4App

Step 4 - Linking an existing User to Google Sign-in

Another linkWith possible use is to link an existing user with another auth provider, in this case, Google. Add this function that calls linkWith the same way as logging in to your UserLogIn component. The only difference here is that instead of calling the method from an empty Parse.User, you will use it from the currently logged-in user object.

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
const handleGoogleLoginLinkResponse = async function(response) {
  // Check if response has an error
  if (response.error !== undefined) {
    console.log(`Error: ${response.error}`);
    return false;
  } else {
    try {
      // Gather Google user info
      const userGoogleId = response.googleId;
      const userTokenId = response.tokenId;
      // Try to link current Parse user using linkWith and these credentials
      // Get current user
      const userToLink = await Parse.User.current();
      try {
        let loggedInUser = await userToLink
        .linkWith('google', {
          authData: {id: userGoogleId, id_token: userTokenId},
        });
        // logIn returns the corresponding ParseUser object
        alert(
          `Success! User ${loggedInUser.get(
            'username',
          )} has successfully linked his Google account!`,
        );
        // Update state variable holding current user
        getCurrentUser();
        return true;
      } catch (error) {
        // Error can be caused by wrong parameters or lack of Internet connection
        alert(`Error! ${error.message}`);
        return false;
      }
    } catch (error) {
      console.log("Error gathering Google user info, please try again!")
      return false;
    }
  }
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
const handleGoogleLoginLinkResponse = async function(response: any): Promise<boolean> {
  // Check if response has an error
  if (response.error !== undefined) {
    console.log(`Error: ${response.error}`);
    return false;
  } else {
    try {
      // Gather Google user info
      const userGoogleId: string = response.googleId;
      const userTokenId: string = response.tokenId;
      // Try to link current Parse user using linkWith and these credentials
      // Get current user
      const userToLink: Parse.User = await Parse.User.current();
      try {
        let loggedInUser: Parse.User = await userToLink
        .linkWith('google', {
          authData: {id: userGoogleId, id_token: userTokenId},
        });
        // logIn returns the corresponding ParseUser object
        alert(
          `Success! User ${loggedInUser.get(
            'username',
          )} has successfully linked his Google account!`,
        );
        // Update state variable holding current user
        getCurrentUser();
        return true;
      } catch (error: any) {
        // Error can be caused by wrong parameters or lack of Internet connection
        alert(`Error! ${error.message}`);
        return false;
      }
    } catch (error: any) {
      console.log("Error gathering Google user info, please try again!")
      return false;
    }
  }
}

Assign this function to another react-google-login component in your home screen, which is shown only when there is a current user logged in in your app. Test your new function, noting that the Parse.User object authData value will be updated with the new auth provider data. Verify if the user has indeed updated in your Parse server dashboard.

React Back4App

Conclusion

At the end of this guide, you learned how to log in, sign up or link existing Parse users on React using Google Sign-in with react-google-login. In the next guide, we will show you how to use Apple sign-in.