2021-03-16 09:00:00 +11:00
--- a/net/minecraft/server/level/WorldServer.java
+++ b/net/minecraft/server/level/WorldServer.java
2021-12-11 00:11:02 +11:00
@@ -149,6 +149,20 @@
2014-11-26 08:32:16 +11:00
import org.apache.logging.log4j.LogManager;
import org.apache.logging.log4j.Logger;
+// CraftBukkit start
2021-03-16 09:00:00 +11:00
+import net.minecraft.world.entity.monster.EntityDrowned;
2021-11-23 18:39:43 +11:00
+import net.minecraft.world.level.dimension.WorldDimension;
2021-03-16 09:00:00 +11:00
+import net.minecraft.world.level.storage.WorldDataServer;
2019-04-23 12:00:00 +10:00
+import org.bukkit.Bukkit;
2014-11-26 08:32:16 +11:00
+import org.bukkit.WeatherType;
2019-04-23 12:00:00 +10:00
+import org.bukkit.craftbukkit.event.CraftEventFactory;
2020-06-25 10:00:00 +10:00
+import org.bukkit.craftbukkit.util.WorldUUID;
2019-04-23 12:00:00 +10:00
+import org.bukkit.event.entity.CreatureSpawnEvent;
+import org.bukkit.event.server.MapInitializeEvent;
2014-11-26 08:32:16 +11:00
+import org.bukkit.event.weather.LightningStrikeEvent;
2019-12-20 17:06:18 -08:00
+import org.bukkit.event.world.TimeSkipEvent;
2014-11-26 08:32:16 +11:00
+// CraftBukkit end
+
2020-06-25 10:00:00 +10:00
public class WorldServer extends World implements GeneratorAccessSeed {
2021-06-11 15:00:00 +10:00
public static final BlockPosition END_SPAWN_POINT = new BlockPosition(100, 50, 0);
2021-12-11 00:11:02 +11:00
@@ -166,7 +180,7 @@
2021-06-11 15:00:00 +10:00
final List<EntityPlayer> players;
private final ChunkProviderServer chunkSource;
2020-06-25 10:00:00 +10:00
private final MinecraftServer server;
2021-06-11 15:00:00 +10:00
- public final IWorldDataServer serverLevelData;
+ public final WorldDataServer serverLevelData; // CraftBukkit - type
final EntityTickList entityTickList;
2021-07-07 00:00:00 +10:00
public final PersistentEntitySectionManager<Entity> entityManager;
2021-06-11 15:00:00 +10:00
public boolean noSave;
2021-12-11 00:11:02 +11:00
@@ -189,9 +203,29 @@
2021-11-22 09:00:00 +11:00
private final StructureCheck structureCheck;
2021-06-11 15:00:00 +10:00
private final boolean tickTime;
2020-06-25 10:00:00 +10:00
2020-08-12 07:00:00 +10:00
- public WorldServer(MinecraftServer minecraftserver, Executor executor, Convertable.ConversionSession convertable_conversionsession, IWorldDataServer iworlddataserver, ResourceKey<World> resourcekey, DimensionManager dimensionmanager, WorldLoadListener worldloadlistener, ChunkGenerator chunkgenerator, boolean flag, long i, List<MobSpawner> list, boolean flag1) {
2021-06-11 15:00:00 +10:00
- Objects.requireNonNull(minecraftserver);
2021-11-22 09:00:00 +11:00
- super(iworlddataserver, resourcekey, dimensionmanager, minecraftserver::getProfiler, false, flag, i);
2020-06-25 10:00:00 +10:00
+
2015-02-26 22:41:06 +00:00
+ // CraftBukkit start
2019-04-23 12:00:00 +10:00
+ private int tickPosition;
2020-06-25 10:00:00 +10:00
+ public final Convertable.ConversionSession convertable;
+ public final UUID uuid;
2014-11-26 08:32:16 +11:00
+
2020-06-25 10:00:00 +10:00
+ public Chunk getChunkIfLoaded(int x, int z) {
2021-11-22 09:00:00 +11:00
+ return this.chunkSource.getChunk(x, z, false);
2020-06-25 10:00:00 +10:00
+ }
2019-04-23 12:00:00 +10:00
+
2021-11-23 18:39:43 +11:00
+ @Override
+ public ResourceKey<WorldDimension> getTypeKey() {
+ return convertable.dimensionType;
+ }
+
2020-06-25 10:00:00 +10:00
+ // Add env and gen to constructor, WorldData -> WorldDataServer
SPIGOT-5880, SPIGOT-5567: New ChunkGenerator API
## **Current API**
The current world generation API is very old and limited when you want to make more complex world generation. Resulting in some hard to fix bugs such as that you cannot modify blocks outside the chunk in the BlockPopulator (which should and was per the docs possible), or strange behavior such as SPIGOT-5880.
## **New API**
With the new API, the generation is more separate in multiple methods and is more in line with Vanilla chunk generation. The new API is designed to as future proof as possible. If for example a new generation step is added it can easily also be added as a step in API by simply creating the method for it. On the other side if a generation step gets removed, the method can easily be called after another, which is the case with surface and bedrock. The new API and changes are also fully backwards compatible with old chunk generators.
### **Changes in the new api**
**Extra generation steps:**
Noise, surface, bedrock and caves are added as steps. With those generation steps three extra methods for Vanilla generation are also added. Those new methods provide the ChunkData instead of returning one. The reason for this is, that the ChunkData is now backed by a ChunkAccess. With this, each step has the information of the step before and the Vanilla information (if chosen by setting a 'should' method to true). The old method is deprecated.
**New class BiomeProvider**
The BiomeProvider acts as Biome source and wrapper for the NMS class WorldChunkManager. With this the underlying Vanilla ChunkGeneration knows which Biome to use for the structure and decoration generation. (Fixes: SPIGOT-5880). Although the List of Biomes which is required in BiomeProvider, is currently not much in use in Vanilla, I decided to add it to future proof the API when it may be required in later versions of Minecraft.
The BiomeProvider is also separated from the ChunkGenerator for plugins which only want to change the biome map, such as single Biome worlds or if some biomes should be more present than others.
**Deprecated isParallelCapable**
Mojang has and is pushing to a more multi threaded chunk generation. This should also be the case for custom chunk generators. This is why the new API only supports multi threaded generation. This does not affect the old API, which is still checking this.
**Base height method added**
This method was added to also bring the Minecraft generator and Bukkit generator more in line. With this it is possible to return the max height of a location (before decorations). This is useful to let most structures know were to place them. This fixes SPIGOT-5567. (This fixes not all structures placement, desert pyramids for example are still way up at y-level 64, This however is more a vanilla bug and should be fixed at Mojangs end).
**WorldInfo Class**
The World object was swapped for a WorldInfo object. This is because many methods of the World object won't work during world generation and would mostly likely result in a deadlock. It contains any information a plugin should need to identify the world.
**BlockPopulator Changes**
Instead of directly manipulating a chunk, changes are now made to a new class LimitedRegion, this class provides methods to populated the chunk and its surrounding area. The wrapping is done so that the population can be moved into the place where Minecraft generates decorations. Where there is no chunk to access yet. By moving it into this place the generation is now async and the surrounding area of the chunk can also be used.
For common methods between the World and LimitedRegion a RegionAccessor was added.
By: DerFrZocker <derrieple@gmail.com>
2021-08-15 08:08:16 +10:00
+ public WorldServer(MinecraftServer minecraftserver, Executor executor, Convertable.ConversionSession convertable_conversionsession, IWorldDataServer iworlddataserver, ResourceKey<World> resourcekey, DimensionManager dimensionmanager, WorldLoadListener worldloadlistener, ChunkGenerator chunkgenerator, boolean flag, long i, List<MobSpawner> list, boolean flag1, org.bukkit.World.Environment env, org.bukkit.generator.ChunkGenerator gen, org.bukkit.generator.BiomeProvider biomeProvider) {
2021-06-11 15:00:00 +10:00
+ // Objects.requireNonNull(minecraftserver); // CraftBukkit - decompile error
2021-11-22 09:00:00 +11:00
+ super(iworlddataserver, resourcekey, dimensionmanager, minecraftserver::getProfiler, false, flag, i, gen, biomeProvider, env);
+ this.pvpMode = minecraftserver.isPvpAllowed();
2020-06-25 10:00:00 +10:00
+ convertable = convertable_conversionsession;
2021-06-11 15:00:00 +10:00
+ uuid = WorldUUID.getUUID(convertable_conversionsession.levelPath.toFile());
2014-11-26 08:32:16 +11:00
+ // CraftBukkit end
2021-06-11 15:00:00 +10:00
this.players = Lists.newArrayList();
this.entityTickList = new EntityTickList();
2021-11-22 09:00:00 +11:00
this.blockTicks = new TickListServer<>(this::isPositionTickingWithEntitiesLoaded, this.getProfilerSupplier());
2021-12-11 00:11:02 +11:00
@@ -203,7 +237,13 @@
2021-06-11 15:00:00 +10:00
this.tickTime = flag1;
2020-06-25 10:00:00 +10:00
this.server = minecraftserver;
2021-06-11 15:00:00 +10:00
this.customSpawners = list;
- this.serverLevelData = iworlddataserver;
2020-06-25 10:00:00 +10:00
+ // CraftBukkit start
2021-06-11 15:00:00 +10:00
+ this.serverLevelData = (WorldDataServer) iworlddataserver;
+ serverLevelData.world = this;
2020-06-25 10:00:00 +10:00
+ if (gen != null) {
+ chunkgenerator = new org.bukkit.craftbukkit.generator.CustomChunkGenerator(this, chunkgenerator, gen);
+ }
+ // CraftBukkit end
2021-11-22 09:00:00 +11:00
boolean flag2 = minecraftserver.forceSynchronousWrites();
DataFixer datafixer = minecraftserver.getFixerUpper();
EntityPersistentStorage<Entity> entitypersistentstorage = new EntityStorage(this, convertable_conversionsession.getDimensionPath(resourcekey).resolve("entities"), datafixer, flag2, minecraftserver);
2021-12-11 00:11:02 +11:00
@@ -234,14 +274,15 @@
2021-11-22 09:00:00 +11:00
long l = minecraftserver.getWorldData().worldGenSettings().seed();
this.structureCheck = new StructureCheck(this.chunkSource.chunkScanner(), this.registryAccess(), minecraftserver.getStructureManager(), resourcekey, chunkgenerator, this, chunkgenerator.getBiomeSource(), l, datafixer);
- this.structureFeatureManager = new StructureManager(this, minecraftserver.getWorldData().worldGenSettings(), this.structureCheck);
+ this.structureFeatureManager = new StructureManager(this, this.serverLevelData.worldGenSettings(), structureCheck); // CraftBukkit
if (this.dimensionType().createDragonFight()) {
- this.dragonFight = new EnderDragonBattle(this, l, minecraftserver.getWorldData().endDragonFightData());
+ this.dragonFight = new EnderDragonBattle(this, this.serverLevelData.worldGenSettings().seed(), this.serverLevelData.endDragonFightData()); // CraftBukkit
2020-06-25 10:00:00 +10:00
} else {
2021-06-11 15:00:00 +10:00
this.dragonFight = null;
2015-03-16 11:32:53 +00:00
}
2021-06-11 15:00:00 +10:00
this.sleepStatus = new SleepStatus();
2021-06-11 21:33:49 +10:00
+ this.getCraftServer().addWorld(this.getWorld()); // CraftBukkit
2019-12-20 17:06:18 -08:00
}
2021-11-22 09:00:00 +11:00
public void setWeatherParameters(int i, int j, boolean flag, boolean flag1) {
2021-12-11 00:11:02 +11:00
@@ -273,12 +314,20 @@
2021-11-22 09:00:00 +11:00
long j;
2019-04-23 12:00:00 +10:00
2021-11-22 09:00:00 +11:00
if (this.sleepStatus.areEnoughSleeping(i) && this.sleepStatus.areEnoughDeepSleeping(i, this.players)) {
2019-12-20 17:06:18 -08:00
+ // CraftBukkit start
2021-11-22 09:00:00 +11:00
+ j = this.levelData.getDayTime() + 24000L;
+ TimeSkipEvent event = new TimeSkipEvent(this.getWorld(), TimeSkipEvent.SkipReason.NIGHT_SKIP, (j - j % 24000L) - this.getDayTime());
2021-06-11 15:00:00 +10:00
if (this.getGameRules().getBoolean(GameRules.RULE_DAYLIGHT)) {
2021-11-22 09:00:00 +11:00
- j = this.levelData.getDayTime() + 24000L;
- this.setDayTime(j - j % 24000L);
2021-06-11 21:33:49 +10:00
+ getCraftServer().getPluginManager().callEvent(event);
2019-12-20 17:06:18 -08:00
+ if (!event.isCancelled()) {
+ this.setDayTime(this.getDayTime() + event.getSkipAmount());
+ }
}
2021-11-22 09:00:00 +11:00
- this.wakeUpAllPlayers();
2019-12-20 17:06:18 -08:00
+ if (!event.isCancelled()) {
2021-11-22 09:00:00 +11:00
+ this.wakeUpAllPlayers();
2019-12-20 17:06:18 -08:00
+ }
+ // CraftBukkit end
2021-11-22 09:00:00 +11:00
if (this.getGameRules().getBoolean(GameRules.RULE_WEATHER_CYCLE) && this.isRaining()) {
this.resetWeatherCycle();
2019-12-20 17:06:18 -08:00
}
2021-12-11 00:11:02 +11:00
@@ -304,7 +353,7 @@
2021-11-22 09:00:00 +11:00
this.runBlockEvents();
2021-06-11 15:00:00 +10:00
this.handlingTick = false;
2021-11-22 09:00:00 +11:00
gameprofilerfiller.pop();
- boolean flag = !this.players.isEmpty() || !this.getForcedChunks().isEmpty();
+ boolean flag = true || !this.players.isEmpty() || !this.getForcedChunks().isEmpty(); // CraftBukkit - this prevents entity cleanup, other issues on servers with no players
2019-04-23 12:00:00 +10:00
2021-11-22 09:00:00 +11:00
if (flag) {
2019-04-23 12:00:00 +10:00
this.resetEmptyTime();
2021-12-11 00:11:02 +11:00
@@ -320,7 +369,7 @@
2019-04-23 12:00:00 +10:00
2021-11-22 09:00:00 +11:00
this.entityTickList.forEach((entity) -> {
2021-06-11 15:00:00 +10:00
if (!entity.isRemoved()) {
2021-11-22 09:00:00 +11:00
- if (this.shouldDiscardEntity(entity)) {
+ if (false && this.shouldDiscardEntity(entity)) { // CraftBukkit - We prevent spawning in general, so this butchering is not needed
entity.discard();
2021-06-11 15:00:00 +10:00
} else {
2021-11-22 09:00:00 +11:00
gameprofilerfiller.push("checkDespawn");
2021-12-11 00:11:02 +11:00
@@ -392,7 +441,7 @@
2019-12-11 09:00:00 +11:00
2021-11-22 09:00:00 +11:00
private void wakeUpAllPlayers() {
this.sleepStatus.removeAllSleepers();
2019-12-11 09:00:00 +11:00
- ((List) this.players.stream().filter(EntityLiving::isSleeping).collect(Collectors.toList())).forEach((entityplayer) -> {
+ (this.players.stream().filter(EntityLiving::isSleeping).collect(Collectors.toList())).forEach((entityplayer) -> { // CraftBukkit - decompile error
2021-11-22 09:00:00 +11:00
entityplayer.stopSleepInBed(false, false);
2019-12-11 09:00:00 +11:00
});
}
2021-12-11 00:11:02 +11:00
@@ -419,14 +468,14 @@
2021-11-22 09:00:00 +11:00
entityhorseskeleton.setTrap(true);
entityhorseskeleton.setAge(0);
entityhorseskeleton.setPos((double) blockposition.getX(), (double) blockposition.getY(), (double) blockposition.getZ());
- this.addFreshEntity(entityhorseskeleton);
+ this.addFreshEntity(entityhorseskeleton, org.bukkit.event.entity.CreatureSpawnEvent.SpawnReason.LIGHTNING); // CraftBukkit
2019-04-23 12:00:00 +10:00
}
2021-11-22 09:00:00 +11:00
EntityLightning entitylightning = (EntityLightning) EntityTypes.LIGHTNING_BOLT.create(this);
2020-06-25 10:00:00 +10:00
2021-11-22 09:00:00 +11:00
entitylightning.moveTo(Vec3D.atBottomCenterOf(blockposition));
entitylightning.setVisualOnly(flag1);
- this.addFreshEntity(entitylightning);
2020-06-25 10:00:00 +10:00
+ this.strikeLightning(entitylightning, org.bukkit.event.weather.LightningStrikeEvent.Cause.WEATHER); // CraftBukkit
2019-04-23 12:00:00 +10:00
}
}
2021-12-11 00:11:02 +11:00
@@ -437,12 +486,12 @@
2019-04-23 12:00:00 +10:00
BiomeBase biomebase = this.getBiome(blockposition);
2021-11-22 09:00:00 +11:00
if (biomebase.shouldFreeze(this, blockposition1)) {
- this.setBlockAndUpdate(blockposition1, Blocks.ICE.defaultBlockState());
+ org.bukkit.craftbukkit.event.CraftEventFactory.handleBlockFormEvent(this, blockposition1, Blocks.ICE.defaultBlockState(), null); // CraftBukkit
2019-04-23 12:00:00 +10:00
}
2021-06-11 15:00:00 +10:00
if (flag) {
2021-11-22 09:00:00 +11:00
if (biomebase.shouldSnow(this, blockposition)) {
- this.setBlockAndUpdate(blockposition, Blocks.SNOW.defaultBlockState());
+ org.bukkit.craftbukkit.event.CraftEventFactory.handleBlockFormEvent(this, blockposition, Blocks.SNOW.defaultBlockState(), null); // CraftBukkit
2021-06-11 15:00:00 +10:00
}
2014-11-26 08:32:16 +11:00
2021-11-22 09:00:00 +11:00
IBlockData iblockdata = this.getBlockState(blockposition1);
2021-12-11 00:11:02 +11:00
@@ -638,6 +687,7 @@
2021-11-22 09:00:00 +11:00
this.rainLevel = MathHelper.clamp(this.rainLevel, 0.0F, 1.0F);
}
+ /* CraftBukkit start
if (this.oRainLevel != this.rainLevel) {
this.server.getPlayerList().broadcastAll(new PacketPlayOutGameStateChange(PacketPlayOutGameStateChange.RAIN_LEVEL_CHANGE, this.rainLevel), this.dimension());
}
2021-12-11 00:11:02 +11:00
@@ -656,14 +706,47 @@
2021-11-22 09:00:00 +11:00
this.server.getPlayerList().broadcastAll(new PacketPlayOutGameStateChange(PacketPlayOutGameStateChange.RAIN_LEVEL_CHANGE, this.rainLevel));
this.server.getPlayerList().broadcastAll(new PacketPlayOutGameStateChange(PacketPlayOutGameStateChange.THUNDER_LEVEL_CHANGE, this.thunderLevel));
}
+ // */
+ for (int idx = 0; idx < this.players.size(); ++idx) {
+ if (((EntityPlayer) this.players.get(idx)).level == this) {
+ ((EntityPlayer) this.players.get(idx)).tickWeather();
+ }
+ }
+
+ if (flag != this.isRaining()) {
+ // Only send weather packets to those affected
+ for (int idx = 0; idx < this.players.size(); ++idx) {
+ if (((EntityPlayer) this.players.get(idx)).level == this) {
+ ((EntityPlayer) this.players.get(idx)).setPlayerWeather((!flag ? WeatherType.DOWNFALL : WeatherType.CLEAR), false);
+ }
+ }
+ }
+ for (int idx = 0; idx < this.players.size(); ++idx) {
+ if (((EntityPlayer) this.players.get(idx)).level == this) {
+ ((EntityPlayer) this.players.get(idx)).updateWeather(this.oRainLevel, this.rainLevel, this.oThunderLevel, this.thunderLevel);
+ }
+ }
+ // CraftBukkit end
2014-11-26 08:32:16 +11:00
}
2021-11-22 09:00:00 +11:00
private void resetWeatherCycle() {
- this.serverLevelData.setRainTime(0);
2018-07-15 10:00:00 +10:00
+ // CraftBukkit start
2021-11-22 09:00:00 +11:00
this.serverLevelData.setRaining(false);
- this.serverLevelData.setThunderTime(0);
2015-01-05 09:50:48 +11:00
+ // If we stop due to everyone sleeping we should reset the weather duration to some other random value.
+ // Not that everyone ever manages to get the whole server to sleep at the same time....
2021-11-22 09:00:00 +11:00
+ if (!this.serverLevelData.isRaining()) {
+ this.serverLevelData.setRainTime(0);
2014-11-26 08:32:16 +11:00
+ }
2015-01-05 09:50:48 +11:00
+ // CraftBukkit end
2021-06-11 15:00:00 +10:00
this.serverLevelData.setThundering(false);
2015-01-05 09:50:48 +11:00
+ // CraftBukkit start
+ // If we stop due to everyone sleeping we should reset the weather duration to some other random value.
+ // Not that everyone ever manages to get the whole server to sleep at the same time....
2021-06-11 15:00:00 +10:00
+ if (!this.serverLevelData.isThundering()) {
2021-11-22 09:00:00 +11:00
+ this.serverLevelData.setThunderTime(0);
2014-11-26 08:32:16 +11:00
+ }
+ // CraftBukkit end
}
2019-04-23 12:00:00 +10:00
public void resetEmptyTime() {
2021-12-11 00:11:02 +11:00
@@ -698,6 +781,7 @@
2021-06-11 15:00:00 +10:00
});
2021-11-22 09:00:00 +11:00
gameprofilerfiller.incrementCounter("tickNonPassenger");
2021-06-11 15:00:00 +10:00
entity.tick();
+ entity.postTick(); // CraftBukkit
2021-11-22 09:00:00 +11:00
this.getProfiler().pop();
2021-06-11 15:00:00 +10:00
Iterator iterator = entity.getPassengers().iterator();
2021-12-11 00:11:02 +11:00
@@ -721,6 +805,7 @@
2019-04-23 12:00:00 +10:00
});
2021-11-22 09:00:00 +11:00
gameprofilerfiller.incrementCounter("tickPassenger");
entity1.rideTick();
2021-06-11 15:00:00 +10:00
+ entity1.postTick(); // CraftBukkit
2021-11-22 09:00:00 +11:00
gameprofilerfiller.pop();
2021-06-11 15:00:00 +10:00
Iterator iterator = entity1.getPassengers().iterator();
2014-11-26 08:32:16 +11:00
2021-12-11 00:11:02 +11:00
@@ -745,6 +830,7 @@
2021-11-22 09:00:00 +11:00
ChunkProviderServer chunkproviderserver = this.getChunkSource();
2014-11-26 08:32:16 +11:00
2019-04-23 12:00:00 +10:00
if (!flag1) {
2014-12-03 14:35:33 -05:00
+ org.bukkit.Bukkit.getPluginManager().callEvent(new org.bukkit.event.world.WorldSaveEvent(getWorld())); // CraftBukkit
2014-11-26 08:32:16 +11:00
if (iprogressupdate != null) {
2021-11-22 09:00:00 +11:00
iprogressupdate.progressStartNoAbort(new ChatMessage("menu.savingLevel"));
2014-12-03 14:35:33 -05:00
}
2021-12-11 00:11:02 +11:00
@@ -762,11 +848,19 @@
2021-06-11 15:00:00 +10:00
}
2019-06-25 21:02:13 +10:00
}
+
+ // CraftBukkit start - moved from MinecraftServer.saveChunks
+ WorldServer worldserver1 = this;
+
2021-11-22 09:00:00 +11:00
+ serverLevelData.setWorldBorder(worldserver1.getWorldBorder().createSettings());
+ serverLevelData.setCustomBossEvents(this.server.getCustomBossEvents().save());
+ convertable.saveDataTag(this.server.registryHolder, this.serverLevelData, this.server.getPlayerList().getSingleplayerData());
2019-06-25 21:02:13 +10:00
+ // CraftBukkit end
}
2021-11-22 09:00:00 +11:00
private void saveLevelData() {
2021-06-11 15:00:00 +10:00
if (this.dragonFight != null) {
2021-11-22 09:00:00 +11:00
- this.server.getWorldData().setEndDragonFightData(this.dragonFight.saveData());
+ this.serverLevelData.setEndDragonFightData(this.dragonFight.saveData()); // CraftBukkit
2020-07-17 08:56:56 +10:00
}
2021-11-22 09:00:00 +11:00
this.getChunkSource().getDataStorage().save();
2021-12-11 00:11:02 +11:00
@@ -812,15 +906,34 @@
2014-11-26 08:32:16 +11:00
2019-04-23 12:00:00 +10:00
@Override
2021-11-22 09:00:00 +11:00
public boolean addFreshEntity(Entity entity) {
- return this.addEntity(entity);
2019-04-23 12:00:00 +10:00
+ // CraftBukkit start
2021-11-22 09:00:00 +11:00
+ return this.addFreshEntity(entity, CreatureSpawnEvent.SpawnReason.DEFAULT);
2019-04-23 12:00:00 +10:00
+ }
+
+ @Override
2021-11-22 09:00:00 +11:00
+ public boolean addFreshEntity(Entity entity, CreatureSpawnEvent.SpawnReason reason) {
+ return this.addEntity(entity, reason);
2019-04-23 12:00:00 +10:00
+ // CraftBukkit end
2016-10-15 23:20:12 -07:00
}
2021-11-22 09:00:00 +11:00
public boolean addWithUUID(Entity entity) {
- return this.addEntity(entity);
2019-04-23 12:00:00 +10:00
+ // CraftBukkit start
2021-11-22 09:00:00 +11:00
+ return this.addWithUUID(entity, CreatureSpawnEvent.SpawnReason.DEFAULT);
2019-04-23 12:00:00 +10:00
+ }
+
2021-11-22 09:00:00 +11:00
+ public boolean addWithUUID(Entity entity, CreatureSpawnEvent.SpawnReason reason) {
+ return this.addEntity(entity, reason);
2019-04-23 12:00:00 +10:00
+ // CraftBukkit end
2016-10-15 23:20:12 -07:00
}
2021-11-22 09:00:00 +11:00
public void addDuringTeleport(Entity entity) {
- this.addEntity(entity);
2021-06-11 15:00:00 +10:00
+ // CraftBukkit start
2021-11-22 09:00:00 +11:00
+ this.addDuringTeleport(entity, CreatureSpawnEvent.SpawnReason.DEFAULT);
2021-06-11 15:00:00 +10:00
+ }
+
2021-11-22 09:00:00 +11:00
+ public void addDuringTeleport(Entity entity, CreatureSpawnEvent.SpawnReason reason) {
+ this.addEntity(entity, reason);
2021-06-11 15:00:00 +10:00
+ // CraftBukkit end
}
2021-11-22 09:00:00 +11:00
public void addDuringCommandTeleport(EntityPlayer entityplayer) {
2021-12-11 00:11:02 +11:00
@@ -851,24 +964,36 @@
2021-11-22 09:00:00 +11:00
this.entityManager.addNewEntity(entityplayer);
2019-04-23 12:00:00 +10:00
}
2016-03-01 08:32:46 +11:00
2021-11-22 09:00:00 +11:00
- private boolean addEntity(Entity entity) {
2019-04-23 12:00:00 +10:00
+ // CraftBukkit start
2021-11-22 09:00:00 +11:00
+ private boolean addEntity(Entity entity, CreatureSpawnEvent.SpawnReason spawnReason) {
2021-06-11 15:00:00 +10:00
if (entity.isRemoved()) {
2021-11-22 09:00:00 +11:00
- WorldServer.LOGGER.warn("Tried to add entity {} but it was marked as removed already", EntityTypes.getKey(entity.getType()));
+ // WorldServer.LOGGER.warn("Tried to add entity {} but it was marked as removed already", EntityTypes.getKey(entity.getType())); // CraftBukkit
2019-04-23 12:00:00 +10:00
return false;
2016-03-01 08:32:46 +11:00
} else {
2019-04-23 12:00:00 +10:00
+ if (!CraftEventFactory.doEntityAddEventCalling(this, entity, spawnReason)) {
+ return false;
+ }
+ // CraftBukkit end
2021-06-11 15:00:00 +10:00
+
2021-11-22 09:00:00 +11:00
return this.entityManager.addNewEntity(entity);
2019-04-23 12:00:00 +10:00
}
}
2020-08-12 07:00:00 +10:00
2021-11-22 09:00:00 +11:00
public boolean tryAddFreshEntityWithPassengers(Entity entity) {
- Stream stream = entity.getSelfAndPassengers().map(Entity::getUUID);
2020-08-12 07:00:00 +10:00
+ // CraftBukkit start
2021-11-22 09:00:00 +11:00
+ return this.tryAddFreshEntityWithPassengers(entity, org.bukkit.event.entity.CreatureSpawnEvent.SpawnReason.DEFAULT);
2020-08-12 07:00:00 +10:00
+ }
+
2021-11-22 09:00:00 +11:00
+ public boolean tryAddFreshEntityWithPassengers(Entity entity, org.bukkit.event.entity.CreatureSpawnEvent.SpawnReason reason) {
2020-08-12 07:00:00 +10:00
+ // CraftBukkit end
2021-11-22 09:00:00 +11:00
+ Stream<UUID> stream = entity.getSelfAndPassengers().map(Entity::getUUID); // CraftBukkit - decompile error
2021-06-11 15:00:00 +10:00
PersistentEntitySectionManager persistententitysectionmanager = this.entityManager;
Objects.requireNonNull(this.entityManager);
2021-11-22 09:00:00 +11:00
if (stream.anyMatch(persistententitysectionmanager::isLoaded)) {
2020-08-12 07:00:00 +10:00
return false;
} else {
2021-11-22 09:00:00 +11:00
- this.addFreshEntityWithPassengers(entity);
+ this.addFreshEntityWithPassengers(entity, reason); // CraftBukkit
2020-11-17 13:41:47 +11:00
return true;
2020-08-12 07:00:00 +10:00
}
}
2021-12-11 00:11:02 +11:00
@@ -882,10 +1007,32 @@
2021-11-22 09:00:00 +11:00
entityplayer.remove(entity_removalreason);
2014-11-26 08:32:16 +11:00
}
2020-06-25 10:00:00 +10:00
+ // CraftBukkit start
+ public boolean strikeLightning(Entity entitylightning) {
+ return this.strikeLightning(entitylightning, LightningStrikeEvent.Cause.UNKNOWN);
2018-09-15 19:16:42 +10:00
+ }
+
2020-06-25 10:00:00 +10:00
+ public boolean strikeLightning(Entity entitylightning, LightningStrikeEvent.Cause cause) {
2021-05-09 16:33:34 +10:00
+ LightningStrikeEvent lightning = CraftEventFactory.callLightningStrikeEvent((org.bukkit.entity.LightningStrike) entitylightning.getBukkitEntity(), cause);
2014-11-26 08:32:16 +11:00
+
+ if (lightning.isCancelled()) {
2020-06-25 10:00:00 +10:00
+ return false;
2014-11-26 08:32:16 +11:00
+ }
2020-06-25 10:00:00 +10:00
+
2021-11-22 09:00:00 +11:00
+ return this.addFreshEntity(entitylightning);
2020-06-25 10:00:00 +10:00
+ }
+ // CraftBukkit end
+
@Override
2021-11-22 09:00:00 +11:00
public void destroyBlockProgress(int i, BlockPosition blockposition, int j) {
2019-04-23 12:00:00 +10:00
Iterator iterator = this.server.getPlayerList().getPlayers().iterator();
+ // CraftBukkit start
+ EntityHuman entityhuman = null;
+ Entity entity = this.getEntity(i);
+ if (entity instanceof EntityHuman) entityhuman = (EntityHuman) entity;
+ // CraftBukkit end
+
while (iterator.hasNext()) {
EntityPlayer entityplayer = (EntityPlayer) iterator.next();
2021-12-11 00:11:02 +11:00
@@ -894,6 +1041,12 @@
2021-11-22 09:00:00 +11:00
double d1 = (double) blockposition.getY() - entityplayer.getY();
double d2 = (double) blockposition.getZ() - entityplayer.getZ();
2019-04-23 12:00:00 +10:00
+ // CraftBukkit start
2021-11-29 09:18:52 +11:00
+ if (entityhuman != null && !entityplayer.getBukkitEntity().canSee(entityhuman.getBukkitEntity())) {
2019-04-23 12:00:00 +10:00
+ continue;
+ }
+ // CraftBukkit end
+
if (d0 * d0 + d1 * d1 + d2 * d2 < 1024.0D) {
2021-11-22 09:00:00 +11:00
entityplayer.connection.send(new PacketPlayOutBlockBreakAnimation(i, blockposition, j));
2019-04-23 12:00:00 +10:00
}
2021-12-11 00:11:02 +11:00
@@ -948,7 +1101,18 @@
2021-06-11 15:00:00 +10:00
Iterator iterator = this.navigatingMobs.iterator();
2021-02-14 10:08:27 +11:00
while (iterator.hasNext()) {
2021-06-11 15:00:00 +10:00
- EntityInsentient entityinsentient = (EntityInsentient) iterator.next();
2021-02-14 10:08:27 +11:00
+ // CraftBukkit start - fix SPIGOT-6362
2021-06-11 15:00:00 +10:00
+ EntityInsentient entityinsentient;
2021-02-14 10:08:27 +11:00
+ try {
2021-06-11 15:00:00 +10:00
+ entityinsentient = (EntityInsentient) iterator.next();
2021-02-14 10:08:27 +11:00
+ } catch (java.util.ConcurrentModificationException ex) {
+ // This can happen because the pathfinder update below may trigger a chunk load, which in turn may cause more navigators to register
+ // In this case we just run the update again across all the iterators as the chunk will then be loaded
+ // As this is a relative edge case it is much faster than copying navigators (on either read or write)
2021-11-22 09:00:00 +11:00
+ sendBlockUpdated(blockposition, iblockdata, iblockdata1, i);
2021-02-14 10:08:27 +11:00
+ return;
+ }
+ // CraftBukkit end
2021-06-11 15:00:00 +10:00
NavigationAbstract navigationabstract = entityinsentient.getNavigation();
2021-02-14 10:08:27 +11:00
2021-12-11 00:11:02 +11:00
if (navigationabstract.shouldRecomputePath(blockposition)) {
@@ -984,10 +1148,20 @@
2014-11-26 08:32:16 +11:00
2019-04-23 12:00:00 +10:00
@Override
2021-11-22 09:00:00 +11:00
public Explosion explode(@Nullable Entity entity, @Nullable DamageSource damagesource, @Nullable ExplosionDamageCalculator explosiondamagecalculator, double d0, double d1, double d2, float f, boolean flag, Explosion.Effect explosion_effect) {
2014-11-26 08:32:16 +11:00
+ // CraftBukkit start
2021-11-22 09:00:00 +11:00
+ Explosion explosion = super.explode(entity, damagesource, explosiondamagecalculator, d0, d1, d2, f, flag, explosion_effect);
2014-11-26 08:32:16 +11:00
+
+ if (explosion.wasCanceled) {
+ return explosion;
+ }
2015-02-26 22:41:06 +00:00
+
2014-11-26 08:32:16 +11:00
+ /* Remove
2020-06-25 10:00:00 +10:00
Explosion explosion = new Explosion(this, entity, damagesource, explosiondamagecalculator, d0, d1, d2, f, flag, explosion_effect);
2018-07-15 10:00:00 +10:00
2021-11-22 09:00:00 +11:00
explosion.explode();
explosion.finalizeExplosion(false);
2014-11-26 08:32:16 +11:00
+ */
+ // CraftBukkit end - TODO: Check if explosions are still properly implemented
2019-04-23 12:00:00 +10:00
if (explosion_effect == Explosion.Effect.NONE) {
2021-11-22 09:00:00 +11:00
explosion.clearToBlow();
2014-11-26 08:32:16 +11:00
}
2021-12-11 00:11:02 +11:00
@@ -1068,13 +1242,20 @@
2014-11-26 08:32:16 +11:00
}
2021-11-22 09:00:00 +11:00
public <T extends ParticleParam> int sendParticles(T t0, double d0, double d1, double d2, int i, double d3, double d4, double d5, double d6) {
2019-12-11 09:00:00 +11:00
- PacketPlayOutWorldParticles packetplayoutworldparticles = new PacketPlayOutWorldParticles(t0, false, d0, d1, d2, (float) d3, (float) d4, (float) d5, (float) d6, i);
2014-11-26 08:32:16 +11:00
+ // CraftBukkit - visibility api support
2018-08-26 18:26:09 +10:00
+ return sendParticles(null, t0, d0, d1, d2, i, d3, d4, d5, d6, false);
2014-11-26 08:32:16 +11:00
+ }
2015-02-26 22:41:06 +00:00
+
2018-08-26 18:26:09 +10:00
+ public <T extends ParticleParam> int sendParticles(EntityPlayer sender, T t0, double d0, double d1, double d2, int i, double d3, double d4, double d5, double d6, boolean force) {
2019-12-11 09:00:00 +11:00
+ PacketPlayOutWorldParticles packetplayoutworldparticles = new PacketPlayOutWorldParticles(t0, force, d0, d1, d2, (float) d3, (float) d4, (float) d5, (float) d6, i);
2014-11-26 08:32:16 +11:00
+ // CraftBukkit end
2018-07-15 10:00:00 +10:00
int j = 0;
2014-11-26 08:32:16 +11:00
2018-07-15 10:00:00 +10:00
for (int k = 0; k < this.players.size(); ++k) {
EntityPlayer entityplayer = (EntityPlayer) this.players.get(k);
2014-12-28 14:56:18 +00:00
+ if (sender != null && !entityplayer.getBukkitEntity().canSee(sender.getBukkitEntity())) continue; // CraftBukkit
2014-11-26 08:32:16 +11:00
2021-11-22 09:00:00 +11:00
- if (this.sendParticles(entityplayer, false, d0, d1, d2, packetplayoutworldparticles)) {
+ if (this.sendParticles(entityplayer, force, d0, d1, d2, packetplayoutworldparticles)) { // CraftBukkit
2018-07-15 10:00:00 +10:00
++j;
2018-08-26 18:26:09 +10:00
}
}
2021-12-11 00:11:02 +11:00
@@ -1125,7 +1306,7 @@
2020-06-30 10:29:41 +10:00
@Nullable
2021-11-22 09:00:00 +11:00
public BlockPosition findNearestMapFeature(StructureGenerator<?> structuregenerator, BlockPosition blockposition, int i, boolean flag) {
- return !this.server.getWorldData().worldGenSettings().generateFeatures() ? null : this.getChunkSource().getGenerator().findNearestMapFeature(this, structuregenerator, blockposition, i, flag);
+ return !this.serverLevelData.worldGenSettings().generateFeatures() ? null : this.getChunkSource().getGenerator().findNearestMapFeature(this, structuregenerator, blockposition, i, flag); // CraftBukkit
2020-06-30 10:29:41 +10:00
}
@Nullable
2021-12-11 00:11:02 +11:00
@@ -1162,11 +1343,21 @@
2021-06-11 15:00:00 +10:00
@Nullable
2019-04-23 12:00:00 +10:00
@Override
2021-11-22 09:00:00 +11:00
public WorldMap getMapData(String s) {
- return (WorldMap) this.getServer().overworld().getDataStorage().get(WorldMap::load, s);
+ // CraftBukkit start
+ return (WorldMap) this.getServer().overworld().getDataStorage().get((nbttagcompound) -> {
2019-04-23 12:00:00 +10:00
+ // We only get here when the data file exists, but is not a valid map
2021-11-22 09:00:00 +11:00
+ WorldMap newMap = WorldMap.load(nbttagcompound);
2021-06-11 15:00:00 +10:00
+ newMap.id = s;
2019-04-23 12:00:00 +10:00
+ MapInitializeEvent event = new MapInitializeEvent(newMap.mapView);
+ Bukkit.getServer().getPluginManager().callEvent(event);
+ return newMap;
2021-06-11 15:00:00 +10:00
+ }, s);
2021-11-22 09:00:00 +11:00
+ // CraftBukkit end
2019-04-23 12:00:00 +10:00
}
2021-06-11 15:00:00 +10:00
@Override
2021-11-22 09:00:00 +11:00
public void setMapData(String s, WorldMap worldmap) {
2021-06-11 15:00:00 +10:00
+ worldmap.id = s; // CraftBukkit
2021-11-22 09:00:00 +11:00
this.getServer().overworld().getDataStorage().set(s, worldmap);
2021-03-16 09:00:00 +11:00
}
2021-12-11 00:11:02 +11:00
@@ -1478,6 +1669,11 @@
2020-06-25 10:00:00 +10:00
@Override
2021-11-22 09:00:00 +11:00
public void blockUpdated(BlockPosition blockposition, Block block) {
if (!this.isDebug()) {
2020-06-25 10:00:00 +10:00
+ // CraftBukkit start
+ if (populating) {
+ return;
+ }
+ // CraftBukkit end
2021-11-22 09:00:00 +11:00
this.updateNeighborsAt(blockposition, block);
2020-06-25 10:00:00 +10:00
}
2021-12-11 00:11:02 +11:00
@@ -1497,12 +1693,12 @@
2020-06-25 10:00:00 +10:00
}
2021-11-22 09:00:00 +11:00
public boolean isFlat() {
- return this.server.getWorldData().worldGenSettings().isFlatWorld();
+ return this.serverLevelData.worldGenSettings().isFlatWorld(); // CraftBukkit
2020-06-25 10:00:00 +10:00
}
@Override
public long getSeed() {
2021-11-22 09:00:00 +11:00
- return this.server.getWorldData().worldGenSettings().seed();
+ return this.serverLevelData.worldGenSettings().seed(); // CraftBukkit
2020-06-25 10:00:00 +10:00
}
@Nullable
2021-12-11 00:11:02 +11:00
@@ -1530,7 +1726,7 @@
2021-11-22 09:00:00 +11:00
private static <T> String getTypeCount(Iterable<T> iterable, Function<T, String> function) {
2020-11-03 07:00:00 +11:00
try {
2021-06-11 15:00:00 +10:00
Object2IntOpenHashMap<String> object2intopenhashmap = new Object2IntOpenHashMap();
- Iterator iterator = iterable.iterator();
+ Iterator<T> iterator = iterable.iterator(); // CraftBukkit - decompile error
2020-11-03 07:00:00 +11:00
while (iterator.hasNext()) {
T t0 = iterator.next();
2021-12-11 00:11:02 +11:00
@@ -1539,7 +1735,7 @@
2021-06-11 15:00:00 +10:00
object2intopenhashmap.addTo(s, 1);
2020-11-03 07:00:00 +11:00
}
2021-06-11 15:00:00 +10:00
- return (String) object2intopenhashmap.object2IntEntrySet().stream().sorted(Comparator.comparing(Entry::getIntValue).reversed()).limit(5L).map((entry) -> {
+ return (String) object2intopenhashmap.object2IntEntrySet().stream().sorted(Comparator.comparing(Entry<String>::getIntValue).reversed()).limit(5L).map((entry) -> { // CraftBukkit - decompile error
String s1 = (String) entry.getKey();
return s1 + ":" + entry.getIntValue();
2021-12-11 00:11:02 +11:00
@@ -1550,17 +1746,33 @@
2020-06-25 10:00:00 +10:00
}
2021-11-22 09:00:00 +11:00
public static void makeObsidianPlatform(WorldServer worldserver) {
2020-06-25 10:00:00 +10:00
+ // CraftBukkit start
2021-11-22 09:00:00 +11:00
+ WorldServer.makeObsidianPlatform(worldserver, null);
2020-06-25 10:00:00 +10:00
+ }
+
2021-11-22 09:00:00 +11:00
+ public static void makeObsidianPlatform(WorldServer worldserver, Entity entity) {
2020-06-25 10:00:00 +10:00
+ // CraftBukkit end
2021-06-11 15:00:00 +10:00
BlockPosition blockposition = WorldServer.END_SPAWN_POINT;
2020-06-25 10:00:00 +10:00
int i = blockposition.getX();
int j = blockposition.getY() - 2;
int k = blockposition.getZ();
+ // CraftBukkit start
+ org.bukkit.craftbukkit.util.BlockStateListPopulator blockList = new org.bukkit.craftbukkit.util.BlockStateListPopulator(worldserver);
2021-11-22 09:00:00 +11:00
BlockPosition.betweenClosed(i - 2, j + 1, k - 2, i + 2, j + 3, k + 2).forEach((blockposition1) -> {
- worldserver.setBlockAndUpdate(blockposition1, Blocks.AIR.defaultBlockState());
+ blockList.setBlock(blockposition1, Blocks.AIR.defaultBlockState(), 3);
2020-06-25 10:00:00 +10:00
});
2021-11-22 09:00:00 +11:00
BlockPosition.betweenClosed(i - 2, j, k - 2, i + 2, j, k + 2).forEach((blockposition1) -> {
- worldserver.setBlockAndUpdate(blockposition1, Blocks.OBSIDIAN.defaultBlockState());
+ blockList.setBlock(blockposition1, Blocks.OBSIDIAN.defaultBlockState(), 3);
2020-06-25 10:00:00 +10:00
});
+ org.bukkit.World bworld = worldserver.getWorld();
+ org.bukkit.event.world.PortalCreateEvent portalEvent = new org.bukkit.event.world.PortalCreateEvent((List<org.bukkit.block.BlockState>) (List) blockList.getList(), bworld, (entity == null) ? null : entity.getBukkitEntity(), org.bukkit.event.world.PortalCreateEvent.CreateReason.END_PLATFORM);
+
2021-06-11 21:33:49 +10:00
+ worldserver.getCraftServer().getPluginManager().callEvent(portalEvent);
2020-06-25 10:00:00 +10:00
+ if (!portalEvent.isCancelled()) {
+ blockList.updateList();
+ }
+ // CraftBukkit end
}
2021-06-11 15:00:00 +10:00
@Override
2021-12-11 00:11:02 +11:00
@@ -1666,6 +1878,7 @@
2021-06-11 15:00:00 +10:00
}
}
+ entity.valid = true; // CraftBukkit
}
2021-11-22 09:00:00 +11:00
public void onTrackingEnd(Entity entity) {
2021-12-11 00:11:02 +11:00
@@ -1707,6 +1920,14 @@
2021-11-22 09:00:00 +11:00
gameeventlistenerregistrar.onListenerRemoved(entity.level);
2021-06-11 15:00:00 +10:00
}
2021-12-04 08:23:26 +11:00
+ // CraftBukkit start
+ entity.valid = false;
+ if (!(entity instanceof EntityPlayer)) {
+ for (EntityPlayer player : players) {
+ player.getBukkitEntity().onEntityRemove(entity);
+ }
+ }
+ // CraftBukkit end
2021-06-11 15:00:00 +10:00
}
}
2020-06-25 10:00:00 +10:00
}