| | 1 | | using System.Collections.Generic; |
| | 2 | | using System.IO; |
| | 3 | | using System.Linq; |
| | 4 | | using System.Text; |
| | 5 | | using System.Text.Json; |
| | 6 | | using Itinero.Network; |
| | 7 | | using Itinero.Network.Enumerators.Edges; |
| | 8 | | using Itinero.Network.Tiles; |
| | 9 | | using Itinero.Network.Tiles.Standalone; |
| | 10 | |
|
| | 11 | | namespace Itinero.IO.Json.GeoJson; |
| | 12 | |
|
| | 13 | | /// <summary> |
| | 14 | | /// Contains extension methods to serialize standalone network tiles to geojson. |
| | 15 | | /// </summary> |
| | 16 | | public static class StandaloneNetworkTileExtensions |
| | 17 | | { |
| | 18 | | /// <summary> |
| | 19 | | /// Gets a geojson version of the network tile. |
| | 20 | | /// </summary> |
| | 21 | | /// <param name="tile">The tile.</param> |
| | 22 | | /// <returns>A string with geojson.</returns> |
| | 23 | | public static string ToGeoJson(this StandaloneNetworkTile tile) |
| 0 | 24 | | { |
| 0 | 25 | | using var stream = new MemoryStream(); |
| 0 | 26 | | using (var jsonWriter = new Utf8JsonWriter(stream)) |
| 0 | 27 | | { |
| 0 | 28 | | jsonWriter.WriteFeatureCollectionStart(); |
| 0 | 29 | | jsonWriter.WriteFeatures(tile); |
| 0 | 30 | | jsonWriter.WriteFeatureCollectionEnd(); |
| 0 | 31 | | } |
| | 32 | |
|
| 0 | 33 | | return Encoding.UTF8.GetString(stream.ToArray()); |
| 0 | 34 | | } |
| | 35 | |
|
| | 36 | | /// <summary> |
| | 37 | | /// Writes features to the given json writer. |
| | 38 | | /// </summary> |
| | 39 | | /// <param name="tile">The tile.</param> |
| | 40 | | /// <param name="jsonWriter">The json writer.</param> |
| | 41 | | public static void WriteFeatures(this Utf8JsonWriter jsonWriter, StandaloneNetworkTile tile) |
| 0 | 42 | | { |
| 0 | 43 | | var edges = new HashSet<EdgeId>(); |
| | 44 | |
|
| 0 | 45 | | var vertex = new VertexId(tile.TileId, 0); |
| 0 | 46 | | var edgeEnumerator = tile.GetEnumerator(); |
| 0 | 47 | | var secondEdgeEnumerator = tile.GetEnumerator(); |
| 0 | 48 | | while (edgeEnumerator.MoveTo(vertex)) |
| 0 | 49 | | { |
| | 50 | | // write vertex features. |
| 0 | 51 | | jsonWriter.WriteVertexFeature(vertex, edgeEnumerator.TailLocation); |
| | 52 | |
|
| | 53 | | // iterate over all edges. |
| 0 | 54 | | while (edgeEnumerator.MoveNext()) |
| 0 | 55 | | { |
| 0 | 56 | | if (edges.Contains(edgeEnumerator.EdgeId)) continue; |
| 0 | 57 | | edges.Add(edgeEnumerator.EdgeId); |
| | 58 | |
|
| | 59 | | // write edge features. |
| 0 | 60 | | jsonWriter.WriteEdgeFeature(edgeEnumerator); |
| | 61 | |
|
| 0 | 62 | | if (edgeEnumerator.HeadOrder != null) |
| 0 | 63 | | { |
| | 64 | | // check out turn costs at head location. |
| 0 | 65 | | secondEdgeEnumerator.MoveTo(edgeEnumerator.Head); |
| 0 | 66 | | while (secondEdgeEnumerator.MoveNext()) |
| 0 | 67 | | { |
| 0 | 68 | | if (secondEdgeEnumerator.EdgeId == edgeEnumerator.EdgeId) continue; |
| 0 | 69 | | if (secondEdgeEnumerator.TailOrder == null) continue; |
| | 70 | |
|
| 0 | 71 | | foreach (var turnCost in |
| 0 | 72 | | edgeEnumerator.GetTurnCostFromHead(secondEdgeEnumerator.TailOrder.Value)) |
| 0 | 73 | | { |
| 0 | 74 | | if (turnCost.cost == 0) continue; |
| | 75 | |
|
| 0 | 76 | | jsonWriter.WriteFeatureStart(); |
| 0 | 77 | | var attributes = turnCost.attributes.ToList(); |
| 0 | 78 | | attributes.AddRange(new (string key, string value)[] |
| 0 | 79 | | { |
| 0 | 80 | | ("edge1_tile_id", edgeEnumerator.EdgeId.TileId.ToString()), |
| 0 | 81 | | ("edge1_local_id", edgeEnumerator.EdgeId.LocalId.ToString()), |
| 0 | 82 | | ("edge2_tile_id", secondEdgeEnumerator.EdgeId.TileId.ToString()), |
| 0 | 83 | | ("edge2_local_id", secondEdgeEnumerator.EdgeId.LocalId.ToString()), |
| 0 | 84 | | ("edges_prefix", string.Join(",", turnCost.prefixEdges.Select(x => x.ToString()))), |
| 0 | 85 | | ("cost", turnCost.cost.ToString()) |
| 0 | 86 | | }); |
| 0 | 87 | | jsonWriter.WriteProperties(attributes); |
| 0 | 88 | | jsonWriter.WritePropertyName("geometry"); |
| 0 | 89 | | jsonWriter.WriteLineString(edgeEnumerator.GetCompleteShape() |
| 0 | 90 | | .Concat(secondEdgeEnumerator.GetCompleteShape())); |
| 0 | 91 | | jsonWriter.WriteFeatureEnd(); |
| 0 | 92 | | } |
| 0 | 93 | | } |
| 0 | 94 | | } |
| | 95 | |
|
| 0 | 96 | | if (edgeEnumerator.TailOrder != null) |
| 0 | 97 | | { |
| | 98 | | // check out turn costs at tail location. |
| 0 | 99 | | secondEdgeEnumerator.MoveTo(edgeEnumerator.Tail); |
| 0 | 100 | | while (secondEdgeEnumerator.MoveNext()) |
| 0 | 101 | | { |
| 0 | 102 | | if (secondEdgeEnumerator.EdgeId == edgeEnumerator.EdgeId) continue; |
| 0 | 103 | | if (secondEdgeEnumerator.TailOrder == null) continue; |
| | 104 | |
|
| 0 | 105 | | foreach (var turnCost in |
| 0 | 106 | | edgeEnumerator.GetTurnCostFromTail(secondEdgeEnumerator.TailOrder.Value)) |
| 0 | 107 | | { |
| 0 | 108 | | if (turnCost.cost == 0) continue; |
| | 109 | |
|
| 0 | 110 | | jsonWriter.WriteFeatureStart(); |
| 0 | 111 | | var attributes = turnCost.attributes.ToList(); |
| 0 | 112 | | attributes.AddRange(new (string key, string value)[] |
| 0 | 113 | | { |
| 0 | 114 | | ("edge1_tile_id", edgeEnumerator.EdgeId.TileId.ToString()), |
| 0 | 115 | | ("edge1_local_id", edgeEnumerator.EdgeId.LocalId.ToString()), |
| 0 | 116 | | ("edge2_tile_id", secondEdgeEnumerator.EdgeId.TileId.ToString()), |
| 0 | 117 | | ("edge2_local_id", secondEdgeEnumerator.EdgeId.LocalId.ToString()), |
| 0 | 118 | | ("edges_prefix", string.Join(",", turnCost.prefixEdges.Select(x => x.ToString()))), |
| 0 | 119 | | ("cost", turnCost.cost.ToString()) |
| 0 | 120 | | }); |
| 0 | 121 | | jsonWriter.WriteProperties(attributes); |
| 0 | 122 | | jsonWriter.WritePropertyName("geometry"); |
| 0 | 123 | | jsonWriter.WriteLineString(edgeEnumerator.GetCompleteShape().Reverse() |
| 0 | 124 | | .Concat(secondEdgeEnumerator.GetCompleteShape())); |
| 0 | 125 | | jsonWriter.WriteFeatureEnd(); |
| 0 | 126 | | } |
| 0 | 127 | | } |
| 0 | 128 | | } |
| 0 | 129 | | } |
| | 130 | |
|
| 0 | 131 | | vertex = new VertexId(vertex.TileId, vertex.LocalId + 1); |
| 0 | 132 | | } |
| 0 | 133 | | } |
| | 134 | | } |