inz-00/Assets/Voronoi/VoronoiGenerator.cs
2019-07-27 22:40:42 +02:00

196 lines
5.6 KiB
C#

using System.Collections.Generic;
using System.Linq;
using Priority_Queue;
namespace Assets.Voronoi
{
public interface IEvent
{
double Priority { get; }
bool IsValid { get; }
}
public class SiteEvent : IEvent
{
public Site Site { get; internal set; }
public double Priority => Site.Point.y;
public bool IsValid => true;
}
public class EdgeEvent : IEvent
{
public RedBlackNode<Parabola> node;
public Point vertex;
public bool IsValid { get; set; } = true;
public double Priority => vertex.y + Point.Dist(vertex, node.Value.Site.Point);
}
public class Site
{
public ISet<Site> Neighbours { get; internal set; } = new HashSet<Site>();
public Point Point { get; internal set; }
public Site(Point point)
{
Point = point;
}
}
public class VoronoiGenerator
{
public IList<Site> Sites { get; }
public IPriorityQueue<IEvent, double> Queue { get; internal set; }
public IList<Point> Vertices { get; internal set; }
public IList<Edge> Edges { get; internal set; }
public BeachLine Line { get; private set; }
public bool Done => Queue.Count == 0;
public VoronoiGenerator(IList<Point> sites)
{
Sites = sites.Select(x => new Site(x)).ToList();
Reset();
}
private void Reset()
{
Queue = new SimplePriorityQueue<IEvent, double>();
Line = new BeachLine();
Vertices = new List<Point>();
Edges = new List<Edge>();
foreach (var site in Sites)
{
var @event = new SiteEvent { Site = site };
Queue.Enqueue(@event, @event.Priority);
}
}
public void Step()
{
while (Queue.Count > 0)
{
var ev = Queue.Dequeue();
if (!ev.IsValid) continue;
Line.Directrix = ev.Priority;
switch (ev)
{
case SiteEvent site:
HandleSiteEvent(site);
return;
case EdgeEvent edge:
HandleEdgeEvent(edge);
return;
}
}
}
public void Generate()
{
Reset();
while (Queue.Count > 0)
Step();
}
private void Enqueue(EdgeEvent @event)
{
if (@event.Priority > Line.Directrix)
Queue.Enqueue(@event, @event.Priority);
}
private void HandleEdgeEvent(EdgeEvent @event)
{
var queue = new Queue<RedBlackNode<Parabola>>();
var node = @event.node;
var parabola = node.Value;
var previous = node.Previous;
var next = node.Next;
if (previous?.Value.Event != null) previous.Value.Event.IsValid = false;
if (next?.Value.Event != null) next.Value.Event.IsValid = false;
Vertices.Add(@event.vertex);
node.Value.LeftEdge.end = @event.vertex;
node.Value.RightEdge.end = @event.vertex;
Edges.Add(node.Value.LeftEdge);
Edges.Add(node.Value.RightEdge);
var newEdge = new Edge() { start = @event.vertex };
node.Previous.Value.RightEdge = newEdge;
node.Next.Value.LeftEdge = newEdge;
Line.RemoveParabola(node);
if (Line.CheckCircleEvent(previous) is EdgeEvent p)
Enqueue(p);
if (Line.CheckCircleEvent(next) is EdgeEvent n)
Enqueue(n);
if (previous != null && next != null)
{
previous.Value.Site.Neighbours.Add(next.Value.Site);
next.Value.Site.Neighbours.Add(previous.Value.Site);
}
}
private void HandleSiteEvent(SiteEvent @event)
{
var site = @event.Site;
var start = new Point(site.Point.x, Line.Eval(site.Point.x));
var above = Line.FindParabola(site.Point.x);
if (above != null)
{
var left = above.Value.LeftEdge;
var right = above.Value.RightEdge;
var node = Line.AddParabola(@event.Site, above);
var newLeft = new Edge() { start = start };
var newRight = new Edge() { start = start };
node.Previous.Value.LeftEdge = left;
node.Previous.Value.RightEdge = newLeft;
node.Value.LeftEdge = newLeft;
node.Value.RightEdge = newRight;
node.Next.Value.LeftEdge = newRight;
node.Next.Value.RightEdge = right;
if (Line.CheckCircleEvent(node.Previous) is EdgeEvent p)
Enqueue(p);
if (Line.CheckCircleEvent(node.Next) is EdgeEvent n)
Enqueue(n);
if (node.Previous != null)
{
@event.Site.Neighbours.Add(node.Previous.Value.Site);
node.Previous.Value.Site.Neighbours.Add(@event.Site);
}
}
else
{
Line.AddParabola(@event.Site);
}
}
}
}