All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.oscim.core.Tile Maven / Gradle / Ivy

/*
 * Copyright 2010, 2011, 2012 mapsforge.org
 * Copyright 2013 Hannes Janetzek
 * Copyright 2016-2017 devemux86
 *
 * This file is part of the OpenScienceMap project (http://www.opensciencemap.org).
 *
 * This program is free software: you can redistribute it and/or modify it under the
 * terms of the GNU Lesser General Public License as published by the Free Software
 * Foundation, either version 3 of the License, or (at your option) any later version.
 *
 * This program is distributed in the hope that it will be useful, but WITHOUT ANY
 * WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A
 * PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details.
 *
 * You should have received a copy of the GNU Lesser General Public License along with
 * this program. If not, see .
 */
package org.oscim.core;

import org.oscim.backend.CanvasAdapter;

/**
 * A tile represents a rectangular part of the world map. All tiles can be
 * identified by their X and Y number together with their zoom level. The actual
 * area that a tile covers on a map depends on the underlying map projection.
 */
public class Tile {

    /**
     * Default tile size in pixels.
     */
    private static final int DEFAULT_TILE_SIZE = 256;

    /**
     * Width and height of a map tile in pixels.
     */
    public static int SIZE = 512;

    /**
     * Tile size multiple in pixels.
     */
    public static int TILE_SIZE_MULTIPLE = 64;

    /**
     * the map size implied by zoom level and tileSize, to avoid multiple computations.
     */
    public final long mapSize;

    /**
     * The X number of this tile.
     */
    public final int tileX;

    /**
     * The Y number of this tile.
     */
    public final int tileY;

    /**
     * The Zoom level of this tile.
     */
    public final byte zoomLevel;

    private BoundingBox boundingBox;
    private Point origin;

    /**
     * @param tileX     the X number of the tile.
     * @param tileY     the Y number of the tile.
     * @param zoomLevel the zoom level of the tile.
     */
    public Tile(int tileX, int tileY, byte zoomLevel) {
        this.tileX = tileX;
        this.tileY = tileY;
        this.zoomLevel = zoomLevel;
        this.mapSize = MercatorProjection.getMapSize(zoomLevel);
    }

    @Override
    public String toString() {
        return new StringBuilder()
                .append("[X:")
                .append(this.tileX)
                .append(", Y:")
                .append(this.tileY)
                .append(", Z:")
                .append(this.zoomLevel)
                .append("]")
                .toString();
    }

    @Override
    public boolean equals(Object obj) {
        if (this == obj)
            return true;

        if (!(obj instanceof Tile))
            return false;

        Tile o = (Tile) obj;

        if (o.tileX == this.tileX && o.tileY == this.tileY
                && o.zoomLevel == this.zoomLevel)
            return true;

        return false;
    }

    private int mHash = 0;

    @Override
    public int hashCode() {
        if (mHash == 0) {
            int result = 7;
            result = 31 * result + this.tileX;
            result = 31 * result + this.tileY;
            result = 31 * result + this.zoomLevel;
            mHash = result;
        }
        return mHash;
    }

    /**
     * Calculate tile size (256px) with a scale factor.
     * Clamp tile size to a preset multiple, e.g. 64px.
     */
    public static int calculateTileSize() {
        float scaled = DEFAULT_TILE_SIZE * CanvasAdapter.getScale();
        return Math.max(TILE_SIZE_MULTIPLE,
                Math.round(scaled / TILE_SIZE_MULTIPLE) * TILE_SIZE_MULTIPLE);
    }

    /**
     * Gets the geographic extend of this Tile as a BoundingBox.
     *
     * @return boundaries of this tile.
     */
    public BoundingBox getBoundingBox() {
        if (this.boundingBox == null) {
            double minLatitude = Math.max(MercatorProjection.LATITUDE_MIN, MercatorProjection.tileYToLatitude(tileY + 1, zoomLevel));
            double minLongitude = Math.max(-180, MercatorProjection.tileXToLongitude(this.tileX, zoomLevel));
            double maxLatitude = Math.min(MercatorProjection.LATITUDE_MAX, MercatorProjection.tileYToLatitude(this.tileY, zoomLevel));
            double maxLongitude = Math.min(180, MercatorProjection.tileXToLongitude(tileX + 1, zoomLevel));
            if (maxLongitude == -180) {
                // fix for dateline crossing, where the right tile starts at -180 and causes an invalid bbox
                maxLongitude = 180;
            }
            this.boundingBox = new BoundingBox(minLatitude, minLongitude, maxLatitude, maxLongitude);
        }
        return this.boundingBox;
    }

    /**
     * Return the BoundingBox of a rectangle of tiles defined by upper left and lower right tile.
     *
     * @param upperLeft  tile in upper left corner.
     * @param lowerRight tile in lower right corner.
     * @return BoundingBox defined by the area around upperLeft and lowerRight Tile.
     */
    public static BoundingBox getBoundingBox(Tile upperLeft, Tile lowerRight) {
        BoundingBox ul = upperLeft.getBoundingBox();
        BoundingBox lr = lowerRight.getBoundingBox();
        return ul.extendBoundingBox(lr);
    }

    /**
     * @return the maximum valid tile number for the given zoom level, 2zoomLevel -1.
     */
    public static int getMaxTileNumber(byte zoomLevel) {
        if (zoomLevel < 0) {
            throw new IllegalArgumentException("zoomLevel must not be negative: " + zoomLevel);
        } else if (zoomLevel == 0) {
            return 0;
        }
        return (2 << zoomLevel - 1) - 1;
    }

    /**
     * Returns the top-left point of this tile in absolute coordinates.
     *
     * @return the top-left point
     */
    public Point getOrigin() {
        if (this.origin == null) {
            double x = MercatorProjection.tileToPixel(this.tileX);
            double y = MercatorProjection.tileToPixel(this.tileY);
            this.origin = new Point(x, y);
        }
        return this.origin;
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy