diff --git a/samples/SmokeTest/Source/Assets/SmokeTest/MainGui.cs b/samples/SmokeTest/Source/Assets/SmokeTest/MainGui.cs index 0436f20d9..5a47872fb 100644 --- a/samples/SmokeTest/Source/Assets/SmokeTest/MainGui.cs +++ b/samples/SmokeTest/Source/Assets/SmokeTest/MainGui.cs @@ -92,7 +92,8 @@ public enum Ui Leaderboards, Video, UserInfo, - PopupGravity + PopupGravity, + Permissions } public void Start() @@ -351,7 +352,11 @@ internal void ShowRegularUi() { SetUI(Ui.PopupGravity); } - else if (GUI.Button(this.CalcGrid(1, 5), "Sign Out")) + else if (GUI.Button(this.CalcGrid(1, 5), "Permissions")) + { + SetUI(Ui.Permissions); + } + else if (GUI.Button(this.CalcGrid(0, 6), "Sign Out")) { this.DoSignOut(); } @@ -936,6 +941,29 @@ private void ShowPopupGravityUi() } } + private void ShowPermissionsUi() + { + DrawStatus(); + DrawTitle("Permissions Ui"); + + if (GUI.Button(CalcGrid(0, 1), "Has Permission - Email")) + { + Status = "Email Permission " + PlayGamesPlatform.Instance.HasPermission("email"); + } + else if (GUI.Button(CalcGrid(1, 1), "Request Permission- Email")) + { + Status = "Asking permission for email"; + PlayGamesPlatform.Instance.RequestPermission( + code => { Status = "Result code " + code; }, + "email"); + } + else if (GUI.Button(CalcGrid(1, 6), "Back")) + { + SetUI(Ui.Main); + ShowEffect(true); + } + } + internal void ShowUserInfoUi() { GUI.Label( @@ -1117,6 +1145,9 @@ internal void OnGUI() case Ui.PopupGravity: ShowPopupGravityUi(); break; + case Ui.Permissions: + ShowPermissionsUi(); + break; default: // check for a status of interest, and if there // is one, then don't touch it. Otherwise diff --git a/source/PluginDev/Assets/GooglePlayGames/BasicApi/DummyClient.cs b/source/PluginDev/Assets/GooglePlayGames/BasicApi/DummyClient.cs index c946e229e..6a79a0abb 100644 --- a/source/PluginDev/Assets/GooglePlayGames/BasicApi/DummyClient.cs +++ b/source/PluginDev/Assets/GooglePlayGames/BasicApi/DummyClient.cs @@ -373,6 +373,28 @@ public void SubmitScore( } } + /// Asks user to give permissions for the given scopes. + /// Callback used to indicate the outcome of the operation. + /// Scope to ask permission for + public void RequestPermissions(Action callback, string[] scopes) + { + LogUsage(); + if (callback != null) + { + callback.Invoke(SignInStatus.Failed); + } + } + + /// Returns whether or not user has given permissions for given scopes. + /// + /// array of scopes + /// true, if given, false otherwise. + public bool HasPermissions(string[] scopes) + { + LogUsage(); + return false; + } + /// /// Returns a real-time multiplayer client. /// diff --git a/source/PluginDev/Assets/GooglePlayGames/BasicApi/IPlayGamesClient.cs b/source/PluginDev/Assets/GooglePlayGames/BasicApi/IPlayGamesClient.cs index f0e3a0918..09f038063 100644 --- a/source/PluginDev/Assets/GooglePlayGames/BasicApi/IPlayGamesClient.cs +++ b/source/PluginDev/Assets/GooglePlayGames/BasicApi/IPlayGamesClient.cs @@ -304,6 +304,21 @@ void SubmitScore(string leaderboardId, long score, void SubmitScore(string leaderboardId, long score, string metadata, Action successOrFailureCalllback); + /// + /// Asks user to give permissions for the given scopes. + /// + /// Callback used to indicate the outcome of the operation. + /// list of scopes to ask permission for + void RequestPermissions(Action callback, string[] scopes); + + /// + /// Returns whether or not user has given permissions for given scopes + /// + /// + /// list of scopes + /// true, if given, false otherwise. + bool HasPermissions(string[] scopes); + /// /// Returns a real-time multiplayer client. /// diff --git a/source/PluginDev/Assets/GooglePlayGames/BasicApi/SignInStatus.cs b/source/PluginDev/Assets/GooglePlayGames/BasicApi/SignInStatus.cs new file mode 100644 index 000000000..22a3535ca --- /dev/null +++ b/source/PluginDev/Assets/GooglePlayGames/BasicApi/SignInStatus.cs @@ -0,0 +1,42 @@ +namespace GooglePlayGames.BasicApi +{ + public enum SignInStatus + { + /// The operation was successful. + Success, + + /// + /// The client attempted to connect to the service but the user is not signed in. The client may + /// choose to continue without using the API. Alternately, if {@link Status#hasResolution} returns + /// {@literal true} the client may call {@link Status#startResolutionForResult(Activity, int)} to + /// prompt the user to sign in. After the sign in activity returns with {@link Activity#RESULT_OK} + /// further attempts should succeed. + /// + UiSignInRequired, + + /// A network error occurred. Retrying should resolve the problem. + NetworkError, + + /// An internal error occurred. + InternalError, + + /// The sign in was canceled. + Canceled, + + /// + /// A sign in process is currently in progress and the current one cannot continue. e.g. the user + /// clicks the SignInButton multiple times and more than one sign in intent was launched. + /// + AlreadyInProgress, + + /// + /// Failure reason is unknown. Check adb log to see details if any. + /// + Failed, + + /// + /// Currently not authenticated. Silent or interactive sign in is required. + /// + NotAuthenticated, + } +} diff --git a/source/PluginDev/Assets/GooglePlayGames/BasicApi/SignInStatus.cs.meta b/source/PluginDev/Assets/GooglePlayGames/BasicApi/SignInStatus.cs.meta new file mode 100644 index 000000000..39f52979a --- /dev/null +++ b/source/PluginDev/Assets/GooglePlayGames/BasicApi/SignInStatus.cs.meta @@ -0,0 +1,3 @@ +fileFormatVersion: 2 +guid: 0992bc2597d741e59dc3f8c963a3ca25 +timeCreated: 1574256557 \ No newline at end of file diff --git a/source/PluginDev/Assets/GooglePlayGames/ISocialPlatform/PlayGamesPlatform.cs b/source/PluginDev/Assets/GooglePlayGames/ISocialPlatform/PlayGamesPlatform.cs index 8a15bd5cb..089ff67e4 100644 --- a/source/PluginDev/Assets/GooglePlayGames/ISocialPlatform/PlayGamesPlatform.cs +++ b/source/PluginDev/Assets/GooglePlayGames/ISocialPlatform/PlayGamesPlatform.cs @@ -1280,6 +1280,53 @@ public void LoadScores(ILeaderboard board, Action callback) (PlayGamesLeaderboard) board, scoreData, callback)); } + /// Asks user to give permissions for the given scopes. + /// Callback used to indicate the outcome of the operation. + /// Scope to ask permission for + public void RequestPermission(Action callback, string scope) + { + RequestPermissions(callback, new string[] {scope}); + } + + /// Asks user to give permissions for the given scopes. + /// Callback used to indicate the outcome of the operation. + /// List of scopes to ask permission for + public void RequestPermissions(Action callback, string[] scopes) + { + if (!IsAuthenticated()) + { + GooglePlayGames.OurUtils.Logger.e( + "HasPermissions can only be called after authentication."); + callback(SignInStatus.NotAuthenticated); + return; + } + + mClient.RequestPermissions(callback, scopes); + } + + /// Returns whether or not user has given permissions for given scopes. + /// scope + /// true, if given, false otherwise. + public bool HasPermission(string scope) + { + return HasPermissions(new string[] {scope}); + } + + /// Returns whether or not user has given permissions for given scopes. + /// array of scopes + /// true, if given, false otherwise. + public bool HasPermissions(string[] scopes) + { + if (!IsAuthenticated()) + { + GooglePlayGames.OurUtils.Logger.e( + "HasPermissions can only be called after authentication."); + return false; + } + + return mClient.HasPermissions(scopes); + } + /// /// Check if the leaderboard is currently loading. /// diff --git a/source/PluginDev/Assets/GooglePlayGames/Platforms/Android/AndroidClient.cs b/source/PluginDev/Assets/GooglePlayGames/Platforms/Android/AndroidClient.cs index 96b61000f..17a76caf3 100644 --- a/source/PluginDev/Assets/GooglePlayGames/Platforms/Android/AndroidClient.cs +++ b/source/PluginDev/Assets/GooglePlayGames/Platforms/Android/AndroidClient.cs @@ -1020,6 +1020,37 @@ public void SubmitScore(string leaderboardId, long score, string metadata, } } + public void RequestPermissions(Action callback, string[] scopes) + { + callback = AsOnGameThreadCallback(callback); + mTokenClient.RequestPermissions((code => + { + UpdateClients(); + callback(code); + }), scopes); + } + + private void UpdateClients() + { + lock (GameServicesLock) + { + var account = mTokenClient.GetAccount(); + mSavedGameClient = new AndroidSavedGameClient(account); + mEventsClient = new AndroidEventsClient(account); + mVideoClient = new AndroidVideoClient(mVideoClient.IsCaptureSupported(), account); + mRealTimeClient = new AndroidRealTimeMultiplayerClient(this, account); + mTurnBasedClient = new AndroidTurnBasedMultiplayerClient(this, account); + mTurnBasedClient.RegisterMatchDelegate(mConfiguration.MatchDelegate); + } + } + + /// Returns whether or not user has given permissions for given scopes. + /// + public bool HasPermissions(string[] scopes) + { + return mTokenClient.HasPermissions(scopes); + } + /// /// public IRealTimeMultiplayerClient GetRtmpClient() diff --git a/source/PluginDev/Assets/GooglePlayGames/Platforms/Android/AndroidTokenClient.cs b/source/PluginDev/Assets/GooglePlayGames/Platforms/Android/AndroidTokenClient.cs index f26812429..066d0c76a 100644 --- a/source/PluginDev/Assets/GooglePlayGames/Platforms/Android/AndroidTokenClient.cs +++ b/source/PluginDev/Assets/GooglePlayGames/Platforms/Android/AndroidTokenClient.cs @@ -18,6 +18,7 @@ namespace GooglePlayGames.Android { using System; + using System.Linq; using BasicApi; using OurUtils; using UnityEngine; @@ -132,6 +133,91 @@ public void FetchTokens(bool silent, Action callback) PlayGamesHelperObject.RunOnGameThread(() => DoFetchToken(silent, callback)); } + public void RequestPermissions(Action callback, string[] scopes) + { + using (var bridgeClass = new AndroidJavaClass(HelperFragmentClass)) + using (var currentActivity = AndroidHelperFragment.GetActivity()) + using (var task = + bridgeClass.CallStatic("showRequestPermissionsUi", currentActivity, + oauthScopes.Union(scopes).ToArray())) + { + AndroidTaskUtils.AddOnSuccessListener(task, /* disposeResult= */ false, + accountWithNewScopes => + { + if (accountWithNewScopes == null) + { + callback(SignInStatus.InternalError); + return; + } + + account = accountWithNewScopes; + email = account.Call("getEmail"); + idToken = account.Call("getIdToken"); + authCode = account.Call("getServerAuthCode"); + oauthScopes = oauthScopes.Union(scopes).ToList(); + callback(SignInStatus.Success); + }); + + AndroidTaskUtils.AddOnFailureListener(task, e => + { + var failCode = ToSignInStatus(e.Call("getStatusCode")); + OurUtils.Logger.e("Exception requesting new permissions: " + failCode); + callback(failCode); + }); + } + } + + private static SignInStatus ToSignInStatus(int code) + { + Dictionary dictionary = new Dictionary() + { + { + /* CommonUIStatus.UI_BUSY */ -12, SignInStatus.AlreadyInProgress + }, + { + /* CommonStatusCodes.SUCCESS */ 0, SignInStatus.Success + }, + { + /* CommonStatusCodes.SIGN_IN_REQUIRED */ 4, SignInStatus.UiSignInRequired + }, + { + /* CommonStatusCodes.NETWORK_ERROR */ 7, SignInStatus.NetworkError + }, + { + /* CommonStatusCodes.INTERNAL_ERROR */ 8, SignInStatus.InternalError + }, + { + /* CommonStatusCodes.CANCELED */ 16, SignInStatus.Canceled + }, + { + /* CommonStatusCodes.API_NOT_CONNECTED */ 17, SignInStatus.Failed + }, + { + /* GoogleSignInStatusCodes.SIGN_IN_FAILED */ 12500, SignInStatus.Failed + }, + { + /* GoogleSignInStatusCodes.SIGN_IN_CANCELLED */ 12501, SignInStatus.Canceled + }, + { + /* GoogleSignInStatusCodes.SIGN_IN_CURRENTLY_IN_PROGRESS */ 12502, SignInStatus.AlreadyInProgress + }, + }; + + return dictionary.ContainsKey(code) ? dictionary[code] : SignInStatus.Failed; + } + + /// Returns whether or not user has given permissions for given scopes. + /// array of scopes + /// true, if given, false otherwise. + public bool HasPermissions(string[] scopes) + { + using (var bridgeClass = new AndroidJavaClass(HelperFragmentClass)) + using (var currentActivity = AndroidHelperFragment.GetActivity()) + { + return bridgeClass.CallStatic("hasPermissions", currentActivity, scopes); + } + } + private void DoFetchToken(bool silent, Action callback) { try diff --git a/source/PluginDev/Assets/GooglePlayGames/Platforms/TokenClient.cs b/source/PluginDev/Assets/GooglePlayGames/Platforms/TokenClient.cs index adf2e9ce7..bb70aa0a6 100644 --- a/source/PluginDev/Assets/GooglePlayGames/Platforms/TokenClient.cs +++ b/source/PluginDev/Assets/GooglePlayGames/Platforms/TokenClient.cs @@ -17,6 +17,7 @@ #if UNITY_ANDROID namespace GooglePlayGames { + using GooglePlayGames.BasicApi; using System; internal interface TokenClient @@ -67,6 +68,10 @@ void GetAnotherServerAuthCode(bool reAuthenticateIfNeeded, void SetHidePopups(bool flag); void FetchTokens(bool silent, Action callback); + + void RequestPermissions(Action callback, string[] scopes); + + bool HasPermissions(string[] scopes); } } #endif \ No newline at end of file diff --git a/source/SupportLib/PlayGamesPluginSupport/src/main/java/com/google/games/bridge/HelperFragment.java b/source/SupportLib/PlayGamesPluginSupport/src/main/java/com/google/games/bridge/HelperFragment.java index 81ed3f1da..de15351ac 100644 --- a/source/SupportLib/PlayGamesPluginSupport/src/main/java/com/google/games/bridge/HelperFragment.java +++ b/source/SupportLib/PlayGamesPluginSupport/src/main/java/com/google/games/bridge/HelperFragment.java @@ -56,6 +56,7 @@ public class HelperFragment extends Fragment static final int RC_INBOX_UI = 9007; static final int RC_SHOW_WAITING_ROOM_UI = 9008; static final int RC_SHOW_INVITATION_INBOX_UI = 9009; + static final int RC_SHOW_REQUEST_PERMISSIONS_UI = 9010; // Pending token request. There can be only one outstanding request at a // time. @@ -204,7 +205,7 @@ public static Task showInvitationInboxUI(Ac return request.getTask(); } - public static Task showInboxUi(Activity parentActivity){ + public static Task showInboxUi(Activity parentActivity) { InboxUiRequest request = new InboxUiRequest(); if(!HelperFragment.startRequest(parentActivity, request)) { @@ -214,6 +215,28 @@ public static Task showInboxUi(Activity parentActivity){ return request.getTask(); } + public static Task showRequestPermissionsUi(Activity parentActivity, String[] scopes) { + RequestPermissionsRequest request = new RequestPermissionsRequest(toScopeList(scopes)); + + if(!HelperFragment.startRequest(parentActivity, request)) { + request.setFailure(CommonUIStatus.UI_BUSY); + } + + return request.getTask(); + } + + public static boolean hasPermissions(Activity parentActivity, String[] scopes) { + return GoogleSignIn.hasPermissions(getAccount(parentActivity), toScopeList(scopes)); + } + + private static Scope[] toScopeList(String[] scopeUris) { + Scope[] scopes = new Scope[scopeUris.length]; + for (int i = 0; i < scopeUris.length; i++) { + scopes[i] = new Scope(scopeUris[i]); + } + return scopes; + } + public static void signOut(Activity activity) { GoogleSignInClient signInClient = GoogleSignIn.getClient(activity, GoogleSignInOptions.DEFAULT_GAMES_SIGN_IN); signInClient.signOut(); diff --git a/source/SupportLib/PlayGamesPluginSupport/src/main/java/com/google/games/bridge/RequestPermissionsRequest.java b/source/SupportLib/PlayGamesPluginSupport/src/main/java/com/google/games/bridge/RequestPermissionsRequest.java new file mode 100644 index 000000000..33a8935fa --- /dev/null +++ b/source/SupportLib/PlayGamesPluginSupport/src/main/java/com/google/games/bridge/RequestPermissionsRequest.java @@ -0,0 +1,104 @@ +package com.google.games.bridge; + +import android.app.Activity; +import android.content.Intent; +import android.util.Log; + +import com.google.android.gms.auth.api.Auth; +import com.google.android.gms.auth.api.signin.GoogleSignIn; +import com.google.android.gms.auth.api.signin.GoogleSignInAccount; +import com.google.android.gms.auth.api.signin.GoogleSignInClient; +import com.google.android.gms.auth.api.signin.GoogleSignInStatusCodes; +import com.google.android.gms.auth.api.signin.GoogleSignInResult; +import com.google.android.gms.auth.api.signin.GoogleSignInOptions; +import com.google.android.gms.common.api.ApiException; +import com.google.android.gms.common.api.CommonStatusCodes; +import com.google.android.gms.common.api.Status; +import com.google.android.gms.common.api.Scope; +import com.google.android.gms.tasks.Task; +import com.google.android.gms.tasks.TaskCompletionSource; +import java.util.ArrayList; + +class RequestPermissionsRequest implements HelperFragment.Request { + private static final String TAG = "RequestPermissions"; + + private final TaskCompletionSource resultTaskSource = new TaskCompletionSource<>(); + private Scope[] scopes; + private HelperFragment helperFragment; + + public RequestPermissionsRequest(Scope[] scopes) { + this.scopes = scopes; + } + + Task getTask() { + return resultTaskSource.getTask(); + } + + public void process(HelperFragment helperFragment) { + this.helperFragment = helperFragment; + final Activity activity = helperFragment.getActivity(); + GoogleSignInAccount account = HelperFragment.getAccount(activity); + Scope[] unauthorizedScopes = getUnauthorizedScopes(account, scopes); + if (unauthorizedScopes.length == 0) { + setSuccess(GoogleSignIn.getAccountForScopes(activity, scopes[0], scopes)); + } else { + Intent intent = getSignInIntentForAccountAndScopes(activity, account, unauthorizedScopes); + helperFragment.startActivityForResult(intent, HelperFragment.RC_SHOW_REQUEST_PERMISSIONS_UI); + } + } + + private Scope[] getUnauthorizedScopes(GoogleSignInAccount account, Scope[] scopes) { + ArrayList unauthorizedScopes = new ArrayList(); + for (Scope scope : scopes) { + if (!GoogleSignIn.hasPermissions(account, scope)) { + unauthorizedScopes.add(scope); + } + } + return unauthorizedScopes.toArray(new Scope[unauthorizedScopes.size()]); + } + + private static Intent getSignInIntentForAccountAndScopes( + /* @NonNull */ Activity activity, /* @Nullable */ GoogleSignInAccount account, /* @NonNull */ Scope... scopes) { + GoogleSignInOptions.Builder optionsBuilder = new GoogleSignInOptions.Builder(); + + if (scopes.length > 0) { + optionsBuilder.requestScopes(scopes[0], scopes); + } + + if (account != null && account.getEmail() != null) { + optionsBuilder.setAccountName(account.getEmail()); + } + + return GoogleSignIn.getClient(activity, optionsBuilder.build()).getSignInIntent(); + } + + public void onActivityResult(int requestCode, int resultCode, Intent data) { + if (requestCode == HelperFragment.RC_SHOW_REQUEST_PERMISSIONS_UI) { + GoogleSignInResult result = Auth.GoogleSignInApi.getSignInResultFromIntent(data); + if (result != null && result.isSuccess()) { + GoogleSignInAccount account = GoogleSignIn.getAccountForScopes(helperFragment.getActivity(), scopes[0], scopes); + setSuccess(account); + } else if (resultCode == Activity.RESULT_CANCELED) { + if (result != null) { + setFailure(result.getStatus().getStatusCode()); + } else { + setFailure(CommonStatusCodes.CANCELED); + } + } else if (result != null) { + setFailure(result.getStatus().getStatusCode()); + } else { + setFailure(CommonStatusCodes.ERROR); + } + } + } + + void setFailure(int code) { + resultTaskSource.setException(new ApiException(new Status(code))); + HelperFragment.finishRequest(this); + } + + private void setSuccess(GoogleSignInAccount account) { + resultTaskSource.setResult(account); + HelperFragment.finishRequest(this); + } +}