servicepoint-tanks/tanks-backend/TanksServer/Interactivity/SendToServicePointDisplay.cs

99 lines
3.2 KiB
C#
Raw Normal View History

using System.Diagnostics;
using System.Net.Sockets;
using DisplayCommands;
using TanksServer.GameLogic;
using TanksServer.Graphics;
namespace TanksServer.Interactivity;
internal sealed class SendToServicePointDisplay : IFrameConsumer
{
2024-04-13 14:08:51 +02:00
private const int ScoresWidth = 12;
private const int ScoresHeight = 20;
2024-04-19 13:41:53 +02:00
private const int ScoresPlayerRows = ScoresHeight - 6;
2024-04-13 14:08:51 +02:00
private readonly IDisplayConnection _displayConnection;
2024-04-19 13:41:53 +02:00
private readonly MapService _mapService;
private readonly ILogger<SendToServicePointDisplay> _logger;
2024-04-13 14:08:51 +02:00
private readonly PlayerServer _players;
private readonly Cp437Grid _scoresBuffer;
private readonly TimeSpan _minFrameTime;
private DateTime _nextFailLogAfter = DateTime.Now;
private DateTime _nextFrameAfter = DateTime.Now;
public SendToServicePointDisplay(
PlayerServer players,
ILogger<SendToServicePointDisplay> logger,
IDisplayConnection displayConnection,
2024-04-19 13:41:53 +02:00
IOptions<HostConfiguration> hostOptions,
MapService mapService
)
{
_players = players;
_logger = logger;
_displayConnection = displayConnection;
2024-04-19 13:41:53 +02:00
_mapService = mapService;
_minFrameTime = TimeSpan.FromMilliseconds(hostOptions.Value.ServicePointDisplayMinFrameTimeMs);
var localIp = _displayConnection.GetLocalIPv4().Split('.');
Debug.Assert(localIp.Length == 4);
_scoresBuffer = new Cp437Grid(12, 20)
{
[00] = "== TANKS! ==",
[01] = "-- scores --",
[17] = "-- join --",
[18] = string.Join('.', localIp[..2]),
[19] = string.Join('.', localIp[2..])
};
}
public async Task OnFrameDoneAsync(GamePixelGrid gamePixelGrid, PixelGrid observerPixels)
{
if (DateTime.Now < _nextFrameAfter)
return;
_nextFrameAfter = DateTime.Now + _minFrameTime;
RefreshScores();
try
{
await _displayConnection.SendBitmapLinearWindowAsync(0, 0, observerPixels);
await _displayConnection.SendCp437DataAsync(MapService.TilesPerRow, 0, _scoresBuffer);
}
catch (SocketException ex)
{
if (DateTime.Now > _nextFailLogAfter)
{
_logger.LogWarning("could not send data to service point display: {}", ex.Message);
_nextFailLogAfter = DateTime.Now + TimeSpan.FromSeconds(5);
}
}
}
private void RefreshScores()
{
var playersToDisplay = _players.GetAll()
2024-04-13 18:35:36 +02:00
.OrderByDescending(p => p.Scores.Kills)
.Take(ScoresPlayerRows);
ushort row = 2;
foreach (var p in playersToDisplay)
{
2024-04-13 18:35:36 +02:00
var score = p.Scores.Kills.ToString();
var nameLength = Math.Min(p.Name.Length, ScoresWidth - score.Length - 1);
var name = p.Name[..nameLength];
var spaces = new string(' ', ScoresWidth - score.Length - nameLength);
_scoresBuffer[row] = name + spaces + score;
row++;
}
2024-04-19 13:41:53 +02:00
for (; row < 16; row++)
_scoresBuffer[row] = string.Empty;
2024-04-19 13:41:53 +02:00
_scoresBuffer[16] = _mapService.Current.Name[..(Math.Min(ScoresWidth, _mapService.Current.Name.Length) - 1)];
}
2024-04-13 14:08:51 +02:00
}