The best way to get started building an Android app with mapsforge is by studying the Samples app, which gives a complete overview of the capabilities of the mapsforge library.
Here, however, we go through a very basic example of an app that simply displays a map rendered from a mapfile in the built-in render style.
Mapsforge currently requires disabling hardware acceleration for the map view. This can be controlled in various levels, better described in Android documentation.
You'll need to have the appropriate permissions in manifest for tile cache to work properly:
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
Before you make any calls on the mapsforge library, you need to initialize the AndroidGraphicFactory. Behind the scenes, this initialization process gathers a bit of information on your device, such as the screen resolution, that allows mapsforge to automatically adapt the rendering for the device.
AndroidGraphicFactory.createInstance(this.getApplication());
If you forget this step, your app will crash. You can place this code, like in the Samples app, in the Android Application class. This ensures it is created before any specific activity. But it can also be created in the onCreate() method in your activity.
A MapView is an Android View (or ViewGroup) that displays a mapsforge map. You can have multiple MapViews in your app or even a single Activity. Have a look at the BasicMapViewerXml on how to create a MapView using the Android XML Layout definitions. Here we create a MapView on the fly and make the content view of the activity the MapView. This means that no other elements make up the content of this activity.
this.mapView = new MapView(this);
setContentView(this.mapView);
We then make some simple adjustments, such as showing a scale bar and setting up the zoom buttons:
this.mapView.setClickable(true);
this.mapView.getMapScaleBar().setVisible(true);
this.mapView.setBuiltInZoomControls(true);
this.mapView.getMapZoomControls().setZoomLevelMin((byte) 10);
this.mapView.getMapZoomControls().setZoomLevelMax((byte) 20);
To avoid redrawing all the tiles all the time, we need to set up a tile cache. A utility method helps with this:
this.tileCache = AndroidUtil.createTileCache(this, "mapcache",mapView.getModel().displayModel.getTileSize(),1f,this.mapView.getModel().frameBufferModel.getOverdrawFactor());
Now we need to set up the process of displaying a map. A map can have several layers, stacked on top of each other. A layer can be a map or some visual elements, such as markers. Here we only show a map based on a mapsforge map file. For this we need a TileRendererLayer. A tileRendererLayer needs a tileCache to hold the generated map tiles, a mapfile from which the tiles are generated and rendertheme that defines the appearance of the map:
MapDataStore mapDataStore = new MapFile(getMapFile());
this.tileRendererLayer = new TileRendererLayer(tileCache, mapDataStore,
this.mapView.getModel().mapViewPosition, false, true, AndroidGraphicFactory.INSTANCE);
tileRendererLayer.setXmlRenderTheme(InternalRenderTheme.OSMARENDER);
On its own a tileRendererLayer does not know where to display the map, so we need to associate it with our mapView:
this.mapView.getLayerManager().getLayers().add(tileRendererLayer);
The map also needs to know which area to display and at what zoom level. This is set via a MapViewPosition:
this.mapView.getModel().mapViewPosition.setCenter(new LatLong(52.517037, 13.38886));
this.mapView.getModel().mapViewPosition.setZoomLevel((byte) 12);
Refer to the Samples app on how to read the initial position out of a mapfile or how to store the current position when a user leaves your app.
Whenever your activity changes, some cleanup operations have to be performed lest your app runs out of memory.
@Override
protected void onDestroy() {
super.onDestroy();
this.mapView.destroyAll();
}
Here comes the whole as a single file:
package org.mapsforge.applications.android.simplemapviewer;
import java.io.File;
import org.mapsforge.core.model.LatLong;
import org.mapsforge.map.android.graphics.AndroidGraphicFactory;
import org.mapsforge.map.android.util.AndroidUtil;
import org.mapsforge.map.android.view.MapView;
import org.mapsforge.map.layer.cache.TileCache;
import org.mapsforge.map.layer.renderer.TileRendererLayer;
import org.mapsforge.map.datastore.MapDataStore;
import org.mapsforge.map.reader.MapFile;
import org.mapsforge.map.rendertheme.InternalRenderTheme;
import android.app.Activity;
import android.os.Bundle;
import android.os.Environment;
public class MainActivity extends Activity {
// name of the map file in the external storage
private static final String MAPFILE = "germany.map";
private MapView mapView;
private TileCache tileCache;
private TileRendererLayer tileRendererLayer;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
AndroidGraphicFactory.createInstance(this.getApplication());
this.mapView = new MapView(this);
setContentView(this.mapView);
this.mapView.setClickable(true);
this.mapView.getMapScaleBar().setVisible(true);
this.mapView.setBuiltInZoomControls(true);
this.mapView.getMapZoomControls().setZoomLevelMin((byte) 10);
this.mapView.getMapZoomControls().setZoomLevelMax((byte) 20);
// create a tile cache of suitable size
this.tileCache = AndroidUtil.createTileCache(this, "mapcache",
mapView.getModel().displayModel.getTileSize(), 1f,
this.mapView.getModel().frameBufferModel.getOverdrawFactor());
}
@Override
protected void onStart() {
super.onStart();
this.mapView.getModel().mapViewPosition.setCenter(new LatLong(52.517037, 13.38886));
this.mapView.getModel().mapViewPosition.setZoomLevel((byte) 12);
// tile renderer layer using internal render theme
MapDataStore mapDataStore = new MapFile(getMapFile());
this.tileRendererLayer = new TileRendererLayer(tileCache, mapDataStore,
this.mapView.getModel().mapViewPosition, false, true, AndroidGraphicFactory.INSTANCE);
tileRendererLayer.setXmlRenderTheme(InternalRenderTheme.OSMARENDER);
// only once a layer is associated with a mapView the rendering starts
this.mapView.getLayerManager().getLayers().add(tileRendererLayer);
}
@Override
protected void onDestroy() {
super.onDestroy();
this.mapView.destroyAll();
}
private File getMapFile() {
File file = new File(Environment.getExternalStorageDirectory(), MAPFILE);
return file;
}
}