diff --git a/core/src/main/java/net/neoforged/fml/ModContainer.java b/core/src/main/java/net/neoforged/fml/ModContainer.java
index 0b3312b06..dfaef7edf 100644
--- a/core/src/main/java/net/neoforged/fml/ModContainer.java
+++ b/core/src/main/java/net/neoforged/fml/ModContainer.java
@@ -5,12 +5,18 @@
package net.neoforged.fml;
+import net.neoforged.bus.api.BusBuilder;
import net.neoforged.bus.api.Event;
+import net.neoforged.bus.api.EventPriority;
+import net.neoforged.bus.api.IEventBus;
import net.neoforged.fml.config.IConfigEvent;
import net.neoforged.fml.config.ModConfig;
import net.neoforged.fml.event.IModBusEvent;
import net.neoforged.fml.loading.progress.ProgressMeter;
import net.neoforged.neoforgespi.language.IModInfo;
+import org.apache.logging.log4j.LogManager;
+import org.apache.logging.log4j.Logger;
+import org.jetbrains.annotations.Nullable;
import java.util.EnumMap;
import java.util.HashMap;
@@ -24,6 +30,8 @@
import java.util.function.Consumer;
import java.util.function.Supplier;
+import static net.neoforged.fml.Logging.LOADING;
+
/**
* The container that wraps around mods in the system.
*
@@ -39,6 +47,8 @@
public abstract class ModContainer
{
+ private static final Logger LOGGER = LogManager.getLogger();
+
protected final String modId;
protected final String namespace;
protected final IModInfo modInfo;
@@ -167,8 +177,49 @@ public void dispatchConfigEvent(IConfigEvent event) {
public abstract Object getMod();
/**
- * Accept an arbitrary event for processing by the mod. Probably posted to an event bus in the lower level container.
+ * {@return the event bus for this mod, if available}
+ *
+ *
Not all mods have an event bus!
+ *
+ * @implNote For custom mod container implementations, the event bus must be built with
+ * {@link BusBuilder#allowPerPhasePost()} or posting via {@link #acceptEvent(EventPriority, Event)} will throw!
+ */
+ @Nullable
+ public abstract IEventBus getEventBus();
+
+ /**
+ * Accept an arbitrary event for processing by the mod. Posted to {@link #getEventBus()}.
* @param e Event to accept
*/
- protected void acceptEvent(T e) {}
+ protected final void acceptEvent(T e) {
+ IEventBus bus = getEventBus();
+ if (bus == null) return;
+
+ try {
+ LOGGER.trace(LOADING, "Firing event for modid {} : {}", this.getModId(), e);
+ bus.post(e);
+ LOGGER.trace(LOADING, "Fired event for modid {} : {}", this.getModId(), e);
+ } catch (Throwable t) {
+ LOGGER.error(LOADING,"Caught exception during event {} dispatch for modid {}", e, this.getModId(), t);
+ throw new ModLoadingException(modInfo, modLoadingStage, "fml.modloading.errorduringevent", t);
+ }
+ }
+
+ /**
+ * Accept an arbitrary event for processing by the mod. Posted to {@link #getEventBus()}.
+ * @param e Event to accept
+ */
+ protected final void acceptEvent(EventPriority phase, T e) {
+ IEventBus bus = getEventBus();
+ if (bus == null) return;
+
+ try {
+ LOGGER.trace(LOADING, "Firing event for phase {} for modid {} : {}", phase, this.getModId(), e);
+ bus.post(phase, e);
+ LOGGER.trace(LOADING, "Fired event for phase {} for modid {} : {}", phase, this.getModId(), e);
+ } catch (Throwable t) {
+ LOGGER.error(LOADING,"Caught exception during event {} dispatch for modid {}", e, this.getModId(), t);
+ throw new ModLoadingException(modInfo, modLoadingStage, "fml.modloading.errorduringevent", t);
+ }
+ }
}
diff --git a/core/src/main/java/net/neoforged/fml/ModList.java b/core/src/main/java/net/neoforged/fml/ModList.java
index c5c36dcfb..e2d416fc5 100644
--- a/core/src/main/java/net/neoforged/fml/ModList.java
+++ b/core/src/main/java/net/neoforged/fml/ModList.java
@@ -223,6 +223,10 @@ public void forEachModContainer(BiConsumer modContainerCon
indexedMods.forEach(modContainerConsumer);
}
+ public List getSortedMods() {
+ return sortedContainers;
+ }
+
public void forEachModInOrder(Consumer containerConsumer) {
this.sortedContainers.forEach(containerConsumer);
}
diff --git a/core/src/main/java/net/neoforged/fml/ModLoader.java b/core/src/main/java/net/neoforged/fml/ModLoader.java
index b4808b78f..07e20ef15 100644
--- a/core/src/main/java/net/neoforged/fml/ModLoader.java
+++ b/core/src/main/java/net/neoforged/fml/ModLoader.java
@@ -7,6 +7,8 @@
import com.google.common.collect.ImmutableList;
import net.neoforged.bus.api.Event;
+import net.neoforged.bus.api.EventPriority;
+import net.neoforged.bus.api.IEventBus;
import net.neoforged.fml.event.IModBusEvent;
import net.neoforged.fml.loading.FMLEnvironment;
import net.neoforged.fml.loading.FMLLoader;
@@ -23,6 +25,7 @@
import net.neoforged.neoforgespi.locating.IModFile;
import org.apache.logging.log4j.LogManager;
import org.apache.logging.log4j.Logger;
+import org.jetbrains.annotations.Nullable;
import java.util.*;
import java.util.concurrent.CompletableFuture;
@@ -315,7 +318,18 @@ public void runEventGenerator(Function mc.acceptEvent(generator.apply(mc)));
+
+ // Construct events
+ List modContainers = ModList.get().getSortedMods();
+ List events = new ArrayList<>(modContainers.size());
+ ModList.get().forEachModInOrder(mc -> events.add(generator.apply(mc)));
+
+ // Post them
+ for (EventPriority phase : EventPriority.values()) {
+ for (int i = 0; i < modContainers.size(); i++) {
+ modContainers.get(i).acceptEvent(phase, events.get(i));
+ }
+ }
}
public void postEvent(T e) {
@@ -323,14 +337,12 @@ public void postEvent(T e) {
LOGGER.error("Cowardly refusing to send event {} to a broken mod state", e.getClass().getName());
return;
}
- ModList.get().forEachModInOrder(mc -> mc.acceptEvent(e));
+ for (EventPriority phase : EventPriority.values()) {
+ ModList.get().forEachModInOrder(mc -> mc.acceptEvent(phase, e));
+ }
}
public T postEventWithReturn(T e) {
- if (!loadingStateValid) {
- LOGGER.error("Cowardly refusing to send event {} to a broken mod state", e.getClass().getName());
- return e;
- }
- ModList.get().forEachModInOrder(mc -> mc.acceptEvent(e));
+ postEvent(e);
return e;
}
public void postEventWrapContainerInModOrder(T event) {
@@ -341,11 +353,13 @@ public void postEventWithWrapInModOrder(T e, Bi
LOGGER.error("Cowardly refusing to send event {} to a broken mod state", e.getClass().getName());
return;
}
- ModList.get().forEachModInOrder(mc -> {
- pre.accept(mc, e);
- mc.acceptEvent(e);
- post.accept(mc, e);
- });
+ for (EventPriority phase : EventPriority.values()) {
+ ModList.get().forEachModInOrder(mc -> {
+ pre.accept(mc, e);
+ mc.acceptEvent(phase, e);
+ post.accept(mc, e);
+ });
+ }
}
public List getWarnings()
@@ -378,5 +392,10 @@ public boolean matches(final Object mod) {
public Object getMod() {
return null;
}
+
+ @Override
+ public @Nullable IEventBus getEventBus() {
+ return null;
+ }
}
}
diff --git a/languages/java/src/main/java/net/neoforged/fml/javafmlmod/FMLModContainer.java b/languages/java/src/main/java/net/neoforged/fml/javafmlmod/FMLModContainer.java
index 6c5a4439e..1c20bc74c 100644
--- a/languages/java/src/main/java/net/neoforged/fml/javafmlmod/FMLModContainer.java
+++ b/languages/java/src/main/java/net/neoforged/fml/javafmlmod/FMLModContainer.java
@@ -45,7 +45,11 @@ public FMLModContainer(IModInfo info, String className, ModFileScanData modFileS
LOGGER.debug(LOADING,"Creating FMLModContainer instance for {}", className);
this.scanResults = modFileScanResults;
activityMap.put(ModLoadingStage.CONSTRUCT, this::constructMod);
- this.eventBus = BusBuilder.builder().setExceptionHandler(this::onEventFailed).markerType(IModBusEvent.class).build();
+ this.eventBus = BusBuilder.builder()
+ .setExceptionHandler(this::onEventFailed)
+ .markerType(IModBusEvent.class)
+ .allowPerPhasePost()
+ .build();
this.configHandler = Optional.of(ce->this.eventBus.post(ce.self()));
final FMLJavaModLoadingContext contextExtension = new FMLJavaModLoadingContext(this);
this.contextExtension = () -> contextExtension;
@@ -138,20 +142,9 @@ public Object getMod()
return modInstance;
}
+ @Override
public IEventBus getEventBus()
{
return this.eventBus;
}
-
- @Override
- protected void acceptEvent(final T e) {
- try {
- LOGGER.trace(LOADING, "Firing event for modid {} : {}", this.getModId(), e);
- this.eventBus.post(e);
- LOGGER.trace(LOADING, "Fired event for modid {} : {}", this.getModId(), e);
- } catch (Throwable t) {
- LOGGER.error(LOADING,"Caught exception during event {} dispatch for modid {}", e, this.getModId(), t);
- throw new ModLoadingException(modInfo, modLoadingStage, "fml.modloading.errorduringevent", t);
- }
- }
}
diff --git a/languages/lowcode/src/main/java/net/neoforged/fml/lowcodemod/LowCodeModContainer.java b/languages/lowcode/src/main/java/net/neoforged/fml/lowcodemod/LowCodeModContainer.java
index 810d2ac36..437851d76 100644
--- a/languages/lowcode/src/main/java/net/neoforged/fml/lowcodemod/LowCodeModContainer.java
+++ b/languages/lowcode/src/main/java/net/neoforged/fml/lowcodemod/LowCodeModContainer.java
@@ -6,12 +6,12 @@
package net.neoforged.fml.lowcodemod;
import com.mojang.logging.LogUtils;
-import net.neoforged.bus.api.Event;
+import net.neoforged.bus.api.IEventBus;
import net.neoforged.fml.IExtensionPoint;
import net.neoforged.fml.ModContainer;
-import net.neoforged.fml.event.IModBusEvent;
import net.neoforged.neoforgespi.language.IModInfo;
import net.neoforged.neoforgespi.language.ModFileScanData;
+import org.jetbrains.annotations.Nullable;
import org.slf4j.Logger;
import static net.neoforged.fml.loading.LogMarkers.LOADING;
@@ -45,7 +45,7 @@ public Object getMod()
}
@Override
- protected void acceptEvent(final T e)
- {
+ public @Nullable IEventBus getEventBus() {
+ return null;
}
}
diff --git a/languages/minecraft/src/main/java/net/neoforged/fml/mclanguageprovider/MinecraftModContainer.java b/languages/minecraft/src/main/java/net/neoforged/fml/mclanguageprovider/MinecraftModContainer.java
index be4671004..9f0cc0a5d 100644
--- a/languages/minecraft/src/main/java/net/neoforged/fml/mclanguageprovider/MinecraftModContainer.java
+++ b/languages/minecraft/src/main/java/net/neoforged/fml/mclanguageprovider/MinecraftModContainer.java
@@ -5,8 +5,10 @@
package net.neoforged.fml.mclanguageprovider;
+import net.neoforged.bus.api.IEventBus;
import net.neoforged.fml.ModContainer;
import net.neoforged.neoforgespi.language.IModInfo;
+import org.jetbrains.annotations.Nullable;
import java.util.Objects;
@@ -27,4 +29,9 @@ public boolean matches(final Object mod) {
public Object getMod() {
return MCMODINSTANCE;
}
+
+ @Override
+ public @Nullable IEventBus getEventBus() {
+ return null;
+ }
}