Commit e38f7f248651a15b730929f96b622539e5930f97
1 parent
813a72f5
LiteLoader 1.6.4_02 - experimental - add Exposable interface and support for mod config via Gson
Showing
10 changed files
with
625 additions
and
84 deletions
java/com/mumfrey/liteloader/ExposeConfig.java
0 → 100644
1 | +package com.mumfrey.liteloader; | |
2 | + | |
3 | +import java.lang.annotation.ElementType; | |
4 | +import java.lang.annotation.Retention; | |
5 | +import java.lang.annotation.RetentionPolicy; | |
6 | +import java.lang.annotation.Target; | |
7 | + | |
8 | +import com.mumfrey.liteloader.modconfig.ConfigStrategy; | |
9 | + | |
10 | +/** | |
11 | + * Annotation which can be a applied to mod classes to indicate they should be serialised with Gson | |
12 | + * | |
13 | + * @author Adam Mummery-Smith | |
14 | + */ | |
15 | +@Target(ElementType.TYPE) | |
16 | +@Retention(RetentionPolicy.RUNTIME) | |
17 | +public @interface ExposeConfig | |
18 | +{ | |
19 | + /** | |
20 | + * Configuration strategy to use | |
21 | + */ | |
22 | + ConfigStrategy strategy() default ConfigStrategy.Unversioned; | |
23 | + | |
24 | + /** | |
25 | + * Config file name, if not specified the mod class name is used | |
26 | + */ | |
27 | + String filename() default ""; | |
28 | +} | ... | ... |
java/com/mumfrey/liteloader/LiteMod.java
... | ... | @@ -2,12 +2,14 @@ package com.mumfrey.liteloader; |
2 | 2 | |
3 | 3 | import java.io.File; |
4 | 4 | |
5 | +import com.mumfrey.liteloader.modconfig.Exposable; | |
6 | + | |
5 | 7 | /** |
6 | 8 | * Base interface for mods |
7 | 9 | * |
8 | 10 | * @author Adam Mummery-Smith |
9 | 11 | */ |
10 | -public interface LiteMod | |
12 | +public interface LiteMod extends Exposable | |
11 | 13 | { |
12 | 14 | /** |
13 | 15 | * Get the mod's display name | ... | ... |
java/com/mumfrey/liteloader/core/LiteLoader.java
... | ... | @@ -40,7 +40,8 @@ import com.mumfrey.liteloader.crashreport.CallableLiteLoaderBrand; |
40 | 40 | import com.mumfrey.liteloader.crashreport.CallableLiteLoaderMods; |
41 | 41 | import com.mumfrey.liteloader.gui.GuiControlsPaginated; |
42 | 42 | import com.mumfrey.liteloader.gui.GuiScreenModInfo; |
43 | -import com.mumfrey.liteloader.modconfig.ConfigPanelManager; | |
43 | +import com.mumfrey.liteloader.modconfig.ConfigManager; | |
44 | +import com.mumfrey.liteloader.modconfig.Exposable; | |
44 | 45 | import com.mumfrey.liteloader.permissions.PermissionsManagerClient; |
45 | 46 | import com.mumfrey.liteloader.util.PrivateFields; |
46 | 47 | |
... | ... | @@ -170,9 +171,9 @@ public final class LiteLoader |
170 | 171 | private final PermissionsManagerClient permissionsManager = PermissionsManagerClient.getInstance(); |
171 | 172 | |
172 | 173 | /** |
173 | - * Configuration panel manager/registry | |
174 | + * Mod configuration manager | |
174 | 175 | */ |
175 | - private final ConfigPanelManager configPanelManager; | |
176 | + private final ConfigManager configManager; | |
176 | 177 | |
177 | 178 | /** |
178 | 179 | * Flag which keeps track of whether late initialisation has been done |
... | ... | @@ -279,7 +280,7 @@ public final class LiteLoader |
279 | 280 | this.enabledModsList = EnabledModsList.createFrom(this.enabledModsFile); |
280 | 281 | this.enabledModsList.processModsList(bootstrap.getProfile(), modNameFilter); |
281 | 282 | |
282 | - this.configPanelManager = new ConfigPanelManager(); | |
283 | + this.configManager = new ConfigManager(); | |
283 | 284 | } |
284 | 285 | |
285 | 286 | /** |
... | ... | @@ -787,6 +788,37 @@ public final class LiteLoader |
787 | 788 | } |
788 | 789 | |
789 | 790 | /** |
791 | + * @param modClass | |
792 | + */ | |
793 | + public void writeConfig(LiteMod mod) | |
794 | + { | |
795 | + if (mod != null) | |
796 | + { | |
797 | + this.writeConfig(mod.getClass()); | |
798 | + } | |
799 | + } | |
800 | + | |
801 | + /** | |
802 | + * @param exposableClass | |
803 | + */ | |
804 | + public void writeConfig(Class<? extends Exposable> exposableClass) | |
805 | + { | |
806 | + this.configManager.invalidateConfig(exposableClass); | |
807 | + } | |
808 | + | |
809 | + /** | |
810 | + * Register an arbitrary Exposable | |
811 | + * | |
812 | + * @param exposable Exposable object to register | |
813 | + * @param fileName | |
814 | + */ | |
815 | + public void registerExposable(Exposable exposable, String fileName) | |
816 | + { | |
817 | + this.configManager.registerExposable(exposable, fileName, true); | |
818 | + this.configManager.initConfig(exposable); | |
819 | + } | |
820 | + | |
821 | + /** | |
790 | 822 | * Create mod instances from the enumerated classes |
791 | 823 | * |
792 | 824 | * @param modsToLoad List of mods to load |
... | ... | @@ -898,6 +930,9 @@ public final class LiteLoader |
898 | 930 | { |
899 | 931 | LiteLoader.logInfo("Initialising mod %s version %s", mod.getName(), mod.getVersion()); |
900 | 932 | |
933 | + // register mod config panel if configurable | |
934 | + this.configManager.registerMod(mod); | |
935 | + | |
901 | 936 | try |
902 | 937 | { |
903 | 938 | this.handleModVersionUpgrade(mod); |
... | ... | @@ -907,6 +942,9 @@ public final class LiteLoader |
907 | 942 | LiteLoader.logWarning("Error performing settings upgrade for %s. Settings may not be properly migrated", mod.getName()); |
908 | 943 | } |
909 | 944 | |
945 | + // Init mod config if there is any | |
946 | + this.configManager.initConfig(mod); | |
947 | + | |
910 | 948 | // initialise the mod |
911 | 949 | mod.init(this.commonConfigFolder); |
912 | 950 | |
... | ... | @@ -916,9 +954,6 @@ public final class LiteLoader |
916 | 954 | // add mod to permissions manager if permissible |
917 | 955 | this.permissionsManager.registerMod(mod); |
918 | 956 | |
919 | - // register mod config panel if configurable | |
920 | - this.configPanelManager.registerMod(mod); | |
921 | - | |
922 | 957 | this.loadedMods.add(mod); |
923 | 958 | this.loadedModsList += String.format("\n - %s version %s", mod.getName(), mod.getVersion()); |
924 | 959 | } |
... | ... | @@ -934,6 +969,11 @@ public final class LiteLoader |
934 | 969 | if (LiteLoaderBootstrap.VERSION.getLoaderRevision() > lastModVersion.getLoaderRevision()) |
935 | 970 | { |
936 | 971 | LiteLoader.logInfo("Performing config upgrade for mod %s. Upgrading %s to %s...", mod.getName(), lastModVersion, LiteLoaderBootstrap.VERSION); |
972 | + | |
973 | + // Migrate versioned config if any is present | |
974 | + this.configManager.migrateModConfig(mod, this.versionConfigFolder, this.inflectVersionedConfigPath(lastModVersion)); | |
975 | + | |
976 | + // Let the mod upgrade | |
937 | 977 | mod.upgradeSettings(LiteLoaderBootstrap.VERSION.getMinecraftVersion(), this.versionConfigFolder, this.inflectVersionedConfigPath(lastModVersion)); |
938 | 978 | |
939 | 979 | this.bootstrap.storeLastKnownModRevision(modKey); |
... | ... | @@ -1038,7 +1078,7 @@ public final class LiteLoader |
1038 | 1078 | // If we're at the main menu, prepare the overlay |
1039 | 1079 | if (this.modInfoScreen == null || this.modInfoScreen.getMenu() != this.minecraft.currentScreen) |
1040 | 1080 | { |
1041 | - this.modInfoScreen = new GuiScreenModInfo(this.minecraft, (GuiMainMenu)this.minecraft.currentScreen, this, this.enabledModsList, this.configPanelManager); | |
1081 | + this.modInfoScreen = new GuiScreenModInfo(this.minecraft, (GuiMainMenu)this.minecraft.currentScreen, this, this.enabledModsList, this.configManager); | |
1042 | 1082 | } |
1043 | 1083 | |
1044 | 1084 | this.modInfoScreen.drawScreen(mouseX, mouseY, partialTicks); |
... | ... | @@ -1051,7 +1091,7 @@ public final class LiteLoader |
1051 | 1091 | } |
1052 | 1092 | else if (this.minecraft.currentScreen instanceof GuiMainMenu && Keyboard.isKeyDown(Keyboard.KEY_LCONTROL) && Keyboard.isKeyDown(Keyboard.KEY_LSHIFT) && Keyboard.isKeyDown(Keyboard.KEY_TAB)) |
1053 | 1093 | { |
1054 | - this.minecraft.displayGuiScreen(new GuiScreenModInfo(this.minecraft, (GuiMainMenu)this.minecraft.currentScreen, this, this.enabledModsList, this.configPanelManager)); | |
1094 | + this.minecraft.displayGuiScreen(new GuiScreenModInfo(this.minecraft, (GuiMainMenu)this.minecraft.currentScreen, this, this.enabledModsList, this.configManager)); | |
1055 | 1095 | } |
1056 | 1096 | } |
1057 | 1097 | |
... | ... | @@ -1064,14 +1104,30 @@ public final class LiteLoader |
1064 | 1104 | // Tick the permissions manager |
1065 | 1105 | this.permissionsManager.onTick(this.minecraft, partialTicks, inGame); |
1066 | 1106 | |
1107 | + // Tick the config manager | |
1108 | + this.configManager.onTick(); | |
1109 | + | |
1067 | 1110 | this.checkAndStoreKeyBindings(); |
1068 | 1111 | |
1069 | 1112 | if (this.modInfoScreen != null && this.minecraft.currentScreen != this.modInfoScreen) |
1070 | 1113 | { |
1071 | 1114 | this.modInfoScreen.updateScreen(); |
1072 | 1115 | } |
1116 | + | |
1117 | + if (!PrivateFields.gameIsRunning.get(this.minecraft)) | |
1118 | + { | |
1119 | + this.onShutDown(); | |
1120 | + } | |
1073 | 1121 | } |
1074 | - | |
1122 | + | |
1123 | + private void onShutDown() | |
1124 | + { | |
1125 | + LiteLoader.logInfo("LiteLoader is shutting down, syncing configuration"); | |
1126 | + | |
1127 | + this.storeBindings(); | |
1128 | + this.configManager.syncConfig(); | |
1129 | + } | |
1130 | + | |
1075 | 1131 | /** |
1076 | 1132 | * Register a key for a mod |
1077 | 1133 | * | ... | ... |
java/com/mumfrey/liteloader/gui/GuiScreenModInfo.java
... | ... | @@ -20,7 +20,7 @@ import com.mumfrey.liteloader.core.EnabledModsList; |
20 | 20 | import com.mumfrey.liteloader.core.LiteLoader; |
21 | 21 | import com.mumfrey.liteloader.core.ModFile; |
22 | 22 | import com.mumfrey.liteloader.modconfig.ConfigPanel; |
23 | -import com.mumfrey.liteloader.modconfig.ConfigPanelManager; | |
23 | +import com.mumfrey.liteloader.modconfig.ConfigManager; | |
24 | 24 | |
25 | 25 | import net.minecraft.src.DynamicTexture; |
26 | 26 | import net.minecraft.src.GuiButton; |
... | ... | @@ -136,7 +136,7 @@ public class GuiScreenModInfo extends GuiScreen |
136 | 136 | */ |
137 | 137 | private GuiCheckbox chkEnabled; |
138 | 138 | |
139 | - private ConfigPanelManager configPanelManager; | |
139 | + private ConfigManager configManager; | |
140 | 140 | |
141 | 141 | /** |
142 | 142 | * Configuration panel |
... | ... | @@ -149,11 +149,11 @@ public class GuiScreenModInfo extends GuiScreen |
149 | 149 | * @param loader |
150 | 150 | * @param enabledModsList |
151 | 151 | */ |
152 | - public GuiScreenModInfo(Minecraft minecraft, GuiMainMenu mainMenu, LiteLoader loader, EnabledModsList enabledModsList, ConfigPanelManager configPanelManager) | |
152 | + public GuiScreenModInfo(Minecraft minecraft, GuiMainMenu mainMenu, LiteLoader loader, EnabledModsList enabledModsList, ConfigManager configManager) | |
153 | 153 | { |
154 | 154 | this.mc = minecraft; |
155 | 155 | this.mainMenu = mainMenu; |
156 | - this.configPanelManager = configPanelManager; | |
156 | + this.configManager = configManager; | |
157 | 157 | |
158 | 158 | // Spawn the texture resource if we haven't already |
159 | 159 | if (aboutTexture == null) |
... | ... | @@ -484,7 +484,7 @@ public class GuiScreenModInfo extends GuiScreen |
484 | 484 | this.btnToggle.drawButton = true; |
485 | 485 | this.btnToggle.displayString = this.selectedMod.willBeEnabled() ? "Disable mod" : "Enable mod"; |
486 | 486 | |
487 | - this.btnConfig.drawButton = this.configPanelManager.hasPanel(this.selectedMod.getModClass()); | |
487 | + this.btnConfig.drawButton = this.configManager.hasPanel(this.selectedMod.getModClass()); | |
488 | 488 | } |
489 | 489 | } |
490 | 490 | |
... | ... | @@ -679,7 +679,7 @@ public class GuiScreenModInfo extends GuiScreen |
679 | 679 | { |
680 | 680 | if (this.selectedMod != null && this.selectedMod.getModClass() != null) |
681 | 681 | { |
682 | - ConfigPanel panel = this.configPanelManager.getPanel(this.selectedMod.getModClass()); | |
682 | + ConfigPanel panel = this.configManager.getPanel(this.selectedMod.getModClass()); | |
683 | 683 | if (panel != null) |
684 | 684 | { |
685 | 685 | if (this.configPanel != null) | ... | ... |
java/com/mumfrey/liteloader/modconfig/ConfigManager.java
0 → 100644
1 | +package com.mumfrey.liteloader.modconfig; | |
2 | + | |
3 | +import java.io.File; | |
4 | +import java.io.IOException; | |
5 | +import java.util.HashMap; | |
6 | +import java.util.LinkedList; | |
7 | +import java.util.List; | |
8 | +import java.util.Map; | |
9 | + | |
10 | +import com.google.common.base.Strings; | |
11 | +import com.google.common.io.Files; | |
12 | +import com.mumfrey.liteloader.Configurable; | |
13 | +import com.mumfrey.liteloader.LiteMod; | |
14 | +import com.mumfrey.liteloader.ExposeConfig; | |
15 | + | |
16 | +/** | |
17 | + * Registry where we keep the mod config panel classes | |
18 | + * | |
19 | + * @author Adam Mummery-Smith | |
20 | + */ | |
21 | +public class ConfigManager | |
22 | +{ | |
23 | + /** | |
24 | + * Mod config panel classes | |
25 | + */ | |
26 | + private Map<Class<? extends LiteMod>, Class<? extends ConfigPanel>> configPanels = new HashMap<Class<? extends LiteMod>, Class<? extends ConfigPanel>>(); | |
27 | + | |
28 | + /** | |
29 | + * Mod config writers | |
30 | + */ | |
31 | + private Map<Class<? extends Exposable>, ExposableConfigWriter> configWriters = new HashMap<Class<? extends Exposable>, ExposableConfigWriter>(); | |
32 | + | |
33 | + /** | |
34 | + * List of config writers, for faster iteration in onTick | |
35 | + */ | |
36 | + private List<ExposableConfigWriter> configWriterList = new LinkedList<ExposableConfigWriter>(); | |
37 | + | |
38 | + /** | |
39 | + * Register a mod, adds the config panel class to the map if the mod implements Configurable | |
40 | + */ | |
41 | + public void registerMod(LiteMod mod) | |
42 | + { | |
43 | + if (mod instanceof Configurable) | |
44 | + { | |
45 | + Class<? extends ConfigPanel> panelClass = ((Configurable)mod).getConfigPanelClass(); | |
46 | + if (panelClass != null) this.configPanels.put(mod.getClass(), panelClass); | |
47 | + } | |
48 | + | |
49 | + this.registerExposable(mod, null, false); | |
50 | + } | |
51 | + | |
52 | + /** | |
53 | + * @param exposable | |
54 | + */ | |
55 | + public void registerExposable(Exposable exposable, String fileName, boolean force) | |
56 | + { | |
57 | + ExposeConfig exposeConfig = exposable.getClass().<ExposeConfig>getAnnotation(ExposeConfig.class); | |
58 | + if (exposeConfig != null) | |
59 | + { | |
60 | + if (fileName == null) exposeConfig.filename(); | |
61 | + this.initConfigWriter(exposable, fileName, exposeConfig.strategy()); | |
62 | + } | |
63 | + else if (force) | |
64 | + { | |
65 | + this.initConfigWriter(exposable, fileName, ConfigStrategy.Versioned); | |
66 | + } | |
67 | + } | |
68 | + | |
69 | + /** | |
70 | + * Create a config writer instance for the specified mod | |
71 | + * | |
72 | + * @param exposable | |
73 | + * @param fileName | |
74 | + * @param strategy | |
75 | + */ | |
76 | + private void initConfigWriter(Exposable exposable, String fileName, ConfigStrategy strategy) | |
77 | + { | |
78 | + if (this.configWriters.containsKey(exposable.getClass())) | |
79 | + { | |
80 | + throw new IllegalArgumentException("Cannot register multiple Exposable instances of the same class or the Exposable already registered"); | |
81 | + } | |
82 | + | |
83 | + if (Strings.isNullOrEmpty(fileName)) | |
84 | + { | |
85 | + fileName = exposable.getClass().getSimpleName().toLowerCase(); | |
86 | + | |
87 | + if (fileName.startsWith("litemod")) | |
88 | + fileName = fileName.substring(7); | |
89 | + } | |
90 | + | |
91 | + ExposableConfigWriter configWriter = ExposableConfigWriter.create(exposable, strategy, fileName); | |
92 | + if (configWriter != null) | |
93 | + { | |
94 | + this.configWriters.put(exposable.getClass(), configWriter); | |
95 | + this.configWriterList.add(configWriter); | |
96 | + } | |
97 | + } | |
98 | + | |
99 | + /** | |
100 | + * Initialise the config writer for the specified mod | |
101 | + * | |
102 | + * @param exposable | |
103 | + */ | |
104 | + public void initConfig(Exposable exposable) | |
105 | + { | |
106 | + Class<? extends Exposable> exposableClass = exposable.getClass(); | |
107 | + if (exposableClass != null && this.configWriters.containsKey(exposableClass)) | |
108 | + { | |
109 | + this.configWriters.get(exposableClass).init(); | |
110 | + } | |
111 | + } | |
112 | + | |
113 | + /** | |
114 | + * If the specified mod has a versioned config strategy, attempt to copy the config | |
115 | + * | |
116 | + * @param mod | |
117 | + * @param newConfigPath | |
118 | + * @param oldConfigPath | |
119 | + */ | |
120 | + public void migrateModConfig(LiteMod mod, File newConfigPath, File oldConfigPath) | |
121 | + { | |
122 | + Class<? extends Exposable> exposableClass = mod.getClass(); | |
123 | + if (exposableClass != null && this.configWriters.containsKey(exposableClass)) | |
124 | + { | |
125 | + ExposableConfigWriter writer = this.configWriters.get(exposableClass); | |
126 | + if (writer.isVersioned()) | |
127 | + { | |
128 | + File newConfigFile = writer.getConfigFile(); | |
129 | + File legacyConfigFile = new File(oldConfigPath, newConfigFile.getName()); | |
130 | + | |
131 | + if (legacyConfigFile.exists() && !newConfigFile.exists()) | |
132 | + { | |
133 | + try | |
134 | + { | |
135 | + Files.copy(legacyConfigFile, newConfigFile); | |
136 | + } | |
137 | + catch (IOException ex) | |
138 | + { | |
139 | + ex.printStackTrace(); | |
140 | + } | |
141 | + } | |
142 | + } | |
143 | + } | |
144 | + } | |
145 | + | |
146 | + /** | |
147 | + * Check whether a config panel is available for the specified class | |
148 | + * | |
149 | + * @param modClass | |
150 | + * @return | |
151 | + */ | |
152 | + public boolean hasPanel(Class<? extends LiteMod> modClass) | |
153 | + { | |
154 | + return modClass != null && this.configPanels.containsKey(modClass); | |
155 | + } | |
156 | + | |
157 | + /** | |
158 | + * Instance a new config panel for the specified mod class if one is available | |
159 | + * | |
160 | + * @param modClass | |
161 | + * @return | |
162 | + */ | |
163 | + public ConfigPanel getPanel(Class<? extends LiteMod> modClass) | |
164 | + { | |
165 | + if (modClass != null && this.configPanels.containsKey(modClass)) | |
166 | + { | |
167 | + try | |
168 | + { | |
169 | + return this.configPanels.get(modClass).newInstance(); | |
170 | + } | |
171 | + catch (InstantiationException ex) {} | |
172 | + catch (IllegalAccessException ex) {} | |
173 | + | |
174 | + // If instantiation fails, remove the panel | |
175 | + this.configPanels.remove(modClass); | |
176 | + } | |
177 | + | |
178 | + return null; | |
179 | + } | |
180 | + | |
181 | + /** | |
182 | + * Invalidate the specified mod config, cause it to be written to disk or scheduled for writing | |
183 | + * if it has been written recent | |
184 | + * | |
185 | + * @param exposableClass | |
186 | + */ | |
187 | + public void invalidateConfig(Class<? extends Exposable> exposableClass) | |
188 | + { | |
189 | + if (exposableClass != null && this.configWriters.containsKey(exposableClass)) | |
190 | + { | |
191 | + this.configWriters.get(exposableClass).invalidate(); | |
192 | + } | |
193 | + } | |
194 | + | |
195 | + /** | |
196 | + * Tick all of the configuration writers, handles latent writes for anti-hammer strategy | |
197 | + */ | |
198 | + public void onTick() | |
199 | + { | |
200 | + for (ExposableConfigWriter writer : this.configWriterList) | |
201 | + { | |
202 | + writer.onTick(); | |
203 | + } | |
204 | + } | |
205 | + | |
206 | + /** | |
207 | + * Force all mod configs to be flushed to disk | |
208 | + */ | |
209 | + public void syncConfig() | |
210 | + { | |
211 | + for (ExposableConfigWriter writer : this.configWriterList) | |
212 | + { | |
213 | + writer.sync(); | |
214 | + } | |
215 | + } | |
216 | +} | ... | ... |
java/com/mumfrey/liteloader/modconfig/ConfigPanelManager.java deleted
100644 → 0
1 | -package com.mumfrey.liteloader.modconfig; | |
2 | - | |
3 | -import java.util.HashMap; | |
4 | -import java.util.Map; | |
5 | - | |
6 | -import com.mumfrey.liteloader.Configurable; | |
7 | -import com.mumfrey.liteloader.LiteMod; | |
8 | - | |
9 | -/** | |
10 | - * Registry where we keep the mod config panel classes | |
11 | - * | |
12 | - * @author Adam Mummery-Smith | |
13 | - */ | |
14 | -public class ConfigPanelManager | |
15 | -{ | |
16 | - /** | |
17 | - * Mod config panel classes | |
18 | - */ | |
19 | - private Map<Class<? extends LiteMod>, Class<? extends ConfigPanel>> configPanels = new HashMap<Class<? extends LiteMod>, Class<? extends ConfigPanel>>(); | |
20 | - | |
21 | - /** | |
22 | - * Register a mod, adds the config panel class to the map if the mod implements Configurable | |
23 | - */ | |
24 | - public void registerMod(LiteMod mod) | |
25 | - { | |
26 | - if (mod instanceof Configurable) | |
27 | - { | |
28 | - Class<? extends ConfigPanel> panelClass = ((Configurable)mod).getConfigPanelClass(); | |
29 | - if (panelClass != null) this.configPanels.put(mod.getClass(), panelClass); | |
30 | - } | |
31 | - } | |
32 | - | |
33 | - /** | |
34 | - * Check whether a config panel is available for the specified class | |
35 | - * | |
36 | - * @param modClass | |
37 | - * @return | |
38 | - */ | |
39 | - public boolean hasPanel(Class<? extends LiteMod> modClass) | |
40 | - { | |
41 | - return modClass != null && this.configPanels.containsKey(modClass); | |
42 | - } | |
43 | - | |
44 | - /** | |
45 | - * Instance a new config panel for the specified mod class if one is available | |
46 | - * | |
47 | - * @param modClass | |
48 | - * @return | |
49 | - */ | |
50 | - public ConfigPanel getPanel(Class<? extends LiteMod> modClass) | |
51 | - { | |
52 | - if (modClass != null && this.configPanels.containsKey(modClass)) | |
53 | - { | |
54 | - try | |
55 | - { | |
56 | - return this.configPanels.get(modClass).newInstance(); | |
57 | - } | |
58 | - catch (InstantiationException ex) {} | |
59 | - catch (IllegalAccessException ex) {} | |
60 | - | |
61 | - // If instantiation fails, remove the panel | |
62 | - this.configPanels.remove(modClass); | |
63 | - } | |
64 | - | |
65 | - return null; | |
66 | - } | |
67 | -} |
java/com/mumfrey/liteloader/modconfig/ConfigStrategy.java
0 → 100644
1 | +package com.mumfrey.liteloader.modconfig; | |
2 | + | |
3 | +import java.io.File; | |
4 | + | |
5 | +import com.mumfrey.liteloader.core.LiteLoader; | |
6 | + | |
7 | +/** | |
8 | + * Configuration management strategy | |
9 | + * | |
10 | + * @author Adam Mummery-Smith | |
11 | + */ | |
12 | +public enum ConfigStrategy | |
13 | +{ | |
14 | + /** | |
15 | + * Use the unversioned "common" config folder | |
16 | + */ | |
17 | + Unversioned, | |
18 | + | |
19 | + /** | |
20 | + * Use the versioned config folder | |
21 | + */ | |
22 | + Versioned; | |
23 | + | |
24 | + public File getFileForStrategy(String fileName) | |
25 | + { | |
26 | + if (this == ConfigStrategy.Versioned) | |
27 | + { | |
28 | + return new File(LiteLoader.getConfigFolder(), fileName); | |
29 | + } | |
30 | + | |
31 | + return new File(LiteLoader.getCommonConfigFolder(), fileName); | |
32 | + } | |
33 | +} | ... | ... |
java/com/mumfrey/liteloader/modconfig/Exposable.java
0 → 100644
java/com/mumfrey/liteloader/modconfig/ExposableConfigWriter.java
0 → 100644
1 | +package com.mumfrey.liteloader.modconfig; | |
2 | + | |
3 | +import java.io.File; | |
4 | +import java.io.FileReader; | |
5 | +import java.io.FileWriter; | |
6 | +import java.io.IOException; | |
7 | +import java.lang.reflect.Type; | |
8 | + | |
9 | +import com.google.gson.Gson; | |
10 | +import com.google.gson.GsonBuilder; | |
11 | +import com.google.gson.InstanceCreator; | |
12 | + | |
13 | +/** | |
14 | + * Manages serialisation of exposable properties to a JSON config file via Gson | |
15 | + * | |
16 | + * @author Adam Mummery-Smith | |
17 | + */ | |
18 | +public final class ExposableConfigWriter implements InstanceCreator<Exposable> | |
19 | +{ | |
20 | + /** | |
21 | + * Minimum number of milliseconds that must elapse between writes | |
22 | + */ | |
23 | + private static final long ANTI_HAMMER_DELAY = 1000L; | |
24 | + | |
25 | + /** | |
26 | + * Exposable instance which we will serialise exposed properties | |
27 | + */ | |
28 | + private final Exposable exposable; | |
29 | + | |
30 | + /** | |
31 | + * JSON file to write to | |
32 | + */ | |
33 | + private final File configFile; | |
34 | + | |
35 | + /** | |
36 | + * True if this is a versioned config strategy | |
37 | + */ | |
38 | + private final boolean versioned; | |
39 | + | |
40 | + /** | |
41 | + * Gson instance | |
42 | + */ | |
43 | + private final Gson gson; | |
44 | + | |
45 | + /** | |
46 | + * True if a config write has been requested but anti-hammer has prevented the write from occurring | |
47 | + */ | |
48 | + private volatile boolean dirty = false; | |
49 | + | |
50 | + /** | |
51 | + * Last time the config was written, used for anti-hammer | |
52 | + */ | |
53 | + private volatile long lastWrite = 0L; | |
54 | + | |
55 | + /** | |
56 | + * It's possible that writes may be requested from different threads, lock object to prevent cross-thread derp | |
57 | + */ | |
58 | + private Object readWriteLock = new Object(); | |
59 | + | |
60 | + /** | |
61 | + * @param exposable | |
62 | + * @param configFile | |
63 | + */ | |
64 | + private ExposableConfigWriter(Exposable exposable, File configFile, boolean versioned) | |
65 | + { | |
66 | + this.exposable = exposable; | |
67 | + this.configFile = configFile; | |
68 | + this.versioned = versioned; | |
69 | + | |
70 | + GsonBuilder gsonBuilder = new GsonBuilder(); | |
71 | + gsonBuilder.setPrettyPrinting(); | |
72 | + gsonBuilder.serializeNulls(); | |
73 | + gsonBuilder.excludeFieldsWithoutExposeAnnotation(); | |
74 | + gsonBuilder.registerTypeAdapter(exposable.getClass(), this); | |
75 | + | |
76 | + this.gson = gsonBuilder.create(); | |
77 | + } | |
78 | + | |
79 | + /** | |
80 | + * Get the config file underlying this writer | |
81 | + */ | |
82 | + File getConfigFile() | |
83 | + { | |
84 | + return this.configFile; | |
85 | + } | |
86 | + | |
87 | + /** | |
88 | + * Returns true if this writer is using a versioned strategy | |
89 | + */ | |
90 | + boolean isVersioned() | |
91 | + { | |
92 | + return this.versioned; | |
93 | + } | |
94 | + | |
95 | + /** | |
96 | + * Returns true if this writer has been invalidated but not yet been flushed to disk | |
97 | + */ | |
98 | + boolean isDirty() | |
99 | + { | |
100 | + return this.dirty; | |
101 | + } | |
102 | + | |
103 | + /* (non-Javadoc) | |
104 | + * @see com.google.gson.InstanceCreator#createInstance(java.lang.reflect.Type) | |
105 | + */ | |
106 | + @Override | |
107 | + public Exposable createInstance(Type type) | |
108 | + { | |
109 | + return this.exposable; | |
110 | + } | |
111 | + | |
112 | + /** | |
113 | + * Initialise the config, reads from file and writes the initial config file if not present | |
114 | + */ | |
115 | + void init() | |
116 | + { | |
117 | + // Read the config | |
118 | + this.read(); | |
119 | + | |
120 | + // If the config doesn't exist yet, seed the config | |
121 | + if (!this.configFile.exists()) | |
122 | + { | |
123 | + this.write(); | |
124 | + } | |
125 | + } | |
126 | + | |
127 | + /** | |
128 | + * Read the config from the file | |
129 | + */ | |
130 | + void read() | |
131 | + { | |
132 | + synchronized (this.readWriteLock) | |
133 | + { | |
134 | + if (this.configFile.exists()) | |
135 | + { | |
136 | + FileReader reader = null; | |
137 | + | |
138 | + try | |
139 | + { | |
140 | + reader = new FileReader(this.configFile); | |
141 | + | |
142 | + // Normally GSON would produce a new object by calling the default constructor, but we | |
143 | + // trick it into deserialising properties on the existing object instance by implementing | |
144 | + // an InstanceCreator which just returns the object instance which we already have | |
145 | + this.gson.fromJson(reader, this.exposable.getClass()); | |
146 | + } | |
147 | + catch (Exception ex) | |
148 | + { | |
149 | + ex.printStackTrace(); | |
150 | + } | |
151 | + finally | |
152 | + { | |
153 | + try | |
154 | + { | |
155 | + if (reader != null) | |
156 | + reader.close(); | |
157 | + } | |
158 | + catch (IOException ex) | |
159 | + { | |
160 | + ex.printStackTrace(); | |
161 | + } | |
162 | + } | |
163 | + } | |
164 | + } | |
165 | + } | |
166 | + | |
167 | + /** | |
168 | + * Write the config to the file | |
169 | + */ | |
170 | + void write() | |
171 | + { | |
172 | + synchronized (this.readWriteLock) | |
173 | + { | |
174 | + FileWriter writer = null; | |
175 | + try | |
176 | + { | |
177 | + writer = new FileWriter(this.configFile); | |
178 | + this.gson.toJson(this.exposable, writer); | |
179 | + | |
180 | + this.dirty = false; | |
181 | + this.lastWrite = System.currentTimeMillis(); | |
182 | + } | |
183 | + catch (Exception ex) | |
184 | + { | |
185 | + ex.printStackTrace(); | |
186 | + } | |
187 | + finally | |
188 | + { | |
189 | + try | |
190 | + { | |
191 | + if (writer != null) | |
192 | + writer.close(); | |
193 | + } | |
194 | + catch (IOException ex) | |
195 | + { | |
196 | + ex.printStackTrace(); | |
197 | + } | |
198 | + } | |
199 | + } | |
200 | + } | |
201 | + | |
202 | + /** | |
203 | + * Write the config to file, respecting anti-hammer and queuing the write if not | |
204 | + * enough time has elapsed | |
205 | + */ | |
206 | + void invalidate() | |
207 | + { | |
208 | + long sinceLastWrite = System.currentTimeMillis() - this.lastWrite; | |
209 | + if (sinceLastWrite < ANTI_HAMMER_DELAY) | |
210 | + { | |
211 | + this.dirty = true; | |
212 | + return; | |
213 | + } | |
214 | + | |
215 | + this.write(); | |
216 | + } | |
217 | + | |
218 | + /** | |
219 | + * Handle latent writes if the config was previously invalidated | |
220 | + */ | |
221 | + void onTick() | |
222 | + { | |
223 | + if (this.dirty) | |
224 | + { | |
225 | + long sinceLastWrite = System.currentTimeMillis() - this.lastWrite; | |
226 | + if (sinceLastWrite >= ANTI_HAMMER_DELAY) | |
227 | + { | |
228 | + this.write(); | |
229 | + } | |
230 | + } | |
231 | + } | |
232 | + | |
233 | + /** | |
234 | + * Force a write if dirty | |
235 | + */ | |
236 | + void sync() | |
237 | + { | |
238 | + if (this.dirty) | |
239 | + { | |
240 | + this.write(); | |
241 | + } | |
242 | + } | |
243 | + | |
244 | + /** | |
245 | + * Factory method which creates and intialises a new ExposableConfigWriter for the specified exposable object and strategy | |
246 | + * | |
247 | + * @param exposable | |
248 | + * @param strategy | |
249 | + * @param fileName | |
250 | + * @return | |
251 | + */ | |
252 | + static ExposableConfigWriter create(Exposable exposable, ConfigStrategy strategy, String fileName) | |
253 | + { | |
254 | + if (!fileName.toLowerCase().endsWith(".json")) | |
255 | + fileName = fileName + ".json"; | |
256 | + | |
257 | + File configFile = strategy.getFileForStrategy(fileName); | |
258 | + ExposableConfigWriter writer = new ExposableConfigWriter(exposable, configFile, strategy == ConfigStrategy.Versioned); | |
259 | + | |
260 | + return writer; | |
261 | + } | |
262 | +} | |
0 | 263 | \ No newline at end of file | ... | ... |
java/com/mumfrey/liteloader/util/PrivateFields.java
... | ... | @@ -144,6 +144,7 @@ public class PrivateFields<P, T> |
144 | 144 | public static final PrivateFields<Minecraft, Timer> minecraftTimer = new PrivateFields<Minecraft, Timer> (Minecraft.class, "timer", "S", "field_71428_T"); // Minecraft/timer |
145 | 145 | public static final PrivateFields<Minecraft, Profiler> minecraftProfiler = new PrivateFields<Minecraft, Profiler> (Minecraft.class, "mcProfiler", "C", "field_71424_I"); // Minecraft/mcProfiler |
146 | 146 | public static final PrivateFields<Minecraft, List<ResourcePack>> defaultResourcePacks = new PrivateFields<Minecraft, List<ResourcePack>> (Minecraft.class, "defaultResourcePacks", "aq", "field_110449_ao"); // Minecraft/defaultResourcePacks |
147 | + public static final PrivateFields<Minecraft, Boolean> gameIsRunning = new PrivateFields<Minecraft, Boolean> (Minecraft.class, "running", "D", "field_71425_J"); // Minecraft/running | |
147 | 148 | public static final PrivateFields<RenderManager, Map> entityRenderMap = new PrivateFields<RenderManager, Map> (RenderManager.class, "entityRenderMap", "q", "field_78729_o"); // RenderManager/entityRenderMap |
148 | 149 | public static final PrivateFields<GuiControls, GuiScreen> guiControlsParentScreen = new PrivateFields<GuiControls, GuiScreen> (GuiControls.class, "parentScreen", "b", "field_73909_b"); // GuiControls/parentScreen |
149 | 150 | public static final PrivateFields<PlayerUsageSnooper, IPlayerUsage> playerStatsCollector = new PrivateFields<PlayerUsageSnooper, IPlayerUsage>(PlayerUsageSnooper.class, "playerStatsCollector", "d", "field_76478_d"); // PlayerUsageSnooper/playerStatsCollector | ... | ... |