-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathT2Module.cs
245 lines (214 loc) · 12.7 KB
/
T2Module.cs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
using BepInEx.Configuration;
using R2API;
using RoR2;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Reflection;
using UnityEngine.Networking;
using static TILER2.MiscUtil;
namespace TILER2 {
public abstract class T2Module<T>:T2Module where T : T2Module<T> {
public static T instance {get;private set;}
protected T2Module() {
if(instance != null) throw new InvalidOperationException("Singleton class \"" + typeof(T).Name + "\" inheriting Module was instantiated twice");
instance = this as T;
}
}
/// <summary>
/// Provides a relatively low-extra-code pattern for dividing a mod into smaller modules, each of which has its own config category managed by TILER2.AutoConfig.
/// </summary>
public abstract class T2Module : AutoConfigContainer {
internal static void SetupModuleClass() {
On.RoR2.Run.Start += On_RunStart;
On.RoR2.Language.SetCurrentLanguage += Language_SetCurrentLanguage;
}
private static void On_RunStart(On.RoR2.Run.orig_Start orig, Run self) {
orig(self);
if(!NetworkServer.active) return;
var rngGenerator = new Xoroshiro128Plus(self.seed);
foreach(var module in _allModules)
module.rng = new Xoroshiro128Plus(rngGenerator.nextUlong);
}
private static void Language_SetCurrentLanguage(On.RoR2.Language.orig_SetCurrentLanguage orig, string newCurrentLanguageName) {
orig(newCurrentLanguageName);
foreach(var module in allModules) {
module.RefreshPermanentLanguage();
if(module.enabled) {
if(module.languageInstalled)
module.UninstallLanguage();
module.InstallLanguage();
}
}
AutoConfigModule.globalLanguageDirty = false;
}
private static readonly FilingDictionary<T2Module> _allModules = new();
public static readonly ReadOnlyFilingDictionary<T2Module> allModules = _allModules.AsReadOnly();
public bool enabled { get; protected internal set; } = true;
public readonly string name;
///<summary>If true, Module.enabled will be registered as a config entry.</summary>
public virtual bool managedEnable => true;
///<summary>If true and managedEnabled is true, Module.enabled will be registered as a Risk Of Options option.</summary>
public virtual bool managedEnableRoO => true;
///<summary>If managedEnable is true, this will be appended to the module's enable/disable config description.</summary>
public virtual string enabledConfigDescription => null;
///<summary>If managedEnable is true, this will be used for the resultant config entry.</summary>
public virtual AutoConfigFlags enabledConfigFlags => AutoConfigFlags.PreventNetMismatch;
///<summary>If managedEnable is true, this will be used for the resultant config entry.</summary>
public virtual AutoConfigUpdateActionTypes enabledConfigUpdateActionTypes => AutoConfigUpdateActionTypes.InvalidateLanguage;
protected readonly List<LanguageAPI.LanguageOverlay> languageOverlays = new();
protected readonly List<LanguageAPI.LanguageOverlay> permanentLanguageOverlays = new();
protected readonly Dictionary<string, string> genericLanguageTokens = new();
protected readonly Dictionary<string, Dictionary<string, string>> specificLanguageTokens = new();
protected readonly Dictionary<string, string> permanentGenericLanguageTokens = new();
protected readonly Dictionary<string, Dictionary<string, string>> permanentSpecificLanguageTokens = new();
public bool languageInstalled { get; private set; } = false;
public bool permanentLanguageInstalled { get; private set; } = false;
///<summary>A server-only rng instance based on the current run's seed.</summary>
public Xoroshiro128Plus rng { get; internal set; }
///<summary>Contains various information relating to the mod owning this module.</summary>
public ModInfo modInfo {get; private set;}
///<summary>Will be prepended to the category name of all config entries of this subclass of T2Module.</summary>
public virtual string configCategoryPrefix => "Modules.";
/// <summary>
/// Implement to handle AutoItemConfig binding and other related actions. With standard base plugin setup, will be performed before SetupAttributes and SetupBehavior.
/// </summary>
public virtual void SetupConfig() {
var moduleConfigName = $"{configCategoryPrefix}{name}";
if(managedEnable) {
Bind(typeof(T2Module).GetProperty(nameof(enabled)), modInfo.mainConfigFile, modInfo.displayName, moduleConfigName, new AutoConfigAttribute(
$"{((enabledConfigDescription != null) ? (enabledConfigDescription + "\n") : "")}Set to False to disable this module, and as much of its content as can be disabled after initial load. Doing so may cause changes in other modules as well.",
enabledConfigFlags), enabledConfigUpdateActionTypes != AutoConfigUpdateActionTypes.None ? new AutoConfigUpdateActionsAttribute(enabledConfigUpdateActionTypes) : null);
if(managedEnableRoO && Compat_RiskOfOptions.enabled) {
var binding = bindings.First(x => x.boundProperty == typeof(T2Module).GetProperty(nameof(enabled)));
BindRoO(binding, new AutoConfigRoOCheckboxAttribute());
}
}
BindAll(modInfo.mainConfigFile, modInfo.displayName, moduleConfigName);
ConfigEntryChanged += (sender, args) => {
if(args.target.boundProperty.Name == nameof(enabled)) {
if((bool)args.newValue == true) {
Install();
} else {
Uninstall();
if(languageInstalled)
UninstallLanguage();
}
RefreshPermanentLanguage();
}
if(args.flags.HasFlag(AutoConfigUpdateActionTypes.InvalidateLanguage)) {
if(enabled) {
if(languageInstalled)
UninstallLanguage();
InstallLanguage();
}
RefreshPermanentLanguage();
}
};
}
///<summary>
///Implement to handle registration with RoR2 catalogs.
///</summary>
public virtual void SetupAttributes() {}
///<summary>Third stage of setup. Should be used to apply permanent hooks and other similar things.</summary>
public virtual void SetupBehavior() {}
///<summary>Fourth stage of setup. Will be performed after all catalogs have initialized.</summary>
public virtual void SetupLate() {}
///<summary>Fifth stage of setup. Should be used to perform final, non-permanent hooks and changes.</summary>
public virtual void Install() {}
///<summary>Should undo EVERY change made by Install.</summary>
public virtual void Uninstall() {}
///<summary>Will be called once after initial language setup, and also if/when the module is installed after setup. Automatically loads tokens from genericLanguageTokens/specificLanguageTokens into R2API Language Overlays in languageOverlays.</summary>
public virtual void InstallLanguage() {
languageOverlays.Add(LanguageAPI.AddOverlay(genericLanguageTokens));
languageOverlays.Add(LanguageAPI.AddOverlay(specificLanguageTokens));
languageInstalled = true;
AutoConfigModule.globalLanguageDirty = true;
}
///<summary>Will be called if/when the module is uninstalled after setup, and before any language installation after the first. Automatically undoes all R2API Language Overlays registered to languageOverlays.</summary>
public virtual void UninstallLanguage() {
foreach(var overlay in languageOverlays) {
overlay.Remove();
}
languageOverlays.Clear();
languageInstalled = false;
AutoConfigModule.globalLanguageDirty = true;
}
///<summary>Handles language that should never be uninstalled for an extended period (e.g. item tokens), contained in permanentGenericLanguageTokens/permanentSpecificLanguageTokens.</summary>
public virtual void RefreshPermanentLanguage() {
if(permanentLanguageInstalled) {
foreach(var overlay in permanentLanguageOverlays)
overlay.Remove();
}
permanentLanguageOverlays.Clear();
permanentLanguageOverlays.Add(LanguageAPI.AddOverlay(permanentGenericLanguageTokens));
permanentLanguageOverlays.Add(LanguageAPI.AddOverlay(permanentSpecificLanguageTokens));
AutoConfigModule.globalLanguageDirty = true;
}
/// <summary>
/// Call to scan your plugin's assembly for classes inheriting directly from a specific subtype of Module, initialize all of them, and prepare a list for further setup.
/// Has special handling for the MyClass : ModuleOrModuleSubclass<MyClass> pattern.
/// </summary>
/// <returns>A FilingDictionary containing all instances that this method just initialized.</returns>
public static FilingDictionary<T> InitDirect<T>(ModInfo modInfo) where T:T2Module {
return InitAll<T>(modInfo, Assembly.GetCallingAssembly(), t => (t.BaseType.IsGenericType
? (t.BaseType.GenericTypeArguments[0] == t && t.BaseType.BaseType == typeof(T))
: t.BaseType == typeof(T)));
}
/// <summary>
/// Call to scan your plugin's assembly for classes inheriting directly or indirectly from a specific subtype of Module, initialize all of them, and prepare a list for further setup.
/// </summary>
/// <returns>A FilingDictionary containing all instances that this method just initialized.</returns>
public static FilingDictionary<T> InitAll<T>(ModInfo modInfo, Func<Type, bool> extraTypeChecks = null) where T:T2Module {
return InitAll<T>(modInfo, Assembly.GetCallingAssembly(), extraTypeChecks);
}
private static FilingDictionary<T> InitAll<T>(ModInfo modInfo, Assembly callingAssembly, Func<Type, bool> extraTypeChecks) where T:T2Module {
var f = new FilingDictionary<T>();
foreach(Type type in callingAssembly.GetTypes().Where(t => t.IsClass && !t.IsAbstract && t.IsSubclassOf(typeof(T)) && (extraTypeChecks?.Invoke(t) ?? true))) {
var newModule = (T)Activator.CreateInstance(type, nonPublic: true);
newModule.modInfo = modInfo;
f.Add(newModule);
}
return f;
}
/// <summary>
/// Call to scan your plugin's assembly for classes inheriting directly from Module, initialize all of them, and prepare a list for further setup.
/// </summary>
/// <returns>A FilingDictionary containing all instances that this method just initialized.</returns>
public static FilingDictionary<T2Module> InitModules(ModInfo modInfo) {
return InitAll<T2Module>(modInfo, Assembly.GetCallingAssembly(), t => (t.BaseType.IsGenericType
? (t.BaseType.GenericTypeArguments[0] == t && t.BaseType.BaseType == typeof(T2Module))
: t.BaseType == typeof(T2Module)));
}
public static void SetupAll_PluginAwake(IEnumerable<T2Module> modulesToSetup) {
foreach(var module in modulesToSetup) {
module.SetupConfig();
}
foreach(var module in modulesToSetup) {
module.SetupAttributes();
}
foreach(var module in modulesToSetup) {
module.SetupBehavior();
}
}
public static void SetupAll_PluginStart(IEnumerable<T2Module> modulesToSetup, bool installUnmanaged = false) {
foreach(var module in modulesToSetup) {
module.SetupLate();
}
foreach(var module in modulesToSetup) {
if((installUnmanaged || module.managedEnable) && module.enabled)
module.Install();
}
}
protected T2Module() {
name = GetType().Name;
_allModules.Add(this);
}
public struct ModInfo {
public string displayName;
public string longIdentifier;
public string shortIdentifier;
public ConfigFile mainConfigFile;
}
}
}