Constants.cs 36 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122
  1. using System;
  2. using System.Collections.Generic;
  3. using System.IO;
  4. using System.Linq;
  5. using System.Xml.Linq;
  6. using MyRoomCustom;
  7. using Newtonsoft.Json;
  8. using UnityEngine;
  9. using wf;
  10. using static MeidoPhotoStudio.Plugin.MenuFileUtility;
  11. namespace MeidoPhotoStudio.Plugin;
  12. public static class Constants
  13. {
  14. public const string customPoseDirectory = "Custom Poses";
  15. public const string customHandDirectory = "Hand Presets";
  16. public const string customFaceDirectory = "Face Presets";
  17. public const string sceneDirectory = "Scenes";
  18. public const string kankyoDirectory = "Environments";
  19. public const string configDirectory = "MeidoPhotoStudio";
  20. public const string translationDirectory = "Translations";
  21. public const string databaseDirectory = "Database";
  22. public enum Window { Call, Pose, Face, BG, BG2, Main, Message, Save, SaveModal, Settings }
  23. public enum Scene { Daily = 3, Edit = 5 }
  24. public enum DoguCategory { Other, Mob, Desk, HandItem, BGSmall }
  25. public static readonly string customPosePath;
  26. public static readonly string customHandPath;
  27. public static readonly string customFacePath;
  28. public static readonly string scenesPath;
  29. public static readonly string kankyoPath;
  30. public static readonly string configPath;
  31. public static readonly string databasePath;
  32. public static readonly int mainWindowID = 765;
  33. public static readonly int messageWindowID = 961;
  34. public static readonly int sceneManagerWindowID = 876;
  35. public static readonly int sceneManagerModalID = 283;
  36. public static readonly int dropdownWindowID = 777;
  37. public static readonly List<string> PoseGroupList = new();
  38. public static readonly Dictionary<string, List<string>> PoseDict = new();
  39. public static readonly List<string> CustomPoseGroupList = new();
  40. public static readonly Dictionary<string, List<string>> CustomPoseDict = new();
  41. public static readonly List<string> CustomHandGroupList = new();
  42. public static readonly Dictionary<string, List<string>> CustomHandDict = new();
  43. public static readonly List<string> FaceGroupList = new();
  44. public static readonly Dictionary<string, List<string>> FaceDict = new();
  45. public static readonly List<string> CustomFaceGroupList = new();
  46. public static readonly Dictionary<string, List<string>> CustomFaceDict = new();
  47. public static readonly List<string> BGList = new();
  48. public static readonly List<KeyValuePair<string, string>> MyRoomCustomBGList = new();
  49. public static readonly List<string> DoguCategories = new();
  50. public static readonly Dictionary<string, List<string>> DoguDict = new();
  51. public static readonly List<string> MyRoomPropCategories = new();
  52. public static readonly Dictionary<string, List<MyRoomItem>> MyRoomPropDict = new();
  53. public static readonly Dictionary<string, List<ModItem>> ModPropDict =
  54. new(StringComparer.InvariantCultureIgnoreCase);
  55. public static readonly List<string> SceneDirectoryList = new();
  56. public static readonly List<string> KankyoDirectoryList = new();
  57. public static readonly List<MpnAttachProp> MpnAttachPropList = new();
  58. public static readonly Dictionary<DoguCategory, string> customDoguCategories =
  59. new()
  60. {
  61. [DoguCategory.Other] = "other",
  62. [DoguCategory.Mob] = "mob",
  63. [DoguCategory.Desk] = "desk",
  64. [DoguCategory.HandItem] = "handItem",
  65. [DoguCategory.BGSmall] = "bgSmall"
  66. };
  67. public static int MyRoomCustomBGIndex { get; private set; } = -1;
  68. public static bool HandItemsInitialized { get; private set; }
  69. public static bool MpnAttachInitialized { get; private set; }
  70. public static bool MenuFilesInitialized { get; private set; }
  71. public static event EventHandler<MenuFilesEventArgs> MenuFilesChange;
  72. public static event EventHandler<PresetChangeEventArgs> CustomPoseChange;
  73. public static event EventHandler<PresetChangeEventArgs> CustomHandChange;
  74. public static event EventHandler<PresetChangeEventArgs> CustomFaceChange;
  75. private static bool beginHandItemInit;
  76. private static bool beginMpnAttachInit;
  77. static Constants()
  78. {
  79. configPath = Path.Combine(BepInEx.Paths.ConfigPath, configDirectory);
  80. var presetPath = Path.Combine(configPath, "Presets");
  81. customPosePath = Path.Combine(presetPath, customPoseDirectory);
  82. customHandPath = Path.Combine(presetPath, customHandDirectory);
  83. customFacePath = Path.Combine(presetPath, customFaceDirectory);
  84. scenesPath = Path.Combine(configPath, sceneDirectory);
  85. kankyoPath = Path.Combine(configPath, kankyoDirectory);
  86. databasePath = Path.Combine(configPath, databaseDirectory);
  87. var directories =
  88. new[] { customPosePath, customHandPath, scenesPath, kankyoPath, configPath, customFacePath, databasePath };
  89. foreach (var directory in directories)
  90. if (!Directory.Exists(directory))
  91. Directory.CreateDirectory(directory);
  92. }
  93. public static void Initialize()
  94. {
  95. InitializeSceneDirectories();
  96. InitializeKankyoDirectories();
  97. InitializePoses();
  98. InitializeHandPresets();
  99. InitializeFaceBlends();
  100. InitializeBGs();
  101. InitializeDogu();
  102. InitializeMyRoomProps();
  103. InitializeMpnAttachProps();
  104. }
  105. public static void AddFacePreset(Dictionary<string, float> faceData, string filename, string directory)
  106. {
  107. filename = Utility.SanitizePathPortion(filename);
  108. directory = Utility.SanitizePathPortion(directory);
  109. if (string.IsNullOrEmpty(filename))
  110. filename = "face_preset";
  111. if (directory.Equals(customFaceDirectory, StringComparison.InvariantCultureIgnoreCase))
  112. directory = string.Empty;
  113. directory = Path.Combine(customFacePath, directory);
  114. if (!Directory.Exists(directory))
  115. Directory.CreateDirectory(directory);
  116. var fullPath = Path.Combine(directory, filename);
  117. if (File.Exists($"{fullPath}.xml"))
  118. fullPath += $"_{Utility.Timestamp}";
  119. fullPath = Path.GetFullPath($"{fullPath}.xml");
  120. if (!fullPath.StartsWith(customFacePath))
  121. {
  122. Utility.LogError($"Could not save face preset! Path is invalid: '{fullPath}'");
  123. return;
  124. }
  125. var rootElement = new XElement("FaceData");
  126. foreach (var kvp in faceData)
  127. rootElement.Add(new XElement("elm", kvp.Value.ToString("G9"), new XAttribute("name", kvp.Key)));
  128. var fullDocument = new XDocument(
  129. new XDeclaration("1.0", "utf-8", "true"),
  130. new XComment("MeidoPhotoStudio Face Preset"),
  131. rootElement
  132. );
  133. fullDocument.Save(fullPath);
  134. var fileInfo = new FileInfo(fullPath);
  135. var category = fileInfo.Directory.Name;
  136. var faceGroup = CustomFaceGroupList.Find(
  137. group => string.Equals(category, group, StringComparison.InvariantCultureIgnoreCase)
  138. );
  139. if (string.IsNullOrEmpty(faceGroup))
  140. {
  141. CustomFaceGroupList.Add(category);
  142. CustomFaceDict[category] = new();
  143. CustomFaceGroupList.Sort((a, b) => KeepAtTop(a, b, customFaceDirectory));
  144. }
  145. else
  146. category = faceGroup;
  147. CustomFaceDict[category].Add(fullPath);
  148. CustomFaceDict[category].Sort(WindowsLogicalComparer.StrCmpLogicalW);
  149. CustomFaceChange?.Invoke(null, new(fullPath, category));
  150. }
  151. public static void AddPose(byte[] anmBinary, string filename, string directory)
  152. {
  153. // TODO: Consider writing a file system monitor
  154. filename = Utility.SanitizePathPortion(filename);
  155. directory = Utility.SanitizePathPortion(directory);
  156. if (string.IsNullOrEmpty(filename))
  157. filename = "custom_pose";
  158. if (directory.Equals(customPoseDirectory, StringComparison.InvariantCultureIgnoreCase))
  159. directory = string.Empty;
  160. directory = Path.Combine(customPosePath, directory);
  161. if (!Directory.Exists(directory))
  162. Directory.CreateDirectory(directory);
  163. var fullPath = Path.Combine(directory, filename);
  164. if (File.Exists($"{fullPath}.anm"))
  165. fullPath += $"_{Utility.Timestamp}";
  166. fullPath = Path.GetFullPath($"{fullPath}.anm");
  167. if (!fullPath.StartsWith(customPosePath))
  168. {
  169. Utility.LogError($"Could not save pose! Path is invalid: '{fullPath}'");
  170. return;
  171. }
  172. File.WriteAllBytes(fullPath, anmBinary);
  173. var fileInfo = new FileInfo(fullPath);
  174. var category = fileInfo.Directory.Name;
  175. var poseGroup = CustomPoseGroupList.Find(
  176. group => string.Equals(category, group, StringComparison.InvariantCultureIgnoreCase)
  177. );
  178. if (string.IsNullOrEmpty(poseGroup))
  179. {
  180. CustomPoseGroupList.Add(category);
  181. CustomPoseDict[category] = new();
  182. CustomPoseGroupList.Sort((a, b) => KeepAtTop(a, b, customPoseDirectory));
  183. }
  184. else
  185. category = poseGroup;
  186. CustomPoseDict[category].Add(fullPath);
  187. CustomPoseDict[category].Sort(WindowsLogicalComparer.StrCmpLogicalW);
  188. CustomPoseChange?.Invoke(null, new(fullPath, category));
  189. }
  190. public static void AddHand(byte[] handBinary, bool right, string filename, string directory)
  191. {
  192. filename = Utility.SanitizePathPortion(filename);
  193. directory = Utility.SanitizePathPortion(directory);
  194. if (string.IsNullOrEmpty(filename))
  195. filename = "custom_hand";
  196. if (directory.Equals(customHandDirectory, StringComparison.InvariantCultureIgnoreCase))
  197. directory = string.Empty;
  198. directory = Path.Combine(customHandPath, directory);
  199. if (!Directory.Exists(directory))
  200. Directory.CreateDirectory(directory);
  201. var fullPath = Path.Combine(directory, filename);
  202. if (File.Exists($"{fullPath}.xml"))
  203. fullPath += $"_{Utility.Timestamp}";
  204. fullPath = Path.GetFullPath($"{fullPath}.xml");
  205. if (!fullPath.StartsWith(customHandPath))
  206. {
  207. Utility.LogError($"Could not save hand! Path is invalid: '{fullPath}'");
  208. return;
  209. }
  210. // TODO: This does not actually do what I think it does.
  211. var gameVersion = Misc.GAME_VERSION; // get game version from user's Assembly-CSharp
  212. var finalXml = new XDocument(
  213. new XDeclaration("1.0", "utf-8", "true"),
  214. new XComment("CM3D2 FingerData"),
  215. new XElement(
  216. "FingerData",
  217. new XElement("GameVersion", gameVersion),
  218. new XElement("RightData", right),
  219. new XElement("BinaryData", Convert.ToBase64String(handBinary))
  220. )
  221. );
  222. finalXml.Save(fullPath);
  223. var fileInfo = new FileInfo(fullPath);
  224. var category = fileInfo.Directory.Name;
  225. var handGroup = CustomHandGroupList.Find(
  226. group => string.Equals(category, group, StringComparison.InvariantCultureIgnoreCase)
  227. );
  228. if (string.IsNullOrEmpty(handGroup))
  229. {
  230. CustomHandGroupList.Add(category);
  231. CustomHandDict[category] = new();
  232. CustomHandGroupList.Sort((a, b) => KeepAtTop(a, b, customHandDirectory));
  233. }
  234. else
  235. category = handGroup;
  236. CustomHandDict[category].Add(fullPath);
  237. CustomHandDict[category].Sort(WindowsLogicalComparer.StrCmpLogicalW);
  238. CustomHandChange?.Invoke(null, new(fullPath, category));
  239. }
  240. public static void InitializeSceneDirectories()
  241. {
  242. SceneDirectoryList.Clear();
  243. SceneDirectoryList.Add(sceneDirectory);
  244. foreach (var directory in Directory.GetDirectories(scenesPath))
  245. SceneDirectoryList.Add(new DirectoryInfo(directory).Name);
  246. SceneDirectoryList.Sort((a, b) => KeepAtTop(a, b, sceneDirectory));
  247. }
  248. public static void InitializeKankyoDirectories()
  249. {
  250. KankyoDirectoryList.Clear();
  251. KankyoDirectoryList.Add(kankyoDirectory);
  252. foreach (var directory in Directory.GetDirectories(kankyoPath))
  253. KankyoDirectoryList.Add(new DirectoryInfo(directory).Name);
  254. KankyoDirectoryList.Sort((a, b) => KeepAtTop(a, b, kankyoDirectory));
  255. }
  256. public static void InitializePoses()
  257. {
  258. // Load Poses
  259. var poseListPath = Path.Combine(databasePath, "mm_pose_list.json");
  260. try
  261. {
  262. var poseListJson = File.ReadAllText(poseListPath);
  263. foreach (var poseList in JsonConvert.DeserializeObject<List<SerializePoseList>>(poseListJson))
  264. {
  265. PoseDict[poseList.UIName] = poseList.PoseList;
  266. PoseGroupList.Add(poseList.UIName);
  267. }
  268. }
  269. catch (IOException e)
  270. {
  271. Utility.LogError($"Could not open pose database because {e.Message}");
  272. Utility.LogMessage("Pose list will be serverely limited.");
  273. AddDefaultPose();
  274. }
  275. catch (Exception e)
  276. {
  277. Utility.LogError($"Could not parse pose database because {e.Message}");
  278. AddDefaultPose();
  279. }
  280. // Get Other poses that'll go into Normal, Normal 2 and Ero 2
  281. var com3d2MotionList = GameUty.FileSystem.GetList("motion", AFileSystemBase.ListType.AllFile);
  282. if (com3d2MotionList?.Length > 0)
  283. {
  284. var poseSet = new HashSet<string>();
  285. foreach (var poses in PoseDict.Values)
  286. poseSet.UnionWith(poses);
  287. var newCategories = new[] { "normal2", "ero2" };
  288. foreach (var category in newCategories)
  289. if (!PoseDict.ContainsKey(category))
  290. PoseDict[category] = new();
  291. // TODO: Try to group these poses into more than "normal2" and "ero2"
  292. foreach (var path in com3d2MotionList)
  293. {
  294. if (Path.GetExtension(path) is not ".anm")
  295. continue;
  296. var file = Path.GetFileNameWithoutExtension(path);
  297. if (poseSet.Contains(file))
  298. continue;
  299. if (file.StartsWith("edit_"))
  300. PoseDict["normal"].Add(file);
  301. else if (file is not ("dance_cm3d2_001_zoukin" or "dance_cm3d2_001_mop" or "aruki_1_idougo_f"
  302. or "sleep2" or "stand_akire2") && !file.EndsWith("_3_") && !file.EndsWith("_5_")
  303. && !file.StartsWith("vr_") && !file.StartsWith("dance_mc") && !file.Contains("_kubi_")
  304. && !file.Contains("a01_") && !file.Contains("b01_") && !file.Contains("b02_")
  305. && !file.EndsWith("_m2") && !file.EndsWith("_m2_once_") && !file.StartsWith("h_")
  306. && !file.StartsWith("event_") && !file.StartsWith("man_") && !file.EndsWith("_m")
  307. && !file.Contains("_m_") && !file.Contains("_man_")
  308. )
  309. {
  310. if (path.Contains(@"\sex\"))
  311. PoseDict["ero2"].Add(file);
  312. else
  313. PoseDict["normal2"].Add(file);
  314. }
  315. }
  316. foreach (var category in newCategories)
  317. {
  318. if (PoseDict[category].Count > 0)
  319. {
  320. if (!PoseGroupList.Contains(category))
  321. PoseGroupList.Add(category);
  322. }
  323. else
  324. PoseDict.Remove(category);
  325. }
  326. }
  327. InitializeCustomPoses();
  328. static void AddDefaultPose()
  329. {
  330. if (!PoseDict.ContainsKey("normal"))
  331. PoseDict["normal"] = new() { "maid_stand01" };
  332. if (!PoseGroupList.Contains("normal"))
  333. PoseGroupList.Insert(0, "normal");
  334. }
  335. }
  336. public static void InitializeCustomPoses()
  337. {
  338. CustomPoseGroupList.Clear();
  339. CustomPoseDict.Clear();
  340. CustomPoseGroupList.Add(customPoseDirectory);
  341. CustomPoseDict[customPoseDirectory] = new();
  342. GetPoses(customPosePath);
  343. foreach (var directory in Directory.GetDirectories(customPosePath))
  344. GetPoses(directory);
  345. CustomPoseGroupList.Sort((a, b) => KeepAtTop(a, b, customPoseDirectory));
  346. CustomPoseChange?.Invoke(null, PresetChangeEventArgs.Empty);
  347. static void GetPoses(string directory)
  348. {
  349. var poseList = Directory.GetFiles(directory)
  350. .Where(file => Path.GetExtension(file) is ".anm");
  351. if (poseList.Any())
  352. {
  353. var poseGroupName = new DirectoryInfo(directory).Name;
  354. if (poseGroupName != customPoseDirectory)
  355. CustomPoseGroupList.Add(poseGroupName);
  356. CustomPoseDict[poseGroupName] = poseList.ToList();
  357. CustomPoseDict[poseGroupName].Sort(WindowsLogicalComparer.StrCmpLogicalW);
  358. }
  359. }
  360. }
  361. public static void InitializeHandPresets()
  362. {
  363. CustomHandGroupList.Clear();
  364. CustomHandDict.Clear();
  365. CustomHandGroupList.Add(customHandDirectory);
  366. CustomHandDict[customHandDirectory] = new();
  367. GetPresets(customHandPath);
  368. foreach (var directory in Directory.GetDirectories(customHandPath))
  369. GetPresets(directory);
  370. CustomHandGroupList.Sort((a, b) => KeepAtTop(a, b, customHandDirectory));
  371. CustomHandChange?.Invoke(null, PresetChangeEventArgs.Empty);
  372. static void GetPresets(string directory)
  373. {
  374. var presetList = Directory.GetFiles(directory)
  375. .Where(file => Path.GetExtension(file) is ".xml");
  376. if (!presetList.Any())
  377. return;
  378. var presetCategory = new DirectoryInfo(directory).Name;
  379. if (presetCategory != customHandDirectory)
  380. CustomHandGroupList.Add(presetCategory);
  381. CustomHandDict[presetCategory] = presetList.ToList();
  382. CustomHandDict[presetCategory].Sort(WindowsLogicalComparer.StrCmpLogicalW);
  383. }
  384. }
  385. public static void InitializeFaceBlends()
  386. {
  387. PhotoFaceData.Create();
  388. FaceGroupList.AddRange(PhotoFaceData.popup_category_list.Select(kvp => kvp.Key));
  389. foreach (var kvp in PhotoFaceData.category_list)
  390. FaceDict[kvp.Key] = kvp.Value.ConvertAll(data => data.setting_name);
  391. InitializeCustomFaceBlends();
  392. }
  393. public static void InitializeCustomFaceBlends()
  394. {
  395. CustomFaceGroupList.Clear();
  396. CustomFaceDict.Clear();
  397. CustomFaceGroupList.Add(customFaceDirectory);
  398. CustomFaceDict[customFaceDirectory] = new();
  399. GetFacePresets(customFacePath);
  400. foreach (var directory in Directory.GetDirectories(customFacePath))
  401. GetFacePresets(directory);
  402. CustomFaceGroupList.Sort((a, b) => KeepAtTop(a, b, customFaceDirectory));
  403. CustomFaceChange?.Invoke(null, PresetChangeEventArgs.Empty);
  404. static void GetFacePresets(string directory)
  405. {
  406. IEnumerable<string> presetList = Directory.GetFiles(directory)
  407. .Where(file => Path.GetExtension(file) is ".xml").ToList();
  408. if (presetList.Any())
  409. {
  410. var faceGroupName = new DirectoryInfo(directory).Name;
  411. if (faceGroupName != customFaceDirectory)
  412. CustomFaceGroupList.Add(faceGroupName);
  413. CustomFaceDict[faceGroupName] = presetList.ToList();
  414. CustomFaceDict[faceGroupName].Sort(WindowsLogicalComparer.StrCmpLogicalW);
  415. }
  416. }
  417. }
  418. public static void InitializeBGs()
  419. {
  420. // Load BGs
  421. PhotoBGData.Create();
  422. // COM3D2 BGs
  423. foreach (var bgData in PhotoBGData.data)
  424. {
  425. if (string.IsNullOrEmpty(bgData.create_prefab_name))
  426. continue;
  427. var bg = bgData.create_prefab_name;
  428. BGList.Add(bg);
  429. }
  430. // CM3D2 BGs
  431. if (GameUty.IsEnabledCompatibilityMode)
  432. {
  433. using var csvParser = OpenCsvParser("phot_bg_list.nei", GameUty.FileSystemOld);
  434. for (var cell_y = 1; cell_y < csvParser.max_cell_y; cell_y++)
  435. {
  436. if (!csvParser.IsCellToExistData(3, cell_y))
  437. continue;
  438. var bg = csvParser.GetCellAsString(3, cell_y);
  439. BGList.Add(bg);
  440. }
  441. }
  442. // Set index regardless of there being myRoom bgs or not
  443. MyRoomCustomBGIndex = BGList.Count;
  444. var saveDataDict = CreativeRoomManager.GetSaveDataDic();
  445. if (saveDataDict is not null)
  446. MyRoomCustomBGList.AddRange(saveDataDict);
  447. }
  448. public static void InitializeDogu()
  449. {
  450. foreach (var customCategory in customDoguCategories.Values)
  451. DoguDict[customCategory] = new();
  452. InitializeDeskItems();
  453. InitializePhotoBGItems();
  454. InitializeOtherDogu();
  455. InitializeHandItems();
  456. foreach (var category in PhotoBGObjectData.popup_category_list.Select(kvp => kvp.Key))
  457. {
  458. if (category is "マイオブジェクト")
  459. continue;
  460. DoguCategories.Add(category);
  461. }
  462. foreach (DoguCategory category in Enum.GetValues(typeof(DoguCategory)))
  463. DoguCategories.Add(customDoguCategories[category]);
  464. }
  465. public static List<ModItem> GetModPropList(string category)
  466. {
  467. if (!PropManager.ModItemsOnly && !MenuFilesReady)
  468. {
  469. Utility.LogMessage("Menu files are not ready yet");
  470. return null;
  471. }
  472. if (!MenuFilesInitialized)
  473. InitializeModProps();
  474. if (!ModPropDict.ContainsKey(category))
  475. return null;
  476. var selectedList = ModPropDict[category];
  477. if (selectedList[0].Icon == null)
  478. {
  479. selectedList.Sort((a, b) =>
  480. {
  481. var res = a.Priority.CompareTo(b.Priority);
  482. if (res is 0)
  483. res = string.Compare(a.Name, b.Name);
  484. return res;
  485. });
  486. var previousMenuFile = string.Empty;
  487. selectedList.RemoveAll(item =>
  488. {
  489. if (item.Icon != null)
  490. return false;
  491. Texture2D icon;
  492. var iconFile = item.IconFile;
  493. if (string.IsNullOrEmpty(iconFile))
  494. {
  495. Utility.LogWarning($"Could not find icon '{iconFile}' for menu '{item.MenuFile}");
  496. return true;
  497. }
  498. try
  499. {
  500. icon = ImportCM.CreateTexture(iconFile);
  501. }
  502. catch
  503. {
  504. try
  505. {
  506. icon = ImportCM.CreateTexture($"tex\\{iconFile}");
  507. }
  508. catch
  509. {
  510. Utility.LogWarning($"Could not load '{iconFile}' for menu '{item.MenuFile}");
  511. return true;
  512. }
  513. }
  514. item.Icon = icon;
  515. return false;
  516. });
  517. }
  518. return selectedList;
  519. }
  520. private static void InitializeOtherDogu()
  521. {
  522. DoguDict[customDoguCategories[DoguCategory.BGSmall]] = BGList;
  523. DoguDict[customDoguCategories[DoguCategory.Mob]].AddRange(
  524. new[]
  525. {
  526. "Mob_Man_Stand001", "Mob_Man_Stand002", "Mob_Man_Stand003", "Mob_Man_Sit001", "Mob_Man_Sit002",
  527. "Mob_Man_Sit003", "Mob_Girl_Stand001", "Mob_Girl_Stand002", "Mob_Girl_Stand003", "Mob_Girl_Sit001",
  528. "Mob_Girl_Sit002", "Mob_Girl_Sit003"
  529. }
  530. );
  531. var DoguList = DoguDict[customDoguCategories[DoguCategory.Other]];
  532. // bg object extend
  533. var doguHashSet = new HashSet<string>(StringComparer.InvariantCultureIgnoreCase);
  534. doguHashSet.UnionWith(BGList);
  535. try
  536. {
  537. var ignoreListPath = Path.Combine(databasePath, "bg_ignore_list.json");
  538. var ignoreListJson = File.ReadAllText(ignoreListPath);
  539. var ignoreList = JsonConvert.DeserializeObject<string[]>(ignoreListJson);
  540. doguHashSet.UnionWith(ignoreList);
  541. }
  542. catch (IOException e)
  543. {
  544. Utility.LogWarning($"Could not open ignored BG database because {e.Message}");
  545. }
  546. catch { }
  547. foreach (var doguList in DoguDict.Values)
  548. doguHashSet.UnionWith(doguList);
  549. foreach (var path in GameUty.FileSystem.GetList("bg", AFileSystemBase.ListType.AllFile))
  550. {
  551. if (Path.GetExtension(path) is not ".asset_bg" || path.Contains("myroomcustomize"))
  552. continue;
  553. var file = Path.GetFileNameWithoutExtension(path);
  554. if (doguHashSet.Contains(file) || file.EndsWith("_hit"))
  555. continue;
  556. DoguList.Add(file);
  557. doguHashSet.Add(file);
  558. }
  559. // Get cherry picked dogu that I can't find in the game files
  560. try
  561. {
  562. var doguExtendPath = Path.Combine(databasePath, "extra_dogu.json");
  563. var doguExtendJson = File.ReadAllText(doguExtendPath);
  564. DoguList.AddRange(JsonConvert.DeserializeObject<IEnumerable<string>>(doguExtendJson));
  565. }
  566. catch (IOException e)
  567. {
  568. Utility.LogError($"Could not open extra prop database because {e.Message}");
  569. }
  570. catch (Exception e)
  571. {
  572. Utility.LogError($"Could not parse extra prop database because {e.Message}");
  573. }
  574. foreach (var path in GameUty.FileSystemOld.GetList("bg", AFileSystemBase.ListType.AllFile))
  575. {
  576. if (Path.GetExtension(path) is not ".asset_bg")
  577. continue;
  578. var file = Path.GetFileNameWithoutExtension(path);
  579. if (!doguHashSet.Contains(file) && !file.EndsWith("_not_optimisation"))
  580. DoguList.Add(file);
  581. }
  582. }
  583. private static void InitializeDeskItems()
  584. {
  585. var enabledIDs = new HashSet<int>();
  586. CsvCommonIdManager.ReadEnabledIdList(
  587. CsvCommonIdManager.FileSystemType.Normal, true, "desk_item_enabled_id", ref enabledIDs
  588. );
  589. CsvCommonIdManager.ReadEnabledIdList(
  590. CsvCommonIdManager.FileSystemType.Old, true, "desk_item_enabled_id", ref enabledIDs
  591. );
  592. var com3d2DeskDogu = DoguDict[customDoguCategories[DoguCategory.Desk]];
  593. void GetDeskItems(AFileSystemBase fs)
  594. {
  595. using var csvParser = OpenCsvParser("desk_item_detail.nei", fs);
  596. for (var cell_y = 1; cell_y < csvParser.max_cell_y; cell_y++)
  597. {
  598. if (!csvParser.IsCellToExistData(0, cell_y))
  599. continue;
  600. var cell = csvParser.GetCellAsInteger(0, cell_y);
  601. if (!enabledIDs.Contains(cell))
  602. continue;
  603. var dogu = string.Empty;
  604. if (csvParser.IsCellToExistData(3, cell_y))
  605. dogu = csvParser.GetCellAsString(3, cell_y);
  606. else if (csvParser.IsCellToExistData(4, cell_y))
  607. dogu = csvParser.GetCellAsString(4, cell_y);
  608. if (!string.IsNullOrEmpty(dogu))
  609. com3d2DeskDogu.Add(dogu);
  610. }
  611. }
  612. GetDeskItems(GameUty.FileSystem);
  613. }
  614. private static void InitializePhotoBGItems()
  615. {
  616. PhotoBGObjectData.Create();
  617. var photoBGObjectList = PhotoBGObjectData.data;
  618. var doguCategories = new List<string>();
  619. var addedCategories = new HashSet<string>(StringComparer.InvariantCultureIgnoreCase);
  620. foreach (var photoBGObject in photoBGObjectList)
  621. {
  622. var category = photoBGObject.category;
  623. if (!addedCategories.Contains(category))
  624. {
  625. addedCategories.Add(category);
  626. doguCategories.Add(category);
  627. }
  628. if (!DoguDict.ContainsKey(category))
  629. DoguDict[category] = new();
  630. var dogu = string.Empty;
  631. if (!string.IsNullOrEmpty(photoBGObject.create_prefab_name))
  632. dogu = photoBGObject.create_prefab_name;
  633. else if (!string.IsNullOrEmpty(photoBGObject.create_asset_bundle_name))
  634. dogu = photoBGObject.create_asset_bundle_name;
  635. else if (!string.IsNullOrEmpty(photoBGObject.direct_file))
  636. dogu = photoBGObject.direct_file;
  637. if (!string.IsNullOrEmpty(dogu))
  638. DoguDict[category].Add(dogu);
  639. }
  640. DoguDict["パーティクル"].AddRange(
  641. new[]
  642. {
  643. "Particle/pLineY", "Particle/pLineP02", "Particle/pHeart01",
  644. "Particle/pLine_act2", "Particle/pstarY_act2"
  645. }
  646. );
  647. }
  648. private static void InitializeHandItems()
  649. {
  650. if (HandItemsInitialized)
  651. return;
  652. if (!MenuFilesReady)
  653. {
  654. if (!beginHandItemInit)
  655. MenuFilesReadyChange += (s, a) =>
  656. InitializeHandItems();
  657. beginHandItemInit = true;
  658. return;
  659. }
  660. var menuDataBase = GameMain.Instance.MenuDataBase;
  661. var doguHashSet = new HashSet<string>(StringComparer.InvariantCultureIgnoreCase);
  662. doguHashSet.UnionWith(BGList);
  663. try
  664. {
  665. var ignoreListPath = Path.Combine(databasePath, "bg_ignore_list.json");
  666. var ignoreListJson = File.ReadAllText(ignoreListPath);
  667. var ignoreList = JsonConvert.DeserializeObject<string[]>(ignoreListJson);
  668. doguHashSet.UnionWith(ignoreList);
  669. }
  670. catch (IOException e)
  671. {
  672. Utility.LogWarning($"Could not open ignored BG database because {e.Message}");
  673. }
  674. catch (Exception e)
  675. {
  676. Utility.LogError($"Could not parse ignored BG database because {e.Message}");
  677. }
  678. foreach (var doguList in DoguDict.Values)
  679. doguHashSet.UnionWith(doguList);
  680. var category = customDoguCategories[DoguCategory.HandItem];
  681. for (var i = 0; i < menuDataBase.GetDataSize(); i++)
  682. {
  683. menuDataBase.SetIndex(i);
  684. if ((MPN)menuDataBase.GetCategoryMpn() is not MPN.handitem)
  685. continue;
  686. var menuFileName = menuDataBase.GetMenuFileName();
  687. if (menuDataBase.GetBoDelOnly() || menuFileName.EndsWith("_del.menu"))
  688. continue;
  689. var handItemAsOdogu = Utility.HandItemToOdogu(menuFileName);
  690. var isolatedHandItem = menuFileName.Substring(menuFileName.IndexOf('_') + 1);
  691. if (doguHashSet.Contains(handItemAsOdogu) || doguHashSet.Contains(isolatedHandItem))
  692. continue;
  693. doguHashSet.Add(isolatedHandItem);
  694. DoguDict[category].Add(menuFileName);
  695. // Check for a half deck of cards to add the full deck as well
  696. if (menuFileName is "handitemd_cards_i_.menu")
  697. DoguDict[category].Add("handiteml_cards_i_.menu");
  698. }
  699. HandItemsInitialized = true;
  700. OnMenuFilesChange(MenuFilesEventArgs.EventType.HandItems);
  701. }
  702. private static void InitializeMpnAttachProps()
  703. {
  704. if (MpnAttachInitialized)
  705. return;
  706. if (!MenuFilesReady)
  707. {
  708. if (!beginMpnAttachInit)
  709. MenuFilesReadyChange += (s, a) =>
  710. InitializeMpnAttachProps();
  711. beginMpnAttachInit = true;
  712. return;
  713. }
  714. var menuDataBase = GameMain.Instance.MenuDataBase;
  715. var attachMpn = new[] { MPN.kousoku_lower, MPN.kousoku_upper };
  716. for (var i = 0; i < menuDataBase.GetDataSize(); i++)
  717. {
  718. menuDataBase.SetIndex(i);
  719. var itemMpn = (MPN)menuDataBase.GetCategoryMpn();
  720. if (attachMpn.Any(mpn => mpn == itemMpn))
  721. {
  722. var menuFileName = menuDataBase.GetMenuFileName();
  723. var mpnTag = menuDataBase.GetCategoryMpnText();
  724. if (menuDataBase.GetBoDelOnly() || menuFileName.EndsWith("_del.menu"))
  725. continue;
  726. MpnAttachPropList.Add(new(itemMpn, menuFileName));
  727. }
  728. }
  729. MpnAttachInitialized = true;
  730. OnMenuFilesChange(MenuFilesEventArgs.EventType.MpnAttach);
  731. }
  732. private static void InitializeMyRoomProps()
  733. {
  734. PlacementData.CreateData();
  735. var myRoomData = PlacementData.GetAllDatas(false);
  736. myRoomData.Sort((a, b) =>
  737. {
  738. var res = a.categoryID.CompareTo(b.categoryID);
  739. if (res is 0)
  740. res = a.ID.CompareTo(b.ID);
  741. return res;
  742. }
  743. );
  744. foreach (var data in myRoomData)
  745. {
  746. var category = PlacementData.GetCategoryName(data.categoryID);
  747. if (!MyRoomPropDict.ContainsKey(category))
  748. {
  749. MyRoomPropCategories.Add(category);
  750. MyRoomPropDict[category] = new();
  751. }
  752. var asset = !string.IsNullOrEmpty(data.resourceName) ? data.resourceName : data.assetName;
  753. var item = new MyRoomItem() { PrefabName = asset, ID = data.ID };
  754. MyRoomPropDict[category].Add(item);
  755. }
  756. }
  757. private static void InitializeModProps()
  758. {
  759. for (var i = 1; i < MenuCategories.Length; i++)
  760. ModPropDict[MenuCategories[i]] = new();
  761. if (!PropManager.ModItemsOnly)
  762. {
  763. var menuDatabase = GameMain.Instance.MenuDataBase;
  764. for (var i = 0; i < menuDatabase.GetDataSize(); i++)
  765. {
  766. menuDatabase.SetIndex(i);
  767. var modItem = new ModItem();
  768. if (ParseNativeMenuFile(i, modItem))
  769. ModPropDict[modItem.Category].Add(modItem);
  770. }
  771. }
  772. var cache = new MenuFileCache();
  773. foreach (var modMenuFile in GameUty.ModOnlysMenuFiles)
  774. {
  775. ModItem modItem;
  776. if (cache.Has(modMenuFile))
  777. modItem = cache[modMenuFile];
  778. else
  779. {
  780. modItem = ModItem.Mod(modMenuFile);
  781. ParseMenuFile(modMenuFile, modItem);
  782. cache[modMenuFile] = modItem;
  783. }
  784. if (ValidBG2MenuFile(modItem))
  785. ModPropDict[modItem.Category].Add(modItem);
  786. }
  787. cache.Serialize();
  788. foreach (var modFile in Menu.GetModFiles())
  789. {
  790. var modItem = ModItem.OfficialMod(modFile);
  791. if (ParseModMenuFile(modFile, modItem))
  792. ModPropDict[modItem.Category].Add(modItem);
  793. }
  794. MenuFilesInitialized = true;
  795. }
  796. // TODO: This could leak on failure.
  797. private static CsvParser OpenCsvParser(string nei, AFileSystemBase fs)
  798. {
  799. try
  800. {
  801. if (!fs.IsExistentFile(nei))
  802. return null;
  803. var file = fs.FileOpen(nei);
  804. var csvParser = new CsvParser();
  805. if (csvParser.Open(file))
  806. return csvParser;
  807. file?.Dispose();
  808. return null;
  809. }
  810. catch { }
  811. return null;
  812. }
  813. private static void OnMenuFilesChange(MenuFilesEventArgs.EventType eventType) =>
  814. MenuFilesChange?.Invoke(null, new(eventType));
  815. private static int KeepAtTop(string a, string b, string topItem)
  816. {
  817. if (a == b)
  818. return 0;
  819. if (a == topItem)
  820. return -1;
  821. if (b == topItem)
  822. return 1;
  823. return WindowsLogicalComparer.StrCmpLogicalW(a, b);
  824. }
  825. private class SerializePoseList
  826. {
  827. public string UIName { get; set; }
  828. public List<string> PoseList { get; set; }
  829. }
  830. }
  831. public class MenuFilesEventArgs : EventArgs
  832. {
  833. public EventType Type { get; }
  834. public enum EventType { HandItems, MenuFiles, MpnAttach }
  835. public MenuFilesEventArgs(EventType type) =>
  836. Type = type;
  837. }
  838. public class PresetChangeEventArgs : EventArgs
  839. {
  840. public static new PresetChangeEventArgs Empty { get; } = new(string.Empty, string.Empty);
  841. public string Category { get; }
  842. public string Path { get; }
  843. public PresetChangeEventArgs(string path, string category)
  844. {
  845. Path = path;
  846. Category = category;
  847. }
  848. }
  849. public readonly struct MpnAttachProp
  850. {
  851. public MPN Tag { get; }
  852. public string MenuFile { get; }
  853. public MpnAttachProp(MPN tag, string menuFile)
  854. {
  855. Tag = tag;
  856. MenuFile = menuFile;
  857. }
  858. }