namespace Guru { using UnityEngine; using System; using System.Collections; using System.Collections.Generic; using System.IO; public partial class GuruSDK: MonoBehaviour { public const string Version = "1.0.9"; public const string Tag = "[Guru]"; public const string ServicesConfigKey = "guru_services"; private static GuruSDK _instance; /// /// 单利引用 /// public static GuruSDK Instance { get { if(null == _instance) { _instance = CreateInstance(); } return _instance; } } private GuruSDKInitConfig _initConfig; private Action _onCompleteCallback; private static GuruSDKModel _model; internal static GuruSDKInitConfig InitConfig => Instance._initConfig; internal static GuruSDKModel Model => GuruSDKModel.Instance; private static GuruServicesConfig _appServicesConfig; private static GuruSettings _guruSettings; private static GuruSettings GuruSettings { get { if (_guruSettings == null) _guruSettings = GuruSettings.Instance; return _guruSettings; } } private static DateTime _initTime; private static bool _isDebugEnabled = false; /// /// Debug Mode /// public static bool IsDebugMode { get { #if UNITY_EDITOR || DEBUG return true; #endif return _isDebugEnabled; } } /// /// 初始化成功标志位 /// public static bool IsInitialSuccess { get; private set; } = false; #region 初始化 private static GuruSDK CreateInstance() { var go = new GameObject(nameof(GuruSDK)); DontDestroyOnLoad(go); _instance = go.AddComponent(); return _instance; } public static GuruSDKInitConfig BuildConfig( bool useCustomConsent = false, bool autoLoadAds = true, bool iapEnabled = true, bool autoRecordFinishedLevels = true, bool debugMode = false, Dictionary defaultRemoteData = null, byte[] googleKeys = null, byte[] appleRootCerts = null) { var config = GuruSDKInitConfig.Build(useCustomConsent, autoLoadAds, iapEnabled, autoRecordFinishedLevels, debugMode, defaultRemoteData, googleKeys, appleRootCerts); return config; } public static void Init(Action onComplete) { Init(GuruSDKInitConfig.Build(), onComplete); } public static void Init(GuruSDKInitConfig config, Action onComplete) { _initTime = DateTime.Now.ToUniversalTime(); LogI($"#1 ---- Guru SDK init ----\n{config.ToString()}"); Instance.StartWithConfig(config, onComplete); } /// /// 启动SDK /// /// /// private void StartWithConfig(GuruSDKInitConfig config, Action onComplete) { Model.PropBLevel.OnValueChanged += OnBLevelChanged; Model.PropBPlay.OnValueChanged += OnBPlayChanged; IsInitialSuccess = false; _initConfig = config; _onCompleteCallback = onComplete; _isDebugEnabled = config.DebugMode; } void Start() { //---- Init All tools ---- LogI($"#2 --- InitFirebase ---"); //---------- Start Firebase ------------ FirebaseUtil.onInitComplete += OnFirebaseReady; FirebaseUtil.InitFirebase(null); // 确保所有的逻辑提前被调用到 LogI($"#2.1 --- InitFacebook ---"); //---------- Start Facebook ------------ FBService.Instance.StartService(); LogI($"#2.2 --- Call SDK init complete -> callback: { (_onCompleteCallback == null ? "Null" : _onCompleteCallback.ToString()) } ---"); IsInitialSuccess = true; _onCompleteCallback?.Invoke(true); } /// /// 开始各种组件初始化 /// private void OnFirebaseReady(bool success) { FirebaseUtil.onInitComplete -= OnFirebaseReady; LogI($"#3 --- On FirebaseDeps:{success} ---"); Callbacks.SDK._onFirebaseReady?.Invoke(success); LogFirebaseDeps(success); LogI($"#3.5 --- Call InitRemoteConfig ---"); // 开始Remote Manager初始化 RemoteConfigManager.Init(BuildDefaultRemoteData(_initConfig.DefaultRemoteData)); RemoteConfigManager.OnFetchCompleted += OnFetchRemoteCallback; LogI($"#4 --- Apply remote services config ---"); // 根据缓存的云控配置来初始化参数 InitAllServices(); LogI($"#5 --- sync sdk time ---"); var sp = DateTime.Now.ToUniversalTime() - _initTime; LogSDKInitTime(sp.TotalSeconds); } /// /// 注入云控参数基础数据 /// /// /// private Dictionary BuildDefaultRemoteData(Dictionary dict) { if (dict == null) dict = new Dictionary(3); // 注入默认的 Services 配置值 string json = Model.LoadDefaltServicesConfigJson(); if (!string.IsNullOrEmpty(json)) dict[ServicesConfigKey] = json; return dict; } /// /// 拉取云控参数完成 /// /// private void OnFetchRemoteCallback(bool success) { LogI($"#6 --- Remote fetch complete: {success} ---"); ABTestManager.Init(); // 启动AB测试解析器 Callbacks.Remote._onRemoteFetchComplete?.Invoke(success); } #endregion #region App Remote Update /// /// Apply Cloud guru-service configs for sdk assets /// private void InitAllServices() { bool useKeywords = false; bool useIAP = true; bool appleReview = false; var services = GetRemoteServicesConfig(); if (services != null) { _appServicesConfig = services; useKeywords = _appServicesConfig.KeywordsEnabled(); useIAP = _appServicesConfig.IsIAPEnabled(); Try(() => { LogI($"#4.1 --- Init apply services ---"); //---------------------------------------------------------------- // 自打点设置错误上报 // GuruAnalytics.EnableErrorLog = _appServicesConfig.EnableErrorLog(); // adjust 事件设置 if (null != _appServicesConfig.adjust_settings && null != GuruSettings) { // 更新 Adjust Tokens GuruSettings.UpdateAdjustTokens( _appServicesConfig.adjust_settings.AndroidToken(), _appServicesConfig.adjust_settings.iOSToken()); // 更新 Adjust Events GuruSettings.UpdateAdjustEvents(_appServicesConfig.adjust_settings.events); } LogI($"#4.2 --- Init GuruSttings ---"); // GuruSettings 设置 if (null != _appServicesConfig.app_settings) { if (_appServicesConfig.Tch02Value() > 0) { Analytics.EnableTch02Event = true; Analytics.SetTch02TargetValue(_appServicesConfig.Tch02Value()); } // 设置获取设备 UUID 的方法 if (_appServicesConfig.UseUUID()) { IPMConfig.UsingUUID = true; // 开始使用 UUID 作为 DeviceID 标识 } #if UNITY_IOS // 苹果审核标志位 appleReview = _appServicesConfig.IsAppReview(); #endif if (null != GuruSettings) { // 更新和升级 GuruSettings 对应的值 GuruSettings.UpdateAppSettings( _appServicesConfig.app_settings.bundle_id, _appServicesConfig.fb_settings?.fb_app_id ?? "", _appServicesConfig.app_settings.support_email, _appServicesConfig.app_settings.privacy_url, _appServicesConfig.app_settings.terms_url, _appServicesConfig.app_settings.android_store, _appServicesConfig.app_settings.ios_store); } } //--------------------------------- }, ex => { Debug.LogError($"--- ERROR on apply services: {ex.Message}"); }); } if (useIAP) { // InitIAP(_initConfig.GoogleKeys, _initConfig.AppleRootCerts); // 初始化IAP Try(() => { LogI($"#4.3 --- Init IAP ---"); InitIAP(_initConfig.GoogleKeys, _initConfig.AppleRootCerts); // 初始化IAP }, ex => { Debug.LogError($"--- ERROR on useIAP: {ex.Message}"); }); } if (useKeywords) { // KeywordsManager.Install(Model.IsIAPUser, Model.SuccessLevelCount); // 启动Keyword管理器 Try(() => { LogI($"#4.4 --- Init Keywords ---"); KeywordsManager.Install(Model.IsIAPUser, Model.SuccessLevelCount); // 启动Keyword管理器 }, ex => { Debug.LogError($"--- ERROR on Keywords: {ex.Message}"); }); } #if UNITY_IOS if (appleReview) { // StartAppleReviewFlow(); // 直接显示 ATT 弹窗, 跳过 Consent 流程 Try(() => { LogI($"#4.5.0 --- StartAppleReviewFlow ---"); StartAppleReviewFlow(); // 直接显示 ATT 弹窗, 跳过 Consent 流程 }, ex => { Debug.LogError($"--- ERROR on StartAppleReviewFlow: {ex.Message}"); }); return; } #endif if (!InitConfig.UseCustomConsent && !appleReview) { // LogI($"--- #3 Start Consent Flow ---"); // StartConsentFlow(); Try(() => { LogI($"#4.5 --- StartConsentFlow ---"); StartConsentFlow(); }, ex => { Debug.LogError($"--- ERROR on StartConsentFlow: {ex.Message}"); }); } // 中台服务初始化结束 Callbacks.SDK._onGuruServiceReady?.Invoke(); } /// /// Get the guru-service cloud config value; /// User can fetch the cloud guru-service config by using Custom Service Key /// /// private GuruServicesConfig GetRemoteServicesConfig() { string key = ServicesConfigKey; if (!string.IsNullOrEmpty(_initConfig.CustomServiceKey)) { key = _initConfig.CustomServiceKey; } var json = GetRemoteString(key); if (!string.IsNullOrEmpty(json)) { return JsonParser.ToObject(json); } return null; } private void Try(Action method, Action onException = null, Action onFinal = null) { if (method == null) return; try { method.Invoke(); } catch (Exception ex) { LogException(ex); // ignored onException?.Invoke(ex); } finally { // Finally onFinal?.Invoke(); } } #endregion #region Apple 审核流程逻辑 #if UNITY_IOS private void StartAppleReviewFlow() { CheckAttStatus(); } #endif #endregion #region 数据 private void OnBLevelChanged(int blevel) { SetUserBLevel(blevel); } private void OnBPlayChanged(int bplay) { SetUserBPlay(bplay); } #endregion #region Misc /// /// 打开页面 /// /// public static void OpenURL(string url) { GuruWebview.OpenPage(url); } #endregion #region Logging internal static void LogI(object message) { Debug.Log($"{Tag} {message}"); } internal static void LogW(object message) { Debug.LogWarning($"{Tag} {message}"); } internal static void LogE(object message) { Debug.LogError($"{Tag} {message}"); } internal static void LogException(string message) { LogException( new Exception($"{Tag} {message}")); } internal static void LogException(Exception e) { Debug.LogException(e); } /// /// 上报崩溃信息 /// /// public static void Report(string message) { Analytics.LogCrashlytics(message, false); } /// /// 上报异常 /// /// public static void ReportException(string message) { Analytics.LogCrashlytics(message); } /// /// 上报异常 Exception /// /// public static void ReportException(Exception ex) { Analytics.LogCrashlytics(ex); } #endregion #region 生命周期 /// /// 暂停时处理 /// /// private void OnAppPauseHandler(bool paused) { if(paused) Model.Save(true); // 强制保存数据 Callbacks.App._onAppPaused?.Invoke(paused); } private void OnApplicationPause(bool paused) { OnAppPauseHandler(paused); } private void OnApplicationFocus(bool hasFocus) { OnAppPauseHandler(!hasFocus); } private void OnApplicationQuit() { Model.Save(true); Callbacks.App._onAppQuit?.Invoke(); } #endregion #region 延迟处理 /// /// 启动协程 /// /// /// public static Coroutine DoCoroutine(IEnumerator enumerator) { return Instance != null ? Instance.StartCoroutine(enumerator) : null; } public static void KillCoroutine(Coroutine coroutine) { if(coroutine != null) Instance.StopCoroutine(coroutine); } /// /// 延时执行 /// /// /// public static Coroutine Delay(float seconds, Action callback) { return DoCoroutine(Instance.OnDelayCall(seconds, callback)); } private IEnumerator OnDelayCall(float delay, Action callback) { if (delay > 0) { yield return new WaitForSeconds(delay); } else { yield return null; } callback?.Invoke(); } #endregion } }