Intermediate: Huawei Activity Identification Service | HMS location kit

Search This thread

lokeshsuryan

Member
Mar 2, 2021
45
1
1621338461277.png

Introduction

Nowadays, everybody is using smartphones to do daily tasks like taking photos, looking up movie times, making calls etc. The best part of Android apps on mobile phones is that they are trying more and more to get to know their users. Many applications today take users' locations to provide users with locational feeds. One common example is a normal news app, where the app takes your current location and shows the news by location.

If you're a developer, you need to understand users better to give users a better experience of the application. You should know at any time what your users do. The more you know about your users, the better application for your users can build. For example, a distance calculator app lunches by itself when you start driving yourcar or bike and stops when you stop driving. Health and fitness app also uses this service to determine how many meters/kilometers you have covered on particular day.

What is Activity Identification Service?

Activity Identification Service does the heavy lifting using acceleration sensor, cellular network information and magnetometer from device to identify user’s current activity. Your app receives a list of detected activities, each of which includes possibility and identity properties.

The Activity Identification Service can detect following activities:

  • STILL: When the mobile device will be still, that is, the user is either sitting at someplace or the mobile device is having no motion, then the Activity Recognition Client will detect the STILL activity.
  • FOOT: When the mobile device is moving at a normal speed , that is, the user carrying the mobile device is either walking or running then the Activity Identification Service will detect the FOOT activity.
  • WALKING: This is a sub-activity of the FOOT activity which is detected by the Activity Identification Service when the user carrying the mobile device is walking.
  • RUNNING: This is also a sub-activity of FOOT activity which is detected by the Activity Recognition Client when the user carrying the mobile device is running.
  • VEHICLE: This activity detected when the mobile device is on the bus or car or some other kind of vehicle or the user holding the mobile device is present in the vehicle.
  • OTHERS: The Activity Identification service will show this result when the device is unable to detect any activity on the mobile device.
In this article, we will create a sample application to show user activity. When user clicks start button, we will identify user activity status along with possibility level and display the status in Textview and Imageview. And when user clicks on stop button, we will stop requesting activity identification updates.
1621338495833.png
1621338523120.png


Development Overview

Prerequisite


1. Must have a Huawei Developer Account.

2. Must have Android Studio 3.0 or later.

3. Must have Huawei phone running EMUI 5.0 or later.

4. EMUI 5.0 or later.

Software Requirements

1. Java SDK 1.7 or later.

2. Android 5.0 or later.

Preparation

1. Create an app or project in the Huawei App Gallery Connect.

2. Provide the SHA Key and App Package name of the project in App Information Section and enable the Location Kit API.

3. Download the agconnect-services.json file.

4. Create an Android project.

Integration

1. Add below to build.gradle (project) file under buildscript/repositories and allprojects/repositories.

Code:
// Top-level build file where you can add configuration options common to all sub-projects/modules.
buildscript {
    repositories {
        google()
        jcenter()
        maven {url 'https://developer.huawei.com/repo/'}
    }
    dependencies {
        classpath "com.android.tools.build:gradle:4.0.1"
        classpath 'com.huawei.agconnect:agcp:1.4.2.300'
        // NOTE: Do not place your application dependencies here; they belong
        // in the individual module build.gradle files
    }
}


allprojects {
    repositories {
        google()
        jcenter()
        maven {url 'https://developer.huawei.com/repo/'}
    }
}


task clean(type: Delete) {
    delete rootProject.buildDir
}

2. Add below to build.gradle (app) file, under dependencies to use the Location kit SDK.

Code:
apply plugin: 'com.huawei.agconnect'
dependencies {
  
     implementation 'com.huawei.hms:location:5.0.5.300'
}

Tip: Minimum android version supported for these kits is 19.

3. Add below permissions to manifest file.

For version earlier than android Q

Code:
<uses-permission android:name="com.huawei.hms.permission.ACTIVITY_RECOGNITION"/>

For version Android Q and later

Code:
<uses-permission android:name="android.permission.ACTIVITY_RECOGNITION" />


Note: The above permissions are dangerous permission and need to be requested dynamically. Requesting permission dynamically is not covered in this article.

Development

We need to register static broadcast receiver in AndroidManifest.xmlto listen to activity status update identified by Activity Identification Service.
Code:
<receiver
     android:name=".LocationReceiver"
     android:exported="true">
     <intent-filter>
         <action android:name="com.huawei.hmssample.location.LocationBroadcastReceiver.ACTION_PROCESS_LOCATION" />
     </intent-filter>
</receiver>

Now the next step is to add the UI for our Main Activity. In our application, we will be having one TextView to display the name of the current activity and display corresponding image on ImageView and one TextView to display the possibility of Activity. We will have two Buttons to start and stop activity identification tracking. So, the activity_main.xml file looks something like this:

XML:
<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
     xmlns:app="http://schemas.android.com/apk/res-auto"
     xmlns:tools="http://schemas.android.com/tools"
     android:layout_width="match_parent"
     android:layout_height="match_parent"
     android:background="#FAF0E6"
     tools:context=".MainActivity">

     <ImageView
         android:id="@+id/ivDisplay"
         android:layout_width="250dp"
         android:layout_height="250dp"
         android:layout_centerInParent="true"
         android:scaleType="centerInside"
         android:src="@drawable/ic_still" />

     <TextView
         android:id="@+id/tvidentity"
         android:layout_width="wrap_content"
         android:layout_height="wrap_content"
         android:layout_below="@+id/ivDisplay"
         android:layout_marginTop="5dp"
         android:textStyle="bold"
         android:textColor="#192841"
         android:textSize="25sp"
         android:layout_centerHorizontal="true"
         app:layout_constraintBottom_toBottomOf="parent"
         app:layout_constraintLeft_toLeftOf="parent"
         app:layout_constraintRight_toRightOf="parent"
         app:layout_constraintTop_toTopOf="parent" />
     <TextView
         android:id="@+id/tvpossiblity"
         android:layout_width="wrap_content"
         android:layout_height="wrap_content"
         android:layout_below="@+id/tvidentity"
         android:textSize="20sp"
         android:textColor="#192841"
         android:layout_centerHorizontal="true"
         app:layout_constraintBottom_toBottomOf="parent"
         app:layout_constraintLeft_toLeftOf="parent"
         app:layout_constraintRight_toRightOf="parent"
         app:layout_constraintTop_toTopOf="parent" />

     <LinearLayout
         android:layout_width="match_parent"
         android:layout_height="wrap_content"
         android:layout_alignParentBottom="true"
         android:orientation="horizontal">
         <Button
             android:layout_width="wrap_content"
             android:layout_height="wrap_content"
             android:id="@+id/bStart"
             android:layout_weight="1"
             android:layout_margin="5dp"
             android:text="Start Tracking"
             android:textColor="@color/upsdk_white"
             android:background="#192841"/>
         <Button
             android:layout_width="wrap_content"
             android:layout_height="wrap_content"
             android:id="@+id/bStop"
             android:layout_margin="5dp"
             android:layout_weight="1"
             android:text="Stop Tracking"
             android:textColor="@color/upsdk_white"
             android:background="#192841"/>
     </LinearLayout>

</RelativeLayout>

Now let’s create instance of ActivityIdentificationService in onCreate() method of MainActivity.java

Java:
private PendingIntent mPendingIntent;
private ActivityIdentificationService identificationService;

@Override
protected void onCreate(Bundle savedInstanceState) {
     super.onCreate(savedInstanceState);        intializeTracker();  }

private void intializeTracker() {
     identificationService = ActivityIdentification.getService(this);
     mPendingIntent = obtainPendingIntent();
}

To obtain PendingIntent object

Java:
private PendingIntent obtainPendingIntent() {
     Intent intent = new Intent(this, LocationReceiver.class);
     intent.setAction(LocationReceiver.ACTION_NAME);
     return PendingIntent.getBroadcast(this, 0, intent, PendingIntent.FLAG_UPDATE_CURRENT);
}

When user clicks on Start Tracking Button, we will request activity identification updates by calling createActivityIdentificationUpdates() method.

identificationService.createActivityIdentificationUpdates(5000, mPendingIntent)

.addOnSuccessListener(new OnSuccessListener<Void>() {
@override
public void onSuccess(Void aVoid) {
Log.i(TAG, "createActivityIdentificationUpdates onSuccess");
}
})
// Define callback for request failure.
.addOnFailureListener(new OnFailureListener() {
@override
public void onFailure(Exception e) {
Log.e(TAG, "createActivityIdentificationUpdates onFailure:" + e.getMessage());
}
});
This method has two parameters: detectionIntervalMillis and pendingIntent, which indicate the detection interval (in milliseconds) and action to perform, respectively.



On click of Stop Tracking Button, we will stop activity identification updates.
Java:
identificationService.deleteActivityIdentificationUpdates(mPendingIntent)


         .addOnSuccessListener(new OnSuccessListener<Void>() {
             @Override
             public void onSuccess(Void aVoid) {
                 Log.i(TAG, "deleteActivityIdentificationUpdates onSuccess");
             }
         })
      
         .addOnFailureListener(new OnFailureListener() {
             @Override
             public void onFailure(Exception e) {
                 Log.e(TAG, "deleteActivityIdentificationUpdates onFailure:" + e.getMessage());
             }
         });

Finally, We can get activity identification result (containing identity and possibility) from intent received by the broadcast receiver.
Java:
public class LocationReceiver extends BroadcastReceiver {

     public static final String ACTION_NAME = "com.huawei.hms.location.ACTION_PROCESS_LOCATION";
    
     @Override
     public void onReceive(Context context, Intent intent) {
         if (intent != null) {
             final String action = intent.getAction();
             if (ACTION_NAME.equals(action)) {
                 // Obtains ActivityIdentificationResponse from extras of the intent sent by the activity identification service.
                 ActivityIdentificationResponse activityIdentificationResponse = ActivityIdentificationResponse.getDataFromIntent(intent);
                 if(activityIdentificationResponse!= null) {
                
                     List<ActivityIdentificationData> list = activityIdentificationResponse.getActivityIdentificationDatas();
                    
                     ActivityIdentificationData identificationData = list.get(list.size() -1);
                     int identificationIdentity =  identificationData.getIdentificationActivity();
                     int possibility =  identificationData.getPossibility();
                     Intent i = new Intent("activityIdentificationReceiver");
                     i.putExtra("identity", identificationIdentity);
                     i.putExtra("possibility", possibility);
                     context.sendBroadcast(i);
                 }
             }
         }
     }
 }
getActivityIdentificationDatas() API is used to obtain the list of activitiy identification list. The activity identifications are sorted by most probable activity first.

We have created Utils.java class to obtain activity status from identity code obtained from LocationReceiver
Java:
public class Utils {

    public static String getActivityIdentityName(int code) {

        switch(code) {
            case ActivityIdentificationData.VEHICLE:
                return "VEHICLE";
            case ActivityIdentificationData.BIKE:
                return "BIKE";
            case ActivityIdentificationData.FOOT:
                return "FOOT";
            case ActivityIdentificationData.STILL:
                return "STILL";
            case ActivityIdentificationData.OTHERS:
                return "OTHERS";
            case ActivityIdentificationData.WALKING:
                return "WALKING";
            case ActivityIdentificationData.RUNNING:
                return "RUNNING";
            default:
                return "No Data Available";
        }
    }

    public static int getActivityIdentityDrawableID(int code) {

        switch(code) {
            case ActivityIdentificationData.VEHICLE:
                return R.drawable.ic_driving;
            case ActivityIdentificationData.BIKE:
                return R.drawable.ic_on_bicycle;
            case ActivityIdentificationData.FOOT:
                return R.drawable.ic_still;
            case ActivityIdentificationData.STILL:
                return R.drawable.ic_still;
            case ActivityIdentificationData.OTHERS:
                return R.drawable.ic_unknown;
            case ActivityIdentificationData.WALKING:
                return R.drawable.ic_walking;
            case ActivityIdentificationData.RUNNING:
                return R.drawable.ic_running;
            default:
                return R.drawable.ic_unknown;
        }
    }
}
Code snippet of MainActivity.java
Java:
public class MainActivity extends AppCompatActivity {
    private static final String TAG = "MainActivity";
    private ActivityConversionRequest request;
    private Button bStart, bStop;
    private TextView tvPossiblity, tvIdentity;
    private ImageView ivDisplay;
    private PendingIntent mPendingIntent;
    private ActivityIdentificationService identificationService;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        intializeTracker();
        bStart = findViewById(R.id.bStart);
        bStop = findViewById(R.id.bStop);
        tvIdentity = findViewById(R.id.tvidentity);
        tvPossiblity = findViewById(R.id.tvpossiblity);
        ivDisplay = findViewById(R.id.ivDisplay);
        bStart.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View view) {


                identificationService.createActivityIdentificationUpdates(5000, mPendingIntent)

                        .addOnSuccessListener(new OnSuccessListener<Void>() {
                            @Override
                            public void onSuccess(Void aVoid) {
                                Log.i(TAG, "createActivityIdentificationUpdates onSuccess");
                            }
                        })

                        .addOnFailureListener(new OnFailureListener() {
                            @Override
                            public void onFailure(Exception e) {
                                Log.e(TAG, "createActivityIdentificationUpdates onFailure:" + e.getMessage());
                            }
                        });

            }
        });

        bStop.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View view) {

                identificationService.deleteActivityIdentificationUpdates(mPendingIntent)

                        .addOnSuccessListener(new OnSuccessListener<Void>() {
                            @Override
                            public void onSuccess(Void aVoid) {
                                Log.i(TAG, "deleteActivityIdentificationUpdates onSuccess");
                            }
                        })

                        .addOnFailureListener(new OnFailureListener() {
                            @Override
                            public void onFailure(Exception e) {
                                Log.e(TAG, "deleteActivityIdentificationUpdates onFailure:" + e.getMessage());
                            }
                        });

            }
        });
    }

    private void intializeTracker() {
        identificationService = ActivityIdentification.getService(this);
        mPendingIntent = obtainPendingIntent();
    }

    // Get PendingIntent associated with the custom static broadcast class LocationBroadcastReceiver.
    private PendingIntent obtainPendingIntent() {
        Intent intent = new Intent(this, LocationReceiver.class);
        intent.setAction(LocationReceiver.ACTION_NAME);
        return PendingIntent.getBroadcast(this, 0, intent, PendingIntent.FLAG_UPDATE_CURRENT);
    }

    @Override
    protected void onResume() {
        super.onResume();
        IntentFilter filter = new IntentFilter();
        filter.addAction("activityIdentificationReceiver");
        registerReceiver(mIdentificationReceiver , filter);
    }

    @Override
    protected void onPause() {
        super.onPause();
        try {
            if(mIdentificationReceiver != null){
                unregisterReceiver(mIdentificationReceiver);
            }
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

    private BroadcastReceiver mIdentificationReceiver = new BroadcastReceiver(){

        @Override
        public void onReceive(Context context, Intent intent) {
            int possibility = intent.getIntExtra("possibility", 0);
            int identity = intent.getIntExtra("identity", 103);
            tvIdentity.setText(Utils.getActivityIdentityName(identity));
            tvPossiblity.setText("Possibility : " +  String.valueOf(possibility));

            ivDisplay.setImageResource(Utils.getActivityIdentityDrawableID(identity));
        }
    };
}
Tips and Tricks
1.During writing of this article, the activity identification service cannot identify the cycling and riding activities on devices outside the Chinese mainland.
2. ACTIVITY_RECOGNITION is dangerous permission and should be requested dynamically.

Conclusion
In this article, we have learnt how to use the Activity Identification Service in our application to determine the activities that users are doing at any given time. The Activity Identification Service determines the ongoing activities based on a possibility value that tells you which activity is currently taking place.
Hope you found this story useful and interesting.
Happy coding! 😃 💻

References
https://developer.huawei.com/consum...troduction-0000001050706106-V5?ha_source=hms1
 
Last edited: