From 49baa3c19c550b7f1197931af724f18e5e352c5a Mon Sep 17 00:00:00 2001 From: CraftBukkit/Spigot Date: Sun, 2 Feb 2020 19:27:10 +1100 Subject: [PATCH] #619: Allow delegation of certain elements to Vanilla when using a custom ChunkGenerator Allows delegation of caves, decorations, mobs and structures to the Vanilla generation algorithms. Overriding these methods to return true enables that aspect of Vanilla generation, which is applied after the ChunkGenerator's custom generation. By: konsolas --- .../generator/CustomChunkGenerator.java | 181 ++++++++++++------ 1 file changed, 123 insertions(+), 58 deletions(-) diff --git a/paper-server/src/main/java/org/bukkit/craftbukkit/generator/CustomChunkGenerator.java b/paper-server/src/main/java/org/bukkit/craftbukkit/generator/CustomChunkGenerator.java index cda41d0678..44408efad1 100644 --- a/paper-server/src/main/java/org/bukkit/craftbukkit/generator/CustomChunkGenerator.java +++ b/paper-server/src/main/java/org/bukkit/craftbukkit/generator/CustomChunkGenerator.java @@ -8,25 +8,28 @@ import net.minecraft.server.BiomeManager; import net.minecraft.server.BiomeStorage; import net.minecraft.server.Block; import net.minecraft.server.BlockPosition; +import net.minecraft.server.ChunkProviderGenerate; +import net.minecraft.server.ChunkProviderHell; +import net.minecraft.server.ChunkProviderTheEnd; import net.minecraft.server.ChunkSection; +import net.minecraft.server.DefinedStructureManager; import net.minecraft.server.EnumCreatureType; import net.minecraft.server.GeneratorAccess; import net.minecraft.server.GeneratorSettingsDefault; +import net.minecraft.server.GeneratorSettingsEnd; +import net.minecraft.server.GeneratorSettingsNether; +import net.minecraft.server.GeneratorSettingsOverworld; import net.minecraft.server.HeightMap; import net.minecraft.server.IChunkAccess; import net.minecraft.server.ITileEntity; -import net.minecraft.server.MobSpawnerCat; -import net.minecraft.server.MobSpawnerPatrol; -import net.minecraft.server.MobSpawnerPhantom; import net.minecraft.server.ProtoChunk; import net.minecraft.server.RegionLimitedWorldAccess; import net.minecraft.server.StructureGenerator; import net.minecraft.server.TileEntity; -import net.minecraft.server.VillageSiege; import net.minecraft.server.World; +import net.minecraft.server.WorldChunkManager; import net.minecraft.server.WorldGenFeatureConfiguration; import net.minecraft.server.WorldGenStage; -import net.minecraft.server.WorldGenerator; import net.minecraft.server.WorldServer; import org.bukkit.block.Biome; import org.bukkit.craftbukkit.block.CraftBlock; @@ -35,15 +38,11 @@ import org.bukkit.generator.ChunkGenerator.BiomeGrid; import org.bukkit.generator.ChunkGenerator.ChunkData; public class CustomChunkGenerator extends InternalChunkGenerator { + private final net.minecraft.server.ChunkGenerator delegate; private final ChunkGenerator generator; private final WorldServer world; private final long seed; private final Random random; - private final StructureGenerator strongholdGen = WorldGenerator.STRONGHOLD; - private final MobSpawnerPhantom mobSpawnerPhantom = new MobSpawnerPhantom(); - private final MobSpawnerPatrol mobSpawnerPatrol = new MobSpawnerPatrol(); - private final MobSpawnerCat mobSpawnerCat = new MobSpawnerCat(); - private final VillageSiege villageSiege = new VillageSiege(); private class CustomBiomeGrid implements BiomeGrid { @@ -78,6 +77,20 @@ public class CustomChunkGenerator extends InternalChunkGenerator C getFeatureConfiguration(BiomeBase biomebase, StructureGenerator structuregenerator) { + return (C) delegate.getFeatureConfiguration(biomebase, structuregenerator); + } + + @Override + public WorldChunkManager getWorldChunkManager() { + return delegate.getWorldChunkManager(); + } + + @Override + public void storeStructures(GeneratorAccess generatoraccess, IChunkAccess ichunkaccess) { + delegate.storeStructures(generatoraccess, ichunkaccess); + } + + @Override + public int getSeaLevel() { + return delegate.getSeaLevel(); + } + + @Override + public void createStructures(BiomeManager biomemanager, IChunkAccess ichunkaccess, net.minecraft.server.ChunkGenerator chunkgenerator, DefinedStructureManager definedstructuremanager) { + // Call the bukkit ChunkGenerator before structure generation so correct biome information is available. int x = ichunkaccess.getPos().x; int z = ichunkaccess.getPos().z; random.setSeed((long) x * 341873128712L + (long) z * 132897987541L); @@ -136,73 +180,94 @@ public class CustomChunkGenerator extends InternalChunkGenerator> 2, 0, 9 >> 2); + delegate.createStructures(new BiomeManager(null, 0, null) { - @Override - public void buildNoise(GeneratorAccess generatoraccess, IChunkAccess ichunkaccess) { - } + @Override + public BiomeManager a(WorldChunkManager worldchunkmanager) { + return this; + } - @Override - public int getBaseHeight(int i, int j, HeightMap.Type heightmap_type) { - return 0; - } - - @Override - public List getMobsFor(EnumCreatureType type, BlockPosition position) { - BiomeBase biomebase = world.getBiome(position); - - return biomebase == null ? null : biomebase.getMobs(type); - } - - @Override - public void addDecorations(RegionLimitedWorldAccess regionlimitedworldaccess) { - } - - @Override - public void addMobs(RegionLimitedWorldAccess regionlimitedworldaccess) { - } - - @Override - public BlockPosition findNearestMapFeature(World world, String type, BlockPosition position, int i, boolean flag) { - return "Stronghold".equals(type) && this.strongholdGen != null ? this.strongholdGen.getNearestGeneratedFeature(world, this, position, i, flag) : null; - } - - @Override - public GeneratorSettingsDefault getSettings() { - return settings; - } - - @Override - public void doMobSpawning(WorldServer worldserver, boolean flag, boolean flag1) { - if (worldserver.getWorldProvider().isOverworld()) { - this.mobSpawnerPhantom.a(worldserver, flag, flag1); - this.mobSpawnerPatrol.a(worldserver, flag, flag1); - this.mobSpawnerCat.a(worldserver, flag, flag1); - this.villageSiege.a(worldserver, flag, flag1); + @Override + public BiomeBase a(BlockPosition blockposition) { + return biome; + } + }, ichunkaccess, chunkgenerator, definedstructuremanager); } } + @Override + public void doCarving(BiomeManager biomemanager, IChunkAccess ichunkaccess, WorldGenStage.Features worldgenstage_features) { + if (generator.shouldGenerateCaves()) { + delegate.doCarving(biomemanager, ichunkaccess, worldgenstage_features); + } + } + + @Override + public void buildNoise(GeneratorAccess generatoraccess, IChunkAccess ichunkaccess) { + // Disable vanilla generation + } + + @Override + public int getBaseHeight(int i, int j, HeightMap.Type heightmap_type) { + return delegate.getBaseHeight(i, j, heightmap_type); + } + + @Override + public List getMobsFor(EnumCreatureType enumcreaturetype, BlockPosition blockposition) { + return delegate.getMobsFor(enumcreaturetype, blockposition); + } + + @Override + public void addDecorations(RegionLimitedWorldAccess regionlimitedworldaccess) { + if (generator.shouldGenerateDecorations()) { + delegate.addDecorations(regionlimitedworldaccess); + } + } + + @Override + public void addMobs(RegionLimitedWorldAccess regionlimitedworldaccess) { + if (generator.shouldGenerateMobs()) { + delegate.addMobs(regionlimitedworldaccess); + } + } + + @Override + public BlockPosition findNearestMapFeature(World world, String s, BlockPosition blockposition, int i, boolean flag) { + return delegate.findNearestMapFeature(world, s, blockposition, i, flag); + } + + @Override + public GeneratorSettingsDefault getSettings() { + return delegate.getSettings(); + } + + @Override + public void doMobSpawning(WorldServer worldserver, boolean flag, boolean flag1) { + delegate.doMobSpawning(worldserver, flag, flag1); + } + @Override public boolean canSpawnStructure(BiomeBase biomebase, StructureGenerator structuregenerator) { - return biomebase.a(structuregenerator); + return delegate.canSpawnStructure(biomebase, structuregenerator); } @Override public long getSeed() { - return seed; + return delegate.getSeed(); } @Override public int getSpawnHeight() { - return world.getSeaLevel() + 1; + return delegate.getSpawnHeight(); } @Override public int getGenerationDepth() { - return world.getHeight(); + return delegate.getGenerationDepth(); } }