Skip to content

Basic Usage

This section describes the VIRNECT Track setup for Android application development with Android Studio. All code samples are written in Kotlin.

Limitations

This guide describes the essential module integration and initialization of VIRNECT Track for an existing Application. For a guide on visualization of tracking results, please refer to the Sample Project implementation.

1. Import Library

  • Open or create a Android Studio project
  • Download the AAR library and extract the binaries to your project folder
  • In the Android Studio menu, select File | Project Structure
  • Select section Modules and add a new module with +
  • In the dialog, select Import JAR/.AAR Package and click on Next
  • Select the path to the downloaded Track.aar file and set the module name Track
  • Ensure the line include ':Track' was added to your settings.gradle file
  • Add the line implementation project(":Track") to the build.gradle file of your app module
  • Perform a gradle-sync

After setting up the project configuration, VIRNECT Track can be imported to your APP with the following statement:

import com.virnect.Track

2. Request Permissions

Please ensure to request the following APP permissions before using VIRNECT Track in your APP.

  • CAMERA needed to access the back-facing camera
  • WRITE_EXTERNAL_STORAGE needed to write target files to cache
  • READ_EXTERNAL_STORAGE needed to read target files from file system or cache

3. Register the License Key

To enable usage of the Track framework, your License Key needs to be registered first:

// Set license key
Track.setLicenseKey(licenseKey)

Calling this method will perform a network request to check the validity of your key.

Internet access required

Your APP always needs internet access to validate the license key before running the Track framework.

How to improve key security

When your source code is managed via a version control software (e.g. git), the defined license key might be exposed. Also, when the application is compiled without further code obfuscation, the API key may be extracted with reverse engineering attacks. To protect your license key, we strongly encourage you to encrypt your license key instead of saving it as clear text.

4. Specify Tracking Targets

Download or train one or multiple trackable targets. See Section Target Trainer for more detail.

To specify which targets should be tracked, use the designated Target name without the TRACK_ prefix:

// Setup Target configuration for framework
Track.setTargetNames(arrayOf<String>("VARIMG_000", "VARQR_000"))

Place the target files in a designated targets folder on your device, or add them to a resources folder of your APP:

When the targets are saved on the device, they can be accessed directly:

// Use direct path to file system
Track.setTargetDataPath("/sdcard/TrackTargets/")

When the targets are embedded as a Resource, the FileAccessHelper needs to be used:

import com.virnect.FileAccessHelper

// Copy target files from APP assets to cache directory 
Track.setTargetDataPath(FileAccessHelper.prepareTargetData(this, "assetPath/targetFolder/"))

5. Initialize the Framework

After setting up the configuration, the framework can be initialized with the application context

// Initialize Track framework
val success = Track.initialize(context)

The initialize function will return the result of the initialization process. If the initialization returns false, the reason can be determined by retrieving the framework log messages.

Framework initialization will enable the back-facing camera as an image source. Each camera frame will be processed by the Track framework automatically.

To use a file sequence as an image source please refer to this guide.

6. Register a Callback Function

A callback function can be registered to retrieve the latest tracking result. The framework calls this callback function after processing each frame.

// Define callback function
val callbackFunction  = Runnable{
    // Prepare target result array from "targets" string array
    val trackerResults = targets.map { Common.TrackerResult(it) }.toTypedArray()

    // Retrieve tracking results with empty array
    Track.getTrackingResult(trackerResults)

    // Initialize result placeholder
    var logText = ""

    // Get status log for each tracking result
    for (i in trackerResults.indices) 
        trackerResults[i].also {
            logText += "%s - %s\n".format(it.mTargetName, it.mStatus.name)
        }

    // Display tracking result as text
    runOnUiThread {
        TextView.text = logText;
    }
}

Track.setProcessingCallback(callbackFunction)

Warning

Note that any UI access from within the callback is restricted and need to be performed on the UI thread

7. Start processing

Call Track.startProcessing() to start VIRNECT Track, Track.stopProcessing() to stop again.

8. Handle Android Lifecycle Events

To comply with the Android Activity Lifecycle, the processing loop of the tracking framework needs to be stopped when the app gets paused. When the app exits, the framework needs to be closed to free all resources:

/**
 * Stop processing on pause
 */
override fun onPause() {
    super.onPause()
    Track.stopProcessing()
}

/**
 * Free all resources on destruction
 */
override fun onStop() {
    super.onStop()
    Track.close()
}

Minimum working example:

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
// by VIRNECT CO., LTD.

import android.content.Context
import android.widget.TextView
import com.virnect.Common
import com.virnect.Track

fun initializeAndStartTrackFramework( context: Context){
    // Set license key
    Track.setLicenseKey("XXXX-XXXX-XXXX-XXXX")
    // Setup Target configuration for framework
    Track.setTargetNames(arrayOf<String>("VARIMG_000"))
    // Use direct path to file system
    Track.setTargetDataPath("/sdcard/target/")
    // Initialize Track framework
    Track.initialize(context)
    // Define callback function
    val callbackFunction  = Runnable{
        // Prepare target result array from "targets" string array
        val trackerResults = targets.map { Common.TrackerResult(it) }.toTypedArray()
        // Retrieve tracking results with empty array
        Track.getTrackingResult(trackerResults)
        // Initialize result placeholder
        var logText = ""
        // Get status log for each tracking result
        for (i in trackerResults.indices)
            trackerResults[i].also {
                logText += "%s - %s\n".format(it.mTargetName, it.mStatus.name)
            }
        // Display tracking result as text
        runOnUiThread {
            findViewById<TextView>(R.id.textView).text = logText;
        }
    }
    // Add callback function
    Track.setProcessingCallback(callbackFunction)
    // Start framework
    Track.startProcessing()
}
Back to top