Dieser Commit ist enthalten in:
Ursprung
45fd36907c
Commit
69a0e02ab9
@ -120,6 +120,22 @@ public abstract class AI {
|
|||||||
Chat.broadcastChat("PARTICIPANT_CHAT", team.getColoredName(), entity.getName(), message);
|
Chat.broadcastChat("PARTICIPANT_CHAT", team.getColoredName(), entity.getName(), message);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
protected Vector toAIPosition(Vector location) {
|
||||||
|
Region extend = team.getExtendRegion();
|
||||||
|
if(Fight.getUnrotated() == team)
|
||||||
|
return new Vector(
|
||||||
|
location.getX() - extend.getMinX(),
|
||||||
|
location.getY() - team.getSchemRegion().getMinY(),
|
||||||
|
location.getZ() - extend.getMinZ()
|
||||||
|
);
|
||||||
|
else
|
||||||
|
return new Vector(
|
||||||
|
extend.getMaxX() - location.getX(),
|
||||||
|
location.getY() - team.getSchemRegion().getMinY(),
|
||||||
|
extend.getMaxZ() - location.getZ()
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
protected Vector getPosition() {
|
protected Vector getPosition() {
|
||||||
Location location = entity.getLocation();
|
Location location = entity.getLocation();
|
||||||
Region extend = team.getExtendRegion();
|
Region extend = team.getExtendRegion();
|
||||||
@ -205,15 +221,17 @@ public abstract class AI {
|
|||||||
public void run() {
|
public void run() {
|
||||||
Location location = entity.getLocation();
|
Location location = entity.getLocation();
|
||||||
Location target = translate(pos, false);
|
Location target = translate(pos, false);
|
||||||
|
/*
|
||||||
if(Math.abs(location.getX() - target.getX()) > 1 || Math.abs(location.getY() - target.getY()) > 1.2 || Math.abs(location.getZ() - target.getZ()) > 1) {
|
if(Math.abs(location.getX() - target.getX()) > 1 || Math.abs(location.getY() - target.getY()) > 1.2 || Math.abs(location.getZ() - target.getZ()) > 1) {
|
||||||
FightSystem.getPlugin().getLogger().log(Level.INFO, () -> entity.getName() + ": Overdistance movement " + location.toVector() + " " + target.toVector());
|
FightSystem.getPlugin().getLogger().log(Level.INFO, () -> entity.getName() + ": Overdistance movement " + location.toVector() + " " + target.toVector());
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
*/
|
||||||
|
|
||||||
if(!team.getFightPlayer(entity).canEntern() && !team.getExtendRegion().inRegion(target))
|
if(!team.getFightPlayer(entity).canEntern() && !team.getExtendRegion().inRegion(target))
|
||||||
return;
|
return;
|
||||||
|
|
||||||
entity.teleport(target, PlayerTeleportEvent.TeleportCause.COMMAND);
|
entity.teleport(target, PlayerTeleportEvent.TeleportCause.PLUGIN);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
@ -22,6 +22,7 @@ package de.steamwar.fightsystem.ai;
|
|||||||
import de.steamwar.entity.REntityServer;
|
import de.steamwar.entity.REntityServer;
|
||||||
import de.steamwar.fightsystem.Config;
|
import de.steamwar.fightsystem.Config;
|
||||||
import de.steamwar.fightsystem.ai.navmesh.NavMesh;
|
import de.steamwar.fightsystem.ai.navmesh.NavMesh;
|
||||||
|
import de.steamwar.fightsystem.fight.Fight;
|
||||||
import de.steamwar.fightsystem.fight.FightTeam;
|
import de.steamwar.fightsystem.fight.FightTeam;
|
||||||
import de.steamwar.sql.SchematicNode;
|
import de.steamwar.sql.SchematicNode;
|
||||||
import de.steamwar.sql.SteamwarUser;
|
import de.steamwar.sql.SteamwarUser;
|
||||||
@ -33,6 +34,7 @@ import java.util.Random;
|
|||||||
|
|
||||||
public class LixfelAI extends AI {
|
public class LixfelAI extends AI {
|
||||||
|
|
||||||
|
private final Random random = new Random();
|
||||||
private final REntityServer entityServer = new REntityServer();
|
private final REntityServer entityServer = new REntityServer();
|
||||||
private final FightTeam team;
|
private final FightTeam team;
|
||||||
private final NavMesh navMesh;
|
private final NavMesh navMesh;
|
||||||
@ -45,9 +47,10 @@ public class LixfelAI extends AI {
|
|||||||
|
|
||||||
@Override
|
@Override
|
||||||
public SchematicNode chooseSchematic() {
|
public SchematicNode chooseSchematic() {
|
||||||
List<SchematicNode> publics = SchematicNode.getAllSchematicsOfType(0, Config.SchematicType.toDB());
|
return SchematicNode.byIdAndUser(SteamwarUser.get(0), 111476);
|
||||||
SchematicNode schem = publics.get(new Random().nextInt(publics.size()));
|
// List<SchematicNode> publics = SchematicNode.getAllSchematicsOfType(0, Config.SchematicType.toDB());
|
||||||
return schem;
|
// SchematicNode schem = publics.get(new Random().nextInt(publics.size()));
|
||||||
|
// return schem;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
@ -61,5 +64,20 @@ public class LixfelAI extends AI {
|
|||||||
@Override
|
@Override
|
||||||
protected void plan() {
|
protected void plan() {
|
||||||
setReady();
|
setReady();
|
||||||
|
|
||||||
|
if (navMesh == null) return;
|
||||||
|
|
||||||
|
List<Vector> walkableBlocks = navMesh.getWalkableBlocks(getEntity().getLocation().toVector());
|
||||||
|
if (walkableBlocks.isEmpty()) return;
|
||||||
|
Vector destination = walkableBlocks.get(random.nextInt(walkableBlocks.size()));
|
||||||
|
List<Vector> path = navMesh.path(getEntity().getLocation().toVector(), destination);
|
||||||
|
if (path.isEmpty()) return;
|
||||||
|
chat(getEntity().getLocation().toVector() + " -> " + destination + " = " + path.size());
|
||||||
|
for(Vector p : path) {
|
||||||
|
move(toAIPosition(p));
|
||||||
|
}
|
||||||
|
for (int i = 0; i < 40; i++) {
|
||||||
|
move(path.get(path.size() - 1));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -46,7 +46,18 @@ public class NavMesh implements Listener {
|
|||||||
private static final double PLAYER_JUMP_HEIGHT = 1.25;
|
private static final double PLAYER_JUMP_HEIGHT = 1.25;
|
||||||
private static final double PLAYER_HEIGHT = 1.8125;
|
private static final double PLAYER_HEIGHT = 1.8125;
|
||||||
private static final BoundingBox PLAYER_SHADOW = new BoundingBox(0.2, 0, 0.2, 0.8, 1, 0.8);
|
private static final BoundingBox PLAYER_SHADOW = new BoundingBox(0.2, 0, 0.2, 0.8, 1, 0.8);
|
||||||
private static final BlockFace[] FACES = new BlockFace[]{BlockFace.NORTH, BlockFace.SOUTH, BlockFace.WEST, BlockFace.EAST};
|
private static final Set<Pos> RELATIVE_BLOCKS_TO_CHECK = new HashSet<>();
|
||||||
|
|
||||||
|
static {
|
||||||
|
for (int y = -2; y <= 2; y++) {
|
||||||
|
RELATIVE_BLOCKS_TO_CHECK.add(new Pos(BlockFace.NORTH.getDirection().setY(y)));
|
||||||
|
RELATIVE_BLOCKS_TO_CHECK.add(new Pos(BlockFace.SOUTH.getDirection().setY(y)));
|
||||||
|
RELATIVE_BLOCKS_TO_CHECK.add(new Pos(BlockFace.EAST.getDirection().setY(y)));
|
||||||
|
RELATIVE_BLOCKS_TO_CHECK.add(new Pos(BlockFace.WEST.getDirection().setY(y)));
|
||||||
|
}
|
||||||
|
RELATIVE_BLOCKS_TO_CHECK.add(new Pos(BlockFace.UP.getDirection()));
|
||||||
|
RELATIVE_BLOCKS_TO_CHECK.add(new Pos(BlockFace.DOWN.getDirection()));
|
||||||
|
}
|
||||||
|
|
||||||
private FightTeam fightTeam;
|
private FightTeam fightTeam;
|
||||||
|
|
||||||
@ -56,17 +67,19 @@ public class NavMesh implements Listener {
|
|||||||
new OneShotStateDependent(ArenaMode.All, FightState.PostSchemSetup, () -> {
|
new OneShotStateDependent(ArenaMode.All, FightState.PostSchemSetup, () -> {
|
||||||
Bukkit.getScheduler().runTaskLater(FightSystem.getPlugin(), () -> {
|
Bukkit.getScheduler().runTaskLater(FightSystem.getPlugin(), () -> {
|
||||||
long time = System.currentTimeMillis();
|
long time = System.currentTimeMillis();
|
||||||
fightTeam.getExtendRegion().forEach(this::checkWalkable);
|
fightTeam.getExtendRegion().forEach((x, y, z) -> {
|
||||||
|
if (y < fightTeam.getSchemRegion().getMinY()) return;
|
||||||
|
checkWalkable(x, y, z);
|
||||||
|
});
|
||||||
floorBlock.forEach(this::checkNeighbouring);
|
floorBlock.forEach(this::checkNeighbouring);
|
||||||
System.out.println(System.currentTimeMillis() - time + " ms");
|
System.out.println(System.currentTimeMillis() - time + " ms");
|
||||||
// For neighbour map -2-+2 blocks
|
// For neighbour map -2-+2 blocks
|
||||||
System.out.println(fightTeam + " " + floorBlock.size());
|
System.out.println(fightTeam + " " + floorBlock.size());
|
||||||
iterateWalkableBlocks((vector, ceilingOffset, neightbourConnections) -> {
|
iterateWalkableBlocks((vector, ceilingOffset) -> {
|
||||||
String connecting = neightbourConnections.stream().map(face -> face.name().substring(0, 1)).collect(Collectors.joining());
|
|
||||||
RArmorStand armorStand = new RArmorStand(entityServer, vector.toLocation(WORLD), RArmorStand.Size.MARKER);
|
RArmorStand armorStand = new RArmorStand(entityServer, vector.toLocation(WORLD), RArmorStand.Size.MARKER);
|
||||||
armorStand.setNoGravity(true);
|
armorStand.setNoGravity(true);
|
||||||
armorStand.setInvisible(true);
|
armorStand.setInvisible(true);
|
||||||
armorStand.setDisplayName("+" + (ceilingOffset == null ? "∞" : ceilingOffset) + " " + connecting);
|
armorStand.setDisplayName("+" + (ceilingOffset == null ? "∞" : ceilingOffset));
|
||||||
});
|
});
|
||||||
}, 20);
|
}, 20);
|
||||||
});
|
});
|
||||||
@ -86,6 +99,12 @@ public class NavMesh implements Listener {
|
|||||||
this.z = z;
|
this.z = z;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public Pos(Vector vector) {
|
||||||
|
this.x = vector.getBlockX();
|
||||||
|
this.y = vector.getBlockY();
|
||||||
|
this.z = vector.getBlockZ();
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public boolean equals(Object o) {
|
public boolean equals(Object o) {
|
||||||
if (this == o) return true;
|
if (this == o) return true;
|
||||||
@ -98,11 +117,19 @@ public class NavMesh implements Listener {
|
|||||||
public int hashCode() {
|
public int hashCode() {
|
||||||
return Objects.hash(x, y, z);
|
return Objects.hash(x, y, z);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public Vector toVector() {
|
||||||
|
return new Vector(x, y, z);
|
||||||
|
}
|
||||||
|
|
||||||
|
public Pos offset(Pos pos) {
|
||||||
|
return new Pos(x + pos.x, y + pos.y, z + pos.z);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private Map<Pos, Double> floorBlock = new HashMap<>();
|
private Map<Pos, Double> floorBlock = new HashMap<>();
|
||||||
private Map<Pos, Double> ceilingOffset = new HashMap<>();
|
private Map<Pos, Double> ceilingOffset = new HashMap<>();
|
||||||
private Map<Pos, Set<BlockFace>> neighbourConnections = new HashMap<>();
|
private Map<Pos, Set<Pos>> neighbourConnections = new HashMap<>();
|
||||||
|
|
||||||
private void checkWalkable(int x, int y, int z) {
|
private void checkWalkable(int x, int y, int z) {
|
||||||
Block block = WORLD.getBlockAt(x, y, z);
|
Block block = WORLD.getBlockAt(x, y, z);
|
||||||
@ -151,33 +178,37 @@ public class NavMesh implements Listener {
|
|||||||
}
|
}
|
||||||
|
|
||||||
private void checkNeighbouring(Pos pos, double posFloorHeight) {
|
private void checkNeighbouring(Pos pos, double posFloorHeight) {
|
||||||
for (BlockFace blockFace : FACES) {
|
for (Pos relativeCheck : RELATIVE_BLOCKS_TO_CHECK) {
|
||||||
// TODO: Check FenceGates, Doors, usw.
|
Pos other = new Pos(pos.x + relativeCheck.x, pos.y + relativeCheck.y, pos.z + relativeCheck.z);
|
||||||
for (int cy = pos.y - 2; cy <= pos.y + 2; cy++) {
|
Double otherFloorHeight = floorBlock.get(other);
|
||||||
// TODO: Fix Ladder in Underground
|
if (otherFloorHeight == null) continue;
|
||||||
Pos other = new Pos(pos.x + blockFace.getModX(), cy, pos.z + blockFace.getModZ());
|
double floorDiff = Math.abs(posFloorHeight - otherFloorHeight);
|
||||||
Double otherFloorHeight = floorBlock.get(other);
|
if (floorDiff > PLAYER_JUMP_HEIGHT) continue;
|
||||||
if (otherFloorHeight == null) continue;
|
|
||||||
double floorDiff = Math.abs(posFloorHeight - otherFloorHeight);
|
|
||||||
if (floorDiff > PLAYER_JUMP_HEIGHT) continue;
|
|
||||||
|
|
||||||
Double posCeilingOffset = ceilingOffset.get(pos);
|
Double posCeilingOffset = ceilingOffset.get(pos);
|
||||||
Double otherCeilingOffset = ceilingOffset.get(other);
|
Double otherCeilingOffset = ceilingOffset.get(other);
|
||||||
if (posCeilingOffset == null && otherCeilingOffset == null) {
|
if (posCeilingOffset == null && otherCeilingOffset == null) {
|
||||||
neighbourConnections.computeIfAbsent(pos, __ -> new LinkedHashSet<>()).add(blockFace);
|
neighbourConnections.computeIfAbsent(pos, __ -> new LinkedHashSet<>()).add(relativeCheck);
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (posCeilingOffset != null && posFloorHeight + posCeilingOffset - otherFloorHeight >= PLAYER_HEIGHT) {
|
if (posCeilingOffset != null && otherCeilingOffset == null) {
|
||||||
neighbourConnections.computeIfAbsent(pos, __ -> new LinkedHashSet<>()).add(blockFace);
|
if (posFloorHeight + posCeilingOffset - otherFloorHeight >= PLAYER_HEIGHT) {
|
||||||
|
neighbourConnections.computeIfAbsent(pos, __ -> new LinkedHashSet<>()).add(relativeCheck);
|
||||||
}
|
}
|
||||||
if (otherCeilingOffset != null && otherFloorHeight + otherCeilingOffset - posFloorHeight >= PLAYER_HEIGHT) {
|
continue;
|
||||||
neighbourConnections.computeIfAbsent(pos, __ -> new LinkedHashSet<>()).add(blockFace);
|
}
|
||||||
|
if (otherCeilingOffset != null && posCeilingOffset == null) {
|
||||||
|
if (otherFloorHeight + otherCeilingOffset - posFloorHeight >= PLAYER_HEIGHT) {
|
||||||
|
neighbourConnections.computeIfAbsent(pos, __ -> new LinkedHashSet<>()).add(relativeCheck);
|
||||||
}
|
}
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (posFloorHeight + posCeilingOffset - otherFloorHeight >= PLAYER_HEIGHT && otherFloorHeight + otherCeilingOffset - posFloorHeight >= PLAYER_HEIGHT) {
|
||||||
|
neighbourConnections.computeIfAbsent(pos, __ -> new LinkedHashSet<>()).add(relativeCheck);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// TODO: Ladder movement
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private boolean overlaps(VoxelShape voxelShape, double x, double y, double z) {
|
private boolean overlaps(VoxelShape voxelShape, double x, double y, double z) {
|
||||||
@ -187,10 +218,10 @@ public class NavMesh implements Listener {
|
|||||||
return overlaps;
|
return overlaps;
|
||||||
}
|
}
|
||||||
|
|
||||||
public void iterateWalkableBlocks(TriConsumer<Vector, Double, Set<BlockFace>> consumer) {
|
private void iterateWalkableBlocks(BiConsumer<Vector, Double> consumer) {
|
||||||
floorBlock.forEach((pos, aDouble) -> {
|
floorBlock.forEach((pos, aDouble) -> {
|
||||||
Vector vector = new Vector(pos.x + 0.5, aDouble, pos.z + 0.5);
|
Vector vector = new Vector(pos.x + 0.5, aDouble, pos.z + 0.5);
|
||||||
consumer.accept(vector, ceilingOffset.get(pos), neighbourConnections.getOrDefault(pos, new HashSet<>()));
|
consumer.accept(vector, ceilingOffset.get(pos));
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -198,4 +229,83 @@ public class NavMesh implements Listener {
|
|||||||
|
|
||||||
void accept(A a, B b, C c);
|
void accept(A a, B b, C c);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private Pos toPos(Vector vector) {
|
||||||
|
Pos pos = new Pos(vector);
|
||||||
|
if (floorBlock.containsKey(pos)) return pos;
|
||||||
|
pos = new Pos(pos.x, pos.y - 1, pos.z);
|
||||||
|
if (floorBlock.containsKey(pos)) return pos;
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
public List<Vector> getAllWalkableBlocks() {
|
||||||
|
return floorBlock.keySet().stream().map(Pos::toVector).collect(Collectors.toList());
|
||||||
|
}
|
||||||
|
|
||||||
|
public List<Vector> getWalkableBlocks(Vector fromVector) {
|
||||||
|
Pos from = toPos(fromVector);
|
||||||
|
if (from == null) {
|
||||||
|
return Collections.emptyList();
|
||||||
|
}
|
||||||
|
|
||||||
|
Set<Pos> checked = new HashSet<>();
|
||||||
|
List<Pos> checking = new ArrayList<>();
|
||||||
|
checking.add(from);
|
||||||
|
while (!checking.isEmpty()) {
|
||||||
|
Pos pos = checking.remove(0);
|
||||||
|
checked.add(pos);
|
||||||
|
|
||||||
|
neighbourConnections.getOrDefault(pos, new HashSet<>()).forEach(p -> {
|
||||||
|
Pos n = pos.offset(p);
|
||||||
|
if (checked.contains(n)) return;
|
||||||
|
if (checking.contains(n)) return;
|
||||||
|
checking.add(n);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
return checked.stream().map(Pos::toVector).collect(Collectors.toList());
|
||||||
|
}
|
||||||
|
|
||||||
|
public List<Vector> path(Vector fromVector, Vector toVector) {
|
||||||
|
Pos from = toPos(fromVector);
|
||||||
|
Pos to = toPos(toVector);
|
||||||
|
if (from == null || to == null) {
|
||||||
|
return Collections.emptyList();
|
||||||
|
}
|
||||||
|
|
||||||
|
List<Pos> checking = new ArrayList<>(Arrays.asList(to));
|
||||||
|
Map<Pos, Pos> route = new HashMap<>();
|
||||||
|
while (!checking.isEmpty()) {
|
||||||
|
Set<Pos> toCheck = new HashSet<>();
|
||||||
|
for (Pos pos : checking) {
|
||||||
|
boolean foundFrom = neighbourConnections.getOrDefault(pos, new HashSet<>()).stream()
|
||||||
|
.map(pos::offset)
|
||||||
|
.filter(next -> !route.containsKey(next))
|
||||||
|
.anyMatch(next -> {
|
||||||
|
route.put(next, pos);
|
||||||
|
toCheck.add(next);
|
||||||
|
return next.equals(from);
|
||||||
|
});
|
||||||
|
|
||||||
|
if (foundFrom) {
|
||||||
|
List<Pos> path = new ArrayList<>();
|
||||||
|
path.add(from);
|
||||||
|
|
||||||
|
while (path.get(path.size() - 1) != to) {
|
||||||
|
path.add(route.get(path.get(path.size() - 1)));
|
||||||
|
}
|
||||||
|
|
||||||
|
return path.stream().map(p -> {
|
||||||
|
double floorHeight = floorBlock.get(p);
|
||||||
|
return new Vector(p.x + 0.5, floorHeight, p.z + 0.5);
|
||||||
|
}).collect(Collectors.toList());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
checking.clear();
|
||||||
|
checking.addAll(toCheck);
|
||||||
|
}
|
||||||
|
|
||||||
|
return Collections.emptyList();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
In neuem Issue referenzieren
Einen Benutzer sperren