Big Cleanup (#4)

Renamed Utils to ModLoader
Changed static class to static instance
Moved logging to class and added file logging
Added more protection for modloader properties
Fixed RefreshModFiles now unloads mods before clearing
Fixed trying to load abstract classes and using IsAssignableFrom
Various small improvements
This commit is contained in:
Levi--G 2019-08-04 03:33:33 +02:00 committed by dagoogle
parent 994d2ef04f
commit 54ff3ca085
6 changed files with 251 additions and 162 deletions

View file

@ -1,4 +1,4 @@
<?xml version="1.0" encoding="utf-8"?>
<?xml version="1.0" encoding="utf-8"?>
<Project DefaultTargets="Build" ToolsVersion="4.0" xmlns="http://schemas.microsoft.com/developer/msbuild/2003">
<PropertyGroup>
<Configuration Condition=" '$(Configuration)' == '' ">Debug</Configuration>
@ -35,14 +35,13 @@
<ItemGroup>
<Compile Include="Properties\AssemblyInfo.cs" />
<Compile Include="src\Loader.cs" />
<Compile Include="src\ModLogger.cs" />
<Compile Include="src\ModGUI.cs" />
<Compile Include="src\Utils.cs" />
<Compile Include="src\ModLoader.cs" />
</ItemGroup>
<ItemGroup>
<None Include="packages.config" />
</ItemGroup>
<ItemGroup>
<Folder Include="src\" />
</ItemGroup>
<ItemGroup />
<Import Project="$(MSBuildBinPath)\Microsoft.CSharp.targets" />
</Project>

View file

@ -3,40 +3,47 @@ using System.Collections.Generic;
using System.IO;
using System.Reflection;
using UnityEngine;
namespace ModLoader
namespace BWModLoader
{
public static class Loader
{
public static void Load()
{
Utils.Log("Starting mod loader...");
Utils.DebugLog("Mods dir: "+Utils.modsPath);
string logfile = null;
#if DEBUG
logfile = ModLoader.LogPath + "\\modloader.log";
#endif
ModLogger logger = new ModLogger("[BWML]", logfile);
logger.ClearLog();
logger.Log("Starting mod loader...");
logger.DebugLog("Mods dir: " + ModLoader.ModsPath);
if (!Directory.Exists(Utils.modsPath))
if (!Directory.Exists(ModLoader.ModsPath))
{
Directory.CreateDirectory(Utils.modsPath);
Directory.CreateDirectory(ModLoader.ModsPath);
}
if (!Directory.Exists(Utils.assetsPath))
if (!Directory.Exists(ModLoader.AssetsPath))
{
Directory.CreateDirectory(Utils.assetsPath);
Directory.CreateDirectory(ModLoader.AssetsPath);
}
Utils.modObjects = new GameObject();
//For each DLL in "Blackwake/Blackwake_Data/Managed/Mods/"
//Open them, Get the mod class, then add it in the game.
Utils.RefreshModFiles();
foreach(FileInfo file in Utils.allMods.Keys)
ModLoader loader = new ModLoader(logger);
ModLoader.Instance = loader;
loader.RefreshModFiles();
foreach (FileInfo file in loader.GetAllMods().Keys)
{
Utils.Load(file);
loader.Load(file);
}
Utils.Log("All Mods have been Loaded!");
Utils.modObjects.AddComponent<ModGUI.ModGUI>();
Utils.Log("GUI has been loaded");
logger.Log("All Mods have been Loaded!");
loader.ModObjects.AddComponent<ModGUI.ModGUI>();
logger.Log("GUI has been loaded");
//Keep mods active
UnityEngine.Object.DontDestroyOnLoad(Utils.modObjects);
UnityEngine.Object.DontDestroyOnLoad(loader.ModObjects);
}
}
}

View file

@ -1,10 +1,10 @@
using System;
using System;
using System.Collections;
using System.Collections.Generic;
using System.Collections.Generic;
using System.IO;
using System.Linq;
using UnityEngine;
namespace ModGUI
namespace BWModLoader.ModGUI
{
public class ModGUI : MonoBehaviour
{
@ -57,51 +57,52 @@ namespace ModGUI
void LogWindow()
{
GUI.Label(new Rect(0, 100, 100, 25), "LogWindow");
int logNum = 0;
if (ModLoader.Utils.logs.Any())
{
GUI.Label(new Rect(0, 100, 100, 25), "LogWindow");
int logNum = 0;
if (ModLoader.Instance.Logger.Logs.Any())
{
scrollPosition = GUI.BeginScrollView(new Rect(0, 100, size.x, size.y - 100), scrollPosition, new Rect(0, 0, size.x, 50));
foreach (string log in ModLoader.Utils.logs)
foreach (string log in ModLoader.Instance.Logger.Logs)
{
logNum++;
GUI.Label(new Rect(0, 25 * logNum, 1000, 25), log);
}
GUI.EndScrollView();
}
GUI.EndScrollView();
}
}
void ModWindow()
{
if (GUI.Button(new Rect(0, 100, 100, 25), "Reload all mods"))
{
ModLoader.Utils.RefreshModFiles();
ModLoader.Instance.RefreshModFiles();
}
scrollPosition = GUI.BeginScrollView(new Rect(0, 100, size.x, size.y-100), scrollPosition, new Rect(0, 0, size.x, 50));
int modNum = 0;
foreach (FileInfo file in ModLoader.Utils.allMods.Keys)
{
foreach (Type mod in ModLoader.Utils.allMods[file])
var allmods = ModLoader.Instance.GetAllMods();
foreach (FileInfo file in allmods.Keys)
{
foreach (Type mod in allmods[file])
{
modNum++;
GUI.Label(new Rect(0, modNum * 25, 100, 25), mod.Name);
if (!ModLoader.Utils.IsLoaded(mod))
GUI.Label(new Rect(0, modNum * 25, 100, 25), mod.Name);
if (!ModLoader.Instance.IsLoaded(mod))
{
if (GUI.Button(new Rect(100, modNum * 25, 100, 25), "Enable"))
{
ModLoader.Utils.Load(file);
}
}
ModLoader.Instance.Load(file);
}
}
else
{
if (GUI.Button(new Rect(100, modNum * 25, 100, 25), "Disable"))
{
ModLoader.Utils.Unload(file);
}
ModLoader.Instance.Unload(file);
}
}
if (GUI.Button(new Rect(200, modNum * 25, 100, 25), "Reload"))
{
ModLoader.Utils.Unload(file);
ModLoader.Utils.RefreshModFiles();
ModLoader.Instance.Unload(file);
ModLoader.Instance.RefreshModFiles();
}
}
}

141
src/ModLoader.cs Normal file
View file

@ -0,0 +1,141 @@
using System;
using System.Collections.Generic;
using System.IO;
using System.Reflection;
using UnityEngine;
namespace BWModLoader
{
public class ModLoader
{
public static ModLoader Instance { get; internal set; }
private static readonly string dllpath = new Uri(Assembly.GetExecutingAssembly().CodeBase).LocalPath;
private static readonly string folderPath = Path.GetDirectoryName(dllpath);
public static string ModsPath => folderPath + "\\Mods";
public static string AssetsPath => ModsPath + "\\Assets";
public static string LogPath => ModsPath + "\\Logs";
public ModLogger Logger;
/// <summary>
/// All known mods files
/// </summary>
private readonly Dictionary<FileInfo, List<Type>> allMods = new Dictionary<FileInfo, List<Type>>();
/// <summary>
/// Gets all known mods files
/// </summary>
/// Prevent modification from outside by making copy
public Dictionary<FileInfo, List<Type>> GetAllMods() => new Dictionary<FileInfo, List<Type>>(allMods);
/// <summary>
/// GameObject that holds our mods
/// </summary>
public GameObject ModObjects { get; } = new GameObject();
public ModLoader(ModLogger logger)
{
this.Logger = logger;
}
/// <summary>
/// Checks if a mod is loaded
/// </summary>
/// <param name="mod"></param>
/// <returns></returns>
public bool IsLoaded(Type mod)
{
return ModObjects.GetComponent(mod) != null;
}
/// <summary>
/// Refresh all known mods
/// </summary>
public void RefreshModFiles()
{
DirectoryInfo dir = new DirectoryInfo(ModsPath);
//Unloads & clears known mods
foreach (var mod in allMods)
{
Unload(mod.Key);
}
allMods.Clear();
//Find all files to refresh
foreach (FileInfo file in dir.GetFiles("*.dll"))
{
//Save mod types and file path
allMods.Add(file, LoadModTypes(file));
Logger.Log("Found dll: " + file.Name);
}
}
/// <summary>
/// Finds and loads all mod classes in a file
/// </summary>
/// <param name="file">The file to load</param>
/// <returns></returns>
private List<Type> LoadModTypes(FileInfo file)
{
List<Type> mods = new List<Type>();
try
{
Assembly modDll = Assembly.LoadFrom(file.FullName);
Type[] modType = modDll.GetTypes();
foreach (Type t in modType)
{
Logger.Log("Found type in " + file.Name + ": " + t.Name);
if (t.IsClass && typeof(MonoBehaviour).IsAssignableFrom(t) && !t.IsAbstract && t.IsPublic)
{
mods.Add(t);
}
}
}
catch (Exception e)
{
Logger.Log("Exception raised while loading mod " + file.Name);
Logger.Log(e.Message);
Logger.Log("Skipped loading this mod");
}
return mods;
}
/// <summary>
/// will load a mod from memory
/// </summary>
/// <param name="file"></param>
public void Load(FileInfo file)
{
if (allMods.TryGetValue(file, out var types))
{
foreach (Type mod in types)
{
//if mod is not loaded
if (!IsLoaded(mod))
{
ModObjects.AddComponent(mod);
Logger.Log("Loaded: " + mod.Name + " From file: " + file.Name);
}
}
}
}
//Unloads a mod from game
public void Unload(FileInfo file)
{
if (allMods.TryGetValue(file, out var types))
{
foreach (Type mod in types)
{
//if mod is loaded
if (IsLoaded(mod))
{
UnityEngine.Object.Destroy(ModObjects.GetComponent(mod));
Logger.Log("Unloaded: " + mod.Name + " From file: " + file.Name);
}
}
}
}
}
}

59
src/ModLogger.cs Normal file
View file

@ -0,0 +1,59 @@
using System;
using System.Collections.Generic;
using System.IO;
using System.Linq;
using System.Text;
namespace BWModLoader
{
public class ModLogger
{
string prefix;
string file;
/// <summary>
/// Log history
/// </summary>
public List<string> Logs { get; } = new List<string>();
public ModLogger(string prefix, string file = null)
{
this.prefix = prefix;
this.file = file;
if (!string.IsNullOrEmpty(file) && !File.Exists(file))
{
Directory.CreateDirectory(Path.GetDirectoryName(file));
File.WriteAllText(file, "");
}
}
public void ClearLog()
{
Logs.Clear();
if (!string.IsNullOrEmpty(file) && File.Exists(file))
{
File.Delete(file);
}
}
/// <summary>
/// Log to debugger and output.txt
/// </summary>
public void Log(string output)
{
Console.WriteLine(prefix + output);
Logs.Add(prefix + output);
if (!string.IsNullOrEmpty(file))
{
File.AppendAllText(file, prefix + output + Environment.NewLine);
}
}
public void DebugLog(string output)
{
#if DEBUG
Log(output);
#endif // DEBUG
}
}
}

View file

@ -1,118 +0,0 @@
using System;
using System.Collections.Generic;
using System.IO;
using System.Reflection;
using UnityEngine;
namespace ModLoader
{
public static class Utils
{
static readonly string dllpath = new System.Uri(System.Reflection.Assembly.GetExecutingAssembly().CodeBase).LocalPath;
static readonly string path = new FileInfo(dllpath).Directory.FullName;
public static string modsPath = path + "\\Mods";
public static string assetsPath = modsPath + "\\Assets";
//all known mods files and their classes
public static Dictionary<FileInfo, List<Type>> allMods = new Dictionary<FileInfo, List<Type>>();
//Log history
public static List<string> logs = new List<string>();
//Object that holds our mods
public static GameObject modObjects;
//checks if mod is loaded
public static bool IsLoaded(Type mod)
{
if (modObjects.GetComponent(mod) != null)
{
return true;
}
return false;
}
//Refresh all known mods
public static void RefreshModFiles()
{
DirectoryInfo dir = new DirectoryInfo(modsPath);
//Clears known mods
allMods.Clear();
//Find all files to refresh
foreach (FileInfo file in dir.GetFiles("*.dll"))
{
allMods.Add(file, new List<Type>());
foreach(Type mod in FindModTypes(file))
{
//Save mod type and file path
allMods[file].Add(mod);
}
Log("Found dll: " + file.Name);
}
}
//Find all mod classes in a file
static List<Type> FindModTypes(FileInfo file)
{
List<Type> mods = new List<Type>();
try
{
Assembly modDll = Assembly.LoadFrom(file.FullName);
Type[] modType = modDll.GetTypes();
foreach (Type t in modType)
{
Log("Found type in " + file.Name + ": " + t.Name);
if (t.IsClass && t.IsSubclassOf(typeof(MonoBehaviour)))
{
mods.Add(t);
}
}
}
catch (Exception e)
{
Log("Exception raised while loading mod " + file.Name);
Log(e.Message);
Log("Skipped loading this mod");
}
return mods;
}
//will load a mod from memory
public static void Load(FileInfo file)
{
if (allMods.ContainsKey(file))
{
foreach (Type mod in allMods[file])
{
//if mod is not loaded
if (modObjects.GetComponent(mod) == null)
{
modObjects.AddComponent(mod);
Log("Loaded: " + mod.Name + " From file: " + file.Name);
}
}
}
}
//Unloads a mod from game
public static void Unload(FileInfo file)
{
foreach(Type mod in allMods[file])
{
//if mod is loaded
if (modObjects.GetComponent(mod) != null)
{
UnityEngine.Object.Destroy(modObjects.GetComponent(mod));
Log("Unloaded: " + mod.Name + " From file: " + file.Name);
}
}
}
//Log to debugger and output.txt
public static void Log(string output)
{
Console.WriteLine("[BWML]" + output);
logs.Add(output);
}
public static void DebugLog(string output)
{
#if DEBUG
Log(output);
#endif // DEBUG
}
}
}