Public
Edited
Nov 23, 2022
Importers
Insert cell
Insert cell
EarthRadius = 6378137
Insert cell
MinLatitude = -85.05112878
Insert cell
MaxLatitude = 85.05112878
Insert cell
MinLongitude = -180
Insert cell
MaxLongitude = 180
Insert cell
/**
* Clips a number to the specified minimum and maximum values.
* @param n The number to clip.
* @param minValue Minimum allowable value.
* @param maxValue Maximum allowable value.
* @returns The clipped value.
*/
function Clip(n, minValue, maxValue) {
return Math.min(Math.max(n, minValue), maxValue);
}
Insert cell
/**
* Calculates width and height of the map in pixels at a specific zoom level from -180 degrees to 180 degrees.
* @param zoom Zoom Level to calculate width at.
* @param tileSize The size of the tiles in the tile pyramid.
* @returns Width and height of the map in pixels.
*/
function MapSize(zoom, tileSize) {
return Math.ceil(tileSize * Math.pow(2, zoom));
}
Insert cell
/**
* Calculates the Ground resolution at a specific degree of latitude in the meters per pixel.
* @param latitude Degree of latitude to calculate resolution at.
* @param zoom Zoom level.
* @param tileSize The size of the tiles in the tile pyramid.
* @returns Ground resolution in meters per pixels.
*/
function GroundResolution(latitude, zoom, tileSize) {
latitude = Clip(latitude, MinLatitude, MaxLatitude);
return (
(Math.cos((latitude * Math.PI) / 180) * 2 * Math.PI * EarthRadius) /
MapSize(zoom, tileSize)
);
}
Insert cell
/**
* Determines the map scale at a specified latitude, level of detail, and screen resolution.
* @param latitude Latitude (in degrees) at which to measure the map scale.
* @param zoom Zoom level.
* @param screenDpi Resolution of the screen, in dots per inch.
* @param tileSize The size of the tiles in the tile pyramid.
* @returns The map scale, expressed as the denominator N of the ratio 1 : N.
*/
function MapScale(latitude, zoom, screenDpi, tileSize) {
return (GroundResolution(latitude, zoom, tileSize) * screenDpi) / 0.0254;
}
Insert cell
/**
* Global Converts a Pixel coordinate into a geospatial coordinate at a specified zoom level.
* Global Pixel coordinates are relative to the top left corner of the map (90, -180).
* @param pixel Pixel coordinates in the format of [x, y].
* @param zoom Zoom level.
* @param tileSize The size of the tiles in the tile pyramid.
* @returns A position value in the format [longitude, latitude].
*/
function GlobalPixelToPosition(pixel, zoom, tileSize) {
var mapSize = MapSize(zoom, tileSize);

var x = Clip(pixel[0], 0, mapSize - 1) / mapSize - 0.5;
var y = 0.5 - Clip(pixel[1], 0, mapSize - 1) / mapSize;

return [
360 * x, //Longitude
90 - (360 * Math.atan(Math.exp(-y * 2 * Math.PI))) / Math.PI //Latitude
];
}
Insert cell
/**
* Converts a point from latitude/longitude WGS-84 coordinates (in degrees) into pixel XY coordinates at a specified level of detail.
* @param position Position coordinate in the format [longitude, latitude].
* @param zoom Zoom level.
* @param tileSize The size of the tiles in the tile pyramid.
* @returns A pixel coordinate
*/
function PositionToGlobalPixel(position, zoom, tileSize) {
var latitude = Clip(position[1], MinLatitude, MaxLatitude);
var longitude = Clip(position[0], MinLongitude, MaxLongitude);

var x = (longitude + 180) / 360;
var sinLatitude = Math.sin((latitude * Math.PI) / 180);
var y = 0.5 - Math.log((1 + sinLatitude) / (1 - sinLatitude)) / (4 * Math.PI);

var mapSize = MapSize(zoom, tileSize);

return [
Clip(x * mapSize + 0.5, 0, mapSize - 1),
Clip(y * mapSize + 0.5, 0, mapSize - 1)
];
}
Insert cell
/**
* Converts pixel XY coordinates into tile XY coordinates of the tile containing the specified pixel.
* @param pixel Pixel coordinates in the format of [x, y].
* @param tileSize The size of the tiles in the tile pyramid.
* @returns Tile XY coordinates.
*/
function GlobalPixelToTileXY(pixel, tileSize) {
return {
tileX: Math.round(pixel[0] / tileSize),
tileY: Math.round(pixel[1] / tileSize)
};
}
Insert cell
/**
* Performs a scale transform on a global pixel value from one zoom level to another.
* @param pixel Pixel coordinates in the format of [x, y].
* @param oldZoom The zoom level in which the input global pixel value is from.
* @param newZoom The new zoom level in which the output global pixel value should be aligned with.
*/
function ScaleGlobalPixel(pixel, oldZoom, newZoom) {
var scale = Math.pow(2, oldZoom - newZoom);

return [pixel[0] * scale, pixel[1] * scale];
}
Insert cell
/**
* Performs a scale transform on a set of global pixel values from one zoom level to another.
* @param points A set of global pixel value from the old zoom level. Points are in the format [x,y].
* @param oldZoom The zoom level in which the input global pixel values is from.
* @param newZoom The new zoom level in which the output global pixel values should be aligned with.
* @returns A set of global pixel values that has been scaled for the new zoom level.
*/
function ScaleGlobalPixels(pixels, oldZoom, newZoom) {
var scale = Math.pow(2, oldZoom - newZoom);

var output = [];
for (var i = 0, len = pixels.length; i < len; i++) {
output.push([pixels[i][0] * scale, pixels[i][1] * scale]);
}

return output;
}
Insert cell
/**
* Converts tile XY coordinates into a global pixel XY coordinates of the upper-left pixel of the specified tile.
* @param tileX Tile X coordinate.
* @param tileY Tile Y coordinate.
* @param tileSize The size of the tiles in the tile pyramid.
* @returns Pixel coordinates in the format of [x, y].
*/
function TileXYToGlobalPixel(tileX, tileY, tileSize) {
return [tileX * tileSize, tileY * tileSize];
}
Insert cell
/**
* Converts tile XY coordinates into a quadkey at a specified level of detail.
* @param tileX Tile X coordinate.
* @param tileY Tile Y coordinate.
* @param zoom Zoom level.
* @returns A string containing the quadkey.
*/
function TileXYToQuadKey(tileX, tileY, zoom) {
var quadKey = [];
for (var i = zoom; i > 0; i--) {
var digit = 0;
var mask = 1 << (i - 1);

if ((tileX & mask) != 0) {
digit++;
}

if ((tileY & mask) != 0) {
digit += 2;
}

quadKey.push(digit);
}
return quadKey.join("");
}
Insert cell
/**
* Converts a quadkey into tile XY coordinates.
* @param quadKey Quadkey of the tile.
* @returns Tile XY cocorindates and zoom level for the specified quadkey.
*/
function QuadKeyToTileXY(quadKey) {
var tileX = 0;
var tileY = 0;
var zoom = quadKey.length;

for (var i = zoom; i > 0; i--) {
var mask = 1 << (i - 1);
switch (quadKey[zoom - i]) {
case "0":
break;

case "1":
tileX |= mask;
break;

case "2":
tileY |= mask;
break;

case "3":
tileX |= mask;
tileY |= mask;
break;

default:
throw "Invalid QuadKey digit sequence.";
}
}

return {
tileX: tileX,
tileY: tileY,
zoom: zoom
};
}
Insert cell
/**
* Calculates the XY tile coordinates that a coordinate falls into for a specific zoom level.
* @param position Position coordinate in the format [longitude, latitude].
* @param zoom Zoom level.
* @param tileSize The size of the tiles in the tile pyramid.
* @returns Tiel XY coordinates.
*/
function PositionToTileXY(position, zoom, tileSize) {
var latitude = Clip(position[1], MinLatitude, MaxLatitude);
var longitude = Clip(position[0], MinLongitude, MaxLongitude);

var x = (longitude + 180) / 360;
var sinLatitude = Math.sin((latitude * Math.PI) / 180);
var y = 0.5 - Math.log((1 + sinLatitude) / (1 - sinLatitude)) / (4 * Math.PI);

//tileSize needed in calculations as in rare cases the multiplying/rounding/dividing can make the difference of a pixel which can result in a completely different tile.
var mapSize = MapSize(zoom, tileSize);

return {
tileX: Math.floor(Clip(x * mapSize + 0.5, 0, mapSize - 1) / tileSize),
tileY: Math.floor(Clip(y * mapSize + 0.5, 0, mapSize - 1) / tileSize)
};
}
Insert cell
/**
* Calculates the tile quadkey strings that are within a bounding box at a specific zoom level.
* @param bounds A bounding box defined as an array of numbers in the format of [west, south, east, north].
* @param zoom Zoom level to calculate tiles for.
* @param tileSize The size of the tiles in the tile pyramid.
* @returns A list of quadkey strings.
*/
function GetQuadkeysInBoundingBox(bounds, zoom, tileSize) {
var keys = [];

if (bounds != null && bounds.length >= 4) {
var tl = PositionToTileXY([bounds[0], bounds[3]], zoom, tileSize);
var br = PositionToTileXY([bounds[2], bounds[1]], zoom, tileSize);

for (var x = tl[0]; x <= br[0]; x++) {
for (var y = tl[1]; y <= br[1]; y++) {
keys.push(TileXYToQuadKey(x, y, zoom));
}
}
}

return keys;
}
Insert cell
/**
* Calculates the tile quadkey strings that are within a specified viewport.
* @param position Position coordinate in the format [longitude, latitude].
* @param zoom Zoom level.
* @param width The width of the map viewport in pixels.
* @param height The height of the map viewport in pixels.
* @param tileSize The size of the tiles in the tile pyramid.
* @returns A list of quadkey strings that are within the specified viewport.
*/
function GetQuadkeysInView(position, zoom, width, height, tileSize) {
var p = PositionToGlobalPixel(position, zoom, tileSize);

var top = p[1] - height * 0.5;
var left = p[0] - width * 0.5;

var bottom = p[1] + height * 0.5;
var right = p[0] + width * 0.5;

var tl = GlobalPixelToPosition([left, top], zoom, tileSize);
var br = GlobalPixelToPosition([right, bottom], zoom, tileSize);

//Boudning box in the format: [west, south, east, north];
var bounds = [tl[0], br[1], br[0], tl[1]];

return GetQuadkeysInBoundingBox(bounds, zoom, tileSize);
}
Insert cell
/**
* Calculates the bounding box of a tile.
* @param tileX Tile X coordinate.
* @param tileY Tile Y coordinate.
* @param zoom Zoom level.
* @param tileSize The size of the tiles in the tile pyramid.
* @returns A bounding box of the tile defined as an array of numbers in the format of [west, south, east, north].
*/
function TileXYToBoundingBox(tileX, tileY, zoom, tileSize) {
//Top left corner pixel coordinates
var x1 = tileX * tileSize;
var y1 = tileY * tileSize;

//Bottom right corner pixel coordinates
var x2 = x1 + tileSize;
var y2 = y1 + tileSize;

var nw = GlobalPixelToPosition([x1, y1], zoom, tileSize);
var se = GlobalPixelToPosition([x2, y2], zoom, tileSize);

return [nw[0], se[1], se[0], nw[1]];
}
Insert cell
/**
* Calculates the best map view (center, zoom) for a bounding box on a map.
* @param bounds A bounding box defined as an array of numbers in the format of [west, south, east, north].
* @param mapWidth Map width in pixels.
* @param mapHeight Map height in pixels.
* @param padding Width in pixels to use to create a buffer around the map. This is to keep markers from being cut off on the edge.
* @param tileSize The size of the tiles in the tile pyramid.
* @returns The center and zoom level to best position the map view over the provided bounding box.
*/
function BestMapView(bounds, mapWidth, mapHeight, padding, tileSize) {
if (bounds == null || bounds.length < 4) {
return {
center: [0, 0],
zoom: 1
};
}

var boundsDeltaX;
var centerLat;
var centerLon;

//Check if east value is greater than west value which would indicate that bounding box crosses the antimeridian.
if (bounds[2] > bounds[0]) {
boundsDeltaX = bounds[2] - bounds[0];
centerLon = (bounds[2] + bounds[0]) / 2;
} else {
boundsDeltaX = 360 - (bounds[0] - bounds[2]);
centerLon = (((bounds[2] + bounds[0]) / 2 + 360) % 360) - 180;
}

var ry1 = Math.log(
(Math.sin((bounds[1] * Math.PI) / 180) + 1) /
Math.cos((bounds[1] * Math.PI) / 180)
);
var ry2 = Math.log(
(Math.sin((bounds[3] * Math.PI) / 180) + 1) /
Math.cos((bounds[3] * Math.PI) / 180)
);
var ryc = (ry1 + ry2) / 2;

centerLat = (Math.atan(Math.sinh(ryc)) * 180) / Math.PI;

var resolutionHorizontal = boundsDeltaX / (mapWidth - padding * 2);

var vy0 = Math.log(Math.tan(Math.PI * (0.25 + centerLat / 360)));
var vy1 = Math.log(Math.tan(Math.PI * (0.25 + bounds[3] / 360)));
var zoomFactorPowered =
(mapHeight * 0.5 - padding) / (40.7436654315252 * (vy1 - vy0));
var resolutionVertical = 360.0 / (zoomFactorPowered * tileSize);

var resolution = Math.max(resolutionHorizontal, resolutionVertical);

var zoom = Math.log2(360 / (resolution * tileSize));

return {
center: [centerLon, centerLat],
zoom: zoom
};
}
Insert cell

One platform to build and deploy the best data apps

Experiment and prototype by building visualizations in live JavaScript notebooks. Collaborate with your team and decide which concepts to build out.
Use Observable Framework to build data apps locally. Use data loaders to build in any language or library, including Python, SQL, and R.
Seamlessly deploy to Observable. Test before you ship, use automatic deploy-on-commit, and ensure your projects are always up-to-date.
Learn more