Meshtastic-Android/app/src/main/java/com/geeksville/mesh/ui/MapFragment.kt

221 lines
7.6 KiB
Kotlin
Raw Normal View History

2020-03-11 14:45:49 -07:00
package com.geeksville.mesh.ui
2020-03-30 15:00:18 -07:00
import android.graphics.Color
2020-03-30 11:56:59 -07:00
import android.os.Bundle
2020-04-07 11:27:51 -07:00
import android.view.LayoutInflater
import android.view.View
import android.view.ViewGroup
import androidx.fragment.app.activityViewModels
import androidx.lifecycle.Observer
2020-04-11 13:20:30 -07:00
import com.geeksville.android.GeeksvilleApplication
2020-03-30 10:26:16 -07:00
import com.geeksville.android.Logging
import com.geeksville.mesh.NodeInfo
2020-03-11 18:13:44 -07:00
import com.geeksville.mesh.R
import com.geeksville.mesh.model.UIViewModel
2020-03-30 12:47:01 -07:00
import com.mapbox.geojson.Feature
import com.mapbox.geojson.FeatureCollection
import com.mapbox.geojson.Point
import com.mapbox.mapboxsdk.camera.CameraPosition
2020-03-30 13:06:41 -07:00
import com.mapbox.mapboxsdk.camera.CameraUpdateFactory
import com.mapbox.mapboxsdk.geometry.LatLng
import com.mapbox.mapboxsdk.geometry.LatLngBounds
2020-03-30 10:26:16 -07:00
import com.mapbox.mapboxsdk.maps.MapView
2020-04-07 12:13:50 -07:00
import com.mapbox.mapboxsdk.maps.MapboxMap
2020-03-30 11:56:59 -07:00
import com.mapbox.mapboxsdk.maps.Style
2020-03-30 15:00:18 -07:00
import com.mapbox.mapboxsdk.style.expressions.Expression
2020-03-30 12:47:01 -07:00
import com.mapbox.mapboxsdk.style.layers.Property
2020-03-30 15:00:18 -07:00
import com.mapbox.mapboxsdk.style.layers.Property.TEXT_ANCHOR_TOP
import com.mapbox.mapboxsdk.style.layers.Property.TEXT_JUSTIFY_AUTO
import com.mapbox.mapboxsdk.style.layers.PropertyFactory.*
2020-03-30 12:47:01 -07:00
import com.mapbox.mapboxsdk.style.layers.SymbolLayer
import com.mapbox.mapboxsdk.style.sources.GeoJsonSource
2020-03-11 14:45:49 -07:00
2020-04-07 11:27:51 -07:00
class MapFragment : ScreenFragment("Map"), Logging {
2020-04-07 12:13:50 -07:00
private val model: UIViewModel by activityViewModels()
2020-04-07 12:13:50 -07:00
private val nodeSourceId = "node-positions"
private val nodeLayerId = "node-layer"
private val labelLayerId = "label-layer"
private val markerImageId = "my-marker-image"
private val nodePositions = GeoJsonSource(nodeSourceId)
private val nodeLayer = SymbolLayer(nodeLayerId, nodeSourceId).withProperties(
iconImage(markerImageId),
iconAnchor(Property.ICON_ANCHOR_BOTTOM),
iconAllowOverlap(true)
)
private val labelLayer = SymbolLayer(labelLayerId, nodeSourceId).withProperties(
textField(Expression.get("name")),
textSize(12f),
textColor(Color.RED),
textVariableAnchor(arrayOf(TEXT_ANCHOR_TOP)),
textJustify(TEXT_JUSTIFY_AUTO),
textAllowOverlap(true)
)
private fun onNodesChanged(map: MapboxMap, nodes: Collection<NodeInfo>) {
val nodesWithPosition = nodes.filter { it.validPosition != null }
/**
* Using the latest nodedb, generate geojson features
*/
fun getCurrentNodes(): FeatureCollection {
// Find all nodes with valid locations
2020-04-07 12:13:50 -07:00
val locations = nodesWithPosition.map { node ->
val p = node.position!!
debug("Showing on map: $node")
2020-04-07 12:13:50 -07:00
val f = Feature.fromGeometry(
Point.fromLngLat(
p.longitude,
p.latitude
)
2020-04-07 12:13:50 -07:00
)
node.user?.let {
2020-05-29 13:58:38 -07:00
f.addStringProperty("name", it.longName)
}
f
2020-04-07 12:13:50 -07:00
}
return FeatureCollection.fromFeatures(locations)
}
2020-04-07 12:13:50 -07:00
fun zoomToNodes(map: MapboxMap) {
if (nodesWithPosition.isNotEmpty()) {
val update = if (nodesWithPosition.size >= 2) {
// Multiple nodes, make them all fit on the map view
val bounds = LatLngBounds.Builder()
// Add all positions
2020-04-09 12:22:41 -07:00
bounds.includes(nodesWithPosition.map { it.position!! }
.map { LatLng(it.latitude, it.longitude) })
CameraUpdateFactory.newLatLngBounds(bounds.build(), 150)
} else {
// Only one node, just zoom in on it
val it = nodesWithPosition[0].position!!
val cameraPos = CameraPosition.Builder().target(
LatLng(it.latitude, it.longitude)
).zoom(9.0).build()
CameraUpdateFactory.newCameraPosition(cameraPos)
}
map.animateCamera(update, 1000)
2020-04-07 12:13:50 -07:00
}
}
nodePositions.setGeoJson(getCurrentNodes()) // Update node positions
zoomToNodes(map)
2020-04-07 12:13:50 -07:00
}
2020-04-07 11:27:51 -07:00
override fun onCreateView(
inflater: LayoutInflater, container: ViewGroup?,
savedInstanceState: Bundle?
2020-04-11 13:20:30 -07:00
): View? {
// We can't allow mapbox if user doesn't want analytics
val id =
if ((requireContext().applicationContext as GeeksvilleApplication).isAnalyticsAllowed) {
// Mapbox Access token
R.layout.map_view
} else {
R.layout.map_not_allowed
}
return inflater.inflate(id, container, false)
}
2020-03-11 14:45:49 -07:00
2020-04-11 13:20:30 -07:00
var mapView: MapView? = null
/**
* Mapbox native code can crash painfully if you ever call a mapbox view function while the view is not actively being show
*/
private val isViewVisible: Boolean
get() = view != null && isResumed
override fun onViewCreated(viewIn: View, savedInstanceState: Bundle?) {
2020-06-10 17:19:17 -07:00
super.onViewCreated(viewIn, savedInstanceState)
2020-03-30 11:56:59 -07:00
2020-04-11 13:20:30 -07:00
// We might not have a real mapview if running with analytics
if ((requireContext().applicationContext as GeeksvilleApplication).isAnalyticsAllowed) {
val vIn = viewIn.findViewById<MapView>(R.id.mapView)
mapView = vIn
vIn.onCreate(savedInstanceState)
2020-03-30 11:56:59 -07:00
mapView?.let { v ->
// Each time the pane is shown start fetching new map info (we do this here instead of
// onCreate because getMapAsync can die in native code if the view goes away)
v.getMapAsync { map ->
if (view != null) { // it might have gone away by now
// val markerIcon = BitmapFactory.decodeResource(context.resources, R.drawable.ic_twotone_person_pin_24)
val markerIcon =
requireActivity().getDrawable(R.drawable.ic_twotone_person_pin_24)!!
map.setStyle(Style.OUTDOORS) { style ->
style.addSource(nodePositions)
style.addImage(markerImageId, markerIcon)
style.addLayer(nodeLayer)
style.addLayer(labelLayer)
}
//map.uiSettings.isScrollGesturesEnabled = true
//map.uiSettings.isZoomGesturesEnabled = true
// Provide initial positions
model.nodeDB.nodes.value?.let { nodes ->
onNodesChanged(map, nodes.values)
}
}
// Any times nodes change update our map
model.nodeDB.nodes.observe(viewLifecycleOwner, Observer { nodes ->
debug("Nodes updated! map visible = $isViewVisible")
if (isViewVisible)
onNodesChanged(map, nodes.values)
})
2020-04-11 13:20:30 -07:00
}
}
2020-03-30 10:26:16 -07:00
}
2020-03-11 14:45:49 -07:00
}
2020-04-07 11:27:51 -07:00
override fun onPause() {
2020-04-11 13:20:30 -07:00
mapView?.onPause()
2020-04-07 11:27:51 -07:00
super.onPause()
}
override fun onStart() {
super.onStart()
2020-04-11 13:20:30 -07:00
mapView?.onStart()
2020-04-07 11:27:51 -07:00
}
override fun onStop() {
2020-04-11 13:20:30 -07:00
mapView?.onStop()
2020-04-07 11:27:51 -07:00
super.onStop()
}
override fun onResume() {
super.onResume()
2020-04-11 13:20:30 -07:00
mapView?.onResume()
2020-04-07 11:27:51 -07:00
}
override fun onDestroy() {
2020-04-11 13:20:30 -07:00
mapView?.onDestroy()
2020-04-07 11:27:51 -07:00
super.onDestroy()
}
2020-03-11 14:45:49 -07:00
2020-04-07 11:27:51 -07:00
override fun onSaveInstanceState(outState: Bundle) {
2020-04-11 13:20:30 -07:00
mapView?.onSaveInstanceState(outState)
2020-04-07 11:27:51 -07:00
super.onSaveInstanceState(outState)
2020-03-11 14:45:49 -07:00
}
}
2020-04-07 11:27:51 -07:00