2018-08-12 14:03:07 +00:00
|
|
|
package com.boydti.fawe.object.pattern;
|
|
|
|
|
|
|
|
import com.boydti.fawe.FaweCache;
|
|
|
|
import com.boydti.fawe.object.PseudoRandom;
|
|
|
|
import com.sk89q.worldedit.blocks.BaseBlock;
|
|
|
|
import com.sk89q.worldedit.world.block.BlockState;
|
|
|
|
import com.sk89q.worldedit.function.pattern.AbstractPattern;
|
|
|
|
import com.sk89q.worldedit.function.pattern.Pattern;
|
|
|
|
import com.sk89q.worldedit.function.visitor.BreadthFirstSearch;
|
2018-12-23 16:19:33 +00:00
|
|
|
import com.sk89q.worldedit.math.BlockVector3;
|
|
|
|
import com.sk89q.worldedit.math.MutableBlockVector;
|
2018-08-12 14:03:07 +00:00
|
|
|
import com.sk89q.worldedit.world.block.BlockStateHolder;
|
|
|
|
|
|
|
|
import java.io.IOException;
|
|
|
|
|
|
|
|
public class SurfaceRandomOffsetPattern extends AbstractPattern {
|
|
|
|
private final Pattern pattern;
|
|
|
|
private int moves;
|
|
|
|
|
|
|
|
private transient MutableBlockVector cur;
|
|
|
|
private transient MutableBlockVector[] buffer;
|
|
|
|
private transient MutableBlockVector[] allowed;
|
|
|
|
private transient MutableBlockVector next;
|
|
|
|
|
|
|
|
public SurfaceRandomOffsetPattern(Pattern pattern, int distance) {
|
|
|
|
this.pattern = pattern;
|
|
|
|
this.moves = Math.min(255, distance);
|
|
|
|
init();
|
|
|
|
}
|
|
|
|
|
|
|
|
private void init() {
|
|
|
|
cur = new MutableBlockVector();
|
|
|
|
this.buffer = new MutableBlockVector[BreadthFirstSearch.DIAGONAL_DIRECTIONS.length];
|
|
|
|
for (int i = 0; i < buffer.length; i++) {
|
|
|
|
buffer[i] = new MutableBlockVector();
|
|
|
|
}
|
|
|
|
allowed = new MutableBlockVector[buffer.length];
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
2018-12-23 16:19:33 +00:00
|
|
|
public BlockStateHolder apply(BlockVector3 position) {
|
2018-08-12 14:03:07 +00:00
|
|
|
return pattern.apply(travel(position));
|
|
|
|
}
|
|
|
|
|
2018-12-23 16:19:33 +00:00
|
|
|
private BlockVector3 travel(BlockVector3 pos) {
|
2018-08-12 14:03:07 +00:00
|
|
|
cur.setComponents(pos);
|
|
|
|
for (int move = 0; move < moves; move++) {
|
|
|
|
int index = 0;
|
|
|
|
for (int i = 0; i < allowed.length; i++) {
|
|
|
|
next = buffer[i];
|
2018-12-23 16:19:33 +00:00
|
|
|
BlockVector3 dir = BreadthFirstSearch.DIAGONAL_DIRECTIONS[i];
|
2018-08-12 14:03:07 +00:00
|
|
|
next.setComponents(cur.getBlockX() + dir.getBlockX(), cur.getBlockY() + dir.getBlockY(), cur.getBlockZ() + dir.getBlockZ());
|
2019-01-09 07:13:44 +00:00
|
|
|
if (allowed(next.toBlockVector3())) {
|
2018-08-12 14:03:07 +00:00
|
|
|
allowed[index++] = next;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (index == 0) {
|
2019-01-09 07:13:44 +00:00
|
|
|
return cur.toBlockVector3();
|
2018-08-12 14:03:07 +00:00
|
|
|
}
|
|
|
|
next = allowed[PseudoRandom.random.nextInt(index)];
|
|
|
|
cur.setComponents(next.getBlockX(), next.getBlockY(), next.getBlockZ());
|
|
|
|
}
|
2019-01-09 07:13:44 +00:00
|
|
|
return cur.toBlockVector3();
|
2018-08-12 14:03:07 +00:00
|
|
|
}
|
|
|
|
|
2018-12-23 16:19:33 +00:00
|
|
|
private boolean allowed(BlockVector3 bv) {
|
|
|
|
MutableBlockVector v = new MutableBlockVector(bv);
|
2019-01-09 07:13:44 +00:00
|
|
|
BlockStateHolder block = pattern.apply(bv);
|
2018-08-12 14:03:07 +00:00
|
|
|
if (!block.getBlockType().getMaterial().isMovementBlocker()) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
int x = v.getBlockX();
|
|
|
|
int y = v.getBlockY();
|
|
|
|
int z = v.getBlockZ();
|
|
|
|
v.mutY(y + 1);
|
2019-01-09 07:13:44 +00:00
|
|
|
if (canPassthrough(v.toBlockVector3())) {
|
2018-08-12 14:03:07 +00:00
|
|
|
v.mutY(y);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
v.mutY(y - 1);
|
2019-01-09 07:13:44 +00:00
|
|
|
if (canPassthrough(v.toBlockVector3())) {
|
2018-08-12 14:03:07 +00:00
|
|
|
v.mutY(y);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
v.mutY(y);
|
|
|
|
v.mutX(x + 1);
|
2019-01-09 07:13:44 +00:00
|
|
|
if (canPassthrough(v.toBlockVector3())) {
|
2018-08-12 14:03:07 +00:00
|
|
|
v.mutX(x);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
v.mutX(x - 1);
|
2019-01-09 07:13:44 +00:00
|
|
|
if (canPassthrough(v.toBlockVector3())) {
|
2018-08-12 14:03:07 +00:00
|
|
|
v.mutX(x);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
v.mutX(x);
|
|
|
|
v.mutZ(z + 1);
|
2019-01-09 07:13:44 +00:00
|
|
|
if (canPassthrough(v.toBlockVector3())) {
|
2018-08-12 14:03:07 +00:00
|
|
|
v.mutZ(z);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
v.mutZ(z - 1);
|
2019-01-09 07:13:44 +00:00
|
|
|
if (canPassthrough(v.toBlockVector3())) {
|
2018-08-12 14:03:07 +00:00
|
|
|
v.mutZ(z);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
v.mutZ(z);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2018-12-23 16:19:33 +00:00
|
|
|
private boolean canPassthrough(BlockVector3 v) {
|
2018-08-12 14:03:07 +00:00
|
|
|
BlockStateHolder block = pattern.apply(v);
|
|
|
|
return !block.getBlockType().getMaterial().isMovementBlocker();
|
|
|
|
}
|
|
|
|
|
|
|
|
private void readObject(java.io.ObjectInputStream stream) throws IOException, ClassNotFoundException {
|
|
|
|
stream.defaultReadObject();
|
|
|
|
init();
|
|
|
|
}
|
|
|
|
}
|