Unity 与 Android交互通信 之OPPO篇

前言

  本人是Android SDK方向的开发者,在游戏发行公司工作,因公司业务需求经常与Unity进行交互,借此机会让大家伙了解下Unity与Android交互的一些基础知识。

1.开发环境说明

  Unity和Android Studio所涉及到的SDK、JDK、NDK安装步骤新建工程等操作的不做说明

  Android Studio(AS)版本: 3.2.1

  Unity版本: 2018.2.0f2 破解版

2.实现效果

 Unity调用Andoroid网游OPPO SDK API 实现 登录 支付 退出 等交互 效果如下图

oppo_login.png
oppo_pay.png

oppo_exit.png

3.Android篇

要在Unity游戏项目中调用安卓API,有两种方式:

  1. Unity项目导出为Android工程(Build System选择Gradle),然后在AS中进行二次开发,添加交互功能。这样的方式开发起来很灵活,改动起来也很方便,但是就是很麻烦,因为每次改动都要打一回安卓工程。

  2. 将扩展功能打成jar包,然后将jar包导入到Unity中,直接使用。这样的方式,一次性写好通用交互层 ,不用频繁打安卓工程。

下面给大家说下第二种方式

3.1Android 工程

1.既然要和Android oppo SDK交互 ,需要去开发者申请对应参数和 SDK资源文档

 OPPO开发者网游SDK下载地址:https://open.oppomobile.com/wiki/doc#id=10470

  1. AS新建一个工程,因为需要和Unity交互(用到其中的类),因此需要把它提供的class.jar包放到AS工程libs下
    我Unity安装默认路径在D盘 D:\Unity 所以下面路径取决你Unity安装路径
    D:\Unity\Editor\Data\PlaybackEngines\AndroidPlayer\Variations\mono\Development\Classes\class.jar

  2. 需要按oppo 开发者文档要求 放入oppo资源(assets、libs和AndroidManifest.xml),然后可以按文档接口接入


    androidPro.png


3.2AS清单文件配置注意点

清单文件配置一定要有<meta-data android:name="unityplayer.UnityActivity" android:value="true" />,
不然会有莫名其妙的奇怪问题

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="com.unity.demo">

    <uses-permission android:name="android.permission.INTERNET" />
    <uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
    <uses-permission android:name="android.permission.CHANGE_NETWORK_STATE" />
    <uses-permission android:name="android.permission.ACCESS_WIFI_STATE" />
    <uses-permission android:name="android.permission.READ_PHONE_STATE" />
    <uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" />
    <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
    <uses-permission android:name="android.permission.MOUNT_UNMOUNT_FILESYSTEMS" />
    <uses-permission android:name="android.permission.SYSTEM_ALERT_WINDOW"/>
    <uses-permission android:name="android.permission.GET_TASKS" />
    <uses-permission android:name="android.permission.GET_ACCOUNTS" />
    <uses-permission android:name="android.permission.USE_CREDENTIALS" />
    <uses-permission android:name="android.permission.REQUEST_INSTALL_PACKAGES" />

    <application
        android:allowBackup="true"
        android:icon="@mipmap/ic_launcher"
        android:label="@string/app_name"
        android:roundIcon="@mipmap/ic_launcher_round">
        <activity
            android:name="com.unity.demo.MainActivity"
            android:screenOrientation="portrait"
            android:theme="@android:style/Theme.Translucent.NoTitleBar"
            android:configChanges="orientation|keyboardHidden|screenSize|screenSize|navigation">

            <intent-filter>
                <action android:name="android.intent.action.MAIN" />
                <action android:name="android.intent.action.VIEW"/>
                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
            <meta-data android:name="unityplayer.UnityActivity" android:value="true" />
        </activity>
       
        <!--oppo配置开始-->
        <meta-data
            android:name="debug_mode"
            android:value="false"/>
        <!--  日志开关,发布时候设置为false  -->
        <meta-data
            android:name="is_offline_game"
            android:value="false"/>
        <!--  true:单机游戏   false:网游  -->
        <meta-data
            android:name="app_key"
            android:value=""/>
        <!-- appKey,游戏上线时请务必替换成游戏自身的appkey -->
        <uses-library android:name="org.apache.http.legacy" android:required="false" />
        <!--oppo配置结束-->
    </application>
</manifest>

3.3Android 交互层代码编写

  首先需要让MainActivity继承UnityPlayerActivity,因为Unity导出的app的视图展示需要在UnityPlayerActivity下。

import android.os.Bundle;
import android.util.Log;
import android.view.KeyEvent;
import android.widget.Toast;
import com.nearme.game.sdk.GameCenterSDK;
import com.nearme.game.sdk.callback.ApiCallback;
import com.nearme.game.sdk.callback.GameExitCallback;
import com.nearme.game.sdk.common.model.biz.PayInfo;
import com.nearme.game.sdk.common.model.biz.ReportUserGameInfoParam;
import com.unity3d.player.UnityPlayer;
import com.unity3d.player.UnityPlayerActivity;
import org.json.JSONException;
import org.json.JSONObject;
import java.io.UnsupportedEncodingException;
import java.net.URLEncoder;
import java.util.HashMap;
import java.util.Map;
import java.util.TreeMap;
/**
  *android 与Unity交互层  数据均为模拟游戏数据
  */
public class MainActivity extends UnityPlayerActivity {
    private static String appSecret = ""; //此处应该填写 oppo后台申请下来的appSecret参数
    public final  String TAG="UnityForOppo";

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        doInit();  
    }
    /**
     * 初始化
     */
    public void doInit(){
        Log.i(TAG,"doInit");
        GameCenterSDK.init(appSecret, this);
        Toast.makeText(MainActivity.this,"初始化成功",Toast.LENGTH_LONG).show();
    }

    /**
     * 登录
     */
    public void doLogin(){
        Log.i(TAG,"doLogin");
        GameCenterSDK.getInstance().doLogin(this, new ApiCallback() {
            //登录成功的回调
            @Override
            public void onSuccess(String msg) {
                Toast.makeText(MainActivity.this,"获取登录状态:====="+msg.toString(),Toast.LENGTH_LONG).show();
                GameCenterSDK.getInstance().doGetTokenAndSsoid(new ApiCallback() {
                    @Override
                    public void onSuccess(String resultMsg) {
                        Log.i(TAG,"登录成功获取的msg:====="+resultMsg.toString());
                        String jsonString = "";
                        try {
                            JSONObject   json = new JSONObject(resultMsg);
                            String token = json.getString("token");
                            String ssoid = json.getString("ssoid");
                            json.put("token", URLEncoder.encode(token, "UTF-8"));
                            json.put("ssoid",json.getString("ssoid"));
                       
                            jsonString=json.toString();
                            Log.i(TAG,"登录成功向Unity发送消息:====="+jsonString);
                            /**
                             * 向Unity传递消息
                             * 第1个参数为Unity场景中用于接收android消息的对象名称
                             * 第2个参数为对象上的脚本的一个成员方法名称(脚本名称不限制)
                             * 第3个参数为Unity方法的参数
                             */
                            UnityPlayer.UnitySendMessage("AndroidSDKListener", "LoginCallback", jsonString);
                        } catch (JSONException e) {
                            e.printStackTrace();
                        } catch (UnsupportedEncodingException e) {
                            e.printStackTrace();
                        }
                    }

                    @Override
                    public void onFailure(String resultMsg, int code) {
                        UnityPlayer.UnitySendMessage("AndroidSDKListener", "LoginCallback", resultMsg);
                    }
                });

            }
            //登录失败的回调
            @Override
            public void onFailure(String msg, int code) {
                //回调oppo登录失败,把失败的消息发给Unity
                UnityPlayer.UnitySendMessage("AndroidSDKListener", "LoginCallback", msg);
            }
        });
    }

    /**
     * 支付(唤起支付参数为模拟数据 真实应该由游戏服务器端传入)
     */
    public void doPay(){
        Log.i(TAG,"doPay");
        String orderId=System.currentTimeMillis()+"";//订单号 建议由游戏服务器提供
        String productDesc="优惠月礼包"; //商品描述
        String  productName="钻石"; //商品名称
        String  GAME_OPPO_URL="http://192.168.1.1:8080/order/oppo";//支付回调地址,由服务器提供
        //参数1  游戏订单号  参数2 附加参数 传什么都可以 这里传入订单号  参数3 为支付金额 单位分
        PayInfo payInfo = new PayInfo(orderId, orderId, 1);
        payInfo.setProductDesc(productDesc);
        payInfo.setProductName(productName);
        payInfo.setCallbackUrl(GAME_OPPO_URL);
        GameCenterSDK.getInstance().doPay(this, payInfo, new ApiCallback() {

            @Override
            public void onSuccess(String msg) {
                Log.i(TAG,"PAY SUC");
            }

            @Override
            public void onFailure(String msg, int code) {
                Log.i(TAG,"PAY Fail========"+msg);
            }
        });
    }

    /**
     * 上报游戏数据 给OPPO
     */
    public  void doSubUserInfo (){
        Map<String, String> mRoleInfo =new HashMap<String, String>();
        mRoleInfo.put("type", "createRole");// 以下场景必传[enterServer(登录),levelUp(升级),createRole(创建角色),exitServer(退出)]
        mRoleInfo.put("roleId", "123456");// 当前登录的玩家角色ID,若无,可传入userid
        mRoleInfo.put("roleName", "天下第一");// 当前登录的玩家角色名,不能空
        mRoleInfo.put("roleLevel", "10");// 当前登录的玩家角色等级,不能为空,必须为数字,且不能为null,若无,传入0
        mRoleInfo.put("serverId", "1001");// 当前登录的游戏区服ID,不能为空,必须为数字,若无,传入0
        mRoleInfo.put("serverName", "Oppo32服");// 当前登录的游戏区服名称,不能为空,长度不超过50,不能为null,若无,传入“无”

        if (mRoleInfo != null && "createRole".equals(mRoleInfo.get("type")) || ("enterServer".equals(mRoleInfo.get("type"))) || ("levelUp".equals(mRoleInfo.get("type")))) {
            String serverId = mRoleInfo.get("serverId");
            String serverName = mRoleInfo.get("serverName");
            String roleId = mRoleInfo.get("roleId");
            String roleName = mRoleInfo.get("roleName");
            String roleLevel = mRoleInfo.get("roleLevel");
            GameCenterSDK.getInstance().doReportUserGameInfoData(new ReportUserGameInfoParam(roleId, roleName, Integer.valueOf(roleLevel), roleId, roleName, "", new TreeMap<String, Number>()), new ApiCallback() {
                @Override
                public void onSuccess(String msg) {
                }
                @Override
                public void onFailure(String msg, int code) {

                }
            });
        }

    }
    /**
     * 退出游戏
     */
    public void  doExitGame(){
        Log.i(TAG,"DoExitGame");
        GameCenterSDK.getInstance().onExit(this,new GameExitCallback() {
            @Override
            public void exitGame() {
                finish();
                System.exit(0);
                android.os.Process.killProcess(android.os.Process.myPid());
            }
        });
    }

    @Override
    public boolean onKeyDown(int keyCode, KeyEvent event) {
        Log.i(TAG, "onKeyDown");
        if ((keyCode == KeyEvent.KEYCODE_BACK)) {
            Log.i(TAG, "DoExitGameDoExitGameDoExitGame");
            doExitGame();
            return false;
        } else {
            return super.onKeyDown(keyCode, event);
        }
    }

3.4AS生成JAR包

 app/build.gradle 下加入生成jar包Task

task makeJar(type: Copy) {
    def jarPath = 'build/intermediates/packaged-classes/debug/'  
    def jarName = 'classes.jar'
    from(jarPath)
    into('build/outputs/')  //输出路径为outputs/
    include(jarName)
    rename(jarName, 'unityPlug-1.4.jar') //重名为xxx.jar
}



建议: 先Rebuild Project一下生成packaged-classes文件,然后在Gradle \other下能看到makeJar 点击执行Task

makeJar.png

4.Unity篇

  1. 新建一个Unity工程,在Assets目录下新建Plugins/Android/ 目录
  2. 将Android中的assets、res、AndroidManifest.xml 、libs下的gamesdk-20190910.aar还有生成的unityPlug-1.4.jar 放入Assets/Plugins/Android目录下(在Unity这些东西都会被当做资源处理)
  3. 在Assets/Scenes/ 建立一个场景,在场景上创建一个Canvas,并创建一个名为"AndroidSDKListener"的对象,在AndroidSDKListener之下再放三个按钮触发安卓 登录 支付 退出游戏
  4. 创建一个btnClick.cs文件,将脚本挂载在AndroidSDKListener对象 如图所示


android.png
scenes.png
unityPro.png


4.1 Unity中C#代码实现

using System.Collections;
using System.Collections.Generic;
using UnityEngine;
using UnityEngine.UI;
public class btnClick : MonoBehaviour
{

    private AndroidJavaClass jc;
    private AndroidJavaObject jo;
    private Button btnLogin;
    private Button btnPay;
    private Button btnExitGame;
   
    public void Start()
    {   
        //固定写法
        jc = new AndroidJavaClass("com.unity3d.player.UnityPlayer");
        jo = jc.GetStatic<AndroidJavaObject>("currentActivity");
        btnLogin = transform.Find("BtnLogin").GetComponent<Button>(); //登录
        btnPay= transform.Find("BtnPay").GetComponent<Button>();   //支付
        btnExitGame = transform.Find("BtnExit").GetComponent<Button>();   //退出游戏
        btnLogin.onClick.AddListener(OnBtnLoginClickHandler);
        btnPay.onClick.AddListener(OnBtnPayClickHandler);
        btnExitGame.onClick.AddListener(OnBtnExitGameClickHandler);
    }
  
    /**
      * 登录调用
     */
    private void OnBtnLoginClickHandler()
    {
        jo.Call("doLogin");
    }
   
  /**
      * 支付调用
     */
     private void OnBtnPayClickHandler()
     {
         jo.Call("doPay");
     }

    /**
     * 退出游戏点击调用
     */
    private void OnBtnExitGameClickHandler()
    {
        jo.Call("doExitGame");
       // Application.Quit();//调用C#退出应用
    }
 
    /**
      *接收从安卓端传过来消息
      *方法体 UnityPlayer.UnitySendMessage("AndroidSDKListener", "LoginCallback", msg);和参数2对应
     */
    public  void LoginCallback(string msg)
    {
        Debug.Log("Login_with_msg : " + msg);
    }
}

4.2 打包测试

File ----Build Settings----Android----Player Settings -----Build System-----选择Internal (直接生成APK整包)
配置好签名(jdk sdk ndk均早配置好了) 选择bulid 静候几秒 完整的APK就搞定了
登录、支付、 退出效果 上面已经展示了,看下安卓登录回调成功后向Unity发送的消息

login_msg.png

结语

  记录下自己的学习和工作经验,分享给有需要的人。如果有那里写的不对或者不理解,欢迎大家的指正。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 158,117评论 4 360
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 66,963评论 1 290
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 107,897评论 0 240
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 43,805评论 0 203
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,208评论 3 286
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,535评论 1 216
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,797评论 2 311
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,493评论 0 197
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,215评论 1 241
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,477评论 2 244
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 31,988评论 1 258
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,325评论 2 252
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 32,971评论 3 235
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,055评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,807评论 0 194
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,544评论 2 271
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,455评论 2 266