#if UNITY_WEBGL && WEIXINMINIGAME using System; using System.Collections.Generic; using System.Linq; using UnityEngine; using YooAsset; using WeChatWASM; public static class WechatFileSystemCreater { public static FileSystemParameters CreateWechatFileSystemParameters(IRemoteServices remoteServices, string packageRoot) { string fileSystemClass = $"{nameof(WechatFileSystem)},YooAsset.RuntimeExtension"; var fileSystemParams = new FileSystemParameters(fileSystemClass, packageRoot); fileSystemParams.AddParameter("REMOTE_SERVICES", remoteServices); return fileSystemParams; } } /// /// 微信小游戏文件系统 /// 参考:https://wechat-miniprogram.github.io/minigame-unity-webgl-transform/Design/UsingAssetBundle.html /// internal class WechatFileSystem : IFileSystem { private class WebRemoteServices : IRemoteServices { private readonly string _webPackageRoot; protected readonly Dictionary _mapping = new Dictionary(10000); public WebRemoteServices(string buildinPackRoot) { _webPackageRoot = buildinPackRoot; } string IRemoteServices.GetRemoteMainURL(string fileName) { return GetFileLoadURL(fileName); } string IRemoteServices.GetRemoteFallbackURL(string fileName) { return GetFileLoadURL(fileName); } private string GetFileLoadURL(string fileName) { if (_mapping.TryGetValue(fileName, out string url) == false) { string filePath = PathUtility.Combine(_webPackageRoot, fileName); url = DownloadSystemHelper.ConvertToWWWPath(filePath); _mapping.Add(fileName, url); } return url; } } private readonly HashSet _recorders = new HashSet(); private readonly Dictionary _cacheFilePaths = new Dictionary(10000); private WXFileSystemManager _fileSystemMgr; private string _wxCacheRoot = string.Empty; /// /// 包裹名称 /// public string PackageName { private set; get; } /// /// 文件根目录 /// public string FileRoot { get { return _wxCacheRoot; } } /// /// 文件数量 /// public int FileCount { get { return _recorders.Count; } } #region 自定义参数 /// /// 自定义参数:远程服务接口 /// public IRemoteServices RemoteServices { private set; get; } = null; #endregion public WechatFileSystem() { } public virtual FSInitializeFileSystemOperation InitializeFileSystemAsync() { var operation = new WXFSInitializeOperation(this); OperationSystem.StartOperation(PackageName, operation); return operation; } public virtual FSLoadPackageManifestOperation LoadPackageManifestAsync(string packageVersion, int timeout) { var operation = new WXFSLoadPackageManifestOperation(this, packageVersion, timeout); OperationSystem.StartOperation(PackageName, operation); return operation; } public virtual FSRequestPackageVersionOperation RequestPackageVersionAsync(bool appendTimeTicks, int timeout) { var operation = new WXFSRequestPackageVersionOperation(this, appendTimeTicks, timeout); OperationSystem.StartOperation(PackageName, operation); return operation; } public virtual FSClearCacheFilesOperation ClearCacheFilesAsync(PackageManifest manifest, string clearMode, object clearParam) { if (clearMode == EFileClearMode.ClearAllBundleFiles.ToString()) { var operation = new WXFSClearAllBundleFilesOperation(this); OperationSystem.StartOperation(PackageName, operation); return operation; } else if (clearMode == EFileClearMode.ClearUnusedBundleFiles.ToString()) { var operation = new WXFSClearUnusedBundleFilesAsync(this, manifest); OperationSystem.StartOperation(PackageName, operation); return operation; } else { string error = $"Invalid clear mode : {clearMode}"; var operation = new FSClearCacheFilesCompleteOperation(error); OperationSystem.StartOperation(PackageName, operation); return operation; } } public virtual FSDownloadFileOperation DownloadFileAsync(PackageBundle bundle, DownloadParam param) { param.MainURL = RemoteServices.GetRemoteMainURL(bundle.FileName); param.FallbackURL = RemoteServices.GetRemoteFallbackURL(bundle.FileName); var operation = new WXFSDownloadFileOperation(this, bundle, param); OperationSystem.StartOperation(PackageName, operation); return operation; } public virtual FSLoadBundleOperation LoadBundleFile(PackageBundle bundle) { if (bundle.BundleType == (int)EBuildBundleType.AssetBundle) { var operation = new WXFSLoadBundleOperation(this, bundle); OperationSystem.StartOperation(PackageName, operation); return operation; } else { string error = $"{nameof(WechatFileSystem)} not support load bundle type : {bundle.BundleType}"; var operation = new FSLoadBundleCompleteOperation(error); OperationSystem.StartOperation(PackageName, operation); return operation; } } public virtual void SetParameter(string name, object value) { if (name == "REMOTE_SERVICES") { RemoteServices = (IRemoteServices)value; } else { YooLogger.Warning($"Invalid parameter : {name}"); } } public virtual void OnCreate(string packageName, string rootDirectory) { PackageName = packageName; _wxCacheRoot = rootDirectory; if (string.IsNullOrEmpty(_wxCacheRoot)) { throw new System.Exception("请配置微信小游戏缓存根目录!"); } // 注意:CDN服务未启用的情况下,使用微信WEB服务器 if (RemoteServices == null) { string webRoot = PathUtility.Combine(Application.streamingAssetsPath, YooAssetSettingsData.Setting.DefaultYooFolderName, packageName); RemoteServices = new WebRemoteServices(webRoot); } _fileSystemMgr = WX.GetFileSystemManager(); } public virtual void OnUpdate() { } public virtual bool Belong(PackageBundle bundle) { return true; } public virtual bool Exists(PackageBundle bundle) { string filePath = GetCacheFileLoadPath(bundle); return _recorders.Contains(filePath); } public virtual bool NeedDownload(PackageBundle bundle) { if (Belong(bundle) == false) return false; return Exists(bundle) == false; } public virtual bool NeedUnpack(PackageBundle bundle) { return false; } public virtual bool NeedImport(PackageBundle bundle) { return false; } public virtual string GetBundleFilePath(PackageBundle bundle) { return GetCacheFileLoadPath(bundle); } public virtual byte[] ReadBundleFileData(PackageBundle bundle) { string filePath = GetCacheFileLoadPath(bundle); if (CheckCacheFileExist(filePath)) return _fileSystemMgr.ReadFileSync(filePath); else return Array.Empty(); } public virtual string ReadBundleFileText(PackageBundle bundle) { string filePath = GetCacheFileLoadPath(bundle); if (CheckCacheFileExist(filePath)) return _fileSystemMgr.ReadFileSync(filePath, "utf8"); else return string.Empty; } #region 内部方法 public WXFileSystemManager GetFileSystemMgr() { return _fileSystemMgr; } public bool CheckCacheFileExist(string filePath) { string result = _fileSystemMgr.AccessSync(filePath); return result.Equals("access:ok"); } public string GetCacheFileLoadPath(PackageBundle bundle) { if (_cacheFilePaths.TryGetValue(bundle.BundleGUID, out string filePath) == false) { filePath = PathUtility.Combine(_wxCacheRoot, bundle.FileName); _cacheFilePaths.Add(bundle.BundleGUID, filePath); } return filePath; } #endregion #region 本地记录 public List GetAllRecords() { return _recorders.ToList(); } public bool RecordBundleFile(string filePath) { if (_recorders.Contains(filePath)) { YooLogger.Error($"{nameof(WechatFileSystem)} has element : {filePath}"); return false; } _recorders.Add(filePath); return true; } public void TryRecordBundle(PackageBundle bundle) { string filePath = GetCacheFileLoadPath(bundle); if (_recorders.Contains(filePath) == false) { _recorders.Add(filePath); } } public void ClearAllRecords() { _recorders.Clear(); } public void ClearRecord(string filePath) { if (_recorders.Contains(filePath)) { _recorders.Remove(filePath); } } #endregion } #endif