Skip to content

Latest commit

 

History

History
320 lines (218 loc) · 15.1 KB

README.md

File metadata and controls

320 lines (218 loc) · 15.1 KB

UnityPlayerActivity Extension

The solution to Android plugins collision.

Support this project for Unity to implement

Vote here: Unity Feedback Ticket

Watch it here: Unity Community Post

Description

When we want to create a Unity android plugin that will use any method of the Activity class we are forced to extend UnityPlayerActivity class.

That is not a problem for us, it's possible and simple. Following Unity documentation to extend this class and implement it in our project should take us 10-20 minutes to have a simple plugin in our project running.

But what happen when we want to create antoher plugin that also will need to listen Activity methods. Well nevermind, we can merge two plugins into one right? Messy but works...

But the real problem comes when 3rd party plugins also extend from UnityPlayerActivity and merging our plugins and others it would be unsustainable and a problem for us. Imagine that every time a plugin provider release a new version that changes something of it class extending from UnityPlayerActivity, we will have to:

  • Find plugin source code (if it's open source).
  • Create our UnityActivityPlayer.
  • Merge plugin codes with our UnityPlayerActivity custom class.
  • Build our .aar
  • Modify the AndroidManifest.xml
  • Test that all lives in harmony

Really tedious right?

And if you don't know anything about Android or Java or exporting .aar files you will be in a bigger problem.

What this project do?

This project solves this problem. But to really work it must be a standard, and what better way to be a standard than to implement Unity itself.

To give you a summary of what this project does, it provide the way to listen an Activity lifecycle method without extending from UnityPlayerActivity.

Bad Example

Imagine you want to create your first android plugin for Unity, let's call it MyHelloWorldPlugin. It will be an awesome plugin that will Log every Activity method that is executed.

Since we need to know about Activity events, following Unity documentation tells us to extend UnityPlayerActivity to have interaction with Android OS and Unity Application

When developing a Unity Android application, it is possible to extend the standard UnityPlayerActivity class (the primary Java class for the Unity Player on Android, similar to AppController.mm on Unity iOS) by using plug-ins. An application can override any and all of the basic interaction between the Android OS and the Unity Android application.

So we extend from UnityPlayerActivity, let's call this class MyCustomUnityPlayerActivity, and we log every Activity method with cool mesages.

So when the Activity onCreate method is executed we will log

    onCreate: Wow! I've been created from MyHelloWorldPlugin!
    onStart: Nice! Now we can start doing things!
    .
    .
    .
    onDestroy: Ou! We are been destroyed! Bye World!

Cool right?

We build our .aar (Android library) file containing our new MyCustomUnityPlayerActivity, we place it under Assets/Plugins/Android and now we have to tell Android to use our custom Activity instead of Unity Activity. So we create an AndroidManifest.xml Plugins/Android folder.

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android" package="com.company.product">
  <application android:icon="@drawable/app_icon" android:label="@string/app_name">
  <!-- Here we tell Android to use OUR activity -->
    <activity android:name="com.unity.myhelloworldplugin.MyCustomUnityPlayerActivity" 
             android:label="@string/app_name"
             android:configChanges="fontScale|keyboard|keyboardHidden|locale|mnc|mcc|navigation|orientation|screenLayout|screenSize|smallestScreenSize|uiMode|touchscreen">
        <intent-filter>
            <action android:name="android.intent.action.MAIN" />
            <category android:name="android.intent.category.LAUNCHER" />
        </intent-filter>
    </activity>
  </application>
</manifest>

We build the .apk file, watch the logs and everything is working fine. So now we published to the world!

Let's imagine that Ed, a developer who is just starting using Unity, saw your awesome plugin and wants to integrate it. He will have to:

  • Import your plugin
  • Change his AndroidManifest.xml main Activity to yours.

But imagine Ed already has another plugin that also need to be the main activity. Ed doesn't know how to merge those plugins so he spend weeks learning how to create a Android project, how to extend from UnityPlayerActivity, being in forums asking for help etc.

Time wasted...poor Ed.

The Happy Path

Imagine Unity provides us this project feature, to listen the Main Activity events. Let's create the same MyHelloWorldPlugin but now instead of extending UnityPlayerActivity i will do the following steps:

Create what should be my "CustomUnityPlayerActivity" but instead i will create a "MyHelloWorldActivityListener" and will extend from a class that "Unity provide us" called UnityActivityListener that will facilitate the methods of an activity. Lets see a code example

package com.unity.myhelloworldplugin;

import android.content.Intent;
import android.os.Bundle;
import android.util.Log;

import martingonzalez.com.unityplayeractivityextension.UnityActivityListener;

public class MyHelloWorldActivityListener extends UnityActivityListener {
    private static final String MY_PLUGIN_TAG = "MyPluginTag";

    @Override
    public void onCreate(Bundle savedInstanceState) {
        Log.d(MY_PLUGIN_TAG, "Hello World from onCreate Method!");
    }

    @Override
    public void onNewIntent(Intent intent) {
        Log.d(MY_PLUGIN_TAG, "Hello World from onNewIntent Method! Yes im listening this too! Awesome!");
    }

    @Override
    public void onStop() {
        Log.d(MY_PLUGIN_TAG, "Ow! The app was stopped so Bye World!");
    }

    @Override
    public void onResume() {
        Log.d(MY_PLUGIN_TAG, "OHH! We are back! HELLO AGAIN!");
    }
}

Here we are lying to our plugin, telling him it will be like an activity, but it is not, it is listening to the Main Activity.

After doing this i will add something to my plugin AndroidManifest.xml so when Unity merges every AndroidManifests of the project, we will be sure that this meta-data is on it.

<manifest 
    xmlns:android="http://schemas.android.com/apk/res/android" package="com.unity.myhelloworldplugin">
    <application>
        <meta-data android:name="com.unity.activity.listener.MyHelloWorld" android:value="com.unity.myhelloworldplugin.MyHelloWorldActivityListener" />
    </application>
</manifest>

See how i created a meta-data node where it has:

key: com.unity.activity.listener.MyHelloWorld

value: com.unity.myhelloworldplugin.MyHelloWorldActivityListener

What "Unity will do" (remember Unity does not have this project feature) in its activity is to find all the keys of its metadata that contain the prefix com.unity.activity.listener following by an identifier for your plugin. In this case our plugin id is .MyHelloWorld and create an instance of the class that we are setting as a value in the meta-data. In this case Unity will create an instance of our com.unity.myhelloworldplugin.MyHelloWorldActivityListener class.

Nice! Now we can create our .aar file and import it into Unity under 'Assets/Plugins/Android' without touching the project AndroidManifest.xml and even we are not forcing to set any Main Activity!

Imagine everybody following this rule, there will be no collision between plugins!

Usage

Since Unity does not have this feature implemented we will need to override the UnityPlayerActivity, but this time for a good reason.

You can use the unityActivityListener.package to import the .aar file in your project or find a .aar file in the UnityExampleAndroidPlugin folder called com.unity.extended.aar, that will be our MainActivity now,

Let's check the AndroidManifest.xml file also in that folder.

<?xml version="1.0" encoding="utf-8"?>
<manifest 
    xmlns:android="http://schemas.android.com/apk/res/android" 
    xmlns:tools="http://schemas.android.com/tools" package="${applicationId}" android:versionCode="1" android:versionName="1.0">
    <application android:label="@string/app_name" android:icon="@drawable/app_icon">
    <!-- We override here UnityPlayerActivity [For a good reason] -->
        <activity android:name="martingonzalez.com.unityplayeractivityextension.UnityPlayerActivityExtension" android:label="@string/app_name" android:icon="@drawable/app_icon" android:configChanges="fontScale|keyboard|keyboardHidden|locale|mnc|mcc|navigation|orientation|screenLayout|screenSize|smallestScreenSize|uiMode|touchscreen">
            <intent-filter>
                <action android:name="android.intent.action.MAIN"/>
                <category android:name="android.intent.category.LAUNCHER"/>
            </intent-filter>
        </activity>
    </application>
</manifest>

I know, it's not cool the path martingonzalez.com.unityplayeractivityextension.UnityPlayerActivityExtension but imagine if Unity integrate this in it's own UnityPlayerActivity, we don't even have to create an AndroidManifest in Plugins/Android.

Importing this into your projects you can manage several plugins that require listening to an activity.

But since this is not a standard if you are using a plugin that extends from UnityPlayerActivity you will need to modify the plugin Activity. Yes, it's not right but thats why i rise a ticket asking for this feature.

From Extending UnityPlayerActivity to Listen Activity events

Let's create a real example how to transform a plugin that is extending from UntiyPlayerActivity and add it into our project without changing anything.

Clone this repository.

You will find a folder called badexample. Inside has a small plugin project that is extending from UnityPlayerActivity.

package martingonzalez.com.badexample;

import android.os.Bundle;
import android.util.Log;

import com.unity3d.player.UnityPlayerActivity;

public class MyCustomUnityPlayerActivity extends UnityPlayerActivity {
    private static final String MY_BAD_TAG = "BadTag";

    @Override
    protected void onCreate(Bundle bundle) {
        super.onCreate(bundle);

        Log.d(MY_BAD_TAG, "I've override onCreate! Nice!");
    }
}

If we want to use this plugin we would need to change our AndroidManifest.xml in our project saying that the MAIN Activity will be martingonzalez.com.badexample.MyCustomUnityPlayerActivity but we don't want that! So...

Using AndroidStudio i will open this project and do the following steps:

  • Go to badexample/build.gradle file and change

this:

dependencies {
    implementation fileTree(include: ['*.jar'], excludes: ['UnityClasses-2017.4.1f1.jar'], dir: 'libs')
    implementation 'com.android.support:appcompat-v7:27.1.1'
    compileOnly files('libs/UnityClasses-2017.4.1f1.jar')
}

to:

dependencies {
    implementation fileTree(include: ['*.jar'], excludes: ['UnityClasses-2017.4.1f1.jar'], dir: 'libs')
    implementation 'com.android.support:appcompat-v7:27.1.1'
    compileOnly project(path: ':unityplayeractivityextension')
}

We are changing the dependency of which library we have to use. In this case we need to have a reference to unityplayeractivityextension project, but if Unity would provide us this feature we would only need to integrate Unity classes.jar in our plugin libs folder as a library.

Then we need to take out the UnityPlayerActivity extension from our MyCustomUnityPlayerActivity.java and instead we are going to extend from 'UnityActivityListener'

So our java class now would be like this:

package martingonzalez.com.badexample;

import android.os.Bundle;
import android.util.Log;

import martingonzalez.com.unityplayeractivityextension.UnityActivityListener;

public class MyCustomUnityPlayerActivity extends UnityActivityListener {
    private static final String MY_BAD_TAG = "BadTag";

    @Override
    public void onCreate(Bundle bundle) {
        Log.d(MY_BAD_TAG, "I've override onCreate! Nice!");
    }
}

And one las step is to add the prefix com.unity.activity.listener into our plugin AndroidManifest.xml.

So go to badexample/src/main/AndroidManifest.xml and configure it like this:

<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="martingonzalez.com.badexample">

    <application>
        <meta-data
            android:name="com.unity.activity.listener.MyCustomPlugin"
            android:value="martingonzalez.com.badexample.MyCustomUnityPlayerActivity" />
    </application>

</manifest>

That's all!

Now hit on Gradle at the right side of the editor, expand :badexample -> Tasks -> Build -> Double click on build task and after build suceed you can find .aar file on badexample/build/outputs/aar/com.me.myhelloworld-plugin.aar

Take that .aar file and import it into UnityPlayerActivityListenersExample/Assets/Plugins/Android. (Project was made with Unity 2014.1f1)

Build the .apk and run it on a device or emulator and you will see this in the logcat

04-15 16:04:15.702 1622-3564/? I/ActivityManager: START u0 {act=android.intent.action.MAIN cat=[android.intent.category.LAUNCHER] flg=0x10200000 cmp=com.martingonzalez.unityplayeractivityextension/martingonzalez.com.unityplayeractivityextension.UnityPlayerActivityExtension} from uid 2000
04-15 16:04:15.729 1622-2247/? I/ActivityManager: Start proc 23683:com.martingonzalez.unityplayeractivityextension/u0a91 for activity com.martingonzalez.unityplayeractivityextension/martingonzalez.com.unityplayeractivityextension.UnityPlayerActivityExtension
04-15 16:04:15.813 23683-23683/? D/Unity: #### onCreate from UnityPlayerActivityExtension
    #### Creating Activity Listeners
04-15 16:04:15.815 23683-23683/? D/Unity: #### Activity Listener Found: 
    ####                          
    #### |_: martingonzalez.com.badexample.MyCustomUnityPlayerActivity
    #### |_: com.unity.myhelloworldplugin.MyHelloWorldActivityListener
    ##############################
04-15 16:04:15.816 23683-23683/? D/Unity: #### onStart from UnityPlayerActivityExtension
04-15 16:04:15.818 23683-23683/? D/Unity: #### onResume from UnityPlayerActivityExtension
04-15 16:04:16.079 23683-23683/? D/Unity: #### onWindowFocusChanged from UnityPlayerActivityExtension

See how our UnityPlayerActivityExtension found martingonzalez.com.badexample.MyCustomUnityPlayerActivity

And if we filter the log with out plugin tag that is "BadTag" we will see this:

04-15 16:04:15.815 23683-23683/? D/BadTag: I've override onCreate! Nice!

That's all!

Remember to vote the Unity Feedback Ticket so this can be part of Unity itself.