Skip to content

Sample app for using MapLibre with Android Auto and Automotive

License

Notifications You must be signed in to change notification settings

flitsmeister/MapLibre-Android-Auto-Sample

Repository files navigation

MapLibre Sample app for Android Auto and Automotive

This sample app will demonstrate how to use MapLibre on Android Auto and Android Automotive. (Also known as AAOS, Android for Cars)

So how does this work?

Android Auto uses a Template system. It will not allow you to use regular views. MapLibre will provide the map as a View. If we want to use this MapView on Android Auto, we have to render it on the provided Surface. See also: https://developer.android.com/training/cars/apps#declare-surface-permission

To get the MapView to render on the surface, we do the following; We render the MapView offscreen, create a 'screenshot' of this view. And we'll draw this Bitmap to the Surface via a Canvas. Repeat the process 30 times per second, and you've got a working MapView on Android Auto.

This process is not very efficient, but this is what we got to work with. Maybe in the future, we can find a solution to make MapLibre draw directly unto the Surface, and thereby skip the workaround. Which is drawing a Bitmap on a Canvas.

How is the code structured?

The map can be drawn on Android Auto, and Android Automotive. A lot of code (most of it), can be shared between the two. Therefore, this bulk of the code is in a shared module, called car_common. This is used by the app (Smartphone app, uses Android Auto) and the automotive module. The automotive module actually has no code at all, just some config and resources. Everything it uses comes from the car_common module.

flowchart TD
    A[car_common] 
    B[app]
    C[automotive]
    
    B --> A
    C --> A
Loading

Enough words; Show me the code!

We have a class called CarMapContainer, which will take care of the MapView code;
This is to create the MapView, enable texture mode, and add to the WindowManager. This is needed to ensure the MapView will render, and we can get the Texture later for rendering it onto the Canvas.

val mapView = MapView(carContext, MapLibreMapOptions.createFromAttributes(carContext).apply {
    // Set the textureMode to true, so a TextureView is created
    // We can extract this TextureView to draw on the Android Auto surface
    textureMode(true)
}).apply {
    setLayerType(View.LAYER_TYPE_HARDWARE, Paint())
}
//(...)
carContext.windowManager.addView(
    mapView /* the MapView */,
    getWindowManagerLayoutParams()
)

Furthermore, we have a class called CarMapRenderer, which will render the Map on the provided Surface.

//Make sure to get the Surface by registering as SurfaceCallback
override fun onCreate(owner: LifecycleOwner) {
    super.onCreate(owner)
    try {
        carContext.appManager.setSurfaceCallback(this) //Make sure we get the Surface from Android Auto
    } catch (e: Exception) {
        Log.e(LOG_TAG, "Could not set surface callback", e)
        return
    }
}

//In onSurfaceAvailable, we get the Surface onto which we need to draw the map

    override fun onSurfaceAvailable(surfaceContainer: SurfaceContainer) {
        Log.v(LOG_TAG, "CarMapRenderer.onSurfaceAvailable")
        this.surfaceContainer = surfaceContainer
        mapContainer.setSurfaceSize(surfaceContainer.width, surfaceContainer.height)

        // // Start rendering the map on the Android Auto surface when any behavior changes are detected in the map.
        mapContainer.mapViewInstance?.apply {
            addOnDidBecomeIdleListener { drawOnSurface() }
            addOnWillStartRenderingFrameListener {
                drawOnSurface()
            }
        }
        runOnMainThread {
            // Start drawing the map on the android auto surface
            drawOnSurface()
        }
    }

//Using a canvas, draw the map
private fun drawOnSurface() {
    val mapView = mapContainer.mapViewInstance ?: return
    val surface = surfaceContainer?.surface ?: return
    val canvas = surface.lockHardwareCanvas()
    drawMapOnCanvas(mapView, canvas)
    surface.unlockCanvasAndPost(canvas)
}

//The actual drawing of the map
private fun drawMapOnCanvas(mapView: MapView, canvas: Canvas) {
    val mapViewTextureView = mapView.takeIf { it.childCount > 0 }?.getChildAt(0) as? TextureView
    mapViewTextureView?.bitmap?.let {
        canvas.drawBitmap(it, 0f, 0f, null)
    }
}

Help! I don't see any MapTiles!

That's correct, you can fix this by setting your own Style. Set this in the following places:

Set your own Style in: CarMapContainer.kt (car_common);

getMapAsync {
    mapViewInstance = this
    mapLibreMapInstance = it
    it.setStyle(
        //TODO: Set your own style here
        Style.Builder().fromJson(ResourceUtils.readRawResource(carContext, R.raw.local_style))
    )
}

And in MainActivity (app);

private fun initMap(map: MapLibreMap) {
    try {
        map.setStyle(
            //TODO: Set your own style here!
            Style.Builder().fromJson(ResourceUtils.readRawResource(this, R.raw.local_style))
        )
    } catch (e: Exception) {
        Log.e("MapLibreCar", "Error setting local style", e)
    }
}

Main MapLibre website

https://maplibre.org/

Discussing

This project can be discussed on the OSM-US Slack, in channel #maplibre-android, invite link: https://slack.openstreetmap.us/

Contributing

We welcome contributions to this project. Please read the Contributing Guidelines for more information.

License

MIT

Contributor

Flitsmeister

MapLibre

VietMap

Copyright (c) 2025 MapLibre contributors

About

Sample app for using MapLibre with Android Auto and Automotive

Resources

License

Code of conduct

Security policy

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published

Contributors 3

  •  
  •  
  •  

Languages