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

commonMain.earth.worldwind.shape.TiledSurfaceImage.kt Maven / Gradle / Ivy

Go to download

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.shape

import earth.worldwind.draw.DrawableSurfaceTexture
import earth.worldwind.geom.Angle
import earth.worldwind.geom.Matrix3
import earth.worldwind.geom.Sector
import earth.worldwind.globe.Globe
import earth.worldwind.render.AbstractRenderable
import earth.worldwind.render.RenderContext
import earth.worldwind.render.Texture
import earth.worldwind.render.image.ImageOptions
import earth.worldwind.render.image.ImageSource
import earth.worldwind.render.image.ImageTile
import earth.worldwind.render.program.SurfaceTextureProgram
import earth.worldwind.util.*
import kotlinx.coroutines.*
import kotlin.time.Duration.Companion.seconds

open class TiledSurfaceImage(tileFactory: TileFactory, levelSet: LevelSet): AbstractRenderable("Tiled Surface Image") {
    /**
     * Tile factory implementation.
     */
    var tileFactory = tileFactory
        set(value) {
            field = value
            invalidateTiles()
        }
    /**
     * Tile pyramid representation.
     */
    var levelSet = levelSet
        set(value) {
            field = value
            invalidateTiles()
        }
    /**
     * Additional image texture options.
     */
    var imageOptions: ImageOptions? = null
        set(value) {
            field = value
            invalidateTiles()
        }
    /**
     * Define imagery level of details. It controls tile pixel density on the screen.
     */
    var detailControl = 1.0
    /**
     * Retrieve top level tiles to avoid black holes when navigating and zooming out camera
     */
    var retrieveTopLevelTiles = true
    /**
     * Use ancestor tile texture as a fallback for descendants
     */
    var useAncestorTileTexture = true
    /**
     * Define cache tiles factory implementation.
     */
    var cacheTileFactory: CacheTileFactory? = null
    /**
     * Configures tiled surface image to retrieve only the cache source.
     */
    var isCacheOnly = false

    /**
     * Memory cache for this layer's subdivision tiles. Each entry contains an array of four image tiles corresponding
     * to the subdivision of the group's common parent tile. The cache is configured to hold 1200 groups, a number
     * empirically determined to be sufficient for storing the tiles needed to navigate a small region.
     */
    protected var tileCache = LruMemoryCache>(1200)
    protected var activeProgram: SurfaceTextureProgram? = null
    protected var ancestorTile: ImageTile? = null
    protected var ancestorTexture: Texture? = null
    protected val ancestorTexCoordMatrix = Matrix3()
    protected val topLevelTiles = mutableListOf()
    protected var lastGlobeState: Globe.State? = null

    /**
     * Makes a copy of this tiled surface image
     */
    open fun clone() = TiledSurfaceImage(tileFactory, LevelSet(levelSet)).also {
        it.imageOptions = imageOptions
    }

    /**
     * Cache size should be adjusted in case of levelSet or detailControl changed.
     */
    fun setupTileCache(capacity: Long, lowWater: Long = (capacity * 0.75).toLong()) {
        tileCache = LruMemoryCache(capacity, lowWater)
    }

    override fun doRender(rc: RenderContext) {
        if (rc.terrain.sector.isEmpty) return // no terrain surface to render on
        checkGlobeState(rc)
        determineActiveProgram(rc)
        assembleTiles(rc)
        activeProgram = null // clear the active program to avoid leaking render resources
        ancestorTile = null // clear the ancestor tile and texture
        ancestorTexture = null
    }

    @Throws(IllegalStateException::class, IllegalArgumentException::class)
    open fun launchBulkRetrieval(
        scope: CoroutineScope, sector: Sector, resolution: ClosedRange, onProgress: ((Long, Long, Long) -> Unit)?,
        retrieveTile: suspend (imageSource: ImageSource, cacheSource: ImageSource, options: ImageOptions?) -> Boolean
    ): Job {
        val cacheTileFactory = cacheTileFactory ?: error("Cache not configured")
        require(sector.intersect(levelSet.sector)) { "Sector does not intersect tiled surface image sector" }
        return scope.launch(Dispatchers.Default) {
            val minLevel = levelSet.levelForResolution(resolution.endInclusive)
            val maxLevel = levelSet.levelForResolution(resolution.start)
            val tileCount = levelSet.tileCount(sector, minLevel, maxLevel)
            var downloaded = 0L
            var skipped = 0L
            if (topLevelTiles.isEmpty()) createTopLevelTiles()
            topLevelTiles.forEach { topLevelTile ->
                processAndSubdivideTile(topLevelTile as ImageTile, sector, minLevel, maxLevel) { tile ->
                    var attempt = 0
                    // Retry download attempts till success or 404 not fond or job canceled
                    while(true) {
                        // Check if a job canceled
                        ensureActive()
                        // No image source indicates an empty level or an image missing from the tiled data store
                        val imageSource = tile.imageSource ?: break
                        // If cache tile factory is specified, then create a cache source and store it in tile
                        val cacheSource = tile.cacheSource
                            ?: (cacheTileFactory.createTile(tile.sector, tile.level, tile.row, tile.column) as ImageTile)
                                .imageSource?.also { tile.cacheSource = it } ?: break
                        // Attempt to download tile
                        try {
                            ++attempt
                            if (retrieveTile(imageSource, cacheSource, imageOptions)) {
                                // Tile successfully downloaded
                                onProgress?.invoke(++downloaded, skipped, tileCount)
                            } else {
                                // Received data cannot be decoded as image
                                onProgress?.invoke(downloaded, ++skipped, tileCount)
                            }
                            break // Continue downloading the next tile
                        } catch (_: Throwable) {
                            delay(if (attempt % makeLocalRetries == 0) makeLocalTimeoutLong else makeLocalTimeoutShort)
                        }
                    }
                }
            }
        }
    }

    protected open suspend fun processAndSubdivideTile(
        tile: ImageTile, sector: Sector, minLevel: Level, maxLevel: Level, process: suspend (ImageTile) -> Unit
    ) {
        if (!tile.intersectsSector(sector)) return // Ignore tiles and its descendants outside the specified sector
        val levelNumber = tile.level.levelNumber
        // Skip tiles with level less than specified offset from the process
        if (levelNumber >= minLevel.levelNumber && levelNumber >= levelSet.levelOffset) process(tile)
        // Do not subdivide if specified level or last available level reached
        if (levelNumber < maxLevel.levelNumber && !tile.level.isLastLevel) {
            tile.subdivide(tileFactory).forEach { processAndSubdivideTile(it as ImageTile, sector, minLevel, maxLevel, process) }
        }
    }

    protected open fun determineActiveProgram(rc: RenderContext) {
        activeProgram = rc.getShaderProgram { SurfaceTextureProgram() }
    }

    protected open fun assembleTiles(rc: RenderContext) {
        // TODO
        // The need to create Tiles with a defined image source couples the need to determine a tile's visibility with
        // he need to know its image source. Decoupling the two would mean we only need to know the image source when
        // the texture is actually requested Could the tile-based operations done here be implicit on level/row/column,
        // or use transient pooled tile objects not tied to an image source?
        if (topLevelTiles.isEmpty()) createTopLevelTiles()
        for (i in topLevelTiles.indices) addTileOrDescendants(rc, topLevelTiles[i] as ImageTile)
    }

    protected open fun createTopLevelTiles() = Tile.assembleTilesForLevel(levelSet.firstLevel, tileFactory, topLevelTiles)

    protected open fun addTileOrDescendants(rc: RenderContext, tile: ImageTile) {
        // ignore tiles which do not fit projection limits
        if (rc.globe.projectionLimits?.let { tile.intersectsSector(it) } == false) return
        // ignore the tile and its descendants if it's not needed or not visible
        if (!tile.intersectsSector(levelSet.sector) || !tile.intersectsSector(rc.terrain.sector) || !tile.intersectsFrustum(rc)) return
        // Do not retrieve tiles bigger than level size because they can be simulated by downscaling more detailed tiles
        // TODO Remove this restriction when GeoPackage will be able to correctly align tiles bigger than level size
        val validSize = tile.level.levelWidth >= tile.level.tileWidth && tile.level.levelHeight >= tile.level.tileHeight
        // Do not retrieve tiles from levels before level offset
        val retrieveCurrentLevel = validSize && tile.level.levelNumber >= levelSet.levelOffset
        // Use current tile if it must not subdivide and should be retrieved, or it is the last level tile
        val isLastLevel = tile.level.isLastLevel
        val mustSubdivide = tile.mustSubdivide(rc, detailControl)
        if (isLastLevel || retrieveCurrentLevel && !mustSubdivide) {
            // Skip using last level tile on more detailed levels if using ancestor tiles is switched off
            if (!isLastLevel || !mustSubdivide || useAncestorTileTexture) addTile(rc, tile)
            return  // use the tile if it does not need to be subdivided
        }
        val currentAncestorTile = ancestorTile
        val currentAncestorTexture = ancestorTexture
        getTexture(rc, tile, retrieveTopLevelTiles && retrieveCurrentLevel)?.let { tileTexture ->
            // tile has a texture; use it as a fallback tile for descendants
            ancestorTile = tile
            ancestorTexture = tileTexture
        }
        // each tile has a cached size of 1, recursively process the tile's children
        val children = tile.subdivideToCache(tileFactory, tileCache, 4)
        for (i in children.indices) addTileOrDescendants(rc, children[i] as ImageTile)
        ancestorTile = currentAncestorTile // restore the last fallback tile, even if it was null
        ancestorTexture = currentAncestorTexture
    }

    protected open fun addTile(rc: RenderContext, tile: ImageTile) {
        val texture = getTexture(rc, tile)
        val ancestorTile = ancestorTile
        val ancestorTexture = ancestorTexture
        val imageSource = tile.imageSource
        val absentResourceList = rc.renderResourceCache.absentResourceList
        val opacity = if (rc.isPickMode) 1f else rc.currentLayer.opacity
        if (texture != null) {
            // use the tile's own texture
            val pool = rc.getDrawablePool()
            val drawable = DrawableSurfaceTexture.obtain(pool).set(
                activeProgram, tile.sector, opacity, texture, texture.coordTransform, rc.globe.offset
            )
            rc.offerSurfaceDrawable(drawable, 0.0 /*z-order*/)
        } else if (ancestorTile != null && ancestorTexture != null && (
            // Use ancestor tile if it is allowed or previous level tile is still loading
            useAncestorTileTexture || tile.level.levelNumber - ancestorTile.level.levelNumber <= 1
                    && imageSource != null && !absentResourceList.isResourceAbsent(imageSource.hashCode())
        )) {
            // use the ancestor tile's texture, transformed to fill the tile sector
            ancestorTexCoordMatrix.copy(ancestorTexture.coordTransform)
            ancestorTexCoordMatrix.multiplyByTileTransform(tile.sector, ancestorTile.sector)
            val pool = rc.getDrawablePool()
            val drawable = DrawableSurfaceTexture.obtain(pool).set(
                activeProgram, tile.sector, opacity, ancestorTexture, ancestorTexCoordMatrix, rc.globe.offset
            )
            rc.offerSurfaceDrawable(drawable, 0.0 /*z-order*/)
        }
    }

    // TODO If cache source retrieved but it is outdated, than try to retrieve original image source anyway to refresh cache
    protected open fun getTexture(rc: RenderContext, tile: ImageTile, retrieve: Boolean = true): Texture? {
        // No image source indicates an empty level or an image missing from the tiled data store
        val imageSource = tile.imageSource ?: return null
        // If cache tile factory is specified, then create a cache source and store it in tile
        val cacheSource = tile.cacheSource ?: cacheTileFactory?.run {
            (createTile(tile.sector, tile.level, tile.row, tile.column) as ImageTile).imageSource?.also { tile.cacheSource = it }
        }
        // Read cache only in case of tile factory and cache factory are the same
        val isCacheOnly = isCacheOnly || tileFactory == cacheTileFactory
        // If a cache source is not absent, then retrieve it instead of an original image source
        val isCacheAbsent = cacheSource == null || rc.renderResourceCache.absentResourceList.isResourceAbsent(cacheSource.hashCode())
        return rc.getTexture(
            if (isCacheAbsent) imageSource else cacheSource, imageOptions, retrieve && (!isCacheOnly || !isCacheAbsent)
        )
    }

    protected open fun checkGlobeState(rc: RenderContext) {
        // Invalidate tiles cache when globe state changes
        if (rc.globeState != lastGlobeState) {
            invalidateTiles()
            lastGlobeState = rc.globeState
        }
    }

    open fun invalidateTiles() {
        topLevelTiles.clear()
        tileCache.clear()
    }

    companion object {
        /**
         * Number of reties of bulk tile retrieval before long timeout
         */
        var makeLocalRetries = 3
        /**
         * Short timeout on bulk tile retrieval failed
         */
        var makeLocalTimeoutShort = 5.seconds
        /**
         * Long timeout on bulk tile retrieval failed
         */
        var makeLocalTimeoutLong = 15.seconds
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy