commonMain.earth.worldwind.util.Tile.kt Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of worldwind-jvm Show documentation
Show all versions of worldwind-jvm Show documentation
The WorldWind Kotlin SDK (WWK) includes the library, examples and tutorials for building multiplatform 3D virtual globe applications for Android, Web and Java.
The newest version!
package earth.worldwind.util
import earth.worldwind.geom.Sector
import earth.worldwind.geom.Vec3
import earth.worldwind.render.RenderContext
import kotlin.jvm.JvmStatic
import kotlin.math.abs
import kotlin.math.cos
/**
* Geographically rectangular tile within a [LevelSet], typically representing terrain or imagery. Provides a base
* class for tiles used by tiled image layers and elevation models. Applications typically do not interact with this
* class.
*/
open class Tile protected constructor(
/**
* The sector spanned by this tile.
*/
sector: Sector,
/**
* The level at which this tile lies within a [LevelSet].
*/
val level: Level,
/**
* The tile's row within its level.
*/
val row: Int,
/**
* The tile's column within its level.
*/
val column: Int
) : AbstractTile(sector) {
/**
* A key that uniquely identifies this tile within a level set. Tile keys are not unique to a specific level set.
*/
val tileKey = "${level.levelNumber}.$row.$column"
private val scratchVector = Vec3()
/**
* Indicates whether this tile should be subdivided based on the current navigation state and a specified detail
* factor.
*
* @param rc the current render context
* @param detailFactor the detail factor to consider
*
* @return true if the tile should be subdivided, otherwise false
*/
open fun mustSubdivide(rc: RenderContext, detailFactor: Double): Boolean {
var texelSize = level.texelSize * rc.globe.equatorialRadius // Compute texel size in meters
val pixelSize = if (rc.globe.is2D) rc.pixelSize else {
// Consider that texels are laid out continuously on the arc of constant latitude connecting the tile's
// east and west edges and passing through its centroid.
texelSize *= cos(sector.centroidLatitude.inRadians)
// Get distance from nearest tile point to camera
val nearestPoint = nearestPoint(rc)
val distanceToCamera = nearestPoint.distanceTo(rc.cameraPoint)
// Accelerate the degradation of tile details depending on the viewing angle to tile normal
if (isAccelerateDegradation && level.tileDelta.latitude.inDegrees <= 5.625) {
val viewingVector = nearestPoint.subtract(rc.cameraPoint)
val normalVector =
rc.globe.geographicToCartesianNormal(sector.centroidLatitude, sector.centroidLongitude, scratchVector)
val dot = viewingVector.dot(normalVector)
texelSize *= abs(dot / (viewingVector.magnitude * normalVector.magnitude))
}
// Use individual pixel size based on tile distance to camera
rc.pixelSizeAtDistance(distanceToCamera)
}
// Adjust the subdivision factory when the display density is low.
return texelSize > pixelSize * detailFactor * rc.densityFactor
}
/**
* Returns the four children formed by subdividing this tile. This tile's sector is subdivided into four quadrants
* as follows: Southwest; Southeast; Northwest; Northeast. A new tile is then constructed for each quadrant and
* configured with the next level within this tile's LevelSet and its corresponding row and column within that
* level. This returns null if this tile's level is the last level within its [LevelSet].
*
* @param tileFactory the tile factory to use to create the children
*
* @return an array containing the four child tiles, or null if this tile's level is the last level
*/
open fun subdivide(tileFactory: TileFactory): Array {
val childLevel = level.nextLevel ?: return emptyArray()
val latMin = sector.minLatitude
val lonMin = sector.minLongitude
val latMid = sector.centroidLatitude
val lonMid = sector.centroidLongitude
val latMax = sector.maxLatitude
val lonMax = sector.maxLongitude
var childRow = 2 * row
var childCol = 2 * column
var childSector = Sector(latMin, latMid, lonMin, lonMid)
val child0 = tileFactory.createTile(childSector, childLevel, childRow, childCol) // Southwest
childRow = 2 * row
childCol = 2 * column + 1
childSector = Sector(latMin, latMid, lonMid, lonMax)
val child1 = tileFactory.createTile(childSector, childLevel, childRow, childCol) // Southeast
childRow = 2 * row + 1
childCol = 2 * column
childSector = Sector(latMid, latMax, lonMin, lonMid)
val child2 = tileFactory.createTile(childSector, childLevel, childRow, childCol) // Northwest
childRow = 2 * row + 1
childCol = 2 * column + 1
childSector = Sector(latMid, latMax, lonMid, lonMax)
val child3 = tileFactory.createTile(childSector, childLevel, childRow, childCol) // Northeast
return arrayOf(child0, child1, child2, child3)
}
/**
* Returns the four children formed by subdividing this tile, drawing those children from a specified cache. The
* cache is checked for a child collection prior to subdividing. If one exists in the cache it is returned rather
* than creating a new collection of children. If a new collection is created in the same manner as [subdivide] and added to the cache.
*
* @param tileFactory the tile factory to use to create the children
* @param cache a memory cache that may contain pre-existing child tiles.
* @param cacheSize the cached size of the four child tiles
*
* @return an array containing the four child tiles, or null if this tile's level is the last level
*/
open fun subdivideToCache(
tileFactory: TileFactory, cache: LruMemoryCache>, cacheSize: Int
) = cache[tileKey] ?: subdivide(tileFactory).also { cache.put(tileKey, it, cacheSize) }
companion object {
/**
* Accelerate the degradation of tile details depending on the viewing angle to tile normal.
* This option dramatically increases performance, but degrades scene background level of details.
*/
var isAccelerateDegradation = true
/**
* Creates all tiles for a specified level within a [LevelSet].
*
* @param level the level to create the tiles for
* @param tileFactory the tile factory to use for creating tiles.
* @param result an pre-allocated Collection in which to store the results
*
* @return the result argument populated with the tiles for the specified level
*/
@JvmStatic
fun assembleTilesForLevel(level: Level, tileFactory: TileFactory, result: MutableList): List {
val sector = level.parent.sector
val tileOrigin = level.parent.tileOrigin
val tileDelta = level.tileDelta
val firstRow = tileOrigin.computeRow(tileDelta.latitude, sector.minLatitude)
val lastRow = tileOrigin.computeLastRow(tileDelta.latitude, sector.maxLatitude)
val firstCol = tileOrigin.computeColumn(tileDelta.longitude, sector.minLongitude)
val lastCol = tileOrigin.computeLastColumn(tileDelta.longitude, sector.maxLongitude)
val firstRowLat = tileOrigin.minLatitude.plusDegrees(firstRow * tileDelta.latitude.inDegrees)
val firstColLon = tileOrigin.minLongitude.plusDegrees(firstCol * tileDelta.longitude.inDegrees)
var minLat = firstRowLat
for (row in firstRow..lastRow) {
val maxLat = minLat + tileDelta.latitude
var minLon = firstColLon
for (col in firstCol..lastCol) {
val maxLon = minLon + tileDelta.longitude
val tileSector = Sector(minLat, maxLat, minLon, maxLon)
result.add(tileFactory.createTile(tileSector, level, row, col))
minLon = maxLon
}
minLat = maxLat
}
return result
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy