2026-03-08 16:55:49 +01:00
|
|
|
using GlobalClassLib;
|
2026-03-22 18:31:05 +01:00
|
|
|
using ONDServer.Map;
|
2026-03-08 16:55:49 +01:00
|
|
|
|
2026-03-22 18:31:05 +01:00
|
|
|
namespace ONDServer.Enemies;
|
2026-03-08 16:55:49 +01:00
|
|
|
|
2026-03-19 20:10:45 +01:00
|
|
|
public class RoamingPathfinder : Pathfinder {
|
2026-03-17 20:14:29 +01:00
|
|
|
|
2026-03-19 20:10:45 +01:00
|
|
|
public Predicate<MapTile> AdditionalTileFilter{ get; set; } = _ => true;
|
|
|
|
|
public Predicate<TileConnector> AdditionalConnectorFilter{ get; set; } = _ => true;
|
2026-03-17 20:14:29 +01:00
|
|
|
|
2026-03-19 20:10:45 +01:00
|
|
|
protected int tolerance;
|
|
|
|
|
public List<MapTile> TakenPath { get; } = new();
|
|
|
|
|
|
|
|
|
|
private Random random = new();
|
2026-03-25 16:37:18 +01:00
|
|
|
|
|
|
|
|
// private Queue<MapTile> tilesToCrawl = new();
|
|
|
|
|
// private Dictionary<MapTile, int> distances = new();
|
2026-03-17 20:14:29 +01:00
|
|
|
// private Func<TileConnector, MapTile, bool> defaultConnectorFilter;
|
2026-03-19 20:10:45 +01:00
|
|
|
public RoamingPathfinder(Enemy enemy, int tolerance) : base(enemy) {
|
|
|
|
|
this.tolerance = tolerance;
|
2026-03-08 16:55:49 +01:00
|
|
|
}
|
|
|
|
|
|
2026-03-25 16:37:18 +01:00
|
|
|
protected Dictionary<MapTile, int> Search(MapTile startingTile) {
|
|
|
|
|
Dictionary<MapTile, int> distances = new(){ [startingTile] = 0 };
|
|
|
|
|
Queue<MapTile> tilesToSearch = new();
|
|
|
|
|
tilesToSearch.Enqueue(startingTile);
|
2026-03-08 16:55:49 +01:00
|
|
|
|
2026-03-25 16:37:18 +01:00
|
|
|
while (tilesToSearch.Count > 0){
|
|
|
|
|
MapTile currentTile = tilesToSearch.Dequeue();
|
2026-03-08 16:55:49 +01:00
|
|
|
|
2026-03-25 16:37:18 +01:00
|
|
|
List<MapTile> neighbours = GetNeighbours(currentTile,
|
|
|
|
|
c =>
|
|
|
|
|
AdditionalConnectorFilter(c) &&
|
|
|
|
|
(!c.Blocked || c.Type == ConnectorType.DOOR_OFFICE) &&
|
|
|
|
|
(!distances.ContainsKey(c.OtherTile(currentTile)) || distances[c.OtherTile(currentTile)] > distances[currentTile] + c.Value),
|
|
|
|
|
t =>
|
|
|
|
|
AdditionalTileFilter(t) &&
|
|
|
|
|
(!Enemy.BlocksTile || EnemyManager.GetByLocation(t).All(e => !e.BlocksTile || e == Enemy)) &&
|
|
|
|
|
Server.Players.All(p => p.Value.state.officeTileId != t.Id));
|
2026-03-08 16:55:49 +01:00
|
|
|
|
2026-03-25 16:37:18 +01:00
|
|
|
neighbours.ForEach(t => {
|
|
|
|
|
distances[t] = distances[currentTile] + currentTile.GetConnector(t)!.Value;
|
|
|
|
|
tilesToSearch.Enqueue(t);
|
|
|
|
|
});
|
2026-03-08 16:55:49 +01:00
|
|
|
}
|
2026-03-25 16:37:18 +01:00
|
|
|
|
|
|
|
|
return distances;
|
2026-03-08 16:55:49 +01:00
|
|
|
}
|
2026-03-25 16:37:18 +01:00
|
|
|
|
|
|
|
|
// private void CrawlStep(MapTile tile) {
|
|
|
|
|
// List<MapTile> neighbours = GetNeighbours(tile,
|
|
|
|
|
// c =>
|
|
|
|
|
// AdditionalConnectorFilter(c) &&
|
|
|
|
|
// (!c.Blocked || c.Type == ConnectorType.DOOR_OFFICE) &&
|
|
|
|
|
// tile != Enemy.Location &&
|
|
|
|
|
// (!distances.ContainsKey(c.OtherTile(tile)) || distances[c.OtherTile(tile)] > distances[tile] + c.Value),
|
|
|
|
|
// t =>
|
|
|
|
|
// AdditionalTileFilter(t) &&
|
|
|
|
|
// (!Enemy.BlocksTile || EnemyManager.GetByLocation(t).All(e => !e.BlocksTile || e == Enemy)) &&
|
|
|
|
|
// Server.Players.All(p => p.Value.state.officeTileId != t.Id));
|
|
|
|
|
//
|
|
|
|
|
// neighbours.ForEach(t => distances[t] = distances[tile] + tile.GetConnector(t)!.Value);
|
|
|
|
|
//
|
|
|
|
|
// if (neighbours.Count != 0){
|
|
|
|
|
// neighbours.ForEach(t => CrawlStep(t));
|
|
|
|
|
// }
|
|
|
|
|
// }
|
2026-03-19 20:10:45 +01:00
|
|
|
|
|
|
|
|
public override Decision DecideNext(MapTile goal) {
|
2026-03-25 16:37:18 +01:00
|
|
|
Dictionary<MapTile, int> distances = Search(goal);
|
2026-03-19 20:10:45 +01:00
|
|
|
|
|
|
|
|
List<MapTile> closerTiles = GetNeighbours(Enemy.Location,
|
|
|
|
|
c => AdditionalConnectorFilter(c) && !c.Blocked || c.Type == ConnectorType.DOOR_OFFICE,
|
|
|
|
|
t => AdditionalTileFilter(t) && distances.ContainsKey(t) && distances[t] + t.GetConnector(Enemy.Location).Value <= distances[Enemy.Location] + tolerance);
|
|
|
|
|
if (closerTiles.Contains(goal)){
|
|
|
|
|
if (Enemy.Location.GetConnector(goal)!.Blocked){
|
|
|
|
|
return Decision.Reset();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
IEnumerable<ServerPlayer> players = Server.Players.Values.Where(p => p.state.officeTileId == goal.Id);
|
|
|
|
|
if (players.Count() == 1){
|
|
|
|
|
return Decision.Attack(players.First());
|
|
|
|
|
}
|
|
|
|
|
return Decision.Move(goal);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (closerTiles.Count != 0 && closerTiles.All(t => TakenPath.Contains(t))){
|
|
|
|
|
TakenPath.Clear();
|
|
|
|
|
return DecideNext(goal);
|
|
|
|
|
}
|
|
|
|
|
closerTiles.RemoveAll(t => TakenPath.Contains(t));
|
|
|
|
|
|
|
|
|
|
closerTiles.ForEach(t => distances[t] += Enemy.Location.GetConnector(t)!.Value);
|
|
|
|
|
double roll = random.NextDouble() * closerTiles.Sum(t => 1.0 / distances[t]);
|
|
|
|
|
foreach (var tile in closerTiles){
|
|
|
|
|
double value = 1.0 / distances[tile];
|
|
|
|
|
if (roll <= value){
|
|
|
|
|
TakenPath.Add(tile);
|
|
|
|
|
return Decision.Move(tile);
|
|
|
|
|
}
|
|
|
|
|
roll -= value;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return Decision.Wait();
|
|
|
|
|
}
|
2026-03-08 16:55:49 +01:00
|
|
|
}
|