37 lines
1.2 KiB
Kotlin
37 lines
1.2 KiB
Kotlin
package centre.sciprog.maps
|
|
|
|
import kotlin.math.roundToInt
|
|
|
|
/**
|
|
* Observable position on the map. Includes observation coordinate and [zoom] factor
|
|
*/
|
|
data class MapViewPoint(
|
|
val focus: GeodeticMapCoordinates,
|
|
val zoom: Double,
|
|
) {
|
|
val scaleFactor by lazy { WebMercatorProjection.scaleFactor(zoom.roundToInt()) }
|
|
}
|
|
|
|
fun MapViewPoint.move(deltaX: Float, deltaY: Float): MapViewPoint {
|
|
val newCoordinates = GeodeticMapCoordinates.ofRadians(
|
|
(focus.latitude + deltaY / scaleFactor).coerceIn(
|
|
-MercatorProjection.MAXIMUM_LATITUDE,
|
|
MercatorProjection.MAXIMUM_LATITUDE
|
|
),
|
|
focus.longitude + deltaX / scaleFactor
|
|
)
|
|
return MapViewPoint(newCoordinates, zoom)
|
|
}
|
|
|
|
fun MapViewPoint.move(delta: GeodeticMapCoordinates): MapViewPoint {
|
|
val newCoordinates = GeodeticMapCoordinates.ofRadians(
|
|
(focus.latitude + delta.latitude).coerceIn(
|
|
-MercatorProjection.MAXIMUM_LATITUDE,
|
|
MercatorProjection.MAXIMUM_LATITUDE
|
|
),
|
|
focus.longitude + delta.longitude
|
|
)
|
|
return MapViewPoint(newCoordinates, zoom)
|
|
}
|
|
|
|
fun MapViewPoint.zoom(zoomDelta: Double): MapViewPoint = copy(zoom = (zoom + zoomDelta).coerceIn(2.0, 18.0)) |