cn.nukkit.scheduler.BlockUpdateScheduler Maven / Gradle / Ivy
package cn.nukkit.scheduler;
import cn.nukkit.block.Block;
import cn.nukkit.level.Level;
import cn.nukkit.math.AxisAlignedBB;
import cn.nukkit.math.NukkitMath;
import cn.nukkit.math.Vector3;
import cn.nukkit.utils.BlockUpdateEntry;
import com.google.common.collect.Maps;
import java.util.*;
public class BlockUpdateScheduler {
private final Level level;
private long lastTick;
private Map> queuedUpdates;
private Set pendingUpdates;
public BlockUpdateScheduler(Level level, long currentTick) {
queuedUpdates = Maps.newHashMap(); // Change to ConcurrentHashMap if this needs to be concurrent
lastTick = currentTick;
this.level = level;
}
public synchronized void tick(long currentTick) {
// Should only perform once, unless ticks were skipped
if (currentTick - lastTick < Short.MAX_VALUE) {// Arbitrary
for (long tick = lastTick + 1; tick <= currentTick; tick++) {
perform(tick);
}
} else {
ArrayList times = new ArrayList<>(queuedUpdates.keySet());
Collections.sort(times);
for (long tick : times) {
if (tick <= currentTick) {
perform(tick);
} else {
break;
}
}
}
lastTick = currentTick;
}
private void perform(long tick) {
try {
lastTick = tick;
Set updates = pendingUpdates = queuedUpdates.remove(tick);
if (updates != null) {
Iterator updateIterator = updates.iterator();
while (updateIterator.hasNext()) {
BlockUpdateEntry entry = updateIterator.next();
Vector3 pos = entry.pos;
if (level.isChunkLoaded(NukkitMath.floorDouble(pos.x) >> 4, NukkitMath.floorDouble(pos.z) >> 4)) {
Block block = level.getBlock(entry.pos, entry.block.layer);
updateIterator.remove();
if (Block.equals(block, entry.block, false) && entry.checkBlockWhenUpdate) {
block.onUpdate(Level.BLOCK_UPDATE_SCHEDULED);
}else{
block.onUpdate(Level.BLOCK_UPDATE_SCHEDULED);
}
} else {
level.scheduleUpdate(entry.block, entry.pos, 0);
}
}
}
} finally {
pendingUpdates = null;
}
}
public Set getPendingBlockUpdates(AxisAlignedBB boundingBox) {
Set set = null;
for (Map.Entry> tickEntries : this.queuedUpdates.entrySet()) {
LinkedHashSet tickSet = tickEntries.getValue();
for (BlockUpdateEntry update : tickSet) {
Vector3 pos = update.pos;
if (pos.getX() >= boundingBox.getMinX() && pos.getX() < boundingBox.getMaxX() && pos.getZ() >= boundingBox.getMinZ() && pos.getZ() < boundingBox.getMaxZ()) {
if (set == null) {
set = new LinkedHashSet<>();
}
set.add(update);
}
}
}
return set;
}
public boolean isBlockTickPending(Vector3 pos, Block block) {
Set tmpUpdates = pendingUpdates;
if (tmpUpdates == null || tmpUpdates.isEmpty()) return false;
return tmpUpdates.contains(new BlockUpdateEntry(pos, block));
}
private long getMinTime(BlockUpdateEntry entry) {
return Math.max(entry.delay, lastTick + 1);
}
public void add(BlockUpdateEntry entry) {
long time = getMinTime(entry);
LinkedHashSet updateSet = queuedUpdates.get(time);
if (updateSet == null) {
LinkedHashSet tmp = queuedUpdates.putIfAbsent(time, updateSet = new LinkedHashSet<>());
if (tmp != null) updateSet = tmp;
}
updateSet.add(entry);
}
public boolean contains(BlockUpdateEntry entry) {
for (Map.Entry> tickUpdateSet : queuedUpdates.entrySet()) {
if (tickUpdateSet.getValue().contains(entry)) {
return true;
}
}
return false;
}
public boolean remove(BlockUpdateEntry entry) {
for (Map.Entry> tickUpdateSet : queuedUpdates.entrySet()) {
if (tickUpdateSet.getValue().remove(entry)) {
return true;
}
}
return false;
}
public boolean remove(Vector3 pos) {
for (Map.Entry> tickUpdateSet : queuedUpdates.entrySet()) {
if (tickUpdateSet.getValue().remove(pos)) {
return true;
}
}
return false;
}
}