Remove SpawnListener and adjust egg drop mechanics for difficulty-based spawning.

This commit is contained in:
akastijn 2026-04-05 17:36:32 +02:00
parent f25e696448
commit 465cbd17fc
9 changed files with 83 additions and 121 deletions

View File

@ -4,10 +4,8 @@ import com.alttd.easter.commands.Command;
import com.alttd.easter.config.Config; import com.alttd.easter.config.Config;
import com.alttd.easter.config.Messages; import com.alttd.easter.config.Messages;
import com.alttd.easter.data.DataManager; import com.alttd.easter.data.DataManager;
import com.alttd.easter.glow.GlowManager;
import com.alttd.easter.gui.TurnInGuiManager; import com.alttd.easter.gui.TurnInGuiManager;
import com.alttd.easter.listeners.DeathListener; import com.alttd.easter.listeners.DeathListener;
import com.alttd.easter.listeners.SpawnListener;
import com.alttd.easter.npc.RabbitNpcManager; import com.alttd.easter.npc.RabbitNpcManager;
import org.bukkit.plugin.PluginManager; import org.bukkit.plugin.PluginManager;
import org.bukkit.plugin.java.JavaPlugin; import org.bukkit.plugin.java.JavaPlugin;
@ -16,19 +14,16 @@ public final class Easter extends JavaPlugin {
private DataManager dataManager; private DataManager dataManager;
private RabbitNpcManager rabbitNpcManager; private RabbitNpcManager rabbitNpcManager;
private GlowManager glowManager;
private TurnInGuiManager turnInGuiManager; private TurnInGuiManager turnInGuiManager;
@Override @Override
public void onEnable() { public void onEnable() {
reloadConfigs(); reloadConfigs();
this.dataManager = new DataManager(this); this.dataManager = new DataManager(this);
this.glowManager = new GlowManager(this);
this.turnInGuiManager = new TurnInGuiManager(dataManager); this.turnInGuiManager = new TurnInGuiManager(dataManager);
this.rabbitNpcManager = new RabbitNpcManager(this, turnInGuiManager, dataManager); this.rabbitNpcManager = new RabbitNpcManager(this, turnInGuiManager, dataManager);
registerCommands(); registerCommands();
registerEvents(); registerEvents();
registerSchedulers();
} }
@Override @Override
@ -44,7 +39,6 @@ public final class Easter extends JavaPlugin {
private void registerEvents() { private void registerEvents() {
PluginManager pluginManager = getServer().getPluginManager(); PluginManager pluginManager = getServer().getPluginManager();
pluginManager.registerEvents(new SpawnListener(glowManager), this);
pluginManager.registerEvents(new DeathListener(), this); pluginManager.registerEvents(new DeathListener(), this);
pluginManager.registerEvents(rabbitNpcManager, this); pluginManager.registerEvents(rabbitNpcManager, this);
pluginManager.registerEvents(turnInGuiManager, this); pluginManager.registerEvents(turnInGuiManager, this);
@ -57,8 +51,4 @@ public final class Easter extends JavaPlugin {
rabbitNpcManager.reload(); rabbitNpcManager.reload();
} }
} }
private void registerSchedulers() {
// No schedulers required currently
}
} }

View File

@ -31,11 +31,19 @@ public class Config extends AbstractConfig {
public static class EGGS { public static class EGGS {
private static final String prefix = "eggs."; private static final String prefix = "eggs.";
public static double SPAWN_CHANCE = 0.02; // ~2 per night per player public static double EASY_CHANCE = 0.05;
public static double NORMAL_CHANCE = 0.10;
public static double HARD_CHANCE = 0.25;
@SuppressWarnings("unused") @SuppressWarnings("unused")
private static void load() { private static void load() {
SPAWN_CHANCE = config.getDouble(prefix, "spawn-chance", SPAWN_CHANCE); EASY_CHANCE = config.getDouble(prefix, "easy-chance", EASY_CHANCE);
NORMAL_CHANCE = config.getDouble(prefix, "normal-chance", NORMAL_CHANCE);
HARD_CHANCE = config.getDouble(prefix, "hard-chance", HARD_CHANCE);
// ensure paths exist in config file with their current values
config.yaml.addDefault(prefix + "easy-chance", EASY_CHANCE);
config.yaml.addDefault(prefix + "normal-chance", NORMAL_CHANCE);
config.yaml.addDefault(prefix + "hard-chance", HARD_CHANCE);
} }
} }
@ -123,19 +131,5 @@ public class Config extends AbstractConfig {
} }
config.set(prefix, "next-index", NEXT_INDEX); config.set(prefix, "next-index", NEXT_INDEX);
} }
public static ItemStack getNextPrize() {
if (LIST.isEmpty()) {
log.warn("[Easter] No prizes configured. Staff should configure prizes using /easter setprize.");
return null;
}
if (NEXT_INDEX < 0 || NEXT_INDEX >= LIST.size()) {
NEXT_INDEX = 0;
}
ItemStack prize = LIST.get(NEXT_INDEX).clone();
NEXT_INDEX = (NEXT_INDEX + 1) % LIST.size();
config.set(prefix, "next-index", NEXT_INDEX);
return prize;
}
} }
} }

View File

@ -70,7 +70,7 @@ public class Messages extends AbstractConfig {
public static String NO_EGGS = "<yellow>Please come back when you've found some eggs!</yellow>"; public static String NO_EGGS = "<yellow>Please come back when you've found some eggs!</yellow>";
public static String THANK_YOU = "<green>Thank you for finding my eggs <player>! Keep looking for more!</green>"; public static String THANK_YOU = "<green>Thank you for finding my eggs <player>! Keep looking for more!</green>";
public static String ONE_EGG = "<green>Thank you for finding my eggs <player>! I have received the <egg1> from you. Please keep looking for more!</green>"; public static String ONE_EGG = "<green>Thank you for finding my eggs <player>! I have received the <egg1> from you. Please keep looking for more!</green>";
public static String MULTI_EGGS = "<green>Thank you for finding my eggs <player>! I have received the <egg1>, <egg2>, and <egg3> from you. Please keep looking for more!</green>"; public static String MULTI_EGGS = "<green>Thank you for finding my eggs <player>! I have received <eggs> from you. Please keep looking for more!</green>";
public static String REWARD = "<green>Thank you! You've turned in 4 unique eggs and earned a prize!</green>"; public static String REWARD = "<green>Thank you! You've turned in 4 unique eggs and earned a prize!</green>";
public static String DUPLICATE_EGGS = "<yellow>Those are duplicates I already have. Bring me new colors!</yellow>"; public static String DUPLICATE_EGGS = "<yellow>Those are duplicates I already have. Bring me new colors!</yellow>";
public static String GUI_TITLE = "<gold>Turn in Eggs</gold>"; public static String GUI_TITLE = "<gold>Turn in Eggs</gold>";

View File

@ -6,22 +6,24 @@ import org.bukkit.enchantments.Enchantment;
import org.bukkit.entity.EntityType; import org.bukkit.entity.EntityType;
public enum EggType { public enum EggType {
BROWN_HUSK(Material.BROWN_GLAZED_TERRACOTTA, EntityType.HUSK, Enchantment.FIRE_PROTECTION, 2), BROWN_HUSK(Material.BROWN_GLAZED_TERRACOTTA, EntityType.HUSK, Enchantment.FIRE_PROTECTION, 2, Difficulty.EASY),
RED_SPIDER(Material.RED_GLAZED_TERRACOTTA, EntityType.SPIDER, Enchantment.BANE_OF_ARTHROPODS, 2), RED_SPIDER(Material.RED_GLAZED_TERRACOTTA, EntityType.SPIDER, Enchantment.BANE_OF_ARTHROPODS, 2, Difficulty.EASY),
ORANGE_BLAZE(Material.ORANGE_GLAZED_TERRACOTTA, EntityType.BLAZE, Enchantment.FIRE_ASPECT, 2), ORANGE_BLAZE(Material.ORANGE_GLAZED_TERRACOTTA, EntityType.BLAZE, Enchantment.FIRE_ASPECT, 2, Difficulty.NORMAL),
YELLOW_PIGLIN(Material.YELLOW_GLAZED_TERRACOTTA, EntityType.PIGLIN, Enchantment.FORTUNE, 2), YELLOW_PIGLIN(Material.YELLOW_GLAZED_TERRACOTTA, EntityType.PIGLIN, Enchantment.FORTUNE, 2, Difficulty.NORMAL),
LIME_CREEPER(Material.LIME_GLAZED_TERRACOTTA, EntityType.CREEPER, Enchantment.BLAST_PROTECTION, 2), LIME_CREEPER(Material.LIME_GLAZED_TERRACOTTA, EntityType.CREEPER, Enchantment.BLAST_PROTECTION, 2, Difficulty.NORMAL),
GREEN_ZOMBIE(Material.GREEN_GLAZED_TERRACOTTA, EntityType.ZOMBIE, Enchantment.FEATHER_FALLING, 2), GREEN_ZOMBIE(Material.GREEN_GLAZED_TERRACOTTA, EntityType.ZOMBIE, Enchantment.FEATHER_FALLING, 2, Difficulty.EASY),
LIGHT_BLUE_STRAY(Material.LIGHT_BLUE_GLAZED_TERRACOTTA, EntityType.STRAY, Enchantment.FROST_WALKER, 2), LIGHT_BLUE_STRAY(Material.LIGHT_BLUE_GLAZED_TERRACOTTA, EntityType.STRAY, Enchantment.FROST_WALKER, 2, Difficulty.NORMAL),
CYAN_GUARDIAN(Material.CYAN_GLAZED_TERRACOTTA, EntityType.GUARDIAN, Enchantment.THORNS, 2), CYAN_GUARDIAN(Material.CYAN_GLAZED_TERRACOTTA, EntityType.GUARDIAN, Enchantment.THORNS, 2, Difficulty.HARD),
BLUE_WARDEN(Material.BLUE_GLAZED_TERRACOTTA, EntityType.WARDEN, Enchantment.SWIFT_SNEAK, 2), BLUE_WARDEN(Material.BLUE_GLAZED_TERRACOTTA, EntityType.WARDEN, Enchantment.SWIFT_SNEAK, 2, Difficulty.HARD),
PURPLE_ENDER_DRAGON(Material.PURPLE_GLAZED_TERRACOTTA, EntityType.ENDER_DRAGON, Enchantment.POWER, 2), PURPLE_ENDER_DRAGON(Material.PURPLE_GLAZED_TERRACOTTA, EntityType.ENDER_DRAGON, Enchantment.POWER, 2, Difficulty.HARD),
MAGENTA_ENDERMAN(Material.MAGENTA_GLAZED_TERRACOTTA, EntityType.ENDERMAN, Enchantment.INFINITY, 2), MAGENTA_ENDERMAN(Material.MAGENTA_GLAZED_TERRACOTTA, EntityType.ENDERMAN, Enchantment.INFINITY, 2, Difficulty.NORMAL),
PINK_HOGLIN(Material.PINK_GLAZED_TERRACOTTA, EntityType.HOGLIN, Enchantment.PIERCING, 2), PINK_HOGLIN(Material.PINK_GLAZED_TERRACOTTA, EntityType.HOGLIN, Enchantment.PIERCING, 2, Difficulty.NORMAL),
WHITE_SKELETON(Material.WHITE_GLAZED_TERRACOTTA, EntityType.SKELETON, Enchantment.MULTISHOT, 2), WHITE_SKELETON(Material.WHITE_GLAZED_TERRACOTTA, EntityType.SKELETON, Enchantment.MULTISHOT, 2, Difficulty.EASY),
LIGHT_GRAY_SILVERFISH(Material.LIGHT_GRAY_GLAZED_TERRACOTTA, EntityType.SILVERFISH, Enchantment.EFFICIENCY, 2), LIGHT_GRAY_SILVERFISH(Material.LIGHT_GRAY_GLAZED_TERRACOTTA, EntityType.SILVERFISH, Enchantment.EFFICIENCY, 2, Difficulty.EASY),
GRAY_WITHER(Material.GRAY_GLAZED_TERRACOTTA, EntityType.WITHER, Enchantment.SMITE, 2), GRAY_WITHER(Material.GRAY_GLAZED_TERRACOTTA, EntityType.WITHER, Enchantment.SMITE, 2, Difficulty.HARD),
BLACK_WITHER_SKELETON(Material.BLACK_GLAZED_TERRACOTTA, EntityType.WITHER_SKELETON, Enchantment.LOOTING, 2); BLACK_WITHER_SKELETON(Material.BLACK_GLAZED_TERRACOTTA, EntityType.WITHER_SKELETON, Enchantment.LOOTING, 2, Difficulty.HARD);
public enum Difficulty { EASY, NORMAL, HARD }
@Getter @Getter
private final Material material; private final Material material;
@ -30,12 +32,15 @@ public enum EggType {
private final Enchantment enchantment; private final Enchantment enchantment;
@Getter @Getter
private final int enchantLevel; private final int enchantLevel;
@Getter
private final Difficulty difficulty;
EggType(Material material, EntityType entityType, Enchantment enchantment, int enchantLevel) { EggType(Material material, EntityType entityType, Enchantment enchantment, int enchantLevel, Difficulty difficulty) {
this.material = material; this.material = material;
this.entityType = entityType; this.entityType = entityType;
this.enchantment = enchantment; this.enchantment = enchantment;
this.enchantLevel = enchantLevel; this.enchantLevel = enchantLevel;
this.difficulty = difficulty;
} }
public static EggType fromEntityType(EntityType entityType) { public static EggType fromEntityType(EntityType entityType) {

View File

@ -4,6 +4,7 @@ import com.alttd.easter.egg.EggType;
import net.kyori.adventure.text.format.NamedTextColor; import net.kyori.adventure.text.format.NamedTextColor;
import org.bukkit.Bukkit; import org.bukkit.Bukkit;
import org.bukkit.entity.Entity; import org.bukkit.entity.Entity;
import org.bukkit.plugin.Plugin;
import org.bukkit.scoreboard.Scoreboard; import org.bukkit.scoreboard.Scoreboard;
import org.bukkit.scoreboard.Team; import org.bukkit.scoreboard.Team;
@ -11,7 +12,7 @@ public class GlowManager {
private final Scoreboard scoreboard; private final Scoreboard scoreboard;
public GlowManager(org.bukkit.plugin.Plugin plugin) { public GlowManager(Plugin plugin) {
this.scoreboard = Bukkit.getScoreboardManager().getMainScoreboard(); this.scoreboard = Bukkit.getScoreboardManager().getMainScoreboard();
} }

View File

@ -7,6 +7,7 @@ import com.alttd.easter.egg.EggType;
import com.alttd.easter.util.ItemUtils; import com.alttd.easter.util.ItemUtils;
import lombok.extern.slf4j.Slf4j; import lombok.extern.slf4j.Slf4j;
import net.kyori.adventure.text.Component; import net.kyori.adventure.text.Component;
import net.kyori.adventure.text.JoinConfiguration;
import net.kyori.adventure.text.minimessage.MiniMessage; import net.kyori.adventure.text.minimessage.MiniMessage;
import net.kyori.adventure.text.minimessage.tag.resolver.Placeholder; import net.kyori.adventure.text.minimessage.tag.resolver.Placeholder;
import org.bukkit.Bukkit; import org.bukkit.Bukkit;
@ -105,21 +106,36 @@ public class TurnInGuiManager implements Listener {
int newlyAdded = dataManager.addEggs(player.getUniqueId(), submitted); int newlyAdded = dataManager.addEggs(player.getUniqueId(), submitted);
dataManager.save(); dataManager.save();
// Dialog formatting for submission summary // Dialog formatting for submission summary
List<String> colors = submitted.stream().map(ItemUtils::colorName).toList(); List<Component> eggComponents = new ArrayList<>();
if (colors.size() == 1) { for (EggType type : submitted) {
eggComponents.add(Component.text(ItemUtils.colorName(type)));
}
if (eggComponents.size() == 1) {
player.sendRichMessage(Messages.RABBIT.ONE_EGG, player.sendRichMessage(Messages.RABBIT.ONE_EGG,
Placeholder.parsed("player", player.getName()), Placeholder.parsed("player", player.getName()),
Placeholder.parsed("egg1", colors.getFirst()) Placeholder.component("egg1", eggComponents.getFirst())
); );
} else { } else {
String c1 = !colors.isEmpty() ? colors.get(0) : ""; Component eggsJoined;
String c2 = colors.size() > 1 ? colors.get(1) : ""; if (eggComponents.size() == 2) {
String c3 = colors.size() > 2 ? colors.get(2) : ""; eggsJoined = Component.join(
JoinConfiguration.builder()
.lastSeparator(Component.text(" and "))
.build(),
eggComponents
);
} else {
eggsJoined = Component.join(
JoinConfiguration.builder()
.separator(Component.text(", "))
.lastSeparator(Component.text(", and "))
.build(),
eggComponents
);
}
player.sendRichMessage(Messages.RABBIT.MULTI_EGGS, player.sendRichMessage(Messages.RABBIT.MULTI_EGGS,
Placeholder.parsed("player", player.getName()), Placeholder.parsed("player", player.getName()),
Placeholder.parsed("egg1", c1), Placeholder.component("eggs", eggsJoined)
Placeholder.parsed("egg2", c2),
Placeholder.parsed("egg3", c3)
); );
} }
int newTotal = prevTotal + newlyAdded; int newTotal = prevTotal + newlyAdded;

View File

@ -1,30 +1,41 @@
package com.alttd.easter.listeners; package com.alttd.easter.listeners;
import com.alttd.easter.config.Config;
import com.alttd.easter.egg.EggType; import com.alttd.easter.egg.EggType;
import com.alttd.easter.util.ItemUtils; import com.alttd.easter.util.ItemUtils;
import com.alttd.easter.util.Keys;
import org.bukkit.entity.LivingEntity; import org.bukkit.entity.LivingEntity;
import org.bukkit.entity.Mob;
import org.bukkit.event.EventHandler; import org.bukkit.event.EventHandler;
import org.bukkit.event.Listener; import org.bukkit.event.Listener;
import org.bukkit.event.entity.EntityDeathEvent; import org.bukkit.event.entity.EntityDeathEvent;
import org.bukkit.persistence.PersistentDataType;
import java.util.concurrent.ThreadLocalRandom;
public class DeathListener implements Listener { public class DeathListener implements Listener {
@EventHandler @EventHandler
public void onDeath(EntityDeathEvent event) { public void onDeath(EntityDeathEvent event) {
LivingEntity entity = event.getEntity(); LivingEntity entity = event.getEntity();
String eggName = entity.getPersistentDataContainer().get(Keys.EGG_ENTITY, PersistentDataType.STRING); // Only consider AI-capable mobs
if (eggName == null) { if (!(entity instanceof Mob mob) || !mob.isAware()) {
return; return;
} }
EggType type; EggType eggType = EggType.fromEntityType(entity.getType());
try { if (eggType == null) {
type = EggType.valueOf(eggName);
} catch (IllegalArgumentException ex) {
return; return;
} }
// Ensure the egg drops double chance;
event.getDrops().add(ItemUtils.createEggItem(type)); switch (eggType.getDifficulty()) {
case EASY -> chance = Config.EGGS.EASY_CHANCE;
case NORMAL -> chance = Config.EGGS.NORMAL_CHANCE;
case HARD -> chance = Config.EGGS.HARD_CHANCE;
default -> chance = 0.0;
}
if (chance <= 0) {
return;
}
if (ThreadLocalRandom.current().nextDouble() < chance) {
event.getDrops().add(ItemUtils.createEggItem(eggType));
}
} }
} }

View File

@ -1,55 +0,0 @@
package com.alttd.easter.listeners;
import com.alttd.easter.config.Config;
import com.alttd.easter.egg.EggType;
import com.alttd.easter.glow.GlowManager;
import com.alttd.easter.util.ItemUtils;
import com.alttd.easter.util.Keys;
import org.bukkit.entity.Creature;
import org.bukkit.entity.EntityType;
import org.bukkit.event.EventHandler;
import org.bukkit.event.Listener;
import org.bukkit.event.entity.CreatureSpawnEvent;
import org.bukkit.inventory.EntityEquipment;
import org.bukkit.inventory.ItemStack;
import org.bukkit.persistence.PersistentDataType;
import java.util.concurrent.ThreadLocalRandom;
@SuppressWarnings("ClassCanBeRecord")
public class SpawnListener implements Listener {
private final GlowManager glowManager;
public SpawnListener(GlowManager glowManager) {
this.glowManager = glowManager;
}
@EventHandler
public void onNaturalSpawn(CreatureSpawnEvent event) {
if (event.getSpawnReason() != CreatureSpawnEvent.SpawnReason.NATURAL) {
return;
}
if (!(event.getEntity() instanceof Creature creature)) {
return;
}
EntityType type = event.getEntityType();
EggType eggType = EggType.fromEntityType(type);
if (eggType == null) {
return;
}
double chance = Config.EGGS.SPAWN_CHANCE;
if (ThreadLocalRandom.current().nextDouble() >= chance) {
return;
}
ItemStack eggItem = ItemUtils.createEggItem(eggType);
EntityEquipment equipment = creature.getEquipment();
equipment.setHelmet(eggItem);
equipment.setHelmetDropChance(1.0f);
// tag entity so we know it's an egg carrier
creature.getPersistentDataContainer().set(Keys.EGG_ENTITY, PersistentDataType.STRING, eggType.name());
creature.setGlowing(true);
glowManager.applyGlow(creature, eggType);
}
}

View File

@ -28,8 +28,8 @@ public final class ItemUtils {
case MAGENTA_ENDERMAN -> "Magenta"; case MAGENTA_ENDERMAN -> "Magenta";
case PINK_HOGLIN -> "Pink"; case PINK_HOGLIN -> "Pink";
case WHITE_SKELETON -> "White"; case WHITE_SKELETON -> "White";
case LIGHT_GRAY_SILVERFISH -> "Light Grey"; case LIGHT_GRAY_SILVERFISH -> "Light Gray";
case GRAY_WITHER -> "Grey"; case GRAY_WITHER -> "Gray";
case BLACK_WITHER_SKELETON -> "Black"; case BLACK_WITHER_SKELETON -> "Black";
}; };
} }