All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.cryptomorin.xseries.XEnchantment Maven / Gradle / Ivy

/*
 * The MIT License (MIT)
 *
 * Copyright (c) 2024 Crypto Morin
 *
 * Permission is hereby granted, free of charge, to any person obtaining a copy
 * of this software and associated documentation files (the "Software"), to deal
 * in the Software without restriction, including without limitation the rights
 * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
 * copies of the Software, and to permit persons to whom the Software is
 * furnished to do so, subject to the following conditions:
 *
 * The above copyright notice and this permission notice shall be included in
 * all copies or substantial portions of the Software.
 *
 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED,
 * INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR
 * PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE
 * FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
 * ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
 */
package com.cryptomorin.xseries;

import com.google.common.base.Enums;
import org.bukkit.Material;
import org.bukkit.NamespacedKey;
import org.bukkit.Registry;
import org.bukkit.enchantments.Enchantment;
import org.bukkit.entity.EntityType;
import org.bukkit.inventory.ItemStack;
import org.bukkit.inventory.meta.EnchantmentStorageMeta;

import javax.annotation.Nonnull;
import javax.annotation.Nullable;
import java.util.*;
import java.util.stream.Collectors;

/**
 * Enchantment support with multiple aliases.
 * Uses EssentialsX enchantment list for aliases.
 * Enchantment levels do not start from 0, they start from 1
 * 

* EssentialsX Enchantment: https://github.com/Bukkit/Bukkit/blob/master/src/main/java/org/bukkit/enchantments/Enchantment.java * Enchantment: https://hub.spigotmc.org/javadocs/bukkit/org/bukkit/enchantments/Enchantment.html * Enchanting: https://minecraft.wiki/w/Enchanting * * @author Crypto Morin * @version 3.0.0 * @see Enchantment */ public enum XEnchantment { AQUA_AFFINITY("WATER_WORKER", "WATER_WORKER", "AQUA_AFFINITY", "WATER_MINE"), BANE_OF_ARTHROPODS("DAMAGE_ARTHROPODS", "ARDMG", "BANE_OF_ARTHROPOD", "ARTHROPOD"), BINDING_CURSE("BINDING_CURSE", "BIND_CURSE", "BINDING", "BIND"), BLAST_PROTECTION("PROTECTION_EXPLOSIONS", "BLAST_PROTECT", "EXPLOSIONS_PROTECTION", "EXPLOSION_PROTECTION", "BLAST_PROTECTION"), BREACH, CHANNELING("CHANNELLING", "CHANELLING", "CHANELING", "CHANNEL"), DENSITY, DEPTH_STRIDER("DEPTH", "STRIDER"), EFFICIENCY("DIG_SPEED", "MINE_SPEED", "CUT_SPEED"), FEATHER_FALLING("PROTECTION_FALL", "FALL_PROT", "FEATHER_FALL", "FALL_PROTECTION", "FEATHER_FALLING"), FIRE_ASPECT("FIRE", "MELEE_FIRE", "MELEE_FLAME"), FIRE_PROTECTION("PROTECTION_FIRE", "FIRE_PROT", "FIRE_PROTECT", "FIRE_PROTECTION", "FLAME_PROTECTION", "FLAME_PROTECT", "FLAME_PROT"), FLAME("ARROW_FIRE", "FLAME_ARROW", "FIRE_ARROW"), FORTUNE("LOOT_BONUS_BLOCKS", "BLOCKS_LOOT_BONUS", "FORT"), FROST_WALKER("FROST", "WALKER"), IMPALING("IMPALE", "OCEAN_DAMAGE", "OCEAN_DMG"), INFINITY("ARROW_INFINITE", "INF_ARROWS", "INFINITE_ARROWS", "INFINITE", "UNLIMITED", "UNLIMITED_ARROWS"), KNOCKBACK("K_BACK"), LOOTING("LOOT_BONUS_MOBS", "MOB_LOOT", "MOBS_LOOT_BONUS"), LOYALTY("LOYAL", "RETURN"), LUCK_OF_THE_SEA("LUCK", "LUCK_OF_SEA", "LUCK_OF_SEAS", "ROD_LUCK"), LURE("ROD_LURE"), MENDING, MULTISHOT("TRIPLE_SHOT"), PIERCING, POWER("ARROW_DAMAGE", "ARROW_POWER"), PROJECTILE_PROTECTION("PROTECTION_PROJECTILE", "PROJECTILE_PROTECTION", "PROJ_PROT"), PROTECTION("PROTECTION_ENVIRONMENTAL", "PROTECT"), PUNCH("ARROW_KNOCKBACK", "ARROWKB", "ARROW_PUNCH"), QUICK_CHARGE("QUICKCHARGE", "QUICK_DRAW", "FAST_CHARGE", "FAST_DRAW"), RESPIRATION("OXYGEN", "BREATH", "BREATHING"), RIPTIDE("RIP", "TIDE", "LAUNCH"), SHARPNESS("DAMAGE_ALL", "ALL_DAMAGE", "ALL_DMG", "SHARP"), SILK_TOUCH("SOFT_TOUCH"), SMITE("DAMAGE_UNDEAD", "UNDEAD_DAMAGE"), SOUL_SPEED("SPEED_SOUL", "SOUL_RUNNER"), SWEEPING_EDGE("SWEEPING", "SWEEPING_EDGE", "SWEEP_EDGE"), SWIFT_SNEAK("SNEAK_SWIFT"), THORNS("HIGHCRIT", "THORN", "HIGHERCRIT"), UNBREAKING("DURABILITY", "DURA"), VANISHING_CURSE("VANISHING_CURSE", "VANISH_CURSE", "VANISHING", "VANISH"), WIND_BURST; /** * Cached list of {@link XEnchantment#values()} to avoid allocating memory for * * @since 1.0.0 */ public static final XEnchantment[] VALUES = values(); /** * Entity types that {@link #SMITE} enchantment is effective against. * This set is unmodifiable. * * @since 1.2.0 */ public static final Set EFFECTIVE_SMITE_ENTITIES; /** * Entity types that {@link #BANE_OF_ARTHROPODS} enchantment is effective against. * This set is unmodifiable. * * @since 1.2.0 */ public static final Set EFFECTIVE_BANE_OF_ARTHROPODS_ENTITIES; static { EntityType bee = Enums.getIfPresent(EntityType.class, "BEE").orNull(); EntityType phantom = Enums.getIfPresent(EntityType.class, "PHANTOM").orNull(); EntityType drowned = Enums.getIfPresent(EntityType.class, "DROWNED").orNull(); EntityType witherSkeleton = Enums.getIfPresent(EntityType.class, "WITHER_SKELETON").orNull(); EntityType skeletonHorse = Enums.getIfPresent(EntityType.class, "SKELETON_HORSE").orNull(); EntityType stray = Enums.getIfPresent(EntityType.class, "STRAY").orNull(); EntityType husk = Enums.getIfPresent(EntityType.class, "HUSK").orNull(); Set arthorposEffective = EnumSet.of(EntityType.SPIDER, EntityType.CAVE_SPIDER, EntityType.SILVERFISH, EntityType.ENDERMITE); if (bee != null) arthorposEffective.add(bee); EFFECTIVE_BANE_OF_ARTHROPODS_ENTITIES = Collections.unmodifiableSet(arthorposEffective); Set smiteEffective = EnumSet.of(EntityType.ZOMBIE, EntityType.SKELETON, EntityType.WITHER); if (phantom != null) smiteEffective.add(phantom); if (drowned != null) smiteEffective.add(drowned); if (witherSkeleton != null) smiteEffective.add(witherSkeleton); if (skeletonHorse != null) smiteEffective.add(skeletonHorse); if (stray != null) smiteEffective.add(stray); if (husk != null) smiteEffective.add(husk); EFFECTIVE_SMITE_ENTITIES = Collections.unmodifiableSet(smiteEffective); } @Nullable private final Enchantment enchantment; /** * If an enchantment has {@code self} as it means that * the vanilla enchantment name matches the Bukkit name. * * @see NamespacedKey#getKey() */ XEnchantment(@Nonnull String... aliases) { Enchantment enchantment = getBukkitEnchant(this.name()); Data.NAMES.put(this.name(), this); for (String legacy : aliases) { Data.NAMES.put(legacy, this); if (enchantment == null) { enchantment = getBukkitEnchant(legacy); } } this.enchantment = enchantment; } @SuppressWarnings("deprecation") private static Enchantment getBukkitEnchant(String name) { if (Data.IS_SUPER_FLAT) { return Registry.ENCHANTMENT.get(NamespacedKey.minecraft(name.toLowerCase(Locale.ENGLISH))); } else if (Data.ISFLAT) { return Enchantment.getByKey(NamespacedKey.minecraft(name.toLowerCase(Locale.ENGLISH))); } else { return Enchantment.getByName(name); } } /** * Checks if {@link #SMITE Smite} is effective * against this type of mob. * * @param type the type of the mob. * @return true if smite enchantment is effective against the mob, otherwise false. * @since 1.1.0 */ public static boolean isSmiteEffectiveAgainst(@Nullable EntityType type) { return type != null && EFFECTIVE_SMITE_ENTITIES.contains(type); } /** * Checks if {@link #BANE_OF_ARTHROPODS Bane of Arthropods} is effective * against this type of mob. * * @param type the type of the mob. * @return true if Bane of Arthropods enchantment is effective against the mob, otherwise false. * @since 1.1.0 */ public static boolean isArthropodsEffectiveAgainst(@Nullable EntityType type) { return type != null && EFFECTIVE_BANE_OF_ARTHROPODS_ENTITIES.contains(type); } /** * Attempts to build the string like an enum name.
* Removes all the spaces, numbers and extra non-English characters. Also removes some config/in-game based strings. * While this method is hard to maintain, it's extremely efficient. It's approximately more than x5 times faster than * the normal RegEx + String Methods approach for both formatted and unformatted material names. * * @param name the enchantment name to format. * @return an enum name. * @since 1.0.0 */ @Nonnull private static String format(@Nonnull String name) { int len = name.length(); char[] chs = new char[len]; int count = 0; boolean appendUnderline = false; for (int i = 0; i < len; i++) { char ch = name.charAt(i); if (!appendUnderline && count != 0 && (ch == '-' || ch == ' ' || ch == '_') && chs[count] != '_') appendUnderline = true; else { if ((ch >= 'A' && ch <= 'Z') || (ch >= 'a' && ch <= 'z')) { if (appendUnderline) { chs[count++] = '_'; appendUnderline = false; } chs[count++] = (char) (ch & 0x5f); } } } return new String(chs, 0, count); } /** * Gets an enchantment from Vanilla and bukkit names. * There are also some aliases available. * * @param enchantment the name of the enchantment. * @return an enchantment. * @since 1.0.0 */ @Nonnull public static Optional matchXEnchantment(@Nonnull String enchantment) { if (enchantment == null || enchantment.isEmpty()) throw new IllegalArgumentException("Enchantment name cannot be null or empty"); return Optional.ofNullable(Data.NAMES.get(format(enchantment))); } /** * Gets an enchantment from Vanilla and bukkit names. * There are also some aliases available. * * @param enchantment the enchantment. * @return an enchantment. * @throws IllegalArgumentException may be thrown as an unexpected exception. * @since 1.0.0 */ @Nonnull @SuppressWarnings("deprecation") public static XEnchantment matchXEnchantment(@Nonnull Enchantment enchantment) { Objects.requireNonNull(enchantment, "Cannot parse XEnchantment of a null enchantment"); return Objects.requireNonNull(Data.NAMES.get(enchantment.getName()), () -> "Unsupported enchantment: " + enchantment.getName()); } /** * Gets the enchanted book of this enchantment. * * @param level the level of this enchantment. * @return an enchanted book. * @since 1.0.0 */ @Nonnull public ItemStack getBook(int level) { ItemStack book = new ItemStack(Material.ENCHANTED_BOOK); EnchantmentStorageMeta meta = (EnchantmentStorageMeta) book.getItemMeta(); meta.addStoredEnchant(this.enchantment, level, true); book.setItemMeta(meta); return book; } /** * Parse the Vanilla enchantment. * * @return a Vanilla enchantment. * @since 1.0.0 */ @Nullable public Enchantment getEnchant() { return this.enchantment; } /** * Checks if this enchantment is supported and registered in the current Minecraft version. *

* An invocation of this method yields exactly the same result as the expression: *

*

* {@link #getEnchant()} != null *
* * @return true if the current version has this enchantment, otherwise false. * @since 1.0.0 */ public boolean isSupported() { return enchantment != null; } /** * Checks if this enchantment is supported in the current version and * returns itself if yes. *

* In the other case, the alternate enchantment will get returned, * no matter if it is supported or not. * * @param alternateEnchantment the enchantment to get if this one is not supported. * @return this enchantment or the {@code alternateEnchantment} if not supported. */ @Nullable public XEnchantment or(@Nullable XEnchantment alternateEnchantment) { return isSupported() ? this : alternateEnchantment; } /** * In most cases you should be using {@link #name()} instead. * * @return a friendly readable string name. */ @Override @Nonnull public String toString() { return Arrays.stream(name().split("_")) .map(t -> t.charAt(0) + t.substring(1).toLowerCase()) .collect(Collectors.joining(" ")); } /** * Used for data that need to be accessed during enum initialization. * * @since 2.0.0 */ private static final class Data { private static final boolean ISFLAT, IS_SUPER_FLAT; private static final Map NAMES = new HashMap<>(); static { boolean flat, superFlat; try { Class namespacedKeyClass = Class.forName("org.bukkit.NamespacedKey"); Class enchantmentClass = Class.forName("org.bukkit.enchantments.Enchantment"); enchantmentClass.getDeclaredMethod("getByKey", namespacedKeyClass); flat = true; } catch (ClassNotFoundException | NoSuchMethodException ex) { flat = false; } try { Class.forName("org.bukkit.Registry"); superFlat = true; } catch (ClassNotFoundException ex) { superFlat = false; } ISFLAT = flat; IS_SUPER_FLAT = superFlat; } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy