Paper/src/main/java/net/minecraft/server/World.java

1873 Zeilen
57 KiB
Java

2011-02-07 06:56:07 +01:00
package net.minecraft.server;
import java.util.ArrayList;
import java.util.HashSet;
import java.util.Iterator;
import java.util.List;
import java.util.Random;
import java.util.Set;
import java.util.TreeSet;
2011-02-07 10:43:50 +01:00
// CraftBukkit start
import org.bukkit.craftbukkit.CraftWorld;
import org.bukkit.craftbukkit.event.CraftEventFactory;
2011-02-07 10:43:50 +01:00
import org.bukkit.event.block.BlockCanBuildEvent;
import org.bukkit.event.block.BlockPhysicsEvent;
import org.bukkit.event.entity.CreatureSpawnEvent;
2011-02-07 10:43:50 +01:00
// CraftBukkit end
2011-02-07 06:56:07 +01:00
public class World implements IBlockAccess {
public boolean a = false;
2011-02-23 03:37:56 +01:00
private List u = new ArrayList();
public List entityList = new ArrayList();
2011-02-23 03:37:56 +01:00
private List v = new ArrayList();
private TreeSet w = new TreeSet();
private Set x = new HashSet();
2011-02-07 06:56:07 +01:00
public List c = new ArrayList();
public List players = new ArrayList();
2011-02-23 03:37:56 +01:00
private long y = 16777215L;
public int e = 0;
protected int f = (new Random()).nextInt();
protected int g = 1013904223;
public boolean h = false;
private long z = System.currentTimeMillis();
protected int i = 40;
public int spawnMonsters;
public Random random = new Random();
2011-02-23 03:37:56 +01:00
public boolean l = false;
public final WorldProvider worldProvider;
2011-02-23 03:37:56 +01:00
protected List n = new ArrayList();
public IChunkProvider chunkProvider; // CraftBukkit protected->public
2011-02-23 03:37:56 +01:00
protected final IDataManager p;
public WorldData worldData; // CraftBukkit protected->public
public boolean isLoading;
2011-02-23 03:37:56 +01:00
private boolean A;
private ArrayList B = new ArrayList();
private int C = 0;
public boolean allowMonsters = true; // CraftBukkit private->public
public boolean allowAnimals = true; // CraftBukkit private->public
2011-02-23 03:37:56 +01:00
static int s = 0;
private Set F = new HashSet();
private int G;
private List H;
2011-02-07 06:56:07 +01:00
public boolean isStatic;
public WorldChunkManager getWorldChunkManager() {
return this.worldProvider.b;
2011-02-07 06:56:07 +01:00
}
2011-02-23 03:37:56 +01:00
public World(IDataManager idatamanager, String s, long i, WorldProvider worldprovider) {
this.G = this.random.nextInt(12000);
2011-02-23 03:37:56 +01:00
this.H = new ArrayList();
2011-02-07 06:56:07 +01:00
this.isStatic = false;
2011-02-23 03:37:56 +01:00
this.p = idatamanager;
this.worldData = idatamanager.c();
this.l = this.worldData == null;
2011-02-07 06:56:07 +01:00
if (worldprovider != null) {
this.worldProvider = worldprovider;
} else if (this.worldData != null && this.worldData.h() == -1) {
this.worldProvider = new WorldProviderHell();
2011-02-23 03:37:56 +01:00
} else {
this.worldProvider = new WorldProvider();
2011-02-07 06:56:07 +01:00
}
boolean flag = false;
if (this.worldData == null) {
this.worldData = new WorldData(i, s);
2011-02-07 06:56:07 +01:00
flag = true;
2011-02-23 03:37:56 +01:00
} else {
this.worldData.a(s);
2011-02-07 06:56:07 +01:00
}
this.worldProvider.a(this);
this.chunkProvider = this.b();
2011-02-07 06:56:07 +01:00
if (flag) {
2011-03-31 22:40:00 +02:00
this.c();
2011-02-07 06:56:07 +01:00
}
2011-03-31 22:40:00 +02:00
this.g();
2011-02-07 06:56:07 +01:00
}
2011-02-23 03:37:56 +01:00
protected IChunkProvider b() {
IChunkLoader ichunkloader = this.p.a(this.worldProvider);
2011-02-23 03:37:56 +01:00
return new ChunkProviderLoadOrGenerate(this, ichunkloader, this.worldProvider.c());
2011-02-07 06:56:07 +01:00
}
2011-03-31 22:40:00 +02:00
protected void c() {
this.isLoading = true;
2011-03-31 22:40:00 +02:00
int i = 0;
byte b0 = 64;
int j;
for (j = 0; !this.worldProvider.a(i, j); j += this.random.nextInt(64) - this.random.nextInt(64)) {
i += this.random.nextInt(64) - this.random.nextInt(64);
2011-03-31 22:40:00 +02:00
}
this.worldData.setSpawn(i, b0, j);
this.isLoading = false;
2011-03-31 22:40:00 +02:00
}
2011-02-07 06:56:07 +01:00
public int a(int i, int j) {
int k;
for (k = 63; !this.isEmpty(i, k + 1, j); ++k) {
;
}
return this.getTypeId(i, k, j);
}
public void save(boolean flag, IProgressUpdate iprogressupdate) {
if (this.chunkProvider.b()) {
2011-02-07 06:56:07 +01:00
if (iprogressupdate != null) {
iprogressupdate.a("Saving level");
}
2011-03-31 22:40:00 +02:00
this.t();
2011-02-07 06:56:07 +01:00
if (iprogressupdate != null) {
iprogressupdate.b("Saving chunks");
}
this.chunkProvider.saveChunks(flag, iprogressupdate);
2011-02-07 06:56:07 +01:00
}
}
2011-03-31 22:40:00 +02:00
private void t() {
this.j();
this.p.a(this.worldData, this.players);
2011-02-07 06:56:07 +01:00
}
public int getTypeId(int i, int j, int k) {
return i >= -32000000 && k >= -32000000 && i < 32000000 && k <= 32000000 ? (j < 0 ? 0 : (j >= 128 ? 0 : this.getChunkAt(i >> 4, k >> 4).getTypeId(i & 15, j, k & 15))) : 0;
2011-02-07 06:56:07 +01:00
}
public boolean isEmpty(int i, int j, int k) {
return this.getTypeId(i, j, k) == 0;
}
public boolean isLoaded(int i, int j, int k) {
return j >= 0 && j < 128 ? this.isChunkLoaded(i >> 4, k >> 4) : false;
2011-02-07 06:56:07 +01:00
}
public boolean a(int i, int j, int k, int l) {
return this.a(i - l, j - l, k - l, i + l, j + l, k + l);
}
public boolean a(int i, int j, int k, int l, int i1, int j1) {
if (i1 >= 0 && j < 128) {
i >>= 4;
j >>= 4;
k >>= 4;
l >>= 4;
i1 >>= 4;
j1 >>= 4;
for (int k1 = i; k1 <= l; ++k1) {
for (int l1 = k; l1 <= j1; ++l1) {
if (!this.isChunkLoaded(k1, l1)) {
2011-02-07 06:56:07 +01:00
return false;
}
}
}
return true;
} else {
return false;
}
}
private boolean isChunkLoaded(int i, int j) {
return this.chunkProvider.isChunkLoaded(i, j);
2011-02-07 06:56:07 +01:00
}
public Chunk b(int i, int j) {
return this.getChunkAt(i >> 4, j >> 4);
2011-02-07 06:56:07 +01:00
}
2011-02-07 09:43:51 +01:00
// CraftBukkit start
2011-02-07 06:56:07 +01:00
Chunk lastChunkAccessed;
int lastXAccessed = Integer.MIN_VALUE;
int lastZAccessed = Integer.MIN_VALUE;
2011-02-10 12:56:30 +01:00
final Object chunkLock = new Object();
public Chunk getChunkAt(int i, int j) {
2011-02-10 12:56:30 +01:00
Chunk result = null;
synchronized (chunkLock) {
if (lastChunkAccessed == null || lastXAccessed != i || lastZAccessed != j) {
lastXAccessed = i;
lastZAccessed = j;
lastChunkAccessed = this.chunkProvider.getOrCreateChunk(i, j);
2011-02-10 12:56:30 +01:00
}
result = lastChunkAccessed;
2011-02-07 06:56:07 +01:00
}
2011-02-10 12:56:30 +01:00
return result;
2011-02-07 06:56:07 +01:00
}
2011-02-07 09:43:51 +01:00
// CraftBukkit end
2011-02-07 06:56:07 +01:00
public boolean setRawTypeIdAndData(int i, int j, int k, int l, int i1) {
2011-02-07 06:56:07 +01:00
if (i >= -32000000 && k >= -32000000 && i < 32000000 && k <= 32000000) {
if (j < 0) {
return false;
} else if (j >= 128) {
return false;
} else {
Chunk chunk = this.getChunkAt(i >> 4, k >> 4);
2011-02-07 06:56:07 +01:00
return chunk.a(i & 15, j, k & 15, l, i1);
}
} else {
return false;
}
}
public boolean setRawTypeId(int i, int j, int k, int l) {
2011-02-07 06:56:07 +01:00
if (i >= -32000000 && k >= -32000000 && i < 32000000 && k <= 32000000) {
if (j < 0) {
return false;
} else if (j >= 128) {
return false;
} else {
Chunk chunk = this.getChunkAt(i >> 4, k >> 4);
2011-02-07 06:56:07 +01:00
return chunk.a(i & 15, j, k & 15, l);
}
} else {
return false;
}
}
public Material getMaterial(int i, int j, int k) {
int l = this.getTypeId(i, j, k);
return l == 0 ? Material.AIR : Block.byId[l].material;
}
public int getData(int i, int j, int k) {
if (i >= -32000000 && k >= -32000000 && i < 32000000 && k <= 32000000) {
if (j < 0) {
return 0;
} else if (j >= 128) {
return 0;
} else {
Chunk chunk = this.getChunkAt(i >> 4, k >> 4);
2011-02-07 06:56:07 +01:00
i &= 15;
k &= 15;
return chunk.getData(i, j, k);
2011-02-07 06:56:07 +01:00
}
} else {
return 0;
}
}
public void setData(int i, int j, int k, int l) {
if (this.setRawData(i, j, k, l)) {
this.update(i, j, k, this.getTypeId(i, j, k));
2011-02-07 06:56:07 +01:00
}
}
public boolean setRawData(int i, int j, int k, int l) {
2011-02-07 06:56:07 +01:00
if (i >= -32000000 && k >= -32000000 && i < 32000000 && k <= 32000000) {
if (j < 0) {
return false;
} else if (j >= 128) {
return false;
} else {
Chunk chunk = this.getChunkAt(i >> 4, k >> 4);
2011-02-07 06:56:07 +01:00
i &= 15;
k &= 15;
chunk.b(i, j, k, l);
return true;
}
} else {
return false;
}
}
public boolean setTypeId(int i, int j, int k, int l) {
2011-03-14 05:47:41 +01:00
// Craftbukkit start
int old = this.getTypeId(i, j, k);
if (this.setRawTypeId(i, j, k, l)) {
this.update(i, j, k, l == 0 ? old : l);
2011-02-07 06:56:07 +01:00
return true;
} else {
return false;
}
2011-03-14 05:47:41 +01:00
// Craftbukkit end
2011-02-07 06:56:07 +01:00
}
public boolean setTypeIdAndData(int i, int j, int k, int l, int i1) {
2011-03-14 05:47:41 +01:00
// Craftbukkit start
int old = this.getTypeId(i, j, k);
if (this.setRawTypeIdAndData(i, j, k, l, i1)) {
this.update(i, j, k, l == 0 ? old : l);
2011-02-07 06:56:07 +01:00
return true;
} else {
return false;
}
2011-03-14 05:47:41 +01:00
// Craftbukkit end
2011-02-07 06:56:07 +01:00
}
public void notify(int i, int j, int k) {
2011-02-23 03:37:56 +01:00
for (int l = 0; l < this.n.size(); ++l) {
((IWorldAccess) this.n.get(l)).a(i, j, k);
2011-02-07 06:56:07 +01:00
}
}
protected void update(int i, int j, int k, int l) {
this.notify(i, j, k);
this.applyPhysics(i, j, k, l);
2011-02-07 06:56:07 +01:00
}
public void g(int i, int j, int k, int l) {
if (k > l) {
int i1 = l;
l = k;
k = i1;
}
this.b(i, k, j, i, l, j);
}
public void h(int i, int j, int k) {
2011-02-23 03:37:56 +01:00
for (int l = 0; l < this.n.size(); ++l) {
((IWorldAccess) this.n.get(l)).a(i, j, k, i, j, k);
2011-02-07 06:56:07 +01:00
}
}
public void b(int i, int j, int k, int l, int i1, int j1) {
2011-02-23 03:37:56 +01:00
for (int k1 = 0; k1 < this.n.size(); ++k1) {
((IWorldAccess) this.n.get(k1)).a(i, j, k, l, i1, j1);
2011-02-07 06:56:07 +01:00
}
}
public void applyPhysics(int i, int j, int k, int l) {
2011-02-23 03:37:56 +01:00
this.k(i - 1, j, k, l);
this.k(i + 1, j, k, l);
this.k(i, j - 1, k, l);
this.k(i, j + 1, k, l);
this.k(i, j, k - 1, l);
this.k(i, j, k + 1, l);
2011-02-07 06:56:07 +01:00
}
2011-02-23 03:37:56 +01:00
private void k(int i, int j, int k, int l) {
if (!this.h && !this.isStatic) {
2011-02-07 06:56:07 +01:00
Block block = Block.byId[this.getTypeId(i, j, k)];
if (block != null) {
// CraftBukkit start
2011-02-07 10:43:50 +01:00
CraftWorld world = ((WorldServer) this).getWorld();
if (world != null) {
BlockPhysicsEvent event = new BlockPhysicsEvent(world.getBlockAt(i, j, k), l);
2011-02-07 10:43:50 +01:00
((WorldServer) this).getServer().getPluginManager().callEvent(event);
2011-02-23 13:56:36 +01:00
if (event.isCancelled()) {
return;
}
}
// CraftBukkit stop
block.doPhysics(this, i, j, k, l);
2011-02-07 06:56:07 +01:00
}
}
}
public boolean isChunkLoaded(int i, int j, int k) {
return this.getChunkAt(i >> 4, k >> 4).c(i & 15, j, k & 15);
2011-02-07 06:56:07 +01:00
}
public int getLightLevel(int i, int j, int k) {
2011-02-07 06:56:07 +01:00
return this.a(i, j, k, true);
}
public int a(int i, int j, int k, boolean flag) {
if (i >= -32000000 && k >= -32000000 && i < 32000000 && k <= 32000000) {
int l;
if (flag) {
l = this.getTypeId(i, j, k);
if (l == Block.STEP.id || l == Block.SOIL.id) {
int i1 = this.a(i, j + 1, k, false);
int j1 = this.a(i + 1, j, k, false);
int k1 = this.a(i - 1, j, k, false);
int l1 = this.a(i, j, k + 1, false);
int i2 = this.a(i, j, k - 1, false);
if (j1 > i1) {
i1 = j1;
}
if (k1 > i1) {
i1 = k1;
}
if (l1 > i1) {
i1 = l1;
}
if (i2 > i1) {
i1 = i2;
}
return i1;
}
}
if (j < 0) {
return 0;
} else if (j >= 128) {
2011-02-23 03:37:56 +01:00
l = 15 - this.e;
2011-02-07 06:56:07 +01:00
if (l < 0) {
l = 0;
}
return l;
} else {
Chunk chunk = this.getChunkAt(i >> 4, k >> 4);
2011-02-07 06:56:07 +01:00
i &= 15;
k &= 15;
2011-02-23 03:37:56 +01:00
return chunk.c(i, j, k, this.e);
2011-02-07 06:56:07 +01:00
}
} else {
return 15;
}
}
public boolean k(int i, int j, int k) {
if (i >= -32000000 && k >= -32000000 && i < 32000000 && k <= 32000000) {
if (j < 0) {
return false;
} else if (j >= 128) {
return true;
} else if (!this.isChunkLoaded(i >> 4, k >> 4)) {
2011-02-07 06:56:07 +01:00
return false;
} else {
Chunk chunk = this.getChunkAt(i >> 4, k >> 4);
2011-02-07 06:56:07 +01:00
i &= 15;
k &= 15;
return chunk.c(i, j, k);
}
} else {
return false;
}
}
public int getHighestBlockYAt(int i, int j) {
2011-02-07 06:56:07 +01:00
if (i >= -32000000 && j >= -32000000 && i < 32000000 && j <= 32000000) {
if (!this.isChunkLoaded(i >> 4, j >> 4)) {
2011-02-07 06:56:07 +01:00
return 0;
} else {
Chunk chunk = this.getChunkAt(i >> 4, j >> 4);
2011-02-07 06:56:07 +01:00
return chunk.b(i & 15, j & 15);
}
} else {
return 0;
}
}
public void a(EnumSkyBlock enumskyblock, int i, int j, int k, int l) {
if (!this.worldProvider.e || enumskyblock != EnumSkyBlock.SKY) {
if (this.isLoaded(i, j, k)) {
2011-02-07 06:56:07 +01:00
if (enumskyblock == EnumSkyBlock.SKY) {
if (this.k(i, j, k)) {
l = 15;
}
} else if (enumskyblock == EnumSkyBlock.BLOCK) {
int i1 = this.getTypeId(i, j, k);
if (Block.s[i1] > l) {
l = Block.s[i1];
}
}
if (this.a(enumskyblock, i, j, k) != l) {
this.a(enumskyblock, i, j, k, i, j, k);
}
}
}
}
public int a(EnumSkyBlock enumskyblock, int i, int j, int k) {
if (j >= 0 && j < 128 && i >= -32000000 && k >= -32000000 && i < 32000000 && k <= 32000000) {
int l = i >> 4;
int i1 = k >> 4;
if (!this.isChunkLoaded(l, i1)) {
2011-02-07 06:56:07 +01:00
return 0;
} else {
Chunk chunk = this.getChunkAt(l, i1);
2011-02-07 06:56:07 +01:00
return chunk.a(enumskyblock, i & 15, j, k & 15);
}
} else {
return enumskyblock.c;
}
}
public void b(EnumSkyBlock enumskyblock, int i, int j, int k, int l) {
if (i >= -32000000 && k >= -32000000 && i < 32000000 && k <= 32000000) {
if (j >= 0) {
if (j < 128) {
if (this.isChunkLoaded(i >> 4, k >> 4)) {
Chunk chunk = this.getChunkAt(i >> 4, k >> 4);
2011-02-07 06:56:07 +01:00
chunk.a(enumskyblock, i & 15, j, k & 15, l);
2011-02-23 03:37:56 +01:00
for (int i1 = 0; i1 < this.n.size(); ++i1) {
((IWorldAccess) this.n.get(i1)).a(i, j, k);
2011-02-07 06:56:07 +01:00
}
}
}
}
}
}
public float l(int i, int j, int k) {
return this.worldProvider.f[this.getLightLevel(i, j, k)];
2011-02-07 06:56:07 +01:00
}
2011-03-31 22:40:00 +02:00
public boolean d() {
2011-02-23 03:37:56 +01:00
return this.e < 4;
2011-02-07 06:56:07 +01:00
}
public MovingObjectPosition a(Vec3D vec3d, Vec3D vec3d1) {
return this.rayTrace(vec3d, vec3d1, false);
2011-02-07 06:56:07 +01:00
}
public MovingObjectPosition rayTrace(Vec3D vec3d, Vec3D vec3d1, boolean flag) {
2011-02-07 06:56:07 +01:00
if (!Double.isNaN(vec3d.a) && !Double.isNaN(vec3d.b) && !Double.isNaN(vec3d.c)) {
if (!Double.isNaN(vec3d1.a) && !Double.isNaN(vec3d1.b) && !Double.isNaN(vec3d1.c)) {
int i = MathHelper.floor(vec3d1.a);
int j = MathHelper.floor(vec3d1.b);
int k = MathHelper.floor(vec3d1.c);
int l = MathHelper.floor(vec3d.a);
int i1 = MathHelper.floor(vec3d.b);
int j1 = MathHelper.floor(vec3d.c);
2011-02-07 06:56:07 +01:00
int k1 = 200;
while (k1-- >= 0) {
if (Double.isNaN(vec3d.a) || Double.isNaN(vec3d.b) || Double.isNaN(vec3d.c)) {
return null;
}
if (l == i && i1 == j && j1 == k) {
return null;
}
double d0 = 999.0D;
double d1 = 999.0D;
double d2 = 999.0D;
if (i > l) {
d0 = (double) l + 1.0D;
}
if (i < l) {
d0 = (double) l + 0.0D;
}
if (j > i1) {
d1 = (double) i1 + 1.0D;
}
if (j < i1) {
d1 = (double) i1 + 0.0D;
}
if (k > j1) {
d2 = (double) j1 + 1.0D;
}
if (k < j1) {
d2 = (double) j1 + 0.0D;
}
double d3 = 999.0D;
double d4 = 999.0D;
double d5 = 999.0D;
double d6 = vec3d1.a - vec3d.a;
double d7 = vec3d1.b - vec3d.b;
double d8 = vec3d1.c - vec3d.c;
if (d0 != 999.0D) {
d3 = (d0 - vec3d.a) / d6;
}
if (d1 != 999.0D) {
d4 = (d1 - vec3d.b) / d7;
}
if (d2 != 999.0D) {
d5 = (d2 - vec3d.c) / d8;
}
boolean flag1 = false;
byte b0;
if (d3 < d4 && d3 < d5) {
if (i > l) {
b0 = 4;
} else {
b0 = 5;
}
vec3d.a = d0;
vec3d.b += d7 * d3;
vec3d.c += d8 * d3;
} else if (d4 < d5) {
if (j > i1) {
b0 = 0;
} else {
b0 = 1;
}
vec3d.a += d6 * d4;
vec3d.b = d1;
vec3d.c += d8 * d4;
} else {
if (k > j1) {
b0 = 2;
} else {
b0 = 3;
}
vec3d.a += d6 * d5;
vec3d.b += d7 * d5;
vec3d.c = d2;
}
Vec3D vec3d2 = Vec3D.create(vec3d.a, vec3d.b, vec3d.c);
2011-02-07 06:56:07 +01:00
l = (int) (vec3d2.a = (double) MathHelper.floor(vec3d.a));
2011-02-07 06:56:07 +01:00
if (b0 == 5) {
--l;
++vec3d2.a;
}
i1 = (int) (vec3d2.b = (double) MathHelper.floor(vec3d.b));
2011-02-07 06:56:07 +01:00
if (b0 == 1) {
--i1;
++vec3d2.b;
}
j1 = (int) (vec3d2.c = (double) MathHelper.floor(vec3d.c));
2011-02-07 06:56:07 +01:00
if (b0 == 3) {
--j1;
++vec3d2.c;
}
int l1 = this.getTypeId(l, i1, j1);
int i2 = this.getData(l, i1, j1);
Block block = Block.byId[l1];
if (l1 > 0 && block.a(i2, flag)) {
MovingObjectPosition movingobjectposition = block.a(this, l, i1, j1, vec3d, vec3d1);
if (movingobjectposition != null) {
return movingobjectposition;
}
}
}
return null;
} else {
return null;
}
} else {
return null;
}
}
public void makeSound(Entity entity, String s, float f, float f1) {
2011-02-23 03:37:56 +01:00
for (int i = 0; i < this.n.size(); ++i) {
((IWorldAccess) this.n.get(i)).a(s, entity.locX, entity.locY - (double) entity.height, entity.locZ, f, f1);
2011-02-07 06:56:07 +01:00
}
}
public void makeSound(double d0, double d1, double d2, String s, float f, float f1) {
2011-02-23 03:37:56 +01:00
for (int i = 0; i < this.n.size(); ++i) {
((IWorldAccess) this.n.get(i)).a(s, d0, d1, d2, f, f1);
2011-02-07 06:56:07 +01:00
}
}
public void a(String s, int i, int j, int k) {
2011-02-23 03:37:56 +01:00
for (int l = 0; l < this.n.size(); ++l) {
((IWorldAccess) this.n.get(l)).a(s, i, j, k);
2011-02-07 06:56:07 +01:00
}
}
public void a(String s, double d0, double d1, double d2, double d3, double d4, double d5) {
2011-02-23 03:37:56 +01:00
for (int i = 0; i < this.n.size(); ++i) {
((IWorldAccess) this.n.get(i)).a(s, d0, d1, d2, d3, d4, d5);
2011-02-07 06:56:07 +01:00
}
}
public boolean addEntity(Entity entity) {
int i = MathHelper.floor(entity.locX / 16.0D);
int j = MathHelper.floor(entity.locZ / 16.0D);
2011-02-07 06:56:07 +01:00
boolean flag = false;
if (entity instanceof EntityHuman) {
flag = true;
}
2011-02-23 13:56:36 +01:00
// CraftBukkit start
if (entity instanceof EntityLiving && !(entity instanceof EntityPlayer)) {
CreatureSpawnEvent event = CraftEventFactory.callCreatureSpawnEvent((EntityLiving) entity);
if (event.isCancelled()) {
return false;
}
}
2011-02-23 03:37:56 +01:00
// CraftBukkit end
if (!flag && !this.isChunkLoaded(i, j)) {
2011-02-07 06:56:07 +01:00
return false;
} else {
if (entity instanceof EntityHuman) {
EntityHuman entityhuman = (EntityHuman) entity;
this.players.add(entityhuman);
this.everyoneSleeping();
2011-02-07 06:56:07 +01:00
}
this.getChunkAt(i, j).a(entity);
this.entityList.add(entity);
2011-02-07 06:56:07 +01:00
this.b(entity);
return true;
}
}
protected void b(Entity entity) {
2011-02-23 03:37:56 +01:00
for (int i = 0; i < this.n.size(); ++i) {
((IWorldAccess) this.n.get(i)).a(entity);
2011-02-07 06:56:07 +01:00
}
}
protected void c(Entity entity) {
2011-02-23 03:37:56 +01:00
for (int i = 0; i < this.n.size(); ++i) {
((IWorldAccess) this.n.get(i)).b(entity);
2011-02-07 06:56:07 +01:00
}
}
public void kill(Entity entity) {
2011-02-07 06:56:07 +01:00
if (entity.passenger != null) {
entity.passenger.mount((Entity) null);
2011-02-07 06:56:07 +01:00
}
if (entity.vehicle != null) {
entity.mount((Entity) null);
2011-02-07 06:56:07 +01:00
}
entity.die();
2011-02-07 06:56:07 +01:00
if (entity instanceof EntityHuman) {
this.players.remove((EntityHuman) entity);
this.everyoneSleeping();
2011-02-07 06:56:07 +01:00
}
}
public void removeEntity(Entity entity) {
entity.die();
2011-02-07 06:56:07 +01:00
if (entity instanceof EntityHuman) {
this.players.remove((EntityHuman) entity);
this.everyoneSleeping();
2011-02-07 06:56:07 +01:00
}
int i = entity.chunkX;
int j = entity.chunkZ;
if (entity.bA && this.isChunkLoaded(i, j)) {
this.getChunkAt(i, j).b(entity);
2011-02-07 06:56:07 +01:00
}
this.entityList.remove(entity);
2011-02-07 06:56:07 +01:00
this.c(entity);
}
public void addIWorldAccess(IWorldAccess iworldaccess) {
2011-02-23 03:37:56 +01:00
this.n.add(iworldaccess);
2011-02-07 06:56:07 +01:00
}
public List getEntities(Entity entity, AxisAlignedBB axisalignedbb) {
2011-02-23 03:37:56 +01:00
this.B.clear();
int i = MathHelper.floor(axisalignedbb.a);
int j = MathHelper.floor(axisalignedbb.d + 1.0D);
int k = MathHelper.floor(axisalignedbb.b);
int l = MathHelper.floor(axisalignedbb.e + 1.0D);
int i1 = MathHelper.floor(axisalignedbb.c);
int j1 = MathHelper.floor(axisalignedbb.f + 1.0D);
2011-02-07 06:56:07 +01:00
for (int k1 = i; k1 < j; ++k1) {
for (int l1 = i1; l1 < j1; ++l1) {
if (this.isLoaded(k1, 64, l1)) {
2011-02-07 06:56:07 +01:00
for (int i2 = k - 1; i2 < l; ++i2) {
Block block = Block.byId[this.getTypeId(k1, i2, l1)];
if (block != null) {
2011-02-23 03:37:56 +01:00
block.a(this, k1, i2, l1, axisalignedbb, this.B);
2011-02-07 06:56:07 +01:00
}
}
}
}
}
double d0 = 0.25D;
List list = this.b(entity, axisalignedbb.b(d0, d0, d0));
for (int j2 = 0; j2 < list.size(); ++j2) {
2011-02-23 03:37:56 +01:00
AxisAlignedBB axisalignedbb1 = ((Entity) list.get(j2)).d();
2011-02-07 06:56:07 +01:00
if (axisalignedbb1 != null && axisalignedbb1.a(axisalignedbb)) {
2011-02-23 03:37:56 +01:00
this.B.add(axisalignedbb1);
2011-02-07 06:56:07 +01:00
}
2011-02-23 03:37:56 +01:00
axisalignedbb1 = entity.a_((Entity) list.get(j2));
2011-02-07 06:56:07 +01:00
if (axisalignedbb1 != null && axisalignedbb1.a(axisalignedbb)) {
2011-02-23 03:37:56 +01:00
this.B.add(axisalignedbb1);
2011-02-07 06:56:07 +01:00
}
}
2011-02-23 03:37:56 +01:00
return this.B;
2011-02-07 06:56:07 +01:00
}
public int a(float f) {
float f1 = this.b(f);
float f2 = 1.0F - (MathHelper.cos(f1 * 3.1415927F * 2.0F) * 2.0F + 0.5F);
2011-02-07 06:56:07 +01:00
if (f2 < 0.0F) {
f2 = 0.0F;
}
if (f2 > 1.0F) {
f2 = 1.0F;
}
return (int) (f2 * 11.0F);
}
public float b(float f) {
return this.worldProvider.a(this.worldData.f(), f);
2011-02-07 06:56:07 +01:00
}
public int e(int i, int j) {
Chunk chunk = this.b(i, j);
int k;
for (k = 127; this.getMaterial(i, k, j).isSolid() && k > 0; --k) {
;
}
i &= 15;
for (j &= 15; k > 0; --k) {
int l = chunk.getTypeId(i, k, j);
2011-02-07 06:56:07 +01:00
if (l != 0 && (Block.byId[l].material.isSolid() || Block.byId[l].material.isLiquid())) {
return k + 1;
}
}
return -1;
}
2011-02-23 03:37:56 +01:00
public void c(int i, int j, int k, int l, int i1) {
2011-02-07 06:56:07 +01:00
NextTickListEntry nextticklistentry = new NextTickListEntry(i, j, k, l);
byte b0 = 8;
if (this.a) {
if (this.a(nextticklistentry.a - b0, nextticklistentry.b - b0, nextticklistentry.c - b0, nextticklistentry.a + b0, nextticklistentry.b + b0, nextticklistentry.c + b0)) {
2011-02-23 03:37:56 +01:00
int j1 = this.getTypeId(nextticklistentry.a, nextticklistentry.b, nextticklistentry.c);
2011-02-07 06:56:07 +01:00
2011-02-23 03:37:56 +01:00
if (j1 == nextticklistentry.d && j1 > 0) {
Block.byId[j1].a(this, nextticklistentry.a, nextticklistentry.b, nextticklistentry.c, this.random);
2011-02-07 06:56:07 +01:00
}
}
} else {
if (this.a(i - b0, j - b0, k - b0, i + b0, j + b0, k + b0)) {
if (l > 0) {
nextticklistentry.a((long) i1 + this.worldData.f());
2011-02-07 06:56:07 +01:00
}
2011-02-23 03:37:56 +01:00
if (!this.x.contains(nextticklistentry)) {
this.x.add(nextticklistentry);
this.w.add(nextticklistentry);
2011-02-07 06:56:07 +01:00
}
}
}
}
public void cleanUp() {
this.entityList.removeAll(this.v);
2011-02-07 06:56:07 +01:00
int i;
Entity entity;
int j;
int k;
2011-02-23 03:37:56 +01:00
for (i = 0; i < this.v.size(); ++i) {
entity = (Entity) this.v.get(i);
2011-02-07 06:56:07 +01:00
j = entity.chunkX;
k = entity.chunkZ;
if (entity.bA && this.isChunkLoaded(j, k)) {
this.getChunkAt(j, k).b(entity);
2011-02-07 06:56:07 +01:00
}
}
2011-02-23 03:37:56 +01:00
for (i = 0; i < this.v.size(); ++i) {
this.c((Entity) this.v.get(i));
2011-02-07 06:56:07 +01:00
}
2011-02-23 03:37:56 +01:00
this.v.clear();
2011-02-07 06:56:07 +01:00
for (i = 0; i < this.entityList.size(); ++i) {
entity = (Entity) this.entityList.get(i);
2011-02-07 06:56:07 +01:00
if (entity.vehicle != null) {
if (!entity.vehicle.dead && entity.vehicle.passenger == entity) {
continue;
}
entity.vehicle.passenger = null;
entity.vehicle = null;
}
if (!entity.dead) {
this.playerJoinedWorld(entity);
2011-02-07 06:56:07 +01:00
}
if (entity.dead) {
j = entity.chunkX;
k = entity.chunkZ;
if (entity.bA && this.isChunkLoaded(j, k)) {
this.getChunkAt(j, k).b(entity);
2011-02-07 06:56:07 +01:00
}
this.entityList.remove(i--);
2011-02-07 06:56:07 +01:00
this.c(entity);
}
}
for (i = 0; i < this.c.size(); ++i) {
TileEntity tileentity = (TileEntity) this.c.get(i);
2011-02-23 03:37:56 +01:00
tileentity.i_();
2011-02-07 06:56:07 +01:00
}
}
public void playerJoinedWorld(Entity entity) {
this.entityJoinedWorld(entity, true);
2011-02-07 06:56:07 +01:00
}
public void entityJoinedWorld(Entity entity, boolean flag) {
int i = MathHelper.floor(entity.locX);
int j = MathHelper.floor(entity.locZ);
2011-02-07 06:56:07 +01:00
byte b0 = 32;
if (!flag || this.a(i - b0, 0, j - b0, i + b0, 128, j + b0)) {
2011-02-23 03:37:56 +01:00
entity.bi = entity.locX;
entity.bj = entity.locY;
entity.bk = entity.locZ;
2011-02-07 06:56:07 +01:00
entity.lastYaw = entity.yaw;
entity.lastPitch = entity.pitch;
2011-02-23 03:37:56 +01:00
if (flag && entity.bA) {
2011-02-07 06:56:07 +01:00
if (entity.vehicle != null) {
2011-03-31 22:40:00 +02:00
entity.o_();
2011-02-07 06:56:07 +01:00
} else {
2011-02-23 03:37:56 +01:00
entity.f_();
2011-02-07 06:56:07 +01:00
}
}
if (Double.isNaN(entity.locX) || Double.isInfinite(entity.locX)) {
2011-02-23 03:37:56 +01:00
entity.locX = entity.bi;
2011-02-07 06:56:07 +01:00
}
if (Double.isNaN(entity.locY) || Double.isInfinite(entity.locY)) {
2011-02-23 03:37:56 +01:00
entity.locY = entity.bj;
2011-02-07 06:56:07 +01:00
}
if (Double.isNaN(entity.locZ) || Double.isInfinite(entity.locZ)) {
2011-02-23 03:37:56 +01:00
entity.locZ = entity.bk;
2011-02-07 06:56:07 +01:00
}
if (Double.isNaN((double) entity.pitch) || Double.isInfinite((double) entity.pitch)) {
entity.pitch = entity.lastPitch;
}
if (Double.isNaN((double) entity.yaw) || Double.isInfinite((double) entity.yaw)) {
entity.yaw = entity.lastYaw;
}
int k = MathHelper.floor(entity.locX / 16.0D);
int l = MathHelper.floor(entity.locY / 16.0D);
int i1 = MathHelper.floor(entity.locZ / 16.0D);
2011-02-07 06:56:07 +01:00
2011-02-23 03:37:56 +01:00
if (!entity.bA || entity.chunkX != k || entity.bC != l || entity.chunkZ != i1) {
if (entity.bA && this.isChunkLoaded(entity.chunkX, entity.chunkZ)) {
this.getChunkAt(entity.chunkX, entity.chunkZ).a(entity, entity.bC);
2011-02-07 06:56:07 +01:00
}
if (this.isChunkLoaded(k, i1)) {
2011-02-23 03:37:56 +01:00
entity.bA = true;
this.getChunkAt(k, i1).a(entity);
2011-02-07 06:56:07 +01:00
} else {
2011-02-23 03:37:56 +01:00
entity.bA = false;
2011-02-07 06:56:07 +01:00
}
}
2011-02-23 03:37:56 +01:00
if (flag && entity.bA && entity.passenger != null) {
2011-02-07 06:56:07 +01:00
if (!entity.passenger.dead && entity.passenger.vehicle == entity) {
this.playerJoinedWorld(entity.passenger);
2011-02-07 06:56:07 +01:00
} else {
entity.passenger.vehicle = null;
entity.passenger = null;
}
}
}
}
public boolean containsEntity(AxisAlignedBB axisalignedbb) {
2011-02-07 06:56:07 +01:00
List list = this.b((Entity) null, axisalignedbb);
for (int i = 0; i < list.size(); ++i) {
Entity entity = (Entity) list.get(i);
2011-03-31 22:40:00 +02:00
if (!entity.dead && entity.aD) {
2011-02-07 06:56:07 +01:00
return false;
}
}
return true;
}
public boolean b(AxisAlignedBB axisalignedbb) {
int i = MathHelper.floor(axisalignedbb.a);
int j = MathHelper.floor(axisalignedbb.d + 1.0D);
int k = MathHelper.floor(axisalignedbb.b);
int l = MathHelper.floor(axisalignedbb.e + 1.0D);
int i1 = MathHelper.floor(axisalignedbb.c);
int j1 = MathHelper.floor(axisalignedbb.f + 1.0D);
2011-02-07 06:56:07 +01:00
if (axisalignedbb.a < 0.0D) {
--i;
}
if (axisalignedbb.b < 0.0D) {
--k;
}
if (axisalignedbb.c < 0.0D) {
--i1;
}
for (int k1 = i; k1 < j; ++k1) {
for (int l1 = k; l1 < l; ++l1) {
for (int i2 = i1; i2 < j1; ++i2) {
Block block = Block.byId[this.getTypeId(k1, l1, i2)];
if (block != null && block.material.isLiquid()) {
return true;
}
}
}
}
return false;
}
public boolean c(AxisAlignedBB axisalignedbb) {
int i = MathHelper.floor(axisalignedbb.a);
int j = MathHelper.floor(axisalignedbb.d + 1.0D);
int k = MathHelper.floor(axisalignedbb.b);
int l = MathHelper.floor(axisalignedbb.e + 1.0D);
int i1 = MathHelper.floor(axisalignedbb.c);
int j1 = MathHelper.floor(axisalignedbb.f + 1.0D);
2011-02-07 06:56:07 +01:00
if (this.a(i, k, i1, j, l, j1)) {
for (int k1 = i; k1 < j; ++k1) {
for (int l1 = k; l1 < l; ++l1) {
for (int i2 = i1; i2 < j1; ++i2) {
int j2 = this.getTypeId(k1, l1, i2);
if (j2 == Block.FIRE.id || j2 == Block.LAVA.id || j2 == Block.STATIONARY_LAVA.id) {
return true;
}
}
}
}
}
return false;
}
public boolean a(AxisAlignedBB axisalignedbb, Material material, Entity entity) {
int i = MathHelper.floor(axisalignedbb.a);
int j = MathHelper.floor(axisalignedbb.d + 1.0D);
int k = MathHelper.floor(axisalignedbb.b);
int l = MathHelper.floor(axisalignedbb.e + 1.0D);
int i1 = MathHelper.floor(axisalignedbb.c);
int j1 = MathHelper.floor(axisalignedbb.f + 1.0D);
2011-02-07 06:56:07 +01:00
if (!this.a(i, k, i1, j, l, j1)) {
return false;
} else {
boolean flag = false;
Vec3D vec3d = Vec3D.create(0.0D, 0.0D, 0.0D);
2011-02-07 06:56:07 +01:00
for (int k1 = i; k1 < j; ++k1) {
for (int l1 = k; l1 < l; ++l1) {
for (int i2 = i1; i2 < j1; ++i2) {
Block block = Block.byId[this.getTypeId(k1, l1, i2)];
if (block != null && block.material == material) {
double d0 = (double) ((float) (l1 + 1) - BlockFluids.c(this.getData(k1, l1, i2)));
if ((double) l >= d0) {
flag = true;
block.a(this, k1, l1, i2, entity, vec3d);
}
}
}
}
}
if (vec3d.c() > 0.0D) {
vec3d = vec3d.b();
double d1 = 0.0040D;
entity.motX += vec3d.a * d1;
entity.motY += vec3d.b * d1;
entity.motZ += vec3d.c * d1;
}
return flag;
}
}
public boolean a(AxisAlignedBB axisalignedbb, Material material) {
int i = MathHelper.floor(axisalignedbb.a);
int j = MathHelper.floor(axisalignedbb.d + 1.0D);
int k = MathHelper.floor(axisalignedbb.b);
int l = MathHelper.floor(axisalignedbb.e + 1.0D);
int i1 = MathHelper.floor(axisalignedbb.c);
int j1 = MathHelper.floor(axisalignedbb.f + 1.0D);
2011-02-07 06:56:07 +01:00
for (int k1 = i; k1 < j; ++k1) {
for (int l1 = k; l1 < l; ++l1) {
for (int i2 = i1; i2 < j1; ++i2) {
Block block = Block.byId[this.getTypeId(k1, l1, i2)];
if (block != null && block.material == material) {
return true;
}
}
}
}
return false;
}
public boolean b(AxisAlignedBB axisalignedbb, Material material) {
int i = MathHelper.floor(axisalignedbb.a);
int j = MathHelper.floor(axisalignedbb.d + 1.0D);
int k = MathHelper.floor(axisalignedbb.b);
int l = MathHelper.floor(axisalignedbb.e + 1.0D);
int i1 = MathHelper.floor(axisalignedbb.c);
int j1 = MathHelper.floor(axisalignedbb.f + 1.0D);
2011-02-07 06:56:07 +01:00
for (int k1 = i; k1 < j; ++k1) {
for (int l1 = k; l1 < l; ++l1) {
for (int i2 = i1; i2 < j1; ++i2) {
Block block = Block.byId[this.getTypeId(k1, l1, i2)];
if (block != null && block.material == material) {
int j2 = this.getData(k1, l1, i2);
double d0 = (double) (l1 + 1);
if (j2 < 8) {
d0 = (double) (l1 + 1) - (double) j2 / 8.0D;
}
if (d0 >= axisalignedbb.b) {
return true;
}
}
}
}
}
return false;
}
public Explosion a(Entity entity, double d0, double d1, double d2, float f) {
return this.createExplosion(entity, d0, d1, d2, f, false);
2011-02-07 06:56:07 +01:00
}
public Explosion createExplosion(Entity entity, double d0, double d1, double d2, float f, boolean flag) {
2011-02-07 06:56:07 +01:00
Explosion explosion = new Explosion(this, entity, d0, d1, d2, f);
explosion.a = flag;
explosion.a();
explosion.b();
return explosion;
}
public float a(Vec3D vec3d, AxisAlignedBB axisalignedbb) {
double d0 = 1.0D / ((axisalignedbb.d - axisalignedbb.a) * 2.0D + 1.0D);
double d1 = 1.0D / ((axisalignedbb.e - axisalignedbb.b) * 2.0D + 1.0D);
double d2 = 1.0D / ((axisalignedbb.f - axisalignedbb.c) * 2.0D + 1.0D);
int i = 0;
int j = 0;
for (float f = 0.0F; f <= 1.0F; f = (float) ((double) f + d0)) {
for (float f1 = 0.0F; f1 <= 1.0F; f1 = (float) ((double) f1 + d1)) {
for (float f2 = 0.0F; f2 <= 1.0F; f2 = (float) ((double) f2 + d2)) {
double d3 = axisalignedbb.a + (axisalignedbb.d - axisalignedbb.a) * (double) f;
double d4 = axisalignedbb.b + (axisalignedbb.e - axisalignedbb.b) * (double) f1;
double d5 = axisalignedbb.c + (axisalignedbb.f - axisalignedbb.c) * (double) f2;
if (this.a(Vec3D.create(d3, d4, d5), vec3d) == null) {
2011-02-07 06:56:07 +01:00
++i;
}
++j;
}
}
}
return (float) i / (float) j;
}
public TileEntity getTileEntity(int i, int j, int k) {
Chunk chunk = this.getChunkAt(i >> 4, k >> 4);
2011-02-07 06:56:07 +01:00
return chunk != null ? chunk.d(i & 15, j, k & 15) : null;
}
public void setTileEntity(int i, int j, int k, TileEntity tileentity) {
Chunk chunk = this.getChunkAt(i >> 4, k >> 4);
2011-02-07 06:56:07 +01:00
if (chunk != null) {
chunk.a(i & 15, j, k & 15, tileentity);
}
}
public void n(int i, int j, int k) {
Chunk chunk = this.getChunkAt(i >> 4, k >> 4);
2011-02-07 06:56:07 +01:00
if (chunk != null) {
chunk.e(i & 15, j, k & 15);
}
}
public boolean d(int i, int j, int k) {
Block block = Block.byId[this.getTypeId(i, j, k)];
return block == null ? false : block.a();
}
public boolean doLighting() {
2011-02-23 03:37:56 +01:00
if (this.C >= 50) {
2011-02-07 06:56:07 +01:00
return false;
} else {
2011-02-23 03:37:56 +01:00
++this.C;
2011-02-07 06:56:07 +01:00
boolean flag;
2011-02-07 06:56:07 +01:00
try {
int i = 500;
2011-02-23 03:37:56 +01:00
while (this.u.size() > 0) {
2011-02-07 06:56:07 +01:00
--i;
if (i <= 0) {
flag = true;
return flag;
}
2011-02-23 03:37:56 +01:00
((MetadataChunkBlock) this.u.remove(this.u.size() - 1)).a(this);
2011-02-07 06:56:07 +01:00
}
flag = false;
} finally {
2011-02-23 03:37:56 +01:00
--this.C;
2011-02-07 06:56:07 +01:00
}
return flag;
2011-02-07 06:56:07 +01:00
}
}
public void a(EnumSkyBlock enumskyblock, int i, int j, int k, int l, int i1, int j1) {
this.a(enumskyblock, i, j, k, l, i1, j1, true);
}
public void a(EnumSkyBlock enumskyblock, int i, int j, int k, int l, int i1, int j1, boolean flag) {
if (!this.worldProvider.e || enumskyblock != EnumSkyBlock.SKY) {
2011-02-23 03:37:56 +01:00
++s;
if (s == 50) {
--s;
2011-02-07 06:56:07 +01:00
} else {
int k1 = (l + i) / 2;
int l1 = (j1 + k) / 2;
if (!this.isLoaded(k1, 64, l1)) {
2011-02-23 03:37:56 +01:00
--s;
2011-02-07 09:43:51 +01:00
} else if (!this.b(k1, l1).g()) {
2011-02-23 03:37:56 +01:00
int i2 = this.u.size();
2011-02-07 06:56:07 +01:00
int j2;
if (flag) {
j2 = 5;
if (j2 > i2) {
j2 = i2;
}
for (int k2 = 0; k2 < j2; ++k2) {
2011-02-23 03:37:56 +01:00
MetadataChunkBlock metadatachunkblock = (MetadataChunkBlock) this.u.get(this.u.size() - k2 - 1);
2011-02-07 06:56:07 +01:00
if (metadatachunkblock.a == enumskyblock && metadatachunkblock.a(i, j, k, l, i1, j1)) {
2011-02-23 03:37:56 +01:00
--s;
2011-02-07 06:56:07 +01:00
return;
}
}
}
2011-02-23 03:37:56 +01:00
this.u.add(new MetadataChunkBlock(enumskyblock, i, j, k, l, i1, j1));
2011-02-07 06:56:07 +01:00
j2 = 1000000;
2011-02-23 03:37:56 +01:00
if (this.u.size() > 1000000) {
2011-02-07 06:56:07 +01:00
System.out.println("More than " + j2 + " updates, aborting lighting updates");
2011-02-23 03:37:56 +01:00
this.u.clear();
2011-02-07 06:56:07 +01:00
}
2011-02-23 03:37:56 +01:00
--s;
2011-02-07 06:56:07 +01:00
}
}
}
}
2011-03-31 22:40:00 +02:00
public void g() {
2011-02-07 06:56:07 +01:00
int i = this.a(1.0F);
2011-02-23 03:37:56 +01:00
if (i != this.e) {
this.e = i;
2011-02-07 06:56:07 +01:00
}
}
public void setSpawnFlags(boolean flag, boolean flag1) {
this.allowMonsters = flag;
this.allowAnimals = flag1;
2011-02-07 06:56:07 +01:00
}
public void doTick() {
2011-02-23 03:37:56 +01:00
long i;
if (this.everyoneDeeplySleeping()) {
2011-02-23 03:37:56 +01:00
boolean flag = false;
if (this.allowMonsters && this.spawnMonsters >= 1) {
flag = SpawnerCreature.a(this, this.players);
2011-02-23 03:37:56 +01:00
}
if (!flag) {
i = this.worldData.f() + 24000L;
this.worldData.a(i - i % 24000L);
2011-03-31 22:40:00 +02:00
this.r();
2011-02-23 03:37:56 +01:00
}
}
// CraftBukkit start -- Only call spawner if we have players online and the world allows for mobs or animals
if ((this.allowMonsters || this.allowAnimals) && (this instanceof WorldServer && ((WorldServer) this).getServer().getHandle().players.size() > 0)) {
SpawnerCreature.spawnEntities(this, this.allowMonsters, this.allowAnimals);
2011-02-23 13:56:36 +01:00
}
// CraftBukkit end
this.chunkProvider.unloadChunks();
2011-02-23 03:37:56 +01:00
int j = this.a(1.0F);
2011-02-07 06:56:07 +01:00
2011-02-23 03:37:56 +01:00
if (j != this.e) {
this.e = j;
2011-02-07 06:56:07 +01:00
2011-02-23 03:37:56 +01:00
for (int k = 0; k < this.n.size(); ++k) {
((IWorldAccess) this.n.get(k)).a();
2011-02-07 06:56:07 +01:00
}
}
i = this.worldData.f() + 1L;
2011-02-23 03:37:56 +01:00
if (i % (long) this.i == 0L) {
this.save(false, (IProgressUpdate) null);
2011-02-07 06:56:07 +01:00
}
this.worldData.a(i);
2011-02-07 06:56:07 +01:00
this.a(false);
2011-03-31 22:40:00 +02:00
this.i();
2011-02-07 06:56:07 +01:00
}
2011-03-31 22:40:00 +02:00
protected void i() {
2011-02-23 03:37:56 +01:00
this.F.clear();
2011-02-07 06:56:07 +01:00
int i;
int j;
int k;
int l;
for (int i1 = 0; i1 < this.players.size(); ++i1) {
EntityHuman entityhuman = (EntityHuman) this.players.get(i1);
2011-02-07 06:56:07 +01:00
i = MathHelper.floor(entityhuman.locX / 16.0D);
j = MathHelper.floor(entityhuman.locZ / 16.0D);
2011-02-07 06:56:07 +01:00
byte b0 = 9;
for (k = -b0; k <= b0; ++k) {
for (l = -b0; l <= b0; ++l) {
2011-02-23 03:37:56 +01:00
this.F.add(new ChunkCoordIntPair(k + i, l + j));
2011-02-07 06:56:07 +01:00
}
}
}
2011-02-23 03:37:56 +01:00
if (this.G > 0) {
--this.G;
2011-02-07 06:56:07 +01:00
}
2011-02-23 03:37:56 +01:00
Iterator iterator = this.F.iterator();
2011-02-07 06:56:07 +01:00
while (iterator.hasNext()) {
ChunkCoordIntPair chunkcoordintpair = (ChunkCoordIntPair) iterator.next();
i = chunkcoordintpair.x * 16;
j = chunkcoordintpair.z * 16;
Chunk chunk = this.getChunkAt(chunkcoordintpair.x, chunkcoordintpair.z);
2011-02-07 06:56:07 +01:00
int j1;
int k1;
int l1;
2011-02-23 03:37:56 +01:00
if (this.G == 0) {
this.f = this.f * 3 + this.g;
k = this.f >> 2;
2011-02-07 06:56:07 +01:00
l = k & 15;
j1 = k >> 8 & 15;
k1 = k >> 16 & 127;
l1 = chunk.getTypeId(l, k1, j1);
2011-02-07 06:56:07 +01:00
l += i;
j1 += j;
if (l1 == 0 && this.getLightLevel(l, k1, j1) <= this.random.nextInt(8) && this.a(EnumSkyBlock.SKY, l, k1, j1) <= 0) {
2011-02-07 06:56:07 +01:00
EntityHuman entityhuman1 = this.a((double) l + 0.5D, (double) k1 + 0.5D, (double) j1 + 0.5D, 8.0D);
if (entityhuman1 != null && entityhuman1.d((double) l + 0.5D, (double) k1 + 0.5D, (double) j1 + 0.5D) > 4.0D) {
this.makeSound((double) l + 0.5D, (double) k1 + 0.5D, (double) j1 + 0.5D, "ambient.cave.cave", 0.7F, 0.8F + this.random.nextFloat() * 0.2F);
this.G = this.random.nextInt(12000) + 6000;
2011-02-07 06:56:07 +01:00
}
}
}
for (k = 0; k < 80; ++k) {
2011-02-23 03:37:56 +01:00
this.f = this.f * 3 + this.g;
l = this.f >> 2;
2011-02-07 06:56:07 +01:00
j1 = l & 15;
k1 = l >> 8 & 15;
l1 = l >> 16 & 127;
2011-03-31 22:40:00 +02:00
int i2 = chunk.b[j1 << 11 | k1 << 7 | l1] & 255;
2011-02-07 06:56:07 +01:00
2011-03-31 22:40:00 +02:00
if (Block.n[i2]) {
Block.byId[i2].a(this, j1 + i, l1, k1 + j, this.random);
2011-02-07 06:56:07 +01:00
}
}
}
}
public boolean a(boolean flag) {
2011-02-23 03:37:56 +01:00
int i = this.w.size();
2011-02-07 06:56:07 +01:00
2011-02-23 03:37:56 +01:00
if (i != this.x.size()) {
2011-02-07 06:56:07 +01:00
throw new IllegalStateException("TickNextTick list out of synch");
} else {
if (i > 1000) {
i = 1000;
}
for (int j = 0; j < i; ++j) {
2011-02-23 03:37:56 +01:00
NextTickListEntry nextticklistentry = (NextTickListEntry) this.w.first();
2011-02-07 06:56:07 +01:00
if (!flag && nextticklistentry.e > this.worldData.f()) {
2011-02-07 06:56:07 +01:00
break;
}
2011-02-23 03:37:56 +01:00
this.w.remove(nextticklistentry);
this.x.remove(nextticklistentry);
2011-02-07 06:56:07 +01:00
byte b0 = 8;
if (this.a(nextticklistentry.a - b0, nextticklistentry.b - b0, nextticklistentry.c - b0, nextticklistentry.a + b0, nextticklistentry.b + b0, nextticklistentry.c + b0)) {
int k = this.getTypeId(nextticklistentry.a, nextticklistentry.b, nextticklistentry.c);
if (k == nextticklistentry.d && k > 0) {
Block.byId[k].a(this, nextticklistentry.a, nextticklistentry.b, nextticklistentry.c, this.random);
2011-02-07 06:56:07 +01:00
}
}
}
2011-02-23 03:37:56 +01:00
return this.w.size() != 0;
2011-02-07 06:56:07 +01:00
}
}
public List b(Entity entity, AxisAlignedBB axisalignedbb) {
2011-02-23 03:37:56 +01:00
this.H.clear();
int i = MathHelper.floor((axisalignedbb.a - 2.0D) / 16.0D);
int j = MathHelper.floor((axisalignedbb.d + 2.0D) / 16.0D);
int k = MathHelper.floor((axisalignedbb.c - 2.0D) / 16.0D);
int l = MathHelper.floor((axisalignedbb.f + 2.0D) / 16.0D);
2011-02-07 06:56:07 +01:00
for (int i1 = i; i1 <= j; ++i1) {
for (int j1 = k; j1 <= l; ++j1) {
if (this.isChunkLoaded(i1, j1)) {
this.getChunkAt(i1, j1).a(entity, axisalignedbb, this.H);
2011-02-07 06:56:07 +01:00
}
}
}
2011-02-23 03:37:56 +01:00
return this.H;
2011-02-07 06:56:07 +01:00
}
public List a(Class oclass, AxisAlignedBB axisalignedbb) {
int i = MathHelper.floor((axisalignedbb.a - 2.0D) / 16.0D);
int j = MathHelper.floor((axisalignedbb.d + 2.0D) / 16.0D);
int k = MathHelper.floor((axisalignedbb.c - 2.0D) / 16.0D);
int l = MathHelper.floor((axisalignedbb.f + 2.0D) / 16.0D);
2011-02-07 06:56:07 +01:00
ArrayList arraylist = new ArrayList();
for (int i1 = i; i1 <= j; ++i1) {
for (int j1 = k; j1 <= l; ++j1) {
if (this.isChunkLoaded(i1, j1)) {
this.getChunkAt(i1, j1).a(oclass, axisalignedbb, arraylist);
2011-02-07 06:56:07 +01:00
}
}
}
return arraylist;
}
public void b(int i, int j, int k, TileEntity tileentity) {
if (this.isLoaded(i, j, k)) {
2011-02-07 06:56:07 +01:00
this.b(i, k).f();
}
2011-02-23 03:37:56 +01:00
for (int l = 0; l < this.n.size(); ++l) {
((IWorldAccess) this.n.get(l)).a(i, j, k, tileentity);
2011-02-07 06:56:07 +01:00
}
}
public int a(Class oclass) {
int i = 0;
for (int j = 0; j < this.entityList.size(); ++j) {
Entity entity = (Entity) this.entityList.get(j);
2011-02-07 06:56:07 +01:00
if (oclass.isAssignableFrom(entity.getClass())) {
++i;
}
}
return i;
}
public void a(List list) {
// CraftBukkit start
Entity entity = null;
2011-02-07 06:56:07 +01:00
for (int i = 0; i < list.size(); ++i) {
entity = (Entity) list.get(i);
// CraftBukkit start
if (entity instanceof EntityLiving && !(entity instanceof EntityPlayer)) {
CreatureSpawnEvent event = CraftEventFactory.callCreatureSpawnEvent((EntityLiving) entity);
if (event.isCancelled()) {
continue;
}
}
this.entityList.add(entity);
// CraftBukkit end
2011-02-07 06:56:07 +01:00
this.b((Entity) list.get(i));
}
}
public void b(List list) {
2011-02-23 03:37:56 +01:00
this.v.addAll(list);
2011-02-07 06:56:07 +01:00
}
public boolean a(int i, int j, int k, int l, boolean flag) {
int i1 = this.getTypeId(j, k, l);
Block block = Block.byId[i1];
Block block1 = Block.byId[i];
AxisAlignedBB axisalignedbb = block1.d(this, j, k, l);
if (flag) {
axisalignedbb = null;
}
2011-02-23 13:56:36 +01:00
// CraftBukkit start - We dont want to allow the user to override the bounding box check
boolean defaultReturn = axisalignedbb != null && !this.containsEntity(axisalignedbb) ? false : (block != Block.WATER && block != Block.STATIONARY_WATER && block != Block.LAVA && block != Block.STATIONARY_LAVA && block != Block.FIRE && block != Block.SNOW ? i > 0 && block == null && block1.canPlace(this, j, k, l) : true);
if (axisalignedbb != null && !this.containsEntity(axisalignedbb)) {
return false;
}
BlockCanBuildEvent event = new BlockCanBuildEvent(((WorldServer) this).getWorld().getBlockAt(j, k, l), i, defaultReturn);
2011-02-07 10:43:50 +01:00
((WorldServer) this).getServer().getPluginManager().callEvent(event);
return event.isBuildable();
// CraftBukkit end
2011-02-07 06:56:07 +01:00
}
public PathEntity findPath(Entity entity, Entity entity1, float f) {
int i = MathHelper.floor(entity.locX);
int j = MathHelper.floor(entity.locY);
int k = MathHelper.floor(entity.locZ);
2011-02-07 06:56:07 +01:00
int l = (int) (f + 16.0F);
int i1 = i - l;
int j1 = j - l;
int k1 = k - l;
int l1 = i + l;
int i2 = j + l;
int j2 = k + l;
ChunkCache chunkcache = new ChunkCache(this, i1, j1, k1, l1, i2, j2);
return (new Pathfinder(chunkcache)).a(entity, entity1, f);
}
public PathEntity a(Entity entity, int i, int j, int k, float f) {
int l = MathHelper.floor(entity.locX);
int i1 = MathHelper.floor(entity.locY);
int j1 = MathHelper.floor(entity.locZ);
2011-02-07 06:56:07 +01:00
int k1 = (int) (f + 8.0F);
int l1 = l - k1;
int i2 = i1 - k1;
int j2 = j1 - k1;
int k2 = l + k1;
int l2 = i1 + k1;
int i3 = j1 + k1;
ChunkCache chunkcache = new ChunkCache(this, l1, i2, j2, k2, l2, i3);
return (new Pathfinder(chunkcache)).a(entity, i, j, k, f);
}
public boolean isBlockFacePowered(int i, int j, int k, int l) {
2011-02-07 06:56:07 +01:00
int i1 = this.getTypeId(i, j, k);
2011-02-23 03:37:56 +01:00
return i1 == 0 ? false : Block.byId[i1].c(this, i, j, k, l);
2011-02-07 06:56:07 +01:00
}
public boolean isBlockPowered(int i, int j, int k) {
return this.isBlockFacePowered(i, j - 1, k, 0) ? true : (this.isBlockFacePowered(i, j + 1, k, 1) ? true : (this.isBlockFacePowered(i, j, k - 1, 2) ? true : (this.isBlockFacePowered(i, j, k + 1, 3) ? true : (this.isBlockFacePowered(i - 1, j, k, 4) ? true : this.isBlockFacePowered(i + 1, j, k, 5)))));
2011-02-07 06:56:07 +01:00
}
public boolean isBlockFaceIndirectlyPowered(int i, int j, int k, int l) {
2011-02-07 06:56:07 +01:00
if (this.d(i, j, k)) {
return this.isBlockPowered(i, j, k);
2011-02-07 06:56:07 +01:00
} else {
int i1 = this.getTypeId(i, j, k);
return i1 == 0 ? false : Block.byId[i1].b(this, i, j, k, l);
2011-02-07 06:56:07 +01:00
}
}
public boolean isBlockIndirectlyPowered(int i, int j, int k) {
return this.isBlockFaceIndirectlyPowered(i, j - 1, k, 0) ? true : (this.isBlockFaceIndirectlyPowered(i, j + 1, k, 1) ? true : (this.isBlockFaceIndirectlyPowered(i, j, k - 1, 2) ? true : (this.isBlockFaceIndirectlyPowered(i, j, k + 1, 3) ? true : (this.isBlockFaceIndirectlyPowered(i - 1, j, k, 4) ? true : this.isBlockFaceIndirectlyPowered(i + 1, j, k, 5)))));
2011-02-07 06:56:07 +01:00
}
public EntityHuman a(Entity entity, double d0) {
return this.a(entity.locX, entity.locY, entity.locZ, d0);
}
public EntityHuman a(double d0, double d1, double d2, double d3) {
double d4 = -1.0D;
EntityHuman entityhuman = null;
for (int i = 0; i < this.players.size(); ++i) {
EntityHuman entityhuman1 = (EntityHuman) this.players.get(i);
2011-02-07 06:56:07 +01:00
double d5 = entityhuman1.d(d0, d1, d2);
if ((d3 < 0.0D || d5 < d3 * d3) && (d4 == -1.0D || d5 < d4)) {
d4 = d5;
entityhuman = entityhuman1;
}
}
return entityhuman;
}
2011-03-31 22:40:00 +02:00
public EntityHuman a(String s) {
for (int i = 0; i < this.players.size(); ++i) {
if (s.equals(((EntityHuman) this.players.get(i)).name)) {
return (EntityHuman) this.players.get(i);
2011-03-31 22:40:00 +02:00
}
}
return null;
}
2011-02-07 06:56:07 +01:00
public byte[] c(int i, int j, int k, int l, int i1, int j1) {
byte[] abyte = new byte[l * i1 * j1 * 5 / 2];
int k1 = i >> 4;
int l1 = k >> 4;
int i2 = i + l - 1 >> 4;
int j2 = k + j1 - 1 >> 4;
int k2 = 0;
int l2 = j;
int i3 = j + i1;
if (j < 0) {
l2 = 0;
}
if (i3 > 128) {
i3 = 128;
}
for (int j3 = k1; j3 <= i2; ++j3) {
int k3 = i - j3 * 16;
int l3 = i + l - j3 * 16;
if (k3 < 0) {
k3 = 0;
}
if (l3 > 16) {
l3 = 16;
}
for (int i4 = l1; i4 <= j2; ++i4) {
int j4 = k - i4 * 16;
int k4 = k + j1 - i4 * 16;
if (j4 < 0) {
j4 = 0;
}
if (k4 > 16) {
k4 = 16;
}
k2 = this.getChunkAt(j3, i4).a(abyte, k3, l2, j4, l3, i3, k4, k2);
2011-02-07 06:56:07 +01:00
}
}
return abyte;
}
2011-03-31 22:40:00 +02:00
public void j() {
2011-02-23 03:37:56 +01:00
this.p.b();
}
2011-02-07 06:56:07 +01:00
public void setTime(long i) {
this.worldData.a(i);
2011-02-23 03:37:56 +01:00
}
public long getSeed() {
return this.worldData.b();
2011-02-23 03:37:56 +01:00
}
public long getTime() {
return this.worldData.f();
2011-02-23 03:37:56 +01:00
}
public ChunkCoordinates getSpawn() {
return new ChunkCoordinates(this.worldData.c(), this.worldData.d(), this.worldData.e());
2011-02-07 06:56:07 +01:00
}
public boolean a(EntityHuman entityhuman, int i, int j, int k) {
return true;
}
public void a(Entity entity, byte b0) {}
2011-03-31 22:40:00 +02:00
public IChunkProvider n() {
return this.chunkProvider;
2011-03-31 22:40:00 +02:00
}
2011-02-23 03:37:56 +01:00
public void d(int i, int j, int k, int l, int i1) {
2011-02-07 06:56:07 +01:00
int j1 = this.getTypeId(i, j, k);
if (j1 > 0) {
Block.byId[j1].a(this, i, j, k, l, i1);
}
}
2011-02-23 03:37:56 +01:00
2011-03-31 22:40:00 +02:00
public IDataManager o() {
2011-02-23 03:37:56 +01:00
return this.p;
}
2011-03-31 22:40:00 +02:00
public WorldData p() {
return this.worldData;
2011-02-23 03:37:56 +01:00
}
public void everyoneSleeping() {
this.A = !this.players.isEmpty();
Iterator iterator = this.players.iterator();
2011-02-23 03:37:56 +01:00
while (iterator.hasNext()) {
EntityHuman entityhuman = (EntityHuman) iterator.next();
// CraftBukkit
if (!entityhuman.isSleeping() && !entityhuman.fauxSleeping) {
2011-02-23 03:37:56 +01:00
this.A = false;
break;
}
}
}
// CraftBukkit start
// Calls the method that checks to see if players are sleeping
// Called by CraftPlayer.setPermanentSleeping()
public void checkSleepStatus() {
if (!isStatic) {
everyoneSleeping();
}
}
// CraftBukkit end
2011-03-31 22:40:00 +02:00
protected void r() {
2011-02-23 03:37:56 +01:00
this.A = false;
Iterator iterator = this.players.iterator();
2011-02-23 03:37:56 +01:00
while (iterator.hasNext()) {
EntityHuman entityhuman = (EntityHuman) iterator.next();
if (entityhuman.isSleeping()) {
2011-03-31 22:40:00 +02:00
entityhuman.a(false, false, false);
2011-02-23 03:37:56 +01:00
}
}
}
public boolean everyoneDeeplySleeping() {
2011-02-23 03:37:56 +01:00
if (this.A && !this.isStatic) {
Iterator iterator = this.players.iterator();
// CraftBukkit start
boolean foundActualSleepers = false;
// This allows us to assume that some people are in bed
// but not really, allowing time to pass in spite of AFKers
2011-02-23 03:37:56 +01:00
EntityHuman entityhuman;
do {
if (!iterator.hasNext()) {
return foundActualSleepers;
2011-02-23 03:37:56 +01:00
}
entityhuman = (EntityHuman) iterator.next();
if (entityhuman.isDeeplySleeping()) {
foundActualSleepers = true;
}
} while (entityhuman.isDeeplySleeping() || entityhuman.fauxSleeping);
// CraftBukkit end
2011-02-23 03:37:56 +01:00
return false;
} else {
return false;
}
}
2011-02-07 06:56:07 +01:00
}