Unity3D研究之打开Activity与调用Java代码传递参数

本文指导如何在Unity for Android中调用Java代码,通过创建Activity并使用Unity脚本打开,实现与Android系统的交互。包括代码示例、打包流程及在Unity中调用的方法。
Unity for Android 比较特殊,Unity for IOS 打包是将XCODE工程直接交给开发者,开发者可以在工程的基础上继续添加新的视图,最后由开发者自行打包生成IPA包,发布程序。而Unity for Android打包直接生成APK包,等于说源代码开发者是看不到的,但是Unity的自身确实有些局限,针对Android平台我们需要学习如何在 Unity中调用Android的JAVA代码。本章我们的目标是使用Unity的脚本打开Activity。首先我们创建一个普通的Android工程,目录结构如下图所示。

[img]
[img]http://dl2.iteye.com/upload/attachment/0092/7062/78aa955c-d2f4-3614-a1a8-5878e66c0bfe.png[/img]
[/img]

因为项目需要使用Unity提供的接口,所以需要将接口classes.jar引入至当前工程但中。接口包的所在地,打开Finder->应用程序->Unity->点击Unity图标,鼠标右键选择“显示包内容”->Contents->PlaybackEngines->AndroidPlayer->bin->classes.jar 。接口包引入工程后,开始编写JAVA代码。

UnityTestActivity.java
 	package com.xys;
02
03 import android.content.Context;
04 import android.content.Intent;
05 import android.os.Bundle;
06
07 import com.unity3d.player.UnityPlayerActivity;
08
09 public class UnityTestActivity extends UnityPlayerActivity {
10 /** Called when the activity is first created. */
11
12 Context mContext = null;
13 @Override
14 public void onCreate(Bundle savedInstanceState) {
15 super.onCreate(savedInstanceState);
16 mContext = this;
17 }
18
19 public void StartActivity0(String name)
20 {
21 Intent intent = new Intent(mContext,TestActivity0.class);
22 intent.putExtra("name", name);
23 this.startActivity(intent);
24 }
25
26 public void StartActivity1(String name)
27 {
28 Intent intent = new Intent(mContext,TestActivity1.class);
29 intent.putExtra("name", name);
30 this.startActivity(intent);
31 }
32 }


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
01 	package com.xys;
02
03 import android.app.Activity;
04 import android.os.Bundle;
05 import android.view.View;
06 import android.view.View.OnClickListener;
07 import android.widget.Button;
08 import android.widget.TextView;
09
10 public class TestActivity0 extends Activity {
11 /** Called when the activity is first created. */
12 @Override
13 public void onCreate(Bundle savedInstanceState) {
14 super.onCreate(savedInstanceState);
15 setContentView(R.layout.main);
16
17 TextView text = (TextView)this.findViewById(R.id.textView1);
18 text.setText(this.getIntent().getStringExtra("name"));
19
20 Button close = (Button)this.findViewById(R.id.button0);
21 close.setOnClickListener(new OnClickListener() {
22
23 @Override
24 public void onClick(View v) {
25
26 TestActivity0.this.finish();
27 }
28 });
29 }
30
31 }


TestActivity1.java
01 	package com.xys;
02
03 import android.app.Activity;
04 import android.os.Bundle;
05 import android.view.View;
06 import android.view.View.OnClickListener;
07 import android.widget.Button;
08 import android.widget.TextView;
09
10 public class TestActivity1 extends Activity {
11 /** Called when the activity is first created. */
12 @Override
13 public void onCreate(Bundle savedInstanceState) {
14 super.onCreate(savedInstanceState);
15 setContentView(R.layout.main);
16
17 TextView text = (TextView)this.findViewById(R.id.textView1);
18 text.setText(this.getIntent().getStringExtra("name"));
19
20 Button close = (Button)this.findViewById(R.id.button0);
21 close.setOnClickListener(new OnClickListener() {
22
23 @Override
24 public void onClick(View v) {
25
26 TestActivity1.this.finish();
27 }
28 });
29 }
30
31 public void Start()
32 {
33
34 }
35
36 }


使用this.getIntent().getStringExtra(“name”)方法,得到上个界面传递过来的字符串,并且显示在屏幕中,用于区分新打开的Activity。TestActivity0 与TestActivity1是两个新打开的Activity,它们属于Unity程序的子Activity所以它不需要继承 UnityPlayerActivity,直接继承Activity即可,在代码中监听了一个按钮,意思是点击按钮后关闭当前的Activity。

接着是代码中打开的一个布局文件,这个应该没什问题,学过Android开发的朋友应该都能看懂,我就不详细解释了。

main.xml
01 	<?xml version="1.0" encoding="utf-8"?>
02 <ScrollView xmlns:android="http://schemas.android.com/apk/res/android" android:id="@+id/screen"
03 android:layout_width="fill_parent" android:layout_height="fill_parent"
04 android:orientation="vertical">
05 <LinearLayout
06 android:layout_width="fill_parent" android:layout_height="fill_parent"
07 android:orientation="vertical">
08 <ImageView
09 android:src="@drawable/jay"
10 android:layout_width="fill_parent"
11 android:layout_height="fill_parent"
12 />
13 <TextView android:id="@+id/textView0"
14 android:layout_width="fill_parent"
15 android:layout_height="wrap_content"
16 android:textColor="#000000"
17 android:textSize="18dip"
18 android:background="#00FF00"
19 android:text="雨松MOMO 带你走进Unity for Android的世界"
20 android:gravity="center_vertical|center_horizontal"
21 />
22 <TextView android:id="@+id/textView1"
23 android:layout_width="fill_parent"
24 android:layout_height="wrap_content"
25 android:textColor="#FFFFFF"
26 android:textSize="18dip"
27 android:background="#0000FF"
28 android:text="Unity与Android之间的交互"
29 android:gravity="center_vertical|center_horizontal"
30 />
31 <Button android:id="@+id/button0"
32 android:layout_width="fill_parent" android:layout_height="wrap_content"
33 android:text="关闭这个Activity"/>
34 </LinearLayout>
35 </ScrollView>


最后是本程序的AndroidManisest.xml,这个很重要,一定要配置。
01 	<?xml version="1.0" encoding="utf-8"?>
02 <manifest xmlns:android="http://schemas.android.com/apk/res/android"
03 package="com.xys"
04 android:versionCode="1"
05 android:versionName="1.0" >
06
07 <uses-sdk android:minSdkVersion="7" />
08 <application
09 android:icon="@drawable/ic_launcher"
10 android:label="@string/app_name" >
11 <activity
12 android:theme="@android:style/Theme.NoTitleBar.Fullscreen"
13 android:name=".UnityTestActivity"
14 android:label="@string/app_name" >
15 <intent-filter>
16 <action android:name="android.intent.action.MAIN" />
17 <category android:name="android.intent.category.LAUNCHER" />
18 </intent-filter>
19 </activity>
20
21 <activity
22 android:name=".TestActivity0"
23 android:theme="@android:style/Theme.NoTitleBar.Fullscreen"
24 >
25 </activity>
26
27 <activity
28 android:name=".TestActivity1"
29 android:theme="@android:style/Theme.NoTitleBar.Fullscreen"
30 >
31 </activity>
32 </application>
33
34 </manifest>



大家请看清楚,这里一共配置了代码中的三个Activity,并且标志UnityTestActivity为主Activity。另外继承了 UnityPlayerActivity后在Eclipse是运行不了的,除非拿到Unity中在真机下才行,请大家继续认真阅读本篇博文。

OK,到这里Android的代码已经写完,下面我们学习如何在Unity中去调用它。首先Build一下当前的Eclipse工程,代码所有的.class文件都生成在了Android工程的bin文件夹中,当前工程的路径是 UnityTestActivity->bin->classes->com->xys->你的.class文件。下面需要对这些.class文件进行打包,苹果系统的话打开电脑的终端,cd到classes文件夹的目录下,接着执行代码

1 jar -cvf class.jar *
这行代码的意思是把当前目录下的所有.class文件打包成.jar文件,保存文件名称为class.jar。接着class.jar文件就生成在 bin->classes->目录中了。如下图所示,请大家仔细看一下解开的包应该与你的Android对应的包名保持一致,我的包名是 com.xys,所以文件夹就是class->com->xys->.class代码。
[img]
[img]http://dl2.iteye.com/upload/attachment/0092/7064/5537641a-e4ba-3108-85f9-b4867cfcbe1b.png[/img]
[/img]


确保无误后,请大家开始创建Unity工程。如下图所示,Unity工程中文件夹的结构如下,Plugins->Android的名称不能修改,必需保持一致。接着把Eclipse中Android的工程文件拷贝至这里,除了Android工程中的src文件夹,将其它文件夹全部拷贝至 Plugins->Android文件夹中。最后在Plugins->Android文件夹中创建bin文件夹,然后将刚刚生成的.jar文件拷贝进来,jar的名称可以随便修改,但是jar包里面必须是com->xys->你的class文件,否则运行程序后提示找不到类文件。
[img]
[img]http://dl2.iteye.com/upload/attachment/0092/7066/8637b0b6-12a5-3569-b405-adfabcf95d92.png[/img]
[/img]


最后在Unity工程中创建一个C#脚本,就是上图中的Test.cs,直接将它绑定在摄像机中,它用来通知界面打开Activity。如下图所示,利用GUI在屏幕中创建两个按钮,点击按钮打开不同的Activity。
[img]
[img]http://dl2.iteye.com/upload/attachment/0092/7068/d0d21a6b-2013-3bec-a713-b760db706a22.png[/img]
[/img]

Test.cs
01 	using UnityEngine;
02 using System.Collections;
03
04 public class Test : MonoBehaviour
05 {
06
07 // Update is called once per frame
08 void Update ()
09 {
10 //当用户按下手机的返回键或home键退出游戏
11 if (Input.GetKeyDown(KeyCode.Escape) || Input.GetKeyDown(KeyCode.Home) )
12 {
13 Application.Quit();
14 }
15 }
16
17 void OnGUI()
18 {
19 if(GUILayout.Button("OPEN Activity01",GUILayout.Height(100)))
20 {
21 //注释1
22 AndroidJavaClass jc = new AndroidJavaClass("com.unity3d.player.UnityPlayer");
23 AndroidJavaObject jo = jc.GetStatic<AndroidJavaObject>("currentActivity");
24 jo.Call("StartActivity0","第一个Activity");
25 }
26 if(GUILayout.Button("OPEN Activity02",GUILayout.Height(100)))
27 {
28 AndroidJavaClass jc = new AndroidJavaClass("com.unity3d.player.UnityPlayer");
29 AndroidJavaObject jo = jc.GetStatic<AndroidJavaObject>("currentActivity");
30 jo.Call("StartActivity1","第二个Activit");
31 }
32 }
33
34 }


注释1:先得到AndroidJavaClass,然后得到AndroidjavaObject就是当前Activity的对象,也就是我们在上面创建的主UnityTestActivity.JAVA。拿到它的对象后调用jo.Call()参数1表示调用UnityTestActivity.JAVA类中的方法名称,参数2表示该方法传递过去的参数。如下图所示:“第一个Activity”与“第二个Activit”就是我在C#中传递过去的字符串。
[img]
[img]http://dl2.iteye.com/upload/attachment/0092/7070/1546e65a-fc77-3446-8723-1d03af7375e4.png[/img]
[/img]
打开的Activity中点击“关闭这个Activity按钮”,程序将继续回到原来的界面。
[img]
[img]http://dl2.iteye.com/upload/attachment/0092/7072/1ab899e5-8033-30c9-bdce-fa98272b2f7f.png[/img]
[/img]


最后大家一定要注意Unity中的包名,要和Android工程保持一致,否则无法调用。如下图所示,Bundle Identifier* 当前项目为com.xys 。另外其它的选项也在其中,请大家仔细阅读。
[img]
[img]http://dl2.iteye.com/upload/attachment/0092/7074/90b78946-dbc2-3524-8291-b5f18a5c6cd1.png[/img]
[/img]
采用PyQt5框架Python编程语言构建图书信息管理平台 本项目基于Python编程环境,结合PyQt5图形界面开发库,设计实现了一套完整的图书信息管理解决方案。该系统主要面向图书馆、书店等机构的日常运营需求,通过模块化设计实现了图书信息的标准化管理流程。 系统架构采用典型的三层设计模式,包含数据存储层、业务逻辑层和用户界面层。数据持久化方案支持SQLite轻量级数据库MySQL企业级数据库的双重配置选项,通过统一的数据库操作接口实现数据存取隔离。在数据建模方面,设计了包含图书基本信息、读者档案、借阅记录等核心数据实体,各实体间通过主外键约束建立关联关系。 核心功能模块包含六大子系统: 1. 图书编目管理:支持国际标准书号、中国图书馆分类法等专业元数据的规范化著录,提供批量导入单条录入两种数据采集方式 2. 库存动态监控:实时追踪在架数量、借出状态、预约队列等流通指标,设置库存预警阈值自动提醒补货 3. 读者服务管理:建立完整的读者信用评价体系,记录借阅历史违规行为,实施差异化借阅权限管理 4. 流通业务处理:涵盖借书登记、归还处理、续借申请、逾期计算等标准业务流程,支持射频识别技术设备集成 5. 统计报表生成:按日/月/年周期自动生成流通统计、热门图书排行、读者活跃度等多维度分析图表 6. 系统维护配置:提供用户权限分级管理、数据备份恢复、操作日志审计等管理功能 在技术实现层面,界面设计遵循Material Design设计规范,采用QSS样式表实现视觉定制化。通过信号槽机制实现前后端数据双向绑定,运用多线程处理技术保障界面响应流畅度。数据验证机制包含前端格式校验后端业务规则双重保障,关键操作均设有二次确认流程。 该系统适用于中小型图书管理场景,通过可扩展的插件架构支持功能模块的灵活组合。开发过程中特别注重代码的可维护性,采用面向对象编程范式实现高内聚低耦合的组件设计,为后续功能迭代奠定技术基础。 资源来源于网络分享,仅用于学习交流使用,请勿用于商业,如有侵权请联系我删除!
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值