commonMain.earth.worldwind.globe.terrain.TerrainTile.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.globe.terrain
import earth.worldwind.geom.Sector
import earth.worldwind.geom.Vec3
import earth.worldwind.globe.Globe
import earth.worldwind.render.RenderContext
import earth.worldwind.render.buffer.FloatBufferObject
import earth.worldwind.util.Level
import earth.worldwind.util.Tile
import earth.worldwind.util.kgl.GL_ARRAY_BUFFER
/**
* Represents a portion of a globe's terrain. Applications typically do not interact directly with this class.
*/
open class TerrainTile(sector: Sector, level: Level, row: Int, column: Int): Tile(sector, level, row, column) {
val origin = Vec3()
val points by lazy { FloatArray((level.tileWidth + 2) * (level.tileHeight + 2) * 3) }
protected val heights by lazy { FloatArray( (level.tileWidth + 2) * (level.tileHeight + 2)) }
protected val heightGrid by lazy { FloatArray( level.tileWidth * level.tileHeight) }
/**
* Minimum elevation value used by the BasicTessellator to determine the terrain mesh edge extension depth (skirt).
* This value is scaled by the vertical exaggeration when the terrain is generated.
*/
protected val minTerrainElevation = -Short.MAX_VALUE.toFloat()
protected var heightTimestamp = 0L
protected var verticalExaggeration = 0.0f
protected var globeState: Globe.State? = null
protected var globeOffset: Globe.Offset? = null
var sortOrder = 0.0
protected set
private var pointBufferKey = "TerrainTile.points.default" // This key will be replaced on prepare of each tile
private var heightBufferKey = "TerrainTile.heights.default" // Use the same height buffer for all tiles by default
public override val heightLimits get() = super.heightLimits
open fun prepare(rc: RenderContext) {
val globe = rc.globe
val tileWidth = level.tileWidth
val tileHeight = level.tileHeight
val timestamp = rc.elevationModelTimestamp
if (timestamp != heightTimestamp) {
heightGrid.fill(0f)
globe.elevationModel.getHeightGrid(sector, tileWidth, tileHeight, heightGrid)
// Calculate height vertex buffer from height grid
for (r in 0 until level.tileHeight) for (c in 0 until level.tileWidth) {
heights[(r + 1) * (level.tileWidth + 2) + c + 1] = heightGrid[r * level.tileWidth + c]
}
if (rc.globe.is2D) {
heightGrid.fill(0f) // Do not show terrain in 2D, but keep height values in vertex for heatmap
calcHeightLimits(globe) // Force calculate height limits for heatmap
}
updateHeightBufferKey()
}
val ve = rc.verticalExaggeration.toFloat()
val state = rc.globeState
val offset = rc.globe.offset
if (timestamp != heightTimestamp || ve != verticalExaggeration || state != globeState || offset != globeOffset) {
val borderHeight = minTerrainElevation * ve
val rowStride = (tileWidth + 2) * 3
globe.geographicToCartesian(sector.centroidLatitude, sector.centroidLongitude, 0.0, origin)
globe.geographicToCartesianGrid(
sector, tileWidth, tileHeight, heightGrid, ve, origin, points, rowStride + 3, rowStride
)
globe.geographicToCartesianBorder(
sector, tileWidth + 2, tileHeight + 2, borderHeight, origin, points
)
updatePointBufferKey()
}
heightTimestamp = timestamp
verticalExaggeration = ve
globeState = state
globeOffset = offset
sortOrder = drawSortOrder(rc)
}
fun getHeightBuffer(rc: RenderContext) = rc.getBufferObject(heightBufferKey) {
FloatBufferObject(GL_ARRAY_BUFFER, heights)
}
fun getPointBuffer(rc: RenderContext) = rc.getBufferObject(pointBufferKey) {
FloatBufferObject(GL_ARRAY_BUFFER, points)
}
protected fun updateHeightBufferKey() {
heightBufferKey = "TerrainTile.heights.$tileKey.${bufferSequence++}"
}
protected fun updatePointBufferKey() {
pointBufferKey = "TerrainTile.points.$tileKey.${bufferSequence++}"
}
companion object {
private var bufferSequence = 0L // Must be static to avoid cache collisions when a tile instance is re-created
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy