AdventOfCode/AdventOfCode.Core/Shared/Grid.cs

50 lines
2.0 KiB
C#

namespace AdventOfCode.Core.Shared
{
public class Grid<T> where T : Node
{
public List<T> DataGrid { get; set; } = [];
public int Columns => DataGrid.Max(n => n.X) + 1;
public int Rows => DataGrid.Max(n => n.Y) + 1;
public Grid() { }
public Grid(T[] data) => DataGrid.AddRange(data);
public Grid(IEnumerable<T> data) : this(data.ToArray()) { }
public IEnumerable<T> GetRow(int rowNumber) => DataGrid.Where(n => n.X == rowNumber);
public string GetRowAsString(int columnNumber) => string.Concat(DataGrid.Where(n => n.Y == columnNumber).OrderBy(n => n.X).Select(n => n.Char));
public IEnumerable<T> GetColumn(int columnNumber) => DataGrid.Where(n => n.Y == columnNumber);
public string GetColumnAsString(int rowNumber) => string.Concat(DataGrid.Where(n => n.X == rowNumber).OrderBy(n => n.Y).Select(n => n.Char));
public T GetNode(int x, int y) => DataGrid.First(n => n.X == x && n.Y == y);
public IEnumerable<T> FindWithValue(char toFind) => DataGrid.Where(n => n.Char == toFind);
public void UpdateNode(T node)
{
T gridNode = DataGrid.First(n => n.X == node.X && n.Y == node.Y);
int nodeLocation = DataGrid.IndexOf(gridNode);
DataGrid[nodeLocation] = node;
}
public IEnumerable<T> GetNeighbors(T source, bool allowDiagonals = true)
{
IEnumerable <T> neighbors = DataGrid.Where(target => Math.Abs(source.X - target.X) <= 1 && Math.Abs(source.Y - target.Y) <= 1);
if (allowDiagonals)
{
return neighbors;
}
return neighbors.Where(target => (Math.Abs(source.X - target.X) <= 1 && source.Y == target.Y)
|| (source.X == target.X && Math.Abs(source.Y - target.Y) <= 1));
}
public IEnumerable<T> GetSection(int fromX, int fromY, int toX, int toY)
{
return DataGrid.Where(node => node.X >= fromX && node.X <= toX && node.Y >= fromY && node.Y <= toY);
}
}
}