268 lines
		
	
	
		
			7.4 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
			
		
		
	
	
			268 lines
		
	
	
		
			7.4 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
| <Query Kind="Program">
 | |
|   <Namespace>System.Drawing</Namespace>
 | |
| </Query>
 | |
| 
 | |
| class Step { public int X,Y; public char Dir; public int P; public List<Step> NextSteps = new List<Step>(); }
 | |
| 
 | |
| int StartX;
 | |
| int StartY;
 | |
| int Width;
 | |
| int Height;
 | |
| bool[,] Doors;
 | |
| bool[,] Reachable;
 | |
| int?[,] Distance;
 | |
| Step RootStep;
 | |
| 
 | |
| void Main()
 | |
| {
 | |
| 	Load(File.ReadAllText(Path.Combine(Path.GetDirectoryName(Util.CurrentQueryPath), @"20_input.txt")).Trim());
 | |
| 	//DumpSteps();
 | |
| 	//int c=0; VerifySteps(RootStep, ref c);c.Dump();
 | |
| 	Walk();
 | |
| 	Flood().Dump();
 | |
| 	DumpMap();
 | |
| }
 | |
| 
 | |
| void VerifySteps(Step s, ref int c)
 | |
| {
 | |
| 	for(;;)
 | |
| 	{
 | |
| 		c++;
 | |
| 		if (s.NextSteps.Any(ns => ns.P>=s.P)) throw new Exception();
 | |
| 		if (s.NextSteps.Count == 1) { s = s.NextSteps.Single(); continue; }
 | |
| 
 | |
| 		foreach (var ns in s.NextSteps) VerifySteps(ns, ref c);
 | |
| 		return;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void DumpSteps()
 | |
| {
 | |
| 	StringBuilder b = new StringBuilder();
 | |
| 	DumpSteps(RootStep, ref b, 0);
 | |
| }
 | |
| 
 | |
| void DumpSteps(Step s, ref StringBuilder b, int w)
 | |
| {
 | |
| 	int sscc = 0;
 | |
| 	for (; ; )
 | |
| 	{
 | |
| 		if (s.NextSteps.Count == 0) { b.Append(sscc.ToString()); w += sscc.ToString().Length; break; }
 | |
| 		if (s.NextSteps.Count == 1) { sscc++; s = s.NextSteps.Single(); continue; }
 | |
| 
 | |
| 		{b.Append(sscc.ToString()); w += sscc.ToString().Length;}
 | |
| 		int l = w;
 | |
| 		foreach (var ns in s.NextSteps)
 | |
| 		{
 | |
| 			b.ToString().Dump();b.Clear();
 | |
| 			b.Append(new string(' ', l));
 | |
| 			DumpSteps(ns, ref b, w);
 | |
| 		}
 | |
| 		return;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void Load(string d)
 | |
| {
 | |
| 	d = d.Substring(1, d.Length - 2);
 | |
| 
 | |
| 	StartX    = d.Count(c => c == 'W') + 16;
 | |
| 	StartY    = d.Count(c => c == 'N') + 16;
 | |
| 	Width     = d.Count(c => c == 'E') + 16 + StartX;
 | |
| 	Height    = d.Count(c => c == 'S') + 16 + StartY;
 | |
| 	Doors     = new bool[Width * 2 + 16, Height + 16];
 | |
| 	Distance  = new int?[Width, Height];
 | |
| 	Reachable = new bool[Width, Height];
 | |
| 
 | |
| 	RootStep = Parse(d);
 | |
| }
 | |
| 
 | |
| Step Parse(string d)
 | |
| {
 | |
| 	Step r = null;
 | |
| 	Parse(new List<Step>(), ref d, ref r);
 | |
| 	if (d!="")throw new Exception();
 | |
| 	return r;
 | |
| }
 | |
| 
 | |
| List<Step> Parse(List<Step> source, ref string d, ref Step root)
 | |
| {
 | |
| 	List<Step> lasts = source;
 | |
| 	for (; ; )
 | |
| 	{
 | |
| 		if (d    == "") return lasts;
 | |
| 		if (d[0] == '|') return lasts;
 | |
| 		if (d[0] == ')') return lasts;
 | |
| 		if (d[0] == '(')
 | |
| 		{
 | |
| 			d = d.Substring(1);
 | |
| 			List<Step> newlasts = new List<Step>();
 | |
| 			for (;;)
 | |
| 			{
 | |
| 				if (d[0] == ')') { d = d.Substring(1); newlasts.AddRange(lasts); break; }
 | |
| 				
 | |
| 				var n = Parse(lasts, ref d, ref root);
 | |
| 				newlasts.AddRange(n);
 | |
| 
 | |
| 				if (d[0] == '|') { d = d.Substring(1); continue; }
 | |
| 				if (d[0] == ')') { d = d.Substring(1); break;    }
 | |
| 				
 | |
| 				throw new Exception();
 | |
| 			}
 | |
| 			lasts = newlasts;
 | |
| 			continue;
 | |
| 		}
 | |
| 
 | |
| 		if (d[0] == 'N') { var s = new Step { X = 00, Y = -1, Dir=d[0], P=d.Length, }; lasts.ForEach(l => l.NextSteps.Add(s)); lasts = new List<Step> {s}; d = d.Substring(1); root = root ?? s; continue; }
 | |
| 		if (d[0] == 'E') { var s = new Step { X = +1, Y = 00, Dir=d[0], P=d.Length, }; lasts.ForEach(l => l.NextSteps.Add(s)); lasts = new List<Step> {s}; d = d.Substring(1); root = root ?? s; continue; }
 | |
| 		if (d[0] == 'S') { var s = new Step { X = 00, Y = +1, Dir=d[0], P=d.Length, }; lasts.ForEach(l => l.NextSteps.Add(s)); lasts = new List<Step> {s}; d = d.Substring(1); root = root ?? s; continue; }
 | |
| 		if (d[0] == 'W') { var s = new Step { X = -1, Y = 00, Dir=d[0], P=d.Length, }; lasts.ForEach(l => l.NextSteps.Add(s)); lasts = new List<Step> {s}; d = d.Substring(1); root = root ?? s; continue; }
 | |
| 
 | |
| 		throw new Exception();
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void Walk()
 | |
| {
 | |
| 	Stack<(int x,int y, Step s)> work = new Stack<(int x, int y, Step s)>();
 | |
| 	work.Push((StartX, StartY, RootStep));
 | |
| 	
 | |
| 	while(work.Any())
 | |
| 	{
 | |
| 		(var x, var y, var step) = work.Pop();
 | |
| 		Reachable[x,y]=true;
 | |
| 		
 | |
| 		if (step.Dir == 'N') Doors[x*2, y]  = true;
 | |
| 		if (step.Dir == 'E') Doors[x*2+1,y] = true;
 | |
| 		if (step.Dir == 'S') Doors[x*2,y+1] = true;
 | |
| 		if (step.Dir == 'W') Doors[x*2-1,y] = true;
 | |
| 
 | |
| 		x += step.X;
 | |
| 		y += step.Y;
 | |
| 		foreach (var ns in step.NextSteps) work.Push((x,y,ns));
 | |
| 		//DumpMap();
 | |
| 	}
 | |
| 
 | |
| }
 | |
| 
 | |
| int Flood()
 | |
| {
 | |
| 	Stack<(int x, int y)> work = new Stack<(int x, int y)>();
 | |
| 	work.Push((StartX, StartY));
 | |
| 	Distance[StartX, StartY]=0;
 | |
| 
 | |
| 	while (work.Any())
 | |
| 	{
 | |
| 		(var x, var y) = work.Pop();
 | |
| 
 | |
| 		if (Doors[x * 2,     y    ] && (Distance[x+0,y-1] == null || Distance[x+0,y-1].Value > Distance[x,y]+1)) { Distance[x+0,y-1] = Distance[x,y]+1; work.Push((x+0,y-1)); }
 | |
| 		if (Doors[x * 2 + 1, y    ] && (Distance[x+1,y+0] == null || Distance[x+1,y+0].Value > Distance[x,y]+1)) { Distance[x+1,y+0] = Distance[x,y]+1; work.Push((x+1,y+0)); }
 | |
| 		if (Doors[x * 2,     y + 1] && (Distance[x+0,y+1] == null || Distance[x+0,y+1].Value > Distance[x,y]+1)) { Distance[x+0,y+1] = Distance[x,y]+1; work.Push((x+0,y+1)); }
 | |
| 		if (Doors[x * 2 - 1, y    ] && (Distance[x-1,y+0] == null || Distance[x-1,y+0].Value > Distance[x,y]+1)) { Distance[x-1,y+0] = Distance[x,y]+1; work.Push((x-1,y+0)); }
 | |
| 	}
 | |
| 	
 | |
| 	int max = int.MinValue;
 | |
| 	for (int xx = 0; xx < Width; xx++) for (int yy = 0; yy < Height; yy++) max = Math.Max(Distance[xx,yy] ?? max, max);
 | |
| 	return max;
 | |
| }
 | |
| 
 | |
| void DumpMap()
 | |
| {
 | |
| 	var ww = Width * 2 + 1;
 | |
| 	var hh = Height * 2 + 1;
 | |
| 	char[,] m = new char[Width * 2 + 1, Height * 2 + 1];
 | |
| 
 | |
| 	int _sx = 0;
 | |
| 	int _ex = 0;
 | |
| 	int _sy = 0;
 | |
| 	int _ey = 0;
 | |
| 	for (int xx = 0; ; xx++)
 | |
| 	{
 | |
| 		var ff = false;
 | |
| 		for (int yy = 0; yy < Height; yy++) if (Reachable[xx, yy]) ff = true;
 | |
| 		if (ff) { _sx = xx; break; }
 | |
| 	}
 | |
| 	for (int xx = Width - 1; ; xx--)
 | |
| 	{
 | |
| 		var ff = false;
 | |
| 		for (int yy = 0; yy < Height; yy++) if (Reachable[xx, yy]) ff = true;
 | |
| 		if (ff) { _ex = xx; break; }
 | |
| 	}
 | |
| 	for (int yy = 0; ; yy++)
 | |
| 	{
 | |
| 		var ff = false;
 | |
| 		for (int xx = 0; xx < Width; xx++) if (Reachable[xx, yy]) ff = true;
 | |
| 		if (ff) { _sy = yy; break; }
 | |
| 	}
 | |
| 	for (int yy = Height - 1; ; yy--)
 | |
| 	{
 | |
| 		var ff = false;
 | |
| 		for (int xx = 0; xx < Width; xx++) if (Reachable[xx, yy]) ff = true;
 | |
| 		if (ff) { _ey = yy; break; }
 | |
| 	}
 | |
| 
 | |
| 	var sy = _sy * 2 + 1 - 1;
 | |
| 	var sx = _sx * 2 + 1 - 1;
 | |
| 	var ey = _ey * 2 + 1 + 2;
 | |
| 	var ex = _ex * 2 + 1 + 2;
 | |
| 	
 | |
| 
 | |
| 	for (int xx = sx; xx <= ex; xx++) for (int yy = sy; yy <= ey; yy++) m[xx, yy] = '?';
 | |
| 
 | |
| 	for (int xx = sx; xx <= ex; xx += 2) for (int yy = sy; yy <= ey; yy++) m[xx, yy] = '#';
 | |
| 	for (int yy = sy; yy <= ey; yy += 2) for (int xx = sx; xx <= ex; xx++) m[xx, yy] = '#';
 | |
| 	
 | |
| 	for (int ix = _sx; ix <= _ex; ix++)
 | |
| 	{
 | |
| 		for (int iy = _sy; iy <= _ey; iy++)
 | |
| 		{
 | |
| 			if (Doors[ix * 2,     iy    ]) { m[ix*2+1, iy*2+1-1]='-'; }
 | |
| 			if (Doors[ix * 2 + 1, iy    ]) { m[ix*2+1+1, iy*2+1]='|'; }
 | |
| 			if (Doors[ix * 2,     iy + 1]) { m[ix*2+1, iy*2+1+1]='-'; }
 | |
| 			if (Doors[ix * 2 - 1, iy    ]) { m[ix*2+1-1, iy*2+1]='|'; }
 | |
| 			
 | |
| 			if (Distance[ix,iy]!= null) { m[ix*2+1, iy*2+1]='.'; }
 | |
| 		}
 | |
| 	}
 | |
| 	
 | |
| 
 | |
| 	m[StartX * 2 + 1, StartY * 2 + 1] = 'X';
 | |
| 
 | |
| 	int f = 6;
 | |
| 	Bitmap bmp = new Bitmap(((ex-sx)+(ex-sx)/2) * f, ((ey-sy)+(ey-sy)/2) * f, System.Drawing.Imaging.PixelFormat.Format24bppRgb);
 | |
| 	using (Graphics g = Graphics.FromImage(bmp))
 | |
| 	{
 | |
| 		g.Clear(Color.Magenta);
 | |
| 		for (int yy = sy; yy < ey; yy++)
 | |
| 		{
 | |
| 			for (int xx = sx; xx < ex; xx++)
 | |
| 			{
 | |
| 				int rx = (xx - sx) + (xx - sx) / 2;
 | |
| 				int ry = (yy - sy) + (yy - sy) / 2;
 | |
| 				int rw = ((xx - sx) % 2) + 1;
 | |
| 				int rh = ((yy - sy) % 2) + 1;
 | |
| 
 | |
| 				var c = Brushes.Magenta;
 | |
| 				if (m[xx, yy] == '#') c = Brushes.Black;
 | |
| 				if (m[xx, yy] == 'X') c = Brushes.Green;
 | |
| 				if (m[xx, yy] == '.') c = Brushes.White;
 | |
| 				if (m[xx, yy] == '?') c = Brushes.DarkGray;
 | |
| 				if (m[xx, yy] == '|') c = Brushes.LightGray;
 | |
| 				if (m[xx, yy] == '-') c = Brushes.LightGray;
 | |
| 				g.FillRectangle(c, rx*f, ry*f, rw*f, rh*f);
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 	bmp.Dump();
 | |
| 
 | |
| 	//StringBuilder b = new StringBuilder();
 | |
| 	//for (int yy = 0; yy < hh; yy++)
 | |
| 	//{
 | |
| 	//	for (int xx = 0; xx < ww; xx++)
 | |
| 	//	{
 | |
| 	//		b.Append(m[xx,yy]);
 | |
| 	//	}
 | |
| 	//	b.AppendLine();
 | |
| 	//}
 | |
| 	//b.ToString().Dump();
 | |
| } |