Използване на бутон за влизане в Google с реакция

Опитвам се да използвам входа в Google в приложение за реакция. Докато използването на бутона за влизане извън самото приложение работи чудесно, когато го използвам в персонализиран компонент за вход, не мога да го накарам да работи според очакванията. Когато потребителят влезе, самият бутон трябва да изпълни метод data-onsuccess. Проблемът е, че изпълнението никога не достига този метод, въпреки че влизането работи.

Вероятно пропускам някаква реакция, но наистина не мога да я намеря. Някаква помощ? Намерете по-долу html, който зарежда всичко и самия компонент.

<head>
    <meta name="google-signin-scope" content="profile email">
    <meta name="google-signin-client_id" content="1234-real-client-id.apps.googleusercontent.com">
    <script src="https://apis.google.com/js/platform.js" async defer></script>
</head>
<body>
    <!-- Here is where everything gets displayed -->
    <div id="app"></div>

    <!-- The file with the js code -->
    <script src="/js/main.js"></script>
</body>


var SignIn = React.createClass({
    onSignIn : function (google_user) {
        // I want this method to be executed
    },

    render : function() {
        return (
            <div className="g-signin2" data-onsuccess={this.onSignIn} data-theme="dark" />
        );
    }
});

Забележете, че не поставих тук неподходящ код ;)


person ThisIsErico    schedule 24.07.2015    source източник
comment
Този видеоклип обяснява Google, facebook и свързаното влизане с reactjs -youtu.be/9MhLHkf7Ifs   -  person Prem    schedule 04.08.2017


Отговори (6)


Опитайте да добавите обратното извикване onSuccess, когато инициализирате компонента в componentDidMount().

componentDidMount: function() {
  gapi.signin2.render('g-signin2', {
    'scope': 'https://www.googleapis.com/auth/plus.login',
    'width': 200,
    'height': 50,
    'longtitle': true,
    'theme': 'dark',
    'onsuccess': this. onSignIn
  });  
},
...

Източници: https://developers.google.com/identity/sign-in/web/build-button, https://github.com/meta-meta/webapp-template/blob/6d3e9c86f5c274656ef799263c84a228bfbe1725/app/Application/signIn.jsx.

person BradByte    schedule 24.07.2015
comment
Това всъщност работи добре! :) Предполагам, че реакцията всъщност не е добавяне на правилния атрибут на данни. Благодаря много Брад! - person ThisIsErico; 24.07.2015
comment
Благодаря ви много, че се свързвате към ангажимент да не /master или /develop, както правят повечето хора. Ти си светец, @BradBumbalough!!!! - person Chris Browne; 01.12.2016
comment
Забележете, премахнете реда scope, ако не искате да подканяте за тези разрешения. - person spencer.sm; 27.02.2019

Трябваше да използвам това във функционален компонент, така че реших да споделя как го адаптирах, трябваше да използвам useEffectHook, който може да се използва вместо componentDidMount

  useEffect(() => {
    window.gapi.signin2.render('g-signin2', {
      'scope': 'https://www.googleapis.com/auth/plus.login',
      'width': 200,
      'height': 50,
      'longtitle': true,
      'theme': 'dark',
      'onsuccess': onSignIn
    })
  })
person Max Carroll    schedule 30.07.2019
comment
Опитах вашия код, но onsuccess не се задейства. Необходима ли е друга настройка? - person Ejaz; 03.09.2019
comment
не съм сигурен какво може да е това, предавам моя метод onsuccess като опора от родителския възел, опитахте ли очевидните неща чрез отстраняване на грешки и гарантиране, че вашето обратно извикване onsuccess е дефинирано като точка на изпълнение, може би залепете точка на прекъсване в прозореца .gapi на ред 2 и се уверете, че null не се предава там, в противен случай тази функция трябва да бъде извикана, след като са завършили процеса на влизане (напр. след като изберат и акаунт, въведат паролата си и одобрят всички обхвати, които се нуждаят от одобрение) - Но за да отговоря на въпроса ви, всичко останало, което е необходимо, е да дефинирате onSignIn - person Max Carroll; 23.09.2019
comment
@Ejaz Може би проверете отговора ми: stackoverflow.com/a/59040581/996314 - person Rokit; 26.11.2019
comment
Разреших моето чрез използване на GoogleAuth.isSignedIn.listen() въз основа на developers.google.com/identity/sign-in/web/reference. Става така window.gapi.load('auth2', () => { const auth2 = window.gapi.auth2.init({ config goes here }); auth2.isSignedIn.listen(() => { your code here }) }). Ако искате да получите потребителския профил, можете да използвате GoogleUser.getBasicProfile(). Надявам се това да помогне. - person Yor; 05.02.2020

Сега можете да използвате този React Google Login NPM пакет, който капсулира цялата работа по настройката . Можете просто да предавате опции само на компонента.

import React from 'react';
import ReactDOM from 'react-dom';
import GoogleLogin from 'react-google-login';
// or
import { GoogleLogin } from 'react-google-login';
 
 
const responseGoogle = (response) => {
  console.log(response);
}
 
ReactDOM.render(
  <GoogleLogin
    clientId="658977310896-knrl3gka66fldh83dao2rhgbblmd4un9.apps.googleusercontent.com"
    buttonText="Login"
    onSuccess={responseGoogle}
    onFailure={responseGoogle}
    cookiePolicy={'single_host_origin'}
  />,
  document.getElementById('googleButton')
);

Кодът е от документацията на пакета.

person AV Paul    schedule 07.08.2019
comment
react-google-login работи страхотно! Трябваше да използвам и двете, react state и localstorage, за да получа опресняване след влизане. т.е.: onSuccess={(response) => { this.setState({ userLoggedIn: true, }, () => { localStorage.setItem('userLoggedIn', true); }); - person Miguel Reyes; 17.08.2019

Може да е малко късно, но искам да споделя начина си на внедряване. Както в моя случай, искам да персонализирам напълно бутона и също така искам да имам някакъв зареждащ инструмент. Реализирах това с помощта на куки

/* istanbul ignore file */
import { useState, useEffect, useCallback } from 'react';
import { APP_IDS } from 'utils/constants';

const initgAuth = () =>
  // eslint-disable-next-line no-unused-vars
  new Promise((resolve, _reject) => {
    // eslint-disable-next-line no-undef
    gapi.load('auth2', () => {
      // eslint-disable-next-line no-undef
      const auth = gapi.auth2.init({
        client_id: APP_IDS.Google,
      });
      resolve(auth);
    });
  });

const initializeGoogle = () =>
  // eslint-disable-next-line no-unused-vars
  new Promise((resolve, _reject) => {
    if (typeof gapi !== 'undefined') {
      window.googleAsyncInit().then(auth => {
        resolve(auth);
      });
    } else {
      // eslint-disable-next-line func-names
      window.googleAsyncInit = async function() {
        const auth = await initgAuth();
        resolve(auth);
        return auth;
      };

      // eslint-disable-next-line func-names
      (function(d, s, id) {
        let js;
        const gjs = d.getElementsByTagName(s)[0];
        if (d.getElementById(id)) {
          return;
        }
        // eslint-disable-next-line prefer-const
        js = d.createElement(s);
        js.id = id;
        js.src = 'https://apis.google.com/js/platform.js';
        js.onload = window.googleAsyncInit;
        gjs.parentNode.insertBefore(js, gjs);
      })(document, 'script', 'google_api');
    }
  });

const useGoogle = () => {
  const [google, setGoogle] = useState([]);
  const [isReady, setReady] = useState(false);

  const initGoogle = useCallback(async () => {
    const auth = await initializeGoogle();
    if (auth !== 'undefined') {
      setGoogle(auth);
      setReady(true);
    }
  });

  useEffect(() => {
    initGoogle();
  }, [initGoogle]);

  return [google, isReady];
};

Така че мога просто да използвам това във всеки компонент като този

  const [GoogleInstance, isGoogleReady] = useGoogle();

   GoogleInstance.signIn().then(response => {
      const authResponse = response.getAuthResponse(true);
      doGoogleLogin(authResponse);
    });

Това е доста удобно, защото мога да се уверя, че SDK е зареден, преди потребителят да може да взаимодейства със самия бутон. Също така може да се използва, ако имате различни компоненти/страници, които трябва да имат вход в Google

person keysl    schedule 21.04.2021

Просто се уверете, че сте добавили platform.js скриптовия маркер на Google към index.html на вашето приложение React:

<script src="https://apis.google.com/js/platform.js" async defer></script>

Вземете своя клиентски идентификатор от:

https://console.developers.google.com/apis/credentials


РЕДАКТИРАНЕ Актуализиран пример с внедряване в реалния свят, което използвам в разгърнати производствени приложения

И ето примерна реализация на функционален компонент SignInWithGoogle в Typescript, така че можете да използвате като вдъхновение:

// window.ts
export {};

declare global {
  interface Window {
    gapi: any;
  }
}


// index.tsx
import './window';

import React, {
  useEffect,
  useState,
} from 'react';

const SignInWithGoogle = () => {
  const [loaded, setLoaded] = useState<boolean>(false);
  const [signedIn, setSignedIn] = useState<boolean>(false);

  const onSignIn = (googleUser: {[k: string]: any}) => {
    if (!signedIn) {
      setSignedIn(true);

      const token = googleUser.getAuthResponse().id_token;

      // do something with the token (like send it to your api)...
    }
  };

  const onFail = ({
    error,
  }: {
    error: string;
  }) => {
    // do something
  };

  useEffect(() => {
    const { gapi } = window;

    if (!loaded) {
      gapi.load('auth2', () => {
        gapi.auth2.init().then((auth: any) => {
          auth.signOut().then(() => {
            gapi.signin2.render('google-signin-button', {
              width: 232,
              height: 40,
              longtitle: true,
              onsuccess: onSignIn,
              onfailure: onFail,
            });
          });
        });

        setLoaded(true);
      });
    }
  });

  return (
    <div
      id="google-signin-button"
      className="google-signin-button"
    />
  );
};

export default SignInWithGoogle;
person JeanLescure    schedule 25.11.2020

Нито един от другите отговори не ми помогна. Нещото, което най-накрая проработи, беше използването на id вместо клас за идентификатора на div.

<div id="g-signin2" />

вместо

<div className="g-signin2" />

В допълнение към използването на window.gapi.signin2.render, както предполагат други отговори.

person Alex L    schedule 11.12.2020