Flutter/Social Login

[Flutter] Facebook Login ② - 구현

찌김이 2022. 8. 19. 16:10
728x90
반응형
 

[Flutter] Facebook Login ① - 준비

소셜 로그인 구현 첫번째는 페이스북 로그인 입니다. 파이어베이스 사용여부와 상관없이 세팅방법은 똑같습니다. Meta for Developers Conversations 2022: 제1회 Meta 비즈니스 메시지 콘퍼런스 가장 인기

dalgoodori.tistory.com

 

이전 포스팅에서 페이스북 로그인을 위한 모든 셋팅을 완료하였습니다. 

이번 포스팅에서는 페이스북 로그인을 구현하고 유저의 이메일, 이름을 가져오는 것 까지 구현해보겠습니다.

 

 

pubspec.yaml

 

flutter_facebook_auth | Flutter Package

The easiest way to add facebook login to your flutter app. Feature includes getting user information, profile picture and more. This plugin also supports Web.

pub.dev

dependencies:
  flutter_facebook_auth: ^4.4.1

 

  • 모든 셋팅을 정상적으로 완료하였다면 아래의 코드로 로그인이 진행됩니다.
final LoginResult result = await FacebookAuth.instance.login();

 

  • LoginResult 를 통해 로그인 결과와 토큰을 받을 수 있습니다.
LoginResult({
  required this.status, // LoginStatus
  this.message,
  this.accessToken, // AccessToken
})

enum LoginStatus { success, cancelled, failed, operationInProgress }

AccessToken({
  required this.declinedPermissions,
  required this.grantedPermissions,
  required this.userId,
  required this.expires,
  required this.lastRefresh,
  required this.token,
  required this.applicationId,
  this.graphDomain,
  required this.isExpired,
})

 

login_platform.dart

  • 로그인을 구현할 플랫폼들을 enum 으로 한개 만들어 둡니다.

 

enum LoginPlatform {
  facebook,
  google,
  kakao,
  naver,
  apple,
  none, // logout
}

 

sample_screen.dart

  • 로고 버튼을 눌러 로그인을 실행하고 로그아웃 버튼으로 로그아웃을 처리하는 화면입니다.
  • StatefulWidget 으로 만들고 로그인 상태를 저장할 bool 타입의 변수를 선언합니다.
  • 현재 로그인한 플랫폼을 저장할 변수를 선언합니다. 

 

bool _loginStatus = false;
LoginPlatform _loginPlatform = LoginPlatform.none;

 

 

로그인 처리 후 유저정보 확인

  • 예제에서는 유저정보 확인을 http 패키지로 진행하였습니다.

 

 

http | Dart Package

A composable, multi-platform, Future-based API for HTTP requests.

pub.dev

 

 

  • FacebookAuth.instance.login() 을 통하여 로그인을 진행합니다. 
  • 로그인을 성공했다면 accessToken 으로 token 값을 받아올 수 있습니다.
  • 이 토큰 값으로 유저정보를 확인하는 요청을 보내면 이름과 이메일 정보를 얻을 수 있습니다.
  • 로그인 상태값을 갱신시켜줍니다.

 

  void signInWithFacebook() async {
    final LoginResult result = await FacebookAuth.instance.login();

    if (result.status == LoginStatus.success) {
      final url = Uri.https('graph.facebook.com', '/v2.12/me',
          {'fields': 'email, name', 'access_token': result.accessToken!.token});

      final response = await http.get(url);

      final profileInfo = json.decode(response.body);
      print(profileInfo.toString());
      
      setState(() {
        _loginPlatform = LoginPlatform.facebook;
        _loginStatus = true;
      });
    }
  }

 

 

로그아웃 처리

  • FacebookAuth.instance.logOut() 을 통하여 처리합니다.
  • 로그인 상태값을 false, 현재 로그인 플랫폼을 none 으로 갱신 시켜줍니다.

 

void signOut() async {
    switch (_loginPlatform) {
      case LoginPlatform.facebook:
        await FacebookAuth.instance.logOut();
        break;
      case LoginPlatform.google:
        break;
      case LoginPlatform.kakao:
        break;
      case LoginPlatform.naver:
        break;
      case LoginPlatform.apple:
        break;
      case LoginPlatform.none:
        break;
    }

    setState(() {
      _loginPlatform = LoginPlatform.none;
      _loginStatus = false;
    });
  }

 

 

로그인 버튼 

  • onPressed 나 onTap 이 있는 아무 위젯이나 사용해도 되지만 그렇게하면 기분이 나지 않으니 버튼을 구현해봅시다.
  • 페이스북 로고는 아래 링크에서 다운받으시면 됩니다.
 

Logo | Facebook app | Brand Portal

Audience-Specific Use Advertisers If you use Facebook brand assets in TV, digital advertising or print packaging, you must submit your creative to the Facebook team for review, prior to launch. If you're a Facebook partner or working on a Facebook sponsors

www.facebook.com

 

 

  • 프로젝트에서 asset 폴더와 그 안에 image 폴더를 만들고 다운받은 로고를 넣어줍니다.

 

  • pubspec.yaml 에 assets 를 추가해주세요.
flutter:
  assets:
    - asset/image/

 

 

  • 버튼에 그림자 효과를 주기위해 Card 위젯을 사용했습니다.
  • 버튼에 ripple 효과를 주기위해 Card 위젯 자식으로 Ink 위젯을 사용했습니다.
  • 그리고 onTap 사용을 위해 InkWell 위젯을 사용했습니다.
  • 이미지 경로는 path 부분을 빼고 모두 공통으로 처리할 것이기 때문에 이미지의 이름만 path 로 받습니다.
  • VoidCallback 을 통해 로그인 처리를 하는 함수를 받습니다.

 

  Widget _loginButton(String path, VoidCallback onTap) {
    return Card(
      elevation: 18.0,
      shape: const CircleBorder(),
      clipBehavior: Clip.antiAlias,
      child: Ink.image(
        image: AssetImage('asset/image/$path.png'),
        width: 60,
        height: 60,
        child: InkWell(
          borderRadius: const BorderRadius.all(
            Radius.circular(35.0),
          ),
          onTap: onTap,
        ),
      ),
    );
  }

 

 

로그아웃 버튼

  Widget _logoutButton() {
    return ElevatedButton(
      onPressed: signOut,
      style: ButtonStyle(
        backgroundColor: MaterialStateProperty.all(
          const Color(0xff0165E1),
        ),
      ),
      child: const Text('로그아웃'),
    );
  }

 

 

전체 코드

sample_screen.dart

import 'dart:convert';

import 'package:flutter/material.dart';
import 'package:flutter_facebook_auth/flutter_facebook_auth.dart';
import 'package:google_sign_in/google_sign_in.dart';
import 'package:http/http.dart' as http;
import 'package:social_login_app/login_platform.dart';


class SampleScreen extends StatefulWidget {
  const SampleScreen({Key? key}) : super(key: key);

  @override
  State<SampleScreen> createState() => _SampleScreenState();
}

class _SampleScreenState extends State<SampleScreen> {
  bool _loginStatus = false;
  LoginPlatform _loginPlatform = LoginPlatform.none;

  void signInWithFacebook() async {
    final LoginResult result = await FacebookAuth.instance.login();

    if (result.status == LoginStatus.success) {
      final url = Uri.https('graph.facebook.com', '/v2.12/me', {
        'fields': 'id, email, name',
        'access_token': result.accessToken!.token
      });

      final response = await http.get(url);

      final profileInfo = json.decode(response.body);
      print(profileInfo.toString());
      setState(() {
        _loginPlatform = LoginPlatform.facebook;
        _loginStatus = true;
      });
    } else {}
  }

  void signOut() async {
    switch (_loginPlatform) {
      case LoginPlatform.facebook:
        await FacebookAuth.instance.logOut();
        break;
      case LoginPlatform.google:
        break;
      case LoginPlatform.kakao:
        break;
      case LoginPlatform.naver:
        break;
      case LoginPlatform.apple:
        break;
      case LoginPlatform.none:
        break;
    }

    setState(() {
      _loginPlatform = LoginPlatform.none;
      _loginStatus = false;
    });
  }

  @override
  Widget build(BuildContext context) {
    return Scaffold(
      body: Center(
          child: _loginStatus
              ? _logoutButton()
              : Row(
                  mainAxisAlignment: MainAxisAlignment.center,
                  children: [
                    _loginButton(
                      'facebook_logo',
                      signInWithFacebook,
                    ),
                  ],
                )),
    );
  }

  Widget _loginButton(String path, VoidCallback onTap) {
    return Card(
      elevation: 18.0,
      shape: const CircleBorder(),
      clipBehavior: Clip.antiAlias,
      child: Ink.image(
        image: AssetImage('asset/image/$path.png'),
        width: 60,
        height: 60,
        child: InkWell(
          borderRadius: const BorderRadius.all(
            Radius.circular(35.0),
          ),
          onTap: onTap,
        ),
      ),
    );
  }

  Widget _logoutButton() {
    return ElevatedButton(
      onPressed: signOut,
      style: ButtonStyle(
        backgroundColor: MaterialStateProperty.all(
          const Color(0xff0165E1),
        ),
      ),
      child: const Text('로그아웃'),
    );
  }
}

 

login_platform.dart

enum LoginPlatform {
  facebook,
  google,
  kakao,
  naver,
  apple,
  none,
}

 

 

 

페이스북 로그인 뿐만 아니라 다른 소셜 로그인 구현도 포스팅 해놓았습니다!! 

 

코딩전사의 기록

 

dalgoodori.tistory.com

 

728x90
반응형