This commit is contained in:
luojian 2025-09-02 14:56:54 +08:00
commit c5928b8b64
41 changed files with 679 additions and 455 deletions

View File

@ -1,5 +1,7 @@
using System; using System;
using System.Collections.Generic; using System.Collections.Generic;
using System.Security.Cryptography;
using System.Text;
using UnityEngine; using UnityEngine;
namespace EFSDK namespace EFSDK
@ -10,8 +12,9 @@ namespace EFSDK
public class EFSdk public class EFSdk
{ {
private static EFSdk _mEfSdk; private static EFSdk _mEfSdk;
private static string mappingInfo = @"{""items"":[{""key"":""_sdk_float_balloon.png"",""value"":""aoa38ay.png""}]}";
// 保持变量名不变
private static string mappingInfo = "";
public static EFSdk get() public static EFSdk get()
{ {
@ -28,18 +31,81 @@ namespace EFSDK
public EFSdk() public EFSdk()
{ {
Debug.Log($"GetNewSDKClass():{GetNewSDKClass()}");
// java interface class // java interface class
using (AndroidJavaClass jc = new AndroidJavaClass("com.earn.push._SDK")) using (AndroidJavaClass jc = new AndroidJavaClass(GetNewSDKClass()))
{ {
jo = jc.GetStatic<AndroidJavaObject>("INSTANCE"); jo = jc.GetStatic<AndroidJavaObject>("INSTANCE");
} }
} }
private static string oriSDKPName = "com.earn.push";
private static string oriSDK = "_SDK";
private static string GetNewSDKClass()
{
return GetSDKPackage() + GenerateAndroidName(oriSDK);
}
private static string GetSDKPackage()
{
string[] parts = oriSDKPName.Split('.');
string[] parts2 = new string[parts.Length];
for (int i = 0; i < parts.Length; i++)
{
parts2[i] = GenerateAndroidName(parts[i]);
}
string newPName = "";
for (int i = 0; i < parts2.Length; i++)
{
newPName+=parts2[i]+".";
}
return newPName;
}
private static string GenerateAndroidName(string oriString)
{
string md5Str = GetFirstEightWithUnderscore(GetMD5Hash(Application.identifier + oriString));
return md5Str;
}
public static string GetMD5Hash(string input)
{
using (var md5 = MD5.Create())
{
var inputBytes = Encoding.ASCII.GetBytes(input);
var hashBytes = md5.ComputeHash(inputBytes);
var builder = new StringBuilder();
foreach (var t in hashBytes)
{
builder.Append(t.ToString("x2")); // Convert byte to hexadecimal string
}
return builder.ToString();
}
}
static string GetFirstEightWithUnderscore(string str)
{
if (string.IsNullOrEmpty(str)) return str;
string sub = str.Length <= 8 ? str : str.Substring(0, 8);
if (char.IsDigit(sub[0]))
{
sub = "a" + sub;
}
return sub;
}
private T SDKCall<T>(string _method, params object[] _param) private T SDKCall<T>(string _method, params object[] _param)
{ {
try try
{ {
return jo.Call<T>(_method, _param); string newMethod = GenerateAndroidName(_method);
Debug.Log($"SDKCall<T> newMethod:{newMethod}");
return jo.Call<T>(newMethod, _param);
} }
catch (Exception e) catch (Exception e)
{ {
@ -53,7 +119,9 @@ namespace EFSDK
{ {
try try
{ {
jo.Call(_method, _param); string newMethod = GenerateAndroidName(_method);
Debug.Log($"SDKCall newMethod:{newMethod}");
jo.Call(newMethod, _param);
} }
catch (Exception e) catch (Exception e)
{ {
@ -63,11 +131,8 @@ namespace EFSDK
public enum ActionType public enum ActionType
{ {
COIN_CLICK, //点击金币 SDK_INIT_Succ, //GAM页面加载成功
BALLOON_CLICK, //点击气球 H5_Load_Succ, //H5页面加载成功
COIN_SHOW, //金币展示出来了
BOX_SHOW, //气球/宝箱展示出来了
GAM_LOAD_SUCC, //GAM页面加载成功
ON_RESUME, //游戏可见时回调, ON_RESUME, //游戏可见时回调,
// CAN_GOBACK, //游戏可见时回调, // CAN_GOBACK, //游戏可见时回调,
} }
@ -111,8 +176,8 @@ namespace EFSDK
private void SDKInit() private void SDKInit()
{ {
// SDKCall("init"); SDKCall("initSDK", mappingInfo);
ActionCallback?.Invoke(ActionType.GAM_LOAD_SUCC, string.Empty); ActionCallback?.Invoke(ActionType.SDK_INIT_Succ, string.Empty);
} }
/// <summary> /// <summary>
@ -170,77 +235,6 @@ namespace EFSDK
SDKCall("goHome"); SDKCall("goHome");
} }
/// <summary>
/// 是否手动控制漂浮道具显示/隐藏
/// SDK内默认当H5页面加载完成后自动显示漂浮道具
/// </summary>
/// <param name="autoShow">true: 自动显示/隐藏道具 false: 游戏主动控制道具显示/隐藏</param>
/// <returns></returns>
public void AutoShowFloat(bool autoShow)
{
SDKCall("autoShowFloat", autoShow);
}
/// <summary>
/// 飘金币
/// </summary>
/// <param name="id"></param>
/// <returns></returns>
public void ShowFloatCoin(int id)
{
SDKCall("showFloatCoin", id);
}
/// <summary>
/// 飘金币
/// </summary>
/// <param name="id"></param>
/// <param name="res">悬浮金币按钮的图片资源,传字符串 0 或 1 0:金币图 1:红点宝箱图 </param>
/// <returns></returns>
public void ShowFloatCoin(int id, String res)
{
SDKCall("showFloatCoin", id, res);
}
/// <summary>
/// 设置悬浮金币按钮的图片资源
/// </summary>
/// <param name="res">传字符串 0 或 1 0:金币图 1:红点宝箱图</param>
public void SetFloatCoinRes(String res)
{
SDKCall("setFloatCoinRes", res);
}
/// <summary>
/// 隐藏金币
/// </summary>
/// <returns></returns>
public void HideFloatCoin()
{
SDKCall("hideFloatCoin");
}
/// <summary>
/// 飘气球
/// </summary>
/// <param name="startId"></param>
/// <param name="endId"></param>
/// <param name="fly_first_time"></param>
/// <param name="fly_gap_time"></param>
/// <returns></returns>
public void ShowBalloon(int startId, int endId, int fly_first_time, int fly_gap_time)
{
SDKCall("showBalloon", startId, endId, fly_first_time, fly_gap_time);
}
/// <summary>
/// 隐藏气球
/// </summary>
/// <returns></returns>
public void HideBalloon()
{
SDKCall("hideBalloon");
}
/// <summary> /// <summary>
/// ///
@ -340,14 +334,6 @@ namespace EFSDK
SDKCall("setGameName", gameName); SDKCall("setGameName", gameName);
} }
// /// <summary>
// /// 设置推送 消息通知 的文案
// /// </summary>
// /// <param name="message"></param>
// public void SetCommPushMessage(string message)
// {
// SDKCall("setCommPushMessage", message);
// }
/// <summary> /// <summary>
/// 设置当前游戏语言是否是 西班牙语 /// 设置当前游戏语言是否是 西班牙语

View File

@ -6,11 +6,7 @@ namespace EFSDK
public class EFSdkAndroid : MonoBehaviour public class EFSdkAndroid : MonoBehaviour
{ {
private string COIN_CLICK = "coin_click"; private string H5_Load_Succ = "Gam_Load_Succ";
private string BALLOON_CLICK = "balloon_click";
private string Coin_Show = "Coin_Show";
private string Box_Show = "Box_Show";
private string Gam_Load_Succ = "Gam_Load_Succ";
private string On_Resume = "onResume"; private string On_Resume = "onResume";
private string Can_Goback = "canGoback"; private string Can_Goback = "canGoback";
@ -28,34 +24,11 @@ namespace EFSDK
EFSdk.get().mCanGobackAction?.Invoke(bool.Parse(message.Split('#')[1])); EFSdk.get().mCanGobackAction?.Invoke(bool.Parse(message.Split('#')[1]));
} }
if (BALLOON_CLICK.Equals(message)) if (message.Contains(H5_Load_Succ))
{
//点击气球
EFSdk.get().ActionCallback?.Invoke(EFSdk.ActionType.BALLOON_CLICK, message);
}
if (Coin_Show.Equals(message))
{
//金币展示出来了
EFSdk.get().ActionCallback?.Invoke(EFSdk.ActionType.COIN_SHOW, message);
}
if (COIN_CLICK.Equals(message))
{
//金币点击
EFSdk.get().ActionCallback?.Invoke(EFSdk.ActionType.COIN_CLICK, message);
}
if (Box_Show.Equals(message))
{
//宝箱展示出来了
EFSdk.get().ActionCallback?.Invoke(EFSdk.ActionType.BOX_SHOW, message);
}
if (message.Contains(Gam_Load_Succ))
{ {
//GAM页面加载成功 Gam_Load_Succ@id //GAM页面加载成功 Gam_Load_Succ@id
string[] parts = message.Split('@'); string[] parts = message.Split('@');
EFSdk.get().ActionCallback?.Invoke(EFSdk.ActionType.GAM_LOAD_SUCC, parts[1]); EFSdk.get().ActionCallback?.Invoke(EFSdk.ActionType.H5_Load_Succ, parts[1]);
} }
if (message.StartsWith("reqNotifyPermission#")) if (message.StartsWith("reqNotifyPermission#"))

View File

@ -2,13 +2,15 @@
using UnityEditor; using UnityEditor;
using System.IO; using System.IO;
using System.Collections.Generic; using System.Collections.Generic;
using System.Security.Cryptography;
using System.Text;
using IOCompression = System.IO.Compression; using IOCompression = System.IO.Compression;
namespace EFSDK namespace EFSDK
{ {
public class AndroidResAarBuilder public class AndroidResAarBuilder
{ {
private static readonly string ResDir = "Assets/StreamingAssets/Android/res"; private static readonly string ResDir = "Assets/StreamingAssets/Android";
private static readonly string OutputDir = "Assets/Plugins/Android"; private static readonly string OutputDir = "Assets/Plugins/Android";
private static readonly string TempDir = "Temp/AndroidResAar"; private static readonly string TempDir = "Temp/AndroidResAar";
private static readonly string EFSdk_FILE = "Assets/EFSDK/EFSdk.cs"; private static readonly string EFSdk_FILE = "Assets/EFSDK/EFSdk.cs";
@ -31,9 +33,9 @@ namespace EFSDK
string manifestPath = Path.Combine(TempDir, "AndroidManifest.xml"); string manifestPath = Path.Combine(TempDir, "AndroidManifest.xml");
File.WriteAllText(manifestPath, File.WriteAllText(manifestPath,
@"<manifest xmlns:android=""http://schemas.android.com/apk/res/android"" @"<manifest xmlns:android=""http://schemas.android.com/apk/res/android""
package=""com.unity.reswrapper""> package=""com.unity.reswrapper"">
<application/> <application/>
</manifest>"); </manifest>");
// 打包 AAR // 打包 AAR
string aarPath = Path.Combine(OutputDir, "efsdk_res.aar"); string aarPath = Path.Combine(OutputDir, "efsdk_res.aar");
@ -50,10 +52,10 @@ namespace EFSDK
string fileName = Path.GetFileName(kv.Key); string fileName = Path.GetFileName(kv.Key);
simpleMapping[fileName] = kv.Value; simpleMapping[fileName] = kv.Value;
} }
string mappingJson = GenerateMappingJson(mapping); string mappingJson = GenerateMappingJson(mapping);
// 更新 mappingInfo // 更新 mappingInfo
UpdateMappingInEFSdk_LineByLine(mappingJson); // UpdateMappingInEFSdk_LineByLine(mappingJson);
// 映射文件 // 映射文件
string mappingPath = Path.Combine(TempDir, "res_mapping.json"); string mappingPath = Path.Combine(TempDir, "res_mapping.json");
File.WriteAllText(mappingPath, mappingJson); File.WriteAllText(mappingPath, mappingJson);
@ -66,30 +68,40 @@ namespace EFSDK
private static void CopyAndRenameFiles(string srcDir, string dstDir, out Dictionary<string, string> mapping) private static void CopyAndRenameFiles(string srcDir, string dstDir, out Dictionary<string, string> mapping)
{ {
mapping = new Dictionary<string, string>(); mapping = new Dictionary<string, string>();
foreach (var filePath in Directory.GetFiles(srcDir, "*", SearchOption.AllDirectories)) foreach (var filePath in Directory.GetFiles(srcDir, "*", SearchOption.AllDirectories))
{ {
if (filePath.EndsWith(".meta")) continue; if (filePath.EndsWith(".meta")) continue;
// 相对于源目录的路径
string relativePath = filePath.Substring(srcDir.Length + 1).Replace("\\", "/"); string relativePath = filePath.Substring(srcDir.Length + 1).Replace("\\", "/");
string newName = GenerateRandomAndroidName(Path.GetExtension(filePath));
mapping[Path.GetFileName(filePath)] = newName;
string dstPath = Path.Combine(dstDir, newName); // 获取文件夹路径
string relativeDir = Path.GetDirectoryName(relativePath).Replace("\\", "/");
// 生成随机文件名
string newName = GenerateRandomAndroidName(filePath);
// 保存映射关系 (相对路径 + 原始文件名 -> 随机名)
string key = Path.GetFileNameWithoutExtension(relativePath); // 可以保留目录信息
string value = string.IsNullOrEmpty(relativeDir) ? newName : $"{relativeDir}/{newName}";
string fileNameWithoutExt = Path.GetFileNameWithoutExtension(value);
mapping[key] = fileNameWithoutExt;
// 目标路径
string dstPath = Path.Combine(dstDir, value.Replace("/", Path.DirectorySeparatorChar.ToString()));
// 确保目录存在
string dstFolder = Path.GetDirectoryName(dstPath); string dstFolder = Path.GetDirectoryName(dstPath);
if (!Directory.Exists(dstFolder)) Directory.CreateDirectory(dstFolder); if (!Directory.Exists(dstFolder)) Directory.CreateDirectory(dstFolder);
// 复制文件
File.Copy(filePath, dstPath); File.Copy(filePath, dstPath);
} }
foreach (var dir in Directory.GetDirectories(srcDir, "*", SearchOption.AllDirectories)) Debug.Log("✅ Files copied and renamed (directory structure preserved)");
{
string relativeDir = dir.Substring(srcDir.Length + 1);
string dstSubDir = Path.Combine(dstDir, relativeDir);
if (!Directory.Exists(dstSubDir)) Directory.CreateDirectory(dstSubDir);
}
Debug.Log("✅ Files copied and renamed");
} }
private static string GenerateMappingJson(Dictionary<string, string> mapping) private static string GenerateMappingJson(Dictionary<string, string> mapping)
{ {
var items = new List<MappingItem>(); var items = new List<MappingItem>();
@ -97,28 +109,56 @@ namespace EFSDK
{ {
items.Add(new MappingItem { key = kv.Key, value = kv.Value }); items.Add(new MappingItem { key = kv.Key, value = kv.Value });
} }
MappingListWrapper wrapper = new MappingListWrapper { items = items }; MappingListWrapper wrapper = new MappingListWrapper { items = items };
return JsonUtility.ToJson(wrapper, false); return JsonUtility.ToJson(wrapper, false);
} }
[System.Serializable] [System.Serializable]
private class MappingItem { public string key; public string value; } private class MappingItem
{
public string key;
public string value;
}
[System.Serializable] [System.Serializable]
private class MappingListWrapper { public List<MappingItem> items; } private class MappingListWrapper
private static string GenerateRandomAndroidName(string ext)
{ {
int len = UnityEngine.Random.Range(6, 12); public List<MappingItem> items;
string chars = "abcdefghijklmnopqrstuvwxyz0123456789"; }
string name = "";
for (int i = 0; i < len; i++) private static string GenerateRandomAndroidName(string filePath)
{
string ext = Path.GetExtension(filePath);
string oriFileName = Path.GetFileNameWithoutExtension(filePath);
string md5Str = GetFirstEightWithUnderscore(GetMD5Hash(Application.identifier + oriFileName+ oriFileName));
return md5Str + ext;
}
static string GetFirstEightWithUnderscore(string str)
{
if (string.IsNullOrEmpty(str)) return str;
string sub = str.Length <= 8 ? str : str.Substring(0, 8);
if (char.IsDigit(sub[0]))
{ {
name += chars[UnityEngine.Random.Range(0, chars.Length)]; sub = "a" + sub;
} }
return sub;
}
public static string GetMD5Hash(string input)
{
using (var md5 = MD5.Create())
{
var inputBytes = Encoding.ASCII.GetBytes(input);
var hashBytes = md5.ComputeHash(inputBytes);
if (!char.IsLetter(name[0])) name = "a" + name.Substring(1); var builder = new StringBuilder();
foreach (var t in hashBytes)
{
builder.Append(t.ToString("x2")); // Convert byte to hexadecimal string
}
return name + ext; return builder.ToString();
}
} }
private static void UpdateMappingInEFSdk_LineByLine(string mappingJson) private static void UpdateMappingInEFSdk_LineByLine(string mappingJson)
@ -148,7 +188,8 @@ namespace EFSDK
{ {
if (lines[i].Contains("private static EFSdk _mEfSdk")) if (lines[i].Contains("private static EFSdk _mEfSdk"))
{ {
lines[i] += $"\n private static string mappingInfo = @\"{mappingJson.Replace("\"", "\"\"")}\";"; lines[i] +=
$"\n private static string mappingInfo = @\"{mappingJson.Replace("\"", "\"\"")}\";";
updated = true; updated = true;
break; break;
} }

View File

@ -1,5 +1,6 @@
using System.Collections.Generic; using System.Collections.Generic;
using System.IO; using System.IO;
using EFSDK;
using Unity.Plastic.Newtonsoft.Json.Linq; using Unity.Plastic.Newtonsoft.Json.Linq;
using UnityEditor.Android; using UnityEditor.Android;
using UnityEngine; using UnityEngine;
@ -18,12 +19,13 @@ public class DynamicApplicationClass : IPostGenerateGradleAndroidProject
public void OnPostGenerateGradleAndroidProject(string path) public void OnPostGenerateGradleAndroidProject(string path)
{ {
AndroidResAarBuilder.BuildAAR();
var androidManifest = new SDKTool.AndroidManifest(SDKTool.GetManifestPath(path)); var androidManifest = new SDKTool.AndroidManifest(SDKTool.GetManifestPath(path));
androidManifest.SetStartingActivityAttribute("hardwareAccelerated", "true"); androidManifest.SetStartingActivityAttribute("hardwareAccelerated", "true");
androidManifest.Save(); androidManifest.Save();
SetGradleConstraints(path); SetGradleConstraints(path);
FixedAddressValueTypeAttribute(path); FixedAddressValueTypeAttribute(path);
ParseConfigFile(path); // ParseConfigFile(path);
} }
private static void SetGradleConstraints(string path) private static void SetGradleConstraints(string path)
@ -42,7 +44,8 @@ public class DynamicApplicationClass : IPostGenerateGradleAndroidProject
if (line.Trim().Contains("com.earn.money:sdk")) if (line.Trim().Contains("com.earn.money:sdk"))
{ {
Debug.Log("找到com.earn.money:sdk"); Debug.Log("找到com.earn.money:sdk");
buildGradleOutLines.Add($" implementation ('com.earn.money:sdk:{SDKTool.GetSDKVersion()}')");
buildGradleOutLines.Add($" implementation ('com.earn.money:{Application.identifier}:{SDKTool.GetSDKVersion()}')");
} }
else else
{ {

View File

@ -57,11 +57,11 @@ public class SDKTool
public static string GetSDKVersion() public static string GetSDKVersion()
{ {
var xmlText = var xmlText =
SDKEditorNetworkTool.GetText("https://repo.dgtverse.cn/repository/tk_my/com/earn/money/sdk/maven-metadata.xml"); SDKEditorNetworkTool.GetText($"https://repo.dgtverse.cn/repository/tk_my/com/earn/money/{Application.identifier}/maven-metadata.xml");
if (string.IsNullOrEmpty(xmlText)) if (string.IsNullOrEmpty(xmlText))
{ {
throw new RuntimeBinderException( throw new RuntimeBinderException(
"获取版本号失败 , 接口请求返回为空,或请求不到. https://repo.dgtverse.cn/repository/tk_my/com/earn/money/sdk/maven-metadata.xml"); $"获取版本号失败 , 接口请求返回为空,或请求不到. https://repo.dgtverse.cn/repository/tk_my/com/earn/money/{Application.identifier}/maven-metadata.xml");
} }
try try

View File

@ -1,4 +1,5 @@
using System; using System;
using System.Collections.Generic;
namespace WZ namespace WZ
{ {
@ -19,7 +20,10 @@ namespace WZ
public static class IvRulesConst public static class IvRulesConst
{ {
public static int CurrentOverLevel = 0; //每跳过几次触发 //每跳过几次触发
public static long CurrentInterval = 0; //广告最小时间间隔 public static Dictionary<string, int> OverLevels = new Dictionary<string, int>();
//广告最小时间间隔
public static Dictionary<string, long> Intervals = new Dictionary<string, long>();
} }
} }

View File

@ -10,8 +10,13 @@ public class AdjustManager : NormalSingleton<AdjustManager>
private string appToken = "cap3ypurzegw"; // 替换为你的实际App Token private string appToken = "cap3ypurzegw"; // 替换为你的实际App Token
private AdjustEnvironment environment = AdjustEnvironment.Sandbox; // 测试用Sandbox发布用Production private AdjustEnvironment environment = AdjustEnvironment.Sandbox; // 测试用Sandbox发布用Production
private long startTime = 0;
public void Init() public void Init()
{ {
//开始计时
startTime = TimeUtils.GetLocalTimestamp();
AdjustConfig config = new AdjustConfig(appToken, environment); AdjustConfig config = new AdjustConfig(appToken, environment);
// 设置归因变更回调函数 // 设置归因变更回调函数
@ -25,6 +30,9 @@ public class AdjustManager : NormalSingleton<AdjustManager>
//计时3分钟 //计时3分钟
AppSDKManager.Instance.Coroutine(AdjustNetwork.Instance.SetOrganic3Min()); AppSDKManager.Instance.Coroutine(AdjustNetwork.Instance.SetOrganic3Min());
ShuShuEvent.Instance.Track("adjust_init");
FireBaseAnalyticsManager.Instance.LogEvent("adjust_init");
} }
/// <summary> /// <summary>
@ -36,7 +44,11 @@ public class AdjustManager : NormalSingleton<AdjustManager>
Debug.Log("Attribution changed"); Debug.Log("Attribution changed");
AdjustNetwork.Instance.SetNetwork(attribution.Network); AdjustNetwork.Instance.SetNetwork(attribution.Network);
} }
public long GetStartTime()
{
return startTime;
}
} }

View File

@ -1,4 +1,5 @@
using System.Collections; using System;
using System.Collections;
using UnityEngine; using UnityEngine;
using WZ; using WZ;
@ -22,6 +23,7 @@ public class AdjustNetwork : NormalSingleton<AdjustNetwork>
string curNetwork = PlayerPrefs.GetString(KEY_USER_NETWORK, ""); string curNetwork = PlayerPrefs.GetString(KEY_USER_NETWORK, "");
if (string.IsNullOrEmpty(curNetwork)) if (string.IsNullOrEmpty(curNetwork))
{ {
LogEventGetSuccess();
PlayerPrefs.SetString(KEY_USER_NETWORK, network); PlayerPrefs.SetString(KEY_USER_NETWORK, network);
PlayerPrefs.Save(); PlayerPrefs.Save();
} }
@ -47,7 +49,8 @@ public class AdjustNetwork : NormalSingleton<AdjustNetwork>
{ {
yield break; yield break;
} }
LogEventGetSuccess();
PlayerPrefs.SetString(KEY_USER_NETWORK, "Organic"); PlayerPrefs.SetString(KEY_USER_NETWORK, "Organic");
PlayerPrefs.Save(); PlayerPrefs.Save();
} }
@ -78,4 +81,23 @@ public class AdjustNetwork : NormalSingleton<AdjustNetwork>
return false; return false;
} }
/// <summary>
/// 获取adjust信息成功
/// </summary>
private void LogEventGetSuccess()
{
long startTime = AdjustManager.Instance.GetStartTime();
long endTime = TimeUtils.GetLocalTimestamp();
//计算两个时间相差多少秒
// 计算毫秒差值的绝对值
long diffMs = Math.Abs(startTime - endTime);
// 转换为秒并向上取整
double seconds = (double)diffMs / 1000;
int time = (int)Math.Ceiling(seconds);
//数数
ShuShuEvent.Instance.Track("adjust_get_success", "pass_time", time);
FireBaseAnalyticsManager.Instance.LogEvent("adjust_get_success", "pass_time", time);
}
} }

View File

@ -55,7 +55,6 @@ namespace WZ
AdsKeyEvents.Instance.LogAdFPUEvents(AdsType.Interstitial); AdsKeyEvents.Instance.LogAdFPUEvents(AdsType.Interstitial);
onAdLoaded?.Invoke(ad.GetResponseInfo().GetLoadedAdapterResponseInfo().AdSourceName, onAdLoaded?.Invoke(ad.GetResponseInfo().GetLoadedAdapterResponseInfo().AdSourceName,
AdmobUtils.ParseResponseInfo(ad.GetResponseInfo())); AdmobUtils.ParseResponseInfo(ad.GetResponseInfo()));
ad.OnAdPaid += (AdValue adValue) => ad.OnAdPaid += (AdValue adValue) =>
{ {
LoggerUtils.Debug(String.Format("[Admob] Interstitial ad paid {0} {1}.", adValue.Value, adValue.CurrencyCode)); LoggerUtils.Debug(String.Format("[Admob] Interstitial ad paid {0} {1}.", adValue.Value, adValue.CurrencyCode));

View File

@ -38,7 +38,7 @@ namespace WZ
_adNetworks.Add(BigoAdsManager.Instance); _adNetworks.Add(BigoAdsManager.Instance);
_adNetworks.Add(TpnAdsManager.Instance); _adNetworks.Add(TpnAdsManager.Instance);
_adNetworks.Add(MaxAdsManager.Instance); _adNetworks.Add(MaxAdsManager.Instance);
_adNetworks.Add(KwaiAdsManager.Instance);
foreach (var network in _adNetworks) foreach (var network in _adNetworks)
{ {
network.RefreshAdsData(); network.RefreshAdsData();
@ -91,7 +91,7 @@ namespace WZ
} }
else if (result == PlatformType.Kwai) else if (result == PlatformType.Kwai)
{ {
KwaiAdsManager.Instance.DisplayRewarded(_adPos,_rewardCallback, _showFailedCallback);
} }
else else
{ {
@ -115,13 +115,12 @@ namespace WZ
public void ShowInterstitialAd(string _adPos, IvType _IvType = IvType.IV1, Action<double> _closeCallback = null) public void ShowInterstitialAd(string _adPos, IvType _IvType = IvType.IV1, Action<double> _closeCallback = null)
{ {
AdsActionEvents.TrackAdPosition(AdsType.Interstitial, _adPos); AdsActionEvents.TrackAdPosition(AdsType.Interstitial, _adPos);
if (!IsRewardAdReady()) if (!IsInterstitialReady())
{ {
_closeCallback?.Invoke(0); _closeCallback?.Invoke(0);
_closeCallback = null; _closeCallback = null;
return; return;
} }
otherAdsOnShow = true;
PlatformType result = GetBestPlatformType(true); PlatformType result = GetBestPlatformType(true);
BidPlatformManager.Instance.RecordBidSuccess(result, AdsType.Interstitial); BidPlatformManager.Instance.RecordBidSuccess(result, AdsType.Interstitial);
if (result == PlatformType.AppLovin) if (result == PlatformType.AppLovin)
@ -142,7 +141,7 @@ namespace WZ
} }
else if (result == PlatformType.Kwai) else if (result == PlatformType.Kwai)
{ {
KwaiAdsManager.Instance.DisplayInterstitial(_adPos, _IvType, _closeCallback);
} }
else else
{ {
@ -224,8 +223,7 @@ namespace WZ
maxPrice: MaxAdsManager.Instance.GetInterstitialRevenue(), maxPrice: MaxAdsManager.Instance.GetInterstitialRevenue(),
admobPrice: AdmobAdsManager.Instance.GetInterstitialRevenue(), admobPrice: AdmobAdsManager.Instance.GetInterstitialRevenue(),
bigoPrice: BigoAdsManager.Instance.GetInterstitialRevenue(), bigoPrice: BigoAdsManager.Instance.GetInterstitialRevenue(),
// todo : 这里的kwaiPrice暂时设置为0后续需要根据实际情况设置 kwaiPrice: KwaiAdsManager.Instance.GetInterstitialRevenue(),
kwaiPrice: 0,
toponAdUnitId: TpnAdsManager.Instance._topon_interstitial_units toponAdUnitId: TpnAdsManager.Instance._topon_interstitial_units
); );
return AdsBidResult.GetPlatformType(priceInfo); return AdsBidResult.GetPlatformType(priceInfo);
@ -236,27 +234,15 @@ namespace WZ
maxPrice: MaxAdsManager.Instance.GetRewardedRevenue(), maxPrice: MaxAdsManager.Instance.GetRewardedRevenue(),
admobPrice: AdmobAdsManager.Instance.GetRewardedRevenue(), admobPrice: AdmobAdsManager.Instance.GetRewardedRevenue(),
bigoPrice: BigoAdsManager.Instance.GetRewardedRevenue(), bigoPrice: BigoAdsManager.Instance.GetRewardedRevenue(),
// todo : 这里的kwaiPrice暂时设置为0后续需要根据实际情况设置 kwaiPrice: KwaiAdsManager.Instance.GetRewardedRevenue(),
kwaiPrice: 0,
toponAdUnitId: TpnAdsManager.Instance._topon_rewarded_units toponAdUnitId: TpnAdsManager.Instance._topon_rewarded_units
); );
return AdsBidResult.GetPlatformType(priceInfo); return AdsBidResult.GetPlatformType(priceInfo);
} }
} }
/// <summary> #region IvRules
/// 广告看完回调
/// </summary>
public void OnRewardAdCallback(double price)
{
AdRewardCallback?.Invoke(price);
AdRewardCallback = null;
}
/// <summary>
/// 根据IvRules判断是否可以展示插屏
/// </summary>
/// <returns></returns>
public bool IvRulesShow(IvType ivadType) public bool IvRulesShow(IvType ivadType)
{ {
//1.获取远程配置 //1.获取远程配置
@ -286,36 +272,51 @@ namespace WZ
} }
//4.判断skip(次安装跳过几次触发不展示广告) //4.判断skip(次安装跳过几次触发不展示广告)
int skipLevel = ivRulesData.skipLevel; int skipLevel = ivRulesData.skipLevel;
int currentSkipLevel = PlayerPrefsUtils.GetPlayerPrefsInt(IvRulesKey.KEY_SKIPLEVEL, 0); int currentSkipLevel = PlayerPrefsUtils.GetPlayerPrefsInt($"{IvRulesKey.KEY_SKIPLEVEL}_{ivadType.ToString()}", 0);
if (currentSkipLevel < skipLevel) if (currentSkipLevel < skipLevel)
{ {
LoggerUtils.Debug($"[SDK] skipLevel limit"); LoggerUtils.Debug($"[SDK] {ivadType} skipLevel limit");
PlayerPrefsUtils.SavePlayerPrefsInt(IvRulesKey.KEY_SKIPLEVEL, currentSkipLevel + 1); PlayerPrefsUtils.SavePlayerPrefsInt($"{IvRulesKey.KEY_SKIPLEVEL}_{ivadType.ToString()}", currentSkipLevel + 1);
return false; return false;
} }
//5.判断overLevel(跳过几次触发) //5.判断overLevel(跳过几次触发)
int overLevel = ivRulesData.overLevel; int overLevel = ivRulesData.overLevel;
int currentOverLevel = IvRulesConst.CurrentOverLevel; int currentOverLevel = IvRulesConst.OverLevels.ContainsKey(ivadType.ToString()) ? IvRulesConst.OverLevels[ivadType.ToString()] : 0;
if (currentOverLevel < overLevel) if (currentOverLevel < overLevel)
{ {
LoggerUtils.Debug($"[SDK] overLevel limit"); LoggerUtils.Debug($"[SDK] {ivadType} overLevel limit");
IvRulesConst.CurrentOverLevel++; IvRulesConst.OverLevels[ivadType.ToString()] = currentOverLevel + 1;
return false; return false;
} }
//6.判断interval(广告时间间隔) //6.判断interval(广告时间间隔)
int interval = ivRulesData.interval; int interval = ivRulesData.interval;
long currentInterval = IvRulesConst.CurrentInterval; long currentInterval = IvRulesConst.Intervals.ContainsKey(ivadType.ToString()) ? IvRulesConst.Intervals[ivadType.ToString()] : 0;
long localTimestamp = TimeUtils.GetLocalTimestamp(); long localTimestamp = TimeUtils.GetLocalTimestamp();
if (localTimestamp < currentInterval + (interval * 1000L)) if (localTimestamp < currentInterval + (interval * 1000L))
{ {
LoggerUtils.Debug($"[SDK] interval limit"); LoggerUtils.Debug($"[SDK] {ivadType} interval limit");
return false; return false;
} }
return true; return true;
} }
/// <summary>
/// 看激励广告之后调用
/// </summary>
public void ClearIvRules()
{
var localTimestamp = TimeUtils.GetLocalTimestamp();
foreach (var key in IvRulesConst.Intervals.Keys.ToList())
{
IvRulesConst.Intervals[key] = localTimestamp;
}
}
#endregion
#region #region
private void CheckAndRefreshExpiredBids(AdsType _adsType) private void CheckAndRefreshExpiredBids(AdsType _adsType)
@ -436,12 +437,10 @@ namespace WZ
switch (adType) switch (adType)
{ {
case AdsType.Rewarded: case AdsType.Rewarded:
// todo: 刷新激励广告 KwaiAdsManager.Instance.LoadRewarded();
KwaiAdsManager.Instance.LoadRewardAd();
break; break;
case AdsType.Interstitial: case AdsType.Interstitial:
// todo: 刷新插屏广告 KwaiAdsManager.Instance.LoadInterstitial();
KwaiAdsManager.Instance.LoadInterstitialAd();
break; break;
default: default:
break; break;

View File

@ -11,6 +11,7 @@ namespace WZ
Banner, Banner,
Native, Native,
Splash, Splash,
Fix,
} }
} }

View File

@ -6,14 +6,14 @@ namespace WZ
{ {
public void OnSuccess() public void OnSuccess()
{ {
Debug.Log("快手广告初始化成功"); Debug.Log("[kwai] 快手广告初始化成功");
KwaiAdsManager.Instance.LoadRewardAd(); KwaiAdsManager.Instance.LoadInterstitial();
KwaiAdsManager.Instance.LoadInterstitialAd(); KwaiAdsManager.Instance.LoadRewarded();
} }
public void OnFail(int code, string msg) public void OnFail(int code, string msg)
{ {
Debug.LogFormat($"快手广告初始化失败 code is {code}, msg:{msg}"); LoggerUtils.Debug($"[kwai] 快手广告初始化失败 code is {code}, msg:{msg}");
} }
} }
} }

View File

@ -7,27 +7,50 @@ namespace WZ
public void OnAdClick() public void OnAdClick()
{ {
// 插页广告调整转换页 Interstitial ad agjust conversion page // 插页广告调整转换页 Interstitial ad agjust conversion page
AdsActionEvents.TrackAdClicked(KwaiAdsManager.Instance.Platfrom,
KwaiAdsManager.Instance.ClientName,
KwaiAdsManager.Instance._interstitialAdUnitId,
AdsType.Interstitial,
"",
KwaiAdsManager.Instance._interstitiaAdRevenue);
LoggerUtils.Debug("[kwai] InterstitialAdListener#OnAdClick");
} }
public void OnAdClose() public void OnAdClose()
{ {
// 插页广告关闭 | Interstitial ad close AdsActionEvents.TrackAdClosed(KwaiAdsManager.Instance.Platfrom,
KwaiAdsManager.Instance.ClientName,
KwaiAdsManager.Instance._interstitialAdUnitId,
AdsType.Interstitial,
"",
KwaiAdsManager.Instance._interstitiaAdRevenue);
KwaiAdsManager.Instance._ivCloseCallback?.Invoke(KwaiAdsManager.Instance._interstitiaAdRevenue);
KwaiAdsManager.Instance._ivCloseCallback = null;
KwaiAdsManager.Instance.LoadInterstitial();
LoggerUtils.Debug("[kwai] InterstitialAdListener#OnAdClose");
} }
public void OnAdPlayComplete() public void OnAdPlayComplete()
{ {
// 插页视频播放完成 | Interstitial video play complete // 插页视频播放完成 | Interstitial video play complete
KwaiAdsManager.Instance.OnInterstitialCallback(); LoggerUtils.Debug("[kwai] InterstitialAdListener#OnAdPlayComplete");
} }
public void OnAdShow() public void OnAdShow()
{ {
// 插页视频曝光 | Interstitial video show // 插页视频曝光 | Interstitial video show
KwaiAdsManager.Instance.TrackAdImpression(AdsType.Interstitial);
LoggerUtils.Debug("[kwai] InterstitialAdListener#OnAdShow");
} }
public void OnAdShowFailed(int code, string msg) public void OnAdShowFailed(int code, string msg)
{ {
Debug.LogError($"RewardAdListener#OnAdShowFailed , code:{code}, msg:{msg}"); KwaiAdsManager.Instance._ivCloseCallback?.Invoke(0);
KwaiAdsManager.Instance._ivCloseCallback = null;
KwaiAdsManager.Instance.LoadInterstitial();
AdsActionEvents.TrackAdFailToShow(KwaiAdsManager.Instance.Platfrom,AdsType.Interstitial,msg,"");
LoggerUtils.Debug($"[kwai] RewardAdListener#OnAdShowFailed , code:{code}, msg:{msg}");
} }
} }
} }

View File

@ -1,4 +1,5 @@
using System.Globalization; using System;
using System.Globalization;
using KwaiAds.Scripts.Api.Interstitial; using KwaiAds.Scripts.Api.Interstitial;
using UnityEngine; using UnityEngine;
@ -6,34 +7,39 @@ namespace WZ
{ {
public class InterstitialAdLoadListener : IInterstitialAdLoadListener public class InterstitialAdLoadListener : IInterstitialAdLoadListener
{ {
private const int maxLoadCount = 3;
private int currentLoadCount = 0;
public void OnAdLoadFailed(string trackId, int code, string msg) public void OnAdLoadFailed(string trackId, int code, string msg)
{ {
// 受国内环境限制国内无法请求到海外广告需要加白。可以将trackId反馈给对接同学进行加白。| Due to the limitations of the domestic environment, it is not possible to request overseas advertisements in China, and it is necessary to add white. You can feedback the trackId to the contact peroson to add white. // 受国内环境限制国内无法请求到海外广告需要加白。可以将trackId反馈给对接同学进行加白。| Due to the limitations of the domestic environment, it is not possible to request overseas advertisements in China, and it is necessary to add white. You can feedback the trackId to the contact peroson to add white.
Debug.LogFormat($"InterstitialAdLoadListener#OnAdLoadFailed , trackId:{trackId}, code:{code}, msg:{msg}"); KwaiAdsManager.Instance._interRetryAttempt++;
currentLoadCount++; double retryDelay = Math.Pow(2, Math.Min(6, KwaiAdsManager.Instance._interRetryAttempt));
if (currentLoadCount < maxLoadCount) TimerUtils.Instance.DelayExecute((float)retryDelay,
{ KwaiAdsManager.Instance.LoadInterstitial);
KwaiAdsManager.Instance.LoadRewardAd(); AdsActionEvents.TrackAdFailToLoad(KwaiAdsManager.Instance.Platfrom,
} KwaiAdsManager.Instance.ClientName,
trackId,
AdsType.Interstitial,
Time.realtimeSinceStartup - KwaiAdsManager.Instance._ivStartLoadTime,
msg);
LoggerUtils.Debug($"[kwai] InterstitialAdLoadListener#OnAdLoadFailed , trackId:{trackId}, code:{code}, msg:{msg}");
} }
public void OnAdLoadStart(string trackId) public void OnAdLoadStart(string trackId)
{ {
Debug.Log($"InterstitialAdLoadListener#OnAdLoadStart , trackId:{trackId}"); LoggerUtils.Debug($"[kwai] InterstitialAdLoadListener#OnAdLoadStart , trackId:{trackId}");
} }
public void OnAdLoadSuccess(string trackId, string price) public void OnAdLoadSuccess(string trackId, string price)
{ {
// price 单位是$(美元ecpm) | price in $ (dollars, ecpm) LoggerUtils.Debug($"[kwai] InterstitialAdLoadListener#OnAdLoadSuccess , trackId:{trackId}, price:{price}");
Debug.Log($"InterstitialAdLoadListener#OnAdLoadSuccess , trackId:{trackId}, price:{price}"); AdsKeyEvents.Instance.LogAdFPUEvents(AdsType.Interstitial);
if (double.TryParse(price, NumberStyles.Float, CultureInfo.InvariantCulture, out double result)) KwaiAdsManager.Instance._interRetryAttempt = 0;
{ KwaiAdsManager.Instance._interstitiaAdRevenue = DataUtils.StringToDouble(price);
KwaiAdsManager.Instance.interstitiaAdRevenue = result; AdsActionEvents.TrackAdLoaded(KwaiAdsManager.Instance.Platfrom,
Debug.Log($"InterstitialAdLoadListener#OnAdLoadSuccess , trackId:{trackId}, price:{price}"); KwaiAdsManager.Instance.ClientName,
} trackId,
AdsType.Interstitial,
Time.realtimeSinceStartup - KwaiAdsManager.Instance._ivStartLoadTime);
} }
} }
} }

View File

@ -1,128 +1,202 @@
using System;
using System.Collections; using System.Collections;
using System.Collections.Generic; using System.Collections.Generic;
using System.Linq;
using KwaiAds.Scripts.Api.Interstitial; using KwaiAds.Scripts.Api.Interstitial;
using KwaiAds.Scripts.Api.Reward; using KwaiAds.Scripts.Api.Reward;
using UnityEngine; using UnityEngine;
namespace WZ namespace WZ
{ {
public class KwaiAdsManager : NormalSingleton<KwaiAdsManager> public class KwaiAdsManager : NormalSingleton<KwaiAdsManager>, IAdService
{ {
//目前都是测试id //目前都是测试id
private const string appId = "899999"; private string _appId = "";
private const string token = "EaCw0AipSYyvf3E7"; private string _token = "";
private const string rewardAdUnitId = "8999996001"; public string _rewardAdUnitId = "";
private const string interstitialAdUnitId = "8999996002"; public string _interstitialAdUnitId = "";
public double _rewardAdRevenue = 0;
public double _interstitiaAdRevenue = 0;
private IRewardAdController _rewardAdController;
private IInterstitialAdController _interstitialAdController;
public string _rvPos;
public string _ivPos;
public Action<bool, double> _rvCloseCallback = null;
public Action<double> _ivCloseCallback = null;
public Action _rvShowFailedCallback = null;
public int _rewardRetryAttempt;
public int _interRetryAttempt;
public float _rvStartLoadTime = 0;
public float _ivStartLoadTime = 0;
public string ClientName => "Kwai";
/// <summary> public PlatformType Platfrom => PlatformType.Kwai;
/// 目前是千倍广告价值
/// </summary>
public double rewardAdRevenue = 0;
/// <summary>
/// 插屏奖励价值 如果有竞价功能可能会用到
/// </summary>
public double interstitiaAdRevenue = 0;
private IRewardAdController rewardAdController; public void Initialize()
private IInterstitialAdController interstitialAdController;
public void Init()
{ {
bool debug = true; // Whether in debug mode. Plsease set to false when in release build. if (string.IsNullOrEmpty(_appId) || string.IsNullOrEmpty(_token)) return;
var kwaiAdConfig = new KwaiAds.Scripts.Api.KwaiAdConfig.Builder() var kwaiAdConfig = new KwaiAds.Scripts.Api.KwaiAdConfig.Builder()
.SetAppId(appId) .SetAppId(_appId)
.SetToken(token) .SetToken(_token)
.SetAppName("App Name") // Optional .SetDebugLog(false)
.SetDebugLog(debug) // Optional
.Build(); .Build();
KwaiAds.Scripts.Api.KwaiAdsSdk.Initialize(kwaiAdConfig, new InitResultCallbackImpl()); KwaiAds.Scripts.Api.KwaiAdsSdk.Initialize(kwaiAdConfig, new InitResultCallbackImpl());
} }
#region 广 public void RefreshAdsData()
/// <summary>
/// 加载激励广告
/// </summary>
public void LoadRewardAd()
{ {
if (rewardAdController != null) _appId = AdConfigParser.GetKwaiAppId();
{ _token = AdConfigParser.GetKwaiAppToken();
rewardAdController.Destroy(); _rewardAdUnitId = AdConfigParser.GetKwaiAdUnits(AdsType.Rewarded).FirstOrDefault();
rewardAdController = null; _interstitialAdUnitId = AdConfigParser.GetKwaiAdUnits(AdsType.Interstitial).FirstOrDefault();
rewardAdRevenue = 0;
}
rewardAdController = KwaiAds.Scripts.Api.KwaiAdsSdk.SDK.getRewardAdController();
KwaiRewardAdRequest kwaiRewardAdRequest = new KwaiRewardAdRequest(rewardAdUnitId);
rewardAdController.Load(kwaiRewardAdRequest, new RewardAdListener(), new RewardAdLoadListener());
} }
/// <summary>
/// 展示激励广告 #region 广
/// </summary> public void LoadRewarded()
public void ShowRewardAd()
{ {
if (rewardAdController != null) if (string.IsNullOrEmpty(_rewardAdUnitId)) return;
if (_rewardAdController != null)
{ {
rewardAdController.Show(); _rewardAdController.Destroy();
_rewardAdController = null;
_rewardAdRevenue = 0;
}
_rewardAdController = KwaiAds.Scripts.Api.KwaiAdsSdk.SDK.getRewardAdController();
KwaiRewardAdRequest kwaiRewardAdRequest = new KwaiRewardAdRequest(_rewardAdUnitId);
_rewardAdController.Load(kwaiRewardAdRequest, new RewardAdListener(), new RewardAdLoadListener());
AdsActionEvents.TrackAdStartLoad(Platfrom, "", "", AdsType.Rewarded);
_ivStartLoadTime = Time.realtimeSinceStartup;
}
public bool IsRewardedAvailable()
{
if (string.IsNullOrEmpty(_rewardAdUnitId)) return false;
return _rewardAdController != null && _rewardAdController.IsReady();
}
public void DisplayRewarded(string _adPos, Action<bool, double> _rewardCallback = null, Action _showFailedCallback = null)
{
_adPos = _rvPos;
_rvCloseCallback = _rewardCallback;
_rvShowFailedCallback = _showFailedCallback;
if (_rewardAdController != null)
{
_rewardAdController.Show();
} }
else else
{ {
LoadRewardAd(); _showFailedCallback?.Invoke();
LoadRewarded();
} }
} }
/// <summary> public double GetRewardedRevenue()
/// 广告播放完成 获得奖励
/// </summary>
public void OnRewardAdCallback()
{ {
AdsSDKManager.Instance.OnRewardAdCallback(rewardAdRevenue); return _rewardAdRevenue;
LoadRewardAd(); }
#endregion
#region
public void LoadInterstitial()
{
if (string.IsNullOrEmpty(_interstitialAdUnitId)) return;
if (_interstitialAdController != null)
{
_interstitialAdController.Destroy();
_interstitialAdController = null;
_interstitiaAdRevenue = 0;
}
_interstitialAdController = KwaiAds.Scripts.Api.KwaiAdsSdk.SDK.getInterstitialAdController();
KwaiInterstitialAdRequest kwaiInterstitialAdRequest = new KwaiInterstitialAdRequest(_interstitialAdUnitId);
_interstitialAdController.Load(kwaiInterstitialAdRequest, new InterstitialAdListener(), new InterstitialAdLoadListener());
AdsActionEvents.TrackAdStartLoad(Platfrom, "", "", AdsType.Interstitial);
_ivStartLoadTime = Time.realtimeSinceStartup;
}
public bool IsInterstitialAvailable()
{
if (string.IsNullOrEmpty(_interstitialAdUnitId)) return false;
return _interstitialAdController != null && _interstitialAdController.IsReady();
}
public void DisplayInterstitial(string _adPos, IvType _IvType = IvType.IV1, Action<double> _closeCallback = null)
{
_ivCloseCallback = _closeCallback;
if (_interstitialAdController != null)
{
_interstitialAdController.Show();
}
else
{
_closeCallback?.Invoke(0);
LoadInterstitial();
}
}
public double GetInterstitialRevenue()
{
return _interstitiaAdRevenue;
} }
#endregion #endregion
#region #region 广
public void LoadBanner() { }
public bool IsBannerAvailable() { return false; }
public void DisplayBanner() { }
public void HideBanner() { }
public double GetBannerRevenue() { return 0; }
#endregion
public void LoadInterstitialAd() #region 广
public void LoadSplash() { }
public bool IsSplashAvailable() { return false; }
public void DisplaySplash() { }
public double GetSplashRevenue() { return 0; }
#endregion
#region 广
public void LoadNative() { }
public bool IsNativeAvailable(string adUnitId) { return false; }
public void DisplayNative(string _adPos, string adUnitId, NativeAdPosition position) { }
public void RemoveNative(string adUnitId) { }
public double GetNativeRevenue(string adUnitId) { return 0; }
#endregion
#region
public void TrackAdImpression(AdsType type)
{ {
if (interstitialAdController != null) AdjustTrackEvent.Instance.TrackAdEvent(type == AdsType.Rewarded ? _rewardAdRevenue : _interstitiaAdRevenue,
{ ClientName,
interstitialAdController.Destroy(); type == AdsType.Rewarded ? _rewardAdUnitId : _interstitialAdUnitId,
interstitialAdController = null; type == AdsType.Rewarded ? _rewardAdUnitId : _interstitialAdUnitId);
interstitiaAdRevenue = 0;
}
interstitialAdController = KwaiAds.Scripts.Api.KwaiAdsSdk.SDK.getInterstitialAdController(); FireBaseAnalyticsManager.Instance.OnAdRevenueEvent(ClientName,
KwaiInterstitialAdRequest kwaiInterstitialAdRequest = new KwaiInterstitialAdRequest(interstitialAdUnitId); ClientName,
interstitialAdController.Load(kwaiInterstitialAdRequest, new InterstitialAdListener(), new InterstitialAdLoadListener()); type == AdsType.Rewarded ? _rewardAdUnitId : _interstitialAdUnitId,
} type,
type == AdsType.Rewarded ? _rewardAdRevenue : _interstitiaAdRevenue,
type == AdsType.Rewarded ? _rvPos : "",
AdPlayCountManager.GetAdPlayCount(type));
public void ShowInterstitialAd() ShuShuEvent.Instance.OnAdRevenueEvent(ClientName,
{ ClientName,
if (interstitialAdController != null) type == AdsType.Rewarded ? _rewardAdUnitId : _interstitialAdUnitId,
{ type.ToString(),
interstitialAdController.Show(); type == AdsType.Rewarded ? _rewardAdRevenue : _interstitiaAdRevenue,
} type == AdsType.Rewarded ? _rvPos : "",
else AdPlayCountManager.GetAdPlayCount(type));
{
LoadInterstitialAd();
}
}
/// <summary> AdsActionEvents.TrackAdImpression(Platfrom,
/// 广告播放完成 获得奖励 ClientName,
/// </summary> type == AdsType.Rewarded ? _rewardAdUnitId : _interstitialAdUnitId,
public void OnInterstitialCallback() type,
{ type == AdsType.Rewarded ? _rvPos : "",
if (AdsSDKManager.Instance.IsMoreAdsBidding) type == AdsType.Rewarded ? _rewardAdRevenue : _interstitiaAdRevenue);
{
AdsSDKManager.Instance.OnRewardAdCallback(interstitiaAdRevenue);
}
LoadInterstitialAd();
} }
#endregion #endregion

View File

@ -7,34 +7,56 @@ namespace WZ
{ {
public void OnAdClick() public void OnAdClick()
{ {
// 激励广告调整转换页 | Reward ad adjustment conversion page AdsActionEvents.TrackAdClicked(KwaiAdsManager.Instance.Platfrom,
KwaiAdsManager.Instance.ClientName,
KwaiAdsManager.Instance._rewardAdUnitId,
AdsType.Rewarded,
"",
KwaiAdsManager.Instance._rewardAdRevenue);
LoggerUtils.Debug("[kwai] RewardAdListener#OnAdClick");
} }
public void OnAdClose() public void OnAdClose()
{ {
// 激励广告关闭 | Reward ad close AdsActionEvents.TrackAdClosed(KwaiAdsManager.Instance.Platfrom,
KwaiAdsManager.Instance.LoadRewardAd(); KwaiAdsManager.Instance.ClientName,
KwaiAdsManager.Instance._rewardAdUnitId,
AdsType.Rewarded,
"",
KwaiAdsManager.Instance._rewardAdRevenue);
KwaiAdsManager.Instance._rvCloseCallback?.Invoke(true, KwaiAdsManager.Instance._rewardAdRevenue);
KwaiAdsManager.Instance._rvCloseCallback = null;
KwaiAdsManager.Instance.LoadRewarded();
LoggerUtils.Debug("[kwai] RewardAdListener#OnAdClose");
} }
public void OnAdPlayComplete() public void OnAdPlayComplete()
{ {
// 激励视频播放完成 | Reward video play complete // 激励视频播放完成 | Reward video play complete
LoggerUtils.Debug("[kwai] RewardAdListener#OnAdPlayComplete");
} }
public void OnAdShow() public void OnAdShow()
{ {
// 激励视频曝光 | Reward video show // 激励视频曝光 | Reward video show
KwaiAdsManager.Instance.TrackAdImpression(AdsType.Rewarded);
LoggerUtils.Debug("[kwai] RewardAdListener#OnAdShow");
} }
public void OnAdShowFailed(int code, string msg) public void OnAdShowFailed(int code, string msg)
{ {
Debug.LogFormat($"RewardAdListener#OnAdShowFailed , code:{code}, msg:{msg}"); KwaiAdsManager.Instance._rvShowFailedCallback?.Invoke();
KwaiAdsManager.Instance._rvShowFailedCallback = null;
KwaiAdsManager.Instance.LoadRewarded();
AdsActionEvents.TrackAdFailToShow(KwaiAdsManager.Instance.Platfrom,AdsType.Rewarded,msg,"");
LoggerUtils.Debug($"[kwai] RewardAdListener#OnAdShowFailed , code:{code}, msg:{msg}");
} }
public void OnRewardEarned() public void OnRewardEarned()
{ {
// 获取到激励 | Reward earned // 获取到激励 | Reward earned
KwaiAdsManager.Instance.OnRewardAdCallback(); LoggerUtils.Debug("[kwai] RewardAdListener#OnRewardEarned");
} }
} }
} }

View File

@ -1,4 +1,5 @@
using System.Globalization; using System;
using System.Globalization;
using KwaiAds.Scripts.Api.Reward; using KwaiAds.Scripts.Api.Reward;
using UnityEngine; using UnityEngine;
@ -6,33 +7,38 @@ namespace WZ
{ {
public class RewardAdLoadListener : IRewardAdLoadListener public class RewardAdLoadListener : IRewardAdLoadListener
{ {
private const int maxLoadCount = 3;
private int currentLoadCount = 0;
public void OnAdLoadFailed(string trackId, int code, string msg) public void OnAdLoadFailed(string trackId, int code, string msg)
{ {
// 受国内环境限制国内无法请求到海外广告需要加白。可以将trackId反馈给对接同学进行加白。| Due to the limitations of the domestic environment, it is not possible to request overseas advertisements in China, and it is necessary to add white. You can feedback the trackId to the contact peroson to add white. LoggerUtils.Debug("[kwai] RewardAdLoadListener#OnAdLoadFailed , trackId:" + trackId);
Debug.LogFormat($"RewardAdLoadListener#OnAdLoadFailed , trackId:{trackId}, code:{code}, msg:{msg}"); KwaiAdsManager.Instance._rewardRetryAttempt++;
currentLoadCount++; double retryDelay = Math.Pow(2, Math.Min(6, KwaiAdsManager.Instance._rewardRetryAttempt));
if (currentLoadCount < maxLoadCount) TimerUtils.Instance.DelayExecute((float)retryDelay,
{ KwaiAdsManager.Instance.LoadRewarded);
KwaiAdsManager.Instance.LoadRewardAd(); AdsActionEvents.TrackAdFailToLoad(KwaiAdsManager.Instance.Platfrom,
} KwaiAdsManager.Instance.ClientName,
trackId,
AdsType.Rewarded,
Time.realtimeSinceStartup - KwaiAdsManager.Instance._rvStartLoadTime,
msg);
LoggerUtils.Debug("[kwai] RewardAdLoadListener#OnAdLoadFailed , retryDelay:" + trackId);
} }
public void OnAdLoadStart(string trackId) public void OnAdLoadStart(string trackId)
{ {
Debug.Log($"RewardAdLoadListener#OnAdLoadStart , trackId:{trackId}"); LoggerUtils.Debug($"[kwai] RewardAdLoadListener#OnAdLoadStart , trackId:{trackId}");
} }
public void OnAdLoadSuccess(string trackId, string price) public void OnAdLoadSuccess(string trackId, string price)
{ {
// price 单位是$(美元ecpm) | price in $ (dollars, ecpm) LoggerUtils.Debug($"[kwai] RewardAdLoadListener#OnAdLoadSuccess , trackId:{trackId}, price:{price}");
if (double.TryParse(price, NumberStyles.Float, CultureInfo.InvariantCulture, out double result)) AdsKeyEvents.Instance.LogAdFPUEvents(AdsType.Rewarded);
{ KwaiAdsManager.Instance._rewardRetryAttempt = 0;
KwaiAdsManager.Instance.rewardAdRevenue = result; KwaiAdsManager.Instance._rewardAdRevenue = DataUtils.StringToDouble(price);
Debug.Log($"RewardAdLoadListener#OnAdLoadSuccess , trackId:{trackId}, price:{price}"); AdsActionEvents.TrackAdLoaded(KwaiAdsManager.Instance.Platfrom,
} KwaiAdsManager.Instance.ClientName,
trackId,
AdsType.Rewarded,
Time.realtimeSinceStartup - KwaiAdsManager.Instance._rvStartLoadTime);
} }
} }
} }

View File

@ -325,7 +325,7 @@ namespace WZ
AdPlayCountManager.GetAdPlayCount(type)); AdPlayCountManager.GetAdPlayCount(type));
AdsActionEvents.TrackAdImpression(Platfrom, AdsActionEvents.TrackAdImpression(Platfrom,
adInfo.NetworkName, adInfo.NetworkName,
adInfo.AdUnitIdentifier, adInfo.AdUnitIdentifier,
type, type,
type == AdsType.Rewarded ? _rvPos : "", type == AdsType.Rewarded ? _rvPos : "",

View File

@ -36,6 +36,20 @@ public class AppSDKManager : D_MonoSingleton<AppSDKManager>
#region ad #region ad
/// <summary>
/// 是否已缓存激励广告
/// </summary>
/// <returns></returns>
public bool IsRewardAdReady()
{
return AdsSDKManager.Instance.IsRewardAdReady();;
}
/// <summary>
/// 展示激励广告
/// </summary>
/// <param name="position"></param>
/// <param name="callback"></param>
public void ShowRewardAd(string position, Action<bool,double> callback = null) public void ShowRewardAd(string position, Action<bool,double> callback = null)
{ {
bool isRewardAdReady = AdsSDKManager.Instance.IsRewardAdReady(); bool isRewardAdReady = AdsSDKManager.Instance.IsRewardAdReady();
@ -45,12 +59,35 @@ public class AppSDKManager : D_MonoSingleton<AppSDKManager>
{ {
if (isReward) if (isReward)
{ {
//callback?.Invoke(); AdsSDKManager.Instance.ClearIvRules();
callback?.Invoke(true, revenue); callback?.Invoke(true, revenue);
} }
else
{
callback?.Invoke(false, 0);
}
}); });
} }
else
{
callback?.Invoke(false, 0);
}
} }
/// <summary>
/// 是否已缓存插屏
/// </summary>
public bool IsInterstitialAdReady()
{
return AdsSDKManager.Instance.IsInterstitialReady();
}
/// <summary>
/// 展示插屏广告
/// </summary>
/// <param name="position"></param>
/// <param name="ivadType"></param>
/// <param name="callback"></param>
public void ShowInterstitial(string position, IvType ivadType = IvType.IV1, Action<double> callback = null) public void ShowInterstitial(string position, IvType ivadType = IvType.IV1, Action<double> callback = null)
{ {
//插屏展示逻辑 //插屏展示逻辑
@ -64,8 +101,9 @@ public class AppSDKManager : D_MonoSingleton<AppSDKManager>
AdsSDKManager.Instance.ShowInterstitialAd(position, ivadType, (revenue) => AdsSDKManager.Instance.ShowInterstitialAd(position, ivadType, (revenue) =>
{ {
//展示完一个插屏之后调用 //展示完一个插屏之后调用
IvRulesConst.CurrentOverLevel = 0; IvRulesConst.OverLevels[ivadType.ToString()] = 0;
IvRulesConst.CurrentInterval = TimeUtils.GetLocalTimestamp(); IvRulesConst.Intervals[ivadType.ToString()] = TimeUtils.GetLocalTimestamp();
callback?.Invoke(revenue);
}); });
} }
else else
@ -79,10 +117,7 @@ public class AppSDKManager : D_MonoSingleton<AppSDKManager>
} }
} }
public bool IsRewardAdReady()
{
return AdsSDKManager.Instance.IsRewardAdReady();;
}
#endregion #endregion
@ -128,7 +163,6 @@ public class AppSDKManager : D_MonoSingleton<AppSDKManager>
#endregion #endregion
#region 线 #region 线
public bool GetRemoteConfigBool(string key, bool defaultValue = false) public bool GetRemoteConfigBool(string key, bool defaultValue = false)
@ -169,22 +203,24 @@ public class AppSDKManager : D_MonoSingleton<AppSDKManager>
#endregion #endregion
#region EFSDK #region EFSDK
/// <summary> /// <summary>
/// 展示WebView /// 展示WebView
/// </summary> /// </summary>
/// <param name="id">标签id,存在多个WebView时,用于标识WebView</param> /// <param name="id">标签id,存在多个WebView时,用于标识WebView</param>
/// <param name="pos">广告位</param>
/// <param name="url">网址</param> /// <param name="url">网址</param>
/// <param name="pRect">WebView展示区域的RectTransform</param> /// <param name="pRect">WebView展示区域的RectTransform</param>
/// <param name="pCam">可不传;传值的话要传正交相机</param> /// <param name="pCam">可不传;传值的话要传正交相机</param>
public void ShowWebView(int id, string url, RectTransform pRect, Camera pCam = null) public void ShowWebView(int id, string pos, string url, RectTransform pRect, Camera pCam = null)
{ {
if (Application.isEditor) if (Application.isEditor)
{ {
return; return;
} }
EFSdkManager.Instance.SetOkspinShowPos(pos);
EFSdk.get().ShowWebView(id, url, pRect, pCam); EFSdk.get().ShowWebView(id, url, pRect, pCam);
} }
@ -232,91 +268,8 @@ public class AppSDKManager : D_MonoSingleton<AppSDKManager>
} }
}); });
} }
/// <summary>
/// 是否手动控制漂浮道具显示/隐藏
/// SDK内默认当H5页面加载完成后自动显示漂浮道具
/// </summary>
/// <param name="autoShow">true: 自动显示/隐藏道具 false: 游戏主动控制道具显示/隐藏</param>
/// <returns></returns>
public void AutoShowFloat(bool autoShow)
{
if (Application.isEditor)
{
return;
}
EFSdk.get().AutoShowFloat(false);
}
/// <summary>
/// 新增接口飘金币
/// </summary>
/// <param name="id"></param>
/// <param name="res">悬浮金币按钮的图片资源,传字符串 0 或 1 0:金币图 1:红点宝箱图 </param>
/// <returns></returns>
public void ShowFloatCoin(String res)
{
if (Application.isEditor)
{
return;
}
var remoteConfig = FireBaseRemoteConfigManager.Instance.GetRemoteConfigInt("coin_position", 3);
if (remoteConfig <= 0)
{
remoteConfig = 3;
}
if (remoteConfig > 10)
{
remoteConfig = 3;
}
EFSdk.get().SetFloatCoinRes(res);
EFSdk.get().ShowFloatCoin(remoteConfig);
}
/// <summary>
/// 隐藏金币
/// </summary>
/// <returns></returns>
public void HideFloatCoin()
{
if (Application.isEditor)
{
return;
}
EFSdk.get().HideFloatCoin();
}
/// <param name="startId">宝箱动画起始位置</param>
/// <param name="endId">宝箱动画移动结束位置</param>
/// <param name="fly_first_time">首次delay时间</param>
/// <param name="fly_gap_time">每次漂浮移动的时间间隔</param>
/// <returns></returns>
public void ShowBalloon(int startId, int endId, int fly_first_time, int fly_gap_time)
{
if (Application.isEditor)
{
return;
}
var startFlyIndex = FireBaseRemoteConfigManager.Instance.GetRemoteConfigInt("start_fly", 40);
var endFlyIndex = FireBaseRemoteConfigManager.Instance.GetRemoteConfigInt("end_fly", 60);
var flyFirstTime = FireBaseRemoteConfigManager.Instance.GetRemoteConfigInt("fly_first_time", 3);
var flyGapTime = FireBaseRemoteConfigManager.Instance.GetRemoteConfigInt("fly_gap_time", 15);
EFSdk.get().ShowBalloon(startFlyIndex, endFlyIndex, flyFirstTime, flyGapTime);
}
/// <summary>
/// 隐藏气球
/// </summary>
/// <returns></returns>
public void HideBalloon()
{
if (Application.isEditor)
{
return;
}
EFSdk.get().HideBalloon();
}
/// <summary> /// <summary>
/// 设置推送开关, SDK默认关闭通知 /// 设置推送开关, SDK默认关闭通知
/// </summary> /// </summary>
@ -575,4 +528,13 @@ public class AppSDKManager : D_MonoSingleton<AppSDKManager>
} }
#endregion #endregion
/// <summary>
/// 是否是自然量用户
/// </summary>
/// <returns></returns>
public bool InOrganic()
{
return AdjustNetwork.Instance.InOrganic();
}
} }

View File

@ -6,32 +6,14 @@ namespace WZ
{ {
public class EFSdkManager : NormalSingleton<EFSdkManager> public class EFSdkManager : NormalSingleton<EFSdkManager>
{ {
private const string KEY_OKSPIN_SHOW_COUNT = "OKSPIN_SHOW_COUNT";
//互动广告位
private string okspinShowPos = "";
public void Init() public void Init()
{ {
EFSdk.get().Init((actionType, str) => EFSdk.get().Init((actionType, str) => { });
{
if (EFSdk.ActionType.COIN_CLICK == actionType)
{
//TOTO 游戏在此处理 点击金币弹广告的逻辑或其他
}
if (EFSdk.ActionType.BALLOON_CLICK == actionType)
{
//TOTO 游戏在此处理 点击宝箱弹广告的逻辑或其他
}
if (EFSdk.ActionType.COIN_SHOW == actionType)
{
}
if (EFSdk.ActionType.BOX_SHOW == actionType)
{
}
if (EFSdk.ActionType.GAM_LOAD_SUCC == actionType)
{
// 标签id,标识哪个WebView加载成功了
int id = int.Parse(str);
}
});
SetSDKEventCallback(); SetSDKEventCallback();
SetHdH5ImpressionCallback(); SetHdH5ImpressionCallback();
@ -54,17 +36,38 @@ namespace WZ
{ {
EFSdk.get().SetHdH5ImpressionCallback((url) => EFSdk.get().SetHdH5ImpressionCallback((url) =>
{ {
//TODO 判断是okspin还是appluck //判断链接不为空
string h5ad_okspinrev = FireBaseRemoteConfigManager.Instance.GetRemoteConfigString("H5ad_okspinrev", "0"); if (string.IsNullOrEmpty(url))
string h5ad_appluckrev= FireBaseRemoteConfigManager.Instance.GetRemoteConfigString("H5ad_appluckrev", "0"); {
//TODO 上报 return;
}
//判断链接是互动广告
if (!url.Contains("gamifyspace"))
{
return;
}
//展示次数
int count = PlayerPrefsUtils.GetPlayerPrefsInt(KEY_OKSPIN_SHOW_COUNT, 0);
count++;
PlayerPrefsUtils.SavePlayerPrefsInt(KEY_OKSPIN_SHOW_COUNT, count);
//互动广告只有okSpin
float revenue = FireBaseRemoteConfigManager.Instance.GetRemoteConfigFloat("rev_okspin", 0);
//adjust //adjust
//AdjustTrackEvent.Instance.TrackAdEvent(); AdjustTrackEvent.Instance.TrackAdEvent(revenue, "H5ad_game", url, url);
//firebase //firebase
//FireBaseAnalyticsManager.Instance.OnAdRevenueEvent(); FireBaseAnalyticsManager.Instance.OnAdRevenueEvent("H5ad_game", "H5ad_game", url, AdsType.Fix, revenue, okspinShowPos, count);
//数数 //数数
//ShuShuEvent.Instance.OnAdRevenueEvent(); ShuShuEvent.Instance.OnAdRevenueEvent("H5ad_game", "H5ad_game", url, AdsType.Fix.ToString(), revenue, okspinShowPos, count);
}); });
} }
public void SetOkspinShowPos(string pos)
{
okspinShowPos = pos;
}
} }
} }

View File

@ -22,6 +22,9 @@ namespace WZ
AdConfigParser.Parse(GetRemoteConfigString("ad_config")); AdConfigParser.Parse(GetRemoteConfigString("ad_config"));
// 刷新广告位信息 // 刷新广告位信息
AdsSDKManager.Instance.RefreshAdsData(); AdsSDKManager.Instance.RefreshAdsData();
//AB测试分组参数
GroupSet();
// adjust卸载监控 // adjust卸载监控
/* 执行到这时表示firebase接入正常能获取到远端在线参数 */ /* 执行到这时表示firebase接入正常能获取到远端在线参数 */
@ -37,6 +40,15 @@ namespace WZ
// }); // });
} }
private void GroupSet()
{
string value = GetRemoteConfigString("group_set");
//数数
ShuShuEvent.Instance.Track($"group_set_{value}");
//firebase
FireBaseAnalyticsManager.Instance.LogEvent($"group_set_{value}");
}
/// <summary> /// <summary>
/// 获取int参数 /// 获取int参数

View File

@ -1,9 +1,10 @@
using System.Collections; using System.Collections;
using System.Collections.Generic; using System.Collections.Generic;
using System.Globalization;
using UnityEngine; using UnityEngine;
namespace WZ namespace WZ
{ {
public static class DataUtils public static class DataUtils
{ {
[System.Serializable] [System.Serializable]
@ -18,6 +19,20 @@ namespace WZ
Wrapper<T> wrapper = JsonUtility.FromJson<Wrapper<T>>(wrappedJson); Wrapper<T> wrapper = JsonUtility.FromJson<Wrapper<T>>(wrappedJson);
return wrapper.items; return wrapper.items;
} }
public static double StringToDouble(string str)
{
double result = 0;
if (double.TryParse(str, NumberStyles.Any, CultureInfo.InvariantCulture, out result))
{
Debug.Log("转换成功: " + result);
}
else
{
Debug.Log("转换失败:字符串格式不正确");
}
return result;
}
} }
} }

View File

@ -0,0 +1,3 @@
fileFormatVersion: 2
guid: fa22f34439f14c8cbaa5fae4c4dc402b
timeCreated: 1742203865

View File

@ -0,0 +1,3 @@
fileFormatVersion: 2
guid: f322db129aac4ea1aa9919e8c8f92a08
timeCreated: 1756693458

View File

@ -0,0 +1,3 @@
fileFormatVersion: 2
guid: 8af1d23b4bce430aa8bfb3e1b6b3b291
timeCreated: 1756693481

Binary file not shown.

After

Width:  |  Height:  |  Size: 280 KiB

View File

@ -0,0 +1,7 @@
fileFormatVersion: 2
guid: 22d4b822110d1bd43932b2b737f61dce
DefaultImporter:
externalObjects: {}
userData:
assetBundleName:
assetBundleVariant:

Binary file not shown.

After

Width:  |  Height:  |  Size: 9.4 KiB

View File

@ -0,0 +1,7 @@
fileFormatVersion: 2
guid: 63a7e60d370bb1c48a1138e45315d8aa
DefaultImporter:
externalObjects: {}
userData:
assetBundleName:
assetBundleVariant:

Binary file not shown.

After

Width:  |  Height:  |  Size: 9.4 KiB

View File

@ -0,0 +1,7 @@
fileFormatVersion: 2
guid: 9cb4efb8034485741852541144c5a7a0
DefaultImporter:
externalObjects: {}
userData:
assetBundleName:
assetBundleVariant:

Binary file not shown.

After

Width:  |  Height:  |  Size: 12 KiB

View File

@ -0,0 +1,7 @@
fileFormatVersion: 2
guid: 9934c44d94c1030438702e646cc1a5b1
DefaultImporter:
externalObjects: {}
userData:
assetBundleName:
assetBundleVariant:

Binary file not shown.

After

Width:  |  Height:  |  Size: 11 KiB

View File

@ -0,0 +1,7 @@
fileFormatVersion: 2
guid: 9cfb6b46fea6c7845999842064fef0dd
DefaultImporter:
externalObjects: {}
userData:
assetBundleName:
assetBundleVariant:

Binary file not shown.

After

Width:  |  Height:  |  Size: 21 KiB

View File

@ -0,0 +1,7 @@
fileFormatVersion: 2
guid: a0fa3d13d2c47b64c8a10c49839f888c
DefaultImporter:
externalObjects: {}
userData:
assetBundleName:
assetBundleVariant:

Binary file not shown.

After

Width:  |  Height:  |  Size: 22 KiB

View File

@ -0,0 +1,7 @@
fileFormatVersion: 2
guid: 10def9eb660635b468ed7fc2277a4e6f
DefaultImporter:
externalObjects: {}
userData:
assetBundleName:
assetBundleVariant:

Binary file not shown.

After

Width:  |  Height:  |  Size: 89 KiB

View File

@ -0,0 +1,3 @@
fileFormatVersion: 2
guid: f2d7efc9e85a47e98bae42a19e1d52a3
timeCreated: 1756693525