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

net.badbird5907.blib.util.XMaterial Maven / Gradle / Ivy

/*
 * The MIT License (MIT)
 *
 * Copyright (c) 2018 Hex_27
 * Copyright (c) 2021 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 net.badbird5907.blib.util;

import com.google.common.base.Enums;
import com.google.common.cache.Cache;
import com.google.common.cache.CacheBuilder;
import com.google.common.cache.CacheLoader;
import com.google.common.cache.LoadingCache;
import org.apache.commons.lang.StringUtils;
import org.apache.commons.lang.Validate;
import org.apache.commons.lang.WordUtils;
import org.bukkit.Bukkit;
import org.bukkit.Material;
import org.bukkit.inventory.ItemStack;

import javax.annotation.Nonnull;
import javax.annotation.Nullable;
import java.util.*;
import java.util.concurrent.TimeUnit;
import java.util.regex.Pattern;
import java.util.regex.PatternSyntaxException;

/**
 * XMaterial - Data Values/Pre-flattening
* 1.13 and above as priority. *

* This class is mainly designed to support {@link ItemStack}. If you want to use it on blocks, you'll have to use * XBlock *

* Pre-flattening: https://minecraft.gamepedia.com/Java_Edition_data_values/Pre-flattening * Materials: https://hub.spigotmc.org/javadocs/spigot/org/bukkit/Material.html * Materials (1.12): https://helpch.at/docs/1.12.2/index.html?org/bukkit/Material.html * Material IDs: https://minecraft-ids.grahamedgecombe.com/ * Material Source Code: https://hub.spigotmc.org/stash/projects/SPIGOT/repos/bukkit/browse/src/main/java/org/bukkit/Material.java * XMaterial v1: https://www.spigotmc.org/threads/329630/ *

* This class will throw a "unsupported material" error if someone tries to use an item with an invalid data value which can only happen in 1.12 servers and below or when the * utility is missing a new material in that specific version. * To get an invalid item, (aka Missing Texture Block) you can use the command * /give @p minecraft:dirt 1 10 where 1 is the item amount, and 10 is the data value. The material {@link #DIRT} with a data value of {@code 10} doesn't exist. * * @author Crypto Morin * @version 10.0.0 * @see Material * @see ItemStack */ public enum XMaterial { ACACIA_BOAT("BOAT_ACACIA"), ACACIA_BUTTON("WOOD_BUTTON"), ACACIA_DOOR("ACACIA_DOOR", "ACACIA_DOOR_ITEM"), ACACIA_FENCE, ACACIA_FENCE_GATE, ACACIA_LEAVES("LEAVES_2"), ACACIA_LOG("LOG_2"), ACACIA_PLANKS(4, "WOOD"), ACACIA_PRESSURE_PLATE("WOOD_PLATE"), ACACIA_SAPLING(4, "SAPLING"), ACACIA_SIGN("SIGN_POST", "SIGN"), ACACIA_SLAB(4, "WOOD_DOUBLE_STEP", "WOOD_STEP", "WOODEN_SLAB"), ACACIA_STAIRS, ACACIA_TRAPDOOR("TRAP_DOOR"), ACACIA_WALL_SIGN("WALL_SIGN"), ACACIA_WOOD("LOG_2"), ACTIVATOR_RAIL, /** * https://minecraft.gamepedia.com/Air * {@link Material#isAir()} * * @see #VOID_AIR * @see #CAVE_AIR */ AIR, ALLIUM(2, "RED_ROSE"), AMETHYST_BLOCK, AMETHYST_CLUSTER, AMETHYST_SHARD, ANCIENT_DEBRIS(16), ANDESITE(5, "STONE"), ANDESITE_SLAB, ANDESITE_STAIRS, ANDESITE_WALL, ANVIL, APPLE, ARMOR_STAND, ARROW, ATTACHED_MELON_STEM(7, "MELON_STEM"), ATTACHED_PUMPKIN_STEM(7, "PUMPKIN_STEM"), AXOLOTL_BUCKET, AXOLOTL_SPAWN_EGG, AZALEA, AZALEA_LEAVES, AZURE_BLUET(3, "RED_ROSE"), BAKED_POTATO, BAMBOO(0, 14), BAMBOO_SAPLING(14), BARREL(0, 14), BARRIER, BASALT(16), BAT_SPAWN_EGG(65, "MONSTER_EGG"), BEACON, BEDROCK, BEEF("RAW_BEEF"), BEEHIVE(15), /** * Beetroot is a known material in pre-1.13 */ BEETROOT("BEETROOT_BLOCK"), BEETROOTS("BEETROOT"), BEETROOT_SEEDS, BEETROOT_SOUP, BEE_NEST(15), BEE_SPAWN_EGG(15), BELL(14), BIG_DRIPLEAF, BIG_DRIPLEAF_STEM, BIRCH_BOAT("BOAT_BIRCH"), BIRCH_BUTTON("WOOD_BUTTON"), BIRCH_DOOR("BIRCH_DOOR", "BIRCH_DOOR_ITEM"), BIRCH_FENCE, BIRCH_FENCE_GATE, BIRCH_LEAVES(2, "LEAVES"), BIRCH_LOG(2, "LOG"), BIRCH_PLANKS(2, "WOOD"), BIRCH_PRESSURE_PLATE("WOOD_PLATE"), BIRCH_SAPLING(2, "SAPLING"), BIRCH_SIGN("SIGN_POST", "SIGN"), BIRCH_SLAB(2, "WOOD_DOUBLE_STEP", "WOOD_STEP", "WOODEN_SLAB"), BIRCH_STAIRS("BIRCH_WOOD_STAIRS"), BIRCH_TRAPDOOR("TRAP_DOOR"), BIRCH_WALL_SIGN("WALL_SIGN"), BIRCH_WOOD(2, "LOG"), BLACKSTONE(16), BLACKSTONE_SLAB(16), BLACKSTONE_STAIRS(16), BLACKSTONE_WALL(16), BLACK_BANNER("STANDING_BANNER", "BANNER"), BLACK_BED(15, "BED_BLOCK", "BED"), BLACK_CANDLE, BLACK_CANDLE_CAKE, BLACK_CARPET(15, "CARPET"), BLACK_CONCRETE(15, "CONCRETE"), BLACK_CONCRETE_POWDER(15, "CONCRETE_POWDER"), BLACK_DYE(0, 14, "INK_SACK", "INK_SAC"), BLACK_GLAZED_TERRACOTTA(15, 12), BLACK_SHULKER_BOX, BLACK_STAINED_GLASS(15, "STAINED_GLASS"), BLACK_STAINED_GLASS_PANE(15, "STAINED_GLASS_PANE"), BLACK_TERRACOTTA(15, "STAINED_CLAY"), BLACK_WALL_BANNER("WALL_BANNER"), BLACK_WOOL(15, "WOOL"), BLAST_FURNACE(0, 14), BLAZE_POWDER, BLAZE_ROD, BLAZE_SPAWN_EGG(61, "MONSTER_EGG"), BLUE_BANNER(4, "STANDING_BANNER", "BANNER"), BLUE_BED(11, "BED_BLOCK", "BED"), BLUE_CANDLE, BLUE_CANDLE_CAKE, BLUE_CARPET(11, "CARPET"), BLUE_CONCRETE(11, "CONCRETE"), BLUE_CONCRETE_POWDER(11, "CONCRETE_POWDER"), BLUE_DYE(4, "INK_SACK", "LAPIS_LAZULI"), BLUE_GLAZED_TERRACOTTA(11, 12), BLUE_ICE(0, 13), BLUE_ORCHID(1, "RED_ROSE"), BLUE_SHULKER_BOX, BLUE_STAINED_GLASS(11, "STAINED_GLASS"), BLUE_STAINED_GLASS_PANE(11, "THIN_GLASS", "STAINED_GLASS_PANE"), BLUE_TERRACOTTA(11, "STAINED_CLAY"), BLUE_WALL_BANNER(4, "WALL_BANNER"), BLUE_WOOL(11, "WOOL"), BONE, BONE_BLOCK, BONE_MEAL(15, "INK_SACK"), BOOK, BOOKSHELF, BOW, BOWL, BRAIN_CORAL(13), BRAIN_CORAL_BLOCK(13), BRAIN_CORAL_FAN(13), BRAIN_CORAL_WALL_FAN, BREAD, BREWING_STAND("BREWING_STAND", "BREWING_STAND_ITEM"), BRICK("CLAY_BRICK"), BRICKS("BRICKS", "BRICK"), BRICK_SLAB(4, "STEP"), BRICK_STAIRS, BRICK_WALL, BROWN_BANNER(3, "STANDING_BANNER", "BANNER"), BROWN_BED(12, "BED_BLOCK", "BED"), BROWN_CANDLE, BROWN_CANDLE_CAKE, BROWN_CARPET(12, "CARPET"), BROWN_CONCRETE(12, "CONCRETE"), BROWN_CONCRETE_POWDER(12, "CONCRETE_POWDER"), BROWN_DYE(3, "INK_SACK", "DYE", "COCOA_BEANS"), BROWN_GLAZED_TERRACOTTA(12, 12), BROWN_MUSHROOM, BROWN_MUSHROOM_BLOCK("BROWN_MUSHROOM", "HUGE_MUSHROOM_1"), BROWN_SHULKER_BOX, BROWN_STAINED_GLASS(12, "STAINED_GLASS"), BROWN_STAINED_GLASS_PANE(12, "THIN_GLASS", "STAINED_GLASS_PANE"), BROWN_TERRACOTTA(12, "STAINED_CLAY"), BROWN_WALL_BANNER(3, "WALL_BANNER"), BROWN_WOOL(12, "WOOL"), BUBBLE_COLUMN(13), BUBBLE_CORAL(13), BUBBLE_CORAL_BLOCK(13), BUBBLE_CORAL_FAN(13), BUBBLE_CORAL_WALL_FAN, BUCKET, BUDDING_AMETHYST, BUNDLE, CACTUS, CAKE("CAKE_BLOCK"), CALCITE, CAMPFIRE(14), CANDLE, CANDLE_CAKE, CARROT("CARROT_ITEM"), CARROTS("CARROT"), CARROT_ON_A_STICK("CARROT_STICK"), CARTOGRAPHY_TABLE(0, 14), CARVED_PUMPKIN(1, 13), CAT_SPAWN_EGG, CAULDRON("CAULDRON", "CAULDRON_ITEM"), /** * 1.13 tag is not added because it's the same thing as {@link #AIR} * * @see #VOID_AIR */ CAVE_AIR("AIR"), CAVE_SPIDER_SPAWN_EGG(59, "MONSTER_EGG"), CAVE_VINES, CAVE_VINES_PLANT, CHAIN(16), CHAINMAIL_BOOTS, CHAINMAIL_CHESTPLATE, CHAINMAIL_HELMET, CHAINMAIL_LEGGINGS, CHAIN_COMMAND_BLOCK("COMMAND", "COMMAND_CHAIN"), CHARCOAL(1, "COAL"), CHEST("LOCKED_CHEST"), CHEST_MINECART("STORAGE_MINECART"), CHICKEN("RAW_CHICKEN"), CHICKEN_SPAWN_EGG(93, "MONSTER_EGG"), CHIPPED_ANVIL(1, "ANVIL"), CHISELED_DEEPSLATE, CHISELED_NETHER_BRICKS(1, "NETHER_BRICKS"), CHISELED_POLISHED_BLACKSTONE(0, 16, "POLISHED_BLACKSTONE"), CHISELED_QUARTZ_BLOCK(1, "QUARTZ_BLOCK"), CHISELED_RED_SANDSTONE(1, "RED_SANDSTONE"), CHISELED_SANDSTONE(1, "SANDSTONE"), CHISELED_STONE_BRICKS(3, "SMOOTH_BRICK"), CHORUS_FLOWER(0, 9), CHORUS_FRUIT(0, 9), CHORUS_PLANT(0, 9), CLAY("HARD_CLAY"), CLAY_BALL, CLOCK("WATCH"), COAL, COAL_BLOCK, COAL_ORE, COARSE_DIRT(1, "DIRT"), COBBLED_DEEPSLATE, COBBLED_DEEPSLATE_SLAB, COBBLED_DEEPSLATE_STAIRS, COBBLED_DEEPSLATE_WALL, COBBLESTONE, COBBLESTONE_SLAB(3, "STEP"), COBBLESTONE_STAIRS, COBBLESTONE_WALL("COBBLE_WALL"), COBWEB("WEB"), COCOA(15), COCOA_BEANS(3, "INK_SACK"), COD("RAW_FISH"), COD_BUCKET(0, 13), COD_SPAWN_EGG(0, 13), COMMAND_BLOCK("COMMAND"), COMMAND_BLOCK_MINECART("COMMAND_MINECART"), /** * Unlike redstone torch and redstone lamp... neither REDTONE_COMPARATOR_OFF nor REDSTONE_COMPARATOR_ON * are items. REDSTONE_COMPARATOR is. * * @see #REDSTONE_TORCH * @see #REDSTONE_LAMP */ COMPARATOR("REDSTONE_COMPARATOR_OFF", "REDSTONE_COMPARATOR_ON", "REDSTONE_COMPARATOR"), COMPASS, COMPOSTER(0, 14), CONDUIT(0, 13, "BEACON"), COOKED_BEEF, COOKED_CHICKEN, COOKED_COD("COOKED_FISH"), COOKED_MUTTON, COOKED_PORKCHOP("PORK", "GRILLED_PORK"), COOKED_RABBIT, COOKED_SALMON(1, "COOKED_FISH"), COOKIE, COPPER_BLOCK, COPPER_INGOT, COPPER_ORE, CORNFLOWER(4, 14), COW_SPAWN_EGG(92, "MONSTER_EGG"), CRACKED_DEEPSLATE_BRICKS, CRACKED_DEEPSLATE_TILES, CRACKED_NETHER_BRICKS(2, "NETHER_BRICKS"), CRACKED_POLISHED_BLACKSTONE_BRICKS(0, 16, "POLISHED_BLACKSTONE_BRICKS"), CRACKED_STONE_BRICKS(2, "SMOOTH_BRICK"), CRAFTING_TABLE("WORKBENCH"), CREEPER_BANNER_PATTERN, CREEPER_HEAD(4, "SKULL", "SKULL_ITEM"), CREEPER_SPAWN_EGG(50, "MONSTER_EGG"), CREEPER_WALL_HEAD(4, "SKULL", "SKULL_ITEM"), CRIMSON_BUTTON(16), CRIMSON_DOOR(16), CRIMSON_FENCE(16), CRIMSON_FENCE_GATE(16), CRIMSON_FUNGUS(16), CRIMSON_HYPHAE(16), CRIMSON_NYLIUM(16), CRIMSON_PLANKS(16), CRIMSON_PRESSURE_PLATE(16), CRIMSON_ROOTS(16), CRIMSON_SIGN(0, 16, "SIGN_POST"), CRIMSON_SLAB(16), CRIMSON_STAIRS(16), CRIMSON_STEM(16), CRIMSON_TRAPDOOR(16), CRIMSON_WALL_SIGN(0, 16, "WALL_SIGN"), CROSSBOW, CRYING_OBSIDIAN(16), CUT_COPPER, CUT_COPPER_SLAB, CUT_COPPER_STAIRS, CUT_RED_SANDSTONE(13), CUT_RED_SANDSTONE_SLAB("STONE_SLAB2"), CUT_SANDSTONE(13), CUT_SANDSTONE_SLAB("STEP"), CYAN_BANNER(6, "STANDING_BANNER", "BANNER"), CYAN_BED(9, "BED_BLOCK", "BED"), CYAN_CANDLE, CYAN_CANDLE_CAKE, CYAN_CARPET(9, "CARPET"), CYAN_CONCRETE(9, "CONCRETE"), CYAN_CONCRETE_POWDER(9, "CONCRETE_POWDER"), CYAN_DYE(6, "INK_SACK"), CYAN_GLAZED_TERRACOTTA(9, 12), CYAN_SHULKER_BOX, CYAN_STAINED_GLASS(9, "STAINED_GLASS"), CYAN_STAINED_GLASS_PANE(9, "STAINED_GLASS_PANE"), CYAN_TERRACOTTA(9, "STAINED_CLAY"), CYAN_WALL_BANNER(6, "WALL_BANNER"), CYAN_WOOL(9, "WOOL"), DAMAGED_ANVIL(2, "ANVIL"), DANDELION("YELLOW_FLOWER"), DARK_OAK_BOAT("BOAT_DARK_OAK"), DARK_OAK_BUTTON("WOOD_BUTTON"), DARK_OAK_DOOR("DARK_OAK_DOOR", "DARK_OAK_DOOR_ITEM"), DARK_OAK_FENCE, DARK_OAK_FENCE_GATE, DARK_OAK_LEAVES(4, "LEAVES", "LEAVES_2"), DARK_OAK_LOG(1, "LOG_2"), DARK_OAK_PLANKS(5, "WOOD"), DARK_OAK_PRESSURE_PLATE("WOOD_PLATE"), DARK_OAK_SAPLING(5, "SAPLING"), DARK_OAK_SIGN("SIGN_POST", "SIGN"), DARK_OAK_SLAB(5, "WOOD_DOUBLE_STEP", "WOOD_STEP", "WOODEN_SLAB"), DARK_OAK_STAIRS, DARK_OAK_TRAPDOOR("TRAP_DOOR"), DARK_OAK_WALL_SIGN("WALL_SIGN"), DARK_OAK_WOOD(1, "LOG_2"), DARK_PRISMARINE(1, "PRISMARINE"), DARK_PRISMARINE_SLAB(13), DARK_PRISMARINE_STAIRS(13), DAYLIGHT_DETECTOR("DAYLIGHT_DETECTOR_INVERTED"), DEAD_BRAIN_CORAL(13), DEAD_BRAIN_CORAL_BLOCK(13), DEAD_BRAIN_CORAL_FAN(13), DEAD_BRAIN_CORAL_WALL_FAN(13), DEAD_BUBBLE_CORAL(13), DEAD_BUBBLE_CORAL_BLOCK(13), DEAD_BUBBLE_CORAL_FAN(13), DEAD_BUBBLE_CORAL_WALL_FAN(13), DEAD_BUSH, DEAD_FIRE_CORAL(13), DEAD_FIRE_CORAL_BLOCK(13), DEAD_FIRE_CORAL_FAN(13), DEAD_FIRE_CORAL_WALL_FAN(13), DEAD_HORN_CORAL(13), DEAD_HORN_CORAL_BLOCK(13), DEAD_HORN_CORAL_FAN(13), DEAD_HORN_CORAL_WALL_FAN(13), DEAD_TUBE_CORAL(13), DEAD_TUBE_CORAL_BLOCK(13), DEAD_TUBE_CORAL_FAN(13), DEAD_TUBE_CORAL_WALL_FAN(13), DEBUG_STICK(0, 13), DEEPSLATE, DEEPSLATE_BRICKS, DEEPSLATE_BRICK_SLAB, DEEPSLATE_BRICK_STAIRS, DEEPSLATE_BRICK_WALL, DEEPSLATE_COAL_ORE, DEEPSLATE_COPPER_ORE, DEEPSLATE_DIAMOND_ORE, DEEPSLATE_EMERALD_ORE, DEEPSLATE_GOLD_ORE, DEEPSLATE_IRON_ORE, DEEPSLATE_LAPIS_ORE, DEEPSLATE_REDSTONE_ORE, DEEPSLATE_TILES, DEEPSLATE_TILE_SLAB, DEEPSLATE_TILE_STAIRS, DEEPSLATE_TILE_WALL, DETECTOR_RAIL, DIAMOND, DIAMOND_AXE, DIAMOND_BLOCK, DIAMOND_BOOTS, DIAMOND_CHESTPLATE, DIAMOND_HELMET, DIAMOND_HOE, DIAMOND_HORSE_ARMOR("DIAMOND_BARDING"), DIAMOND_LEGGINGS, DIAMOND_ORE, DIAMOND_PICKAXE, DIAMOND_SHOVEL("DIAMOND_SPADE"), DIAMOND_SWORD, DIORITE(3, "STONE"), DIORITE_SLAB, DIORITE_STAIRS, DIORITE_WALL, DIRT, /** * Changed in 1.17 */ DIRT_PATH("GRASS_PATH"), DISPENSER, DOLPHIN_SPAWN_EGG(0, 13), DONKEY_SPAWN_EGG(32, "MONSTER_EGG"), DRAGON_BREATH("DRAGONS_BREATH"), DRAGON_EGG, DRAGON_HEAD(5, 9, "SKULL", "SKULL_ITEM"), DRAGON_WALL_HEAD(5, "SKULL", "SKULL_ITEM"), DRIED_KELP(13), DRIED_KELP_BLOCK(13), DRIPSTONE_BLOCK, DROPPER, DROWNED_SPAWN_EGG(0, 13), EGG, ELDER_GUARDIAN_SPAWN_EGG(4, "MONSTER_EGG"), ELYTRA, EMERALD, EMERALD_BLOCK, EMERALD_ORE, ENCHANTED_BOOK, ENCHANTED_GOLDEN_APPLE(1, "GOLDEN_APPLE"), ENCHANTING_TABLE("ENCHANTMENT_TABLE"), ENDERMAN_SPAWN_EGG(58, "MONSTER_EGG"), ENDERMITE_SPAWN_EGG(67, "MONSTER_EGG"), ENDER_CHEST, ENDER_EYE("EYE_OF_ENDER"), ENDER_PEARL, END_CRYSTAL, END_GATEWAY(0, 9), END_PORTAL("ENDER_PORTAL"), END_PORTAL_FRAME("ENDER_PORTAL_FRAME"), END_ROD(0, 9), END_STONE("ENDER_STONE"), END_STONE_BRICKS("END_BRICKS"), END_STONE_BRICK_SLAB(6, "STEP"), END_STONE_BRICK_STAIRS("SMOOTH_STAIRS"), END_STONE_BRICK_WALL, EVOKER_SPAWN_EGG(34, "MONSTER_EGG"), EXPERIENCE_BOTTLE("EXP_BOTTLE"), EXPOSED_COPPER, EXPOSED_CUT_COPPER, EXPOSED_CUT_COPPER_SLAB, EXPOSED_CUT_COPPER_STAIRS, FARMLAND("SOIL"), FEATHER, FERMENTED_SPIDER_EYE, FERN(2, "LONG_GRASS"), /** * For some reasons filled map items are really special. * Their data value starts from 0 and every time a player * creates a new map that maps data value increases. * https://github.com/CryptoMorin/XSeries/issues/91 */ FILLED_MAP("MAP"), FIRE, FIREWORK_ROCKET("FIREWORK"), FIREWORK_STAR("FIREWORK_CHARGE"), FIRE_CHARGE("FIREBALL"), FIRE_CORAL(13), FIRE_CORAL_BLOCK(13), FIRE_CORAL_FAN(13), FIRE_CORAL_WALL_FAN, FISHING_ROD, FLETCHING_TABLE(0, 14), FLINT, FLINT_AND_STEEL, FLOWERING_AZALEA, FLOWERING_AZALEA_LEAVES, FLOWER_BANNER_PATTERN, FLOWER_POT("FLOWER_POT", "FLOWER_POT_ITEM"), FOX_SPAWN_EGG(14), /** * This special material cannot be obtained as an item. */ FROSTED_ICE(0, 9), FURNACE("BURNING_FURNACE"), FURNACE_MINECART("POWERED_MINECART"), GHAST_SPAWN_EGG(56, "MONSTER_EGG"), GHAST_TEAR, GILDED_BLACKSTONE(16), GLASS, GLASS_BOTTLE, GLASS_PANE("THIN_GLASS"), GLISTERING_MELON_SLICE("SPECKLED_MELON"), GLOBE_BANNER_PATTERN, GLOWSTONE, GLOWSTONE_DUST, GLOW_BERRIES, GLOW_INK_SAC, GLOW_ITEM_FRAME, GLOW_LICHEN, GLOW_SQUID_SPAWN_EGG, GOAT_SPAWN_EGG, GOLDEN_APPLE, GOLDEN_AXE("GOLD_AXE"), GOLDEN_BOOTS("GOLD_BOOTS"), GOLDEN_CARROT, GOLDEN_CHESTPLATE("GOLD_CHESTPLATE"), GOLDEN_HELMET("GOLD_HELMET"), GOLDEN_HOE("GOLD_HOE"), GOLDEN_HORSE_ARMOR("GOLD_BARDING"), GOLDEN_LEGGINGS("GOLD_LEGGINGS"), GOLDEN_PICKAXE("GOLD_PICKAXE"), GOLDEN_SHOVEL("GOLD_SPADE"), GOLDEN_SWORD("GOLD_SWORD"), GOLD_BLOCK, GOLD_INGOT, GOLD_NUGGET, GOLD_ORE, GRANITE(1, "STONE"), GRANITE_SLAB, GRANITE_STAIRS, GRANITE_WALL, GRASS(1, "LONG_GRASS"), GRASS_BLOCK("GRASS"), GRAVEL, GRAY_BANNER(8, "STANDING_BANNER", "BANNER"), GRAY_BED(7, "BED_BLOCK", "BED"), GRAY_CANDLE, GRAY_CANDLE_CAKE, GRAY_CARPET(7, "CARPET"), GRAY_CONCRETE(7, "CONCRETE"), GRAY_CONCRETE_POWDER(7, "CONCRETE_POWDER"), GRAY_DYE(8, "INK_SACK"), GRAY_GLAZED_TERRACOTTA(7, 12), GRAY_SHULKER_BOX, GRAY_STAINED_GLASS(7, "STAINED_GLASS"), GRAY_STAINED_GLASS_PANE(7, "THIN_GLASS", "STAINED_GLASS_PANE"), GRAY_TERRACOTTA(7, "STAINED_CLAY"), GRAY_WALL_BANNER(8, "WALL_BANNER"), GRAY_WOOL(7, "WOOL"), GREEN_BANNER(2, "STANDING_BANNER", "BANNER"), GREEN_BED(13, "BED_BLOCK", "BED"), GREEN_CANDLE, GREEN_CANDLE_CAKE, GREEN_CARPET(13, "CARPET"), GREEN_CONCRETE(13, "CONCRETE"), GREEN_CONCRETE_POWDER(13, "CONCRETE_POWDER"), GREEN_DYE(2, "INK_SACK", "CACTUS_GREEN"), GREEN_GLAZED_TERRACOTTA(13, 12), GREEN_SHULKER_BOX, GREEN_STAINED_GLASS(13, "STAINED_GLASS"), GREEN_STAINED_GLASS_PANE(13, "THIN_GLASS", "STAINED_GLASS_PANE"), GREEN_TERRACOTTA(13, "STAINED_CLAY"), GREEN_WALL_BANNER(2, "WALL_BANNER"), GREEN_WOOL(13, "WOOL"), GRINDSTONE(0, 14), GUARDIAN_SPAWN_EGG(68, "MONSTER_EGG"), GUNPOWDER("SULPHUR"), HANGING_ROOTS, HAY_BLOCK, HEART_OF_THE_SEA(13), HEAVY_WEIGHTED_PRESSURE_PLATE("IRON_PLATE"), HOGLIN_SPAWN_EGG(0, 16, "MONSTER_EGG"), HONEYCOMB(15), HONEYCOMB_BLOCK(15), HONEY_BLOCK(0, 15), HONEY_BOTTLE(0, 15), HOPPER, HOPPER_MINECART, HORN_CORAL(13), HORN_CORAL_BLOCK(13), HORN_CORAL_FAN(13), HORN_CORAL_WALL_FAN, HORSE_SPAWN_EGG(100, "MONSTER_EGG"), HUSK_SPAWN_EGG(23, "MONSTER_EGG"), ICE, INFESTED_CHISELED_STONE_BRICKS(5, "MONSTER_EGGS"), INFESTED_COBBLESTONE(1, "MONSTER_EGGS"), INFESTED_CRACKED_STONE_BRICKS(4, "MONSTER_EGGS"), INFESTED_DEEPSLATE, INFESTED_MOSSY_STONE_BRICKS(3, "MONSTER_EGGS"), INFESTED_STONE("MONSTER_EGGS"), INFESTED_STONE_BRICKS(2, "MONSTER_EGGS"), /** * We will only add "INK_SAC" for {@link #BLACK_DYE} since it's * the only material (linked with this material) that is added * after 1.13, which means it can use both INK_SACK and INK_SAC. */ INK_SAC("INK_SACK"), IRON_AXE, IRON_BARS("IRON_FENCE"), IRON_BLOCK, IRON_BOOTS, IRON_CHESTPLATE, IRON_DOOR("IRON_DOOR_BLOCK"), IRON_HELMET, IRON_HOE, IRON_HORSE_ARMOR("IRON_BARDING"), IRON_INGOT, IRON_LEGGINGS, IRON_NUGGET, IRON_ORE, IRON_PICKAXE, IRON_SHOVEL("IRON_SPADE"), IRON_SWORD, IRON_TRAPDOOR, ITEM_FRAME, JACK_O_LANTERN, JIGSAW(0, 14), JUKEBOX, JUNGLE_BOAT("BOAT_JUNGLE"), JUNGLE_BUTTON("WOOD_BUTTON"), JUNGLE_DOOR("JUNGLE_DOOR", "JUNGLE_DOOR_ITEM"), JUNGLE_FENCE, JUNGLE_FENCE_GATE, JUNGLE_LEAVES(3, "LEAVES"), JUNGLE_LOG(3, "LOG"), JUNGLE_PLANKS(3, "WOOD"), JUNGLE_PRESSURE_PLATE("WOOD_PLATE"), JUNGLE_SAPLING(3, "SAPLING"), JUNGLE_SIGN("SIGN_POST", "SIGN"), JUNGLE_SLAB(3, "WOOD_DOUBLE_STEP", "WOOD_STEP", "WOODEN_SLAB"), JUNGLE_STAIRS("JUNGLE_WOOD_STAIRS"), JUNGLE_TRAPDOOR("TRAP_DOOR"), JUNGLE_WALL_SIGN("WALL_SIGN"), JUNGLE_WOOD(3, "LOG"), KELP(13), KELP_PLANT(13), KNOWLEDGE_BOOK(0, 12, "BOOK"), LADDER, LANTERN(0, 14), LAPIS_BLOCK, LAPIS_LAZULI(4, "INK_SACK"), LAPIS_ORE, LARGE_AMETHYST_BUD, LARGE_FERN(3, "DOUBLE_PLANT"), LAVA("STATIONARY_LAVA"), LAVA_BUCKET, LAVA_CAULDRON, LEAD("LEASH"), LEATHER, LEATHER_BOOTS, LEATHER_CHESTPLATE, LEATHER_HELMET, LEATHER_HORSE_ARMOR(0, 14, "IRON_HORSE_ARMOR"), LEATHER_LEGGINGS, LECTERN(0, 14), LEVER, LIGHT, LIGHTNING_ROD, LIGHT_BLUE_BANNER(12, "STANDING_BANNER", "BANNER"), LIGHT_BLUE_BED(3, "BED_BLOCK", "BED"), LIGHT_BLUE_CANDLE, LIGHT_BLUE_CANDLE_CAKE, LIGHT_BLUE_CARPET(3, "CARPET"), LIGHT_BLUE_CONCRETE(3, "CONCRETE"), LIGHT_BLUE_CONCRETE_POWDER(3, "CONCRETE_POWDER"), LIGHT_BLUE_DYE(12, "INK_SACK"), LIGHT_BLUE_GLAZED_TERRACOTTA(3, 12), LIGHT_BLUE_SHULKER_BOX, LIGHT_BLUE_STAINED_GLASS(3, "STAINED_GLASS"), LIGHT_BLUE_STAINED_GLASS_PANE(3, "THIN_GLASS", "STAINED_GLASS_PANE"), LIGHT_BLUE_TERRACOTTA(3, "STAINED_CLAY"), LIGHT_BLUE_WALL_BANNER(12, "WALL_BANNER", "STANDING_BANNER", "BANNER"), LIGHT_BLUE_WOOL(3, "WOOL"), LIGHT_GRAY_BANNER(7, "STANDING_BANNER", "BANNER"), LIGHT_GRAY_BED(8, "BED_BLOCK", "BED"), LIGHT_GRAY_CANDLE, LIGHT_GRAY_CANDLE_CAKE, LIGHT_GRAY_CARPET(8, "CARPET"), LIGHT_GRAY_CONCRETE(8, "CONCRETE"), LIGHT_GRAY_CONCRETE_POWDER(8, "CONCRETE_POWDER"), LIGHT_GRAY_DYE(7, "INK_SACK"), /** * Renamed to SILVER_GLAZED_TERRACOTTA in 1.12 * Renamed to LIGHT_GRAY_GLAZED_TERRACOTTA in 1.14 */ LIGHT_GRAY_GLAZED_TERRACOTTA(0, 12, "STAINED_CLAY", "LIGHT_GRAY_TERRACOTTA", "SILVER_GLAZED_TERRACOTTA"), LIGHT_GRAY_SHULKER_BOX("SILVER_SHULKER_BOX"), LIGHT_GRAY_STAINED_GLASS(8, "STAINED_GLASS"), LIGHT_GRAY_STAINED_GLASS_PANE(8, "THIN_GLASS", "STAINED_GLASS_PANE"), LIGHT_GRAY_TERRACOTTA(8, "STAINED_CLAY"), LIGHT_GRAY_WALL_BANNER(7, "WALL_BANNER"), LIGHT_GRAY_WOOL(8, "WOOL"), LIGHT_WEIGHTED_PRESSURE_PLATE("GOLD_PLATE"), LILAC(1, "DOUBLE_PLANT"), LILY_OF_THE_VALLEY(15, 14), LILY_PAD("WATER_LILY"), LIME_BANNER(10, "STANDING_BANNER", "BANNER"), LIME_BED(5, "BED_BLOCK", "BED"), LIME_CANDLE, LIME_CANDLE_CAKE, LIME_CARPET(5, "CARPET"), LIME_CONCRETE(5, "CONCRETE"), LIME_CONCRETE_POWDER(5, "CONCRETE_POWDER"), LIME_DYE(10, "INK_SACK"), LIME_GLAZED_TERRACOTTA(5, 12), LIME_SHULKER_BOX, LIME_STAINED_GLASS(5, "STAINED_GLASS"), LIME_STAINED_GLASS_PANE(5, "STAINED_GLASS_PANE"), LIME_TERRACOTTA(5, "STAINED_CLAY"), LIME_WALL_BANNER(10, "WALL_BANNER"), LIME_WOOL(5, "WOOL"), LINGERING_POTION, LLAMA_SPAWN_EGG(103, "MONSTER_EGG"), LODESTONE(16), LOOM(14), MAGENTA_BANNER(13, "STANDING_BANNER", "BANNER"), MAGENTA_BED(2, "BED_BLOCK", "BED"), MAGENTA_CANDLE, MAGENTA_CANDLE_CAKE, MAGENTA_CARPET(2, "CARPET"), MAGENTA_CONCRETE(2, "CONCRETE"), MAGENTA_CONCRETE_POWDER(2, "CONCRETE_POWDER"), MAGENTA_DYE(13, "INK_SACK"), MAGENTA_GLAZED_TERRACOTTA(2, 12), MAGENTA_SHULKER_BOX, MAGENTA_STAINED_GLASS(2, "STAINED_GLASS"), MAGENTA_STAINED_GLASS_PANE(2, "THIN_GLASS", "STAINED_GLASS_PANE"), MAGENTA_TERRACOTTA(2, "STAINED_CLAY"), MAGENTA_WALL_BANNER(13, "WALL_BANNER"), MAGENTA_WOOL(2, "WOOL"), MAGMA_BLOCK(0, 10, "MAGMA"), MAGMA_CREAM, MAGMA_CUBE_SPAWN_EGG(62, "MONSTER_EGG"), /** * Adding this to the duplicated list will give you a filled map * for 1.13+ versions and removing it from duplicated list will * still give you a filled map in -1.12 versions. * Since higher versions are our priority I'll keep 1.13+ support * until I can come up with something to fix it. */ MAP("EMPTY_MAP"), MEDIUM_AMETHYST_BUD, MELON("MELON_BLOCK"), MELON_SEEDS, MELON_SLICE("MELON"), MELON_STEM, MILK_BUCKET, MINECART, MOJANG_BANNER_PATTERN, MOOSHROOM_SPAWN_EGG(96, "MONSTER_EGG"), MOSSY_COBBLESTONE, MOSSY_COBBLESTONE_SLAB(3, "STEP"), MOSSY_COBBLESTONE_STAIRS, MOSSY_COBBLESTONE_WALL(1, "COBBLE_WALL", "COBBLESTONE_WALL"), MOSSY_STONE_BRICKS(1, "SMOOTH_BRICK"), MOSSY_STONE_BRICK_SLAB(5, "STEP"), MOSSY_STONE_BRICK_STAIRS("SMOOTH_STAIRS"), MOSSY_STONE_BRICK_WALL, MOSS_BLOCK, MOSS_CARPET, MOVING_PISTON("PISTON_MOVING_PIECE"), MULE_SPAWN_EGG(32, "MONSTER_EGG"), MUSHROOM_STEM("BROWN_MUSHROOM"), MUSHROOM_STEW("MUSHROOM_SOUP"), MUSIC_DISC_11("GOLD_RECORD"), MUSIC_DISC_13("GREEN_RECORD"), MUSIC_DISC_BLOCKS("RECORD_3"), MUSIC_DISC_CAT("RECORD_4"), MUSIC_DISC_CHIRP("RECORD_5"), MUSIC_DISC_FAR("RECORD_6"), MUSIC_DISC_MALL("RECORD_7"), MUSIC_DISC_MELLOHI("RECORD_8"), MUSIC_DISC_PIGSTEP(16), MUSIC_DISC_STAL("RECORD_9"), MUSIC_DISC_STRAD("RECORD_10"), MUSIC_DISC_WAIT("RECORD_11"), MUSIC_DISC_WARD("RECORD_12"), MUTTON, MYCELIUM("MYCEL"), NAME_TAG, NAUTILUS_SHELL(13), NETHERITE_AXE(16), NETHERITE_BLOCK(16), NETHERITE_BOOTS(16), NETHERITE_CHESTPLATE(16), NETHERITE_HELMET(16), NETHERITE_HOE(16), NETHERITE_INGOT(16), NETHERITE_LEGGINGS(16), NETHERITE_PICKAXE(16), NETHERITE_SCRAP(16), NETHERITE_SHOVEL(16), NETHERITE_SWORD(16), NETHERRACK, NETHER_BRICK("NETHER_BRICK_ITEM"), NETHER_BRICKS("NETHER_BRICK"), NETHER_BRICK_FENCE("NETHER_FENCE"), NETHER_BRICK_SLAB(6, "STEP"), NETHER_BRICK_STAIRS, NETHER_BRICK_WALL, NETHER_GOLD_ORE(16), NETHER_PORTAL("PORTAL"), NETHER_QUARTZ_ORE("QUARTZ_ORE"), NETHER_SPROUTS(16), NETHER_STAR, /** * Just like mentioned in https://minecraft.gamepedia.com/Nether_Wart * Nether wart is also known as nether stalk in the code. * NETHER_STALK is the planted state of nether warts. */ NETHER_WART("NETHER_WARTS", "NETHER_STALK"), NETHER_WART_BLOCK, NOTE_BLOCK, OAK_BOAT("BOAT"), OAK_BUTTON("WOOD_BUTTON"), OAK_DOOR("WOODEN_DOOR", "WOOD_DOOR"), OAK_FENCE("FENCE"), OAK_FENCE_GATE("FENCE_GATE"), OAK_LEAVES("LEAVES"), OAK_LOG("LOG"), OAK_PLANKS("WOOD"), OAK_PRESSURE_PLATE("WOOD_PLATE"), OAK_SAPLING("SAPLING"), OAK_SIGN("SIGN_POST", "SIGN"), OAK_SLAB("WOOD_DOUBLE_STEP", "WOOD_STEP", "WOODEN_SLAB"), OAK_STAIRS("WOOD_STAIRS"), OAK_TRAPDOOR("TRAP_DOOR"), OAK_WALL_SIGN("WALL_SIGN"), OAK_WOOD("LOG"), OBSERVER, OBSIDIAN, OCELOT_SPAWN_EGG(98, "MONSTER_EGG"), ORANGE_BANNER(14, "STANDING_BANNER", "BANNER"), ORANGE_BED(1, "BED_BLOCK", "BED"), ORANGE_CANDLE, ORANGE_CANDLE_CAKE, ORANGE_CARPET(1, "CARPET"), ORANGE_CONCRETE(1, "CONCRETE"), ORANGE_CONCRETE_POWDER(1, "CONCRETE_POWDER"), ORANGE_DYE(14, "INK_SACK"), ORANGE_GLAZED_TERRACOTTA(1, 12), ORANGE_SHULKER_BOX, ORANGE_STAINED_GLASS(1, "STAINED_GLASS"), ORANGE_STAINED_GLASS_PANE(1, "STAINED_GLASS_PANE"), ORANGE_TERRACOTTA(1, "STAINED_CLAY"), ORANGE_TULIP(5, "RED_ROSE"), ORANGE_WALL_BANNER(14, "WALL_BANNER"), ORANGE_WOOL(1, "WOOL"), OXEYE_DAISY(8, "RED_ROSE"), OXIDIZED_COPPER, OXIDIZED_CUT_COPPER, OXIDIZED_CUT_COPPER_SLAB, OXIDIZED_CUT_COPPER_STAIRS, PACKED_ICE, PAINTING, PANDA_SPAWN_EGG(14), PAPER, PARROT_SPAWN_EGG(105, "MONSTER_EGG"), PEONY(5, "DOUBLE_PLANT"), PETRIFIED_OAK_SLAB("WOOD_STEP"), PHANTOM_MEMBRANE(13), PHANTOM_SPAWN_EGG(0, 13), PIGLIN_BANNER_PATTERN(16), PIGLIN_BRUTE_SPAWN_EGG(16), PIGLIN_SPAWN_EGG(57, "MONSTER_EGG"), PIG_SPAWN_EGG(90, "MONSTER_EGG"), PILLAGER_SPAWN_EGG(14), PINK_BANNER(9, "STANDING_BANNER", "BANNER"), PINK_BED(6, "BED_BLOCK", "BED"), PINK_CANDLE, PINK_CANDLE_CAKE, PINK_CARPET(6, "CARPET"), PINK_CONCRETE(6, "CONCRETE"), PINK_CONCRETE_POWDER(6, "CONCRETE_POWDER"), PINK_DYE(9, "INK_SACK"), PINK_GLAZED_TERRACOTTA(6, 12), PINK_SHULKER_BOX, PINK_STAINED_GLASS(6, "STAINED_GLASS"), PINK_STAINED_GLASS_PANE(6, "THIN_GLASS", "STAINED_GLASS_PANE"), PINK_TERRACOTTA(6, "STAINED_CLAY"), PINK_TULIP(7, "RED_ROSE"), PINK_WALL_BANNER(9, "WALL_BANNER"), PINK_WOOL(6, "WOOL"), PISTON("PISTON_BASE"), PISTON_HEAD("PISTON_EXTENSION"), PLAYER_HEAD(3, "SKULL", "SKULL_ITEM"), PLAYER_WALL_HEAD(3, "SKULL", "SKULL_ITEM"), PODZOL(2, "DIRT"), POINTED_DRIPSTONE, POISONOUS_POTATO, POLAR_BEAR_SPAWN_EGG(102, "MONSTER_EGG"), POLISHED_ANDESITE(6, "STONE"), POLISHED_ANDESITE_SLAB, POLISHED_ANDESITE_STAIRS, POLISHED_BASALT(16), POLISHED_BLACKSTONE(16), POLISHED_BLACKSTONE_BRICKS(16), POLISHED_BLACKSTONE_BRICK_SLAB(16), POLISHED_BLACKSTONE_BRICK_STAIRS(16), POLISHED_BLACKSTONE_BRICK_WALL(16), POLISHED_BLACKSTONE_BUTTON(16), POLISHED_BLACKSTONE_PRESSURE_PLATE(16), POLISHED_BLACKSTONE_SLAB(16), POLISHED_BLACKSTONE_STAIRS(16), POLISHED_BLACKSTONE_WALL(16), POLISHED_DEEPSLATE, POLISHED_DEEPSLATE_SLAB, POLISHED_DEEPSLATE_STAIRS, POLISHED_DEEPSLATE_WALL, POLISHED_DIORITE(4, "STONE"), POLISHED_DIORITE_SLAB, POLISHED_DIORITE_STAIRS, POLISHED_GRANITE(2, "STONE"), POLISHED_GRANITE_SLAB, POLISHED_GRANITE_STAIRS, POPPED_CHORUS_FRUIT("CHORUS_FRUIT_POPPED"), POPPY("RED_ROSE"), PORKCHOP("PORK"), POTATO("POTATO_ITEM"), POTATOES("POTATO"), POTION, POTTED_ACACIA_SAPLING(4, "SAPLING", "FLOWER_POT"), POTTED_ALLIUM(2, "RED_ROSE", "FLOWER_POT"), POTTED_AZALEA_BUSH, POTTED_AZURE_BLUET(3, "RED_ROSE", "FLOWER_POT"), POTTED_BAMBOO, POTTED_BIRCH_SAPLING(2, "SAPLING", "FLOWER_POT"), POTTED_BLUE_ORCHID(1, "RED_ROSE", "FLOWER_POT"), POTTED_BROWN_MUSHROOM("FLOWER_POT"), POTTED_CACTUS("FLOWER_POT"), POTTED_CORNFLOWER, POTTED_CRIMSON_FUNGUS(16), POTTED_CRIMSON_ROOTS(16), POTTED_DANDELION("YELLOW_FLOWER", "FLOWER_POT"), POTTED_DARK_OAK_SAPLING(5, "SAPLING", "FLOWER_POT"), POTTED_DEAD_BUSH("FLOWER_POT"), POTTED_FERN(2, "LONG_GRASS", "FLOWER_POT"), POTTED_FLOWERING_AZALEA_BUSH, POTTED_JUNGLE_SAPLING(3, "SAPLING", "FLOWER_POT"), POTTED_LILY_OF_THE_VALLEY, POTTED_OAK_SAPLING("SAPLING", "FLOWER_POT"), POTTED_ORANGE_TULIP(5, "RED_ROSE", "FLOWER_POT"), POTTED_OXEYE_DAISY(8, "RED_ROSE", "FLOWER_POT"), POTTED_PINK_TULIP(7, "RED_ROSE", "FLOWER_POT"), POTTED_POPPY("RED_ROSE", "FLOWER_POT"), POTTED_RED_MUSHROOM("FLOWER_POT"), POTTED_RED_TULIP(4, "RED_ROSE", "FLOWER_POT"), POTTED_SPRUCE_SAPLING(1, "SAPLING", "FLOWER_POT"), POTTED_WARPED_FUNGUS(16), POTTED_WARPED_ROOTS(16), POTTED_WHITE_TULIP(6, "RED_ROSE", "FLOWER_POT"), POTTED_WITHER_ROSE, POWDER_SNOW, POWDER_SNOW_BUCKET, POWDER_SNOW_CAULDRON, POWERED_RAIL, PRISMARINE, PRISMARINE_BRICKS(2, "PRISMARINE"), PRISMARINE_BRICK_SLAB(4, "STEP"), PRISMARINE_BRICK_STAIRS(13), PRISMARINE_CRYSTALS, PRISMARINE_SHARD, PRISMARINE_SLAB(13), PRISMARINE_STAIRS(13), PRISMARINE_WALL, PUFFERFISH(3, "RAW_FISH"), PUFFERFISH_BUCKET(0, 13), PUFFERFISH_SPAWN_EGG(0, 13), PUMPKIN, PUMPKIN_PIE, PUMPKIN_SEEDS, PUMPKIN_STEM, PURPLE_BANNER(5, "STANDING_BANNER", "BANNER"), PURPLE_BED(10, "BED_BLOCK", "BED"), PURPLE_CANDLE, PURPLE_CANDLE_CAKE, PURPLE_CARPET(10, "CARPET"), PURPLE_CONCRETE(10, "CONCRETE"), PURPLE_CONCRETE_POWDER(10, "CONCRETE_POWDER"), PURPLE_DYE(5, "INK_SACK"), PURPLE_GLAZED_TERRACOTTA(10, 12), PURPLE_SHULKER_BOX, PURPLE_STAINED_GLASS(10, "STAINED_GLASS"), PURPLE_STAINED_GLASS_PANE(10, "THIN_GLASS", "STAINED_GLASS_PANE"), PURPLE_TERRACOTTA(10, "STAINED_CLAY"), PURPLE_WALL_BANNER(5, "WALL_BANNER"), PURPLE_WOOL(10, "WOOL"), PURPUR_BLOCK, PURPUR_PILLAR, PURPUR_SLAB("PURPUR_DOUBLE_SLAB"), PURPUR_STAIRS, QUARTZ, QUARTZ_BLOCK, QUARTZ_BRICKS(16), QUARTZ_PILLAR(2, "QUARTZ_BLOCK"), QUARTZ_SLAB(7, "STEP"), QUARTZ_STAIRS, RABBIT, RABBIT_FOOT, RABBIT_HIDE, RABBIT_SPAWN_EGG(101, "MONSTER_EGG"), RABBIT_STEW, RAIL("RAILS"), RAVAGER_SPAWN_EGG(14), RAW_COPPER, RAW_COPPER_BLOCK, RAW_GOLD, RAW_GOLD_BLOCK, RAW_IRON, RAW_IRON_BLOCK, REDSTONE, REDSTONE_BLOCK, /** * Unlike redstone torch, REDSTONE_LAMP_ON isn't an item. * The name is just here on the list for matching. * * @see #REDSTONE_TORCH */ REDSTONE_LAMP("REDSTONE_LAMP_ON", "REDSTONE_LAMP_OFF"), REDSTONE_ORE("GLOWING_REDSTONE_ORE"), /** * REDSTONE_TORCH_OFF isn't an item, but a block. * But REDSTONE_TORCH_ON is the item. * The name is just here on the list for matching. */ REDSTONE_TORCH("REDSTONE_TORCH_OFF", "REDSTONE_TORCH_ON"), REDSTONE_WALL_TORCH, REDSTONE_WIRE, RED_BANNER(1, "STANDING_BANNER", "BANNER"), /** * Data value 14 or 0 */ RED_BED(14, "BED_BLOCK", "BED"), RED_CANDLE, RED_CANDLE_CAKE, RED_CARPET(14, "CARPET"), RED_CONCRETE(14, "CONCRETE"), RED_CONCRETE_POWDER(14, "CONCRETE_POWDER"), RED_DYE(1, "INK_SACK", "ROSE_RED"), RED_GLAZED_TERRACOTTA(14, 12), RED_MUSHROOM, RED_MUSHROOM_BLOCK("RED_MUSHROOM", "HUGE_MUSHROOM_2"), RED_NETHER_BRICKS("RED_NETHER_BRICK"), RED_NETHER_BRICK_SLAB(4, "STEP"), RED_NETHER_BRICK_STAIRS, RED_NETHER_BRICK_WALL, RED_SAND(1, "SAND"), RED_SANDSTONE, RED_SANDSTONE_SLAB("DOUBLE_STONE_SLAB2", "STONE_SLAB2"), RED_SANDSTONE_STAIRS, RED_SANDSTONE_WALL, RED_SHULKER_BOX, RED_STAINED_GLASS(14, "STAINED_GLASS"), RED_STAINED_GLASS_PANE(14, "THIN_GLASS", "STAINED_GLASS_PANE"), RED_TERRACOTTA(14, "STAINED_CLAY"), RED_TULIP(4, "RED_ROSE"), RED_WALL_BANNER(1, "WALL_BANNER"), RED_WOOL(14, "WOOL"), REPEATER("DIODE_BLOCK_ON", "DIODE_BLOCK_OFF", "DIODE"), REPEATING_COMMAND_BLOCK("COMMAND", "COMMAND_REPEATING"), RESPAWN_ANCHOR(16), ROOTED_DIRT, ROSE_BUSH(4, "DOUBLE_PLANT"), ROTTEN_FLESH, SADDLE, SALMON(1, "RAW_FISH"), SALMON_BUCKET(0, 13), SALMON_SPAWN_EGG(0, 13), SAND, SANDSTONE, SANDSTONE_SLAB(1, "DOUBLE_STEP", "STEP", "STONE_SLAB"), SANDSTONE_STAIRS, SANDSTONE_WALL, SCAFFOLDING(0, 14), SCULK_SENSOR, SCUTE(13), SEAGRASS(0, 13), SEA_LANTERN, SEA_PICKLE(13), SHEARS, SHEEP_SPAWN_EGG(91, "MONSTER_EGG"), SHIELD, SHROOMLIGHT(16), SHULKER_BOX("PURPLE_SHULKER_BOX"), SHULKER_SHELL, SHULKER_SPAWN_EGG(69, "MONSTER_EGG"), SILVERFISH_SPAWN_EGG(60, "MONSTER_EGG"), SKELETON_HORSE_SPAWN_EGG(28, "MONSTER_EGG"), SKELETON_SKULL("SKULL", "SKULL_ITEM"), SKELETON_SPAWN_EGG(51, "MONSTER_EGG"), SKELETON_WALL_SKULL("SKULL", "SKULL_ITEM"), SKULL_BANNER_PATTERN, SLIME_BALL, SLIME_BLOCK, SLIME_SPAWN_EGG(55, "MONSTER_EGG"), SMALL_AMETHYST_BUD, SMALL_DRIPLEAF, SMITHING_TABLE, SMOKER(0, 14), SMOOTH_BASALT, SMOOTH_QUARTZ(0, 13), SMOOTH_QUARTZ_SLAB(7, "STEP"), SMOOTH_QUARTZ_STAIRS, SMOOTH_RED_SANDSTONE(2, "RED_SANDSTONE"), SMOOTH_RED_SANDSTONE_SLAB("STONE_SLAB2"), SMOOTH_RED_SANDSTONE_STAIRS, SMOOTH_SANDSTONE(2, "SANDSTONE"), SMOOTH_SANDSTONE_SLAB("STEP"), SMOOTH_SANDSTONE_STAIRS, SMOOTH_STONE("STEP"), SMOOTH_STONE_SLAB("STEP"), SNOW, SNOWBALL("SNOW_BALL"), SNOW_BLOCK, SOUL_CAMPFIRE(16), SOUL_FIRE(16), SOUL_LANTERN(16), SOUL_SAND, SOUL_SOIL(16), SOUL_TORCH(16), SOUL_WALL_TORCH(16), SPAWNER("MOB_SPAWNER"), SPECTRAL_ARROW(0, 9), SPIDER_EYE, SPIDER_SPAWN_EGG(52, "MONSTER_EGG"), SPLASH_POTION, SPONGE, SPORE_BLOSSOM, SPRUCE_BOAT("BOAT_SPRUCE"), SPRUCE_BUTTON("WOOD_BUTTON"), SPRUCE_DOOR("SPRUCE_DOOR", "SPRUCE_DOOR_ITEM"), SPRUCE_FENCE, SPRUCE_FENCE_GATE, SPRUCE_LEAVES(1, "LEAVES", "LEAVES_2"), SPRUCE_LOG(1, "LOG"), SPRUCE_PLANKS(1, "WOOD"), SPRUCE_PRESSURE_PLATE("WOOD_PLATE"), SPRUCE_SAPLING(1, "SAPLING"), SPRUCE_SIGN("SIGN_POST", "SIGN"), SPRUCE_SLAB(1, "WOOD_DOUBLE_STEP", "WOOD_STEP", "WOODEN_SLAB"), SPRUCE_STAIRS("SPRUCE_WOOD_STAIRS"), SPRUCE_TRAPDOOR("TRAP_DOOR"), SPRUCE_WALL_SIGN("WALL_SIGN"), SPRUCE_WOOD(1, "LOG"), SPYGLASS, SQUID_SPAWN_EGG(94, "MONSTER_EGG"), STICK, STICKY_PISTON("PISTON_BASE", "PISTON_STICKY_BASE"), STONE, STONECUTTER(14), STONE_AXE, STONE_BRICKS("SMOOTH_BRICK"), STONE_BRICK_SLAB(4, "DOUBLE_STEP", "STEP", "STONE_SLAB"), STONE_BRICK_STAIRS("SMOOTH_STAIRS"), STONE_BRICK_WALL, STONE_BUTTON, STONE_HOE, STONE_PICKAXE, STONE_PRESSURE_PLATE("STONE_PLATE"), STONE_SHOVEL("STONE_SPADE"), STONE_SLAB("DOUBLE_STEP", "STEP"), STONE_STAIRS, STONE_SWORD, STRAY_SPAWN_EGG(6, "MONSTER_EGG"), STRIDER_SPAWN_EGG(16), STRING, STRIPPED_ACACIA_LOG("LOG_2"), STRIPPED_ACACIA_WOOD("LOG_2"), STRIPPED_BIRCH_LOG(2, "LOG"), STRIPPED_BIRCH_WOOD(2, "LOG"), STRIPPED_CRIMSON_HYPHAE(16), STRIPPED_CRIMSON_STEM(16), STRIPPED_DARK_OAK_LOG("LOG"), STRIPPED_DARK_OAK_WOOD("LOG"), STRIPPED_JUNGLE_LOG(3, "LOG"), STRIPPED_JUNGLE_WOOD(3, "LOG"), STRIPPED_OAK_LOG("LOG"), STRIPPED_OAK_WOOD("LOG"), STRIPPED_SPRUCE_LOG(1, "LOG"), STRIPPED_SPRUCE_WOOD(1, "LOG"), STRIPPED_WARPED_HYPHAE(16), STRIPPED_WARPED_STEM(16), STRUCTURE_BLOCK, /** * Originally developers used barrier blocks for its purpose. * So technically this isn't really considered as a suggested material. */ STRUCTURE_VOID(10, "BARRIER"), SUGAR, /** * Sugar Cane is a known material in pre-1.13 */ SUGAR_CANE("SUGAR_CANE_BLOCK"), SUNFLOWER("DOUBLE_PLANT"), SUSPICIOUS_STEW(0, 14), SWEET_BERRIES(14), SWEET_BERRY_BUSH(0, 14), TALL_GRASS(2, "DOUBLE_PLANT"), TALL_SEAGRASS(2, 13), TARGET(16), TERRACOTTA("STAINED_CLAY"), TINTED_GLASS, TIPPED_ARROW(0, 9), TNT, TNT_MINECART("EXPLOSIVE_MINECART"), TORCH, TOTEM_OF_UNDYING("TOTEM"), TRADER_LLAMA_SPAWN_EGG(103, 14), TRAPPED_CHEST, TRIDENT(13), TRIPWIRE, TRIPWIRE_HOOK, TROPICAL_FISH(2, "RAW_FISH"), TROPICAL_FISH_BUCKET(0, 13, "BUCKET", "WATER_BUCKET"), TROPICAL_FISH_SPAWN_EGG(0, 13, "MONSTER_EGG"), TUBE_CORAL(13), TUBE_CORAL_BLOCK(13), TUBE_CORAL_FAN(13), TUBE_CORAL_WALL_FAN, TUFF, TURTLE_EGG(0, 13), TURTLE_HELMET(0, 13), TURTLE_SPAWN_EGG(0, 13), TWISTING_VINES(16), TWISTING_VINES_PLANT(16), VEX_SPAWN_EGG(35, "MONSTER_EGG"), VILLAGER_SPAWN_EGG(120, "MONSTER_EGG"), VINDICATOR_SPAWN_EGG(36, "MONSTER_EGG"), VINE, /** * 1.13 tag is not added because it's the same thing as {@link #AIR} * * @see #CAVE_AIR */ VOID_AIR("AIR"), WALL_TORCH("TORCH"), WANDERING_TRADER_SPAWN_EGG(0, 14), WARPED_BUTTON(16), WARPED_DOOR(16), WARPED_FENCE(16), WARPED_FENCE_GATE(16), WARPED_FUNGUS(16), WARPED_FUNGUS_ON_A_STICK(16), WARPED_HYPHAE(16), WARPED_NYLIUM(16), WARPED_PLANKS(16), WARPED_PRESSURE_PLATE(16), WARPED_ROOTS(16), WARPED_SIGN(0, 16, "SIGN_POST"), WARPED_SLAB(16), WARPED_STAIRS(16), WARPED_STEM(16), WARPED_TRAPDOOR(16), WARPED_WALL_SIGN(0, 16, "WALL_SIGN"), WARPED_WART_BLOCK(16), /** * This is used for blocks only. * In 1.13- WATER will turn into STATIONARY_WATER after it finished spreading. * After 1.13+ this uses * https://hub.spigotmc.org/javadocs/spigot/org/bukkit/block/data/Levelled.html water flowing system. */ WATER("STATIONARY_WATER"), WATER_BUCKET, WATER_CAULDRON, WAXED_COPPER_BLOCK, WAXED_CUT_COPPER, WAXED_CUT_COPPER_SLAB, WAXED_CUT_COPPER_STAIRS, WAXED_EXPOSED_COPPER, WAXED_EXPOSED_CUT_COPPER, WAXED_EXPOSED_CUT_COPPER_SLAB, WAXED_EXPOSED_CUT_COPPER_STAIRS, WAXED_OXIDIZED_COPPER, WAXED_OXIDIZED_CUT_COPPER, WAXED_OXIDIZED_CUT_COPPER_SLAB, WAXED_OXIDIZED_CUT_COPPER_STAIRS, WAXED_WEATHERED_COPPER, WAXED_WEATHERED_CUT_COPPER, WAXED_WEATHERED_CUT_COPPER_SLAB, WAXED_WEATHERED_CUT_COPPER_STAIRS, WEATHERED_COPPER, WEATHERED_CUT_COPPER, WEATHERED_CUT_COPPER_SLAB, WEATHERED_CUT_COPPER_STAIRS, WEEPING_VINES(16), WEEPING_VINES_PLANT(16), WET_SPONGE(1, "SPONGE"), /** * Wheat is a known material in pre-1.13 */ WHEAT("CROPS"), WHEAT_SEEDS("SEEDS"), WHITE_BANNER(15, "STANDING_BANNER", "BANNER"), WHITE_BED("BED_BLOCK", "BED"), WHITE_CANDLE, WHITE_CANDLE_CAKE, WHITE_CARPET("CARPET"), WHITE_CONCRETE("CONCRETE"), WHITE_CONCRETE_POWDER("CONCRETE_POWDER"), WHITE_DYE(15, 14, "INK_SACK", "BONE_MEAL"), WHITE_GLAZED_TERRACOTTA(0, 12, "STAINED_CLAY"), WHITE_SHULKER_BOX, WHITE_STAINED_GLASS("STAINED_GLASS"), WHITE_STAINED_GLASS_PANE("THIN_GLASS", "STAINED_GLASS_PANE"), WHITE_TERRACOTTA("STAINED_CLAY", "TERRACOTTA"), WHITE_TULIP(6, "RED_ROSE"), WHITE_WALL_BANNER(15, "WALL_BANNER"), WHITE_WOOL("WOOL"), WITCH_SPAWN_EGG(66, "MONSTER_EGG"), WITHER_ROSE(0, 14), WITHER_SKELETON_SKULL(1, "SKULL", "SKULL_ITEM"), WITHER_SKELETON_SPAWN_EGG(5, "MONSTER_EGG"), WITHER_SKELETON_WALL_SKULL(1, "SKULL", "SKULL_ITEM"), WOLF_SPAWN_EGG(95, "MONSTER_EGG"), WOODEN_AXE("WOOD_AXE"), WOODEN_HOE("WOOD_HOE"), WOODEN_PICKAXE("WOOD_PICKAXE"), WOODEN_SHOVEL("WOOD_SPADE"), WOODEN_SWORD("WOOD_SWORD"), WRITABLE_BOOK("BOOK_AND_QUILL"), WRITTEN_BOOK, YELLOW_BANNER(11, "STANDING_BANNER", "BANNER"), YELLOW_BED(4, "BED_BLOCK", "BED"), YELLOW_CANDLE, YELLOW_CANDLE_CAKE, YELLOW_CARPET(4, "CARPET"), YELLOW_CONCRETE(4, "CONCRETE"), YELLOW_CONCRETE_POWDER(4, "CONCRETE_POWDER"), YELLOW_DYE(11, "INK_SACK", "DANDELION_YELLOW"), YELLOW_GLAZED_TERRACOTTA(4, 12, "STAINED_CLAY", "YELLOW_TERRACOTTA"), YELLOW_SHULKER_BOX, YELLOW_STAINED_GLASS(4, "STAINED_GLASS"), YELLOW_STAINED_GLASS_PANE(4, "THIN_GLASS", "STAINED_GLASS_PANE"), YELLOW_TERRACOTTA(4, "STAINED_CLAY"), YELLOW_WALL_BANNER(11, "WALL_BANNER"), YELLOW_WOOL(4, "WOOL"), ZOGLIN_SPAWN_EGG(16), ZOMBIE_HEAD(2, "SKULL", "SKULL_ITEM"), ZOMBIE_HORSE_SPAWN_EGG(29, "MONSTER_EGG"), ZOMBIE_SPAWN_EGG(54, "MONSTER_EGG"), ZOMBIE_VILLAGER_SPAWN_EGG(27, "MONSTER_EGG"), ZOMBIE_WALL_HEAD(2, "SKULL", "SKULL_ITEM"), ZOMBIFIED_PIGLIN_SPAWN_EGG(57, "MONSTER_EGG", "ZOMBIE_PIGMAN_SPAWN_EGG"); /** * Cached array of {@link XMaterial#values()} to avoid allocating memory for * calling the method every time. * * @since 2.0.0 */ public static final XMaterial[] VALUES = values(); /** * We don't want to use {@link Enums#getIfPresent(Class, String)} to avoid a few checks. * * @since 5.1.0 */ private static final Map NAMES = new HashMap<>(); /** * Guava (Google Core Libraries for Java)'s cache for performance and timed caches. * For strings that match a certain XMaterial. Mostly cached for configs. * * @since 1.0.0 */ private static final Cache NAME_CACHE = CacheBuilder.newBuilder() .expireAfterAccess(1, TimeUnit.HOURS) .build(); /** * This is used for {@link #isOneOf(Collection)} * * @since 3.4.0 */ private static final LoadingCache CACHED_REGEX = CacheBuilder.newBuilder() .expireAfterAccess(3, TimeUnit.HOURS) .build(new CacheLoader() { @Override public Pattern load(@Nonnull String str) { try { return Pattern.compile(str); } catch (PatternSyntaxException ex) { ex.printStackTrace(); return null; } } }); /** * The maximum data value in the pre-flattening update which belongs to {@link #VILLAGER_SPAWN_EGG}
* https://minecraftitemids.com/types/spawn-egg * * @see #matchXMaterialWithData(String) * @since 8.0.0 */ private static final byte MAX_DATA_VALUE = 120; /** * Used to tell the system that the passed object's (name or material) data value * is not provided or is invalid. * * @since 8.0.0 */ private static final byte UNKNOWN_DATA_VALUE = -1; /** * The maximum material ID before the pre-flattening update which belongs to {@link #MUSIC_DISC_WAIT} * * @since 8.1.0 */ private static final short MAX_ID = 2267; /** * XMaterial Paradox (Duplication Check) *

* A set of duplicated material names in 1.13 and 1.12 that will conflict with the legacy names. * Values are the new material names. This map also contains illegal elements. Check the static initializer for more info. *

* Duplications are not useful at all in versions above the flattening update {@link Data#ISFLAT} * This set is only used for matching materials, for parsing refer to {@link #isDuplicated()} * * @since 3.0.0 */ private static final Set DUPLICATED; static { for (XMaterial material : VALUES) NAMES.put(material.name(), material); } static { if (Data.ISFLAT) { // It's not needed at all if it's the newer version. We can save some memory. DUPLICATED = null; } else { // MELON_SLICE, CARROTS, POTATOES, BEETROOTS, GRASS_BLOCK, BRICKS, NETHER_BRICKS, BROWN_MUSHROOM // Using the constructor to add elements will decide to allocate more size which we don't need. DUPLICATED = new HashSet<>(4); DUPLICATED.add(GRASS.name()); DUPLICATED.add(MELON.name()); DUPLICATED.add(BRICK.name()); DUPLICATED.add(NETHER_BRICK.name()); } } /** * The data value of this material https://minecraft.gamepedia.com/Java_Edition_data_values/Pre-flattening * It's never a negative number. * * @see #getData() */ private final byte data; /** * The version that this material was added in, otherwise 0 if the version is not recorded. * * @see #getMaterialVersion() * @since 7.0.0 */ private final byte version; /** * A list of material names that was being used for older verions. * * @see #getLegacy() */ @Nonnull private final String[] legacy; /** * The cached Bukkit parsed material. * * @see #parseMaterial() * @since 9.0.0 */ @Nullable private final Material material; XMaterial(int data, int version, @Nonnull String... legacy) { this.data = (byte) data; this.version = (byte) version; this.legacy = legacy; Material mat = null; if ((!Data.ISFLAT && this.isDuplicated()) || (mat = Material.getMaterial(this.name())) == null) { for (int i = legacy.length - 1; i >= 0; i--) { mat = Material.getMaterial(legacy[i]); if (mat != null) break; } } this.material = mat; } XMaterial(int data, @Nonnull String... legacy) { this(data, 0, legacy); } XMaterial(int version) { this(0, version); } XMaterial() { this(0, 0); } XMaterial(String... legacy) { this(0, 0, legacy); } /** * Checks if the version is 1.13 Aquatic Update or higher. * An invocation of this method yields the cached result from the expression: *

*

* {@link #supports(int) 13}} *
* * @return true if 1.13 or higher. * @see #getVersion() * @see #supports(int) * @since 1.0.0 * @deprecated Use {@code XMaterial.supports(13)} instead. This method name can be confusing. */ @Deprecated public static boolean isNewVersion() { return Data.ISFLAT; } /** * This is just an extra method that can be used for many cases. * It can be used in {@link org.bukkit.event.player.PlayerInteractEvent} * or when accessing {@link org.bukkit.entity.Player#getMainHand()}, * or other compatibility related methods. *

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

*

* !{@link #supports(int)} 9 *
* * @since 2.0.0 */ public static boolean isOneEight() { return !supports(9); } /** * Gets the XMaterial with this name similar to {@link #valueOf(String)} * without throwing an exception. * * @param name the name of the material. * * @return an optional that can be empty. * @since 5.1.0 */ @Nonnull private static Optional getIfPresent(@Nonnull String name) { return Optional.ofNullable(NAMES.get(name)); } /** * The current version of the server. * * @return the current server version minor number. * @see #supports(int) * @since 2.0.0 */ public static int getVersion() { return Data.VERSION; } /** * When using 1.13+, this helps to find the old material name * with its data value using a cached search for optimization. * * @see #matchDefinedXMaterial(String, byte) * @since 1.0.0 */ @Nullable private static XMaterial requestOldXMaterial(@Nonnull String name, byte data) { String holder = name + data; XMaterial cache = NAME_CACHE.getIfPresent(holder); if (cache != null) return cache; for (XMaterial material : VALUES) { // Not using material.name().equals(name) check is intended. if ((data == UNKNOWN_DATA_VALUE || data == material.data) && material.anyMatchLegacy(name)) { NAME_CACHE.put(holder, material); return material; } } return null; } /** * Parses the given material name as an XMaterial with a given data * value in the string if attached. Check {@link #matchXMaterialWithData(String)} for more info. * * @see #matchXMaterialWithData(String) * @see #matchDefinedXMaterial(String, byte) * @since 2.0.0 */ @Nonnull public static Optional matchXMaterial(@Nonnull String name) { Validate.notEmpty(name, "Cannot match a material with null or empty material name"); Optional oldMatch = matchXMaterialWithData(name); return oldMatch.isPresent() ? oldMatch : matchDefinedXMaterial(format(name), UNKNOWN_DATA_VALUE); } /** * Parses material name and data value from the specified string. * The separator for the material name and its data value is {@code :} * Spaces are allowed. Mostly used when getting materials from config for old school minecrafters. *

* Examples *

     *     {@code INK_SACK:1 -> RED_DYE}
     *     {@code WOOL: 14  -> RED_WOOL}
     * 
* * @param name the material string that consists of the material name, data and separator character. * * @return the parsed XMaterial. * @see #matchXMaterial(String) * @since 3.0.0 */ @Nonnull private static Optional matchXMaterialWithData(@Nonnull String name) { int index = name.indexOf(':'); if (index != -1) { String mat = format(name.substring(0, index)); try { // We don't use Byte.parseByte because we have our own range check. byte data = (byte) Integer.parseInt(StringUtils.deleteWhitespace(name.substring(index + 1))); return data >= 0 && data < MAX_DATA_VALUE ? matchDefinedXMaterial(mat, data) : matchDefinedXMaterial(mat, UNKNOWN_DATA_VALUE); } catch (NumberFormatException ignored) { return matchDefinedXMaterial(mat, UNKNOWN_DATA_VALUE); } } return Optional.empty(); } /** * Parses the given material as an XMaterial. * * @throws IllegalArgumentException may be thrown as an unexpected exception. * @see #matchDefinedXMaterial(String, byte) * @see #matchXMaterial(ItemStack) * @since 2.0.0 */ @Nonnull public static XMaterial matchXMaterial(@Nonnull Material material) { Objects.requireNonNull(material, "Cannot match null material"); return matchDefinedXMaterial(material.name(), UNKNOWN_DATA_VALUE) .orElseThrow(() -> new IllegalArgumentException("Unsupported material with no data value: " + material.name())); } /** * Parses the given item as an XMaterial using its material and data value (durability) * if not a damageable item {@link ItemStack#getDurability()}. * * @param item the ItemStack to match. * * @return an XMaterial if matched any. * @throws IllegalArgumentException may be thrown as an unexpected exception. * @see #matchXMaterial(Material) * @since 2.0.0 */ @Nonnull @SuppressWarnings("deprecation") public static XMaterial matchXMaterial(@Nonnull ItemStack item) { Objects.requireNonNull(item, "Cannot match null ItemStack"); String material = item.getType().name(); byte data = (byte) (Data.ISFLAT || item.getType().getMaxDurability() > 0 ? 0 : item.getDurability()); // Check FILLED_MAP enum for more info. //if (!Data.ISFLAT && item.hasItemMeta() && item.getItemMeta() instanceof org.bukkit.inventory.meta.MapMeta) return FILLED_MAP; return matchDefinedXMaterial(material, data) .orElseThrow(() -> new IllegalArgumentException("Unsupported material from item: " + material + " (" + data + ')')); } /** * The main method that parses the given material name and data value as an XMaterial. * All the values passed to this method will not be null or empty and are formatted correctly. * * @param name the formatted name of the material. * @param data the data value of the material. Is always 0 or {@link #UNKNOWN_DATA_VALUE} when {@link Data#ISFLAT} * * @return an XMaterial (with the same data value if specified) * @see #matchXMaterial(Material) * @see #matchXMaterial(int, byte) * @see #matchXMaterial(ItemStack) * @since 3.0.0 */ @Nonnull protected static Optional matchDefinedXMaterial(@Nonnull String name, byte data) { // if (!Boolean.valueOf(Boolean.getBoolean(Boolean.TRUE.toString())).equals(Boolean.FALSE.booleanValue())) return null; Boolean duplicated = null; boolean isAMap = name.equalsIgnoreCase("MAP"); // Do basic number and boolean checks before accessing more complex enum stuff. if (Data.ISFLAT || (!isAMap && data <= 0 && !(duplicated = isDuplicated(name)))) { Optional xMaterial = getIfPresent(name); if (xMaterial.isPresent()) return xMaterial; } // Usually flat versions wouldn't pass this point, but some special materials do. XMaterial oldXMaterial = requestOldXMaterial(name, data); if (oldXMaterial == null) { // Special case. Refer to FILLED_MAP for more info. return (data >= 0 && isAMap) ? Optional.of(FILLED_MAP) : Optional.empty(); } if (!Data.ISFLAT && oldXMaterial.isPlural() && (duplicated == null ? isDuplicated(name) : duplicated)) return getIfPresent(name); return Optional.of(oldXMaterial); } /** * XMaterial Paradox (Duplication Check) * Checks if the material has any duplicates. *

* Example: *

{@code MELON, CARROT, POTATO, BEETROOT -> true} * * @param name the name of the material to check. * * @return true if there's a duplicated material for this material, otherwise false. * @since 2.0.0 */ private static boolean isDuplicated(@Nonnull String name) { // Don't use matchXMaterial() since this method is being called from matchXMaterial() itself and will cause a StackOverflowError. return DUPLICATED.contains(name); } /** * Gets the XMaterial based on the material's ID (Magic Value) and data value.
* You should avoid using this for performance issues. * * @param id the ID (Magic value) of the material. * @param data the data value of the material. * * @return a parsed XMaterial with the same ID and data value. * @see #matchXMaterial(ItemStack) * @since 2.0.0 * @deprecated this method loops through all the available materials and matches their ID using {@link #getId()} * which takes a really long time. Plugins should no longer support IDs. If you want, you can make a {@link Map} cache yourself. * This method obviously doesn't work for 1.13+ and will not be supported. This is only here for debugging purposes. */ @Nonnull @Deprecated public static Optional matchXMaterial(int id, byte data) { if (id < 0 || id > MAX_ID || data < 0) return Optional.empty(); for (XMaterial materials : VALUES) { if (materials.data == data && materials.getId() == id) return Optional.of(materials); } return Optional.empty(); } /** * Attempts to build the string like an enum name. * Removes all the spaces, 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 material name to modify. * * @return an enum name. * @since 2.0.0 */ @Nonnull protected 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 { boolean number = false; // Old materials have numbers in them. if ((ch >= 'A' && ch <= 'Z') || (ch >= 'a' && ch <= 'z') || (number = (ch >= '0' && ch <= '9'))) { if (appendUnderline) { chs[count++] = '_'; appendUnderline = false; } if (number) chs[count++] = ch; else chs[count++] = (char) (ch & 0x5f); } } } return new String(chs, 0, count); } /** * Checks if the specified version is the same version or higher than the current server version. * * @param version the major version to be checked. "1." is ignored. E.g. 1.12 = 12 | 1.9 = 9 * * @return true of the version is equal or higher than the current version. * @since 2.0.0 */ public static boolean supports(int version) { return Data.VERSION >= version; } /** * Gets the exact major version (..., 1.9, 1.10, ..., 1.14) * In most cases, you shouldn't be using this method. * * @param version Supports {@link Bukkit#getVersion()}, {@link Bukkit#getBukkitVersion()} and normal formats such as "1.14" * * @return the exact major version. * @see #supports(int) * @see #getVersion() * @see #getMaterialVersion() * @since 2.0.0 */ @Nonnull public static String getMajorVersion(@Nonnull String version) { Validate.notEmpty(version, "Cannot get major Minecraft version from null or empty string"); // getVersion() int index = version.lastIndexOf("MC:"); if (index != -1) { version = version.substring(index + 4, version.length() - 1); } else if (version.endsWith("SNAPSHOT")) { // getBukkitVersion() index = version.indexOf('-'); version = version.substring(0, index); } // 1.13.2, 1.14.4, etc... int lastDot = version.lastIndexOf('.'); if (version.indexOf('.') != lastDot) version = version.substring(0, lastDot); return version; } public String[] getLegacy() { return this.legacy; } /** * XMaterial Paradox (Duplication Check) * I've concluded that this is just an infinite loop that keeps * going around the Singular Form and the Plural Form materials. A waste of brain cells and a waste of time. * This solution works just fine anyway. *

* A solution for XMaterial Paradox. * Manually parses the duplicated materials to find the exact material based on the server version. * If the name ends with "S" -> Plural Form Material. * Plural methods are only plural if they're also {@link #DUPLICATED} *

* The only special exceptions are {@link #BRICKS} and {@link #NETHER_BRICKS} * * @return true if this material is a plural form material, otherwise false. * @since 8.0.0 */ private boolean isPlural() { // this.name().charAt(this.name().length() - 1) == 'S' return this == CARROTS || this == POTATOES; } /** * Checks if the list of given material names matches the given base material. * Mostly used for configs. *

* Supports {@link String#contains} {@code CONTAINS:NAME} and Regular Expression {@code REGEX:PATTERN} formats. *

* Example: *

     *     XMaterial material = {@link #matchXMaterial(ItemStack)};
     *     if (material.isOneOf(plugin.getConfig().getStringList("disabled-items")) return;
     * 
*
* {@code CONTAINS} Examples: *
     *     {@code "CONTAINS:CHEST" -> CHEST, ENDERCHEST, TRAPPED_CHEST -> true}
     *     {@code "cOnTaINS:dYe" -> GREEN_DYE, YELLOW_DYE, BLUE_DYE, INK_SACK -> true}
     * 
*

* {@code REGEX} Examples *

     *     {@code "REGEX:^.+_.+_.+$" -> Every Material with 3 underlines or more: SHULKER_SPAWN_EGG, SILVERFISH_SPAWN_EGG, SKELETON_HORSE_SPAWN_EGG}
     *     {@code "REGEX:^.{1,3}$" -> Material names that have 3 letters only: BED, MAP, AIR}
     * 
*

* The reason that there are tags for {@code CONTAINS} and {@code REGEX} is for the performance. * Although RegEx patterns are cached in this method, * please avoid using the {@code REGEX} tag if you can use the {@code CONTAINS} tag instead. * It'll have a huge impact on performance. * Please avoid using {@code (capturing groups)} there's no use for them in this case. * If you want to use groups, use {@code (?: non-capturing groups)}. It's faster. *

* Want to learn RegEx? You can mess around in RegExr website. * * @param materials the material names to check base material on. * * @return true if one of the given material names is similar to the base material. * @since 3.1.1 */ public boolean isOneOf(@Nullable Collection materials) { if (materials == null || materials.isEmpty()) return false; String name = this.name(); for (String comp : materials) { String checker = comp.toUpperCase(Locale.ENGLISH); if (checker.startsWith("CONTAINS:")) { comp = format(checker.substring(9)); if (name.contains(comp)) return true; continue; } if (checker.startsWith("REGEX:")) { comp = comp.substring(6); Pattern pattern = CACHED_REGEX.getUnchecked(comp); if (pattern != null && pattern.matcher(name).matches()) return true; continue; } // Direct Object Equals Optional xMat = matchXMaterial(comp); if (xMat.isPresent() && xMat.get() == this) return true; } return false; } /** * Sets the {@link Material} (and data value on older versions) of an item. * Damageable materials will not have their durability changed. *

* Use {@link #parseItem()} instead when creating new ItemStacks. * * @param item the item to change its type. * * @see #parseItem() * @since 3.0.0 */ @Nonnull @SuppressWarnings("deprecation") public ItemStack setType(@Nonnull ItemStack item) { Objects.requireNonNull(item, "Cannot set material for null ItemStack"); Material material = this.parseMaterial(); Objects.requireNonNull(material, () -> "Unsupported material: " + this.name()); item.setType(material); if (!Data.ISFLAT && material.getMaxDurability() <= 0) item.setDurability(this.data); return item; } /** * Checks if the given material name matches any of this xmaterial's legacy material names. * All the values passed to this method will not be null or empty and are formatted correctly. * * @param name the material name to check. * * @return true if it's one of the legacy names, otherwise false. * @since 2.0.0 */ private boolean anyMatchLegacy(@Nonnull String name) { for (int i = this.legacy.length - 1; i >= 0; i--) { if (name.equals(this.legacy[i])) return true; } return false; } /** * Parses an enum name to a user-friendly name. * These names will have underlines removed and with each word capitalized. *

* Examples: *

     *     {@literal EMERALD                 -> Emerald}
     *     {@literal EMERALD_BLOCK           -> Emerald Block}
     *     {@literal ENCHANTED_GOLDEN_APPLE  -> Enchanted Golden Apple}
     * 
* * @return a more user-friendly enum name. * @since 3.0.0 */ @Override @Nonnull public String toString() { return WordUtils.capitalize(this.name().replace('_', ' ').toLowerCase(Locale.ENGLISH)); } /** * Gets the ID (Magic value) of the material. * https://www.minecraftinfo.com/idlist.htm * * @return the ID of the material or -1 if it's not a legacy material or the server doesn't support the material. * @see #matchXMaterial(int, byte) * @since 2.2.0 */ @SuppressWarnings("deprecation") public int getId() { if (this.data != 0 || this.version >= 13) return -1; Material material = this.parseMaterial(); if (material == null) return -1; if (Data.ISFLAT && !material.isLegacy()) return -1; return material.getId(); } /** * The data value of this material pre-flattening. *

* Can be accessed with {@link ItemStack#getData()} then {@code MaterialData#getData()} * or {@link ItemStack#getDurability()} if not damageable. * * @return data of this material, or 0 if none. * @since 1.0.0 */ @SuppressWarnings("deprecation") public byte getData() { return data; } /** * Parses an item from this XMaterial. * Uses data values on older versions. * * @return an ItemStack with the same material (and data value if in older versions.) * @see #setType(ItemStack) * @since 2.0.0 */ @Nullable @SuppressWarnings("deprecation") public ItemStack parseItem() { Material material = this.parseMaterial(); if (material == null) return null; return Data.ISFLAT ? new ItemStack(material) : new ItemStack(material, 1, this.data); } /** * Parses the material of this XMaterial. * * @return the material related to this XMaterial based on the server version. * @since 1.0.0 */ @Nullable public Material parseMaterial() { return this.material; } /** * Checks if an item has the same material (and data value on older versions). * * @param item item to check. * * @return true if the material is the same as the item's material (and data value if on older versions), otherwise false. * @since 1.0.0 */ @SuppressWarnings("deprecation") public boolean isSimilar(@Nonnull ItemStack item) { Objects.requireNonNull(item, "Cannot compare with null ItemStack"); if (item.getType() != this.parseMaterial()) return false; return Data.ISFLAT || item.getDurability() == this.data || item.getType().getMaxDurability() > 0; } /** * Checks if this material is supported in the current version. * Suggested materials will be ignored. *

* Note that you should use {@link #parseMaterial()} or {@link #parseItem()} and check if it's null * if you're going to parse and use the material/item later. * * @return true if the material exists in {@link Material} list. * @since 2.0.0 */ public boolean isSupported() { return this.material != null; } /** * The version this material was added in. This will return the minor number in the version scheme. * For example, if it was added in 1.16, it'll return 16 * * @return the version number of 0 if the version was not recorded. * @since 7.0.0 */ public byte getMaterialVersion() { return version; } /** * This method is needed due to Java enum initialization limitations. * It's really inefficient yes, but it's only used for initialization. *

* Yes there are many other ways like comparing the hardcoded ordinal or using a boolean in the enum constructor, * but it's not really a big deal. *

* This method should not be called if the version is after the flattening update {@link Data#ISFLAT} * and is only used for parsing materials, not matching, for matching check {@link #DUPLICATED} */ private boolean isDuplicated() { switch (this.name()) { case "MELON": case "CARROT": case "POTATO": case "GRASS": case "BRICK": case "NETHER_BRICK": // Illegal Elements // Since both 1.12 and 1.13 have _DOOR XMaterial will use it // for 1.12 to parse the material, but it needs _DOOR_ITEM. // We'll trick XMaterial into thinking this needs to be parsed // using the old methods. // Some of these materials have their enum name added to the legacy list as well. case "DARK_OAK_DOOR": case "ACACIA_DOOR": case "BIRCH_DOOR": case "JUNGLE_DOOR": case "SPRUCE_DOOR": case "MAP": case "CAULDRON": case "BREWING_STAND": case "FLOWER_POT": return true; default: return false; } } /** * Used for datas that need to be accessed during enum initilization. * * @since 9.0.0 */ private static final class Data { /** * The current version of the server in the a form of a major version. * If the static initialization for this fails, you know something's wrong with the server software. * * @since 1.0.0 */ private static final int VERSION = Integer.parseInt(getMajorVersion(Bukkit.getVersion()).substring(2)); /** * Cached result if the server version is after the v1.13 flattening update. * * @since 3.0.0 */ private static final boolean ISFLAT = supports(13); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy