這是一個很長很長的story!-芝麻粒兒創作
開篇
源碼地址:GitHub
本文目的,將Unity集成到Android端,學完本文后你可以做到
- Android任意布局加載Unity 3D場景
- 任意操作布局中的按鈕/3D物品(縮放旋轉等)
- 互相消息通信(你叼我,我叼你)
- 自由切換Unity中的場景
- 動態加載手機SD卡3D資源,一次開發到處使用。
- 在小白面前裝逼用

故事正題
首要任務就是將Unity項目導出來(已經做好了3D的處理,關於通信和動態加載在下面介紹)

敲黑板,重點Export Project一定要勾選,之后點擊最下方的Export 靜等項目導出。

導出后的結構感覺好熟悉,就跟Android Studio的項目結構一樣(PS:本來就是)打開Studio 以 moudle的形式導入android工程,第一次可能慢一些慢慢導,去喝杯茶。成功后重要的操作來了。打開剛才導入的build.gradle文件,首當其沖的就是gradle版本的修改,跟你的studio版本一致。
dependencies { classpath 'com.android.tools.build:gradle:3.2.0' }
apply plugin: 'com.android.application'
改為
apply plugin: 'com.android.library'
因為我們要以library的形式集成,接着再往下面走就是熟悉的sdkVsersion了,保持和你的anroid項目一致。還有個applicationId,刪掉這個。
有的項目遇見UnityAds.aar文件,不影響。其他修改以及遇到的一些坑不再贅述,有問題可以留言。我的如下:
// GENERATED BY UNITY. REMOVE THIS COMMENT TO PREVENT OVERWRITING WHEN EXPORTING AGAIN buildscript { repositories { google() jcenter() } dependencies { classpath 'com.android.tools.build:gradle:3.2.0' } } allprojects { repositories { google() jcenter() flatDir { dirs 'libs' } } } apply plugin: 'com.android.library' dependencies { implementation fileTree(dir: 'libs', include: ['*.jar']) } android { compileSdkVersion 29 compileOptions { sourceCompatibility JavaVersion.VERSION_1_8 targetCompatibility JavaVersion.VERSION_1_8 } defaultConfig { minSdkVersion 21 targetSdkVersion 29 ndk { abiFilters 'armeabi-v7a', 'x86' } versionCode 1 versionName '1.0' } lintOptions { abortOnError false } aaptOptions { noCompress = ['.unity3d', '.ress', '.resource', '.obb'] } buildTypes { debug { minifyEnabled false useProguard false proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-unity.txt' jniDebuggable true } release { minifyEnabled false useProguard false proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-unity.txt' signingConfig signingConfigs.debug } } packagingOptions { doNotStrip '*/armeabi-v7a/*.so' doNotStrip '*/x86/*.so' } }
接着打開Unity項目的清單文件AndroidManifest.xml,刪減application節點,刪除intent-filter節點,activity增加內容process(解決某某問題)
<application //刪減其他 android:banner="@drawable/app_banner" android:isGame="true"> <activity ..... //刪除下面兩行-否則造成桌面兩個icon //android:label="@string/app_name" //android:launchMode="singleTask" //增加這行 android:process="e.unity3d"> //刪掉intent-filter <!--<intent-filter>--> <!--<action android:name="android.intent.action.MAIN" />--> <!--<category android:name="android.intent.category.LAUNCHER" />--> <!--<category android:name="android.intent.category.LEANBACK_LAUNCHER" />--> <!--</intent-filter>--> <meta-data android:name="unityplayer.UnityActivity" android:value="true" /> </activity> ......
至此,配置完成,已經可以玩了。
但為了玩的順暢,我們再增加一個自定義控件(1.用來解決kill問題 2.增強自己的擴展性)。自定義一個java文件集成UnityPlayer(核心關鍵類)

public class MUnityPlayer extends UnityPlayer { public MUnityPlayer(Context context) { super(context); } @Override protected void kill() { //super.kill(); //unity默認一些返回操作等會直接kill掉進程,覆寫kill方法,去掉super.kill, 不讓他kill } }
配置完成,點擊菜單欄的Build,Rebuild Project unity的lib項目中生成aar文件


Unity導出的項目配置完成,配置自己的Android項目,首先將幾個lib......so復制到你自己的項目jniLibs中。
然后在你需要集成的項目build文件中
implementation(name: 'XingFeiUnity', ext: 'aar')
一波騷操作搞定,跑起來已經基本可以了。但是這就結束了嗎?不可能,騷起來我們就停不下來。
布局渲染
我們需要在任意布局加載3d,怎么個任意法?就是找個view來addview 既不影響3d 還有android原生界面
//xml布局 <LinearLayout android:id="@+id/linear" android:layout_width="match_parent" android:layout_height="match_parent" android:layout_margin="50dp" android:orientation="vertical" /> //java文件 mLinear.removeAllViews(); mLinear.addView(mUnityPlayer.getView()); mUnityPlayer.requestFocus();
操作效果請看圖片,全部具體代碼的話 就直接放到Github了

通信交互
Android調用Unity
//參數二是 Unity中的方法名 參數一是哪個物體掛在了這個C#腳本 參數三 字符串 UnityPlayer.UnitySendMessage("Main Camera", "AndroidCallUnity", "");

Unity調用Android 仔細看注釋
using System.Collections; using System.Collections.Generic; using UnityEngine; public class Manager : MonoBehaviour { private AndroidJavaObject m_androidObj = null; public GameObject diqiu; void Start() { //注意-情況不同 com.unity3d.player.UnityPlayer 可能不同,可參考其他博主 AndroidJavaClass androidClass = new AndroidJavaClass("com.unity3d.player.UnityPlayer"); m_androidObj = androidClass.GetStatic<AndroidJavaObject>("currentActivity"); diqiu.SetActive(false); } //Unity中的某個物體出發此事件 public void UnityCallAndroid() { Debug.Log("調用方法"); if (m_androidObj != null) { Debug.Log("調用方法進來"); // 第一個參數是android里面java代碼的方法名,第二個是攜帶的字符串參數 m_androidObj.Call("CallAndroid", "我是Unity,我給你發消息了"); } } //Android調用Unity-方法名一定要注意 public void AndroidCallUnity(string json) { if (diqiu.activeInHierarchy) { diqiu.SetActive(false); } else { diqiu.SetActive(true); } } }
場景切換
- Unity內部自己去處理,就跟玩游戲一樣,讓Unity開發自己去做
- andorid觸發,unity換場景,這個借助上面說的消息通信來實現
- 這還有一個技巧,如果資源不是很多且在一個場景的話,可以讓Unity一次直接渲染出來存在字典里,然后想顯示哪個Android給Unity發消息,Unity根據訂好的消息,展示不同的內容,這個好處就是切換展示速度極快。重點處理一下剛啟動的時候的耗時即可。
動態資源
態加載資源的問題,因篇幅有限,咱先只提供個思路,Unity支持讀取android設備的存儲文件,讓他們處理即可,然后android發消息告訴他們地址即可
// 參數一是Unity中的物體名稱,參數二Unity中的方法名 參數三路徑字符串 UnityPlayer.UnitySendMessage("AndriodMethodMgr", "CallUnitySetPath", Environment.getExternalStorageDirectory() + "");
啊哈,到這基本就結束了。快了又開心。
坑中帶坑
為了愉快的裝逼,最好還是看一看遇到的這些問題,能至少省幾天時間。
1. 混淆問題,如果你開啟了混淆,切記 切記,把混淆添加進入,這個大坑耽誤我好久啊
2. 如果模型在unity中沒問題,在android端穿幫,可以看看發布質量,將android的設置成高的

3. 如果反復執行的模型動畫不對,怎么不對?舉例心臟跳動,這是非常注重動畫的銜接的,如果銜接時間不對會造成心臟動畫的抖動,這會非常的明顯。
可以看動畫的setting 退出時間,退出時間是比例(如下圖),1代表全部動畫,0.5代表動畫使勁按的一般。過度時間前后動畫重疊(好像默認.95?) 可以改成0,如圖設置

4. 集成到apk后 申請了橫豎屏 但是apk沒作用,是unity發布出的設置導致的,再Unity導出的時候 other setting中設置宣傳方向
5. 權限問題,上面說了會導出一個android項目,你仔細看這個項目的AndroidManifest文件,你會發現也有權限。
注意,這時候比如你的android項目有權限A 這個Unity導出的項目沒有權限A,當你集成合並之后,導致最終的apk沒有權限A,這並不是我們想看到的;
所以為了 解決這個問題,很簡單,我們把兩個清單文件的權限保持一致即可,記住啊,否則怎么哭的都不知道。
6. 還有一個未解決的問題,放到這,有朋友知道的話,感謝指教。
帶有動畫的一個物體,在有的android設備上,動畫表現征程,但是有的會出現動畫跳動的情況,感覺像是電視的進度條那在跳進度一樣。
結尾
最后,別問我為啥知道這么多問題,問就是因為自己跪着走過來的。

