From 8320b2428deec4d7e06a9ef3943505441f12a1fa Mon Sep 17 00:00:00 2001 From: Technici4n <13494793+Technici4n@users.noreply.github.com> Date: Tue, 23 Jan 2024 14:38:01 +0100 Subject: [PATCH] Simplify extension point system --- .../net/neoforged/fml/IExtensionPoint.java | 15 ++-------- .../java/net/neoforged/fml/ModContainer.java | 30 ++++++++++++------- .../net/neoforged/fml/ModLoadingContext.java | 2 +- 3 files changed, 24 insertions(+), 23 deletions(-) diff --git a/loader/src/main/java/net/neoforged/fml/IExtensionPoint.java b/loader/src/main/java/net/neoforged/fml/IExtensionPoint.java index 2665f40eb..c69620ff9 100644 --- a/loader/src/main/java/net/neoforged/fml/IExtensionPoint.java +++ b/loader/src/main/java/net/neoforged/fml/IExtensionPoint.java @@ -13,19 +13,10 @@ * *

An extension point can be registered for a mod container using {@link ModContainer#registerExtensionPoint(Class, Supplier)} * and retrieved (if present) using {@link ModContainer#getCustomExtension(Class)}. An extension point allows a mod to - * supply an arbitrary value as a record class to another mod or framework through their mod container class, avoiding + * supply an arbitrary value to another mod or framework through their mod container class, avoiding * the use of {@link InterModComms} or other external frameworks to pass around these values.

- * - *

The usual way to declare an extension point is to implement this interface on a record class, with the type - * parameter being a reference to the class itself. For example, {@code record MyExtension(...) extends - * IExtensionPoint} would declare an extension point which supplies a {@code MyExtension} object. However, - * there is no hard requirement that an extension point's type parameter must be in reference to itself; the type - * parameter may reference another record class instead.

- * - * @param the type of the record which is held by the extension point */ -@SuppressWarnings("unused") // Type parameter T -public interface IExtensionPoint +public interface IExtensionPoint { /** * Extension point for the compatibility display test used on the server selection screen. @@ -93,7 +84,7 @@ public interface IExtensionPoint * @see net.neoforged.client.ForgeHooksClient#processForgeListPingData(net.minecraft.network.protocol.status.ServerStatus, net.minecraft.client.multiplayer.ServerData) */ @SuppressWarnings("JavadocReference") // reference to NetworkConstants, ForgeHooksClient - record DisplayTest(Supplier suppliedVersion, BiPredicate remoteVersionTest) implements IExtensionPoint { + record DisplayTest(Supplier suppliedVersion, BiPredicate remoteVersionTest) implements IExtensionPoint { public static final String IGNORESERVERONLY = "OHNOES\uD83D\uDE31\uD83D\uDE31\uD83D\uDE31\uD83D\uDE31\uD83D\uDE31\uD83D\uDE31\uD83D\uDE31\uD83D\uDE31\uD83D\uDE31\uD83D\uDE31\uD83D\uDE31\uD83D\uDE31\uD83D\uDE31\uD83D\uDE31\uD83D\uDE31\uD83D\uDE31\uD83D\uDE31"; } } diff --git a/loader/src/main/java/net/neoforged/fml/ModContainer.java b/loader/src/main/java/net/neoforged/fml/ModContainer.java index fddbde09f..9e3b6b188 100644 --- a/loader/src/main/java/net/neoforged/fml/ModContainer.java +++ b/loader/src/main/java/net/neoforged/fml/ModContainer.java @@ -53,7 +53,7 @@ public abstract class ModContainer protected ModLoadingStage modLoadingStage; protected Supplier contextExtension; protected final Map activityMap = new HashMap<>(); - protected final Map>, Supplier> extensionPoints = new IdentityHashMap<>(); + protected final Map, Supplier> extensionPoints = new IdentityHashMap<>(); protected final EnumMap configs = new EnumMap<>(ModConfig.Type.class); public ModContainer(IModInfo info) @@ -65,20 +65,20 @@ public ModContainer(IModInfo info) this.modLoadingStage = ModLoadingStage.CONSTRUCT; final String displayTestString = info.getConfig().getConfigElement("displayTest").orElse("MATCH_VERSION"); // missing defaults to DEFAULT type - Supplier displayTestSupplier = switch (displayTestString) { + var displayTest = switch (displayTestString) { case "MATCH_VERSION" -> // default displaytest checks for version string match - () -> new IExtensionPoint.DisplayTest(() -> this.modInfo.getVersion().toString(), + new IExtensionPoint.DisplayTest(() -> this.modInfo.getVersion().toString(), (incoming, isNetwork) -> Objects.equals(incoming, this.modInfo.getVersion().toString())); case "IGNORE_SERVER_VERSION" -> // Ignores any version information coming from the server - use for server only mods - () -> new IExtensionPoint.DisplayTest(() -> IExtensionPoint.DisplayTest.IGNORESERVERONLY, (incoming, isNetwork) -> true); + new IExtensionPoint.DisplayTest(() -> IExtensionPoint.DisplayTest.IGNORESERVERONLY, (incoming, isNetwork) -> true); case "IGNORE_ALL_VERSION" -> // Ignores all information and provides no information - () -> new IExtensionPoint.DisplayTest(() -> "", (incoming, isNetwork) -> true); + new IExtensionPoint.DisplayTest(() -> "", (incoming, isNetwork) -> true); case "NONE" -> null; // NO display test at all - use this if you're going to do your own display test default -> // any other value throws an exception throw new IllegalArgumentException("Invalid displayTest value supplied in mods.toml"); }; - if (displayTestSupplier != null) - registerExtensionPoint(IExtensionPoint.DisplayTest.class, displayTestSupplier); + if (displayTest != null) + registerExtensionPoint(IExtensionPoint.DisplayTest.class, displayTest); else extensionPoints.remove(IExtensionPoint.DisplayTest.class); } @@ -142,12 +142,22 @@ public IModInfo getModInfo() } @SuppressWarnings("unchecked") - public Optional getCustomExtension(Class> point) { + public Optional getCustomExtension(Class point) { return Optional.ofNullable((T)extensionPoints.getOrDefault(point,()-> null).get()); } - public > void registerExtensionPoint(Class> point, Supplier extension) - { + /** + * Registers an {@link IExtensionPoint} with the mod container. + */ + public void registerExtensionPoint(Class point, T extension) { + extensionPoints.put(point, () -> extension); + } + + /** + * Registers an {@link IExtensionPoint} with the mod container. + * This overload allows passing a supplier that will only be evaluated when the extension is requested. + */ + public void registerExtensionPoint(Class point, Supplier extension) { extensionPoints.put(point, extension); } diff --git a/loader/src/main/java/net/neoforged/fml/ModLoadingContext.java b/loader/src/main/java/net/neoforged/fml/ModLoadingContext.java index ab653ba22..fa45c679e 100644 --- a/loader/src/main/java/net/neoforged/fml/ModLoadingContext.java +++ b/loader/src/main/java/net/neoforged/fml/ModLoadingContext.java @@ -44,7 +44,7 @@ public String getActiveNamespace() { * @param extension An extension operator * @param The type signature of the extension operator */ - public > void registerExtensionPoint(Class> point, Supplier extension) { + public void registerExtensionPoint(Class point, Supplier extension) { getActiveContainer().registerExtensionPoint(point, extension); }