| 
									
										
										
										
											2025-09-18 10:30:57 +00:00
										 |  |  |  | #if UNITY_PURCHASE | 
					
						
							|  |  |  |  | using System; | 
					
						
							|  |  |  |  | using System.Collections; | 
					
						
							|  |  |  |  | using System.Collections.Generic; | 
					
						
							|  |  |  |  | using Unity.Services.Core; | 
					
						
							|  |  |  |  | using Unity.Services.Core.Environments; | 
					
						
							|  |  |  |  | using UnityEngine; | 
					
						
							|  |  |  |  | using UnityEngine.Purchasing; | 
					
						
							|  |  |  |  | using UnityEngine.Purchasing.Extension; | 
					
						
							|  |  |  |  | using static WZ.IAPOrderManager; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | namespace WZ | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     public class IAPPurchaseManager : D_MonoSingleton<IAPPurchaseManager>, IDetailedStoreListener | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |         #region API | 
					
						
							|  |  |  |  |         public void BuyProductByID(string productId, string productName, string gameExtraParam) | 
					
						
							|  |  |  |  |         { | 
					
						
							| 
									
										
										
										
											2025-09-18 13:49:57 +00:00
										 |  |  |  | #if UNITY_EDITOR | 
					
						
							| 
									
										
										
										
											2025-09-18 10:30:57 +00:00
										 |  |  |  |             RushSDKManager.Instance.OnPurchaseComplete(new PurchaseInfo( | 
					
						
							|  |  |  |  |                 productName: productName, | 
					
						
							|  |  |  |  |                 productID: productId, | 
					
						
							|  |  |  |  |                 orderID: "", | 
					
						
							|  |  |  |  |                 currency: "USD", | 
					
						
							|  |  |  |  |                 price: "", | 
					
						
							|  |  |  |  |                 gameExtra: gameExtraParam, | 
					
						
							|  |  |  |  |                 failReason: "", | 
					
						
							|  |  |  |  |                 orderAlreadyExists: false, | 
					
						
							|  |  |  |  |                 purchaseResult: true, | 
					
						
							|  |  |  |  |                 resultType: IAPResultType.PurchasingSuccess, | 
					
						
							|  |  |  |  |                 productType: "" | 
					
						
							|  |  |  |  |             )); | 
					
						
							|  |  |  |  | #else | 
					
						
							|  |  |  |  |             _productName = productName; | 
					
						
							|  |  |  |  |             _gameExtraParam = gameExtraParam; | 
					
						
							|  |  |  |  |             Product m_p = GetProductInfoByID(productId); | 
					
						
							|  |  |  |  |             var currencyCode = ""; | 
					
						
							|  |  |  |  |             var localizedPrice = ""; | 
					
						
							|  |  |  |  |             if (m_p != null) | 
					
						
							|  |  |  |  |             { | 
					
						
							|  |  |  |  |                 currencyCode = m_p.metadata.isoCurrencyCode; | 
					
						
							|  |  |  |  |                 localizedPrice = m_p.metadata.localizedPrice.ToString(); | 
					
						
							|  |  |  |  |             } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |             IAPEvent.LogIAPButtonClick(new PurchaseInfo( | 
					
						
							|  |  |  |  |                         productName: _productName, | 
					
						
							|  |  |  |  |                         productID: productId, | 
					
						
							|  |  |  |  |                         orderID: "", | 
					
						
							|  |  |  |  |                         currency: currencyCode, | 
					
						
							|  |  |  |  |                         price: localizedPrice, | 
					
						
							|  |  |  |  |                         gameExtra: _gameExtraParam, | 
					
						
							|  |  |  |  |                         failReason: "", | 
					
						
							|  |  |  |  |                         orderAlreadyExists: false, | 
					
						
							|  |  |  |  |                         purchaseResult: false, | 
					
						
							|  |  |  |  |                         resultType: IAPResultType.NULL, | 
					
						
							|  |  |  |  |                         productType: "")); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |             if (IsInitialized()) | 
					
						
							|  |  |  |  |             { | 
					
						
							|  |  |  |  |                 if (_inPurchaseProgress) | 
					
						
							|  |  |  |  |                 { | 
					
						
							|  |  |  |  |                     LoggerUtils.Debug("[iap] The payment is in progress, please do not initiate the payment repeatedly."); | 
					
						
							|  |  |  |  |                     return; | 
					
						
							|  |  |  |  |                 } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |                 Product product = _storeController.products.WithID(productId); | 
					
						
							|  |  |  |  |                 if (product != null && product.availableToPurchase) | 
					
						
							|  |  |  |  |                 { | 
					
						
							|  |  |  |  |                     _inPurchaseProgress = true; | 
					
						
							|  |  |  |  |                     LoggerUtils.Debug( | 
					
						
							|  |  |  |  |                         string.Format("[iap] Purchasing product asychronously: '{0}'", product.definition.id)); | 
					
						
							| 
									
										
										
										
											2025-09-18 13:49:57 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-18 10:30:57 +00:00
										 |  |  |  |                     if (_googlePlayConfiguration != null) | 
					
						
							|  |  |  |  |                     { | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |                         if (!string.IsNullOrEmpty(productName)) | 
					
						
							|  |  |  |  |                         { | 
					
						
							|  |  |  |  |                             _googlePlayConfiguration.SetObfuscatedAccountId(productName); | 
					
						
							|  |  |  |  |                         } | 
					
						
							|  |  |  |  |                         _googlePlayConfiguration.SetObfuscatedProfileId(gameExtraParam); | 
					
						
							|  |  |  |  |                         LoggerUtils.Debug($"[iap] [BuyProductByID] 设置成功 userId = {productName} profileId = {gameExtraParam}"); | 
					
						
							|  |  |  |  |                     } | 
					
						
							|  |  |  |  |                     _storeController.InitiatePurchase(product); | 
					
						
							|  |  |  |  |                 } | 
					
						
							|  |  |  |  |                 else | 
					
						
							|  |  |  |  |                 { | 
					
						
							|  |  |  |  |                     IAPEvent.LogPurchaseFail(new PurchaseInfo( | 
					
						
							|  |  |  |  |                         productName: _productName, | 
					
						
							|  |  |  |  |                         productID: productId, | 
					
						
							|  |  |  |  |                         orderID: "", | 
					
						
							|  |  |  |  |                         currency: currencyCode, | 
					
						
							|  |  |  |  |                         price: localizedPrice, | 
					
						
							|  |  |  |  |                         gameExtra: _gameExtraParam, | 
					
						
							|  |  |  |  |                         failReason: "BuyProductID FAIL. Not purchasing product, either is not found or is not available for purchase", | 
					
						
							|  |  |  |  |                         orderAlreadyExists: false, | 
					
						
							|  |  |  |  |                         purchaseResult: false, | 
					
						
							|  |  |  |  |                         resultType: IAPResultType.NULL, | 
					
						
							|  |  |  |  |                         productType: product.definition.type.ToString())); | 
					
						
							|  |  |  |  |                     LoggerUtils.Debug("BuyProductID: FAIL. Not purchasing product, either is not found or is not available for purchase"); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-18 13:49:57 +00:00
										 |  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2025-09-18 10:30:57 +00:00
										 |  |  |  |             } | 
					
						
							|  |  |  |  |             else | 
					
						
							|  |  |  |  |             { | 
					
						
							|  |  |  |  |                 RushSDKManager.Instance.OnPurchaseComplete(new PurchaseInfo( | 
					
						
							|  |  |  |  |                     productName: productName, | 
					
						
							|  |  |  |  |                     productID: productId, | 
					
						
							|  |  |  |  |                     orderID: "", | 
					
						
							|  |  |  |  |                     currency: "", | 
					
						
							|  |  |  |  |                     price: "", | 
					
						
							|  |  |  |  |                     gameExtra: _gameExtraParam, | 
					
						
							|  |  |  |  |                     failReason: "BuyProductID FAIL. IAP Not initialized or Not add product.", | 
					
						
							|  |  |  |  |                     orderAlreadyExists: false, | 
					
						
							|  |  |  |  |                     purchaseResult: false, | 
					
						
							|  |  |  |  |                     resultType: IAPResultType.PurchasingUnavailable, | 
					
						
							|  |  |  |  |                     productType: "")); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |                 IAPEvent.LogPurchaseFail(new PurchaseInfo( | 
					
						
							|  |  |  |  |                         productName: _productName, | 
					
						
							|  |  |  |  |                         productID: productId, | 
					
						
							|  |  |  |  |                         orderID: "", | 
					
						
							|  |  |  |  |                         currency: currencyCode, | 
					
						
							|  |  |  |  |                         price: localizedPrice, | 
					
						
							|  |  |  |  |                         gameExtra: _gameExtraParam, | 
					
						
							|  |  |  |  |                         failReason: "BuyProductID FAIL. Not purchasing product, either is not found or is not available for purchase", | 
					
						
							|  |  |  |  |                         orderAlreadyExists: false, | 
					
						
							|  |  |  |  |                         purchaseResult: false, | 
					
						
							|  |  |  |  |                         resultType: IAPResultType.NULL, | 
					
						
							|  |  |  |  |                         productType: "")); | 
					
						
							|  |  |  |  |                 LoggerUtils.Debug("[iap] BuyProductID FAIL. IAP Not initialized or Not add product."); | 
					
						
							|  |  |  |  |                 LoggerUtils.Debug("[iap] OnPurchaseFailed -> productId : " + productId + " , transactionID : " + "" + " , localizedPrice : " + "" + " , isoCurrencyCode : " + ""); | 
					
						
							|  |  |  |  |             } | 
					
						
							|  |  |  |  | #endif | 
					
						
							|  |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2025-09-18 13:49:57 +00:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |         public string GetProductPriceByID(string pID) | 
					
						
							|  |  |  |  |         { | 
					
						
							|  |  |  |  |             if (_storeController == null && _storeExtensionProvider == null) | 
					
						
							|  |  |  |  |                 return ""; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |             Product[] tProducts = _storeController.products.all; | 
					
						
							|  |  |  |  |             for (int i = 0; i < tProducts.Length; i++) | 
					
						
							|  |  |  |  |             { | 
					
						
							|  |  |  |  |                 Product tItem = tProducts[i]; | 
					
						
							|  |  |  |  |                 if (tItem.definition.id.Equals(pID)) | 
					
						
							|  |  |  |  |                 { | 
					
						
							|  |  |  |  |                     return tItem.metadata.GetGoogleProductMetadata().localizedPriceString; | 
					
						
							|  |  |  |  |                 } | 
					
						
							|  |  |  |  |             } | 
					
						
							|  |  |  |  |             return ""; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-18 10:30:57 +00:00
										 |  |  |  |         public Product GetProductInfoByID(string pID) | 
					
						
							|  |  |  |  |         { | 
					
						
							|  |  |  |  |             if (_storeController == null && _storeExtensionProvider == null) | 
					
						
							|  |  |  |  |                 return null; | 
					
						
							|  |  |  |  |             for (int i = 0; i < _storeController.products.all.Length; i++) | 
					
						
							|  |  |  |  |             { | 
					
						
							|  |  |  |  |                 Product tItem = _storeController.products.all[i]; | 
					
						
							|  |  |  |  |                 if (tItem.definition.id.Equals(pID)) | 
					
						
							|  |  |  |  |                 { | 
					
						
							|  |  |  |  |                     return tItem; | 
					
						
							|  |  |  |  |                 } | 
					
						
							|  |  |  |  |             } | 
					
						
							|  |  |  |  |             return null; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-18 13:49:57 +00:00
										 |  |  |  |         public void AddProductsDynamic(Dictionary<string, ProductType> products, Action<bool, string> onProductsResult = null) | 
					
						
							| 
									
										
										
										
											2025-09-18 10:30:57 +00:00
										 |  |  |  |         { | 
					
						
							|  |  |  |  |             _initProductDic = products; | 
					
						
							|  |  |  |  |             FetchAdditionalProducts(products, onProductsResult); | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-18 13:49:57 +00:00
										 |  |  |  |         public void DoConfirmPendingPurchaseByID(string productId) | 
					
						
							|  |  |  |  |         { | 
					
						
							|  |  |  |  |             Product product = _storeController.products.WithID(productId); | 
					
						
							|  |  |  |  |             if (null == product) return; | 
					
						
							|  |  |  |  |             if (string.IsNullOrEmpty(product.transactionID)) return; | 
					
						
							|  |  |  |  |             if (product != null && product.availableToPurchase) | 
					
						
							|  |  |  |  |             { | 
					
						
							|  |  |  |  |                 _storeController.ConfirmPendingPurchase(product); | 
					
						
							|  |  |  |  |                 _inPurchaseProgress = false; | 
					
						
							|  |  |  |  |             } | 
					
						
							|  |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2025-09-18 10:30:57 +00:00
										 |  |  |  |         #endregion | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         #region  购买成功 | 
					
						
							|  |  |  |  |         public PurchaseProcessingResult ProcessPurchase(PurchaseEventArgs purchaseEvent) | 
					
						
							|  |  |  |  |         { | 
					
						
							|  |  |  |  |             _inPurchaseProgress = false; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |             LoggerUtils.Debug("[iap] Purchase OK: " + purchaseEvent.purchasedProduct.definition.id); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |             var wrapper = (Dictionary<string, object>)MiniJson.JsonDecode(purchaseEvent.purchasedProduct.receipt); | 
					
						
							|  |  |  |  |             var payload = (string)wrapper["Payload"]; | 
					
						
							|  |  |  |  |             var profileId = _gameExtraParam; | 
					
						
							|  |  |  |  |             var _productName = ""; | 
					
						
							|  |  |  |  |             var payloadObj = (Dictionary<string, object>)MiniJson.JsonDecode(payload); | 
					
						
							|  |  |  |  |             var o = (string)payloadObj["json"]; | 
					
						
							|  |  |  |  |             var payloadData = (Dictionary<string, object>)MiniJson.JsonDecode(o); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |             if (payloadData.TryGetValue("obfuscatedAccountId", out var obfuscatedAccountIdValue)) | 
					
						
							|  |  |  |  |             { | 
					
						
							|  |  |  |  |                 var obfuscatedAccountId = (string)obfuscatedAccountIdValue; | 
					
						
							|  |  |  |  |                 if (!string.IsNullOrEmpty(obfuscatedAccountId)) | 
					
						
							|  |  |  |  |                 { | 
					
						
							|  |  |  |  |                     _productName = obfuscatedAccountId; | 
					
						
							|  |  |  |  |                 } | 
					
						
							|  |  |  |  |             } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |             if (payloadData.TryGetValue("obfuscatedProfileId", out var value)) | 
					
						
							|  |  |  |  |             { | 
					
						
							|  |  |  |  |                 var obfuscatedProfileId = (string)value; | 
					
						
							|  |  |  |  |                 if (!string.IsNullOrEmpty(obfuscatedProfileId)) | 
					
						
							|  |  |  |  |                 { | 
					
						
							|  |  |  |  |                     profileId = obfuscatedProfileId; | 
					
						
							|  |  |  |  |                 } | 
					
						
							|  |  |  |  |             } | 
					
						
							|  |  |  |  |             LoggerUtils.Debug("[iap] productName" + _productName + " profileId:" + profileId); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |             string token = ""; | 
					
						
							|  |  |  |  |             string orderId = ""; | 
					
						
							|  |  |  |  |             if (Application.platform == RuntimePlatform.Android) | 
					
						
							|  |  |  |  |             { | 
					
						
							|  |  |  |  |                 var gpDetails = (Dictionary<string, object>)MiniJson.JsonDecode(payload); | 
					
						
							|  |  |  |  |                 var gpJson = (string)gpDetails["json"]; | 
					
						
							|  |  |  |  |                 var tokenJson = (Dictionary<string, object>)MiniJson.JsonDecode(gpJson); | 
					
						
							|  |  |  |  |                 token = (string)tokenJson["purchaseToken"]; | 
					
						
							|  |  |  |  |                 orderId = (string)tokenJson["orderId"]; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |                 LoggerUtils.Debug("[iap] ClientIAPSuccess productId : " + purchaseEvent.purchasedProduct.definition.id | 
					
						
							|  |  |  |  |                                     + " , transactionID : " + orderId | 
					
						
							|  |  |  |  |                                     + " , token : " + token | 
					
						
							|  |  |  |  |                                     + " , localizedPrice : " + purchaseEvent.purchasedProduct.metadata.localizedPriceString | 
					
						
							|  |  |  |  |                                     + " , isoCurrencyCode : " + purchaseEvent.purchasedProduct.metadata.isoCurrencyCode); | 
					
						
							|  |  |  |  |             } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |             IAPDataConfig newData = new IAPDataConfig(); | 
					
						
							|  |  |  |  |             newData.info = new Dictionary<string, string> | 
					
						
							|  |  |  |  |             { | 
					
						
							|  |  |  |  |                 { "price", purchaseEvent.purchasedProduct.metadata.localizedPrice.ToString() }, | 
					
						
							|  |  |  |  |                 { "product_id", purchaseEvent.purchasedProduct.definition.id }, | 
					
						
							|  |  |  |  |                 { "product_name", (string)_productName }, | 
					
						
							|  |  |  |  |                 { "purchase_token", token }, | 
					
						
							|  |  |  |  |                 { "order_id", orderId }, | 
					
						
							|  |  |  |  |                 { "currency", purchaseEvent.purchasedProduct.metadata.isoCurrencyCode }, | 
					
						
							|  |  |  |  |                 { "payment_method", "googleplay" }, | 
					
						
							|  |  |  |  |                 { "product_type", purchaseEvent.purchasedProduct.definition.type.ToString() }, | 
					
						
							|  |  |  |  |                 { "mGameExtraParam", (string)profileId } | 
					
						
							|  |  |  |  |             }; | 
					
						
							|  |  |  |  |             newData.state = IAPDataStateType.def; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |             IAPEvent.LogClientComplete(new PurchaseInfo( | 
					
						
							|  |  |  |  |                 productName: (string)_productName, | 
					
						
							|  |  |  |  |                 productID: purchaseEvent.purchasedProduct.definition.id, | 
					
						
							|  |  |  |  |                 orderID: orderId, | 
					
						
							|  |  |  |  |                 currency: purchaseEvent.purchasedProduct.metadata.isoCurrencyCode, | 
					
						
							|  |  |  |  |                 price: purchaseEvent.purchasedProduct.metadata.localizedPrice.ToString(), | 
					
						
							|  |  |  |  |                 gameExtra: _gameExtraParam, | 
					
						
							|  |  |  |  |                 failReason: "", | 
					
						
							|  |  |  |  |                 orderAlreadyExists: false, | 
					
						
							|  |  |  |  |                 purchaseResult: true, | 
					
						
							|  |  |  |  |                 resultType: IAPResultType.PurchasingSuccess, | 
					
						
							|  |  |  |  |                 productType: purchaseEvent.purchasedProduct.definition.type.ToString() | 
					
						
							|  |  |  |  |             )); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |             IAPOrderManager.Instance.SaveOrder(newData); | 
					
						
							|  |  |  |  |             IAPOrderManager.Instance.VerifyPurchase(newData); | 
					
						
							|  |  |  |  |             return PurchaseProcessingResult.Complete; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         #endregion | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         #region  动态添加商品 | 
					
						
							| 
									
										
										
										
											2025-09-18 13:49:57 +00:00
										 |  |  |  |         private void FetchAdditionalProducts(Dictionary<string, ProductType> ProductDic, Action<bool, string> onProductsResult = null) | 
					
						
							| 
									
										
										
										
											2025-09-18 10:30:57 +00:00
										 |  |  |  |         { | 
					
						
							|  |  |  |  |             if (!IsInitialized()) | 
					
						
							|  |  |  |  |             { | 
					
						
							|  |  |  |  |                 _addProductsDic = ProductDic; | 
					
						
							|  |  |  |  |                 LoggerUtils.Debug("[iap] IAP not init.Now InitUnityPurchase"); | 
					
						
							|  |  |  |  |                 InitUnityPurchase(); | 
					
						
							|  |  |  |  |                 return; | 
					
						
							|  |  |  |  |             } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |             if (_isFetchingAdditionalProducts) | 
					
						
							|  |  |  |  |             { | 
					
						
							|  |  |  |  |                 LoggerUtils.Debug("[iap] Now fetching additional products,don't call repeatedly"); | 
					
						
							|  |  |  |  |                 if (onProductsResult != null) | 
					
						
							|  |  |  |  |                 { | 
					
						
							|  |  |  |  |                     onProductsResult(false, "Now fetching additional products,don't call repeatedly"); | 
					
						
							|  |  |  |  |                 } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |                 return; | 
					
						
							|  |  |  |  |             } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |             _isFetchingAdditionalProducts = true; | 
					
						
							|  |  |  |  |             if (ProductDic != null) | 
					
						
							|  |  |  |  |             { | 
					
						
							|  |  |  |  |                 var additional = new HashSet<ProductDefinition>(); | 
					
						
							|  |  |  |  |                 foreach (string tID in ProductDic.Keys) | 
					
						
							|  |  |  |  |                 { | 
					
						
							|  |  |  |  |                     additional.Add(new ProductDefinition(tID, ProductDic[tID])); | 
					
						
							|  |  |  |  |                 } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |                 Action onSuccess = () => | 
					
						
							|  |  |  |  |                 { | 
					
						
							|  |  |  |  |                     _isFetchingAdditionalProducts = false; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |                     LoggerUtils.Debug("[iap] Fetched successfully!"); | 
					
						
							|  |  |  |  |                     RushSDKManager.Instance.OnGetProductsInfo?.Invoke(_storeController.products.all); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |                     foreach (var product in _storeController.products.all) | 
					
						
							|  |  |  |  |                     { | 
					
						
							|  |  |  |  |                         LoggerUtils.Debug("[iap]" + product.metadata.localizedTitle | 
					
						
							|  |  |  |  |                                             + "|" + product.metadata.localizedPriceString | 
					
						
							|  |  |  |  |                                             + "|" + product.metadata.localizedDescription | 
					
						
							|  |  |  |  |                                             + "|" + product.metadata.isoCurrencyCode); | 
					
						
							|  |  |  |  |                     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |                     if (onProductsResult != null) | 
					
						
							|  |  |  |  |                     { | 
					
						
							|  |  |  |  |                         onProductsResult(true, "Fetched successfully!"); | 
					
						
							|  |  |  |  |                     } | 
					
						
							|  |  |  |  |                 }; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |                 Action<InitializationFailureReason, string> onFailure = (InitializationFailureReason i, string msg) => | 
					
						
							|  |  |  |  |                 { | 
					
						
							|  |  |  |  |                     _isFetchingAdditionalProducts = false; | 
					
						
							|  |  |  |  |                     if (onProductsResult != null) | 
					
						
							|  |  |  |  |                     { | 
					
						
							|  |  |  |  |                         onProductsResult(true, "Fetching failed for the specified reason: " + i + "  msg: " + msg); | 
					
						
							|  |  |  |  |                     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |                     LoggerUtils.Debug("[iap] Fetching failed for the specified reason: " + i + "  msg: " + msg); | 
					
						
							|  |  |  |  |                 }; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |                 _storeController.FetchAdditionalProducts(additional, onSuccess, onFailure); | 
					
						
							|  |  |  |  |             } | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |         #endregion | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         #region  初始化 | 
					
						
							|  |  |  |  |         public void PreInitialize() | 
					
						
							|  |  |  |  |         { | 
					
						
							|  |  |  |  |             LoggerUtils.Debug("[iap] PreInitialize() mServiceInit: " + _serviceInit); | 
					
						
							|  |  |  |  |             if (!_serviceInit) | 
					
						
							|  |  |  |  |             { | 
					
						
							|  |  |  |  |                 InitializeUnityServices(OnSuccess, OnError); | 
					
						
							|  |  |  |  |             } | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         private void InitializeUnityServices(Action onSuccess, Action<string> onError) | 
					
						
							|  |  |  |  |         { | 
					
						
							|  |  |  |  |             try | 
					
						
							|  |  |  |  |             { | 
					
						
							|  |  |  |  |                 var options = new InitializationOptions().SetEnvironmentName("production"); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |                 UnityServices.InitializeAsync(options).ContinueWith(task => onSuccess()); | 
					
						
							|  |  |  |  |             } | 
					
						
							|  |  |  |  |             catch (Exception exception) | 
					
						
							|  |  |  |  |             { | 
					
						
							|  |  |  |  |                 onError(exception.Message); | 
					
						
							|  |  |  |  |             } | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |         private void OnSuccess() | 
					
						
							|  |  |  |  |         { | 
					
						
							|  |  |  |  |             LoggerUtils.Debug("[iap] Congratulations!\nUnity Gaming Services has been successfully initialized."); | 
					
						
							|  |  |  |  |             _serviceInit = true; | 
					
						
							|  |  |  |  |             IAPEvent.LogPurchaseInit(); | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         private void OnError(string message) | 
					
						
							|  |  |  |  |         { | 
					
						
							|  |  |  |  |             LoggerUtils.Debug($"[iap] Unity Gaming Services failed to initialize with error: {message}."); | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         /// <summary> | 
					
						
							|  |  |  |  |         /// 初始化IAP | 
					
						
							|  |  |  |  |         /// </summary> | 
					
						
							|  |  |  |  |         public void Initialize() | 
					
						
							|  |  |  |  |         { | 
					
						
							| 
									
										
										
										
											2025-09-19 14:34:17 +00:00
										 |  |  |  |             LoggerUtils.Debug("[iap] IAP Initialize() _storeController:" + _storeController + "   m_StoreExtensionProvider: " + _storeExtensionProvider); | 
					
						
							| 
									
										
										
										
											2025-09-18 10:30:57 +00:00
										 |  |  |  |             if (_storeController == null && _storeExtensionProvider == null) | 
					
						
							|  |  |  |  |                 InitUnityPurchase(); | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         private void InitUnityPurchase() | 
					
						
							|  |  |  |  |         { | 
					
						
							|  |  |  |  |             LoggerUtils.Debug("[iap] IAP InitUnityPurchase() IsInitialized: " + IsInitialized()); | 
					
						
							|  |  |  |  |             if (IsInitialized()) return; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |             _repeatCountDic = new Dictionary<string, int>(); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |             _module ??= StandardPurchasingModule.Instance(); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |             // 配置模式; | 
					
						
							|  |  |  |  |             _builder ??= ConfigurationBuilder.Instance(_module); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |             int productsNum = 0; | 
					
						
							| 
									
										
										
										
											2025-09-18 13:49:57 +00:00
										 |  |  |  |             foreach (string tID in StaticProductDic.Keys) | 
					
						
							|  |  |  |  |             { | 
					
						
							|  |  |  |  |                 productsNum++; | 
					
						
							|  |  |  |  |                 if (!string.IsNullOrEmpty(tID)) | 
					
						
							|  |  |  |  |                 { | 
					
						
							|  |  |  |  |                     LoggerUtils.Debug($"[iap] Add IAPProducts IAPProducts: {tID}"); | 
					
						
							|  |  |  |  |                     _builder.AddProduct(tID, StaticProductDic[tID]); | 
					
						
							|  |  |  |  |                 } | 
					
						
							|  |  |  |  |             } | 
					
						
							| 
									
										
										
										
											2025-09-18 10:30:57 +00:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |             if (_initProductDic != null && _initProductDic.Count > 0) | 
					
						
							|  |  |  |  |             { | 
					
						
							|  |  |  |  |                 foreach (string tID in _initProductDic.Keys) | 
					
						
							|  |  |  |  |                 { | 
					
						
							|  |  |  |  |                     productsNum++; | 
					
						
							|  |  |  |  |                     if (!string.IsNullOrEmpty(tID)) | 
					
						
							|  |  |  |  |                     { | 
					
						
							|  |  |  |  |                         LoggerUtils.Debug($"[iap] Add InitProductDic APProducts: {tID}"); | 
					
						
							|  |  |  |  |                         _builder.AddProduct(tID, _initProductDic[tID]); | 
					
						
							|  |  |  |  |                     } | 
					
						
							|  |  |  |  |                 } | 
					
						
							|  |  |  |  |             } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |             if (_addProductsDic != null && _addProductsDic.Count > 0) | 
					
						
							|  |  |  |  |             { | 
					
						
							|  |  |  |  |                 foreach (string tID in _addProductsDic.Keys) | 
					
						
							|  |  |  |  |                 { | 
					
						
							|  |  |  |  |                     productsNum++; | 
					
						
							|  |  |  |  |                     if (!string.IsNullOrEmpty(tID)) | 
					
						
							|  |  |  |  |                     { | 
					
						
							|  |  |  |  |                         LoggerUtils.Debug($"[iap] Add AddProductsDic IAPProducts: {tID}"); | 
					
						
							|  |  |  |  |                         _builder.AddProduct(tID, _addProductsDic[tID]); | 
					
						
							|  |  |  |  |                     } | 
					
						
							|  |  |  |  |                 } | 
					
						
							|  |  |  |  |             } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |             _googlePlayConfiguration = _builder.Configure<IGooglePlayConfiguration>(); | 
					
						
							|  |  |  |  |             if (productsNum > 0) | 
					
						
							|  |  |  |  |             { | 
					
						
							|  |  |  |  |                 UnityPurchasing.Initialize(this, _builder); | 
					
						
							|  |  |  |  |             } | 
					
						
							|  |  |  |  |             else | 
					
						
							|  |  |  |  |             { | 
					
						
							|  |  |  |  |                 LoggerUtils.Debug( | 
					
						
							|  |  |  |  |                     "[iap] UnityPurchasing will not initialize.Products is empty,please add product."); | 
					
						
							|  |  |  |  |             } | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         private bool IsInitialized() | 
					
						
							|  |  |  |  |         { | 
					
						
							|  |  |  |  |             return _storeController != null && _storeExtensionProvider != null; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |         #endregion | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         #region  初始化成功 | 
					
						
							|  |  |  |  |         public void OnInitialized(IStoreController controller, IExtensionProvider extensions) | 
					
						
							|  |  |  |  |         { | 
					
						
							|  |  |  |  |             LoggerUtils.Debug("[iap] IAP initialize Succeed!"); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |             _storeController = controller; | 
					
						
							|  |  |  |  |             _storeExtensionProvider = extensions; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |             foreach (var product in _storeController.products.all) | 
					
						
							|  |  |  |  |             { | 
					
						
							|  |  |  |  |                 LoggerUtils.Debug("[iap] " + product.metadata.localizedTitle | 
					
						
							|  |  |  |  |                                     + "|" + product.metadata.localizedPriceString | 
					
						
							|  |  |  |  |                                     + "|" + product.metadata.localizedDescription | 
					
						
							|  |  |  |  |                                     + "|" + product.metadata.isoCurrencyCode | 
					
						
							|  |  |  |  |                                     + "|" + product.definition.id | 
					
						
							|  |  |  |  |                                     + "|" + product.definition.type); | 
					
						
							|  |  |  |  |             } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |             RushSDKManager.Instance.OnGetProductsInfo?.Invoke(_storeController.products.all); | 
					
						
							|  |  |  |  |             RushSDKManager.Instance.OnPurchaseInitComplete?.Invoke(true, ""); | 
					
						
							|  |  |  |  |             IAPEvent.LogPurchaseInit(true); | 
					
						
							| 
									
										
										
										
											2025-09-18 13:49:57 +00:00
										 |  |  |  |             IAPSubscribeManager.Instance.CheckSubscribeReceipt(); | 
					
						
							|  |  |  |  |             IAPOrderManager.Instance.RefreshOrderStatue(); | 
					
						
							| 
									
										
										
										
											2025-09-18 10:30:57 +00:00
										 |  |  |  |         } | 
					
						
							|  |  |  |  |         #endregion | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         #region  初始化失败 | 
					
						
							|  |  |  |  |         public void OnInitializeFailed(InitializationFailureReason error) | 
					
						
							|  |  |  |  |         { | 
					
						
							|  |  |  |  |             OnInitializeFailed(error, ""); | 
					
						
							|  |  |  |  |             RushSDKManager.Instance.OnPurchaseInitComplete?.Invoke(false, error.ToString()); | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         public void OnInitializeFailed(InitializationFailureReason error, string message) | 
					
						
							|  |  |  |  |         { | 
					
						
							|  |  |  |  |             LoggerUtils.Debug("[iap] IAP OnInitializeFailed error:" + error.ToString() + " msg:" + message); | 
					
						
							|  |  |  |  |             switch (error) | 
					
						
							|  |  |  |  |             { | 
					
						
							|  |  |  |  |                 case InitializationFailureReason.AppNotKnown: | 
					
						
							|  |  |  |  |                     LoggerUtils.Debug("[iap] Is your App correctly uploaded on the relevant publisher console?"); | 
					
						
							|  |  |  |  |                     break; | 
					
						
							|  |  |  |  |                 case InitializationFailureReason.PurchasingUnavailable: | 
					
						
							|  |  |  |  |                     LoggerUtils.Debug("[iap] Billing disabled! Ask the user if billing is disabled in device settings."); | 
					
						
							|  |  |  |  |                     break; | 
					
						
							|  |  |  |  |                 case InitializationFailureReason.NoProductsAvailable: | 
					
						
							|  |  |  |  |                     LoggerUtils.Debug("[iap] No products available for purchase! Developer configuration error; check product metadata!"); | 
					
						
							|  |  |  |  |                     break; | 
					
						
							|  |  |  |  |             } | 
					
						
							|  |  |  |  |             IAPEvent.LogPurchaseInit(false, error.ToString()); | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |         #endregion | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         #region  购买失败 | 
					
						
							|  |  |  |  |         public void OnPurchaseFailed(Product product, PurchaseFailureDescription failureDescription) | 
					
						
							|  |  |  |  |         { | 
					
						
							| 
									
										
										
										
											2025-09-18 13:49:57 +00:00
										 |  |  |  |             _inPurchaseProgress = false; | 
					
						
							|  |  |  |  |             IAPOrderManager.Instance.ProgressCacheOrder(); | 
					
						
							| 
									
										
										
										
											2025-09-18 10:30:57 +00:00
										 |  |  |  |             LoggerUtils.Debug("[iap] OnPurchaseFailed productId : " + failureDescription.productId | 
					
						
							|  |  |  |  |                                 + " , transactionID : " + product.transactionID | 
					
						
							|  |  |  |  |                                 + " , localizedPrice : " + product.metadata.localizedPriceString | 
					
						
							|  |  |  |  |                                 + " , isoCurrencyCode : " + product.metadata.isoCurrencyCode | 
					
						
							|  |  |  |  |                                 + "failureDescription" + failureDescription.message); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |             // 失败打点 | 
					
						
							|  |  |  |  |             IAPEvent.LogPurchaseFail(new PurchaseInfo( | 
					
						
							|  |  |  |  |                 productName: product.metadata.localizedTitle, | 
					
						
							|  |  |  |  |                 productID: product.definition.id, | 
					
						
							|  |  |  |  |                 orderID: "", | 
					
						
							|  |  |  |  |                 currency: product.metadata.isoCurrencyCode, | 
					
						
							|  |  |  |  |                 price: product.metadata.localizedPriceString, | 
					
						
							|  |  |  |  |                 gameExtra: _gameExtraParam, | 
					
						
							|  |  |  |  |                 failReason: failureDescription.message, | 
					
						
							|  |  |  |  |                 orderAlreadyExists: false, | 
					
						
							|  |  |  |  |                 purchaseResult: false, | 
					
						
							|  |  |  |  |                 resultType: (IAPResultType)failureDescription.reason, | 
					
						
							|  |  |  |  |                 productType: product.definition.type.ToString() | 
					
						
							|  |  |  |  |             )); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |             // 购买失败回调 | 
					
						
							|  |  |  |  |             RushSDKManager.Instance.OnPurchaseComplete(new PurchaseInfo( | 
					
						
							|  |  |  |  |                 productName: product.metadata.localizedTitle, | 
					
						
							|  |  |  |  |                 productID: product.definition.id, | 
					
						
							|  |  |  |  |                 orderID: "", | 
					
						
							|  |  |  |  |                 currency: product.metadata.isoCurrencyCode, | 
					
						
							|  |  |  |  |                 price: product.metadata.localizedPriceString, | 
					
						
							|  |  |  |  |                 gameExtra: _gameExtraParam, | 
					
						
							|  |  |  |  |                 failReason: failureDescription.message, | 
					
						
							|  |  |  |  |                 orderAlreadyExists: false, | 
					
						
							|  |  |  |  |                 purchaseResult: false, | 
					
						
							|  |  |  |  |                 resultType: (IAPResultType)failureDescription.reason, | 
					
						
							|  |  |  |  |                 productType: product.definition.type.ToString() | 
					
						
							|  |  |  |  |             )); | 
					
						
							| 
									
										
										
										
											2025-09-18 13:49:57 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-18 10:30:57 +00:00
										 |  |  |  |         } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         public void OnPurchaseFailed(Product product, PurchaseFailureReason failureReason) | 
					
						
							|  |  |  |  |         { | 
					
						
							| 
									
										
										
										
											2025-09-18 13:49:57 +00:00
										 |  |  |  |             _inPurchaseProgress = false; | 
					
						
							|  |  |  |  |             IAPOrderManager.Instance.ProgressCacheOrder(); | 
					
						
							| 
									
										
										
										
											2025-09-18 10:30:57 +00:00
										 |  |  |  |             LoggerUtils.Debug("[iap] OnPurchaseFailed productId -> : " + product.definition.id | 
					
						
							|  |  |  |  |                                 + " , transactionID : " + product.transactionID | 
					
						
							|  |  |  |  |                                 + " , localizedPrice : " + product.metadata.localizedPriceString | 
					
						
							|  |  |  |  |                                 + " , isoCurrencyCode : " + product.metadata.isoCurrencyCode | 
					
						
							|  |  |  |  |                                 + " , failureReason : " + failureReason.ToString()); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |             // 失败打点 | 
					
						
							|  |  |  |  |             IAPEvent.LogPurchaseFail(new PurchaseInfo( | 
					
						
							|  |  |  |  |                 productName: product.metadata.localizedTitle, | 
					
						
							|  |  |  |  |                 productID: product.definition.id, | 
					
						
							|  |  |  |  |                 orderID: "", | 
					
						
							|  |  |  |  |                 currency: product.metadata.isoCurrencyCode, | 
					
						
							|  |  |  |  |                 price: product.metadata.localizedPriceString, | 
					
						
							|  |  |  |  |                 gameExtra: _gameExtraParam, | 
					
						
							|  |  |  |  |                 productType: "", | 
					
						
							|  |  |  |  |                 failReason: failureReason.ToString(), | 
					
						
							|  |  |  |  |                 orderAlreadyExists: false, | 
					
						
							|  |  |  |  |                 purchaseResult: false, | 
					
						
							|  |  |  |  |                 resultType: (IAPResultType)failureReason) | 
					
						
							|  |  |  |  |             ); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |             // 购买失败回调 | 
					
						
							|  |  |  |  |             RushSDKManager.Instance.OnPurchaseComplete(new PurchaseInfo( | 
					
						
							|  |  |  |  |                 productName: product.metadata.localizedTitle, | 
					
						
							|  |  |  |  |                 productID: product.definition.id, | 
					
						
							|  |  |  |  |                 orderID: "", | 
					
						
							|  |  |  |  |                 currency: product.metadata.isoCurrencyCode, | 
					
						
							|  |  |  |  |                 price: product.metadata.localizedPriceString, | 
					
						
							|  |  |  |  |                 gameExtra: _gameExtraParam, | 
					
						
							|  |  |  |  |                 failReason: failureReason.ToString(), | 
					
						
							|  |  |  |  |                 orderAlreadyExists: false, | 
					
						
							|  |  |  |  |                 purchaseResult: false, | 
					
						
							|  |  |  |  |                 resultType: (IAPResultType)failureReason, | 
					
						
							|  |  |  |  |                 productType: product.definition.type.ToString() | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |             )); | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |         #endregion | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-18 13:49:57 +00:00
										 |  |  |  |         #region 添加静态商品 | 
					
						
							|  |  |  |  |         public static Dictionary<string, ProductType> StaticProductDic = new Dictionary<string, ProductType>() | 
					
						
							|  |  |  |  |         { | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         }; | 
					
						
							|  |  |  |  |         #endregion | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-18 10:30:57 +00:00
										 |  |  |  |         #region  Properties | 
					
						
							|  |  |  |  |         private StandardPurchasingModule _module; | 
					
						
							|  |  |  |  |         private ConfigurationBuilder _builder; | 
					
						
							| 
									
										
										
										
											2025-09-18 13:49:57 +00:00
										 |  |  |  |         public IStoreController _storeController; | 
					
						
							| 
									
										
										
										
											2025-09-18 10:30:57 +00:00
										 |  |  |  |         private static IExtensionProvider _storeExtensionProvider; | 
					
						
							|  |  |  |  |         public string _gameExtraParam = ""; | 
					
						
							|  |  |  |  |         private static string _productName = ""; | 
					
						
							|  |  |  |  |         private bool _inPurchaseProgress = false; | 
					
						
							|  |  |  |  |         private bool _serviceInit = false; | 
					
						
							|  |  |  |  |         private Dictionary<string, ProductType> _addProductsDic; | 
					
						
							|  |  |  |  |         private Dictionary<string, ProductType> _initProductDic; | 
					
						
							| 
									
										
										
										
											2025-09-18 13:49:57 +00:00
										 |  |  |  |         public Dictionary<string, int> _repeatCountDic; | 
					
						
							| 
									
										
										
										
											2025-09-18 10:30:57 +00:00
										 |  |  |  |         private IGooglePlayConfiguration _googlePlayConfiguration; | 
					
						
							|  |  |  |  |         private bool _isFetchingAdditionalProducts = false; | 
					
						
							| 
									
										
										
										
											2025-09-18 13:49:57 +00:00
										 |  |  |  |          | 
					
						
							|  |  |  |  |          | 
					
						
							| 
									
										
										
										
											2025-09-18 10:30:57 +00:00
										 |  |  |  |         #endregion | 
					
						
							| 
									
										
										
										
											2025-09-18 13:49:57 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-18 10:30:57 +00:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | #endif |