using System; using System.Collections.Generic; using System.IO; using System.Linq; using System.Xml.Linq; using MyRoomCustom; using Newtonsoft.Json; using UnityEngine; using wf; using static MeidoPhotoStudio.Plugin.MenuFileUtility; namespace MeidoPhotoStudio.Plugin; public static class Constants { public const string customPoseDirectory = "Custom Poses"; public const string customHandDirectory = "Hand Presets"; public const string customFaceDirectory = "Face Presets"; public const string sceneDirectory = "Scenes"; public const string kankyoDirectory = "Environments"; public const string configDirectory = "MeidoPhotoStudio"; public const string translationDirectory = "Translations"; public const string databaseDirectory = "Database"; public enum Window { Call, Pose, Face, BG, BG2, Main, Message, Save, SaveModal, Settings } public enum Scene { Daily = 3, Edit = 5 } public enum DoguCategory { Other, Mob, Desk, HandItem, BGSmall } public static readonly string customPosePath; public static readonly string customHandPath; public static readonly string customFacePath; public static readonly string scenesPath; public static readonly string kankyoPath; public static readonly string configPath; public static readonly string databasePath; public static readonly int mainWindowID = 765; public static readonly int messageWindowID = 961; public static readonly int sceneManagerWindowID = 876; public static readonly int sceneManagerModalID = 283; public static readonly int dropdownWindowID = 777; public static readonly List PoseGroupList = new(); public static readonly Dictionary> PoseDict = new(); public static readonly List CustomPoseGroupList = new(); public static readonly Dictionary> CustomPoseDict = new(); public static readonly List CustomHandGroupList = new(); public static readonly Dictionary> CustomHandDict = new(); public static readonly List FaceGroupList = new(); public static readonly Dictionary> FaceDict = new(); public static readonly List CustomFaceGroupList = new(); public static readonly Dictionary> CustomFaceDict = new(); public static readonly List BGList = new(); public static readonly List> MyRoomCustomBGList = new(); public static readonly List DoguCategories = new(); public static readonly Dictionary> DoguDict = new(); public static readonly List MyRoomPropCategories = new(); public static readonly Dictionary> MyRoomPropDict = new(); public static readonly Dictionary> ModPropDict = new(StringComparer.InvariantCultureIgnoreCase); public static readonly List SceneDirectoryList = new(); public static readonly List KankyoDirectoryList = new(); public static readonly List MpnAttachPropList = new(); public static readonly Dictionary customDoguCategories = new() { [DoguCategory.Other] = "other", [DoguCategory.Mob] = "mob", [DoguCategory.Desk] = "desk", [DoguCategory.HandItem] = "handItem", [DoguCategory.BGSmall] = "bgSmall" }; public static int MyRoomCustomBGIndex { get; private set; } = -1; public static bool HandItemsInitialized { get; private set; } public static bool MpnAttachInitialized { get; private set; } public static bool MenuFilesInitialized { get; private set; } public static event EventHandler MenuFilesChange; public static event EventHandler CustomPoseChange; public static event EventHandler CustomHandChange; public static event EventHandler CustomFaceChange; private static bool beginHandItemInit; private static bool beginMpnAttachInit; static Constants() { configPath = Path.Combine(BepInEx.Paths.ConfigPath, configDirectory); var presetPath = Path.Combine(configPath, "Presets"); customPosePath = Path.Combine(presetPath, customPoseDirectory); customHandPath = Path.Combine(presetPath, customHandDirectory); customFacePath = Path.Combine(presetPath, customFaceDirectory); scenesPath = Path.Combine(configPath, sceneDirectory); kankyoPath = Path.Combine(configPath, kankyoDirectory); databasePath = Path.Combine(configPath, databaseDirectory); var directories = new[] { customPosePath, customHandPath, scenesPath, kankyoPath, configPath, customFacePath, databasePath }; foreach (var directory in directories) if (!Directory.Exists(directory)) Directory.CreateDirectory(directory); } public static void Initialize() { InitializeSceneDirectories(); InitializeKankyoDirectories(); InitializePoses(); InitializeHandPresets(); InitializeFaceBlends(); InitializeBGs(); InitializeDogu(); InitializeMyRoomProps(); InitializeMpnAttachProps(); } public static void AddFacePreset(Dictionary faceData, string filename, string directory) { filename = Utility.SanitizePathPortion(filename); directory = Utility.SanitizePathPortion(directory); if (string.IsNullOrEmpty(filename)) filename = "face_preset"; if (directory.Equals(customFaceDirectory, StringComparison.InvariantCultureIgnoreCase)) directory = string.Empty; directory = Path.Combine(customFacePath, directory); if (!Directory.Exists(directory)) Directory.CreateDirectory(directory); var fullPath = Path.Combine(directory, filename); if (File.Exists($"{fullPath}.xml")) fullPath += $"_{Utility.Timestamp}"; fullPath = Path.GetFullPath($"{fullPath}.xml"); if (!fullPath.StartsWith(customFacePath)) { Utility.LogError($"Could not save face preset! Path is invalid: '{fullPath}'"); return; } var rootElement = new XElement("FaceData"); foreach (var kvp in faceData) rootElement.Add(new XElement("elm", kvp.Value.ToString("G9"), new XAttribute("name", kvp.Key))); var fullDocument = new XDocument( new XDeclaration("1.0", "utf-8", "true"), new XComment("MeidoPhotoStudio Face Preset"), rootElement ); fullDocument.Save(fullPath); var fileInfo = new FileInfo(fullPath); var category = fileInfo.Directory.Name; var faceGroup = CustomFaceGroupList.Find( group => string.Equals(category, group, StringComparison.InvariantCultureIgnoreCase) ); if (string.IsNullOrEmpty(faceGroup)) { CustomFaceGroupList.Add(category); CustomFaceDict[category] = new(); CustomFaceGroupList.Sort((a, b) => KeepAtTop(a, b, customFaceDirectory)); } else category = faceGroup; CustomFaceDict[category].Add(fullPath); CustomFaceDict[category].Sort(WindowsLogicalComparer.StrCmpLogicalW); CustomFaceChange?.Invoke(null, new(fullPath, category)); } public static void AddPose(byte[] anmBinary, string filename, string directory) { // TODO: Consider writing a file system monitor filename = Utility.SanitizePathPortion(filename); directory = Utility.SanitizePathPortion(directory); if (string.IsNullOrEmpty(filename)) filename = "custom_pose"; if (directory.Equals(customPoseDirectory, StringComparison.InvariantCultureIgnoreCase)) directory = string.Empty; directory = Path.Combine(customPosePath, directory); if (!Directory.Exists(directory)) Directory.CreateDirectory(directory); var fullPath = Path.Combine(directory, filename); if (File.Exists($"{fullPath}.anm")) fullPath += $"_{Utility.Timestamp}"; fullPath = Path.GetFullPath($"{fullPath}.anm"); if (!fullPath.StartsWith(customPosePath)) { Utility.LogError($"Could not save pose! Path is invalid: '{fullPath}'"); return; } File.WriteAllBytes(fullPath, anmBinary); var fileInfo = new FileInfo(fullPath); var category = fileInfo.Directory.Name; var poseGroup = CustomPoseGroupList.Find( group => string.Equals(category, group, StringComparison.InvariantCultureIgnoreCase) ); if (string.IsNullOrEmpty(poseGroup)) { CustomPoseGroupList.Add(category); CustomPoseDict[category] = new(); CustomPoseGroupList.Sort((a, b) => KeepAtTop(a, b, customPoseDirectory)); } else category = poseGroup; CustomPoseDict[category].Add(fullPath); CustomPoseDict[category].Sort(WindowsLogicalComparer.StrCmpLogicalW); CustomPoseChange?.Invoke(null, new(fullPath, category)); } public static void AddHand(byte[] handBinary, bool right, string filename, string directory) { filename = Utility.SanitizePathPortion(filename); directory = Utility.SanitizePathPortion(directory); if (string.IsNullOrEmpty(filename)) filename = "custom_hand"; if (directory.Equals(customHandDirectory, StringComparison.InvariantCultureIgnoreCase)) directory = string.Empty; directory = Path.Combine(customHandPath, directory); if (!Directory.Exists(directory)) Directory.CreateDirectory(directory); var fullPath = Path.Combine(directory, filename); if (File.Exists($"{fullPath}.xml")) fullPath += $"_{Utility.Timestamp}"; fullPath = Path.GetFullPath($"{fullPath}.xml"); if (!fullPath.StartsWith(customHandPath)) { Utility.LogError($"Could not save hand! Path is invalid: '{fullPath}'"); return; } // TODO: This does not actually do what I think it does. var gameVersion = Misc.GAME_VERSION; // get game version from user's Assembly-CSharp var finalXml = new XDocument( new XDeclaration("1.0", "utf-8", "true"), new XComment("CM3D2 FingerData"), new XElement( "FingerData", new XElement("GameVersion", gameVersion), new XElement("RightData", right), new XElement("BinaryData", Convert.ToBase64String(handBinary)) ) ); finalXml.Save(fullPath); var fileInfo = new FileInfo(fullPath); var category = fileInfo.Directory.Name; var handGroup = CustomHandGroupList.Find( group => string.Equals(category, group, StringComparison.InvariantCultureIgnoreCase) ); if (string.IsNullOrEmpty(handGroup)) { CustomHandGroupList.Add(category); CustomHandDict[category] = new(); CustomHandGroupList.Sort((a, b) => KeepAtTop(a, b, customHandDirectory)); } else category = handGroup; CustomHandDict[category].Add(fullPath); CustomHandDict[category].Sort(WindowsLogicalComparer.StrCmpLogicalW); CustomHandChange?.Invoke(null, new(fullPath, category)); } public static void InitializeSceneDirectories() { SceneDirectoryList.Clear(); SceneDirectoryList.Add(sceneDirectory); foreach (var directory in Directory.GetDirectories(scenesPath)) SceneDirectoryList.Add(new DirectoryInfo(directory).Name); SceneDirectoryList.Sort((a, b) => KeepAtTop(a, b, sceneDirectory)); } public static void InitializeKankyoDirectories() { KankyoDirectoryList.Clear(); KankyoDirectoryList.Add(kankyoDirectory); foreach (var directory in Directory.GetDirectories(kankyoPath)) KankyoDirectoryList.Add(new DirectoryInfo(directory).Name); KankyoDirectoryList.Sort((a, b) => KeepAtTop(a, b, kankyoDirectory)); } public static void InitializePoses() { // Load Poses var poseListPath = Path.Combine(databasePath, "mm_pose_list.json"); try { var poseListJson = File.ReadAllText(poseListPath); foreach (var poseList in JsonConvert.DeserializeObject>(poseListJson)) { PoseDict[poseList.UIName] = poseList.PoseList; PoseGroupList.Add(poseList.UIName); } } catch (IOException e) { Utility.LogError($"Could not open pose database because {e.Message}"); Utility.LogMessage("Pose list will be serverely limited."); AddDefaultPose(); } catch (Exception e) { Utility.LogError($"Could not parse pose database because {e.Message}"); AddDefaultPose(); } // Get Other poses that'll go into Normal, Normal 2 and Ero 2 var com3d2MotionList = GameUty.FileSystem.GetList("motion", AFileSystemBase.ListType.AllFile); if (com3d2MotionList?.Length > 0) { var poseSet = new HashSet(); foreach (var poses in PoseDict.Values) poseSet.UnionWith(poses); var newCategories = new[] { "normal2", "ero2" }; foreach (var category in newCategories) if (!PoseDict.ContainsKey(category)) PoseDict[category] = new(); // TODO: Try to group these poses into more than "normal2" and "ero2" foreach (var path in com3d2MotionList) { if (Path.GetExtension(path) is not ".anm") continue; var file = Path.GetFileNameWithoutExtension(path); if (poseSet.Contains(file)) continue; if (file.StartsWith("edit_")) PoseDict["normal"].Add(file); else if (file is not ("dance_cm3d2_001_zoukin" or "dance_cm3d2_001_mop" or "aruki_1_idougo_f" or "sleep2" or "stand_akire2") && !file.EndsWith("_3_") && !file.EndsWith("_5_") && !file.StartsWith("vr_") && !file.StartsWith("dance_mc") && !file.Contains("_kubi_") && !file.Contains("a01_") && !file.Contains("b01_") && !file.Contains("b02_") && !file.EndsWith("_m2") && !file.EndsWith("_m2_once_") && !file.StartsWith("h_") && !file.StartsWith("event_") && !file.StartsWith("man_") && !file.EndsWith("_m") && !file.Contains("_m_") && !file.Contains("_man_") ) { if (path.Contains(@"\sex\")) PoseDict["ero2"].Add(file); else PoseDict["normal2"].Add(file); } } foreach (var category in newCategories) { if (PoseDict[category].Count > 0) { if (!PoseGroupList.Contains(category)) PoseGroupList.Add(category); } else PoseDict.Remove(category); } } InitializeCustomPoses(); static void AddDefaultPose() { if (!PoseDict.ContainsKey("normal")) PoseDict["normal"] = new() { "maid_stand01" }; if (!PoseGroupList.Contains("normal")) PoseGroupList.Insert(0, "normal"); } } public static void InitializeCustomPoses() { CustomPoseGroupList.Clear(); CustomPoseDict.Clear(); CustomPoseGroupList.Add(customPoseDirectory); CustomPoseDict[customPoseDirectory] = new(); GetPoses(customPosePath); foreach (var directory in Directory.GetDirectories(customPosePath)) GetPoses(directory); CustomPoseGroupList.Sort((a, b) => KeepAtTop(a, b, customPoseDirectory)); CustomPoseChange?.Invoke(null, PresetChangeEventArgs.Empty); static void GetPoses(string directory) { var poseList = Directory.GetFiles(directory) .Where(file => Path.GetExtension(file) is ".anm"); if (poseList.Any()) { var poseGroupName = new DirectoryInfo(directory).Name; if (poseGroupName != customPoseDirectory) CustomPoseGroupList.Add(poseGroupName); CustomPoseDict[poseGroupName] = poseList.ToList(); CustomPoseDict[poseGroupName].Sort(WindowsLogicalComparer.StrCmpLogicalW); } } } public static void InitializeHandPresets() { CustomHandGroupList.Clear(); CustomHandDict.Clear(); CustomHandGroupList.Add(customHandDirectory); CustomHandDict[customHandDirectory] = new(); GetPresets(customHandPath); foreach (var directory in Directory.GetDirectories(customHandPath)) GetPresets(directory); CustomHandGroupList.Sort((a, b) => KeepAtTop(a, b, customHandDirectory)); CustomHandChange?.Invoke(null, PresetChangeEventArgs.Empty); static void GetPresets(string directory) { var presetList = Directory.GetFiles(directory) .Where(file => Path.GetExtension(file) is ".xml"); if (!presetList.Any()) return; var presetCategory = new DirectoryInfo(directory).Name; if (presetCategory != customHandDirectory) CustomHandGroupList.Add(presetCategory); CustomHandDict[presetCategory] = presetList.ToList(); CustomHandDict[presetCategory].Sort(WindowsLogicalComparer.StrCmpLogicalW); } } public static void InitializeFaceBlends() { PhotoFaceData.Create(); FaceGroupList.AddRange(PhotoFaceData.popup_category_list.Select(kvp => kvp.Key)); foreach (var kvp in PhotoFaceData.category_list) FaceDict[kvp.Key] = kvp.Value.ConvertAll(data => data.setting_name); InitializeCustomFaceBlends(); } public static void InitializeCustomFaceBlends() { CustomFaceGroupList.Clear(); CustomFaceDict.Clear(); CustomFaceGroupList.Add(customFaceDirectory); CustomFaceDict[customFaceDirectory] = new(); GetFacePresets(customFacePath); foreach (var directory in Directory.GetDirectories(customFacePath)) GetFacePresets(directory); CustomFaceGroupList.Sort((a, b) => KeepAtTop(a, b, customFaceDirectory)); CustomFaceChange?.Invoke(null, PresetChangeEventArgs.Empty); static void GetFacePresets(string directory) { IEnumerable presetList = Directory.GetFiles(directory) .Where(file => Path.GetExtension(file) is ".xml").ToList(); if (presetList.Any()) { var faceGroupName = new DirectoryInfo(directory).Name; if (faceGroupName != customFaceDirectory) CustomFaceGroupList.Add(faceGroupName); CustomFaceDict[faceGroupName] = presetList.ToList(); CustomFaceDict[faceGroupName].Sort(WindowsLogicalComparer.StrCmpLogicalW); } } } public static void InitializeBGs() { // Load BGs PhotoBGData.Create(); // COM3D2 BGs foreach (var bgData in PhotoBGData.data) { if (string.IsNullOrEmpty(bgData.create_prefab_name)) continue; var bg = bgData.create_prefab_name; BGList.Add(bg); } // CM3D2 BGs if (GameUty.IsEnabledCompatibilityMode) { using var csvParser = OpenCsvParser("phot_bg_list.nei", GameUty.FileSystemOld); for (var cell_y = 1; cell_y < csvParser.max_cell_y; cell_y++) { if (!csvParser.IsCellToExistData(3, cell_y)) continue; var bg = csvParser.GetCellAsString(3, cell_y); BGList.Add(bg); } } // Set index regardless of there being myRoom bgs or not MyRoomCustomBGIndex = BGList.Count; var saveDataDict = CreativeRoomManager.GetSaveDataDic(); if (saveDataDict is not null) MyRoomCustomBGList.AddRange(saveDataDict); } public static void InitializeDogu() { foreach (var customCategory in customDoguCategories.Values) DoguDict[customCategory] = new(); InitializeDeskItems(); InitializePhotoBGItems(); InitializeOtherDogu(); InitializeHandItems(); foreach (var category in PhotoBGObjectData.popup_category_list.Select(kvp => kvp.Key)) { if (category is "マイオブジェクト") continue; DoguCategories.Add(category); } foreach (DoguCategory category in Enum.GetValues(typeof(DoguCategory))) DoguCategories.Add(customDoguCategories[category]); } public static List GetModPropList(string category) { if (!PropManager.ModItemsOnly && !MenuFilesReady) { Utility.LogMessage("Menu files are not ready yet"); return null; } if (!MenuFilesInitialized) InitializeModProps(); if (!ModPropDict.ContainsKey(category)) return null; var selectedList = ModPropDict[category]; if (selectedList[0].Icon == null) { selectedList.Sort((a, b) => { var res = a.Priority.CompareTo(b.Priority); if (res is 0) res = string.Compare(a.Name, b.Name); return res; }); var previousMenuFile = string.Empty; selectedList.RemoveAll(item => { if (item.Icon != null) return false; Texture2D icon; var iconFile = item.IconFile; if (string.IsNullOrEmpty(iconFile)) { Utility.LogWarning($"Could not find icon '{iconFile}' for menu '{item.MenuFile}"); return true; } try { icon = ImportCM.CreateTexture(iconFile); } catch { try { icon = ImportCM.CreateTexture($"tex\\{iconFile}"); } catch { Utility.LogWarning($"Could not load '{iconFile}' for menu '{item.MenuFile}"); return true; } } item.Icon = icon; return false; }); } return selectedList; } private static void InitializeOtherDogu() { DoguDict[customDoguCategories[DoguCategory.BGSmall]] = BGList; DoguDict[customDoguCategories[DoguCategory.Mob]].AddRange( new[] { "Mob_Man_Stand001", "Mob_Man_Stand002", "Mob_Man_Stand003", "Mob_Man_Sit001", "Mob_Man_Sit002", "Mob_Man_Sit003", "Mob_Girl_Stand001", "Mob_Girl_Stand002", "Mob_Girl_Stand003", "Mob_Girl_Sit001", "Mob_Girl_Sit002", "Mob_Girl_Sit003" } ); var DoguList = DoguDict[customDoguCategories[DoguCategory.Other]]; // bg object extend var doguHashSet = new HashSet(StringComparer.InvariantCultureIgnoreCase); doguHashSet.UnionWith(BGList); try { var ignoreListPath = Path.Combine(databasePath, "bg_ignore_list.json"); var ignoreListJson = File.ReadAllText(ignoreListPath); var ignoreList = JsonConvert.DeserializeObject(ignoreListJson); doguHashSet.UnionWith(ignoreList); } catch (IOException e) { Utility.LogWarning($"Could not open ignored BG database because {e.Message}"); } catch { } foreach (var doguList in DoguDict.Values) doguHashSet.UnionWith(doguList); foreach (var path in GameUty.FileSystem.GetList("bg", AFileSystemBase.ListType.AllFile)) { if (Path.GetExtension(path) is not ".asset_bg" || path.Contains("myroomcustomize")) continue; var file = Path.GetFileNameWithoutExtension(path); if (doguHashSet.Contains(file) || file.EndsWith("_hit")) continue; DoguList.Add(file); doguHashSet.Add(file); } // Get cherry picked dogu that I can't find in the game files try { var doguExtendPath = Path.Combine(databasePath, "extra_dogu.json"); var doguExtendJson = File.ReadAllText(doguExtendPath); DoguList.AddRange(JsonConvert.DeserializeObject>(doguExtendJson)); } catch (IOException e) { Utility.LogError($"Could not open extra prop database because {e.Message}"); } catch (Exception e) { Utility.LogError($"Could not parse extra prop database because {e.Message}"); } foreach (var path in GameUty.FileSystemOld.GetList("bg", AFileSystemBase.ListType.AllFile)) { if (Path.GetExtension(path) is not ".asset_bg") continue; var file = Path.GetFileNameWithoutExtension(path); if (!doguHashSet.Contains(file) && !file.EndsWith("_not_optimisation")) DoguList.Add(file); } } private static void InitializeDeskItems() { var enabledIDs = new HashSet(); CsvCommonIdManager.ReadEnabledIdList( CsvCommonIdManager.FileSystemType.Normal, true, "desk_item_enabled_id", ref enabledIDs ); CsvCommonIdManager.ReadEnabledIdList( CsvCommonIdManager.FileSystemType.Old, true, "desk_item_enabled_id", ref enabledIDs ); var com3d2DeskDogu = DoguDict[customDoguCategories[DoguCategory.Desk]]; void GetDeskItems(AFileSystemBase fs) { using var csvParser = OpenCsvParser("desk_item_detail.nei", fs); for (var cell_y = 1; cell_y < csvParser.max_cell_y; cell_y++) { if (!csvParser.IsCellToExistData(0, cell_y)) continue; var cell = csvParser.GetCellAsInteger(0, cell_y); if (!enabledIDs.Contains(cell)) continue; var dogu = string.Empty; if (csvParser.IsCellToExistData(3, cell_y)) dogu = csvParser.GetCellAsString(3, cell_y); else if (csvParser.IsCellToExistData(4, cell_y)) dogu = csvParser.GetCellAsString(4, cell_y); if (!string.IsNullOrEmpty(dogu)) com3d2DeskDogu.Add(dogu); } } GetDeskItems(GameUty.FileSystem); } private static void InitializePhotoBGItems() { PhotoBGObjectData.Create(); var photoBGObjectList = PhotoBGObjectData.data; var doguCategories = new List(); var addedCategories = new HashSet(StringComparer.InvariantCultureIgnoreCase); foreach (var photoBGObject in photoBGObjectList) { var category = photoBGObject.category; if (!addedCategories.Contains(category)) { addedCategories.Add(category); doguCategories.Add(category); } if (!DoguDict.ContainsKey(category)) DoguDict[category] = new(); var dogu = string.Empty; if (!string.IsNullOrEmpty(photoBGObject.create_prefab_name)) dogu = photoBGObject.create_prefab_name; else if (!string.IsNullOrEmpty(photoBGObject.create_asset_bundle_name)) dogu = photoBGObject.create_asset_bundle_name; else if (!string.IsNullOrEmpty(photoBGObject.direct_file)) dogu = photoBGObject.direct_file; if (!string.IsNullOrEmpty(dogu)) DoguDict[category].Add(dogu); } DoguDict["パーティクル"].AddRange( new[] { "Particle/pLineY", "Particle/pLineP02", "Particle/pHeart01", "Particle/pLine_act2", "Particle/pstarY_act2" } ); } private static void InitializeHandItems() { if (HandItemsInitialized) return; if (!MenuFilesReady) { if (!beginHandItemInit) MenuFilesReadyChange += (s, a) => InitializeHandItems(); beginHandItemInit = true; return; } var menuDataBase = GameMain.Instance.MenuDataBase; var doguHashSet = new HashSet(StringComparer.InvariantCultureIgnoreCase); doguHashSet.UnionWith(BGList); try { var ignoreListPath = Path.Combine(databasePath, "bg_ignore_list.json"); var ignoreListJson = File.ReadAllText(ignoreListPath); var ignoreList = JsonConvert.DeserializeObject(ignoreListJson); doguHashSet.UnionWith(ignoreList); } catch (IOException e) { Utility.LogWarning($"Could not open ignored BG database because {e.Message}"); } catch (Exception e) { Utility.LogError($"Could not parse ignored BG database because {e.Message}"); } foreach (var doguList in DoguDict.Values) doguHashSet.UnionWith(doguList); var category = customDoguCategories[DoguCategory.HandItem]; for (var i = 0; i < menuDataBase.GetDataSize(); i++) { menuDataBase.SetIndex(i); if ((MPN)menuDataBase.GetCategoryMpn() is not MPN.handitem) continue; var menuFileName = menuDataBase.GetMenuFileName(); if (menuDataBase.GetBoDelOnly() || menuFileName.EndsWith("_del.menu")) continue; var handItemAsOdogu = Utility.HandItemToOdogu(menuFileName); var isolatedHandItem = menuFileName.Substring(menuFileName.IndexOf('_') + 1); if (doguHashSet.Contains(handItemAsOdogu) || doguHashSet.Contains(isolatedHandItem)) continue; doguHashSet.Add(isolatedHandItem); DoguDict[category].Add(menuFileName); // Check for a half deck of cards to add the full deck as well if (menuFileName is "handitemd_cards_i_.menu") DoguDict[category].Add("handiteml_cards_i_.menu"); } HandItemsInitialized = true; OnMenuFilesChange(MenuFilesEventArgs.EventType.HandItems); } private static void InitializeMpnAttachProps() { if (MpnAttachInitialized) return; if (!MenuFilesReady) { if (!beginMpnAttachInit) MenuFilesReadyChange += (s, a) => InitializeMpnAttachProps(); beginMpnAttachInit = true; return; } var menuDataBase = GameMain.Instance.MenuDataBase; var attachMpn = new[] { MPN.kousoku_lower, MPN.kousoku_upper }; for (var i = 0; i < menuDataBase.GetDataSize(); i++) { menuDataBase.SetIndex(i); var itemMpn = (MPN)menuDataBase.GetCategoryMpn(); if (attachMpn.Any(mpn => mpn == itemMpn)) { var menuFileName = menuDataBase.GetMenuFileName(); var mpnTag = menuDataBase.GetCategoryMpnText(); if (menuDataBase.GetBoDelOnly() || menuFileName.EndsWith("_del.menu")) continue; MpnAttachPropList.Add(new(itemMpn, menuFileName)); } } MpnAttachInitialized = true; OnMenuFilesChange(MenuFilesEventArgs.EventType.MpnAttach); } private static void InitializeMyRoomProps() { PlacementData.CreateData(); var myRoomData = PlacementData.GetAllDatas(false); myRoomData.Sort((a, b) => { var res = a.categoryID.CompareTo(b.categoryID); if (res is 0) res = a.ID.CompareTo(b.ID); return res; } ); foreach (var data in myRoomData) { var category = PlacementData.GetCategoryName(data.categoryID); if (!MyRoomPropDict.ContainsKey(category)) { MyRoomPropCategories.Add(category); MyRoomPropDict[category] = new(); } var asset = !string.IsNullOrEmpty(data.resourceName) ? data.resourceName : data.assetName; var item = new MyRoomItem() { PrefabName = asset, ID = data.ID }; MyRoomPropDict[category].Add(item); } } private static void InitializeModProps() { for (var i = 1; i < MenuCategories.Length; i++) ModPropDict[MenuCategories[i]] = new(); if (!PropManager.ModItemsOnly) { var menuDatabase = GameMain.Instance.MenuDataBase; for (var i = 0; i < menuDatabase.GetDataSize(); i++) { menuDatabase.SetIndex(i); var modItem = new ModItem(); if (ParseNativeMenuFile(i, modItem)) ModPropDict[modItem.Category].Add(modItem); } } var cache = new MenuFileCache(); foreach (var modMenuFile in GameUty.ModOnlysMenuFiles) { ModItem modItem; if (cache.Has(modMenuFile)) modItem = cache[modMenuFile]; else { modItem = ModItem.Mod(modMenuFile); ParseMenuFile(modMenuFile, modItem); cache[modMenuFile] = modItem; } if (ValidBG2MenuFile(modItem)) ModPropDict[modItem.Category].Add(modItem); } cache.Serialize(); foreach (var modFile in Menu.GetModFiles()) { var modItem = ModItem.OfficialMod(modFile); if (ParseModMenuFile(modFile, modItem)) ModPropDict[modItem.Category].Add(modItem); } MenuFilesInitialized = true; } // TODO: This could leak on failure. private static CsvParser OpenCsvParser(string nei, AFileSystemBase fs) { try { if (!fs.IsExistentFile(nei)) return null; var file = fs.FileOpen(nei); var csvParser = new CsvParser(); if (csvParser.Open(file)) return csvParser; file?.Dispose(); return null; } catch { } return null; } private static void OnMenuFilesChange(MenuFilesEventArgs.EventType eventType) => MenuFilesChange?.Invoke(null, new(eventType)); private static int KeepAtTop(string a, string b, string topItem) { if (a == b) return 0; if (a == topItem) return -1; if (b == topItem) return 1; return WindowsLogicalComparer.StrCmpLogicalW(a, b); } private class SerializePoseList { public string UIName { get; set; } public List PoseList { get; set; } } } public class MenuFilesEventArgs : EventArgs { public EventType Type { get; } public enum EventType { HandItems, MenuFiles, MpnAttach } public MenuFilesEventArgs(EventType type) => Type = type; } public class PresetChangeEventArgs : EventArgs { public static new PresetChangeEventArgs Empty { get; } = new(string.Empty, string.Empty); public string Category { get; } public string Path { get; } public PresetChangeEventArgs(string path, string category) { Path = path; Category = category; } } public readonly struct MpnAttachProp { public MPN Tag { get; } public string MenuFile { get; } public MpnAttachProp(MPN tag, string menuFile) { Tag = tag; MenuFile = menuFile; } }