using LinePutScript; using LinePutScript.Converter; using LinePutScript.Dictionary; using LinePutScript.Localization.WPF; using NAudio.SoundFont; using System; using System.Collections.Generic; using System.Diagnostics; using System.IO; using System.Linq; using System.Reflection; using System.Reflection.Metadata; using System.Security.Cryptography.X509Certificates; using System.Windows; using System.Windows.Media; using System.Xml.Linq; using VPet_Simulator.Core; using VPet_Simulator.Windows.Interface; namespace VPet_Simulator.Windows { public class CoreMOD { /// /// 自动启用MOD名称 /// public static readonly string[] OnModDefList = new string[] { "Core", "PCat", "ModMaker" }; public static HashSet LoadedDLL { get; } = new HashSet() { "Panuon.WPF.dll","steam_api.dll","Panuon.WPF.UI.dll","steam_api64.dll", "LinePutScript.dll","Facepunch.Steamworks.Win32.dll", "Facepunch.Steamworks.Win64.dll", "VPet-Simulator.Core.dll","VPet-Simulator.Windows.Interface.dll","LinePutScript.Localization.WPF.dll", "CSCore.dll" }; public static string NowLoading = null; public string Name; public string Author; /// /// 如果是上传至Steam,则为SteamUserID /// public long AuthorID; /// /// 上传至Steam的ItemID /// public ulong ItemID; public string Intro; public DirectoryInfo Path; public int GameVer; public int Ver; public HashSet Tag = new HashSet(); public bool SuccessLoad = true; public DateTime CacheDate; public string ErrorMessage; public static string INTtoVER(int ver) => ver < 10000 ? $"{ver / 100}.{ver % 100:00}" : $"{ver / 10000}.{ver % 10000 / 100}.{ver % 100:00}"; public static void LoadImage(MainWindow mw, DirectoryInfo di, string pre = "") { //加载其他放在文件夹的图片 foreach (FileInfo fi in di.EnumerateFiles("*.png")) { mw.ImageSources.AddSource(pre + fi.Name.ToLower().Substring(0, fi.Name.Length - 4), fi.FullName); } //加载其他放在文件夹中文件夹的图片 foreach (DirectoryInfo fordi in di.EnumerateDirectories()) { LoadImage(mw, fordi, pre + fordi.Name + "_"); } //加载标志好的图片和图片设置 foreach (FileInfo fi in di.EnumerateFiles("*.lps")) { var tmp = new LpsDocument(File.ReadAllText(fi.FullName)); if (fi.Name.ToLower().StartsWith("set_")) foreach (var line in tmp) mw.ImageSources.ImageSetting.AddorReplaceLine(line); else mw.ImageSources.AddImages(tmp, di.FullName); } } public CoreMOD(DirectoryInfo directory, MainWindow mw) { #if !DEBUG try { #endif Path = directory; LpsDocument modlps = new LpsDocument(File.ReadAllText(directory.FullName + @"\info.lps")); Name = modlps.FindLine("vupmod").Info; NowLoading = Name; Intro = modlps.FindLine("intro").Info; GameVer = modlps.FindSub("gamever").InfoToInt; Ver = modlps.FindSub("ver").InfoToInt; Author = modlps.FindSub("author").Info.Split('[').First(); if (modlps.FindLine("authorid") != null) AuthorID = modlps.FindLine("authorid").InfoToInt64; else AuthorID = 0; if (modlps.FindLine("itemid") != null) ItemID = Convert.ToUInt64(modlps.FindLine("itemid").info); else ItemID = 0; CacheDate = modlps.GetDateTime("cachedate", DateTime.MinValue); //MOD未加载时支持翻译 foreach (var line in modlps.FindAllLine("lang")) { List ls = new List(); foreach (var sub in line) { ls.Add(new Line(sub.Name, sub.info)); } LocalizeCore.AddCulture(line.info, ls); } if (mw.CoreMODs.FirstOrDefault(x => x.Name == Name) != null) { Name += $"({"MOD名称重复".Translate()})"; ErrorMessage = "MOD名称重复".Translate(); return; } if (!IsOnMOD(mw)) { Tag.Add("该模组已停用"); foreach (DirectoryInfo di in Path.EnumerateDirectories()) Tag.Add(di.Name.ToLower()); return; } foreach (DirectoryInfo di in Path.EnumerateDirectories()) { switch (di.Name.ToLower()) { case "theme": Tag.Add("theme"); if (Directory.Exists(di.FullName + @"\fonts")) foreach (var str in Directory.EnumerateFiles(di.FullName + @"\fonts", "*.ttf")) { mw.Fonts.Add(new IFont(new FileInfo(str))); } foreach (FileInfo fi in di.EnumerateFiles("*.lps")) { var tmp = new Theme(new LpsDocument(File.ReadAllText(fi.FullName))); var oldtheme = mw.Themes.Find(x => x.xName == tmp.xName); if (oldtheme != null) mw.Themes.Remove(oldtheme); mw.Themes.Add(tmp); //加载图片包 DirectoryInfo tmpdi = new DirectoryInfo(di.FullName + '\\' + tmp.Image); if (tmpdi.Exists) { foreach (FileInfo tmpfi in tmpdi.EnumerateFiles("*.png")) { tmp.Images.AddSource(tmpfi.Name.ToLower().Substring(0, tmpfi.Name.Length - 4), tmpfi.FullName); } foreach (DirectoryInfo fordi in tmpdi.EnumerateDirectories()) { foreach (FileInfo tmpfi in fordi.EnumerateFiles("*.png")) { tmp.Images.AddSource(fordi.Name + '_' + tmpfi.Name.ToLower().Substring(0, tmpfi.Name.Length - 4), tmpfi.FullName); } } } } break; case "pet": //宠物模型 foreach (FileInfo fi in di.EnumerateFiles("*.lps")) { LpsDocument lps = new LpsDocument(File.ReadAllText(fi.FullName)); if (lps.First().Name.ToLower() == "pet") { var name = lps.First().Info; var p = mw.Pets.FirstOrDefault(x => x.Name == name); if (p == null) { Tag.Add("pet"); p = new PetLoader(lps, di); if (p.Config.Works.Count > 0) Tag.Add("work"); mw.Pets.Add(p); } else { if (lps.FindAllLine("work").Length >= 0) { Tag.Add("work"); } var dis = new DirectoryInfo(di.FullName + "\\" + lps.First()["path"].Info); if (dis.Exists && dis.GetDirectories().Length > 0) Tag.Add("pet"); p.path.Add(di.FullName + "\\" + lps.First()["path"].Info); p.Config.Set(lps); } } } break; case "food": Tag.Add("food"); foreach (FileInfo fi in di.EnumerateFiles("*.lps")) { var tmp = new LpsDocument(File.ReadAllText(fi.FullName)); foreach (ILine li in tmp) { if (li.Name != "food") continue; string tmps = li.Find("name").info; mw.Foods.RemoveAll(x => x.Name == tmps); mw.Foods.Add(LPSConvert.DeserializeObject(li)); } } break; case "image": Tag.Add("image"); LoadImage(mw, di); break; case "text": Tag.Add("text"); foreach (FileInfo fi in di.EnumerateFiles("*.lps")) { var tmp = new LpsDocument(File.ReadAllText(fi.FullName)); foreach (ILine li in tmp) { switch (li.Name.ToLower()) { case "lowfoodtext": mw.LowFoodText.Add(LPSConvert.DeserializeObject(li)); Tag.Add("lowtext"); break; case "lowdrinktext": mw.LowDrinkText.Add(LPSConvert.DeserializeObject(li)); Tag.Add("lowtext"); break; case "clicktext": mw.ClickTexts.Add(LPSConvert.DeserializeObject(li)); Tag.Add("clicktext"); break; case "selecttext": mw.SelectTexts.Add(LPSConvert.DeserializeObject(li)); Tag.Add("selecttext"); break; } } } break; case "lang": Tag.Add("lang"); foreach (FileInfo fi in di.EnumerateFiles("*.lps")) { LocalizeCore.AddCulture(fi.Name.Substring(0, fi.Name.Length - fi.Extension.Length), new LPS_D(File.ReadAllText(fi.FullName))); } foreach (DirectoryInfo dis in di.EnumerateDirectories()) { foreach (FileInfo fi in dis.EnumerateFiles("*.lps")) { LocalizeCore.AddCulture(dis.Name, new LPS_D(File.ReadAllText(fi.FullName))); } } if (mw.Set.Language == "null") { LocalizeCore.LoadDefaultCulture(); } else LocalizeCore.LoadCulture(mw.Set.Language); break; case "plugin": Tag.Add("plugin"); SuccessLoad = true; string authtype = ""; foreach (FileInfo tmpfi in di.EnumerateFiles("*.dll")) { #if X64 if (tmpfi.Name.Contains("x86")) { continue; } #else if (tmpfi.Name.Contains("x64")) { continue; } #endif try { var path = tmpfi.Name; if (LoadedDLL.Contains(path)) continue; LoadedDLL.Add(path); X509Certificate2 certificate = new X509Certificate2(tmpfi.FullName); if (certificate != null) { if (certificate.Subject == "CN=\"Shenzhen Lingban Computer Technology Co., Ltd.\", O=\"Shenzhen Lingban Computer Technology Co., Ltd.\", L=Shenzhen, S=Guangdong Province, C=CN, SERIALNUMBER=91440300MA5H8REU3K, OID.2.5.4.15=Private Organization, OID.1.3.6.1.4.1.311.60.2.1.1=Shenzhen, OID.1.3.6.1.4.1.311.60.2.1.2=Guangdong Province, OID.1.3.6.1.4.1.311.60.2.1.3=CN" && certificate.Issuer == "CN=DigiCert Trusted G4 Code Signing RSA4096 SHA384 2021 CA1, O=\"DigiCert, Inc.\", C=US") {//LBGame 信任的证书 if (authtype != "FAIL") authtype = "[认证]".Translate(); } else if (certificate.Subject != "CN=Microsoft Corporation, O=Microsoft Corporation, L=Redmond, S=Washington, C=US" && !IsPassMOD(mw)) {//不是通过模组,不加载 SuccessLoad = false; continue; } else if (authtype != "") { authtype = "[签名]".Translate(); //Intro += $"\nSubject:{certificate.Subject}\nIssuer:{certificate.Subject}"; } } else { authtype = "FAIL"; if (!IsPassMOD(mw)) {//不是通过模组,不加载 SuccessLoad = false; Author = modlps.FindSub("author").Info.Split('[').First(); continue; } } Assembly dll = Assembly.LoadFrom(tmpfi.FullName); var v = dll.GetExportedTypes(); foreach (Type exportedType in v) { if (exportedType.BaseType == typeof(MainPlugin)) { mw.Plugins.Add((MainPlugin)Activator.CreateInstance(exportedType, mw)); } } } catch (Exception e) { ErrorMessage = e.Message; SuccessLoad = false; } } if (authtype != "FAIL") Author += authtype; break; } } #if !DEBUG } catch (Exception e) { ErrorMessage = e.Message; Tag.Add("该模组已损坏"); SuccessLoad = false; } #endif } public bool IsOnMOD(MainWindow mw) => mw.Set.IsOnMod(Name); public bool IsPassMOD(MainWindow mw) => mw.Set.IsPassMOD(Name); public void WriteFile() { LpsDocument modlps = new LpsDocument(File.ReadAllText(Path.FullName + @"\info.lps")); modlps.FindLine("vupmod").Info = Name; modlps.FindLine("intro").Info = Intro; modlps.FindSub("gamever").InfoToInt = GameVer; modlps.FindSub("ver").InfoToInt = Ver; modlps.FindSub("author").Info = Author; modlps.FindorAddLine("authorid").InfoToInt64 = AuthorID; modlps.FindorAddLine("itemid").info = ItemID.ToString(); File.WriteAllText(Path.FullName + @"\info.lps", modlps.ToString()); } } public static class ExtensionSetting { public static void StartURL(string url) { try { var psi = new ProcessStartInfo { FileName = url, UseShellExecute = true }; Process.Start(psi); } catch { ProcessStartInfo startInfo = new ProcessStartInfo(); startInfo.FileName = "explorer.exe"; startInfo.UseShellExecute = false; startInfo.Arguments = url; Process.Start(startInfo); } } /// /// 吃食物 附带倍率 /// /// 存档 /// 食物 /// 默认1倍 public static void EatFood(this IGameSave save, IFood food, double buff) { save.Exp += food.Exp * buff; var tmp = food.Strength / 2 * buff; save.StrengthChange(tmp); save.StoreStrength += tmp; tmp = food.StrengthFood / 2 * buff; save.StrengthChangeFood(tmp); save.StoreStrengthFood += tmp; tmp = food.StrengthDrink / 2 * buff; save.StrengthChangeDrink(tmp); save.StoreStrengthDrink += tmp; tmp = food.Feeling / 2 * buff; save.FeelingChange(tmp); save.StoreFeeling += tmp * buff; save.Health += food.Health * buff; save.Likability += food.Likability * buff; } public static bool IsOnMod(this Setting t, string ModName) { if (CoreMOD.OnModDefList.Contains(ModName)) return true; var line = t.FindLine("onmod"); if (line == null) return false; return line.Find(ModName.ToLower()) != null; } public static bool IsPassMOD(this Setting t, string ModName) { var line = t.FindLine("passmod"); if (line == null) return false; return line.Find(ModName.ToLower()) != null; } public static bool IsMSGMOD(this Setting t, string ModName) { var line = t.FindorAddLine("msgmod"); if (line.GetBool(ModName)) return false; line.SetBool(ModName, true); return true; } public static void OnMod(this Setting t, string ModName) { if (string.IsNullOrWhiteSpace(ModName)) return; t.FindorAddLine("onmod").AddorReplaceSub(new Sub(ModName.ToLower())); } public static void OnModRemove(this Setting t, string ModName) { t.FindorAddLine("onmod").Remove(ModName.ToLower()); } public static void PassMod(this Setting t, string ModName) { t.FindorAddLine("passmod").AddorReplaceSub(new Sub(ModName.ToLower())); } public static void PassModRemove(this Setting t, string ModName) { t.FindorAddLine("passmod").Remove(ModName.ToLower()); } } }