diff --git a/Assets/Common/Point.cs b/Assets/Common/Point.cs index 6892def..b25a061 100644 --- a/Assets/Common/Point.cs +++ b/Assets/Common/Point.cs @@ -7,6 +7,7 @@ namespace Assets.Common { public double x; public double y; + public double Length => Math.Sqrt(x*x+y*y); public Point(double x, double y) { @@ -27,11 +28,14 @@ namespace Assets.Common return hashCode; } + public static Point operator +(Point a, Point b) => new Point(a.x + b.x, a.y + b.y); public static Point operator +(Point a) => a; public static Point operator -(Point a, Point b) => new Point(a.x - b.x, a.y - b.y); public static Point operator -(Point a) => new Point(-a.x, -a.y); public static Point operator *(Point a, double b) => new Point(a.x * b, a.y * b); public static Point operator /(Point a, double b) => new Point(a.x / b, a.y / b); + + } } \ No newline at end of file diff --git a/Assets/Common/PointUtils.cs b/Assets/Common/PointUtils.cs index b8d5753..162b055 100644 --- a/Assets/Common/PointUtils.cs +++ b/Assets/Common/PointUtils.cs @@ -1,3 +1,4 @@ +using System; using System.Collections.Generic; namespace Assets.Common @@ -20,6 +21,14 @@ namespace Assets.Common return result; } + public static double AngleBetween(Point center, Point a, Point b) + { + a = a - center; + b = b - center; + + return Math.Acos((a.x*b.x+a.y*b.y)/(a.Length * b.Length)); + } + public static bool IsClockwise(Point center, Point a, Point b) { var xa = a.x - center.x; diff --git a/Assets/Common/UnityPointExtensions.cs b/Assets/Common/UnityPointExtensions.cs index 15ae434..aa05833 100644 --- a/Assets/Common/UnityPointExtensions.cs +++ b/Assets/Common/UnityPointExtensions.cs @@ -19,5 +19,15 @@ namespace Assets.Common var count = collection.Count(); return collection.ElementAt(generator.Next(count)); } + + public static IEnumerable RotateRight(this IEnumerable collection, int count = 1) + { + var diff = collection.Count() - count; + + var head = collection.Take(diff); + var tail = collection.Skip(diff).Take(count); + + return tail.Concat(head); + } } } \ No newline at end of file diff --git a/Assets/Scripts/CityGenerator.cs b/Assets/Scripts/CityGenerator.cs index 4b080ba..5d82b77 100644 --- a/Assets/Scripts/CityGenerator.cs +++ b/Assets/Scripts/CityGenerator.cs @@ -46,7 +46,7 @@ namespace Assets.Scripts public List fields = new List(); public List<(int, int)> edges; - + public City(Graph startGraph) { this.startGraph = startGraph; @@ -69,7 +69,7 @@ namespace Assets.Scripts private void FixBoundaryEdges(Site site) { var a = edges; - var b = site.Edges.Select(x => x.Item1 < x.Item2 ? x : (x.Item2, x.Item1)); + var b = site.Edges.Select(x => x.Item1 < x.Item2 ? x : (x.Item2, x.Item1)); edges = a.Union(b).Except(a.Intersect(b)).ToList(); } @@ -77,7 +77,7 @@ namespace Assets.Scripts public void CreateRoadGraph(IList points) { var pointsForRoads = sites.SelectMany(site => site.Vertices.Select(i => (i, points[i])).Append((-1, site.Point))).Distinct().ToList(); - var mapping = pointsForRoads.Select((x, i) => x.Item1 == -1 ? (-1, -1) : (i, x.Item1)).Where(x => x.Item1 != -1).ToDictionary(x => x.Item2, x => x.Item1); + var mapping = pointsForRoads.Select((x, i) => x.Item1 == -1 ? (-1, -1) : (i, x.Item1)).Where(x => x.Item1 != -1).ToDictionary(x => x.Item2, x => x.Item1); VoronoiGenerator generator = new VoronoiGenerator(pointsForRoads.Select(x => x.Item2)); generator.Generate(); @@ -109,8 +109,12 @@ namespace Assets.Scripts private Graph _basicGraph; private List verticesForRoads; private List verticesForCitites; + public GraphGeneratorDebug debug; + public double minimumAngle = 0.25; + public List cities = new List(); + Graph roads = new Graph(); Graph mainRoads = new Graph(); @@ -168,6 +172,7 @@ namespace Assets.Scripts } ConnectPointsIntoRoads(newCity); + FixRoadsDensity(newCity, minimumAngle); CreateFieldBoundaries(newCity); return newCity; @@ -216,8 +221,64 @@ namespace Assets.Scripts { foreach (var (a, b) in city.roads.Edges) city.fieldBoundaries.AddEdge(a, b); + + var deadEnds = city.fieldBoundaries.Vertices.Select((_, i) => i).Where(i => city.fieldBoundaries.Neighbours(i).Count() == 1); + + foreach (var deadEnd in deadEnds) + { + var neighbour = city.fieldBoundaries.Neighbours(deadEnd).First(); + var closest = city.fieldBoundaries.Vertices + .Select((_, i) => i) + .OrderBy(i => Point.Dist(city.fieldBoundaries[i], city.fieldBoundaries[deadEnd])) + .Skip(1) + .First(c => c != neighbour); + + city.fieldBoundaries.AddEdge(deadEnd, closest); + city.roads.AddEdge(deadEnd, closest); + } } + + private void FixRoadsDensity(City city, double angle = 0.25f) + { + // dla każdego punktu w grafie dróg + // chyba że ma 2 to pomiń? + // wyznacz punkt 0 + // oblicz odległości kątowe do wszystkich punktów + // posortuj je + // między wszystkimiu parami oblicz kąt + // potem według jakiegoś parametru usuń te, które są za bliskie + + + for (int i = 0; i < city.roads.Vertices.Count(); i++) + { + var p = city.roads[i]; + if (city.roads.Neighbours(i).Count() <= 2) + continue; + + var reference = new Point(p.x, p.y + 30); + + var orderedNeighours = city.roads.Neighbours(i) + .Select(n => (Vertex: n, Point: city.roads[n])) + .Select(x => (Vertex: x.Vertex, Angle: PointUtils.AngleBetween(p, reference, x.Point))) + .OrderBy(x => x.Angle) + .Select(x => x.Vertex); + + foreach (var (a, b) in orderedNeighours.RotateRight(1).Zip(orderedNeighours, (a, b) => (a, b))) + { + if( PointUtils.AngleBetween(p, city.roads[a], city.roads[b]) < angle) + { + city.roads.DeleteEdge(i, a); + } + } + } + } + + + + + + void CreateWorldRoadGraph() { roads.Vertices = verticesForCitites.Select(i => _basicGraph.Vertices[i]).ToList(); diff --git a/ProjectSettings/ProjectVersion.txt b/ProjectSettings/ProjectVersion.txt index a572b7b..7e64146 100644 --- a/ProjectSettings/ProjectVersion.txt +++ b/ProjectSettings/ProjectVersion.txt @@ -1,2 +1,2 @@ -m_EditorVersion: 2019.3.0a8 -m_EditorVersionWithRevision: 2019.3.0a8 (8ea4afdbfa47) +m_EditorVersion: 2019.2.0f1 +m_EditorVersionWithRevision: 2019.2.0f1 (20c1667945cf)