Add concept of pushing/poping levels to ForgeConfigSpec as well as comments on those levels.

This commit is contained in:
LexManos 2018-09-18 12:19:19 -07:00
parent 86d10d355f
commit b25f92b0c0
2 changed files with 125 additions and 79 deletions

View file

@ -22,6 +22,7 @@ package net.minecraftforge.common;
import static net.minecraftforge.fml.Logging.CORE; import static net.minecraftforge.fml.Logging.CORE;
import java.nio.file.Path; import java.nio.file.Path;
import java.nio.file.Paths;
import org.apache.logging.log4j.LogManager; import org.apache.logging.log4j.LogManager;
@ -29,84 +30,91 @@ import com.electronwill.nightconfig.core.file.CommentedFileConfig;
import com.electronwill.nightconfig.core.io.WritingMode; import com.electronwill.nightconfig.core.io.WritingMode;
import net.minecraftforge.common.config.ForgeConfigSpec; import net.minecraftforge.common.config.ForgeConfigSpec;
import net.minecraftforge.fml.common.FMLPaths;
public class ForgeConfig public class ForgeConfig
{ {
private static ForgeConfig INSTANCE = new ForgeConfig(); private static ForgeConfig INSTANCE = new ForgeConfig();
private static ForgeConfigSpec spec = new ForgeConfigSpec.Builder() private static ForgeConfigSpec spec = new ForgeConfigSpec.Builder()
//General //General
.comment("Set to true to disable Forge's version check mechanics. Forge queries a small json file on our server for version information. For more details see the ForgeVersion class in our github.") .comment("General settings that effect both the client and server")
.translation("forge.configgui.disableVersionCheck") .push("general")
.define("general.disableVersionCheck", false) .comment("Set to true to disable Forge's version check mechanics. Forge queries a small json file on our server for version information. For more details see the ForgeVersion class in our github.")
.translation("forge.configgui.disableVersionCheck")
.define("disableVersionCheck", false)
.comment("Set this to true to remove any Entity that throws an error in its update method instead of closing the server and reporting a crash log. BE WARNED THIS COULD SCREW UP EVERYTHING USE SPARINGLY WE ARE NOT RESPONSIBLE FOR DAMAGES.") .comment("Set this to true to remove any Entity that throws an error in its update method instead of closing the server and reporting a crash log. BE WARNED THIS COULD SCREW UP EVERYTHING USE SPARINGLY WE ARE NOT RESPONSIBLE FOR DAMAGES.")
.translation("forge.configgui.removeErroringEntities") .translation("forge.configgui.removeErroringEntities")
.worldRestart() .worldRestart()
.define("general.removeErroringEntities", false) .define("removeErroringEntities", false)
.comment("Set this to true to remove any TileEntity that throws an error in its update method instead of closing the server and reporting a crash log. BE WARNED THIS COULD SCREW UP EVERYTHING USE SPARINGLY WE ARE NOT RESPONSIBLE FOR DAMAGES.") .comment("Set this to true to remove any TileEntity that throws an error in its update method instead of closing the server and reporting a crash log. BE WARNED THIS COULD SCREW UP EVERYTHING USE SPARINGLY WE ARE NOT RESPONSIBLE FOR DAMAGES.")
.translation("forge.configgui.removeErroringTileEntities") .translation("forge.configgui.removeErroringTileEntities")
.worldRestart() .worldRestart()
.define("general.removeErroringTileEntities", false) .define("removeErroringTileEntities", false)
.comment("Set this to true to check the entire entity's collision bounding box for ladders instead of just the block they are in. Causes noticeable differences in mechanics so default is vanilla behavior. Default: false") .comment("Set this to true to check the entire entity's collision bounding box for ladders instead of just the block they are in. Causes noticeable differences in mechanics so default is vanilla behavior. Default: false")
.translation("forge.configgui.fullBoundingBoxLadders") .translation("forge.configgui.fullBoundingBoxLadders")
.worldRestart() .worldRestart()
.define("general.fullBoundingBoxLadders", false) .define("fullBoundingBoxLadders", false)
.comment("Base zombie summoning spawn chance. Allows changing the bonus zombie summoning mechanic.") .comment("Base zombie summoning spawn chance. Allows changing the bonus zombie summoning mechanic.")
.translation("forge.configgui.zombieBaseSummonChance") .translation("forge.configgui.zombieBaseSummonChance")
.worldRestart() .worldRestart()
.defineInRange("general.zombieBaseSummonChance", 0.1D, 0.0D, 1.0D) .defineInRange("zombieBaseSummonChance", 0.1D, 0.0D, 1.0D)
.comment("Chance that a zombie (or subclass) is a baby. Allows changing the zombie spawning mechanic.") .comment("Chance that a zombie (or subclass) is a baby. Allows changing the zombie spawning mechanic.")
.translation("forge.configgui.zombieBabyChance") .translation("forge.configgui.zombieBabyChance")
.worldRestart() .worldRestart()
.defineInRange("general.zombieBabyChance", 0.05D, 0.0D, 1.0D) .defineInRange("zombieBabyChance", 0.05D, 0.0D, 1.0D)
.comment("Log cascading chunk generation issues during terrain population.") .comment("Log cascading chunk generation issues during terrain population.")
.translation("forge.configgui.logCascadingWorldGeneration") .translation("forge.configgui.logCascadingWorldGeneration")
.define("general.logCascadingWorldGeneration", true) .define("logCascadingWorldGeneration", true)
.comment("Fix vanilla issues that cause worldgen cascading. This DOES change vanilla worldgen so DO NOT report bugs related to world differences if this flag is on.") .comment("Fix vanilla issues that cause worldgen cascading. This DOES change vanilla worldgen so DO NOT report bugs related to world differences if this flag is on.")
.translation("forge.configgui.fixVanillaCascading") .translation("forge.configgui.fixVanillaCascading")
.define("general.fixVanillaCascading", false) .define("fixVanillaCascading", false)
.comment("The time in ticks the server will wait when a dimension was queued to unload. This can be useful when rapidly loading and unloading dimensions, like e.g. throwing items through a nether portal a few time per second.") .comment("The time in ticks the server will wait when a dimension was queued to unload. This can be useful when rapidly loading and unloading dimensions, like e.g. throwing items through a nether portal a few time per second.")
.translation("forge.configgui.dimensionUnloadQueueDelay") .translation("forge.configgui.dimensionUnloadQueueDelay")
.defineInRange("general.dimensionUnloadQueueDelay", 0, 0, Integer.MAX_VALUE) .defineInRange("dimensionUnloadQueueDelay", 0, 0, Integer.MAX_VALUE)
.pop()
//Client //Client
.comment("Controls the number threshold at which Packet51 is preferred over Packet52, default and minimum 64, maximum 1024") .comment("Client only settings, mostly things related to rendering")
.translation("forge.configgui.clumpingThreshold") .push("client")
.worldRestart() .comment("Controls the number threshold at which Packet51 is preferred over Packet52, default and minimum 64, maximum 1024")
.defineInRange("client.clumpingThreshold", 64, 64, 1024) .translation("forge.configgui.clumpingThreshold")
.worldRestart()
.defineInRange("clumpingThreshold", 64, 64, 1024)
.comment("Toggle off to make missing model text in the gui fit inside the slot.") .comment("Toggle off to make missing model text in the gui fit inside the slot.")
.translation("forge.configgui.zoomInMissingModelTextInGui") .translation("forge.configgui.zoomInMissingModelTextInGui")
.define("client.zoomInMissingModelTextInGui", false) .define("zoomInMissingModelTextInGui", false)
.comment("Enable uploading cloud geometry to the GPU for faster rendering.") .comment("Enable uploading cloud geometry to the GPU for faster rendering.")
.translation("forge.configgui.forgeCloudsEnabled") .translation("forge.configgui.forgeCloudsEnabled")
.define("client.forgeCloudsEnabled", true) .define("forgeCloudsEnabled", true)
.comment("Disable culling of hidden faces next to stairs and slabs. Causes extra rendering, but may fix some resource packs that exploit this vanilla mechanic.") .comment("Disable culling of hidden faces next to stairs and slabs. Causes extra rendering, but may fix some resource packs that exploit this vanilla mechanic.")
.translation("forge.configgui.disableStairSlabCulling") .translation("forge.configgui.disableStairSlabCulling")
.define("client.disableStairSlabCulling", false) .define("disableStairSlabCulling", false)
.comment("Enable forge to queue all chunk updates to the Chunk Update thread. May increase FPS significantly, but may also cause weird rendering lag. Not recommended for computers without a significant number of cores available.") .comment("Enable forge to queue all chunk updates to the Chunk Update thread.",
.translation("forge.configgui.alwaysSetupTerrainOffThread") "May increase FPS significantly, but may also cause weird rendering lag.",
.define("client.alwaysSetupTerrainOffThread", false) "Not recommended for computers without a significant number of cores available.")
.translation("forge.configgui.alwaysSetupTerrainOffThread")
.define("alwaysSetupTerrainOffThread", false)
.comment("Enable the forge block rendering pipeline - fixes the lighting of custom models.") .comment("Enable the forge block rendering pipeline - fixes the lighting of custom models.")
.translation("forge.configgui.forgeLightPipelineEnabled") .translation("forge.configgui.forgeLightPipelineEnabled")
.define("client.forgeLightPipelineEnabled", true) .define("forgeLightPipelineEnabled", true)
.comment("When enabled, makes specific reload tasks such as language changing quicker to run.") .comment("When enabled, makes specific reload tasks such as language changing quicker to run.")
.translation("forge.configgui.selectiveResourceReloadEnabled") .translation("forge.configgui.selectiveResourceReloadEnabled")
.define("client.selectiveResourceReloadEnabled", true) .define("selectiveResourceReloadEnabled", true)
.pop()
.build(); .build();
@ -114,22 +122,23 @@ public class ForgeConfig
private CommentedFileConfig configData; private CommentedFileConfig configData;
private void loadFrom(final Path configFile) private void loadFrom(final Path configFile)
{ {
configData = CommentedFileConfig.builder(configFile).sync(). configData = CommentedFileConfig.builder(configFile).sync()
autosave().autoreload(). .autosave()
writingMode(WritingMode.REPLACE). //.autoreload()
build(); .writingMode(WritingMode.REPLACE)
.build();
configData.load(); configData.load();
if (!spec.isCorrect(configData)) { if (!spec.isCorrect(configData)) {
LogManager.getLogger().warn(CORE, "Configuration file {} is not correct. Correcting", configFile); LogManager.getLogger().warn(CORE, "Configuration file {} is not correct. Correcting", configFile);
spec.correct(configData, (action, path, incorrectValue, correctedValue) -> spec.correct(configData, (action, path, incorrectValue, correctedValue) ->
LogManager.getLogger().warn(CORE, "Incorrect key {} was corrected from {} to {}", path, incorrectValue, correctedValue)); LogManager.getLogger().warn(CORE, "Incorrect key {} was corrected from {} to {}", path, incorrectValue, correctedValue));
configData.save();
} }
configData.save();
} }
public static void load() public static void load()
{ {
Path configFile = FMLPaths.CONFIGDIR.get().resolve("forge.toml"); Path configFile = Paths.get("config").resolve("forge.toml");
INSTANCE.loadFrom(configFile); INSTANCE.loadFrom(configFile);
LogManager.getLogger().debug(CORE, "Loaded FML config from {}", configFile); LogManager.getLogger().debug(CORE, "Loaded FML config from {}", configFile);
} }
@ -195,20 +204,6 @@ public class ForgeConfig
} }
} }
} }
@SubscribeEvent
public void missingMapping(RegistryEvent.MissingMappings<Item> event)
{
for (MissingMappings.Mapping<Item> entry : event.getAllMappings())
{
if (entry.key.toString().equals("minecraft:totem")) //This item changed from 1.11 -> 1.11.2
{
ResourceLocation newTotem = new ResourceLocation("minecraft:totem_of_undying");
entry.remap(ForgeRegistries.ITEMS.getValue(newTotem));
}
}
}
@net.minecraftforge.eventbus.api.SubscribeEvent @net.minecraftforge.eventbus.api.SubscribeEvent
public void playerLogin(PlayerEvent.PlayerLoggedInEvent event) public void playerLogin(PlayerEvent.PlayerLoggedInEvent event)
{ {

View file

@ -22,9 +22,11 @@ import static com.electronwill.nightconfig.core.ConfigSpec.CorrectionAction.ADD;
import static com.electronwill.nightconfig.core.ConfigSpec.CorrectionAction.REMOVE; import static com.electronwill.nightconfig.core.ConfigSpec.CorrectionAction.REMOVE;
import static com.electronwill.nightconfig.core.ConfigSpec.CorrectionAction.REPLACE; import static com.electronwill.nightconfig.core.ConfigSpec.CorrectionAction.REPLACE;
import java.util.ArrayList;
import java.util.Arrays; import java.util.Arrays;
import java.util.Collection; import java.util.Collection;
import java.util.Collections; import java.util.Collections;
import java.util.HashMap;
import java.util.Iterator; import java.util.Iterator;
import java.util.LinkedList; import java.util.LinkedList;
import java.util.List; import java.util.List;
@ -50,9 +52,10 @@ import com.google.common.collect.Lists;
public class ForgeConfigSpec public class ForgeConfigSpec
{ {
private final Config storage; private final Config storage;
private ForgeConfigSpec(Config storage) private Map<List<String>, String> levelComments = new HashMap<>();
{ private ForgeConfigSpec(Config storage, Map<List<String>, String> levelComments) {
this.storage = storage; this.storage = storage;
this.levelComments = levelComments;
} }
public boolean isCorrect(CommentedConfig config) { public boolean isCorrect(CommentedConfig config) {
@ -102,6 +105,18 @@ public class ForgeConfigSpec
configMap.put(key, newValue); configMap.put(key, newValue);
listener.onCorrect(action, parentPathUnmodifiable, configValue, newValue); listener.onCorrect(action, parentPathUnmodifiable, configValue, newValue);
count++; count++;
count += correct((Config)specValue, newValue, parentPath, parentPathUnmodifiable, listener, dryRun);
}
String newComment = levelComments.get(parentPath);
String oldComment = config.getComment(key);
if (!Objects.equals(oldComment, newComment))
{
if (dryRun)
return 1;
//TODO: Comment correction listener?
config.setComment(key, newComment);
} }
} }
else else
@ -154,6 +169,8 @@ public class ForgeConfigSpec
{ {
private final Config storage = InMemoryFormat.withUniversalSupport().createConfig(); private final Config storage = InMemoryFormat.withUniversalSupport().createConfig();
private BuilderContext context = new BuilderContext(); private BuilderContext context = new BuilderContext();
private Map<List<String>, String> levelComments = new HashMap<>();
private List<String> currentPath = new ArrayList<>();
//Object //Object
public Builder define(String path, Object defaultValue) { public Builder define(String path, Object defaultValue) {
@ -176,6 +193,12 @@ public class ForgeConfigSpec
return define(path, defaultSupplier, validator, Object.class); return define(path, defaultSupplier, validator, Object.class);
} }
public Builder define(List<String> path, Supplier<?> defaultSupplier, Predicate<Object> validator, Class<?> clazz) { // This is the root where everything at the end of the day ends up. public Builder define(List<String> path, Supplier<?> defaultSupplier, Predicate<Object> validator, Class<?> clazz) { // This is the root where everything at the end of the day ends up.
if (!currentPath.isEmpty()) {
List<String> tmp = new ArrayList<>(currentPath.size() + path.size());
tmp.addAll(currentPath);
tmp.addAll(path);
path = tmp;
}
context.setClazz(clazz); context.setClazz(clazz);
storage.set(path, new ValueSpec(defaultSupplier, validator, context)); storage.set(path, new ValueSpec(defaultSupplier, validator, context));
context = new BuilderContext(); context = new BuilderContext();
@ -325,10 +348,37 @@ public class ForgeConfigSpec
return this; return this;
} }
public Builder push(String path) {
return push(split(path));
}
public Builder push(List<String> path) {
currentPath.addAll(path);
if (context.getComment() != null) {
levelComments.put(new ArrayList<String>(currentPath), LINE_JOINER.join(context.getComment()));
context.setComment((String[])null);
}
context.ensureEmpty();
return this;
}
public Builder pop() {
return pop(1);
}
public Builder pop(int count) {
if (count > currentPath.size())
throw new IllegalArgumentException("Attempted to pop " + count + " elements when we only had: " + currentPath);
for (int x = 0; x < count; x++)
currentPath.remove(currentPath.size() - 1);
return this;
}
public ForgeConfigSpec build() public ForgeConfigSpec build()
{ {
context.ensureEmpty(); context.ensureEmpty();
return new ForgeConfigSpec(storage); return new ForgeConfigSpec(storage, levelComments);
} }
} }
@ -419,7 +469,7 @@ public class ForgeConfigSpec
Objects.requireNonNull(supplier, "Default supplier can not be null"); Objects.requireNonNull(supplier, "Default supplier can not be null");
Objects.requireNonNull(validator, "Validator can not be null"); Objects.requireNonNull(validator, "Validator can not be null");
this.comment = Joiner.on("\n").join(context.getComment()); this.comment = LINE_JOINER.join(context.getComment());
this.langKey = context.getTranslationKey(); this.langKey = context.getTranslationKey();
this.range = context.getRange(); this.range = context.getRange();
this.worldRestart = context.needsWorldRestart(); this.worldRestart = context.needsWorldRestart();
@ -444,6 +494,7 @@ public class ForgeConfigSpec
} }
} }
private static final Joiner LINE_JOINER = Joiner.on("\n");
private static final Splitter DOT_SPLITTER = Splitter.on("."); private static final Splitter DOT_SPLITTER = Splitter.on(".");
private static List<String> split(String path) private static List<String> split(String path)
{ {