Unity3D研究院之打开Activity与调用JAVA代码传递参数(十八)
雨松MOMO原创文章如转载,请注明:转载至我的独立域名博客雨松MOMO程序研究院,原文地址:http://www.xuanyusong.com/archives/667
Unity for Android 比较特殊,Unity for IOS 打包是将XCODE工程直接交给开发者,开发者可以在工程的基础上继续添加新的视图,最后由开发者自行打包生成IPA包,发布程序。而Unity for Android打包直接生成APK包,等于说源代码开发者是看不到的,但是Unity的自身确实有些局限,针对Android平台我们需要学习如何在Unity中调用Android的JAVA代码。本章我们的目标是使用Unity的脚本打开Activity。首先我们创建一个普通的Android工程,目录结构如下图所示。
因为项目需要使用Unity提供的接口,所以需要将接口classes.jar引入至当前工程但中。接口包的所在地,打开Finder->应用程序->Unity->点击Unity图标,鼠标右键选择“显示包内容”->Contents->PlaybackEngines->AndroidPlayer->bin->classes.jar 。接口包引入工程后,开始编写JAVA代码。
UnityTestActivity.java
package com.xys; import android.content.Context; import android.content.Intent; import android.os.Bundle; import com.unity3d.player.UnityPlayerActivity; public class UnityTestActivity extends UnityPlayerActivity { /** Called when the activity is first created. */ Context mContext = null; @Override public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); mContext = this; } public void StartActivity0(String name) { Intent intent = new Intent(mContext,TestActivity0.class); intent.putExtra("name", name); this.startActivity(intent); } public void StartActivity1(String name) { Intent intent = new Intent(mContext,TestActivity1.class); intent.putExtra("name", name); this.startActivity(intent); } }
UnityTestActivity是主Activity,Unity程序一起动就会调用这个Activity,它是在AndroidManifest.xml中配置的。它需要继承UnityPlayerActivity,然而它就是刚刚我们引入的classes.jar包中提供的接口类。UnityTestActivity对外提供了两个方法接口,StartActivity0(String name) 方法与StartActivity1(String name)方法,这两个方法是在Unity中使用C#脚本调用的,意思是调用后程序将打开一个新的Activity,参数name也是由C#脚本传递过来的,接着将传递的String参数继续传递给新打开的Activity。
TestActivity0.java
package com.xys; import android.app.Activity; import android.os.Bundle; import android.view.View; import android.view.View.OnClickListener; import android.widget.Button; import android.widget.TextView; public class TestActivity0 extends Activity { /** Called when the activity is first created. */ @Override public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.main); TextView text = (TextView)this.findViewById(R.id.textView1); text.setText(this.getIntent().getStringExtra("name")); Button close = (Button)this.findViewById(R.id.button0); close.setOnClickListener(new OnClickListener() { @Override public void onClick(View v) { TestActivity0.this.finish(); } }); } }
TestActivity1.java
package com.xys; import android.app.Activity; import android.os.Bundle; import android.view.View; import android.view.View.OnClickListener; import android.widget.Button; import android.widget.TextView; public class TestActivity1 extends Activity { /** Called when the activity is first created. */ @Override public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.main); TextView text = (TextView)this.findViewById(R.id.textView1); text.setText(this.getIntent().getStringExtra("name")); Button close = (Button)this.findViewById(R.id.button0); close.setOnClickListener(new OnClickListener() { @Override public void onClick(View v) { TestActivity1.this.finish(); } }); } public void Start() { } }
使用this.getIntent().getStringExtra(“name”)方法,得到上个界面传递过来的字符串,并且显示在屏幕中,用于区分新打开的Activity。TestActivity0与TestActivity1是两个新打开的Activity,它们属于Unity程序的子Activity所以它不需要继承UnityPlayerActivity,直接继承Activity即可,在代码中监听了一个按钮,意思是点击按钮后关闭当前的Activity。
接着是代码中打开的一个布局文件,这个应该没什问题,学过Android开发的朋友应该都能看懂,我就不详细解释了。
main.xml
<?xml version="1.0" encoding="utf-8"?> <ScrollView xmlns:android="http://schemas.android.com/apk/res/android" android:id="@+id/screen" android:layout_width="fill_parent" android:layout_height="fill_parent" android:orientation="vertical"> <LinearLayout android:layout_width="fill_parent" android:layout_height="fill_parent" android:orientation="vertical"> <ImageView android:src="@drawable/jay" android:layout_width="fill_parent" android:layout_height="fill_parent" /> <TextView android:id="@+id/textView0" android:layout_width="fill_parent" android:layout_height="wrap_content" android:textColor="#000000" android:textSize="18dip" android:background="#00FF00" android:text="雨松MOMO 带你走进Unity for Android的世界" android:gravity="center_vertical¦center_horizontal" /> <TextView android:id="@+id/textView1" android:layout_width="fill_parent" android:layout_height="wrap_content" android:textColor="#FFFFFF" android:textSize="18dip" android:background="#0000FF" android:text="Unity与Android之间的交互" android:gravity="center_vertical¦center_horizontal" /> <Button android:id="@+id/button0" android:layout_width="fill_parent" android:layout_height="wrap_content" android:text="关闭这个Activity"/> </LinearLayout> </ScrollView>
最后是本程序的AndroidManisest.xml,这个很重要,一定要配置。
<?xml version="1.0" encoding="utf-8"?> <manifest xmlns:android="http://schemas.android.com/apk/res/android" package="com.xys" android:versionCode="1" android:versionName="1.0" > <uses-sdk android:minSdkVersion="7" /> <application android:icon="@drawable/ic_launcher" android:label="@string/app_name" > <activity android:theme="@android:style/Theme.NoTitleBar.Fullscreen" android:name=".UnityTestActivity" android:label="@string/app_name" > <intent-filter> <action android:name="android.intent.action.MAIN" /> <category android:name="android.intent.category.LAUNCHER" /> </intent-filter> </activity> <activity android:name=".TestActivity0" android:theme="@android:style/Theme.NoTitleBar.Fullscreen" > </activity> <activity android:name=".TestActivity1" android:theme="@android:style/Theme.NoTitleBar.Fullscreen" > </activity> </application> </manifest>
备注:在</intent-filter>下面加上一现面的这行参数,不然在unity导出时会出现警告 Unable to find unity activity in manifest. You need to make sure orientation attribut is set to portrait manually。
<meta-data android:name="unityplayer.UnityActivity" android:value="true" />
大家请看清楚,这里一共配置了代码中的三个Activity,并且标志UnityTestActivity为主Activity。另外继承了UnityPlayerActivity后在Eclipse是运行不了的,除非拿到Unity中在真机下才行,请大家继续认真阅读本篇博文。
OK,到这里Android的代码已经写完,下面我们学习如何在Unity中去调用它。首先Build一下当前的Eclipse工程,代码所有的.class文件都生成在了Android工程的bin文件夹中,当前工程的路径是UnityTestActivity->bin->classes->com->xys->你的.class文件。下面需要对这些.class文件进行打包,苹果系统的话打开电脑的终端,cd到classes文件夹的目录下,接着执行代码
jar
-
cvf
class
.jar
*
这行代码的意思是把当前目录下的所有.class文件打包成.jar文件,保存文件名称为class.jar。接着class.jar文件就生成在bin->classes->目录中了。如下图所示,请大家仔细看一下解开的包应该与你的Android对应的包名保持一致,我的包名是com.xys,所以文件夹就是class->com->xys->.class代码。
确保无误后,请大家开始创建Unity工程。如下图所示,Unity工程中文件夹的结构如下,Plugins->Android的名称不能修改,必需保持一致。接着把Eclipse中Android的工程文件拷贝至这里,除了Android工程中的src文件夹,将其它文件夹全部拷贝至Plugins->Android文件夹中。最后在Plugins->Android文件夹中创建bin文件夹,然后将刚刚生成的.jar文件拷贝进来,jar的名称可以随便修改,但是jar包里面必须是com->xys->你的class文件,否则运行程序后提示找不到类文件。
最后在Unity工程中创建一个C#脚本,就是上图中的Test.cs,直接将它绑定在摄像机中,它用来通知界面打开Activity。如下图所示,利用GUI在屏幕中创建两个按钮,点击按钮打开不同的Activity。
Test.cs
using UnityEngine; using System.Collections; public class Test : MonoBehaviour { // Update is called once per frame void Update () { //当用户按下手机的返回键或home键退出游戏 if (Input.GetKeyDown(KeyCode.Escape) ¦¦ Input.GetKeyDown(KeyCode.Home) ) { Application.Quit(); } } void OnGUI() { if(GUILayout.Button("OPEN Activity01",GUILayout.Height(100))) { //注释1 AndroidJavaClass jc = new AndroidJavaClass("com.unity3d.player.UnityPlayer"); AndroidJavaObject jo = jc.GetStatic<AndroidJavaObject>("currentActivity"); jo.Call("StartActivity0","第一个Activity"); } if(GUILayout.Button("OPEN Activity02",GUILayout.Height(100))) { AndroidJavaClass jc = new AndroidJavaClass("com.unity3d.player.UnityPlayer"); AndroidJavaObject jo = jc.GetStatic<AndroidJavaObject>("currentActivity"); jo.Call("StartActivity1","第二个Activit"); } } }
注释1:先得到AndroidJavaClass,然后得到AndroidjavaObject就是当前Activity的对象,也就是我们在上面创建的主UnityTestActivity.java。拿到它的对象后调用jo.Call()参数1表示调用UnityTestActivity.java类中的方法名称,参数2表示该方法传递过去的参数。如下图所示:“第一个Activity”与“第二个Activit”就是我在C#中传递过去的字符串。
在打开的Activity中点击“关闭这个Activity按钮”,程序将继续回到原来的界面。
最后大家一定要注意Unity中的包名,要和Android工程保持一致,否则无法调用。如下图所示,Bundle Identifier* 当前项目为com.xys 。另外其它的选项也在其中,请大家仔细阅读。
源码下载地址: http://www.xuanyusong.com/archives/667
相关推荐
总结一下,Activity之间的相互调用与传递参数是Android开发中的核心技能。理解启动模式、Intent的使用、回调机制以及生命周期管理,可以帮助我们构建健壮且用户体验良好的应用程序。在实际编码时,应根据具体需求...
通过这个类,我们可以使用Java代码(通过Unity的Java插件系统)来调用Android的Activity。首先,你需要在Unity中创建一个Java接口,然后在Android项目中实现这个接口。接口中可以包含一个方法,用于启动你自定义的...
接着,在Activity的Java代码中初始化并设置`WebView`: ```java WebView webView = findViewById(R.id.webView); webView.getSettings().setJavaScriptEnabled(true); // 开启JavaScript支持 webView.loadUrl("file...
本篇文章将详细讲解如何在Android的两个Activity之间传递参数,并通过给出的`LoginActivity.java`和`MainActivity.java`两个文件为例进行说明。 首先,我们要了解在Android中,有两种主要的方式来传递数据: 1. ...
1. 在Android端的`SinaWeiboHelper`类中,你需要监听新浪SDK的回调事件,如登录成功、分享成功等,然后通过`UnityPlayer.currentActivity`调用Unity的C#方法传递结果。 ```java public void onLoginSuccess() { ...
总之,这个Unity3D工程实例旨在教你如何利用Unity与Android系统的交互来实现读取相册、调用相机和保存图片的功能。通过学习和实践这个项目,开发者将能够更好地掌握Unity与移动设备硬件的集成,进一步提升其在移动...
在Android平台上,Unity3D通过Java原生代码(.java文件)与Android系统进行交互。要接入小米SDK,我们需要创建一个Java原生插件,编写Java代码并将其打包为.AAR或.JAR文件,然后在Unity中导入这个库。 2. **小米SDK...
在开发使用Unity引擎制作的游戏或应用程序时,有时需要在Unity的C#脚本与Android平台上的Java代码之间进行相互调用,以实现特定的功能。这种需求通常出现在接入特定的SDK、使用原生设备功能、或者进行特定业务逻辑...
这需要通过Unity与Android之间的交互来完成,也就是所谓的Unity调用Android Java函数。这个过程涉及到多个步骤和技术细节,包括JNI(Java Native Interface)的使用、Android插件开发以及Unity脚本的编写。 首先,...
然后,在源Activity中实现该接口,并在需要触发数据传递时调用相应的方法。例如,当用户点击一个按钮时: ```java public class ActivityA extends AppCompatActivity implements DataCallback { // ... @...
Activity之间的跳转和参数传递是Android开发中的重要概念,对于理解和构建复杂的Android应用至关重要。本篇文章将详细解析这一主题,并提供源代码下载,帮助开发者深入理解并实践。 首先,Activity之间的跳转是通过...
Unity是一款强大的跨平台游戏引擎,它也可以用于开发Android应用,因此这个主题涉及到的知识点包括Unity与Android原生应用的交互、Intent的使用、以及如何在Unity中处理Android的Intent数据。 首先,我们要理解...
在某些情况下,开发者可能需要实现Unity与Android原生代码的交互,以便进行特定功能的扩展,例如在游戏内更新(热更新)或者引导用户到应用市场进行评分或查看更新。本文将详细讲解如何实现Unity和Android之间的交互...
了解Activity的基础,特别是如何启动、跳转以及传递参数,是每个Android开发者必须掌握的关键技能。 一、Activity的基础概念 Activity是Android系统中负责与用户交互的组件,它负责展示UI和处理用户事件。每个...
4. **Unity与Java交互**:在Unity中,我们可以通过`Application.LoadLevel`或`SceneManager.LoadScene`加载场景时调用一个C#方法,该方法会触发Java插件的方法。例如,在C#中创建一个静态方法: ```csharp ...
- `Classes`: 存放导出的Java代码,主要负责Unity引擎与Android系统的交互,如初始化UnityPlayer、处理Activity生命周期、实现Android接口等。 - `libs`: 包含了针对不同CPU架构的.so库文件,这是Unity引擎在...
在Unity中,你可以使用`UnityPlayer.CurrentActivity`对象来调用Java方法,例如: ```csharp using UnityEngine; using System.Collections; using UnityEngine.Android; public class CameraCapture : ...