using System;
using System.Collections.Generic;
using System.Linq;
using System.Text.RegularExpressions;
using System.Diagnostics;
namespace ConsoleApp1 {
class Program {
private abstract class Board {
public enum Direction {
Left,
Right,
Up,
Down,
DownLeft,
UpLeft
}
protected String String;
public class Cell {
public HashSet<Char> Set;
public Char Current;
public Cell(String String) {
Set = new HashSet<Char>(String.ToArray());
Current = '\0';
}
}
public class Location {
public Int32 Row;
public Int32 Column;
public Location(Int32 Row, Int32 Column) {
this.Row = Row;
this.Column = Column;
}
public Location(Location From) {
Row = From.Row;
Column = From.Column;
}
}
protected Cell[][] Cells;
public abstract Location GetLocation(Direction Direction, Int32 Distance, Location Locaton);
protected abstract void MakeCells(Object Parameter);
public Board(Object Parameter, String String) {
this.String = String;
MakeCells(Parameter);
var Characters = String.ToArray();
foreach(var Row in Cells) {
foreach(var Cell in Row) {
Cell.Set = new HashSet<Char>(Characters);
}
}
}
protected Cell[] MakeCells(Int32 Size) {
var Cells = new Cell[Size];
for(var Index = 0; Index < Cells.Length; Index++) {
Cells[Index] = new Cell(String);
}
return Cells;
}
public IEnumerable<string> Strings(Char [] Characters, Condition Condition, int Distance) {
if(Distance < Condition.NumberOfCells) {
foreach(var Character in this[GetLocation(Condition.Direction, Distance, Condition.Start)].Set) {
Characters[Distance] = Character;
foreach(var String in Strings(Characters, Condition, Distance + 1)){
yield return String;
}
}
} else {
yield return string.Join("", Characters);
}
}
private IEnumerable<String> Strings(Condition Condition) {
var Characters = new Char[Condition.NumberOfCells];
for(var Index = 0; Index < Characters.Length; Index++) {
Characters[Index] = '\0';
}
foreach(var String in Strings(Characters, Condition, 0)) {
yield return String;
}
}
public Cell this[Location Location]{
get {
return Cells[Location.Row][Location.Column];
}
}
public long Reduce(Condition[] Conditions) {
var Count = 1L;
foreach(var Condition in Conditions){
var Characters = new Char[Condition.NumberOfCells];
var Sets = new HashSet<Char>[Condition.NumberOfCells];
for(var Index = 0; Index < Characters.Length; Index++) {
Characters[Index] = '\0';
Sets[Index] = new HashSet<char>();
}
var RegularExpression = new Regex($"^{Condition.RegularExpression}$", RegexOptions.Compiled);
foreach(var String in Strings(Characters, Condition, 0)) {
if(RegularExpression.IsMatch(String)){
for(var Index = 0; Index < Characters.Length; Index++) {
Sets[Index].Add(String[Index]);
}
}
}
long Count1 = 1;
for(var Index = 0; Index < Sets.Length; Index++) {
this[GetLocation(Condition.Direction, Index, Condition.Start)].Set = Sets[Index];
Count1 *= Sets[Index].Count;
}
Console.WriteLine($"{Condition.NumberOfCells:D}: {Count1:###,###,##0}: {Condition.RegularExpression}");
Count *= Count1;
}
return Count;
}
private IEnumerable<Char[][]> Resolve(Condition[] Conditions, Int32 ConditionIndex) {
if(ConditionIndex < Conditions.Length) {
var Condition = Conditions[ConditionIndex];
var RegularExpression = new Regex($"^{Condition.RegularExpression}$", RegexOptions.Compiled);
foreach(var String in Strings(Condition)) {
if(RegularExpression.IsMatch(String)){
var SaveSets = new HashSet<Char>[Condition.NumberOfCells];
for(var Index = 0; Index < SaveSets.Length; Index++) {
var Cell = this[GetLocation(Condition.Direction, Index, Condition.Start)];
SaveSets[Index] = Cell.Set;
Cell.Set = new HashSet<char>();
Cell.Set.Add(String[Index]);
}
foreach(var Result in Resolve(Conditions, ConditionIndex + 1)) {
yield return Result;
}
for(var Index = 0; Index < SaveSets.Length; Index++) {
var Cell = this[GetLocation(Condition.Direction, Index, Condition.Start)];
Cell.Set = SaveSets[Index];
}
}
}
} else {
var Cells = new Char[this.Cells.Length][];
for(var Row = 0; Row < this.Cells.Length; Row++) {
Cells[Row] = new Char[this.Cells[Row].Length];
for(var Column = 0; Column < this.Cells[Row].Length; Column++) {
Cells[Row][Column] = this.Cells[Row][Column].Set.Single();
}
}
yield return Cells;
}
}
public IEnumerable<Char[][]> Resolve(Condition[] Conditions) {
foreach(var Result in Resolve(Conditions, 0)){
yield return Result;
}
}
}
private class HoneycombBoard: Board {
private Int32 Size;
public override Location GetLocation(Direction Direction, int Distance, Location From) {
var To = new Location(From);
switch(Direction) {
case Direction.Right:
break;
case Direction.DownLeft:
To.Row += Distance;
break;
case Direction.UpLeft:
To.Row -= Distance;
break;
default:
Debug.Fail($"{nameof(GetLocation)}: Row: {Direction.ToString()}");
break;
}
switch(Direction) {
case Direction.Right:
To.Column += Distance;
break;
case Direction.DownLeft:
if(Size - 1 < To.Row) {
To.Column -= To.Row - (Size - 1);
}
break;
case Direction.UpLeft:
if(To.Row < Size - 1) {
To.Column -= Size - 1 - To.Row;
}
break;
default:
Debug.Fail($"{nameof(GetLocation)}: Column: {Direction.ToString()}");
break;
}
return To;
}
protected override void MakeCells(Object Parameter) {
Size = (Int32)Parameter;
Cells = new Cell[2 * Size - 1][];
for(var Row = 0; Row < Size - 1; Row++) {
Cells[Row] = MakeCells(Size + Row);
Cells[2 * Size - 1 - Row - 1] = MakeCells(Size + Row);
}
Cells[Size - 1] = MakeCells(2 * Size - 1);
}
public HoneycombBoard(Int32 Size, String String): base(Size, String) {
}
}
private class Condition {
public readonly String RegularExpression;
public readonly Int32 NumberOfCells;
public readonly Board.Location Start;
public readonly Board.Direction Direction;
public Condition(
String RegularExpression, Int32 NumberOfCells, Int32 Row, Int32 Column,
Board.Direction Direction
) {
this.RegularExpression = RegularExpression;
this.NumberOfCells = NumberOfCells;
Start = new Board.Location(Row, Column);
this.Direction = Direction;
}
}
static void Main(string[] args) {
var StartTime = DateTime.Now;
var Board = new HoneycombBoard(5, "ABCDEFGHMNORXYZ");
var Conditions = new Condition[]{
new Condition(@"(X|Y|Z).*\1.*\1", 5, 0, 0, Program.Board.Direction.Right),
new Condition(@"[^XYZ]*[XYZ]*", 6, 1, 0, Program.Board.Direction.Right),
new Condition(@"(.).*(.).*\2.*\1", 7, 2, 0, Program.Board.Direction.Right),
new Condition(@"[^X]*X+[^X]*", 8, 3, 0, Program.Board.Direction.Right),
new Condition(@"([^X]|XXY)*", 9, 4, 0, Program.Board.Direction.Right),
new Condition(@"[ABCD]*YO[EFGH]*", 8, 5, 0, Program.Board.Direction.Right),
new Condition(@"[^RX](R|XX)*", 7, 6, 0, Program.Board.Direction.Right),
new Condition(@"(...?)\1*", 6, 7, 0, Program.Board.Direction.Right),
new Condition(@".*Y.*Z+", 5, 8, 0, Program.Board.Direction.Right),
new Condition(@"[^A]B[^C]D[^E]", 5, 0, 0, Program.Board.Direction.DownLeft),
new Condition(@"[^DB]*DB[^DB]*", 6, 0, 1, Program.Board.Direction.DownLeft),
new Condition(@"[BMX]*", 7, 0, 2, Program.Board.Direction.DownLeft),
new Condition(@"(.)(.).*\1.*\2.*\2\1", 8, 0, 3, Program.Board.Direction.DownLeft),
new Condition(@"(XX|YY|ZZ)*[XYZ]{2}.", 9, 0, 4, Program.Board.Direction.DownLeft),
new Condition(@"Y.*N.*", 8, 1, 5, Program.Board.Direction.DownLeft),
new Condition(@"X*F.*G.*H.*", 7, 2, 6, Program.Board.Direction.DownLeft),
new Condition(@"(HY|RM|FX|EH)*", 6, 3, 7, Program.Board.Direction.DownLeft),
new Condition(@".*RZ.*", 5, 4, 8, Program.Board.Direction.DownLeft),
new Condition(@"[^ABE]*BE[^ABE]*Z.*", 5, 8, 0, Program.Board.Direction.UpLeft),
new Condition(@".*[BD]+", 6, 8, 1, Program.Board.Direction.UpLeft),
new Condition(@"[^A]*A[^A]*", 7, 8, 2, Program.Board.Direction.UpLeft),
new Condition(@".*MM.*", 8, 8, 3, Program.Board.Direction.UpLeft),
new Condition(@"[^XO]*XO[^XO]*OX[^XO]*(.)\1", 9, 8, 4, Program.Board.Direction.UpLeft),
new Condition(@"(.+).*\1", 8, 7, 5, Program.Board.Direction.UpLeft),
new Condition(@".*X.X.X.*", 7, 6, 6, Program.Board.Direction.UpLeft),
new Condition(@"H+(..)\1.*", 6, 5, 7, Program.Board.Direction.UpLeft),
new Condition(@".*(MR|CH|RF)[^X]*", 5, 4, 8, Program.Board.Direction.UpLeft)
};
Array.Sort(
Conditions,
(X, Y) => { return X.NumberOfCells.CompareTo(Y.NumberOfCells); }
);
var PreviouseCount = 1L;
for(;;) {
var Count = Board.Reduce(Conditions);
Console.WriteLine($"---- {PreviouseCount:###,###,##0} ⇒ {Count:###,###,##0} --------------------");
if(PreviouseCount == Count){
break;
}
PreviouseCount = Count;
}
foreach(var Cells in Board.Resolve(Conditions)) {
Console.WriteLine("---------------------------------------");
for(var Row = 0; Row < Cells.Length; Row++) {
Console.WriteLine($"{Row:D}: {String.Join("", Cells[Row])}");
}
}
Console.WriteLine("---------------------------------------");
Console.WriteLine($"Time: {(DateTime.Now - StartTime).TotalSeconds:F}");
}
}
}
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text.RegularExpressions;
using System.Diagnostics;

namespace ConsoleApp1 {
	class Program {
		private abstract class Board {
			public enum Direction {
				Left,
				Right,
				Up,
				Down,
				DownLeft,
				UpLeft
			}
			protected String String;
			public class Cell {
				public HashSet<Char> Set;
				public Char Current;
				public Cell(String String) {
					Set = new HashSet<Char>(String.ToArray());
					Current = '\0';
				}
			}
			public class Location {
				public Int32 Row;
				public Int32 Column;
				public Location(Int32 Row, Int32 Column) {
					this.Row = Row;
					this.Column = Column;
				}
				public Location(Location From) {
					Row = From.Row;
					Column = From.Column;
				}
			}
			protected Cell[][] Cells;
			public abstract Location GetLocation(Direction Direction, Int32 Distance, Location Locaton);
			protected abstract void MakeCells(Object Parameter);
			public Board(Object Parameter, String String) {
				this.String = String;
				MakeCells(Parameter);
				var Characters = String.ToArray();
				foreach(var Row in Cells) {
					foreach(var Cell in Row) {
						Cell.Set = new HashSet<Char>(Characters);
					}
				}
			}
			protected Cell[] MakeCells(Int32 Size) {
				var Cells = new Cell[Size];
				for(var Index = 0; Index < Cells.Length; Index++) {
					Cells[Index] = new Cell(String);
				}
				return Cells;
			}
			public IEnumerable<string> Strings(Char [] Characters, Condition Condition, int Distance) {
				if(Distance < Condition.NumberOfCells) {
					foreach(var Character in this[GetLocation(Condition.Direction, Distance, Condition.Start)].Set) {
						Characters[Distance] = Character;
						foreach(var String in Strings(Characters, Condition, Distance + 1)){
							yield return String;
						}
					}
				} else {
					yield return string.Join("", Characters);
				}
			}
			private IEnumerable<String> Strings(Condition Condition) {
				var Characters = new Char[Condition.NumberOfCells];
				for(var Index = 0; Index < Characters.Length; Index++) {
					Characters[Index] = '\0';
				}
				foreach(var String in Strings(Characters, Condition, 0)) {
					yield return String;
				}
			}
			public Cell this[Location Location]{
				get {
					return Cells[Location.Row][Location.Column];
				}
			}
			public long Reduce(Condition[] Conditions) {
				var Count = 1L;
				foreach(var Condition in Conditions){
					var Characters = new Char[Condition.NumberOfCells];
					var Sets = new HashSet<Char>[Condition.NumberOfCells];
					for(var Index = 0; Index < Characters.Length; Index++) {
						Characters[Index] = '\0';
						Sets[Index] = new HashSet<char>();
					}
					var RegularExpression = new Regex($"^{Condition.RegularExpression}$", RegexOptions.Compiled);
					foreach(var String in Strings(Characters, Condition, 0)) {
						if(RegularExpression.IsMatch(String)){
							for(var Index = 0; Index < Characters.Length; Index++) {
								Sets[Index].Add(String[Index]);
							}
						}
					}
					long Count1 = 1;
					for(var Index = 0; Index < Sets.Length; Index++) {
						this[GetLocation(Condition.Direction, Index, Condition.Start)].Set = Sets[Index];
						Count1 *= Sets[Index].Count;
					}
					Console.WriteLine($"{Condition.NumberOfCells:D}: {Count1:###,###,##0}: {Condition.RegularExpression}");
					Count *= Count1;
				}
				return Count;
			}
			private IEnumerable<Char[][]> Resolve(Condition[] Conditions, Int32 ConditionIndex) {
				if(ConditionIndex < Conditions.Length) {
					var Condition = Conditions[ConditionIndex];
					var RegularExpression = new Regex($"^{Condition.RegularExpression}$", RegexOptions.Compiled);
					foreach(var String in Strings(Condition)) {
						if(RegularExpression.IsMatch(String)){
							var SaveSets = new HashSet<Char>[Condition.NumberOfCells];
							for(var Index = 0; Index < SaveSets.Length; Index++) {
								var Cell = this[GetLocation(Condition.Direction, Index, Condition.Start)];
								SaveSets[Index] = Cell.Set;
								Cell.Set = new HashSet<char>();
								Cell.Set.Add(String[Index]);
							}
							foreach(var Result in Resolve(Conditions, ConditionIndex + 1)) {
								yield return Result;
							}
							for(var Index = 0; Index < SaveSets.Length; Index++) {
								var Cell = this[GetLocation(Condition.Direction, Index, Condition.Start)];
								Cell.Set = SaveSets[Index];
							}
						}
					}
				} else {
					var Cells = new Char[this.Cells.Length][];
					for(var Row = 0; Row < this.Cells.Length; Row++) {
						Cells[Row] = new Char[this.Cells[Row].Length];
						for(var Column = 0; Column < this.Cells[Row].Length; Column++) {
							Cells[Row][Column] = this.Cells[Row][Column].Set.Single();
						}
					}
					yield return Cells;
				}
			}
			public IEnumerable<Char[][]> Resolve(Condition[] Conditions) {
				foreach(var Result in Resolve(Conditions, 0)){
					yield return Result;
				}
			}
		}
		private class HoneycombBoard: Board {
			private Int32 Size;
			public override Location GetLocation(Direction Direction, int Distance, Location From) {
				var To = new Location(From);
				switch(Direction) {
				case Direction.Right:
					break;
				case Direction.DownLeft:
					To.Row += Distance;
					break;
				case Direction.UpLeft:
					To.Row -= Distance;
					break;
				default:
					Debug.Fail($"{nameof(GetLocation)}: Row: {Direction.ToString()}");
					break;
				}
				switch(Direction) {
				case Direction.Right:
					To.Column += Distance;
					break;
				case Direction.DownLeft:
					if(Size - 1 < To.Row) {
						To.Column -= To.Row - (Size - 1);
					}
					break;
				case Direction.UpLeft:
					if(To.Row < Size - 1) {
						To.Column -= Size - 1 - To.Row;
					}
					break;
				default:
					Debug.Fail($"{nameof(GetLocation)}: Column: {Direction.ToString()}");
					break;
				}
				return To;
			}
			protected override void MakeCells(Object Parameter) {
				Size = (Int32)Parameter;
				Cells = new Cell[2 * Size - 1][];
				for(var Row = 0; Row < Size - 1; Row++) {
					Cells[Row] = MakeCells(Size + Row);
					Cells[2 * Size - 1 - Row - 1] = MakeCells(Size + Row);
				}
				Cells[Size - 1] = MakeCells(2 * Size - 1);
			}
			public HoneycombBoard(Int32 Size, String String): base(Size, String) {
			}
		}
		private class Condition {
			public readonly String RegularExpression;
			public readonly Int32 NumberOfCells;
			public readonly Board.Location Start;
			public readonly Board.Direction Direction;
			public Condition(
				String RegularExpression, Int32 NumberOfCells, Int32 Row, Int32 Column,
				Board.Direction Direction
			) {
				this.RegularExpression = RegularExpression;
				this.NumberOfCells = NumberOfCells;
				Start = new Board.Location(Row, Column);
				this.Direction = Direction;
			}
		}
		static void Main(string[] args) {
			var StartTime = DateTime.Now;
			var Board = new HoneycombBoard(5, "ABCDEFGHMNORXYZ");
			var Conditions = new Condition[]{
				new Condition(@"(X|Y|Z).*\1.*\1", 5, 0, 0, Program.Board.Direction.Right),
				new Condition(@"[^XYZ]*[XYZ]*", 6, 1, 0, Program.Board.Direction.Right),
				new Condition(@"(.).*(.).*\2.*\1", 7, 2, 0, Program.Board.Direction.Right),
				new Condition(@"[^X]*X+[^X]*", 8, 3, 0, Program.Board.Direction.Right),
				new Condition(@"([^X]|XXY)*", 9, 4, 0, Program.Board.Direction.Right),
				new Condition(@"[ABCD]*YO[EFGH]*", 8, 5, 0, Program.Board.Direction.Right),
				new Condition(@"[^RX](R|XX)*", 7, 6, 0, Program.Board.Direction.Right),
				new Condition(@"(...?)\1*", 6, 7, 0, Program.Board.Direction.Right),
				new Condition(@".*Y.*Z+", 5, 8, 0, Program.Board.Direction.Right),
				new Condition(@"[^A]B[^C]D[^E]", 5, 0, 0, Program.Board.Direction.DownLeft),
				new Condition(@"[^DB]*DB[^DB]*", 6, 0, 1, Program.Board.Direction.DownLeft),
				new Condition(@"[BMX]*", 7, 0, 2, Program.Board.Direction.DownLeft),
				new Condition(@"(.)(.).*\1.*\2.*\2\1", 8, 0, 3, Program.Board.Direction.DownLeft),
				new Condition(@"(XX|YY|ZZ)*[XYZ]{2}.", 9, 0, 4, Program.Board.Direction.DownLeft),
				new Condition(@"Y.*N.*", 8, 1, 5, Program.Board.Direction.DownLeft),
				new Condition(@"X*F.*G.*H.*", 7, 2, 6, Program.Board.Direction.DownLeft),
				new Condition(@"(HY|RM|FX|EH)*", 6, 3, 7, Program.Board.Direction.DownLeft),
				new Condition(@".*RZ.*", 5, 4, 8, Program.Board.Direction.DownLeft),
				new Condition(@"[^ABE]*BE[^ABE]*Z.*", 5, 8, 0, Program.Board.Direction.UpLeft),
				new Condition(@".*[BD]+", 6, 8, 1, Program.Board.Direction.UpLeft),
				new Condition(@"[^A]*A[^A]*", 7, 8, 2, Program.Board.Direction.UpLeft),
				new Condition(@".*MM.*", 8, 8, 3, Program.Board.Direction.UpLeft),
				new Condition(@"[^XO]*XO[^XO]*OX[^XO]*(.)\1", 9, 8, 4, Program.Board.Direction.UpLeft),
				new Condition(@"(.+).*\1", 8, 7, 5, Program.Board.Direction.UpLeft),
				new Condition(@".*X.X.X.*", 7, 6, 6, Program.Board.Direction.UpLeft),
				new Condition(@"H+(..)\1.*", 6, 5, 7, Program.Board.Direction.UpLeft),
				new Condition(@".*(MR|CH|RF)[^X]*", 5, 4, 8, Program.Board.Direction.UpLeft)
			};
			Array.Sort(
				Conditions,
				(X, Y) => { return X.NumberOfCells.CompareTo(Y.NumberOfCells); }
			);
			var PreviouseCount = 1L;
			for(;;) {
				var Count = Board.Reduce(Conditions);
				Console.WriteLine($"---- {PreviouseCount:###,###,##0} ⇒ {Count:###,###,##0} --------------------");
				if(PreviouseCount == Count){
					break;
				}
				PreviouseCount = Count;
			}
			foreach(var Cells in Board.Resolve(Conditions)) {
				Console.WriteLine("---------------------------------------");
				for(var Row = 0; Row < Cells.Length; Row++) {
					Console.WriteLine($"{Row:D}: {String.Join("", Cells[Row])}");
				}
			}
			Console.WriteLine("---------------------------------------");
			Console.WriteLine($"Time: {(DateTime.Now - StartTime).TotalSeconds:F}");
		}
	}
}