home

Menu
  • ripgrep search

datasette-tiles/datasette_tiles/utils.py

import math
 
 
async def detect_mtiles_databases(datasette):
    await datasette.refresh_schemas()
    internal = datasette.get_database("_internal")
    result = await internal.execute(
        """
    select
      columns.database_name,
      columns.table_name,
      group_concat(columns.name) as columns
    from
      columns
    where
      columns.table_name = "tiles"
    group by
      columns.database_name,
      columns.table_name
    order by
      columns.table_name
    """
    )
    return [
        row["database_name"]
        for row in result.rows
        if set(row["columns"].split(",")).issuperset(
            {"tile_column", "tile_data", "tile_row", "zoom_level"}
        )
    ]
 
 
async def tiles_stack_database_order(datasette):
    config = datasette.plugin_config("datasette-tiles") or {}
    stack_order = config.get("tiles-stack-order")
    if not stack_order:
        mtiles_databases = await detect_mtiles_databases(datasette)
        stack_order = [
            name for name in datasette.databases.keys() if name in mtiles_databases
        ]
    database_order = list(reversed(stack_order))
    # if datasette-basemap is installed, move basemap to the end
    plugins = [
        p["name"] for p in (await datasette.client.get("/-/plugins.json")).json()
    ]
    if (
        not config.get("tiles-stack-order")
        and "datasette-basemap" in plugins
        and "basemap" in database_order
    ):
        database_order.remove("basemap")
        database_order.append("basemap")
    return [datasette.databases[name] for name in database_order]
 
 
def latlon_to_tile(lat, lon, zoom):
    x_tile = (lon + 180) / 360 * 2 ** zoom
    y_tile = (
        (
            1
            - math.log(math.tan(math.radians(lat)) + 1 / math.cos(math.radians(lat)))
            / math.pi
        )
        / 2
        * 2 ** zoom
    )
    return x_tile, y_tile
 
 
# Given a lat/lon, convert it to OSM tile co-ordinates (nearest actual tile,
# adjusted so the point will be near the centre of a 2x2 tiled map).
def latlon_to_tile_with_adjust(lat, lon, zoom):
    x_tile, y_tile = latlon_to_tile(lat, lon, zoom)
 
    # Try and have point near centre of map
    if x_tile - int(x_tile) > 0.5:
        x_tile += 1
    if y_tile - int(y_tile) > 0.5:
        y_tile += 1
 
    return int(x_tile), int(y_tile)
 
 
def tile_to_latlon(x, y, zoom):
    n = 2 ** zoom
    lon = x / n * 360 - 180
    lat = math.degrees(math.atan(math.sinh(math.pi * (1 - 2 * y / n))))
    return {"lat": lat, "lon": lon}
 
Powered by Datasette