Integrate map support in Android applications based on MapBox

Integrate map support in Android applications based on MapBox

one, Introduction

MapBox is an open source vector map SDK. The promise of this framework is to provide quality rendering speed and smoothness in developing video games. If you are interested in integrating mapping capabilities into your application, then MapBox is one of the options worth considering.

2. Permission Settings

First of all, you can download the source code provided in this article from GitHub at https://github.com/sitepoint-editors/MapBox-App.

To use Mapbox in your application, you need an API access token. To do this, you need to create a Mapbox account first, which you can find at https://www.mapbox.com/studio/account/tokens.

Then, add the token to your Android app's strings.xml file:

  1. < string   name = "accessToken" > Your access token </ string >  

Next, add the following Internet and location permissions to the AndroidManifest.xml configuration file:

  1. < uses-permission   android:name = "android.permission.ACCESS_NETWORK_STATE" />    
  2. < uses-permission   android:name = "android.permission.INTERNET" />     
  3. < uses-permission   android:name = "android.permission.ACCESS_COARSE_LOCATION" />  
  4. < uses-permission   android:name = "android.permission.ACCESS_FINE_LOCATION" />  
  5. < uses-permission   android:name = "android.permission.ACCESS_WIFI_STATE" />  

Please note: The above permissions are required at runtime on Android Marshmallow (API 23) and higher.

3. Install MapBox

Next, open the build.gradle configuration file (the one under the Module: app folder) and add a dependency on MapBox. Please refer to the configuration code as shown below:

  1. repositories {
  2. mavenCentral()  
  3. }
  4. dependencies {
  5. . . .   
  6. compile('com.mapbox.mapboxsdk:mapbox-android-sdk:3.2.0@aar') {
  7. transitive = true  
  8. }
  9. compile ('com.mapbox.mapboxsdk:mapbox-android-directions:1.0.0@aar'){  
  10. transitive = true  
  11. }
  12. }

The first configuration in the code above describes Mapbox. The following configuration corresponds to the Directions library description, which is used to provide driving, walking, and biking functions to the application and support drawing their whereabouts on the map.

4. MapBox Layout

Next, open the layout file content_main.xml and replace the original content with the following:

  1. <? xml   version = "1.0"   encoding = "utf-8" ?>  
  2. < RelativeLayout   xmlns:android = "http://schemas.android.com/apk/res/android"  
  3. xmlns:mapbox = "http://schemas.android.com/apk/res-auto"  
  4. xmlns:tools = "http://schemas.android.com/tools"  
  5. android:layout_width = "match_parent"  
  6. android:layout_height = "match_parent"  
  7. tools:context = "com.example.valdio.mapboxintegration.MainActivity"  
  8. tools:showIn = "@layout/activity_main" >  
  9. < com.mapbox.mapboxsdk.views.MapView  
  10. android:id = "@+id/mapview"  
  11. android:layout_width = "fill_parent"  
  12. android:layout_height = "fill_parent"  
  13. mapbox:access_token = "@string/accessToken"   />  
  14. </RelativeLayout>  

Next, initialize the variables you will need to use in the MainActivity file:

  1. public class MainActivity extends AppCompatActivity {
  2. private MapView mapView = null ;
  3. private String MAPBOX_ACCESS_TOKEN = "" ;
  4. private DirectionsRoute currentRoute = null ;
  5. ...

Next, in the MainActivity.java file, delete the code that creates the toolbar and floating button in the onCreate() method and add the following code to initialize the map:

  1. String MAPBOX_ACCESS_TOKEN = getResources ().getString(R.string.accessToken);
  2. // Set up a standard Mapbox map
  3. MapView mapView = (MapView) findViewById(R.id.mapview);
  4. mapView.setAccessToken(MAPBOX_ACCESS_TOKEN);  
  5. mapView.setStyleUrl(Style.MAPBOX_STREETS); // specify the map style  
  6. mapView.setZoom(14); // zoom level  
  7. mapView.onCreate(savedInstanceState);

Mapbox requires the implementation of the Activity's lifecycle methods to avoid runtime errors; therefore, you need to add the following override function:

  1. @Override  
  2. protected void onStart() {  
  3. super.onStart();  
  4. mapView.onStart();  
  5. }
  6. @Override
  7. protected void onStop() {  
  8. super.onStop();  
  9. mapView.onStop();  
  10. }
  11. @Override  
  12. protected void onDestroy() {
  13. super.onDestroy();  
  14. mapView.onDestroy();  
  15. }  
  16. @Override  
  17. protected void onResume() {
  18. super.onResume();
  19. mapView.onResume();
  20. }
  21. @Override
  22. protected void onPause() {  
  23. super.onPause();
  24. mapView.onPause();  
  25. }  
  26. @Override
  27. protected void onSaveInstanceState(Bundle outState) {  
  28. super.onSaveInstanceState(outState);
  29. mapView.onSaveInstanceState(outState);
  30. }

OK. Now that Mapbox is configured, it's time to build the application.

5. Add markers to the map

Now, add the following code to the top of the onCreate method of MainActivity:

  1. @Override
  2. protected void onStart() {
  3. super.onStart();
  4. mapView.onStart();
  5. }
  6. @Override  
  7. protected void onStop() {
  8. super.onStop();
  9. mapView.onStop();
  10. }
  11. @Override
  12. protected void onDestroy() {
  13. super.onDestroy();
  14. mapView.onDestroy();
  15. }
  16. @Override
  17. protected void onResume() {  
  18. super.onResume();  
  19. mapView.onResume();  
  20. }  
  21. @Override  
  22. protected void onPause() {  
  23. super.onPause();
  24. mapView.onPause();  
  25. }  
  26. @Override  
  27. protected void onSaveInstanceState(Bundle outState) {
  28. super.onSaveInstanceState(outState);  
  29. mapView.onSaveInstanceState(outState);  
  30. }

The CameraPosition in this code is a very useful Mapbox class that can be used to set the position, angle, zoom, tilt, and other information of the user's view.

So far, our map looks like this:

6. Get device location

In order for Mapbox to access your location information, your location services must be enabled and your app should have permissions to use them. As mentioned above, these permissions are required at runtime on Android Marshmallow (API 23) and later.

Now, let's create a new function and add the code to get the current position:

  1. private void myLocation() {
  2. if (ActivityCompat.checkSelfPermission(this, Manifest.permission.ACCESS_FINE_LOCATION) != PackageManager.PERMISSION_GRANTED && ActivityCompat.checkSelfPermission(this, Manifest.permission.ACCESS_COARSE_LOCATION) != PackageManager.PERMISSION_GRANTED) {
  3. // TODO: Consider calling  
  4. // ActivityCompat#requestPermissions  
  5. // here to request the missing permissions, and then overriding  
  6. // public void onRequestPermissionsResult(int requestCode, String[] permissions,  
  7. // int[] grantResults)  
  8. // to handle the case where the user grants the permission. See the documentation  
  9. // for ActivityCompat#requestPermissions for more details.  
  10. return;  
  11. }
  12. mapView.setMyLocationEnabled(true);
  13. mapView.setMyLocationTrackingMode(MyLocationTracking.TRACKING_FOLLOW);  
  14. mapView.getMyLocation();  
  15. }

Then, add the following function call after the map initialization code:

  1. ...  
  2. mapView.onCreate(savedInstanceState);  
  3. myLocation();

In this case, we set the tracking mode to TRACKING_FOLLOW, which means that the user's location will be continuously monitored and the map will be continuously updated as changes occur.

7. Draw a route on the map

Now that we have created our markers and locations, it's time to draw our routes on the map.

This uses the second library (Directions library) imported earlier in the dependency settings, which is also developed by the Mapbox team.

Here's how it works:

1. Create two waypoint locations, one for the departure point and one for the destination.

2. Create a MapboxDirections generator object to make a network request to the Mapbox API with the waypoint location information corresponding to the starting and destination points, as well as configuration information about the directions (driving, walking, biking, etc.).

3. Asynchronously execute direction requests. The MapboxDirections class provides a built-in asynchronous request using the Retrofit API. Specifically, the enqueue() function is used to execute the request.

4. The OnResponse() method returns the Retrofit response result, which is a standard JSON API response.

5. The response body contains the location coordinates, which will be plotted on the map later.

6. Use the Mapbox Polyline function to draw the coordinates on the map. A polyline is a geometric feature that forms an open polyline by connecting multiple line segments end to end.

To use the Direction library, we need to get the device's location as the origin waypoint, and the location information of the destination waypoint specified by the user through a long press.

8. Add waypoint locations and add markers at the destination

Now, add the following code to the top of the onCreate method:

  1. mapView.setOnMapLongClickListener(new MapView.OnMapLongClickListener() {  
  2. @Override  
  3. public void onMapLongClick(LatLng point) {  
  4. //Remove previously added markers  
  5. //Marker is an annotation that shows an icon image at a geographical location  
  6. //so all markers can be removed with the removeAllAnnotations() method.
  7.   mapView.removeAllAnnotations();  
  8. // Set the origin waypoint to the devices location  
  9. Waypoint origin = new Waypoint(mapView.getMyLocation().getLongitude(), mapView.getMyLocation().getLatitude());  
  10. // Set the destination waypoint to the location point long clicked by the user  
  11. Waypoint destination = new Waypoint(point.getLongitude(), point.getLatitude());  
  12. // Add marker to the destination waypoint  
  13. mapView.addMarker(new MarkerOptions()  
  14. .position(new LatLng(point))  
  15. .title("Destination Marker")
  16. .snippet("My destination"));
  17. // Get route from API
  18. getRoute(origin, destination); }
  19.  
  20. });

9. Create a MapboxDirections network request and run it asynchronously

Next, create a method as shown below to obtain *** routing information:

  1. private void getRoute(Waypoint origin, Waypoint destination) {
  2. MapboxDirections directions = new MapboxDirections.Builder()  
  3. .setAccessToken(MAPBOX_ACCESS_TOKEN)
  4. .setOrigin(origin)
  5. .setDestination(destination)  
  6. .setProfile(DirectionsCriteria.PROFILE_WALKING)  
  7. .build();  
  8. directions.enqueue(new Callback < DirectionsResponse > () {  
  9. @Override  
  10. public void onResponse(Response < DirectionsResponse > response, Retrofit retrofit) {  
  11. // Display some info about the route  
  12. currentRoute = response .body().getRoutes().get(0);
  13. showToastMessage(String.format("You are %d meters \nfrom your destination", currentRoute.getDistance()));     
  14. // Draw the route on the map  
  15. drawRoute(currentRoute);  
  16. }
  17. @Override  
  18. public void onFailure(Throwable t) {
  19. showToastMessage("Error: " + t.getMessage());  
  20. }  
  21. });  
  22. }

10. Draw a polyline using coordinate points on the map

Add the following method to implement route drawing:

  1. private void drawRoute(DirectionsRoute route) {  
  2. // Convert List < Waypoint > into LatLng[]  
  3. List <Waypoint>   waypoints = route .getGeometry().getWaypoints();  
  4. LatLng[] point = new LatLng[waypoints.size()];  
  5. for (int i = 0 ; i <   waypoints.size (); i++) {  
  6. point[i] = new LatLng(  
  7. waypoints.get(i).getLatitude(),  
  8. waypoints.get(i).getLongitude());  
  9. }
  10.   // Draw Points on MapView  
  11. mapView.addPolyline(new PolylineOptions()  
  12. .add(point)  
  13. .color(Color.parseColor("#38afea"))  
  14. .width(5));  
  15. }
  16. private void showToastMessage(String message) {  
  17. Toast.makeText(this, message, Toast.LENGTH_SHORT).show();  
  18. }

OK, now run your project. Select two points on the map and you will see something similar to the following:

11. Summary

In this article, we introduced the basics of using the MapBox SDK and its Directory library. In fact, MapBox has much more and richer content that can be applied to your program, such as different map styles, customized vector map drawing, and so on.

As a supplement to this article, I also recommend another library Geocoding (https://github.com/mapbox/mapbox-geocoder-android). This library can convert coordinate information into a map, or vice versa. Of course, to learn more about MapBox, it is recommended to study its mobile application examples (https://www.mapbox.com/mobile/). You will find more libraries available at this URL.

<<:  Tencent architect Xiong Pujiang: WeChat's success is not accidental

>>:  Mobile game startup review: Why did the company fail 3 and a half years after its establishment?

Recommend

E-commerce marketing system full process planning

The marketing system is responsible for increasin...

A guide to improving conversion rates in online education communities

The conversion rate of online education is the mo...

Content Operation: How to connect content to users?

Operators often say "content connects users&...

Brand Marketing Promotion丨Why does IKEA sell products for free?

When it comes to marketing, IKEA has a lot to be ...

Activity launch mode (launchMode) detailed explanation

There are four Activity startup modes: standard, ...

The high-end development of the Android camp is stuck

The Android camp's progress in the high-end m...

How to acquire B-side customers?

I have a knowledge planet where I receive many qu...

How can live streaming make products popular?

A live broadcast room can generate sales of over ...

How much does it cost to rent a server for a year to develop an app?

APP is a word that has only appeared in our lives...