separate tick steps

This commit is contained in:
Vinzenz Schroeter 2024-04-07 19:52:16 +02:00
parent 898a9cedc1
commit a9aaf899a2
28 changed files with 239 additions and 194 deletions

View file

@ -1,5 +1,4 @@
using TanksServer.Helpers;
using TanksServer.Services;
namespace TanksServer.DrawSteps;

View file

@ -1,6 +1,4 @@
using TanksServer.Helpers;
using TanksServer.Models;
using TanksServer.Services;
namespace TanksServer.DrawSteps;

View file

@ -1,7 +1,6 @@
using SixLabors.ImageSharp;
using SixLabors.ImageSharp.PixelFormats;
using TanksServer.Helpers;
using TanksServer.Services;
namespace TanksServer.DrawSteps;

View file

@ -3,3 +3,6 @@ global using System.Collections.Generic;
global using System.Linq;
global using System.Threading;
global using System.Threading.Tasks;
global using Microsoft.Extensions.Options;
global using TanksServer.Models;
global using TanksServer.Services;

View file

@ -1,5 +1,4 @@
using System.Text.Json.Serialization;
using TanksServer.Models;
namespace TanksServer.Helpers;

View file

@ -1,7 +1,4 @@
using System.Diagnostics;
using System.Runtime.CompilerServices;
using TanksServer.Models;
using TanksServer.Services;
namespace TanksServer.Helpers;

View file

@ -1,5 +1,3 @@
using TanksServer.Services;
namespace TanksServer.Models;
internal readonly record struct FloatPosition(double X, double Y)

View file

@ -0,0 +1,7 @@
namespace TanksServer.Services;
internal sealed class ServicePointDisplayConfiguration
{
public string Hostname { get; set; } = string.Empty;
public int Port { get; set; }
}

View file

@ -18,4 +18,6 @@ internal sealed class Tank(Player player, FloatPosition spawnPosition)
public FloatPosition Position { get; set; } = spawnPosition;
public DateTime NextShotAfter { get; set; }
public bool Moved { get; set; }
}

View file

@ -6,7 +6,8 @@ using Microsoft.Extensions.DependencyInjection;
using Microsoft.Extensions.FileProviders;
using TanksServer.DrawSteps;
using TanksServer.Helpers;
using TanksServer.Services;
using TanksServer.Servers;
using TanksServer.TickSteps;
namespace TanksServer;
@ -73,31 +74,29 @@ internal static class Program
options.SerializerOptions.TypeInfoResolverChain.Insert(0, new AppSerializerContext());
});
builder.Services.AddSingleton<ServicePointDisplay>();
builder.Services.AddSingleton<MapService>();
builder.Services.AddSingleton<BulletManager>();
builder.Services.AddSingleton<TankManager>();
builder.Services.AddSingleton<SpawnNewTanks>();
builder.Services.AddSingleton<PixelDrawer>();
builder.Services.AddSingleton<ControlsServer>();
builder.Services.AddSingleton<PlayerServer>();
builder.Services.AddSingleton<ClientScreenServer>();
builder.Services.AddHostedService<GameTickService>();
builder.Services.AddSingleton<BulletManager>();
builder.Services.AddSingleton<ITickStep>(sp => sp.GetRequiredService<BulletManager>());
builder.Services.AddSingleton<TankManager>();
builder.Services.AddSingleton<ITickStep>(sp => sp.GetRequiredService<TankManager>());
builder.Services.AddSingleton<ControlsServer>();
builder.Services.AddHostedService(sp => sp.GetRequiredService<ControlsServer>());
builder.Services.AddSingleton<SpawnQueue>();
builder.Services.AddSingleton<ITickStep>(sp => sp.GetRequiredService<SpawnQueue>());
builder.Services.AddSingleton<PixelDrawer>();
builder.Services.AddSingleton<ITickStep>(sp => sp.GetRequiredService<PixelDrawer>());
builder.Services.AddSingleton<ClientScreenServer>();
builder.Services.AddHostedService(sp => sp.GetRequiredService<ClientScreenServer>());
builder.Services.AddSingleton<ITickStep>(sp => sp.GetRequiredService<ClientScreenServer>());
builder.Services.AddSingleton<PlayerServer>();
builder.Services.AddSingleton<ITickStep, MoveBullets>();
builder.Services.AddSingleton<ITickStep, CollideBulletsWithTanks>();
builder.Services.AddSingleton<ITickStep, CollideBulletsWithMap>();
builder.Services.AddSingleton<ITickStep, RotateTanks>();
builder.Services.AddSingleton<ITickStep, MoveTanks>();
builder.Services.AddSingleton<ITickStep, ShootFromTanks>();
builder.Services.AddSingleton<ITickStep>(sp => sp.GetRequiredService<SpawnNewTanks>());
builder.Services.AddSingleton<ITickStep>(sp => sp.GetRequiredService<PixelDrawer>());
builder.Services.AddSingleton<ITickStep, SendToServicePointDisplay>();
builder.Services.AddSingleton<ITickStep, SendToClientScreen>();
builder.Services.AddSingleton<IDrawStep, MapDrawer>();
builder.Services.AddSingleton<IDrawStep, TankDrawer>();

View file

@ -5,15 +5,13 @@ using System.Threading.Channels;
using Microsoft.Extensions.Hosting;
using Microsoft.Extensions.Logging;
using TanksServer.Helpers;
using TanksServer.Services;
namespace TanksServer;
namespace TanksServer.Servers;
internal sealed class ClientScreenServer(
ILogger<ClientScreenServer> logger,
ILoggerFactory loggerFactory,
PixelDrawer drawer
) : IHostedLifecycleService, ITickStep
ILoggerFactory loggerFactory
) : IHostedLifecycleService
{
private readonly ConcurrentDictionary<ClientScreenServerConnection, byte> _connections = new();
private bool _closing;
@ -41,12 +39,6 @@ internal sealed class ClientScreenServer(
return Task.WhenAll(_connections.Keys.Select(c => c.CloseAsync()));
}
public Task TickAsync()
{
logger.LogTrace("Sending buffer to {} clients", _connections.Count);
return Task.WhenAll(_connections.Keys.Select(c => c.SendAsync(drawer.LastFrame)));
}
public Task StartAsync(CancellationToken cancellationToken) => Task.CompletedTask;
public Task StopAsync(CancellationToken cancellationToken) => Task.CompletedTask;
public Task StartedAsync(CancellationToken cancellationToken) => Task.CompletedTask;
@ -54,8 +46,10 @@ internal sealed class ClientScreenServer(
public Task StoppedAsync(CancellationToken cancellationToken) => Task.CompletedTask;
private void Remove(ClientScreenServerConnection connection) => _connections.TryRemove(connection, out _);
public IEnumerable<ClientScreenServerConnection> GetConnections() => _connections.Keys;
private sealed class ClientScreenServerConnection: IDisposable
internal sealed class ClientScreenServerConnection: IDisposable
{
private readonly ByteChannelWebSocket _channel;
private readonly SemaphoreSlim _wantedFrames = new(1);

View file

@ -2,9 +2,8 @@ using System.Net.WebSockets;
using Microsoft.Extensions.Hosting;
using Microsoft.Extensions.Logging;
using TanksServer.Helpers;
using TanksServer.Models;
namespace TanksServer;
namespace TanksServer.Servers;
internal sealed class ControlsServer(ILogger<ControlsServer> logger, ILoggerFactory loggerFactory)
: IHostedLifecycleService

View file

@ -1,12 +1,11 @@
using System.Collections.Concurrent;
using System.Diagnostics.CodeAnalysis;
using Microsoft.Extensions.Logging;
using TanksServer.Models;
using TanksServer.Services;
using TanksServer.TickSteps;
namespace TanksServer;
namespace TanksServer.Servers;
internal sealed class PlayerServer(ILogger<PlayerServer> logger, SpawnQueue spawnQueue)
internal sealed class PlayerServer(ILogger<PlayerServer> logger, SpawnNewTanks spawnNewTanks)
{
private readonly ConcurrentDictionary<string, Player> _players = new();
@ -34,7 +33,7 @@ internal sealed class PlayerServer(ILogger<PlayerServer> logger, SpawnQueue spaw
private Player AddAndSpawn(string name)
{
var player = new Player(name);
spawnQueue.SpawnTankForPlayer(player);
spawnNewTanks.SpawnTankForPlayer(player);
return player;
}
}

View file

@ -1,10 +1,6 @@
using System.Collections;
using TanksServer.Helpers;
using TanksServer.Models;
namespace TanksServer.Services;
internal sealed class BulletManager(MapService map) : ITickStep
internal sealed class BulletManager
{
private readonly HashSet<Bullet> _bullets = new();
@ -12,32 +8,8 @@ internal sealed class BulletManager(MapService map) : ITickStep
public IEnumerable<Bullet> GetAll() => _bullets;
public Task TickAsync()
public void RemoveWhere(Predicate<Bullet> predicate)
{
HashSet<Bullet> bulletsToRemove = new();
foreach (var bullet in _bullets)
{
MoveBullet(bullet);
if (BulletHitsWall(bullet))
bulletsToRemove.Add(bullet);
}
_bullets.RemoveWhere(b => bulletsToRemove.Contains(b));
return Task.CompletedTask;
}
private static void MoveBullet(Bullet bullet)
{
var angle = bullet.Rotation / 16 * 2 * Math.PI;
bullet.Position = new FloatPosition(
X: bullet.Position.X + Math.Sin(angle) * 3,
Y: bullet.Position.Y - Math.Cos(angle) * 3
);
}
private bool BulletHitsWall(Bullet bullet)
{
return map.IsCurrentlyWall(bullet.Position.ToPixelPosition().ToTilePosition());
_bullets.RemoveWhere(predicate);
}
}

View file

@ -1,4 +1,5 @@
using Microsoft.Extensions.Hosting;
using TanksServer.TickSteps;
namespace TanksServer.Services;
@ -36,8 +37,3 @@ internal sealed class GameTickService(IEnumerable<ITickStep> steps) : IHostedSer
_run?.Dispose();
}
}
public interface ITickStep
{
Task TickAsync();
}

View file

@ -1,5 +1,3 @@
using TanksServer.Models;
namespace TanksServer.Services;
internal sealed class MapService

View file

@ -1,11 +1,6 @@
using System.Diagnostics;
using System.Net.Mime;
using Microsoft.Extensions.Logging;
using SixLabors.ImageSharp;
using SixLabors.ImageSharp.PixelFormats;
using TanksServer.DrawSteps;
using TanksServer.Helpers;
using TanksServer.Models;
using TanksServer.TickSteps;
namespace TanksServer.Services;

View file

@ -1,20 +1,12 @@
using System.Collections;
using System.Collections.Concurrent;
using Microsoft.Extensions.Logging;
using Microsoft.Extensions.Options;
using TanksServer.Models;
namespace TanksServer.Services;
internal sealed class TankManager(
ILogger<TankManager> logger,
IOptions<TanksConfiguration> options,
MapService map,
BulletManager bullets
) : ITickStep, IEnumerable<Tank>
internal sealed class TankManager(ILogger<TankManager> logger) : IEnumerable<Tank>
{
private readonly ConcurrentBag<Tank> _tanks = new();
private readonly TanksConfiguration _config = options.Value;
public void Add(Tank tank)
{
@ -22,86 +14,6 @@ internal sealed class TankManager(
_tanks.Add(tank);
}
public Task TickAsync()
{
foreach (var tank in _tanks)
{
if (TryMoveTank(tank))
continue;
Shoot(tank);
}
return Task.CompletedTask;
}
private bool TryMoveTank(Tank tank)
{
logger.LogTrace("moving tank for player {}", tank.Owner.Id);
var player = tank.Owner;
if (player.Controls.TurnLeft)
tank.Rotation -= _config.TurnSpeed;
if (player.Controls.TurnRight)
tank.Rotation += _config.TurnSpeed;
double speed;
switch (player.Controls)
{
case { Forward: false, Backward: false }:
case { Forward: true, Backward: true }:
return false;
case { Forward: true }:
speed = +_config.MoveSpeed;
break;
case { Backward: true }:
speed = -_config.MoveSpeed;
break;
default:
return false;
}
var angle = tank.Rotation / 16d * 2d * Math.PI;
var newX = tank.Position.X + Math.Sin(angle) * speed;
var newY = tank.Position.Y - Math.Cos(angle) * speed;
return TryMove(tank, new FloatPosition(newX, newY))
|| TryMove(tank, tank.Position with { X = newX })
|| TryMove(tank, tank.Position with { Y = newY });
}
private bool TryMove(Tank tank, FloatPosition newPosition)
{
var x0 = (int)Math.Floor(newPosition.X / MapService.TileSize);
var x1 = (int)Math.Ceiling(newPosition.X / MapService.TileSize);
var y0 = (int)Math.Floor(newPosition.Y / MapService.TileSize);
var y1 = (int)Math.Ceiling(newPosition.Y / MapService.TileSize);
TilePosition[] positions = { new(x0, y0), new(x0, y1), new(x1, y0), new(x1, y1) };
if (positions.Any(map.IsCurrentlyWall))
return false;
tank.Position = newPosition;
return true;
}
private void Shoot(Tank tank)
{
if (!tank.Owner.Controls.Shoot)
return;
if (tank.NextShotAfter >= DateTime.Now)
return;
tank.NextShotAfter = DateTime.Now.AddMilliseconds(_config.ShootDelayMs);
var angle = tank.Rotation / 16 * 2 * Math.PI;
var position = new FloatPosition(
X: tank.Position.X + MapService.TileSize / 2d + Math.Sin(angle) * _config.BulletSpeed,
Y: tank.Position.Y + MapService.TileSize / 2d - Math.Cos(angle) * _config.BulletSpeed
);
bullets.Spawn(new Bullet(tank.Owner, position, tank.Rotation));
}
IEnumerator IEnumerable.GetEnumerator() => GetEnumerator();
public IEnumerator<Tank> GetEnumerator() => _tanks.GetEnumerator();
}

View file

@ -0,0 +1,17 @@
using TanksServer.Helpers;
namespace TanksServer.TickSteps;
internal sealed class CollideBulletsWithMap(BulletManager bullets, MapService map) : ITickStep
{
public Task TickAsync()
{
bullets.RemoveWhere(BulletHitsWall);
return Task.CompletedTask;
}
private bool BulletHitsWall(Bullet bullet)
{
return map.IsCurrentlyWall(bullet.Position.ToPixelPosition().ToTilePosition());
}
}

View file

@ -0,0 +1,15 @@
namespace TanksServer.TickSteps;
internal sealed class CollideBulletsWithTanks(BulletManager bullets) : ITickStep
{
public Task TickAsync()
{
bullets.RemoveWhere(BulletHitsTank);
return Task.CompletedTask;
}
private bool BulletHitsTank(Bullet bullet)
{
return false; // TODO
}
}

View file

@ -0,0 +1,6 @@
namespace TanksServer.TickSteps;
public interface ITickStep
{
Task TickAsync();
}

View file

@ -0,0 +1,21 @@
namespace TanksServer.TickSteps;
internal sealed class MoveBullets(BulletManager bullets) : ITickStep
{
public Task TickAsync()
{
foreach (var bullet in bullets.GetAll())
MoveBullet(bullet);
return Task.CompletedTask;
}
private static void MoveBullet(Bullet bullet)
{
var angle = bullet.Rotation / 16 * 2 * Math.PI;
bullet.Position = new FloatPosition(
X: bullet.Position.X + Math.Sin(angle) * 3,
Y: bullet.Position.Y - Math.Cos(angle) * 3
);
}
}

View file

@ -0,0 +1,60 @@
namespace TanksServer.TickSteps;
internal sealed class MoveTanks(
TankManager tanks, IOptions<TanksConfiguration> options, MapService map
) : ITickStep
{
private readonly TanksConfiguration _config = options.Value;
public Task TickAsync()
{
foreach (var tank in tanks)
tank.Moved = TryMoveTank(tank);
return Task.CompletedTask;
}
private bool TryMoveTank(Tank tank)
{
var player = tank.Owner;
double speed;
switch (player.Controls)
{
case { Forward: false, Backward: false }:
case { Forward: true, Backward: true }:
return false;
case { Forward: true }:
speed = +_config.MoveSpeed;
break;
case { Backward: true }:
speed = -_config.MoveSpeed;
break;
default:
return false;
}
var angle = tank.Rotation / 16d * 2d * Math.PI;
var newX = tank.Position.X + Math.Sin(angle) * speed;
var newY = tank.Position.Y - Math.Cos(angle) * speed;
return TryMoveTankTo(tank, new FloatPosition(newX, newY))
|| TryMoveTankTo(tank, tank.Position with { X = newX })
|| TryMoveTankTo(tank, tank.Position with { Y = newY });
}
private bool TryMoveTankTo(Tank tank, FloatPosition newPosition)
{
var x0 = (int)Math.Floor(newPosition.X / MapService.TileSize);
var x1 = (int)Math.Ceiling(newPosition.X / MapService.TileSize);
var y0 = (int)Math.Floor(newPosition.Y / MapService.TileSize);
var y1 = (int)Math.Ceiling(newPosition.Y / MapService.TileSize);
TilePosition[] positions = { new(x0, y0), new(x0, y1), new(x1, y0), new(x1, y1) };
if (positions.Any(map.IsCurrentlyWall))
return false;
tank.Position = newPosition;
return true;
}
}

View file

@ -0,0 +1,21 @@
namespace TanksServer.TickSteps;
internal sealed class RotateTanks(TankManager tanks, IOptions<TanksConfiguration> options) : ITickStep
{
private readonly TanksConfiguration _config = options.Value;
public Task TickAsync()
{
foreach (var tank in tanks)
{
var player = tank.Owner;
if (player.Controls.TurnLeft)
tank.Rotation -= _config.TurnSpeed;
if (player.Controls.TurnRight)
tank.Rotation += _config.TurnSpeed;
}
return Task.CompletedTask;
}
}

View file

@ -0,0 +1,13 @@
using TanksServer.Servers;
namespace TanksServer.TickSteps;
internal sealed class SendToClientScreen(
ClientScreenServer clientScreenServer, PixelDrawer drawer
) : ITickStep
{
public Task TickAsync()
{
return Task.WhenAll(clientScreenServer.GetConnections().Select(c => c.SendAsync(drawer.LastFrame)));
}
}

View file

@ -1,9 +1,8 @@
using System.Net.Sockets;
using Microsoft.Extensions.Options;
namespace TanksServer.Services;
namespace TanksServer.TickSteps;
internal sealed class ServicePointDisplay(
internal sealed class SendToServicePointDisplay(
IOptions<ServicePointDisplayConfiguration> options,
PixelDrawer drawer
) : ITickStep, IDisposable
@ -20,9 +19,3 @@ internal sealed class ServicePointDisplay(
_udpClient.Dispose();
}
}
internal sealed class ServicePointDisplayConfiguration
{
public string Hostname { get; set; } = string.Empty;
public int Port { get; set; }
}

View file

@ -0,0 +1,34 @@
namespace TanksServer.TickSteps;
internal sealed class ShootFromTanks(
TankManager tanks, IOptions<TanksConfiguration> options, BulletManager bulletManager
) : ITickStep
{
private readonly TanksConfiguration _config = options.Value;
public Task TickAsync()
{
foreach (var tank in tanks.Where(t => !t.Moved))
Shoot(tank);
return Task.CompletedTask;
}
private void Shoot(Tank tank)
{
if (!tank.Owner.Controls.Shoot)
return;
if (tank.NextShotAfter >= DateTime.Now)
return;
tank.NextShotAfter = DateTime.Now.AddMilliseconds(_config.ShootDelayMs);
var angle = tank.Rotation / 16 * 2 * Math.PI;
var position = new FloatPosition(
X: tank.Position.X + MapService.TileSize / 2d + Math.Sin(angle) * _config.BulletSpeed,
Y: tank.Position.Y + MapService.TileSize / 2d - Math.Cos(angle) * _config.BulletSpeed
);
bulletManager.Spawn(new Bullet(tank.Owner, position, tank.Rotation));
}
}

View file

@ -1,9 +1,8 @@
using System.Collections.Concurrent;
using TanksServer.Models;
namespace TanksServer.Services;
namespace TanksServer.TickSteps;
internal sealed class SpawnQueue(TankManager tanks, MapService map) : ITickStep
internal sealed class SpawnNewTanks(TankManager tanks, MapService map) : ITickStep
{
private readonly ConcurrentQueue<Player> _playersToSpawn = new();