mirror of
https://github.com/PaperMC/Paper.git
synced 2025-01-17 22:01:26 +01:00
123 lines
3.9 KiB
Java
123 lines
3.9 KiB
Java
package org.bukkit.craftbukkit;
|
|
|
|
import java.lang.ref.WeakReference;
|
|
import net.minecraft.server.ChunkPosition;
|
|
|
|
import net.minecraft.server.WorldServer;
|
|
|
|
import org.bukkit.Chunk;
|
|
import org.bukkit.World;
|
|
import org.bukkit.block.Block;
|
|
import org.bukkit.block.BlockState;
|
|
import org.bukkit.craftbukkit.block.CraftBlock;
|
|
import org.bukkit.entity.Entity;
|
|
import org.bukkit.craftbukkit.util.ConcurrentSoftMap;
|
|
import org.bukkit.ChunkSnapshot;
|
|
|
|
public class CraftChunk implements Chunk {
|
|
private WeakReference<net.minecraft.server.Chunk> weakChunk;
|
|
private final ConcurrentSoftMap<Integer, Block> cache = new ConcurrentSoftMap<Integer, Block>();
|
|
private WorldServer worldServer;
|
|
private int x;
|
|
private int z;
|
|
|
|
public CraftChunk(net.minecraft.server.Chunk chunk) {
|
|
this.weakChunk = new WeakReference<net.minecraft.server.Chunk>(chunk);
|
|
worldServer = (WorldServer) getHandle().world;
|
|
x = getHandle().x;
|
|
z = getHandle().z;
|
|
}
|
|
|
|
public World getWorld() {
|
|
return worldServer.getWorld();
|
|
}
|
|
|
|
public net.minecraft.server.Chunk getHandle() {
|
|
net.minecraft.server.Chunk c = weakChunk.get();
|
|
if (c == null) {
|
|
c = worldServer.getChunkAt(x,z);
|
|
weakChunk = new WeakReference<net.minecraft.server.Chunk>(c);
|
|
}
|
|
return c;
|
|
}
|
|
|
|
void breakLink() {
|
|
weakChunk.clear();
|
|
}
|
|
|
|
public int getX() {
|
|
return x;
|
|
}
|
|
|
|
public int getZ() {
|
|
return z;
|
|
}
|
|
|
|
@Override
|
|
public String toString() {
|
|
return "CraftChunk{" + "x=" + getX() + "z=" + getZ() + '}';
|
|
}
|
|
|
|
public Block getBlock(int x, int y, int z) {
|
|
int pos = (x & 0xF) << 11 | (z & 0xF) << 7 | (y & 0x7F);
|
|
Block block = this.cache.get(pos);
|
|
if (block == null) {
|
|
Block newBlock = new CraftBlock(this, (getX() << 4) | (x & 0xF), y & 0x7F, (getZ() << 4) | (z & 0xF));
|
|
Block oldBlock = this.cache.put(pos, newBlock);
|
|
if(oldBlock == null) {
|
|
block = newBlock;
|
|
} else {
|
|
block = oldBlock;
|
|
}
|
|
}
|
|
return block;
|
|
}
|
|
|
|
public Entity[] getEntities() {
|
|
int count = 0, index = 0;
|
|
net.minecraft.server.Chunk chunk = getHandle();
|
|
for (int i = 0; i < 8; i++) {
|
|
count += chunk.entitySlices[i].size();
|
|
}
|
|
|
|
Entity[] entities = new Entity[count];
|
|
for (int i = 0; i < 8; i++) {
|
|
for (Object obj: chunk.entitySlices[i].toArray()) {
|
|
if (!(obj instanceof net.minecraft.server.Entity)) {
|
|
continue;
|
|
}
|
|
entities[index++] = ((net.minecraft.server.Entity) obj).getBukkitEntity();
|
|
}
|
|
}
|
|
return entities;
|
|
}
|
|
|
|
public BlockState[] getTileEntities() {
|
|
int index = 0;
|
|
net.minecraft.server.Chunk chunk = getHandle();
|
|
BlockState[] entities = new BlockState[chunk.tileEntities.size()];
|
|
for (Object obj : chunk.tileEntities.keySet().toArray()) {
|
|
if (!(obj instanceof ChunkPosition)) {
|
|
continue;
|
|
}
|
|
ChunkPosition position = (ChunkPosition) obj;
|
|
entities[index++] = worldServer.getWorld().getBlockAt(position.x + (chunk.x << 4), position.y, position.z + (chunk.z << 4)).getState();
|
|
}
|
|
return entities;
|
|
}
|
|
|
|
/**
|
|
* Capture thread-safe read-only snapshot of chunk data
|
|
* @return ChunkSnapshot
|
|
*/
|
|
public ChunkSnapshot getChunkSnapshot() {
|
|
net.minecraft.server.Chunk chunk = getHandle();
|
|
byte[] buf = new byte[32768 + 16384 + 16384 + 16384]; // Get big enough buffer for whole chunk
|
|
chunk.a(buf, 0, 0, 0, 16, 128, 16, 0); // Get whole chunk
|
|
byte[] hmap = new byte[256]; // Get copy of height map
|
|
System.arraycopy(chunk.h, 0, hmap, 0, 256);
|
|
World w = getWorld();
|
|
return new CraftChunkSnapshot(getX(), getZ(), w.getName(), w.getFullTime(), buf, hmap);
|
|
}
|
|
|
|
} |