Send block destructions on a per-chunk basis to nearby players

This commit is contained in:
Sollace 2021-02-01 20:33:39 +02:00
parent 1c775bfd65
commit d02cd4226f
2 changed files with 66 additions and 9 deletions

View file

@ -1,13 +1,20 @@
package com.minelittlepony.unicopia; package com.minelittlepony.unicopia;
import java.util.Arrays;
import java.util.List;
import java.util.stream.Collectors;
import com.minelittlepony.unicopia.network.Channel; import com.minelittlepony.unicopia.network.Channel;
import com.minelittlepony.unicopia.network.MsgBlockDestruction; import com.minelittlepony.unicopia.network.MsgBlockDestruction;
import it.unimi.dsi.fastutil.longs.Long2ObjectMap; import it.unimi.dsi.fastutil.longs.Long2ObjectMap;
import it.unimi.dsi.fastutil.longs.Long2ObjectOpenHashMap; import it.unimi.dsi.fastutil.longs.Long2ObjectOpenHashMap;
import net.minecraft.block.BlockState; import net.minecraft.block.BlockState;
import net.minecraft.entity.player.PlayerEntity;
import net.minecraft.server.world.ServerWorld; import net.minecraft.server.world.ServerWorld;
import net.minecraft.server.world.ThreadedAnvilChunkStorage;
import net.minecraft.util.math.BlockPos; import net.minecraft.util.math.BlockPos;
import net.minecraft.util.math.ChunkPos;
import net.minecraft.world.World; import net.minecraft.world.World;
public class BlockDestructionManager { public class BlockDestructionManager {
@ -19,7 +26,7 @@ public class BlockDestructionManager {
private final World world; private final World world;
private final Long2ObjectMap<Destruction> destructions = new Long2ObjectOpenHashMap<>(); private final Long2ObjectMap<Chunk> destructions = new Long2ObjectOpenHashMap<>();
private final Object locker = new Object(); private final Object locker = new Object();
@ -28,7 +35,11 @@ public class BlockDestructionManager {
} }
public int getBlockDestruction(BlockPos pos) { public int getBlockDestruction(BlockPos pos) {
return destructions.getOrDefault(pos.asLong(), emptyDestruction).amount; return getDestructions(pos).getBlockDestruction(pos);
}
private Chunk getDestructions(BlockPos pos) {
return destructions.computeIfAbsent(new ChunkPos(pos).toLong(), Chunk::new);
} }
public void clearBlockDestruction(BlockPos pos) { public void clearBlockDestruction(BlockPos pos) {
@ -37,7 +48,7 @@ public class BlockDestructionManager {
public void setBlockDestruction(BlockPos pos, int amount) { public void setBlockDestruction(BlockPos pos, int amount) {
synchronized (locker) { synchronized (locker) {
destructions.computeIfAbsent(pos.asLong(), p -> new Destruction()).set(amount); getDestructions(pos).setBlockDestruction(pos, amount);
} }
} }
@ -61,15 +72,61 @@ public class BlockDestructionManager {
destructions.long2ObjectEntrySet().removeIf(entry -> entry.getValue().tick()); destructions.long2ObjectEntrySet().removeIf(entry -> entry.getValue().tick());
if (world instanceof ServerWorld) { if (world instanceof ServerWorld) {
Long2ObjectMap<Integer> sent = new Long2ObjectOpenHashMap<>(); destructions.forEach((chunkPos, chunk) -> chunk.sendUpdates((ServerWorld)world));
destructions.forEach((p, item) -> { }
}
}
private class Chunk {
private final Long2ObjectMap<Destruction> destructions = new Long2ObjectOpenHashMap<>();
private final long pos;
Chunk(long pos) {
this.pos = pos;
}
public int getBlockDestruction(BlockPos pos) {
return destructions.getOrDefault(pos.asLong(), emptyDestruction).amount;
}
public void setBlockDestruction(BlockPos pos, int amount) {
destructions.computeIfAbsent(pos.asLong(), p -> new Destruction()).set(amount);
}
boolean tick() {
destructions.long2ObjectEntrySet().removeIf(e -> e.getValue().tick());
return destructions.isEmpty();
}
void sendUpdates(ServerWorld world) {
if (!world.getChunkManager().isChunkLoaded(ChunkPos.getPackedX(pos), ChunkPos.getPackedZ(pos))) {
return;
}
ThreadedAnvilChunkStorage storage = world.getChunkManager().threadedAnvilChunkStorage;
List<PlayerEntity> players = storage.getPlayersWatchingChunk(new ChunkPos(pos), false).collect(Collectors.toList());
if (!players.isEmpty()) {
Long2ObjectOpenHashMap<Integer> values = new Long2ObjectOpenHashMap<>();
destructions.forEach((blockPos, item) -> {
if (item.dirty) { if (item.dirty) {
sent.put(p.longValue(), (Integer)item.amount); item.dirty = false;
values.put(blockPos.longValue(), (Integer)item.amount);
} }
}); });
if (!sent.isEmpty()) {
Channel.SERVER_BLOCK_DESTRUCTION.send(world, new MsgBlockDestruction(sent)); MsgBlockDestruction msg = new MsgBlockDestruction(values);
if (msg.toBuffer().writerIndex() > 1048576) {
throw new IllegalStateException("Payload may not be larger than 1048576 bytes. Here's what we were trying to send: ["
+ values.size() + "]\n"
+ Arrays.toString(values.values().stream().mapToInt(Integer::intValue).toArray()));
} }
players.forEach(player -> Channel.SERVER_BLOCK_DESTRUCTION.send(player, msg));
} }
} }
} }

View file

@ -22,7 +22,7 @@ public interface Channel {
MPacketType<MsgOtherPlayerCapabilities> SERVER_OTHER_PLAYER_CAPABILITIES = serverToClients(new Identifier("unicopia", "other_player_capabilities"), MsgOtherPlayerCapabilities::new); MPacketType<MsgOtherPlayerCapabilities> SERVER_OTHER_PLAYER_CAPABILITIES = serverToClients(new Identifier("unicopia", "other_player_capabilities"), MsgOtherPlayerCapabilities::new);
CPacketType<MsgSpawnProjectile> SERVER_SPAWN_PROJECTILE = serverToClient(new Identifier("unicopia", "projectile_entity"), MsgSpawnProjectile::new); CPacketType<MsgSpawnProjectile> SERVER_SPAWN_PROJECTILE = serverToClient(new Identifier("unicopia", "projectile_entity"), MsgSpawnProjectile::new);
MPacketType<MsgBlockDestruction> SERVER_BLOCK_DESTRUCTION = serverToClients(new Identifier("unicopia", "block_destruction"), MsgBlockDestruction::new); CPacketType<MsgBlockDestruction> SERVER_BLOCK_DESTRUCTION = serverToClient(new Identifier("unicopia", "block_destruction"), MsgBlockDestruction::new);
static void bootstrap() { } static void bootstrap() { }