VPet/VPet-Simulator.Windows/Function/CoreMOD.cs

454 lines
20 KiB
C#
Raw Normal View History

2023-01-08 16:57:10 +00:00
using LinePutScript;
2023-06-08 08:46:53 +00:00
using LinePutScript.Converter;
2023-07-01 07:46:08 +00:00
using LinePutScript.Dictionary;
using LinePutScript.Localization.WPF;
2023-01-08 16:57:10 +00:00
using System;
using System.Collections.Generic;
using System.Diagnostics;
2023-01-08 16:57:10 +00:00
using System.IO;
using System.Linq;
using System.Reflection;
2024-03-13 10:57:16 +00:00
using System.Security.Cryptography.X509Certificates;
2023-01-08 16:57:10 +00:00
using VPet_Simulator.Core;
2023-04-01 19:31:28 +00:00
using VPet_Simulator.Windows.Interface;
2023-01-08 16:57:10 +00:00
namespace VPet_Simulator.Windows
{
internal class CoreMOD : IModInfo
2023-01-08 16:57:10 +00:00
{
2023-12-07 08:31:20 +00:00
/// <summary>
/// 自动启用MOD名称
/// </summary>
2024-03-30 09:08:50 +00:00
public static readonly string[] OnModDefList = new string[] { "Core", "PCat", "ModMaker" };
2023-12-07 08:31:20 +00:00
2023-08-10 15:48:27 +00:00
public static HashSet<string> LoadedDLL { get; } = new HashSet<string>()
2023-04-01 19:31:28 +00:00
{
2023-09-04 05:01:16 +00:00
"Panuon.WPF.dll","steam_api.dll","Panuon.WPF.UI.dll","steam_api64.dll",
"LinePutScript.dll","Facepunch.Steamworks.Win32.dll", "Facepunch.Steamworks.Win64.dll",
2023-08-10 15:48:27 +00:00
"VPet-Simulator.Core.dll","VPet-Simulator.Windows.Interface.dll","LinePutScript.Localization.WPF.dll",
"CSCore.dll"
2023-04-01 19:31:28 +00:00
};
2023-01-08 16:57:10 +00:00
public static string NowLoading = null;
public string Name { get; set; }
public string Author { get; set; }
2023-01-08 16:57:10 +00:00
/// <summary>
/// 如果是上传至Steam,则为SteamUserID
/// </summary>
public long AuthorID { get; set; }
2023-01-08 16:57:10 +00:00
/// <summary>
/// 上传至Steam的ItemID
/// </summary>
public ulong ItemID { get; set; }
public string Intro { get; set; }
public DirectoryInfo Path { get; set; }
public int GameVer { get; set; }
public int Ver { get; set; }
public HashSet<string> Tag { get; set; } = new HashSet<string>();
2023-01-08 16:57:10 +00:00
public bool SuccessLoad = true;
2023-06-16 03:39:29 +00:00
public DateTime CacheDate;
2023-09-21 04:26:38 +00:00
public string ErrorMessage;
2024-02-22 18:46:23 +00:00
public static string INTtoVER(int ver) => ver < 10000 ? $"{ver / 100}.{ver % 100:00}" : $"{ver / 10000}.{ver % 10000 / 100}.{ver % 100:00}";
2024-03-15 05:18:05 +00:00
public static void LoadImage(MainWindow mw, DirectoryInfo di, string pre = "")
2023-06-08 08:46:53 +00:00
{
//加载其他放在文件夹的图片
foreach (FileInfo fi in di.EnumerateFiles("*.png"))
{
2024-03-15 05:18:05 +00:00
mw.ImageSources.AddSource(pre + fi.Name.ToLower().Substring(0, fi.Name.Length - 4), fi.FullName);
2023-06-08 08:46:53 +00:00
}
//加载其他放在文件夹中文件夹的图片
foreach (DirectoryInfo fordi in di.EnumerateDirectories())
{
2024-03-15 05:18:05 +00:00
LoadImage(mw, fordi, pre + fordi.Name + "_");
2023-06-08 08:46:53 +00:00
}
//加载标志好的图片和图片设置
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 static void LoadFile(MainWindow mw, DirectoryInfo di, string pre = "")
{
//加载其他放在文件夹的文件
foreach (FileInfo fi in di.EnumerateFiles())
{
mw.FileSources.AddSource(pre + fi.Name, fi.FullName);
}
//加载其他放在文件夹中文件夹的文件
foreach (DirectoryInfo fordi in di.EnumerateDirectories())
{
LoadFile(mw, fordi, pre + fordi.Name + "_");
}
}
2023-01-08 16:57:10 +00:00
public CoreMOD(DirectoryInfo directory, MainWindow mw)
{
2024-03-13 10:57:16 +00:00
#if !DEBUG
2023-08-19 06:27:00 +00:00
try
2023-07-25 23:32:31 +00:00
{
2024-03-13 10:57:16 +00:00
#endif
2024-03-30 07:00:02 +00:00
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);
2024-06-03 15:45:43 +00:00
foreach (var skip in modlps["dllskip"])
{
LoadedDLL.Add(skip.Name);
}
2024-04-01 08:03:48 +00:00
if (CacheDate > DateTime.Now)
{//去掉不合理的清理缓存日期
CacheDate = DateTime.MinValue;
}
2023-12-21 09:26:43 +00:00
2024-03-30 07:00:02 +00:00
//MOD未加载时支持翻译
foreach (var line in modlps.FindAllLine("lang"))
{
List<ILine> ls = new List<ILine>();
foreach (var sub in line)
2024-03-22 15:05:10 +00:00
{
2024-03-30 07:00:02 +00:00
ls.Add(new Line(sub.Name, sub.info));
2024-03-22 15:05:10 +00:00
}
2024-03-30 07:00:02 +00:00
LocalizeCore.AddCulture(line.info, ls);
}
2023-01-08 16:57:10 +00:00
2024-03-30 07:00:02 +00:00
if (mw.CoreMODs.FirstOrDefault(x => x.Name == Name) != null)
{
Name += $"({"MOD名称重复".Translate()})";
ErrorMessage = "MOD名称重复".Translate();
return;
}
2024-03-13 10:57:16 +00:00
2024-03-30 07:00:02 +00:00
if (!IsOnMOD(mw))
{
Tag.Add("该模组已停用");
2024-03-22 15:05:10 +00:00
foreach (DirectoryInfo di in Path.EnumerateDirectories())
2024-03-30 07:00:02 +00:00
Tag.Add(di.Name.ToLower());
return;
}
foreach (DirectoryInfo di in Path.EnumerateDirectories())
{
switch (di.Name.ToLower())
2023-01-08 16:57:10 +00:00
{
2024-03-30 07:00:02 +00:00
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)));
}
2024-02-28 14:02:28 +00:00
2024-03-30 07:00:02 +00:00
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)
2024-02-28 14:02:28 +00:00
{
2024-03-30 07:00:02 +00:00
foreach (FileInfo tmpfi in tmpdi.EnumerateFiles("*.png"))
2024-02-28 14:02:28 +00:00
{
2024-03-30 07:00:02 +00:00
tmp.Images.AddSource(tmpfi.Name.ToLower().Substring(0, tmpfi.Name.Length - 4), tmpfi.FullName);
2024-02-28 14:02:28 +00:00
}
2024-03-30 07:00:02 +00:00
foreach (DirectoryInfo fordi in tmpdi.EnumerateDirectories())
2023-01-10 10:43:32 +00:00
{
2024-03-30 07:00:02 +00:00
foreach (FileInfo tmpfi in fordi.EnumerateFiles("*.png"))
2024-03-22 15:05:10 +00:00
{
2024-03-30 07:00:02 +00:00
tmp.Images.AddSource(fordi.Name + '_' + tmpfi.Name.ToLower().Substring(0, tmpfi.Name.Length - 4), tmpfi.FullName);
2023-08-19 06:27:00 +00:00
}
2023-01-10 10:43:32 +00:00
}
2023-01-08 16:57:10 +00:00
}
2024-03-30 07:00:02 +00:00
}
break;
case "pet":
//宠物模型
foreach (FileInfo fi in di.EnumerateFiles("*.lps"))
{
LpsDocument lps = new LpsDocument(File.ReadAllText(fi.FullName));
if (lps.First().Name.ToLower() == "pet")
2023-06-08 08:46:53 +00:00
{
2024-03-30 07:00:02 +00:00
var name = lps.First().Info;
if (name == "默认虚拟桌宠")
name = "vup";//旧版本名称兼容
var p = mw.Pets.FirstOrDefault(x => x.Name == name);
if (p == null)
2023-08-19 06:27:00 +00:00
{
2024-03-30 07:00:02 +00:00
Tag.Add("pet");
p = new PetLoader(lps, di);
if (p.Config.Works.Count > 0)
Tag.Add("work");
mw.Pets.Add(p);
2023-08-19 06:27:00 +00:00
}
2024-03-30 07:00:02 +00:00
else
2023-06-18 22:20:06 +00:00
{
2024-03-30 07:00:02 +00:00
if (lps.FindAllLine("work").Length >= 0)
2023-08-19 06:27:00 +00:00
{
2024-03-30 07:00:02 +00:00
Tag.Add("work");
2023-08-19 06:27:00 +00:00
}
2024-03-30 07:00:02 +00:00
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);
2023-06-18 22:20:06 +00:00
}
}
2024-03-30 07:00:02 +00:00
}
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)
2023-08-19 06:27:00 +00:00
{
2024-03-30 07:00:02 +00:00
if (li.Name != "food")
continue;
string tmps = li.Find("name").info;
mw.Foods.RemoveAll(x => x.Name == tmps);
mw.Foods.Add(LPSConvert.DeserializeObject<Food>(li));
2023-08-19 06:27:00 +00:00
}
2024-03-30 07:00:02 +00:00
}
break;
case "image":
Tag.Add("image");
LoadImage(mw, di);
break;
case "file":
Tag.Add("file");
LoadFile(mw, di);
break;
2024-03-30 07:00:02 +00:00
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)
2023-07-01 07:46:08 +00:00
{
2024-03-30 07:00:02 +00:00
switch (li.Name.ToLower())
2023-08-19 06:27:00 +00:00
{
2024-03-30 07:00:02 +00:00
case "lowfoodtext":
mw.LowFoodText.Add(LPSConvert.DeserializeObject<LowText>(li));
Tag.Add("lowtext");
break;
case "lowdrinktext":
mw.LowDrinkText.Add(LPSConvert.DeserializeObject<LowText>(li));
Tag.Add("lowtext");
break;
case "clicktext":
mw.ClickTexts.Add(LPSConvert.DeserializeObject<ClickText>(li));
Tag.Add("clicktext");
break;
case "selecttext":
mw.SelectTexts.Add(LPSConvert.DeserializeObject<SelectText>(li));
Tag.Add("selecttext");
break;
case "schedulepackage":
mw.SchedulePackage.Add(LPSConvert.DeserializeObject<ScheduleTask.PackageFull>(li));
break;
2023-08-19 06:27:00 +00:00
}
2023-07-01 07:46:08 +00:00
}
2024-03-30 07:00:02 +00:00
}
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"))
2024-03-13 10:57:16 +00:00
{
2024-03-30 07:00:02 +00:00
LocalizeCore.AddCulture(dis.Name, new LPS_D(File.ReadAllText(fi.FullName)));
2024-03-13 10:57:16 +00:00
}
2024-03-30 07:00:02 +00:00
}
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"))
{
2024-03-22 15:05:10 +00:00
#if X64
2024-03-30 07:00:02 +00:00
if (tmpfi.Name.Contains("x86"))
{
continue;
}
2024-02-19 09:57:03 +00:00
#else
if (tmpfi.Name.Contains("x64"))
{
continue;
}
#endif
2024-03-30 09:08:50 +00:00
#if !DEBUG5
2024-03-30 07:00:02 +00:00
try
{
2024-03-30 09:08:50 +00:00
#endif
2024-03-30 09:59:45 +00:00
var path = tmpfi.Name;
if (LoadedDLL.Contains(path))
2024-03-30 09:08:50 +00:00
continue;
2024-03-30 09:59:45 +00:00
LoadedDLL.Add(path);
X509Certificate2 certificate;
try
{
certificate = new X509Certificate2(tmpfi.FullName);
2024-03-30 07:00:02 +00:00
}
2024-03-30 09:59:45 +00:00
catch
{
certificate = null;
2024-03-22 15:05:10 +00:00
}
2024-03-30 09:59:45 +00:00
if (certificate != null)
2024-03-22 15:05:10 +00:00
{
2024-03-30 09:59:45 +00:00
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();
}
2024-04-01 08:03:48 +00:00
else if (!(certificate.Issuer.Contains("Microsoft Corporation") || certificate.Issuer.Contains(".NET Foundation Projects"))
2024-03-30 09:59:45 +00:00
&& !IsPassMOD(mw))
{//不是通过模组,不加载
SuccessLoad = false;
continue;
}
}
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))
{
2024-04-20 18:37:24 +00:00
var n = exportedType.FullName.ToLower();
if (!(n.Contains("modmaker") || n.Contains("dlc")))
App.MODType.Add(exportedType.FullName);
2024-03-30 09:59:45 +00:00
mw.Plugins.Add((MainPlugin)Activator.CreateInstance(exportedType, mw));
}
2024-03-22 15:05:10 +00:00
}
2024-03-30 09:08:50 +00:00
#if !DEBUG5
2024-03-30 07:00:02 +00:00
}
catch (Exception e)
{
ErrorMessage = e.Message;
SuccessLoad = false;
2023-04-01 19:31:28 +00:00
}
2024-03-30 09:08:50 +00:00
#endif
2024-03-30 07:00:02 +00:00
}
if (authtype != "FAIL")
Author += authtype;
break;
2023-01-08 16:57:10 +00:00
}
2024-03-30 07:00:02 +00:00
}
2024-03-13 10:57:16 +00:00
#if !DEBUG
}
2023-12-07 08:31:20 +00:00
catch (Exception e)
2023-08-19 06:27:00 +00:00
{
2023-09-21 04:26:38 +00:00
ErrorMessage = e.Message;
2023-08-19 06:27:00 +00:00
Tag.Add("该模组已损坏");
SuccessLoad = false;
}
2024-03-13 10:57:16 +00:00
#endif
2023-01-08 16:57:10 +00:00
}
2023-06-25 18:55:03 +00:00
public bool IsOnMOD(MainWindow mw) => mw.Set.IsOnMod(Name);
2024-04-17 21:09:46 +00:00
#if DEBUG
public bool IsPassMOD(MainWindow mw) => true;
#else
2023-01-08 16:57:10 +00:00
public bool IsPassMOD(MainWindow mw) => mw.Set.IsPassMOD(Name);
2024-04-17 21:09:46 +00:00
#endif
2023-01-08 16:57:10 +00:00
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());
2023-06-08 08:46:53 +00:00
}
2023-05-25 18:44:31 +00:00
}
public static class ExtensionSetting
{
2024-04-01 17:24:29 +00:00
internal static bool IsOnMod(this Setting t, string ModName)
2023-05-25 18:44:31 +00:00
{
2023-12-07 08:31:20 +00:00
if (CoreMOD.OnModDefList.Contains(ModName))
2023-06-25 18:55:03 +00:00
return true;
var line = t.FindLine("onmod");
2023-05-25 18:44:31 +00:00
if (line == null)
return false;
return line.Find(ModName.ToLower()) != null;
}
2024-04-01 17:24:29 +00:00
internal static bool IsPassMOD(this Setting t, string ModName)
2023-05-25 18:44:31 +00:00
{
var line = t.FindLine("passmod");
if (line == null)
return false;
return line.Find(ModName.ToLower()) != null;
}
2024-04-01 17:24:29 +00:00
internal static bool IsMSGMOD(this Setting t, string ModName)
2023-05-25 18:44:31 +00:00
{
var line = t.FindorAddLine("msgmod");
if (line.GetBool(ModName))
return false;
line.SetBool(ModName, true);
return true;
}
2024-04-01 17:24:29 +00:00
internal static void OnMod(this Setting t, string ModName)
2023-05-25 18:44:31 +00:00
{
if (string.IsNullOrWhiteSpace(ModName))
return;
2023-06-25 18:55:03 +00:00
t.FindorAddLine("onmod").AddorReplaceSub(new Sub(ModName.ToLower()));
2023-05-25 18:44:31 +00:00
}
2024-04-01 17:24:29 +00:00
internal static void OnModRemove(this Setting t, string ModName)
2023-05-25 18:44:31 +00:00
{
2023-06-25 18:55:03 +00:00
t.FindorAddLine("onmod").Remove(ModName.ToLower());
2023-05-25 18:44:31 +00:00
}
2024-04-01 17:24:29 +00:00
internal static void PassMod(this Setting t, string ModName)
2023-05-25 18:44:31 +00:00
{
t.FindorAddLine("passmod").AddorReplaceSub(new Sub(ModName.ToLower()));
}
2024-04-01 17:24:29 +00:00
internal static void PassModRemove(this Setting t, string ModName)
2023-05-25 18:44:31 +00:00
{
t.FindorAddLine("passmod").Remove(ModName.ToLower());
2023-01-08 16:57:10 +00:00
}
}
}