The GeoJSON specification does not allow shapes to cross the antemeridian. I'm not a fan of this, luckily there are a couple of tools in Azure Maps to assist with this.
Options 1: Use longitude values that are outside the range of -180/180, this will cause points to render on the adjacent map. The atlas.math.getPathDenormalizedAtAntimerian
function in Azure Maps takes in an array of coordinates for a path, and does this for you. This keeps the geometry a LineString, but not everyone is a fan of using longitude values outside the -180/180 range, so then there is option 2.
Option 2: Go through each line segment and split any that have a longitude difference of more than 180 degrees. This will generate a MultiLineString. The geometry type changes, but the longitude values stay within the -180/180 range. The atlas.math.getPathSplitByAntimeridian
function in Azure Maps takes in an array of coordinates for a path, and does this for you. This is a much more complex calculation than option 1, but unless you are doing this with a ton of lines, this option should work fine. If you do have a ton of lines, you may also want to consider splitting them in your source data ahead of time.
If you plan on exporting the data that's in the map, you will likely find option 2 better as the output would be valid GeoJSON while the output from option 1 is invalid.
Here is a code sample showing all two options:
<!DOCTYPE html>
<html lang="en">
<head>
<title></title>
<meta charset="utf-8" />
<meta http-equiv="x-ua-compatible" content="IE=Edge" />
<meta name="viewport" content="width=device-width, initial-scale=1, shrink-to-fit=no" />
<!-- Add references to the Azure Maps Map control JavaScript and CSS files. -->
<link rel="stylesheet" href="https://atlas.microsoft.com/sdk/javascript/mapcontrol/3/atlas.min.css" type="text/css" />
<script src="https://atlas.microsoft.com/sdk/javascript/mapcontrol/3/atlas.js"></script>
<script type='text/javascript'>
var map, datasource;
var data = {
"type": "Feature",
"properties": {},
"geometry": {
"type": "LineString",
"coordinates": [
[
144.9586,
-37.848381
],
[
-118.1948,
33.767109
]
]
}
};
function GetMap() {
//Initialize a map instance.
map = new atlas.Map('myMap', {
view: 'Auto',
//Add your Azure Maps subscription client ID to the map SDK. Get an Azure Maps client ID at https://azure.com/maps
authOptions: {
authType: 'subscriptionKey',
subscriptionKey: '<Your Azure Maps Key>'
}
});
//Wait until the map resources are ready.
map.events.add('ready', function () {
//Create a data source to add your data to.
datasource = new atlas.source.DataSource();
map.sources.add(datasource);
//Add a layer for rendering data.
map.layers.add(new atlas.layer.LineLayer(datasource));
map.setCamera({
bounds: atlas.data.BoundingBox.fromData(data)
});
});
}
function option1() {
//Denormalize the coordinates that cross the antimeridian. This will result in some longitude values being outside of the -180 to 180 range.
//Docs: https://learn.microsoft.com/en-us/javascript/api/azure-maps-control/atlas.math?view=azure-maps-typescript-latest#azure-maps-control-atlas-math-getpathdenormalizedatantimerian
var newCoords = atlas.math.getPathDenormalizedAtAntimerian(data.geometry.coordinates);
datasource.setShapes(new atlas.data.LineString(newCoords));
}
function option2() {
//This splits the line on the antimeridian if any line segments cross it (distance between longitudes is greater than 180).
//Docs: https://learn.microsoft.com/en-us/javascript/api/azure-maps-control/atlas.math?view=azure-maps-typescript-latest#azure-maps-control-atlas-math-getpathdenormalizedatantimerian
var newCoords = atlas.math.getPathSplitByAntimeridian(data.geometry.coordinates);
datasource.setShapes(new atlas.data.MultiLineString(newCoords));
}
</script>
</head>
<body onload="GetMap()">
<div id="myMap" style="position:relative;width:100%;height:600px;"></div>
<input type="button" onclick="option1()" value="Option 1" />
<input type="button" onclick="option2()" value="Option 2" />
</body>
</html>