Add shouldLoadChunks in BlockBatch#flush

This commit is contained in:
KrystilizeNevaDies 2021-02-12 10:37:53 +10:00
parent 04c792c302
commit ee8815d2f8

View File

@ -6,6 +6,8 @@ import net.minestom.server.instance.Instance;
import net.minestom.server.instance.InstanceContainer; import net.minestom.server.instance.InstanceContainer;
import net.minestom.server.instance.block.CustomBlock; import net.minestom.server.instance.block.CustomBlock;
import net.minestom.server.utils.block.CustomBlockUtils; import net.minestom.server.utils.block.CustomBlockUtils;
import net.minestom.server.utils.chunk.ChunkUtils;
import org.jetbrains.annotations.NotNull; import org.jetbrains.annotations.NotNull;
import org.jetbrains.annotations.Nullable; import org.jetbrains.annotations.Nullable;
@ -13,6 +15,7 @@ import java.util.ArrayList;
import java.util.HashMap; import java.util.HashMap;
import java.util.List; import java.util.List;
import java.util.Map; import java.util.Map;
import java.util.Set;
import java.util.concurrent.atomic.AtomicInteger; import java.util.concurrent.atomic.AtomicInteger;
/** /**
@ -76,44 +79,70 @@ public class BlockBatch implements InstanceBatch {
} }
public void flush(@Nullable Runnable callback) { public void flush(@Nullable Runnable callback) {
this.flush(callback, false);
}
public void flush(@Nullable Runnable callback, boolean shouldLoadChunks) {
synchronized (data) { synchronized (data) {
AtomicInteger counter = new AtomicInteger(); // Load chunks if applicable
for (Map.Entry<Chunk, List<BlockData>> entry : data.entrySet()) { if (shouldLoadChunks) {
final Chunk chunk = entry.getKey();
final List<BlockData> dataList = entry.getValue();
BLOCK_BATCH_POOL.execute(() -> {
synchronized (chunk) {
if (!chunk.isLoaded())
return;
if (batchOption.isFullChunk()) { // Get chunks
chunk.reset(); Chunk[] chunks = data.keySet().toArray(new Chunk[data.size()]);
}
for (BlockData data : dataList) { // Get chunk indexs
data.apply(chunk); Long[] indexs = new Long[chunks.length];
} for (int i = 0; i < chunks.length; i++) {
indexs[i] = ChunkUtils.getChunkIndex(chunks[i].getChunkX(), chunks[i].getChunkZ());
}
// Refresh chunk for viewers // Load all chunks + flush block data
if (batchOption.isFullChunk()) { ChunkUtils.optionalLoadAll(instance, null, null, (chunk) -> {
chunk.sendChunk(); flushBlockData(callback);
} else { });
chunk.sendChunkUpdate(); } else {
} flushBlockData(callback);
}
}
}
final boolean isLast = counter.incrementAndGet() == data.size(); private void flushBlockData(@Nullable Runnable callback) {
AtomicInteger counter = new AtomicInteger();
// Execute the callback if this was the last chunk to process for (Map.Entry<Chunk, List<BlockData>> entry : data.entrySet()) {
if (isLast) { final Chunk chunk = entry.getKey();
this.instance.refreshLastBlockChangeTime(); final List<BlockData> dataList = entry.getValue();
if (callback != null) { BLOCK_BATCH_POOL.execute(() -> {
this.instance.scheduleNextTick(inst -> callback.run()); synchronized (chunk) {
} if (!chunk.isLoaded())
} return;
if (batchOption.isFullChunk()) {
chunk.reset();
} }
});
} for (BlockData data : dataList) {
data.apply(chunk);
}
// Refresh chunk for viewers
if (batchOption.isFullChunk()) {
chunk.sendChunk();
} else {
chunk.sendChunkUpdate();
}
final boolean isLast = counter.incrementAndGet() == data.size();
// Execute the callback if this was the last chunk to process
if (isLast) {
this.instance.refreshLastBlockChangeTime();
if (callback != null) {
this.instance.scheduleNextTick(inst -> callback.run());
}
}
}
});
} }
} }