Google Signin Leaderboards fixed

dev-ali
Ali Sharoz 5 days ago
parent 7282422d5e
commit 863458846e

@ -128,7 +128,7 @@ public class AchievementManager : MonoBehaviour
int coins = PlayerPrefs.GetInt("Coin");
Debug.Log(toAdd + " Coins Rewarded");
coins += toAdd;
PlayerPrefs.SetInt("Coin", coins);
SafePlayerPrefs.SetInt("Coin", coins);
StartCoroutine(startCoinShakeEffect(coins - toAdd, coins, .5f));
}
@ -169,12 +169,12 @@ public class AchievementManager : MonoBehaviour
#region Ali's Code for Reward
PlayerPrefs.SetInt("UIAchievement" + Index + "Claimable", 1);
SafePlayerPrefs.SetInt("UIAchievement" + Index + "Claimable", 1);
int numOfPrevCoins = PlayerPrefs.GetInt("TotalCoinsForReward" + Index, 0);
int subAchievementIndex = PlayerPrefs.GetInt("subAchievementIndex" + Index, 0);
PlayerPrefs.SetInt("TotalCoinsForReward" + Index, numOfPrevCoins + achievementInfromation.CoinRewards[Mathf.Clamp(subAchievementIndex,0, achievementInfromation.CoinRewards.Count-1)]);
SafePlayerPrefs.SetInt("TotalCoinsForReward" + Index, numOfPrevCoins + achievementInfromation.CoinRewards[Mathf.Clamp(subAchievementIndex,0, achievementInfromation.CoinRewards.Count-1)]);
subAchievementIndex++;
PlayerPrefs.SetInt("subAchievementIndex" + Index, subAchievementIndex);
SafePlayerPrefs.SetInt("subAchievementIndex" + Index, subAchievementIndex);
#endregion
@ -273,7 +273,7 @@ public class AchievementManager : MonoBehaviour
{
for (int i = 0; i < States.Count; i++)
{
PlayerPrefs.SetString("AchievementState_" + i, JsonUtility.ToJson(States[i]));
SafePlayerPrefs.SetString("AchievementState_" + i, JsonUtility.ToJson(States[i]));
}
PlayerPrefs.Save();
}

@ -101,7 +101,7 @@ public class AchievenmentListIngame : MonoBehaviour
{
if (PlayerPrefs.GetInt("UIAchievement" + i + "Claimable") == 1)
{
PlayerPrefs.SetInt("UIAchievement" + i + "Claimable", 0);
SafePlayerPrefs.SetInt("UIAchievement" + i + "Claimable", 0);
Stack[i].ClaimButton.gameObject.SetActive(true);
Stack[i].CompleteToClaimButton.SetActive(false);
int numofCoinsToAdd = PlayerPrefs.GetInt("TotalCoinsForReward" + i);
@ -109,7 +109,7 @@ public class AchievenmentListIngame : MonoBehaviour
Stack[i].ClaimButton.onClick.AddListener(() =>
{
AchievementManager.instance.AddCoins(numofCoinsToAdd);
PlayerPrefs.SetInt("TotalCoinsForReward" + i, 0);
SafePlayerPrefs.SetInt("TotalCoinsForReward" + i, 0);
});
}
}

@ -35,7 +35,7 @@ public class UIAchievement : MonoBehaviour
int RewardtoAdd = Information.CoinRewards[subAchievementIndex];
int TotalReward = RewardtoAdd + CurrentTotalReward;
Information.TotalCoinReward = TotalReward;
PlayerPrefs.SetInt("TotalRewards"+index, TotalReward);
SafePlayerPrefs.SetInt("TotalRewards"+index, TotalReward);
}
/// <summary>

@ -1,5 +1,5 @@
fileFormatVersion: 2
guid: a55ee4efaad27d948ba5f03fc6d7bc80
guid: c86158f937dbef3469d4bed1a9cbf3f1
DefaultImporter:
externalObjects: {}
userData:

@ -0,0 +1,8 @@
fileFormatVersion: 2
guid: 942372e0a39170840b5107589bbc8855
folderAsset: yes
DefaultImporter:
externalObjects: {}
userData:
assetBundleName:
assetBundleVariant:

@ -0,0 +1,8 @@
fileFormatVersion: 2
guid: 0349fcb1e5762334391642aca2cd6ec9
folderAsset: yes
DefaultImporter:
externalObjects: {}
userData:
assetBundleName:
assetBundleVariant:

@ -0,0 +1,8 @@
fileFormatVersion: 2
guid: d29c0cdd54ad5bf42961a942a27587b6
folderAsset: yes
DefaultImporter:
externalObjects: {}
userData:
assetBundleName:
assetBundleVariant:

@ -0,0 +1,8 @@
fileFormatVersion: 2
guid: 570dd445e1db5f043b0bed2a4b26a858
folderAsset: yes
DefaultImporter:
externalObjects: {}
userData:
assetBundleName:
assetBundleVariant:

@ -0,0 +1,8 @@
fileFormatVersion: 2
guid: 81ba2e5a7b10b804b9b3f8f44cd08d65
folderAsset: yes
DefaultImporter:
externalObjects: {}
userData:
assetBundleName:
assetBundleVariant:

@ -0,0 +1,8 @@
fileFormatVersion: 2
guid: 4b12ef651963e1647bcf34b06d452359
folderAsset: yes
DefaultImporter:
externalObjects: {}
userData:
assetBundleName:
assetBundleVariant:

@ -0,0 +1,8 @@
fileFormatVersion: 2
guid: 3fbcb798450a9da458bd1ff200efc8e4
folderAsset: yes
DefaultImporter:
externalObjects: {}
userData:
assetBundleName:
assetBundleVariant:

@ -0,0 +1,8 @@
fileFormatVersion: 2
guid: 546d95353e1ea6741b19368cca1ccb30
folderAsset: yes
DefaultImporter:
externalObjects: {}
userData:
assetBundleName:
assetBundleVariant:

@ -0,0 +1,8 @@
fileFormatVersion: 2
guid: d3a26cbe3cb25014bbe2889e4c379bcc
folderAsset: yes
DefaultImporter:
externalObjects: {}
userData:
assetBundleName:
assetBundleVariant:

@ -0,0 +1,29 @@
fileFormatVersion: 2
guid: ed2ae7de4097e26439f6aef5df2ce991
labels:
- gpsr
PluginImporter:
externalObjects: {}
serializedVersion: 2
iconMap: {}
executionOrder: {}
defineConstraints: []
isPreloaded: 0
isOverridable: 0
isExplicitlyReferenced: 0
validateReferences: 1
platformData:
- first:
Any:
second:
enabled: 0
settings: {}
- first:
Editor: Editor
second:
enabled: 1
settings:
DefaultValueInitialized: true
userData:
assetBundleName:
assetBundleVariant:

@ -0,0 +1,8 @@
<?xml version="1.0" encoding="utf-8"?>
<project xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd" xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance">
<modelVersion>4.0.0</modelVersion>
<groupId>com.google.signin</groupId>
<artifactId>google-signin-support</artifactId>
<version>1.0.4</version>
<packaging>aar</packaging>
</project>

@ -0,0 +1,9 @@
fileFormatVersion: 2
guid: 1847715e547531b4d91b87619689c0f1
labels:
- gpsr
DefaultImporter:
externalObjects: {}
userData:
assetBundleName:
assetBundleVariant:

@ -0,0 +1,9 @@
fileFormatVersion: 2
guid: a089c79cc334748a5a29c5f219b7cfd4
folderAsset: yes
timeCreated: 1490807626
licenseType: Pro
DefaultImporter:
userData:
assetBundleName:
assetBundleVariant:

@ -0,0 +1,9 @@
fileFormatVersion: 2
guid: a7e77ece57ef34772969fad4915b1a4c
folderAsset: yes
timeCreated: 1490978071
licenseType: Pro
DefaultImporter:
userData:
assetBundleName:
assetBundleVariant:

@ -0,0 +1,19 @@
<?xml version="1.0" encoding="UTF-8" ?>
<dependencies>
<!-- See https://github.com/googlesamples/unity-jar-resolver#usage for
how to configure dependencies -->
<androidPackages>
<androidPackage spec="com.google.android.gms:play-services-auth:16+">
<androidSdkPackageIds>
<androidSdkPackageId>extra-google-m2repository</androidSdkPackageId>
</androidSdkPackageIds>
</androidPackage>
</androidPackages>
<!-- iOS Cocoapod dependencies can be specified by each iosPod element. -->
<iosPods>
<iosPod name="GoogleSignIn" version=">= 4.0.2" bitcodeEnabled="false"
minTargetSdk="6.0">
</iosPod>
</iosPods>
</dependencies>

@ -0,0 +1,10 @@
fileFormatVersion: 2
guid: 938014c9994164100b26d82840a88fbb
labels:
- gvh
- gvh_version-1.0.4
TextScriptImporter:
externalObjects: {}
userData:
assetBundleName:
assetBundleVariant:

@ -0,0 +1,11 @@
<?xml version="1.0" encoding="UTF-8" ?>
<dependencies>
<!-- Internal library dependency generated at build time. -->
<androidPackages>
<androidPackage spec="com.google.signin:google-signin-support:1.0.4">
<repositories>
<repository>Assets/GoogleSignIn/Editor/m2repository</repository>
</repositories>
</androidPackage>
</androidPackages>
</dependencies>

@ -0,0 +1,10 @@
fileFormatVersion: 2
guid: e8689106c65f445c19cec6044d615c19
labels:
- gvh
- gvh_version-1.0.4
TextScriptImporter:
externalObjects: {}
userData:
assetBundleName:
assetBundleVariant:

@ -0,0 +1,33 @@
Assets/PlayServicesResolver/Editor/Google.VersionHandlerImpl_v1.2.89.0.dll
Assets/PlayServicesResolver/Editor/Google.IOSResolver_v1.2.89.0.dll
Assets/PlayServicesResolver/Editor/Google.VersionHandler.dll
Assets/PlayServicesResolver/Editor/Google.JarResolver_v1.2.89.0.dll
Assets/Plugins/iOS/GoogleSignIn/GoogleSignInAppController.mm
Assets/Plugins/iOS/GoogleSignIn/GoogleSignInAppController.h
Assets/Plugins/iOS/GoogleSignIn/GoogleSignIn.h
Assets/Plugins/iOS/GoogleSignIn/GoogleSignIn.mm
Assets/Parse/LICENSE
Assets/Parse/Plugins/Unity.Compat.dll
Assets/Parse/Plugins/Unity.Tasks.dll
Assets/SignInSample/MainScene.unity
Assets/SignInSample/SigninSampleScript.cs
Assets/GoogleSignIn/Impl/GoogleSignInImpl.cs
Assets/GoogleSignIn/Impl/SignInHelperObject.cs
Assets/GoogleSignIn/Impl/NativeFuture.cs
Assets/GoogleSignIn/Impl/BaseObject.cs
Assets/GoogleSignIn/GoogleSignIn.cs
Assets/GoogleSignIn/GoogleSignInConfiguration.cs
Assets/GoogleSignIn/Future.cs
Assets/GoogleSignIn/GoogleSignInUser.cs
Assets/GoogleSignIn/GoogleSignInStatusCode.cs
Assets/GoogleSignIn/Editor/GoogleSignInDependencies.xml
Assets/GoogleSignIn/Editor/GoogleSignInSupportDependencies.xml
Assets/GoogleSignIn/Editor/m2repository/com/google/signin/google-signin-support/maven-metadata.xml
Assets/GoogleSignIn/Editor/m2repository/com/google/signin/google-signin-support/maven-metadata.xml.md5
Assets/GoogleSignIn/Editor/m2repository/com/google/signin/google-signin-support/maven-metadata.xml.sha1
Assets/GoogleSignIn/Editor/m2repository/com/google/signin/google-signin-support/1.0.4/google-signin-support-1.0.4.pom.md5
Assets/GoogleSignIn/Editor/m2repository/com/google/signin/google-signin-support/1.0.4/google-signin-support-1.0.4.srcaar.sha1
Assets/GoogleSignIn/Editor/m2repository/com/google/signin/google-signin-support/1.0.4/google-signin-support-1.0.4.pom.sha1
Assets/GoogleSignIn/Editor/m2repository/com/google/signin/google-signin-support/1.0.4/google-signin-support-1.0.4.srcaar
Assets/GoogleSignIn/Editor/m2repository/com/google/signin/google-signin-support/1.0.4/google-signin-support-1.0.4.pom
Assets/GoogleSignIn/Editor/m2repository/com/google/signin/google-signin-support/1.0.4/google-signin-support-1.0.4.srcaar.md5

@ -0,0 +1,11 @@
fileFormatVersion: 2
guid: a67d5f0cfc09749e6b1ce13e8b8e6e1d
labels:
- gvh
- gvh_manifest
- gvh_version-1.0.4
TextScriptImporter:
externalObjects: {}
userData:
assetBundleName:
assetBundleVariant:

@ -0,0 +1,10 @@
fileFormatVersion: 2
guid: eefc3dc8b56e545998952bd59ab36247
folderAsset: yes
timeCreated: 1508956004
licenseType: Free
DefaultImporter:
externalObjects: {}
userData:
assetBundleName:
assetBundleVariant:

@ -0,0 +1,10 @@
fileFormatVersion: 2
guid: ade49ef91c70440a0baeac322ecaa2d7
folderAsset: yes
timeCreated: 1508956004
licenseType: Free
DefaultImporter:
externalObjects: {}
userData:
assetBundleName:
assetBundleVariant:

@ -0,0 +1,10 @@
fileFormatVersion: 2
guid: 707de6ade212a46b1ab4bf4d42325176
folderAsset: yes
timeCreated: 1508956004
licenseType: Free
DefaultImporter:
externalObjects: {}
userData:
assetBundleName:
assetBundleVariant:

@ -0,0 +1,10 @@
fileFormatVersion: 2
guid: 34b86fc2e934d482ea5f9d1f5354b0f9
folderAsset: yes
timeCreated: 1508956004
licenseType: Free
DefaultImporter:
externalObjects: {}
userData:
assetBundleName:
assetBundleVariant:

@ -0,0 +1,10 @@
fileFormatVersion: 2
guid: e3becceb8680148d59e5bb93e5eb5a24
folderAsset: yes
timeCreated: 1508956004
licenseType: Free
DefaultImporter:
externalObjects: {}
userData:
assetBundleName:
assetBundleVariant:

@ -0,0 +1,10 @@
fileFormatVersion: 2
guid: 001a15402063b4df983ba8bc4ddb269f
folderAsset: yes
timeCreated: 1537405253
licenseType: Pro
DefaultImporter:
externalObjects: {}
userData:
assetBundleName:
assetBundleVariant:

@ -0,0 +1,9 @@
<?xml version="1.0" encoding="UTF-8"?>
<project xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd" xmlns="http://maven.apache.org/POM/4.0.0"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance">
<modelVersion>4.0.0</modelVersion>
<groupId>com.google.signin</groupId>
<artifactId>google-signin-support</artifactId>
<version>1.0.4</version>
<packaging>srcaar</packaging>
</project>

@ -0,0 +1,10 @@
fileFormatVersion: 2
guid: 907bf7a19cd7848cabf1f641f61343ee
labels:
- gvh
- gvh_version-1.0.4
DefaultImporter:
externalObjects: {}
userData:
assetBundleName:
assetBundleVariant:

@ -0,0 +1,10 @@
fileFormatVersion: 2
guid: a3f74b3d329a24e81a6c6f9f1d2f46f1
labels:
- gvh
- gvh_version-1.0.4
DefaultImporter:
externalObjects: {}
userData:
assetBundleName:
assetBundleVariant:

@ -0,0 +1,10 @@
fileFormatVersion: 2
guid: e1a3ff77f802d43818521ff3db2bf944
labels:
- gvh
- gvh_version-1.0.4
DefaultImporter:
externalObjects: {}
userData:
assetBundleName:
assetBundleVariant:

@ -0,0 +1,10 @@
fileFormatVersion: 2
guid: 79f9d122dd99b483282294f3b1f1bd36
labels:
- gvh
- gvh_version-1.0.4
DefaultImporter:
externalObjects: {}
userData:
assetBundleName:
assetBundleVariant:

@ -0,0 +1,10 @@
fileFormatVersion: 2
guid: 6eab7a3cbdbf34cffaf951dc9210a32c
labels:
- gvh
- gvh_version-1.0.4
DefaultImporter:
externalObjects: {}
userData:
assetBundleName:
assetBundleVariant:

@ -0,0 +1,10 @@
fileFormatVersion: 2
guid: d508eb0e16bb14c8a962b343e755ef01
labels:
- gvh
- gvh_version-1.0.4
DefaultImporter:
externalObjects: {}
userData:
assetBundleName:
assetBundleVariant:

@ -0,0 +1,12 @@
<?xml version="1.0" encoding="UTF-8"?>
<metadata>
<groupId>com.google.signin</groupId>
<artifactId>google-signin-support</artifactId>
<versioning>
<release>1.0.4</release>
<versions>
<version>1.0.4</version>
</versions>
<lastUpdated>20180920010048</lastUpdated>
</versioning>
</metadata>

@ -0,0 +1,10 @@
fileFormatVersion: 2
guid: c020582c1a85d47c7934908f00a0bd37
labels:
- gvh
- gvh_version-1.0.4
DefaultImporter:
externalObjects: {}
userData:
assetBundleName:
assetBundleVariant:

@ -0,0 +1,10 @@
fileFormatVersion: 2
guid: f3911d467587242978192a8723c68395
labels:
- gvh
- gvh_version-1.0.4
TextScriptImporter:
externalObjects: {}
userData:
assetBundleName:
assetBundleVariant:

@ -0,0 +1,10 @@
fileFormatVersion: 2
guid: 16b8065a143bb4277af08381a56157dd
labels:
- gvh
- gvh_version-1.0.4
DefaultImporter:
externalObjects: {}
userData:
assetBundleName:
assetBundleVariant:

@ -0,0 +1,84 @@
// <copyright file="Future.cs" company="Google Inc.">
// Copyright (C) 2017 Google Inc. All Rights Reserved.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
// </copyright>
namespace Google {
using System.Collections;
using System.Threading.Tasks;
using UnityEngine;
/// <summary>
/// Interface for implementations of the Future<T> API.
/// </summary>
internal interface FutureAPIImpl<T> {
bool Pending { get; }
GoogleSignInStatusCode Status { get; }
T Result { get; }
}
/// <summary>
/// Future return value.
/// </summary>
/// <remarks>This class provides a promise of a result from a method call.
/// The typical usage is to check the Pending property until it is false.
/// At this time either the Status or Result will be available for use.
/// Result is only set if the operation was successful.
/// As a convience, a coroutine to complete a Task is provided.
/// </remarks>
public class Future<T> {
private FutureAPIImpl<T> apiImpl;
internal Future(FutureAPIImpl<T> impl) {
apiImpl = impl;
}
/// <summary>
/// Gets a value indicating whether this
/// <see cref="T:Google.Future`1"/> is pending.
/// </summary>
/// <value><c>true</c> if pending; otherwise, <c>false</c>.</value>
public bool Pending { get { return apiImpl.Pending; } }
/// <summary>
/// Gets the status.
/// </summary>
/// <value>The status is set when Pending == false.</value>
GoogleSignInStatusCode Status { get { return apiImpl.Status; } }
/// <summary>
/// Gets the result.
/// </summary>
/// <value>The result is set when Pending == false and there is no error.
/// </value>
T Result { get { return apiImpl.Result; } }
/// <summary>
/// Waits for result then completes the TaskCompleationSource.
/// </summary>
/// <returns>The for result.</returns>
/// <param name="tcs">Tcs.</param>
internal IEnumerator WaitForResult(TaskCompletionSource<T> tcs) {
yield return new WaitUntil(() => !Pending);
if (Status == GoogleSignInStatusCode.Canceled) {
tcs.SetCanceled();
} else if (Status == GoogleSignInStatusCode.Success ||
Status == GoogleSignInStatusCode.SuccessCached) {
tcs.SetResult(Result);
} else {
tcs.SetException(new GoogleSignIn.SignInException(Status));
}
}
}
}

@ -0,0 +1,14 @@
fileFormatVersion: 2
guid: ad3b09fb652fb4ff0a68d1966f13160e
labels:
- gvh
- gvh_version-1.0.4
MonoImporter:
externalObjects: {}
serializedVersion: 2
defaultReferences: []
executionOrder: 0
icon: {instanceID: 0}
userData:
assetBundleName:
assetBundleVariant:

@ -0,0 +1,202 @@
// <copyright file="GoogleSignIn.cs" company="Google Inc.">
// Copyright (C) 2017 Google Inc. All Rights Reserved.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
// </copyright>
namespace Google {
using System;
using System.Runtime.Serialization;
using System.Threading.Tasks;
using Google.Impl;
using UnityEngine;
/// <summary>
/// Google sign in API.
/// </summary>
/// <remarks>This class implements the GoogleSignInAPI for Unity.
/// Typical usage is to set the Configuration options as needed, then
/// get the DefaultInstance and call signIn or signInSilently. See
/// the <a href="https://developers.google.com/identity">
/// Google Sign-In API documentation for more details.</a>
/// <para>
/// <code>
/// private static readonly GoogleSignInConfiguration configuration =
/// new GoogleSignInConfiguration {
/// WebClientId = "<your client id here >",
/// RequestIdToken = true
/// };
///
/// public void OnSignIn() {
/// GoogleSignIn.Configuration = configuration;
/// GoogleSignIn.Configuration.UseGameSignIn = false;
/// GoogleSignIn.Configuration.RequestIdToken = true;
/// GoogleSignIn.DefaultInstance.SignIn().ContinueWith(
/// OnAuthenticationFinished);
/// }
/// </code>
/// </para>
/// </remarks>
public class GoogleSignIn {
#if !UNITY_ANDROID && !UNITY_IOS
static GoogleSignIn() {
Debug.LogError("This platform is not supported");
}
#endif
private static GoogleSignIn theInstance = null;
private static GoogleSignInConfiguration theConfiguration = null;
private ISignInImpl impl;
///<summary> The configuration settings for Google Sign-in.</summary>
///<remarks> The configuration should be set before calling the sign-in
/// methods. Once the configuration is set it cannot be changed.
///</remarks>
public static GoogleSignInConfiguration Configuration {
set {
// Can set the configuration until the singleton is created.
if (theInstance == null || theConfiguration == value || theConfiguration == null) {
theConfiguration = value;
} else {
throw new SignInException(GoogleSignInStatusCode.DeveloperError,
"DefaultInstance already created. " +
" Cannot change configuration after creation.");
}
}
get {
return theConfiguration;
}
}
/// <summary>
/// Singleton instance of this class.
/// </summary>
/// <value>The instance.</value>
public static GoogleSignIn DefaultInstance {
get {
if (theInstance == null) {
#if UNITY_ANDROID || UNITY_IOS
theInstance = new GoogleSignIn(new GoogleSignInImpl(Configuration));
#else
theInstance = new GoogleSignIn(null);
throw new SignInException(
GoogleSignInStatusCode.DeveloperError,
"This platform is not supported by GoogleSignIn");
#endif
}
return theInstance;
}
}
internal GoogleSignIn(GoogleSignInImpl impl) {
this.impl = impl;
}
public void EnableDebugLogging(bool flag) {
impl.EnableDebugLogging(flag);
}
/// <summary>Starts the authentication process.</summary>
/// <remarks>
/// The authenication process is started and may display account picker
/// popups and consent prompts based on the state of authentication and
/// the requested elements.
/// </remarks>
public Task<GoogleSignInUser> SignIn() {
var tcs = new TaskCompletionSource<GoogleSignInUser>();
SignInHelperObject.Instance.StartCoroutine(
impl.SignIn().WaitForResult(tcs));
return tcs.Task;
}
/// <summary>Starts the silent authentication process.</summary>
/// <remarks>
/// The authenication process is started and will attempt to sign in without
/// displaying any UI. If this cannot be done, the developer should call
/// SignIn().
/// </remarks>
public Task<GoogleSignInUser> SignInSilently() {
var tcs = new TaskCompletionSource<GoogleSignInUser>();
SignInHelperObject.Instance.StartCoroutine(
impl.SignInSilently().WaitForResult(tcs));
return tcs.Task;
}
/// <summary>
/// Signs out the User.
/// </summary>
/// <remarks>Future sign-in attempts will require the user to select the
/// account to use when signing in.
/// </remarks>
public void SignOut() {
theConfiguration = null;
impl.SignOut();
}
/// <summary>
/// Disconnect this instance.
/// </summary>
/// <remarks>When the user is disconnected, it revokes all access that may
/// have been granted to this application. This includes any server side
/// access tokens derived from server auth codes. As a result, future
/// sign-in attempts will require the user to re-consent to the requested
/// scopes.
/// </remarks>
public void Disconnect() {
impl.Disconnect();
}
/// <summary>
/// Sign in exception. This is a checked exception for handling specific
/// errors during the sign-in process.
/// </summary>
[Serializable]
public class SignInException : Exception {
internal SignInException(GoogleSignInStatusCode status) {
Status = status;
}
public SignInException(GoogleSignInStatusCode status, string message) :
base(message) {
Status = status;
}
public SignInException(GoogleSignInStatusCode status, string message,
Exception innerException) : base(message, innerException) {
Status = status;
}
protected SignInException(GoogleSignInStatusCode status,
SerializationInfo info,
StreamingContext context) :
base(info, context) {
Status = status;
}
public GoogleSignInStatusCode Status {
get;
internal set;
}
}
}
internal interface ISignInImpl {
Future<GoogleSignInUser> SignIn();
Future<GoogleSignInUser> SignInSilently();
void EnableDebugLogging(bool flag);
void SignOut();
void Disconnect();
}
} // namespace Google

@ -0,0 +1,14 @@
fileFormatVersion: 2
guid: 138b984208e394be797ce8905a44fd54
labels:
- gvh
- gvh_version-1.0.4
MonoImporter:
externalObjects: {}
serializedVersion: 2
defaultReferences: []
executionOrder: 0
icon: {instanceID: 0}
userData:
assetBundleName:
assetBundleVariant:

@ -0,0 +1,63 @@
// <copyright file="GoogleSignInConfiguration.cs" company="Google Inc.">
// Copyright (C) 2017 Google Inc. All Rights Reserved.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
// </copyright>
namespace Google {
using System.Collections.Generic;
/// <summary>
/// Configuration properties for Google Sign-In.
/// </summary>
public class GoogleSignInConfiguration {
/// <summary>Set to true to use games signin, false for default signin.
/// </summary>
/// <remarks>Note: The Games configuration is not supported on non-Android
/// platforms.
/// If games configuration is used, you must also add the
/// play-services-games libraries and dependencies.
/// See the README for more details.
/// </remarks>
public bool UseGameSignIn = false;
/// <summary>Web client id associated with this app.</summary>
/// <remarks>Required for requesting auth code or id token.</remarks>
public string WebClientId = null;
/// <summary>Set to true for getting an auth code when authenticating.
/// </summary>
public bool RequestAuthCode = false;
/// <summary>Set to true to request to reset the refresh token.
/// Causes re-consent.
/// </summary>
public bool ForceTokenRefresh = false;
/// <summary>Request email address, requires consent.</summary>
public bool RequestEmail = false;
/// <summary>Request id token, requires consent.</summary>
public bool RequestIdToken = false;
/// <summary>Request profile information, requires consent.</summary>
public bool RequestProfile = false;
/// <summary>Hides popup UIs from games services.</summary>
/// <remarks>Used with games signin to show or hide the connecting popup UI
/// and to associate an invisible view for other popups. This is
/// recommended for VR applications. This has no effect if UseGameSignIn is
/// false.
/// </remarks>
public bool HidePopups = false;
/// <summary>Account name to use when authenticating,
/// null indicates use default.</summary>
public string AccountName = null;
/// <summary>Additional scopes to request, requires consent.</summary>
public IEnumerable<string> AdditionalScopes = null;
}
}

@ -0,0 +1,14 @@
fileFormatVersion: 2
guid: 0c3999c0f68f04ae08f04fb3bf2a2050
labels:
- gvh
- gvh_version-1.0.4
MonoImporter:
externalObjects: {}
serializedVersion: 2
defaultReferences: []
executionOrder: 0
icon: {instanceID: 0}
userData:
assetBundleName:
assetBundleVariant:

@ -0,0 +1,50 @@
using UnityEngine;
using Google;
using System.Threading.Tasks;
public class GoogleSignInController : MonoBehaviour
{
private GoogleSignInConfiguration configuration;
void Awake()
{
// Set up the Google Sign-In configuration
configuration = new GoogleSignInConfiguration
{
WebClientId = "624507103444-6agok4g1q29bsb615v235jbf0k585ruk.apps.googleusercontent.com",
RequestEmail = true,
RequestIdToken = false, // ❌ We no longer request IdToken, since we use ServerAuthCode instead
RequestAuthCode = true // ✅ Enable ServerAuthCode for secure backend authentication
};
GoogleSignIn.Configuration = configuration;
}
public void SignInWithGoogle()
{
GoogleSignIn.DefaultInstance.SignIn().ContinueWith(OnGoogleSignIn);
}
private void OnGoogleSignIn(Task<GoogleSignInUser> task)
{
if (task.IsFaulted)
{
Debug.LogError("Google Sign-In encountered an error: " + task.Exception);
}
else if (task.IsCanceled)
{
Debug.Log("Google Sign-In was canceled.");
}
else
{
GoogleSignInUser user = task.Result;
Debug.Log("✅ Google Sign-In succeeded!");
Debug.Log("👤 Display Name: " + user.DisplayName);
Debug.Log("📧 Email: " + user.Email);
Debug.Log("🔑 Server Auth Code: " + user.AuthCode);
// 🔹 Send this AuthCode to your backend (PlayFab, Firebase, or custom server)
// The backend will exchange this for an access token.
}
}
}

@ -0,0 +1,11 @@
fileFormatVersion: 2
guid: 8948afce6ec0e984aa137d3f84aba688
MonoImporter:
externalObjects: {}
serializedVersion: 2
defaultReferences: []
executionOrder: 0
icon: {instanceID: 0}
userData:
assetBundleName:
assetBundleVariant:

@ -0,0 +1,72 @@
// <copyright file="GoogleSignInStatusCode.cs" company="Google Inc.">
// Copyright (C) 2017 Google Inc. All Rights Reserved.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
// </copyright>
namespace Google {
/// <summary>
/// Status code for the SignIn operations.
/// </summary>
/// <remarks>All successful status codes are less than or equal to 0.
/// </remarks>
public enum GoogleSignInStatusCode {
/// <summary>The operation was successful, but used the device's cache.
/// </summary>
SuccessCached = -1,
/// <summary>The operation was successful.</summary>
Success = 0,
/// <summary>The client attempted to call a method from an API that
/// failed to connect.</summary>
ApiNotConnected = 1,
/// <summary>The result was canceled either due to client disconnect
/// or cancel().</summary>
Canceled = 2,
/// <summary> A blocking call was interrupted while waiting and did not
/// run to completion.</summary>
Interrupted = 3,
/// <summary> The client attempted to connect to the service with an
/// invalid account name specified. </summary>
InvalidAccount = 4,
/// <summary>Timed out while awaiting the result.</summary>
Timeout = 5,
/// <summary>The application is misconfigured.
/// This error is not recoverable.</summary>
/// <remarks>
/// The developer should look at the logs after this to determine
/// more actionable information.
/// </remarks>
DeveloperError = 6,
/// <summary>An internal error occurred. Retrying should resolve the
/// problem.</summary>
InternalError = 7,
/// <summary>A network error occurred. Retrying should resolve the problem.
/// </summary>
NetworkError = 8,
/// <summary> The operation failed with no more detailed information.
/// </summary>
Error = 9,
}
} // namespace GoogleSignIn

@ -0,0 +1,14 @@
fileFormatVersion: 2
guid: 077933fcaedac412d9762bf3e0a3be68
labels:
- gvh
- gvh_version-1.0.4
MonoImporter:
externalObjects: {}
serializedVersion: 2
defaultReferences: []
executionOrder: 0
icon: {instanceID: 0}
userData:
assetBundleName:
assetBundleVariant:

@ -0,0 +1,76 @@
// <copyright file="GoogleSignInUser.cs" company="Google Inc.">
// Copyright (C) 2017 Google Inc. All Rights Reserved.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
// </copyright>
namespace Google {
using System;
/// <summary> Information for the authenticated user.</summary>
public class GoogleSignInUser {
/// <summary> Server AuthCode to be exchanged for an auth token.</summary>
///<remarks> null if not requested, or if there was an error.</remarks>
public string AuthCode {
get;
internal set;
}
/// <summary> Email address.</summary>
///<remarks> null if not requested, or if there was an error.</remarks>
public string Email {
get;
internal set;
}
/// <summary> Id token.</summary>
///<remarks> null if not requested, or if there was an error.</remarks>
public string IdToken {
get;
internal set;
}
/// <summary> Display Name.</summary>
public string DisplayName {
get;
internal set;
}
/// <summary> Given Name.</summary>
public string GivenName {
get;
internal set;
}
/// <summary> Family Name.</summary>
public string FamilyName {
get;
internal set;
}
/// <summary> Profile photo</summary>
/// <remarks> Can be null if the profile is not requested,
/// or none set.</remarks>
public Uri ImageUrl {
get;
internal set;
}
/// <summary> User ID</summary>
public string UserId {
get;
internal set;
}
}
}

@ -0,0 +1,14 @@
fileFormatVersion: 2
guid: c56b92217d0144af5907627d1235e0a5
labels:
- gvh
- gvh_version-1.0.4
MonoImporter:
externalObjects: {}
serializedVersion: 2
defaultReferences: []
executionOrder: 0
icon: {instanceID: 0}
userData:
assetBundleName:
assetBundleVariant:

@ -0,0 +1,9 @@
fileFormatVersion: 2
guid: 2fbe4f3ec9db4415e849f9bb89e63a92
folderAsset: yes
timeCreated: 1502761839
licenseType: Pro
DefaultImporter:
userData:
assetBundleName:
assetBundleVariant:

@ -0,0 +1,70 @@
// <copyright file="BaseObject.cs" company="Google Inc.">
// Copyright (C) 2017 Google Inc. All Rights Reserved.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
// </copyright>
namespace Google.Impl {
using System;
using System.Runtime.InteropServices;
using System.Text;
using UnityEngine;
/// <summary>
/// Base object manages the pointer to a native object which provides the
/// implementation of a C# object.
/// </summary>
internal abstract class BaseObject : IDisposable {
// handle to native object.
private HandleRef selfHandleRef;
private static HandleRef nullSelf = new HandleRef();
public BaseObject(IntPtr intPtr) {
selfHandleRef = new HandleRef(this, intPtr);
}
protected HandleRef SelfPtr() {
if (selfHandleRef.Equals(nullSelf)) {
throw new InvalidOperationException(
"Attempted to use object after it was cleaned up");
}
return selfHandleRef;
}
public virtual void Dispose() {
selfHandleRef = nullSelf;
}
internal delegate UIntPtr OutStringMethod([In, Out] byte[] out_bytes,
UIntPtr out_size);
internal static String OutParamsToString(OutStringMethod outStringMethod) {
UIntPtr requiredSize = outStringMethod(null, UIntPtr.Zero);
if (requiredSize.Equals(UIntPtr.Zero)) {
return null;
}
string str = null;
try {
byte[] array = new byte[requiredSize.ToUInt32()];
outStringMethod(array, requiredSize);
str = Encoding.UTF8.GetString(array, 0,
(int)requiredSize.ToUInt32() - 1);
} catch (Exception e) {
Debug.LogError("Exception creating string from char array: " + e);
str = string.Empty;
}
return str;
}
}
}

@ -0,0 +1,14 @@
fileFormatVersion: 2
guid: fe250cda690a44cb08f8f7d26c9723b4
labels:
- gvh
- gvh_version-1.0.4
MonoImporter:
externalObjects: {}
serializedVersion: 2
defaultReferences: []
executionOrder: 0
icon: {instanceID: 0}
userData:
assetBundleName:
assetBundleVariant:

@ -0,0 +1,187 @@
// <copyright file="GoogleSignInImpl.cs" company="Google Inc.">
// Copyright (C) 2017 Google Inc. All Rights Reserved.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
// </copyright>
namespace Google.Impl {
using System;
using System.Collections.Generic;
using System.Runtime.InteropServices;
internal class GoogleSignInImpl : BaseObject, ISignInImpl {
#if UNITY_ANDROID
private const string DllName = "native-googlesignin";
#else
private const string DllName = "__Internal";
#endif
internal GoogleSignInImpl(GoogleSignInConfiguration configuration)
: base(GoogleSignIn_Create(GetPlayerActivity())) {
if (configuration != null) {
List<string> scopes = new List<string>();
if (configuration.AdditionalScopes != null) {
scopes.AddRange(configuration.AdditionalScopes);
}
GoogleSignIn_Configure(SelfPtr(), configuration.UseGameSignIn,
configuration.WebClientId,
configuration.RequestAuthCode,
configuration.ForceTokenRefresh,
configuration.RequestEmail,
configuration.RequestIdToken,
configuration.HidePopups,
scopes.ToArray(),
scopes.Count,
configuration.AccountName);
}
}
/// <summary>Enables/Disables verbose logging to help troubleshooting</summary>
public void EnableDebugLogging(bool flag) {
GoogleSignIn_EnableDebugLogging(SelfPtr(), flag);
}
/// <summary>
/// Starts the authentication process.
/// </summary>
/// <remarks>
/// The authenication process is started and may display account picker
/// popups and consent prompts based on the state of authentication and
/// the requested elements.
/// </remarks>
public Future<GoogleSignInUser> SignIn() {
IntPtr nativeFuture = GoogleSignIn_SignIn(SelfPtr());
return new Future<GoogleSignInUser>(new NativeFuture(nativeFuture));
}
/// <summary>
/// Starts the authentication process.
/// </summary>
/// <remarks>
/// The authenication process is started and may display account picker
/// popups and consent prompts based on the state of authentication and
/// the requested elements.
/// </remarks>
public Future<GoogleSignInUser> SignInSilently() {
IntPtr nativeFuture = GoogleSignIn_SignInSilently(SelfPtr());
return new Future<GoogleSignInUser>(new NativeFuture(nativeFuture));
}
/// <summary>
/// Signs out the User.
/// </summary>
public void SignOut() {
GoogleSignIn_Signout(SelfPtr());
}
/// <summary>
/// Disconnects the user from the application and revokes all consent.
/// </summary>
public void Disconnect() {
GoogleSignIn_Disconnect(SelfPtr());
}
/// <summary>
/// Creates an instance of the native Google Sign-In implementation.
/// </summary>
/// <remarks>
/// For Android this must be the JNI raw object for the parentActivity.
/// For iOS it is ignored.
/// </remarks>
/// <returns>The pointer to the instance.</returns>
/// <param name="data">Data used in creating the instance.</param>
[DllImport(DllName)]
static extern IntPtr GoogleSignIn_Create(IntPtr data);
[DllImport(DllName)]
static extern void GoogleSignIn_EnableDebugLogging(HandleRef self, bool flag);
[DllImport(DllName)]
static extern bool GoogleSignIn_Configure(HandleRef self,
bool useGameSignIn, string webClientId,
bool requestAuthCode, bool forceTokenRefresh, bool requestEmail,
bool requestIdToken, bool hidePopups, string[] additionalScopes,
int scopeCount, string accountName);
[DllImport(DllName)]
static extern IntPtr GoogleSignIn_SignIn(HandleRef self);
[DllImport(DllName)]
static extern IntPtr GoogleSignIn_SignInSilently(HandleRef self);
[DllImport(DllName)]
static extern void GoogleSignIn_Signout(HandleRef self);
[DllImport(DllName)]
static extern void GoogleSignIn_Disconnect(HandleRef self);
[DllImport(DllName)]
internal static extern void GoogleSignIn_DisposeFuture(HandleRef self);
[DllImport(DllName)]
internal static extern bool GoogleSignIn_Pending(HandleRef self);
[DllImport(DllName)]
internal static extern IntPtr GoogleSignIn_Result(HandleRef self);
[DllImport(DllName)]
internal static extern int GoogleSignIn_Status(HandleRef self);
[DllImport(DllName)]
internal static extern UIntPtr GoogleSignIn_GetServerAuthCode(
HandleRef self, [In, Out] byte[] bytes, UIntPtr len);
[DllImport(DllName)]
internal static extern UIntPtr GoogleSignIn_GetDisplayName(HandleRef self,
[In, Out] byte[] bytes, UIntPtr len);
[DllImport(DllName)]
internal static extern UIntPtr GoogleSignIn_GetEmail(HandleRef self,
[In, Out] byte[] bytes, UIntPtr len);
[DllImport(DllName)]
internal static extern UIntPtr GoogleSignIn_GetFamilyName(HandleRef self,
[In, Out] byte[] bytes, UIntPtr len);
[DllImport(DllName)]
internal static extern UIntPtr GoogleSignIn_GetGivenName(HandleRef self,
[In, Out] byte[] bytes, UIntPtr len);
[DllImport(DllName)]
internal static extern UIntPtr GoogleSignIn_GetIdToken(HandleRef self,
[In, Out] byte[] bytes, UIntPtr len);
[DllImport(DllName)]
internal static extern UIntPtr GoogleSignIn_GetImageUrl(HandleRef self,
[In, Out] byte[] bytes, UIntPtr len);
[DllImport(DllName)]
internal static extern UIntPtr GoogleSignIn_GetUserId(HandleRef self,
[In, Out] byte[] bytes, UIntPtr len);
// Gets the Unity player activity.
// For iOS, this returns Zero.
private static IntPtr GetPlayerActivity() {
#if UNITY_ANDROID
UnityEngine.AndroidJavaClass jc = new UnityEngine.AndroidJavaClass(
"com.unity3d.player.UnityPlayer");
return jc.GetStatic<UnityEngine.AndroidJavaObject>("currentActivity")
.GetRawObject();
#else
return IntPtr.Zero;
#endif
}
}
}

@ -0,0 +1,14 @@
fileFormatVersion: 2
guid: 9ae1f008f9f994b9c96c1a14067d7b48
labels:
- gvh
- gvh_version-1.0.4
MonoImporter:
externalObjects: {}
serializedVersion: 2
defaultReferences: []
executionOrder: 0
icon: {instanceID: 0}
userData:
assetBundleName:
assetBundleVariant:

@ -0,0 +1,102 @@
// <copyright file="NativeFuture.cs" company="Google Inc.">
// Copyright (C) 2017 Google Inc. All Rights Reserved.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
// </copyright>
namespace Google.Impl {
using System;
using System.Runtime.InteropServices;
/// <summary>
/// Native future is an interal class that implements the FutureAPIImpl
/// by calling native methods which are implemented in the native code.
/// </summary>
internal class NativeFuture : BaseObject, FutureAPIImpl<GoogleSignInUser> {
internal NativeFuture(IntPtr ptr) : base(ptr) {
}
public override void Dispose() {
GoogleSignInImpl.GoogleSignIn_DisposeFuture(SelfPtr());
base.Dispose();
}
public bool Pending {
get {
return GoogleSignInImpl.GoogleSignIn_Pending(SelfPtr());
}
}
public GoogleSignInUser Result {
get {
IntPtr ptr = GoogleSignInImpl.GoogleSignIn_Result(SelfPtr());
if (ptr != IntPtr.Zero) {
GoogleSignInUser user = new GoogleSignInUser();
HandleRef userPtr = new HandleRef(user, ptr);
user.DisplayName = OutParamsToString((out_string, out_size) =>
GoogleSignInImpl.GoogleSignIn_GetDisplayName(userPtr,
out_string,
out_size));
user.Email = OutParamsToString((out_string, out_size) =>
GoogleSignInImpl.GoogleSignIn_GetEmail(userPtr, out_string,
out_size));
user.FamilyName = OutParamsToString((out_string, out_size) =>
GoogleSignInImpl.GoogleSignIn_GetFamilyName(userPtr, out_string,
out_size));
user.GivenName = OutParamsToString((out_string, out_size) =>
GoogleSignInImpl.GoogleSignIn_GetGivenName(userPtr, out_string,
out_size));
user.IdToken = OutParamsToString((out_string, out_size) =>
GoogleSignInImpl.GoogleSignIn_GetIdToken(userPtr, out_string,
out_size));
user.AuthCode = OutParamsToString((out_string, out_size) =>
GoogleSignInImpl.GoogleSignIn_GetServerAuthCode(userPtr, out_string,
out_size));
string url = OutParamsToString((out_string, out_size) =>
GoogleSignInImpl.GoogleSignIn_GetImageUrl(userPtr, out_string,
out_size));
if (url.Length > 0) {
user.ImageUrl = new System.Uri(url);
}
user.UserId = OutParamsToString((out_string, out_size) =>
GoogleSignInImpl.GoogleSignIn_GetUserId(userPtr, out_string,
out_size));
return user;
} else {
return null;
}
}
}
/// <summary>
/// Gets the status.
/// </summary>
/// <remarks>The platform specific implementation maps the platform specific
/// code to one defined in GoogleSignStatusCode.</remarks>
/// <value>The status.</value>
public GoogleSignInStatusCode Status {
get {
return (GoogleSignInStatusCode)GoogleSignInImpl.GoogleSignIn_Status(
SelfPtr());
}
}
}
}

@ -0,0 +1,14 @@
fileFormatVersion: 2
guid: 5f1aae79b1ca4432d9d8ec382c54bf46
labels:
- gvh
- gvh_version-1.0.4
MonoImporter:
externalObjects: {}
serializedVersion: 2
defaultReferences: []
executionOrder: 0
icon: {instanceID: 0}
userData:
assetBundleName:
assetBundleVariant:

@ -0,0 +1,42 @@
// <copyright file="SignInHelperObject.cs" company="Google Inc.">
// Copyright (C) 2017 Google Inc. All Rights Reserved.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
// </copyright>
namespace Google.Impl {
using UnityEngine;
///<summary>Helper object to connect the Sign-in API to the Unity Game Scene.
///</summary>
///<remarks>This class is added to the scene so that the Google Sign-in API
/// can start coroutines.
///</remarks>
public class SignInHelperObject : MonoBehaviour {
private static SignInHelperObject instance;
internal static SignInHelperObject Instance {
get {
if (Application.isPlaying) {
// add an invisible game object to the scene
GameObject obj = new GameObject("GoogleSignInHelperObject");
DontDestroyOnLoad(obj);
instance = obj.AddComponent<SignInHelperObject>();
} else {
instance = new SignInHelperObject();
}
return instance;
}
}
}
}

@ -0,0 +1,14 @@
fileFormatVersion: 2
guid: ad98e5b48888e44eb81dd5884d3a1754
labels:
- gvh
- gvh_version-1.0.4
MonoImporter:
externalObjects: {}
serializedVersion: 2
defaultReferences: []
executionOrder: 0
icon: {instanceID: 0}
userData:
assetBundleName:
assetBundleVariant:

@ -0,0 +1,8 @@
fileFormatVersion: 2
guid: df8ccdbfa53820647be6f67fba435252
folderAsset: yes
DefaultImporter:
externalObjects: {}
userData:
assetBundleName:
assetBundleVariant:

@ -0,0 +1,236 @@
using UnityEngine;
using UnityEngine.UI;
using Google;
using PlayFab;
using PlayFab.ClientModels;
using System;
using System.Collections.Generic;
using System.Threading.Tasks;
public class GoogleSignInManager : MonoBehaviour
{
private GoogleSignInConfiguration configuration;
public Bootstrapper bootstrapper;
public Button googleSignInButton;
public Button guestLoginButton;
void Awake()
{
configuration = new GoogleSignInConfiguration
{
WebClientId = "627393855302-ie99j5loumrkt3gqi5mopa1c1avu7o86.apps.googleusercontent.com",
RequestEmail = true,
RequestAuthCode = true
};
GoogleSignIn.Configuration = configuration;
Application.quitting += SyncPlayerPrefsToPlayFabOnQuit;
}
void Start()
{
googleSignInButton.onClick.RemoveAllListeners();
googleSignInButton.onClick.AddListener(SignInWithGoogle);
guestLoginButton.onClick.RemoveAllListeners();
guestLoginButton.onClick.AddListener(GuestLogin);
if (PlayerPrefs.HasKey("PlayFabID") && !PlayerPrefs.HasKey("GuestMode"))
{
Debug.Log("User previously signed in with Google. Attempting silent login...");
googleSignInButton.gameObject.SetActive(false);
guestLoginButton.gameObject.SetActive(false);
SignInSilently();
}
else if (PlayerPrefs.HasKey("GuestMode"))
{
Debug.Log("Guest mode previously selected. Letting user choose again.");
googleSignInButton.gameObject.SetActive(true);
guestLoginButton.gameObject.SetActive(true);
}
else
{
Debug.Log("No login info found. Showing login options.");
googleSignInButton.gameObject.SetActive(true);
guestLoginButton.gameObject.SetActive(true);
}
}
public void GuestLogin()
{
Debug.Log("Starting game in Guest Mode...");
SafePlayerPrefs.SetInt("GuestMode", 1);
PlayerPrefs.Save();
googleSignInButton.gameObject.SetActive(false);
guestLoginButton.gameObject.SetActive(false);
bootstrapper.StartGame();
}
public void SignInWithGoogle()
{
PlayerPrefs.DeleteKey("GuestMode");
PlayerPrefs.DeleteAll();
PlayerPrefs.Save();
googleSignInButton.interactable = false;
guestLoginButton.interactable = false;
GoogleSignIn.DefaultInstance.SignIn().ContinueWith(OnGoogleSignIn);
}
private void OnGoogleSignIn(Task<GoogleSignInUser> task)
{
if (task.IsFaulted || task.IsCanceled)
{
Debug.LogError("Google Sign-In failed: " + task.Exception);
googleSignInButton.gameObject.SetActive(true);
googleSignInButton.interactable = true;
guestLoginButton.gameObject.SetActive(true);
guestLoginButton.interactable = true;
return;
}
GoogleSignInUser user = task.Result;
string authCode = user.AuthCode;
SafePlayerPrefs.SetString("GoogleAuthCode", authCode);
PlayerPrefs.Save();
LoginToPlayFab(authCode);
}
private void SignInSilently()
{
Debug.Log("Attempting Google Silent Sign-In...");
GoogleSignIn.DefaultInstance.SignInSilently().ContinueWith(task =>
{
if (task.IsFaulted || task.IsCanceled)
{
Debug.LogWarning("Silent Sign-In failed. Showing login buttons.");
googleSignInButton.gameObject.SetActive(true);
googleSignInButton.interactable = true;
guestLoginButton.gameObject.SetActive(true);
guestLoginButton.interactable = true;
return;
}
GoogleSignInUser user = task.Result;
string authCode = user.AuthCode;
SafePlayerPrefs.SetString("GoogleAuthCode", authCode);
PlayerPrefs.Save();
LoginToPlayFab(authCode);
});
}
private void LoginToPlayFab(string authCode)
{
PlayFabSettings.staticSettings.TitleId = "1F528E";
var request = new LoginWithGoogleAccountRequest
{
TitleId = "1F528E",
ServerAuthCode = authCode,
CreateAccount = true
};
PlayFabClientAPI.LoginWithGoogleAccount(request, OnPlayFabLoginSuccess, OnPlayFabLoginFailure);
}
private void OnPlayFabLoginSuccess(LoginResult result)
{
Debug.Log("✅ PlayFab Login Success! PlayFab ID: " + result.PlayFabId);
SafePlayerPrefs.SetString("PlayFabID", result.PlayFabId);
PlayerPrefs.Save();
LoadPlayerPrefsFromPlayFab(() =>
{
googleSignInButton.gameObject.SetActive(false);
guestLoginButton.gameObject.SetActive(false);
bootstrapper.StartGame();
});
}
private void OnPlayFabLoginFailure(PlayFabError error)
{
Debug.LogError("❌ PlayFab Login Failed: " + error.GenerateErrorReport());
googleSignInButton.gameObject.SetActive(true);
googleSignInButton.interactable = true;
guestLoginButton.gameObject.SetActive(true);
guestLoginButton.interactable = true;
}
private void LoadPlayerPrefsFromPlayFab(Action onComplete)
{
PlayFabClientAPI.GetUserData(new GetUserDataRequest(), result =>
{
if (result.Data != null)
{
foreach (var entry in result.Data)
{
string key = entry.Key;
string rawValue = entry.Value.Value;
if (rawValue.StartsWith("int:") && int.TryParse(rawValue.Substring(4), out int i))
SafePlayerPrefs.SetInt(key, i);
else if (rawValue.StartsWith("float:") && float.TryParse(rawValue.Substring(6), out float f))
SafePlayerPrefs.SetFloat(key, f);
else if (rawValue.StartsWith("string:"))
SafePlayerPrefs.SetString(key, rawValue.Substring(7));
else
SafePlayerPrefs.SetString(key, rawValue);
PlayerPrefsKeys.RegisterKey(key);
}
PlayerPrefs.Save();
}
onComplete?.Invoke();
},
error =>
{
Debug.LogError("❌ Failed to load SafePlayerPrefs from PlayFab: " + error.GenerateErrorReport());
onComplete?.Invoke();
});
}
private void SyncPlayerPrefsToPlayFabOnQuit()
{
if (PlayerPrefs.HasKey("GuestMode")) return;
Dictionary<string, string> allPrefs = new Dictionary<string, string>();
foreach (var key in PlayerPrefsKeys.GetAllKeys())
{
allPrefs[key] = PlayerPrefs.GetString(key);
}
var request = new UpdateUserDataRequest
{
Data = allPrefs
};
if(PlayFabClientAPI.IsClientLoggedIn())
{
PlayFabClientAPI.UpdateUserData(request,
result => Debug.Log("✅ Synced SafePlayerPrefs to PlayFab on quit."),
error => Debug.LogError("❌ Failed to sync SafePlayerPrefs: " + error.GenerateErrorReport()));
}
}
public void SignOut()
{
PlayerPrefs.DeleteKey("GoogleAuthCode");
PlayerPrefs.DeleteKey("PlayFabID");
PlayerPrefs.DeleteKey("GuestMode");
PlayerPrefs.Save();
GoogleSignIn.DefaultInstance.SignOut();
Debug.Log("User signed out.");
}
}

@ -0,0 +1,11 @@
fileFormatVersion: 2
guid: aea303bf6931eb64a925ac187863ae99
MonoImporter:
externalObjects: {}
serializedVersion: 2
defaultReferences: []
executionOrder: 0
icon: {instanceID: 0}
userData:
assetBundleName:
assetBundleVariant:

@ -0,0 +1,23 @@
using System.Collections.Generic;
using UnityEngine;
public static class PlayerPrefsKeys
{
private const string KeyRegistry = "_AllPlayerPrefsKeys";
public static void RegisterKey(string key)
{
var keys = PlayerPrefs.GetString(KeyRegistry, "");
if (!keys.Contains(key))
{
keys += key + ";";
SafePlayerPrefs.SetString(KeyRegistry, keys);
PlayerPrefs.Save();
}
}
public static List<string> GetAllKeys()
{
var keysString = PlayerPrefs.GetString(KeyRegistry, "");
return new List<string>(keysString.Split(';', System.StringSplitOptions.RemoveEmptyEntries));
}
}

@ -0,0 +1,11 @@
fileFormatVersion: 2
guid: ffa4debad77245b4d9620a5fc2b07c9d
MonoImporter:
externalObjects: {}
serializedVersion: 2
defaultReferences: []
executionOrder: 0
icon: {instanceID: 0}
userData:
assetBundleName:
assetBundleVariant:

@ -0,0 +1,137 @@
using UnityEngine;
using PlayFab;
using PlayFab.ClientModels;
using System.Collections.Generic;
public class PlayerPrefsSyncManager : MonoBehaviour
{
private static PlayerPrefsSyncManager instance;
private void OnApplicationQuit()
{
SyncPlayerPrefsToPlayFabOnQuit();
}
private void OnApplicationPause()
{
SyncPlayerPrefsToPlayFabOnQuit();
}
private void OnApplicationFocus(bool focus)
{
//if (!focus)
SyncPlayerPrefsToPlayFabOnQuit();
}
void Awake()
{
if (instance == null)
{
instance = this;
DontDestroyOnLoad(this.gameObject);
Application.quitting += SyncPlayerPrefsToPlayFabOnQuit;
}
else
{
Destroy(gameObject);
}
}
private const int MaxKeysPerRequest = 10;
public void SyncPlayerPrefsToPlayFabOnQuit()
{
if (PlayFabClientAPI.IsClientLoggedIn())
{
var keys = PlayerPrefsKeys.GetAllKeys();
if (keys.Count == 0)
{
Debug.Log("No PlayerPrefs keys registered, skipping sync.");
return;
}
Dictionary<string, string> allPrefs = new Dictionary<string, string>();
foreach (var key in keys)
{
string strVal = PlayerPrefs.GetString(key, "__MISSING__");
if (strVal != "__MISSING__")
{
allPrefs[key] = "string:" + strVal;
continue;
}
int intVal = PlayerPrefs.GetInt(key, int.MinValue + 1);
if (intVal != int.MinValue + 1)
{
allPrefs[key] = "int:" + intVal;
continue;
}
float floatVal = PlayerPrefs.GetFloat(key, float.MinValue + 1);
if (floatVal != float.MinValue + 1)
{
allPrefs[key] = "float:" + floatVal.ToString("R");
}
}
foreach (var pair in allPrefs)
{
Debug.Log($"[Sync] {pair.Key} = {pair.Value}");
}
// Split into batches of 10
var batches = new List<Dictionary<string, string>>();
var currentBatch = new Dictionary<string, string>();
foreach (var pair in allPrefs)
{
currentBatch[pair.Key] = pair.Value;
if (currentBatch.Count == MaxKeysPerRequest)
{
batches.Add(currentBatch);
currentBatch = new Dictionary<string, string>();
}
}
if (currentBatch.Count > 0)
{
batches.Add(currentBatch);
}
UploadPlayerPrefsBatches(batches, 0);
}
}
private void UploadPlayerPrefsBatches(List<Dictionary<string, string>> batches, int index)
{
if (index >= batches.Count)
{
Debug.Log("✅ All PlayerPrefs batches synced to PlayFab.");
return;
}
var request = new UpdateUserDataRequest
{
Data = batches[index],
Permission = UserDataPermission.Public
};
if (PlayFabClientAPI.IsClientLoggedIn())
{
PlayFabClientAPI.UpdateUserData(request,
result =>
{
Debug.Log($"✅ Synced batch {index + 1}/{batches.Count}");
UploadPlayerPrefsBatches(batches, index + 1);
},
error =>
{
Debug.LogError($"❌ Failed to sync batch {index + 1}/{batches.Count}: {error.GenerateErrorReport()}");
});
}
}
}

@ -0,0 +1,11 @@
fileFormatVersion: 2
guid: c814388591ae90d4aa18f1259f9c3b10
MonoImporter:
externalObjects: {}
serializedVersion: 2
defaultReferences: []
executionOrder: 0
icon: {instanceID: 0}
userData:
assetBundleName:
assetBundleVariant:

@ -0,0 +1,124 @@
using PlayFab.ClientModels;
using PlayFab;
using System.Collections;
using System.Collections.Generic;
using UnityEngine;
using UnityEngine.UI;
using MS;
using TMPro;
public class Profile : MonoBehaviour
{
// Start is called before the first frame update
public Button EnteredNameButton;
public List<Sprite> AvatarSprites;
public Sprite AvatarSelectedSprite;
public Popup EnterNamePanel;
public Popup MainPanel;
public TMP_InputField nameInputField;
public Image HeaderProfileImg;
public TextMeshProUGUI HeaderProfileName;
public Image LevelFillerImg;
public TextMeshProUGUI LevelFillerText;
private void Start()
{
CheckForDisplayName();
CheckForAvatar();
if (EnteredNameButton != null)
EnteredNameButton.interactable = false;
}
private void CheckForAvatar()
{
int hasSelectedAvatar = PlayerPrefs.GetInt(GameConstants.AvatarSelectedCheckKey, 0);
if (hasSelectedAvatar > 0)
{
//string name = PlayerPrefs.GetString(GameConstants.DisplayNameKey);
//PlayFabLeaderboards.DisplayName = name;
AvatarSelectedSprite = AvatarSprites[PlayerPrefs.GetInt(GameConstants.AvatarSelectedIndex)];
HeaderProfileImg.sprite = AvatarSelectedSprite;
}
else
{
MainPanel.Close();
if (!EnterNamePanel.isOpen)
EnterNamePanel.Open();
}
}
private void CheckForDisplayName()
{
int hasEnteredName = PlayerPrefs.GetInt(GameConstants.NameEnteredCheckKey, 0);
if (hasEnteredName > 0)
{
string name = PlayerPrefs.GetString(GameConstants.DisplayNameKey);
PlayFabLeaderboards.DisplayName = name;
PlayFabManager.Instance.playFabLeaderboards.SetDisplayName(name);
HeaderProfileName.text = name;
LevelFillerSetter();
}
else
{
Debug.Log("Name not found in plaerprefs");
MainPanel.Close();
if (!EnterNamePanel.isOpen)
EnterNamePanel.Open();
}
}
public void LevelFillerSetter()
{
int bestScore = PlayerPrefs.GetInt("BestScore", 0); // Default to 0 if not set
float fillAmount = Mathf.Clamp01((float)bestScore / 100f); // Ensure it stays between 0 and 1
LevelFillerImg.fillAmount = fillAmount;
//LevelFillerText.text = bestScore + "/" + maxScore;
}
public void OnDisplayNameEntered()
{
if (PlayFabClientAPI.IsClientLoggedIn())
{
PlayFabManager.Instance.playFabLeaderboards.SetDisplayName(nameInputField.text);
}
SafePlayerPrefs.SetInt(GameConstants.NameEnteredCheckKey, 1);
SafePlayerPrefs.SetString(GameConstants.DisplayNameKey, nameInputField.text);
HeaderProfileName.text = nameInputField.text;
EnterNamePanel.Close();
LevelFillerSetter();
}
public void OnAvatarSelected(int AvatarID)
{
//PlayFabManager.Instance.playFabLeaderboards.SetDisplayName(nameInputField.text);
EnteredNameButton.interactable = true;
SafePlayerPrefs.SetInt(GameConstants.AvatarSelectedCheckKey, 1);
SafePlayerPrefs.SetInt(GameConstants.AvatarSelectedIndex, AvatarID);
AvatarSelectedSprite = AvatarSprites[AvatarID];
HeaderProfileImg.sprite = AvatarSelectedSprite;
SetAvatarID(AvatarID);
//NameEnterPanel.Close();
}
public void SetAvatarID(int avatarID)
{
var request = new UpdateUserDataRequest
{
Data = new Dictionary<string, string>
{
{ "AvatarID", avatarID.ToString() } // Convert int to string for storage
},
Permission = UserDataPermission.Public
};
if (PlayFabClientAPI.IsClientLoggedIn())
PlayFabClientAPI.UpdateUserData(request, OnDataUpdateSuccess, OnDataUpdateFailure);
}
private void OnDataUpdateSuccess(UpdateUserDataResult result)
{
Debug.Log("Avatar ID updated successfully.");
}
private void OnDataUpdateFailure(PlayFabError error)
{
Debug.LogError("Failed to update Avatar ID: " + error.GenerateErrorReport());
}
}

@ -0,0 +1,11 @@
fileFormatVersion: 2
guid: 21236f4bddba5ff4db559d91c0c804b4
MonoImporter:
externalObjects: {}
serializedVersion: 2
defaultReferences: []
executionOrder: 0
icon: {instanceID: 0}
userData:
assetBundleName:
assetBundleVariant:

@ -0,0 +1,24 @@
using UnityEngine;
public static class SafePlayerPrefs
{
public static void SetInt(string key, int value)
{
PlayerPrefs.SetInt(key, value);
PlayerPrefsKeys.RegisterKey(key);
PlayerPrefs.Save();
}
public static void SetFloat(string key, float value)
{
PlayerPrefs.SetFloat(key, value);
PlayerPrefsKeys.RegisterKey(key);
PlayerPrefs.Save();
}
public static void SetString(string key, string value)
{
PlayerPrefs.SetString(key, value);
PlayerPrefsKeys.RegisterKey(key);
PlayerPrefs.Save();
}
}

@ -0,0 +1,11 @@
fileFormatVersion: 2
guid: 3f16590318a73204e98d26d21f94e188
MonoImporter:
externalObjects: {}
serializedVersion: 2
defaultReferences: []
executionOrder: 0
icon: {instanceID: 0}
userData:
assetBundleName:
assetBundleVariant:

Binary file not shown.

After

Width:  |  Height:  |  Size: 3.2 KiB

@ -0,0 +1,140 @@
fileFormatVersion: 2
guid: 2d53a66c391dadc4b8ad1347e8c7682f
TextureImporter:
internalIDToNameTable: []
externalObjects: {}
serializedVersion: 13
mipmaps:
mipMapMode: 0
enableMipMap: 0
sRGBTexture: 1
linearTexture: 0
fadeOut: 0
borderMipMap: 0
mipMapsPreserveCoverage: 0
alphaTestReferenceValue: 0.5
mipMapFadeDistanceStart: 1
mipMapFadeDistanceEnd: 3
bumpmap:
convertToNormalMap: 0
externalNormalMap: 0
heightScale: 0.25
normalMapFilter: 0
flipGreenChannel: 0
isReadable: 0
streamingMipmaps: 0
streamingMipmapsPriority: 0
vTOnly: 0
ignoreMipmapLimit: 0
grayScaleToAlpha: 0
generateCubemap: 6
cubemapConvolution: 0
seamlessCubemap: 0
textureFormat: 1
maxTextureSize: 2048
textureSettings:
serializedVersion: 2
filterMode: 1
aniso: 1
mipBias: 0
wrapU: 1
wrapV: 1
wrapW: 0
nPOTScale: 0
lightmap: 0
compressionQuality: 50
spriteMode: 1
spriteExtrude: 1
spriteMeshType: 1
alignment: 0
spritePivot: {x: 0.5, y: 0.5}
spritePixelsToUnits: 100
spriteBorder: {x: 0, y: 0, z: 0, w: 0}
spriteGenerateFallbackPhysicsShape: 1
alphaUsage: 1
alphaIsTransparency: 1
spriteTessellationDetail: -1
textureType: 8
textureShape: 1
singleChannelComponent: 0
flipbookRows: 1
flipbookColumns: 1
maxTextureSizeSet: 0
compressionQualitySet: 0
textureFormatSet: 0
ignorePngGamma: 0
applyGammaDecoding: 0
swizzle: 50462976
cookieLightType: 0
platformSettings:
- serializedVersion: 3
buildTarget: DefaultTexturePlatform
maxTextureSize: 2048
resizeAlgorithm: 0
textureFormat: -1
textureCompression: 1
compressionQuality: 50
crunchedCompression: 0
allowsAlphaSplitting: 0
overridden: 0
ignorePlatformSupport: 0
androidETC2FallbackOverride: 0
forceMaximumCompressionQuality_BC6H_BC7: 0
- serializedVersion: 3
buildTarget: Standalone
maxTextureSize: 2048
resizeAlgorithm: 0
textureFormat: -1
textureCompression: 1
compressionQuality: 50
crunchedCompression: 0
allowsAlphaSplitting: 0
overridden: 0
ignorePlatformSupport: 0
androidETC2FallbackOverride: 0
forceMaximumCompressionQuality_BC6H_BC7: 0
- serializedVersion: 3
buildTarget: iPhone
maxTextureSize: 2048
resizeAlgorithm: 0
textureFormat: -1
textureCompression: 1
compressionQuality: 50
crunchedCompression: 0
allowsAlphaSplitting: 0
overridden: 0
ignorePlatformSupport: 0
androidETC2FallbackOverride: 0
forceMaximumCompressionQuality_BC6H_BC7: 0
- serializedVersion: 3
buildTarget: Android
maxTextureSize: 2048
resizeAlgorithm: 0
textureFormat: -1
textureCompression: 1
compressionQuality: 50
crunchedCompression: 0
allowsAlphaSplitting: 0
overridden: 0
ignorePlatformSupport: 0
androidETC2FallbackOverride: 0
forceMaximumCompressionQuality_BC6H_BC7: 0
spriteSheet:
serializedVersion: 2
sprites: []
outline: []
physicsShape: []
bones: []
spriteID: 5e97eb03825dee720800000000000000
internalID: 0
vertices: []
indices:
edges: []
weights: []
secondaryTextures: []
nameFileIdTable: {}
mipmapLimitGroupName:
pSDRemoveMatte: 0
userData:
assetBundleName:
assetBundleVariant:

Binary file not shown.

@ -1,5 +1,5 @@
fileFormatVersion: 2
guid: ed9b95dc6ed6d0647ad7f1a8f305385d
guid: ee723588dbcf9034d82443ab62c094db
DefaultImporter:
externalObjects: {}
userData:

Binary file not shown.

File diff suppressed because it is too large Load Diff

@ -0,0 +1,32 @@
fileFormatVersion: 2
guid: 46cc214f66e2d5845aef9fa4a84a11da
PluginImporter:
externalObjects: {}
serializedVersion: 2
iconMap: {}
executionOrder: {}
defineConstraints: []
isPreloaded: 0
isOverridable: 0
isExplicitlyReferenced: 0
validateReferences: 1
platformData:
- first:
Android: Android
second:
enabled: 1
settings: {}
- first:
Any:
second:
enabled: 0
settings: {}
- first:
Editor: Editor
second:
enabled: 0
settings:
DefaultValueInitialized: true
userData:
assetBundleName:
assetBundleVariant:

@ -0,0 +1,8 @@
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android" package="com.google.unity.ads" android:versionName="1.0" android:versionCode="1">
<application>
<uses-library android:required="false" android:name="org.apache.http.legacy" />
<meta-data android:name="com.google.android.gms.ads.APPLICATION_ID" android:value="ca-app-pub-1518268979235646~4802074656" />
<meta-data android:name="com.google.unity.ads.UNITY_VERSION" android:value="2022.3.49f1" />
</application>
</manifest>

@ -0,0 +1,5 @@
android {
packagingOptions {
pickFirst "META-INF/kotlinx_coroutines_core.version"
}
}

@ -0,0 +1,97 @@
// Copyright (C) 2023 Google LLC
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
import groovy.util.XmlSlurper
import groovy.xml.XmlUtil
import java.util.zip.ZipEntry
import java.util.zip.ZipOutputStream
configurations {
// Configuration used to resolve the artifacts of dependencies.
aarArtifacts.extendsFrom implementation
}
/**
* Validates the Unity GMA plugin dependencies.
* Add the following snippet to Assets/Plugins/Android/mainTemplate.gradle in the Unity Editor or
* unityLibrary/build.gradle in an Android project to use this script:
* <pre>{@code
* gradle.projectsEvaluated {
* apply from: 'GoogleMobileAdsPlugin.androidlib/validate_dependencies.gradle'
* }
* }</pre>
*/
task validateDependencies {
def expandedArchiveDirectory
// List of artifacts resolved from the aarArtifacts configuration.
project.configurations.aarArtifacts.
resolvedConfiguration.lenientConfiguration.
getArtifacts(Specs.satisfyAll()).findResults {
ResolvedArtifact artifact ->
File artifactTargetFile = new File(artifact.file.parent , artifact.file.name)
// Desired artifact - com.google.android.gms:play-services-ads-lite:22.4.0
// Group ID - com.google.android.gms
// Artifact ID - play-services-ads-lite
// Since Gradle has different naming convention for the same artifact in
// * modules-2 cache - play-services-ads-lite-22.4.0.aar
// * transforms-2 cache - com.google.android.gms.play-services-ads-lite-22.4.0
// we look for the common segment.
if (artifact.name.contains("play-services-ads-lite")) {
// Explode the archive to a temporary directory.
FileTree expandedArchive = project.zipTree(artifactTargetFile)
expandedArchive.forEach { File androidManifest ->
if (androidManifest.getName() == "AndroidManifest.xml") {
def xml = new XmlSlurper().parse(androidManifest)
def propertyNode = xml.depthFirst().find { it.name() == 'property' }
if (propertyNode) {
// Replace the <property> node with a comment.
propertyNode.replaceNode {
mkp.comment 'android.adservices.AD_SERVICES_CONFIG property'\
+ ' removed by GoogleMobileAds Unity plugin - Release notes: '\
+ 'https://github.com/googleads/googleads-mobile-unity/releases/'\
+ 'tag/v8.6.0'
}
}
def updatedXml = XmlUtil.serialize(xml)
androidManifest.setWritable(true)
androidManifest.text = updatedXml
expandedArchiveDirectory = androidManifest.parent
}
}
// Update the artifact archive.
artifactTargetFile.withOutputStream { outputStream ->
def zipStream = new ZipOutputStream(outputStream)
file(expandedArchiveDirectory).eachFileRecurse { file ->
if (file.isFile()) {
def entry = new ZipEntry(file.name)
zipStream.putNextEntry(entry)
file.withInputStream { zipStream << it }
zipStream.closeEntry()
}
}
zipStream.close()
}
}
}
// Clean up the temporary directory.
if (expandedArchiveDirectory) delete expandedArchiveDirectory
}
// Run the update task before unityLibrary project is built.
project(':unityLibrary:GoogleMobileAdsPlugin.androidlib') {
tasks.named('preBuild') {
dependsOn validateDependencies
}
}

@ -0,0 +1,32 @@
fileFormatVersion: 2
guid: ff8209b689cd4e54b844eda1ce5184c6
PluginImporter:
externalObjects: {}
serializedVersion: 2
iconMap: {}
executionOrder: {}
defineConstraints: []
isPreloaded: 0
isOverridable: 0
isExplicitlyReferenced: 0
validateReferences: 1
platformData:
- first:
Android: Android
second:
enabled: 1
settings: {}
- first:
Any:
second:
enabled: 0
settings: {}
- first:
Editor: Editor
second:
enabled: 0
settings:
DefaultValueInitialized: true
userData:
assetBundleName:
assetBundleVariant:

Some files were not shown because too many files have changed in this diff Show More

Loading…
Cancel
Save