mirror of
https://github.com/OpenDiablo2/OpenDiablo2
synced 2024-11-05 09:47:18 -05:00
173 lines
6.4 KiB
C#
173 lines
6.4 KiB
C#
using System;
|
|
using System.Collections.Generic;
|
|
using System.Drawing;
|
|
using System.Linq;
|
|
using OpenDiablo2.Common.Enums;
|
|
using OpenDiablo2.Common.Interfaces;
|
|
using OpenDiablo2.Common.Interfaces.Drawing;
|
|
using OpenDiablo2.Common.Models;
|
|
|
|
namespace OpenDiablo2.Core.Map_Engine
|
|
{
|
|
public sealed class MapEngine : IMapEngine
|
|
{
|
|
private readonly IGameState gameState;
|
|
private readonly IRenderWindow renderWindow;
|
|
|
|
private readonly List<ICharacterRenderer> characterRenderers = new List<ICharacterRenderer>();
|
|
|
|
public int FocusedPlayerId { get; set; } = 0;
|
|
|
|
private PointF cameraLocation = new PointF();
|
|
public PointF CameraLocation
|
|
{
|
|
get => cameraLocation;
|
|
set
|
|
{
|
|
if (cameraLocation == value)
|
|
return;
|
|
|
|
cameraLocation = value;
|
|
//cOffX = (int)((cameraLocation.X - cameraLocation.Y) * (cellSizeX / 2));
|
|
//cOffY = (int)((cameraLocation.X + cameraLocation.Y) * (cellSizeY / 2));
|
|
}
|
|
}
|
|
|
|
private const int
|
|
cellSizeX = 160,
|
|
cellSizeY = 80;
|
|
|
|
public MapEngine(
|
|
IGameState gameState,
|
|
IRenderWindow renderWindow,
|
|
ISessionManager sessionManager
|
|
)
|
|
{
|
|
this.gameState = gameState;
|
|
this.renderWindow = renderWindow;
|
|
|
|
sessionManager.OnPlayerInfo += OnPlayerInfo;
|
|
sessionManager.OnLocatePlayers += OnLocatePlayers;
|
|
}
|
|
|
|
private void OnLocatePlayers(int clientHash, IEnumerable<PlayerLocationDetails> playerLocationDetails)
|
|
{
|
|
foreach(var loc in playerLocationDetails)
|
|
{
|
|
var cr = characterRenderers.FirstOrDefault(x => x.LocationDetails.PlayerId == loc.PlayerId);
|
|
var newDirection = loc.MovementDirection != cr.LocationDetails.MovementDirection;
|
|
var stanceChanged = loc.MovementType != cr.LocationDetails.MovementType;
|
|
cr.LocationDetails = loc;
|
|
if (newDirection || stanceChanged)
|
|
cr.ResetAnimationData();
|
|
}
|
|
}
|
|
|
|
private void OnPlayerInfo(int clientHash, IEnumerable<PlayerInfo> playerInfo)
|
|
{
|
|
// Remove character renderers for players that no longer exist...
|
|
characterRenderers.RemoveAll(x => playerInfo.Any(z => z.UID == x.UID));
|
|
|
|
// Update existing character renderers
|
|
foreach (var cr in characterRenderers)
|
|
{
|
|
var info = playerInfo.FirstOrDefault(x => x.UID == cr.UID);
|
|
if (info == null)
|
|
continue;
|
|
|
|
// TODO: This shouldn't be necessary...
|
|
cr.LocationDetails = info.LocationDetails;
|
|
cr.MobMode = info.MobMode;
|
|
cr.WeaponClass = info.WeaponClass;
|
|
cr.Hero = info.Hero;
|
|
cr.ArmorType = info.ArmorType;
|
|
|
|
}
|
|
|
|
// Add character renderers for characters that now exist
|
|
foreach(var info in playerInfo.Where(x => !characterRenderers.Any(z => x.UID == z.UID)))
|
|
{
|
|
var cr = renderWindow.CreateCharacterRenderer();
|
|
cr.UID = info.UID;
|
|
cr.LocationDetails = info.LocationDetails;
|
|
cr.MobMode = info.MobMode;
|
|
cr.WeaponClass = info.WeaponClass;
|
|
cr.Hero = info.Hero;
|
|
cr.ArmorType = info.ArmorType;
|
|
characterRenderers.Add(cr);
|
|
cr.ResetAnimationData();
|
|
}
|
|
}
|
|
|
|
public void Render()
|
|
{
|
|
var xOffset = gameState.CameraOffset;
|
|
|
|
var cx = -(cameraLocation.X - Math.Truncate(cameraLocation.X));
|
|
var cy = -(cameraLocation.Y - Math.Truncate(cameraLocation.Y));
|
|
|
|
for (int ty = -5; ty <= 9; ty++)
|
|
{
|
|
for (int tx = -5; tx <= 9; tx++)
|
|
{
|
|
var ax = tx + Math.Truncate(cameraLocation.X);
|
|
var ay = ty + Math.Truncate(cameraLocation.Y);
|
|
|
|
var px = (tx - ty) * (cellSizeX / 2);
|
|
var py = (tx + ty) * (cellSizeY / 2);
|
|
|
|
var ox = (cx - cy) * (cellSizeX / 2);
|
|
var oy = (cx + cy) * (cellSizeY / 2);
|
|
|
|
|
|
foreach (var cellInfo in gameState.GetMapCellInfo((int)ax, (int)ay, eRenderCellType.Floor))
|
|
renderWindow.DrawMapCell(cellInfo, 320 + px + (int)ox + xOffset, 210 + py + (int)oy);
|
|
|
|
|
|
foreach (var cellInfo in gameState.GetMapCellInfo((int)ax, (int)ay, eRenderCellType.WallLower))
|
|
renderWindow.DrawMapCell(cellInfo, 320 + px + (int)ox + xOffset, 210 + py + (int)oy);
|
|
|
|
foreach (var cellInfo in gameState.GetMapCellInfo((int)ax, (int)ay, eRenderCellType.WallUpper))
|
|
renderWindow.DrawMapCell(cellInfo, 320 + px + (int)ox + xOffset, 210 + py + (int)oy);
|
|
|
|
// TODO: We need to render the characters infront of, or behind the wall properly...
|
|
if (ty == 1 && tx == 1)
|
|
{
|
|
foreach (var character in characterRenderers/*.Where(x => Math.Truncate(x.LocationDetails.PlayerX) == ax && Math.Truncate(x.LocationDetails.PlayerY) == ay)*/)
|
|
{
|
|
// TODO: Temporary hack
|
|
character.Render(400 + gameState.CameraOffset, 280);
|
|
}
|
|
}
|
|
|
|
foreach (var cellInfo in gameState.GetMapCellInfo((int)ax, (int)ay, eRenderCellType.Roof))
|
|
renderWindow.DrawMapCell(cellInfo, 320 + px + (int)ox + xOffset, 210 + py + (int)oy);
|
|
}
|
|
}
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
public void Update(long ms)
|
|
{
|
|
foreach (var character in characterRenderers)
|
|
character.Update(ms);
|
|
|
|
if (FocusedPlayerId != 0)
|
|
{
|
|
var player = gameState.PlayerInfos.FirstOrDefault(x => x.LocationDetails.PlayerId == FocusedPlayerId);
|
|
if (player != null)
|
|
{
|
|
// TODO: Maybe smooth movement? Maybe not?
|
|
CameraLocation = new PointF(player.LocationDetails.PlayerX, player.LocationDetails.PlayerY);
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
}
|
|
}
|