added fixing city roads density
This commit is contained in:
parent
ceb20fcba1
commit
c33de4afa2
@ -7,6 +7,7 @@ namespace Assets.Common
|
|||||||
{
|
{
|
||||||
public double x;
|
public double x;
|
||||||
public double y;
|
public double y;
|
||||||
|
public double Length => Math.Sqrt(x*x+y*y);
|
||||||
|
|
||||||
public Point(double x, double y)
|
public Point(double x, double y)
|
||||||
{
|
{
|
||||||
@ -27,11 +28,14 @@ namespace Assets.Common
|
|||||||
return hashCode;
|
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, 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) => 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, 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) => 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);
|
||||||
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);
|
||||||
|
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
}
|
@ -1,3 +1,4 @@
|
|||||||
|
using System;
|
||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
|
|
||||||
namespace Assets.Common
|
namespace Assets.Common
|
||||||
@ -20,6 +21,14 @@ namespace Assets.Common
|
|||||||
return result;
|
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)
|
public static bool IsClockwise(Point center, Point a, Point b)
|
||||||
{
|
{
|
||||||
var xa = a.x - center.x;
|
var xa = a.x - center.x;
|
||||||
|
@ -19,5 +19,15 @@ namespace Assets.Common
|
|||||||
var count = collection.Count();
|
var count = collection.Count();
|
||||||
return collection.ElementAt(generator.Next(count));
|
return collection.ElementAt(generator.Next(count));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public static IEnumerable<T> RotateRight<T>(this IEnumerable<T> 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);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
@ -46,7 +46,7 @@ namespace Assets.Scripts
|
|||||||
public List<CityField> fields = new List<CityField>();
|
public List<CityField> fields = new List<CityField>();
|
||||||
|
|
||||||
public List<(int, int)> edges;
|
public List<(int, int)> edges;
|
||||||
|
|
||||||
public City(Graph<Site> startGraph)
|
public City(Graph<Site> startGraph)
|
||||||
{
|
{
|
||||||
this.startGraph = startGraph;
|
this.startGraph = startGraph;
|
||||||
@ -69,7 +69,7 @@ namespace Assets.Scripts
|
|||||||
private void FixBoundaryEdges(Site site)
|
private void FixBoundaryEdges(Site site)
|
||||||
{
|
{
|
||||||
var a = edges;
|
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();
|
edges = a.Union(b).Except(a.Intersect(b)).ToList();
|
||||||
}
|
}
|
||||||
@ -77,7 +77,7 @@ namespace Assets.Scripts
|
|||||||
public void CreateRoadGraph(IList<Point> points)
|
public void CreateRoadGraph(IList<Point> points)
|
||||||
{
|
{
|
||||||
var pointsForRoads = sites.SelectMany(site => site.Vertices.Select(i => (i, points[i])).Append((-1, site.Point))).Distinct().ToList();
|
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));
|
VoronoiGenerator generator = new VoronoiGenerator(pointsForRoads.Select(x => x.Item2));
|
||||||
generator.Generate();
|
generator.Generate();
|
||||||
@ -109,8 +109,12 @@ namespace Assets.Scripts
|
|||||||
private Graph<Site> _basicGraph;
|
private Graph<Site> _basicGraph;
|
||||||
private List<int> verticesForRoads;
|
private List<int> verticesForRoads;
|
||||||
private List<int> verticesForCitites;
|
private List<int> verticesForCitites;
|
||||||
|
|
||||||
public GraphGeneratorDebug debug;
|
public GraphGeneratorDebug debug;
|
||||||
|
public double minimumAngle = 0.25;
|
||||||
|
|
||||||
public List<City> cities = new List<City>();
|
public List<City> cities = new List<City>();
|
||||||
|
|
||||||
Graph<Site> roads = new Graph<Site>();
|
Graph<Site> roads = new Graph<Site>();
|
||||||
|
|
||||||
Graph<Point> mainRoads = new Graph<Point>();
|
Graph<Point> mainRoads = new Graph<Point>();
|
||||||
@ -168,6 +172,7 @@ namespace Assets.Scripts
|
|||||||
}
|
}
|
||||||
|
|
||||||
ConnectPointsIntoRoads(newCity);
|
ConnectPointsIntoRoads(newCity);
|
||||||
|
FixRoadsDensity(newCity, minimumAngle);
|
||||||
CreateFieldBoundaries(newCity);
|
CreateFieldBoundaries(newCity);
|
||||||
|
|
||||||
return newCity;
|
return newCity;
|
||||||
@ -216,8 +221,64 @@ namespace Assets.Scripts
|
|||||||
{
|
{
|
||||||
foreach (var (a, b) in city.roads.Edges)
|
foreach (var (a, b) in city.roads.Edges)
|
||||||
city.fieldBoundaries.AddEdge(a, b);
|
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()
|
void CreateWorldRoadGraph()
|
||||||
{
|
{
|
||||||
roads.Vertices = verticesForCitites.Select(i => _basicGraph.Vertices[i]).ToList();
|
roads.Vertices = verticesForCitites.Select(i => _basicGraph.Vertices[i]).ToList();
|
||||||
|
@ -1,2 +1,2 @@
|
|||||||
m_EditorVersion: 2019.3.0a8
|
m_EditorVersion: 2019.2.0f1
|
||||||
m_EditorVersionWithRevision: 2019.3.0a8 (8ea4afdbfa47)
|
m_EditorVersionWithRevision: 2019.2.0f1 (20c1667945cf)
|
||||||
|
Loading…
Reference in New Issue
Block a user