Mob Effects & Potions
Status effects, sometimes known as potion effects and referred to in-code as MobEffect
s, are effects that influence an entity every tick. This article explains how to use them, what the difference between an effect and a potion is, and how to add your own.
Terminology
- A
MobEffect
affects an entity every tick. Like blocks or items,MobEffect
s are registry objects, meaning they must be registered and are singletons.- An instant mob effect is a special kind of mob effect that is designed to be applied for one tick. Vanilla has two instant effects, Instant Health and Instant Harming.
- A
MobEffectInstance
is an instance of aMobEffect
, with a duration, amplifier and some other properties set (see below).MobEffectInstance
s are toMobEffect
s whatItemStack
s are toItem
s. - A
Potion
is a collection ofMobEffectInstance
s. Vanilla mainly uses potions for the four potion items (read on), however, they can be applied to any item at will. It is up to the item if and how the item then uses the potion set on it. - A potion item is an item that is meant to have a potion set on it. This is an informal term, the vanilla
PotionItem
class has nothing to do with this (it refers to the "normal" potion item). Minecraft currently has four potion items: potions, splash potions, lingering potions, and tipped arrows; however more may be added by mods.
MobEffect
s
To create your own MobEffect
, extend the MobEffect
class:
public class MyMobEffect extends MobEffect {
public MyMobEffect(MobEffectCategory category, int color) {
super(category, color);
}
@Override
public boolean applyEffectTick(LivingEntity entity, int amplifier) {
// Apply your effect logic here.
// If this returns false when shouldApplyEffectTickThisTick returns true, the effect will immediately be removed
return true;
}
// Whether the effect should apply this tick. Used e.g. by the Regeneration effect that only applies
// once every x ticks, depending on the tick count and amplifier.
@Override
public boolean shouldApplyEffectTickThisTick(int tickCount, int amplifier) {
return tickCount % 2 == 0; // replace this with whatever check you want
}
// Utility method that is called when the effect is first added to the entity.
// This does not get called again until all instances of this effect have been removed from the entity.
@Override
public void onEffectAdded(LivingEntity entity, int amplifier) {
super.onEffectAdded(entity, amplifier);
}
// Utility method that is called when the effect is added to the entity.
// This gets called every time this effect is added to the entity.
@Override
public void onEffectStarted(LivingEntity entity, int amplifier) {
}
}
Like all registry objects, MobEffect
s must be registered, like so:
//MOB_EFFECTS is a DeferredRegister<MobEffect>
public static final Supplier<MyMobEffect> MY_MOB_EFFECT = MOB_EFFECTS.register("my_mob_effect", () -> new MyMobEffect(
//Can be either BENEFICIAL, NEUTRAL or HARMFUL. Used to determine the potion tooltip color of this effect.
MobEffectCategory.BENEFICIAL,
//The color of the effect particles.
0xffffff
));
The MobEffect
class also provides default functionality for adding attribute modifiers to affected entities. For example, the speed effect adds an attribute modifier for movement speed. Effect attribute modifiers are added like so:
public static final Supplier<MyMobEffect> MY_MOB_EFFECT = MOB_EFFECTS.register("my_mob_effect", () -> new MyMobEffect(...)
.addAttributeModifier(Attribute.ATTACK_DAMAGE, ResourceLocation.fromNamespaceAndPath("examplemod", "effect.strength"), 2.0, AttributeModifier.Operation.ADD_VALUE)
);
InstantenousMobEffect
If you want to create an instant effect, you can use the helper class InstantenousMobEffect
instead of the regular MobEffect
class, like so:
public class MyMobEffect extends InstantenousMobEffect {
public MyMobEffect(MobEffectCategory category, int color) {
super(category, color);
}
@Override
public void applyEffectTick(LivingEntity entity, int amplifier) {
// Apply your effect logic here.
}
}
Then, register your effect like normal.
Events
Many effects have their logic applied in other places. For example, the levitation effect is applied in the living entity movement handler. For modded MobEffect
s, it often makes sense to apply them in an event handler. NeoForge also provides a few events related to effects:
MobEffectEvent.Applicable
is fired when the game checks whether aMobEffectInstance
can be applied to an entity. This event can be used to deny or force adding the effect instance to the target.MobEffectEvent.Added
is fired when theMobEffectInstance
is added to the target. This event contains information about a previousMobEffectInstance
that may have been present on the target.MobEffectEvent.Expired
is fired when theMobEffectInstance
expires, i.e. the timer goes to zero.MobEffectEvent.Remove
is fired when the effect is removed from the entity through means other than expiring, e.g. through drinking milk or via commands.
MobEffectInstance
s
A MobEffectInstance
is, simply put, an effect applied to an entity. Creating a MobEffectInstance
is done by calling the constructor:
MobEffectInstance instance = new MobEffectInstance(
// The mob effect to use.
MobEffects.REGENERATION,
// The duration to use, in ticks. Defaults to 0 if not specified.
500,
// The amplifier to use. This is the "strength" of the effect, i.e. Strength I, Strength II, etc;
// starting at 0. Defaults to 0 if not specified.
0,
// Whether the effect is an "ambient" effect, meaning it is being applied by an ambient source,
// of which Minecraft currently has the beacon and the conduit. Defaults to false if not specified.
false,
// Whether the effect is visible in the inventory. Defaults to true if not specified.
true,
// Whether an effect icon is visible in the top right corner. Defaults to true if not specified.
true
);
Several constructor overloads are available, omitting the last 1-5 parameters, respectively.
MobEffectInstance
s are mutable. If you need a copy, call new MobEffectInstance(oldInstance)
.
Using MobEffectInstance
s
A MobEffectInstance
can be added to an entity like so:
MobEffectInstance instance = new MobEffectInstance(...);
entity.addEffect(instance);
Similarly, MobEffectInstance
can also be removed from an entity. Since a MobEffectInstance
overwrites pre-existing MobEffectInstance
s of the same MobEffect
on the entity, there can only ever be one MobEffectInstance
per MobEffect
and entity. As such, specifying the MobEffect
suffices when removing:
entity.removeEffect(MobEffects.REGENERATION);
MobEffect
s can only be applied to LivingEntity
or its subclasses, i.e. players and mobs. Things like items or thrown snowballs cannot be affected by MobEffect
s.
Potion
s
Potion
s are created by calling the constructor of Potion
with the MobEffectInstance
s you want the potion to have. For example:
//POTIONS is a DeferredRegister<Potion>
public static final Supplier<Potion> MY_POTION = POTIONS.register("my_potion", () -> new Potion(new MobEffectInstance(MY_MOB_EFFECT, 3600)));
Note that the parameter of new Potion
is a vararg. This means that you can add as many effects as you want to the potion. This also means that it is possible to create empty potions, i.e. potions that don't have any effects. Simply call new Potion()
and you're done! (This is how vanilla adds the awkward
potion, by the way.)
The name of the potion can be passed as the first constructor argument. It is used for translating; for example, the long and strong potion variants in vanilla use this to have the same names as their base variant. The name is not required; if it is omitted, the name will be queried from the registry.
The PotionContents
class offers various helper methods related to potion items. Potion item store their PotionContents
via DataComponent#POTION_CONTENTS
.
Brewing
Now that your potion is added, potion items are available for your potion. However, there is no way to obtain your potion in survival, so let's change that!
Potions are traditionally made in the Brewing Stand. Unfortunately, Mojang does not provide datapack support for brewing recipes, so we have to be a little old-fashioned and add our recipes through code via the RegisterBrewingRecipesEvent
event. This is done like so:
// Using some method to listen to the event
@SubscribeEvent
public static void registerBrewingRecipes(RegisterBrewingRecipesEvent event) {
// Gets the builder to add recipes to
PotionBrewing.Builder builder = event.getBuilder();
// Will add brewing recipes for all container potions (e.g. potion, splash potion, lingering potion)
builder.addMix(
// The initial potion to apply to
Potions.AWKWARD,
// The brewing ingredient. This is the item at the top of the brewing stand.
Items.FEATHER,
// The resulting potion
MY_POTION
);
}