From f2aaef5c074a98d3811dcba503276f7831bf18ae Mon Sep 17 00:00:00 2001 From: Bukkit/Spigot Date: Fri, 21 Sep 2018 08:49:55 -0400 Subject: [PATCH] Add API to locate structures By: Senmori --- .../main/java/org/bukkit/StructureType.java | 222 ++++++++++++++++++ paper-api/src/main/java/org/bukkit/World.java | 14 ++ 2 files changed, 236 insertions(+) create mode 100644 paper-api/src/main/java/org/bukkit/StructureType.java diff --git a/paper-api/src/main/java/org/bukkit/StructureType.java b/paper-api/src/main/java/org/bukkit/StructureType.java new file mode 100644 index 0000000000..a8a92607ed --- /dev/null +++ b/paper-api/src/main/java/org/bukkit/StructureType.java @@ -0,0 +1,222 @@ +package org.bukkit; + +import com.google.common.base.Preconditions; +import com.google.common.collect.ImmutableMap; +import org.apache.commons.lang.Validate; +import org.bukkit.map.MapCursor; + +import java.util.HashMap; +import java.util.Map; +import java.util.Objects; + +/** + * This class handles the creation and storage of all structure types for + * Bukkit. Structure Types are the different kinds of structures that can be + * generated during world/chunk generation. These include Villages, Mineshafts, + * Mansions, etc. + *
+ * The registration of new {@link StructureType}s is case-sensitive. + */ +// Order is retrieved from WorldGenFactory +public class StructureType { + + private static final Map structureTypeMap = new HashMap<>(); + + /** + * Mineshafts are underground structures which consist of branching mining + * tunnels with wooden supports and broken rails. + *
+ * They are the only place where cave spider spawners and minecarts with + * chests can be found naturally. + */ + public static final StructureType MINESHAFT = register(new StructureType("Mineshaft", MapCursor.Type.RED_X)); + + /** + * Villages are naturally generating structures that form above ground. + *
+ * They are usually generated in desert, plains, taiga, and savanna biomes + * and are a site for villager spawns, with whom the player can trade. + */ + public static final StructureType VILLAGE = register(new StructureType("Village", MapCursor.Type.MANSION)); + + /** + * Nether fortresses are very large complexes that mainly consist of + * netherbricks. + *
+ * They contain blaze spawners, nether wart farms, and loot chests. They are + * only generated in the nether dimension. + */ + public static final StructureType NETHER_FORTRESS = register(new StructureType("Fortress", MapCursor.Type.RED_X)); + + /** + * Strongholds are underground structures that consist of many rooms, + * libraries, and an end portal room. + *
+ * They can be found using an {@link Material#ENDER_EYE}. + */ + public static final StructureType STRONGHOLD = register(new StructureType("Stronghold", MapCursor.Type.MANSION)); + + /** + * Jungle pyramids (also known as jungle temples) are found in jungles. + *
+ * They are usually composed of cobblestone and mossy cobblestone. They + * consist of three floors, with the bottom floor containing treasure + * chests. + */ + public static final StructureType JUNGLE_PYRAMID = register(new StructureType("Jungle_Pyramid", MapCursor.Type.RED_X)); + + /** + * Ocean ruins are clusters of many different blocks that generate + * underwater in ocean biomes (as well as on the surface of beaches). + *
+ * They come in my different variations. The cold variants consist primarily + * of stone brick, and the warm variants consist of sandstone. + */ + public static final StructureType OCEAN_RUIN = register(new StructureType("Ocean_Ruin", MapCursor.Type.TEMPLE)); + + /** + * Desert pyramids (also known as desert temples) are found in deserts. + *
+ * They are usually composed of sandstone and stained terracotta. + */ + public static final StructureType DESERT_PYRAMID = register(new StructureType("Desert_Pyramid", MapCursor.Type.RED_X)); + + /** + * Igloos are structures that generate in snowy biomes. + *
+ * They consist of the house, as well as a basement. + */ + public static final StructureType IGLOO = register(new StructureType("Igloo", MapCursor.Type.RED_X)); + + /** + * Swamp huts (also known as witch huts) generate in swamp biomes and have + * the ability to spawn witches. + */ + public static final StructureType SWAMP_HUT = register(new StructureType("Swamp_Hut", MapCursor.Type.RED_X)); + + /** + * Ocean monuments are underwater structures. + *
+ * They are usually composed on all three different prismarine types and sea + * lanterns. They are the only place guardians and elder guardians spawn + * naturally. + */ + public static final StructureType OCEAN_MONUMENT = register(new StructureType("Monument", MapCursor.Type.TEMPLE)); + + /** + * End Cities are tall castle-like structures that generate in the outer + * island of the End dimension. + *
+ * They consist primarily of end stone bricks, purpur blocks, and end rods. + * They are the only place where shulkers can be found. + */ + public static final StructureType END_CITY = register(new StructureType("EndCity", MapCursor.Type.RED_X)); + + /** + * Mansions (also known as woodland mansions) are massive house structures + * that generate in dark forests, containing a wide variety of rooms. + *
+ * They are the only place where evokers, vindicators, and vexes spawn + * naturally (but only once) + */ + public static final StructureType WOODLAND_MANSION = register(new StructureType("Mansion", MapCursor.Type.MANSION)); + + /** + * Buried treasure consists of a single chest buried in the beach sand or + * gravel, with random loot in it. + */ + public static final StructureType BURIED_TREASURE = register(new StructureType("Buried_Treasure", MapCursor.Type.RED_X)); + + /** + * Shipwrecks are structures that generate on the floor of oceans or + * beaches. + *
+ * They are made up of wood materials, and contain 1-3 loot chests. They can + * generate sideways, upside-down, or upright. + */ + public static final StructureType SHIPWRECK = register(new StructureType("Shipwreck", MapCursor.Type.RED_X)); + + /* **************** + * STRUCTURE TYPES REGISTERED ABOVE THIS + * **************** + */ + private final String name; + private final MapCursor.Type mapCursor; + + /** + * Create a new StructureType with a given name and map cursor. To indicate + * this structure should not be compatible with explorer maps, use null for + * mapIcon. + * + * @param name the name of the structure, case-sensitive + * @param mapIcon the {@link MapCursor.Type} this structure type should use + * when creating explorer maps. Use null to indicate this structure should + * not be compatible with explorer maps. + */ + private StructureType(String name, MapCursor.Type mapIcon) { + Validate.notEmpty(name, "Structure name cannot be empty"); + this.name = name; + this.mapCursor = mapIcon; + } + + /** + * Get the name of this structure. This is case-sensitive when used in + * commands. + * + * @return the name of this structure + */ + public String getName() { + return name; + } + + /** + * Get the {@link MapCursor.Type} that this structure can use on maps. If + * this is null, this structure will not appear on explorer maps. + * + * @return the {@link MapCursor.Type} or null. + */ + public MapCursor.Type getMapIcon() { + return mapCursor; + } + + @Override + public boolean equals(Object other) { + if (this == other) { + return true; + } + if (!(other instanceof StructureType)) { + return false; + } + StructureType that = (StructureType) other; + return this.name.equals(that.name) && this.mapCursor == that.mapCursor; + } + + @Override + public int hashCode() { + int hash = 7; + hash = 71 * hash + Objects.hashCode(this.name); + hash = 71 * hash + Objects.hashCode(this.mapCursor); + return hash; + } + + @Override + public String toString() { + return "StructureType{name=" + this.name + ", cursor=" + this.mapCursor + "}"; + } + + private static T register(T type) { + Preconditions.checkNotNull(type, "Cannot register null StructureType."); + Preconditions.checkArgument(!structureTypeMap.containsKey(type.getName()), "Cannot register same StructureType twice. %s", type.getName()); + StructureType.structureTypeMap.put(type.getName(), type); + return type; + } + + /** + * Get all registered {@link StructureType}s. + * + * @return an immutable copy of registered structure types. + */ + public static Map getStructureTypes() { + return ImmutableMap.copyOf(structureTypeMap); + } +} diff --git a/paper-api/src/main/java/org/bukkit/World.java b/paper-api/src/main/java/org/bukkit/World.java index 9ce8ecbc26..9efe698956 100644 --- a/paper-api/src/main/java/org/bukkit/World.java +++ b/paper-api/src/main/java/org/bukkit/World.java @@ -1522,6 +1522,20 @@ public interface World extends PluginMessageRecipient, Metadatable { */ public void spawnParticle(Particle particle, double x, double y, double z, int count, double offsetX, double offsetY, double offsetZ, double extra, T data, boolean force); + /** + * Find the closest nearby structure of a given {@link StructureType}. + * Finding unexplored structures can, and will, block if the world is + * looking in chunks that gave not generated yet. This can lead to the world + * temporarily freezing while locating an unexplored structure. + * + * @param origin where to start looking for a structure + * @param structureType the type of structure to find + * @param radius the radius, in chunks, around with to search + * @param findUnexplored true to only find unexplored structures + * @return the closest {@link Location}, or null if no structure of the + * specified type exists. + */ + public Location locateNearestStructure(Location origin, StructureType structureType, int radius, boolean findUnexplored); /** * Represents various map environment types that a world may be