Unity接入多个SDK的通用接口开发与资源管理(二)
接着上篇,本篇对SDK接口进行封装。在开始之前,需要先了解下面知识。
(1)unity与android之间的交互
unity通过下面方式调用android中的函数:
方法一:
AndroidJavaClass jc = new AndroidJavaClass ("com.unity3d.player.UnityPlayer");
AndroidJavaObject _mActivity = jc.GetStatic<AndroidJavaObject>("currentActivity");
_mActivity.Call("函数名");
方法二:
AndroidJavaClass jc = new AndroidJavaClass ("com.unity3d.player.UnityPlayer");//括号里可以是任意jar包的包名+类名
jc.CallStatic("函数名");
上面代码第一种先获得UnityPlayer类,UnityPlayer类是unity导出Android工程后生成的unity-classes.jar里的类。然后再获取静态变量_mActivity,_mActivity 是Activity类型,由此获得了Android当前的Activity,然后再调用此Activity中的方法。我们可以看一下unity自动生成的classes.jar中的源码:

第二种是通过包名+类名获取该类,直接调用类中的静态方法来实现的。可以是任意自定义的包名和类名。但这种就与Android的生命周期无关了。
我们可以这么理解Android应用中一个界面是一个Activity,两个界面的跳转就是两个Activity之间的跳转(类似的iOS是UIController之间的跳转)。其实我们unity游戏在Android上也是一个Android应用,所以它也是运行在一个Activity上的,这个Activity叫UnityPlayerActivity,它继承自Activity,看源码如下,每个Activity初始化的时候会首先执行onCreate()方法,所以通常把SDK的初始化函数也放在这个函数中。它就会随着Android的生命周期进行初始化,这是第一种做法,也是最常用的SDK初始化方法。还有一种SDK初始化的方法,就是自己新建一个jar包,在jar包中新建类,在类中写一些SDK初始化的静态方法,然后通过方法二在unity中调用这些方法进行初始化。我们这里采用大多数人使用的方法一,即在onCreate()中初始化。

有了上面的了解,我们开始进行unity的通用接口的开发。我们需要一个SDKManager类来管理所有调用SDK接口的方法,代码如下,把脚本挂在一个空物体上。
using System.Collections;
using System.Collections.Generic;
using UnityEngine;
using System; public class SDKManager : MonoBehaviour { private static SDKManager _instance = null;
public static SDKManager Instance
{
get
{
return _instance;
}
} private void Awake()
{
if (_instance != null)
{
GameObject.Destroy(this.gameObject);
}
_instance = this; InitSDKManager(); DontDestroyOnLoad(this);
} private void OnDestroy()
{
if (_instance == this) {
_instance = null;
}
} #if UNITY_ANDROID && !UNITY_EDITOR
private static string UNITY_CLASS = "com.unity3d.player.UnityPlayer";
private AndroidJavaObject _mActivity;
#endif
public void InitSDKManager()
{
#if UNITY_ANDROID && !UNITY_EDITOR
AndroidJavaClass jc = new AndroidJavaClass (UNITY_CLASS);
_mActivity = jc.GetStatic<AndroidJavaObject>("currentActivity");
#endif
} //信鸽SDK接口
public void GetPushToken()
{
#if UNITY_ANDROID && !UNITY_EDITOR
_pushToken = _mActivity.Call<string>("GetXGPushTokenNew");
#endif
} public void SetXGPushTag(string tag)
{
#if UNITY_ANDROID && !UNITY_EDITOR
_mActivity.Call("SetXGPushTag",tag);
#endif
} public void AddLocalPushNotificationAndroid(string content, DateTime date, bool isRepeatEveryday = false)
{
#if UNITY_ANDROID && !UNITY_EDITOR
int isRepeat = ;
if (isRepeatEveryday == true)
{
isRepeat = ;
} _mActivity.Call("AddLocalNotification","SDK集合",content, date.ToString("yyyyMMdd"), date.ToString("HH"), date.ToString("mm"),isRepeat);
#endif
} public void ClearAllLocalNotificationAndroid()
{
#if UNITY_ANDROID && !UNITY_EDITOR
_mActivity.Call("ClearAllLocalNotification");
#endif
} //buglySDK接口
//启动c#异常捕获上报功能
public void buglyCatch()
{
Debug.Log("start Bugly report");
BuglyAgent.EnableExceptionHandler(); } }
然后新建一个Test脚本挂在摄像机上,调用这些接口:
using System.Collections;
using System.Collections.Generic;
using UnityEngine;
using System; public class Test : MonoBehaviour
{ void OnGUI()
{
if (GUILayout.Button("信鸽推送", GUILayout.Width(100.0f), GUILayout.Height(50.0f)))
{
DateTime noticeTime = new DateTime(, , , , , );
SDKManager.Instance.AddLocalPushNotificationAndroid("信鸽给你的推送", noticeTime);
}
if (GUILayout.Button("清除本地推送", GUILayout.Width(100.0f), GUILayout.Height(50.0f)))
{ SDKManager.Instance.ClearAllLocalNotificationAndroid();
}
if (GUILayout.Button("开启Bugly异常上报", GUILayout.Width(100.0f), GUILayout.Height(50.0f)))
{
SDKManager.Instance.buglyCatch();
}
}
}
上面代码,我们新建一个SDKManaget单例类,把每个SDK的接口封装成一个函数。像信鸽这种原生的jar包中的方法,我们就用_mActivity.Call()去调用,像bugly这种提供了接口的,我们直接调用提供的接口。通用接口类就介绍到这里,下面我们来实现_mActivity.Call()中调用的Android工程里的方法。使用Android Studio(2.3.3)新建一个工程。我们的目的是做一个jar包,这个jar包作用有两点:(1)用来对接unity调用jar包里的方法。(2)对接unity导出的Android工程的生命周期。这是我们为什么选择上面第一种接入方法的原因。上面说过unity运行的Activity是UnityPlayerActivity,我们新建一个类MainActivity,继承com.unity3d.player.UnityPlayerNativeActivity,而UnityPlayerNativeActivity继承UnityPlayerActivity。然后在AndroidManifest.xml里设置MainActivity为启动的Activity,所以MainActivity就是当前unity运行的Activity。我们通过_mActivity = jc.GetStatic<AndroidJavaObject>("currentActivity")获取到的就是MainActivity,然后我们在MainActivity中实现被unity调用的函数。代码:
public class MainActivity extends com.unity3d.player.UnityPlayerNativeActivity {
private static boolean _isXGRegisterSuccess = false;
private static String _pushToken = "";
@Override
protected void onCreate(Bundle savedInstanceState)
{
if (savedInstanceState == null)
{
savedInstanceState = new Bundle();
}
super.onCreate(savedInstanceState);
Log.d("myGame", "MainActivity onCreate Called");
InitBugly();
InitXG();
}
void InitBugly()
{
CrashReport.initCrashReport(getApplicationContext(),"aaaaaaaa",false);
}
void InitXG(){
XGPushConfig.enableDebug(this,false);
MsgReceiver testReceiver = new MsgReceiver();
IntentFilter intentFilter = new IntentFilter();
intentFilter.addAction("com.abc.def.receiveintent");
registerReceiver(testReceiver,intentFilter);
XGPushManager.registerPush(this, new XGIOperateCallback() {
@Override
public void onSuccess(Object o, int i) {
Log.w("XGPush","XGsuccess:"+o);
_isXGRegisterSuccess = true;
_pushToken = o.toString();
}
@Override
public void onFail(Object o, int i, String s) {
_isXGRegisterSuccess = false;
Log.w("XGPush","failed:"+i+", msg:"+s);
}
});
}
public void AddLocalNotification(String title, String content, String date, String hour, String min, int isRepeatEveryday)
{
if (_isXGRegisterSuccess)
{
XGLocalMessage msg = new XGLocalMessage();
msg.setTitle(title);
msg.setContent(content);
msg.setDate(date);
msg.setHour(hour);
msg.setMin(min);
int notifyIconId = getResources().getIdentifier("notify_icon", "drawable", getPackageName());
Log.e("notify icon id",notifyIconId+"");
msg.setSmall_icon(Integer.toString(notifyIconId));
int _hour = Integer.parseInt(hour);
int _min = Integer.parseInt(min);
msg.setRing(1);
XGPushManager.addLocalNotification(MainActivity.this,msg);
}
}
public void ClearAllLocalNotification()
{
Log.w("ClearAllLocalNotice","success");
if (_isXGRegisterSuccess)
{
XGPushManager.clearLocalNotifications(MainActivity.this);
}
}
}
如果需要在启动unity的Activity(即MainActivity)前做一些处理,你可以新建一个类SplashActivity继承Activity,设置SplashActivity为启动Activity,然后跳转到MainActivity。就可以在SplashActivity中做一些启动游戏前的处理。我们把上述工作做完后,就可以将这些类打包成一个jar包(AS如何打jar包,再说),导入unity导出的Android工程中就可以了。到这里通用接口的开发就完了。还剩下一项工作就是我们之前从unity中提取出来的SDK资源,我们需要添加到Android工程中,下面就介绍如何进行资源管理,把资源添加到unity导出的Android工程中。
注:
AS打jar包,在build.gradle最后加这些代码:
task makeJar(type: Jar, dependsOn:['build']){
archiveName = 'QuicksdkDemo.jar' //包名
from('build/intermediates/classes/debug/')
destinationDir = file('build/libs') //导出路径
exclude('com/taiyouxi/a3k/BuildConfig.class') //不打进包的文件
exclude('com/taiyouxi/a3k/BuildConfig\$*.class') //不打进包的文件
exclude('**/R.class') //不打进包的文件
exclude('**/R\$*.class') //不打进包的文件
include('com/taiyouxi/a3k/*.class') //需要打进包的文件
include('com/quicksdk/unity/*.class') //需要打进包的文件
include('com/weibo/*.class') //需要打进包的文件
include('com/wxapi/*.class') //需要打进包的文件
}
makeJar.dependsOn(build)
然后打开Terminal窗口,输入gradlew makeJar然后回车:
gradlew makeJar
打出的包在build>libs文件夹下如图:

Unity接入多个SDK的通用接口开发与资源管理(二)的更多相关文章
- Unity接入多个SDK的通用接口开发与资源管理(三)
接着上篇,介绍SDK资源的导入.首先介绍一下Android Studio工程. AS工程可以由多个Module组成,我们可以把某个Module作为我们打包的Module,其他的Module当做资源导入 ...
- Unity接入多个SDK的通用接口开发与资源管理(一)
每一款游戏都有接入多个SDK的需求.当接入多个SDK时会面临两个问题: (1)代码层面:每个SDK都有调用各自功能的方法,我们需要统一管理每个SDK提供的方法,这样会使每个SDK调用方便,使代码整洁易 ...
- Senparc.Weixin.MP SDK 微信公众平台开发教程(二):成为开发者
Senparc.Weixin.MP SDK 微信公众平台开发教程(二):成为开发者 这一篇主要讲作为一名使用公众平台接口的开发者,你需要知道的一些东西.其中也涉及到一些微信官方的规定或比较掩蔽的注意点 ...
- Senparc.Weixin.MP SDK 微信公众平台开发教程(二十):使用菜单消息功能
在<Senparc.Weixin.MP SDK 微信公众平台开发教程(十一):高级接口说明>教程中,我们介绍了如何使用“客服接口”,即在服务器后台,在任意时间向微信发送文本.图文.图片等不 ...
- Senparc.Weixin.MP SDK 微信公众平台开发教程(二十一):在小程序中使用 WebSocket (.NET Core)
本文将介绍如何在 .NET Core 环境下,借助 SignalR 在小程序内使用 WebSocket.关于 WebSocket 和 SignalR 的基础理论知识不在这里展开,已经有足够的参考资料, ...
- Senparc.Weixin.MP SDK 微信公众平台开发教程(二十二):如何安装 Nuget(dll) 后使用项目源代码调试
最近碰到开发者问:我使用 nuget 安装了 Senparc.Weixin SDK,但是有一些已经封装好的过程想要调试,我又不想直接附加源代码项目,这样就没有办法同步更新了,我应该怎么办? 这其实是一 ...
- Unity接入微信支付SDK 2022年版安卓篇
最近1年转了UE开发,博客更新的比较少,技术栈宽了不少,以后有空尽量多更新,也方便总结记忆 Unity接入微信支付整个过程坑比较多,网上之前的教程要么比较老,要么比较零碎,只能东拼西凑摸索,跑通后还是 ...
- Senparc.Weixin.MP SDK 微信公众平台开发教程(二十二):在 .NET Core 2.0/3.0 中使用 MessageHandler 中间件
概述 在 <Senparc.Weixin.MP SDK 微信公众平台开发教程(六):了解MessageHandler> 中我们已经了解了 MessageHandler 的运行原理和使用方法 ...
- Senparc.Weixin.MP SDK 微信公众平台开发教程(八):通用接口说明
一.基础说明 这里说的“通用接口(CommonAPIs)”是使用微信公众账号一系列高级功能的必备验证功能(应用于开发模式). 我们通过微信后台唯一的凭证,向通用接口发出请求,得到访问令牌(Access ...
随机推荐
- WKWebView 网络请求Header 丢失
WKWebView 是苹果手机上主要的H5加载控件,它相比UIWebView 有诸多优势.在次不做比较,但是它的坑缺比较多.网上也有很多的例子但是做的比较好的真不多,我在这里推荐俩博客供大家参考.ht ...
- Redis设计与实现——独立功能的实现
发布和订阅 频道的订阅和退订 struct redisServer{ //键是被订阅者频道 ,键是一个链表,记录所有订阅这个频道的客户端 dict *publish_channels } 订阅实现: ...
- jqgrid 获取选中用户的数据插入
因为查询出的表和被插入的表不是在同一个数据库,所以先从前台jqgrid表格中获取到数据后,再插入表中. 实现: 获取到jqgrid选中 的每行数据之后,发ajax请求把数据以json格式传入后台,后台 ...
- 《Java从入门到失业》第二章:Java环境(四):IDE集成环境
2.4IDE集成环境 在掌握了编写.编译和运行Java程序的基本步骤以后,你肯定就在想,这太麻烦了,有没有更好的工具?当然有了,那就是IDE.IDE就是专业的集成开发环境(Integrated Dev ...
- 配置react / antd 按需加载 并且使用less(react v16)
1.开启项目 并且执行 yarn eject 下载好我们需要的插件(babel-plugin-import less less-loader antd react-loadable ...
- Redis高级项目实战,都0202年了,还不会Redis?
导读 大家都听过1万小时定律,可事实真的是这样吗?做了1万小时的CRUD,不还只会CRUD吗,这年头不适当的更新自身下技术栈,出门和别人聊天吹牛的时候,都没拿不出手,(⊙o⊙)…Redis没入门的童鞋 ...
- golang map 声明,赋值
参考链接:https://blog.csdn.net/wide288/article/details/84303511 // 先声明map var m1 map[string]string// 再使用 ...
- edge 修改链接打开方式
我目前的edge版本是 Version 84.0.522.63 (Official build) (64-bit) 每次点击链接, 都是默认在原页面打开新标签, 不符合过往习惯. 修改方式 打开控制面 ...
- 移动web开发之布局
移动web开发流式布局 1.0 移动端基础 1.1浏览器现状 PC端常见浏览器:360浏览器.谷歌浏览器.火狐浏览器.QQ浏览器.百度浏览器.搜狗浏览器.IE浏览器. 移动端常见浏览器:UC浏览器,Q ...
- py_正则表达式练习
正则表达式: #正则表达式 #键盘数字6上的符号,^表示行的开始,$ 表示行的结束 #test = "tm queal Tomorrow Moon" ''' ^tm #匹配tm开头 ...