美文网首页Unity学习小组征服Unity3dunity3D技术分享
5、与iOS、Android的交互 实践篇——主动调用

5、与iOS、Android的交互 实践篇——主动调用

作者: 万士辰 | 来源:发表于2016-08-28 19:07 被阅读2451次

    本文属于「Unity与iOS、Android平台的整合」系列文章之一,转载请注明出处
    主要讲解Unity与iOS、Android平台相互主动调用实现

    零、前言

    由于本文涉及到较多基础知识,建议读者们把前面几篇补完后再往下看

    一、前期工作

    1.控制台

    Debug.Log();

    这个API是我们比较常用的Log输出工具,但是只能在调试的时候看到,所以我自己写了个在设备上显示Log的简单控制台,具体制作和使用大家可以在我之后附带的工程中看到。

    using System;
    using System.Text;
    using UnityEngine;
    using UnityEngine.UI;
    
    public class MyConsole : MonoBehaviour
    {
        private static MyConsole Instance;
        void Awake() { Instance = this; }
        [SerializeField]
        private Text logTxt;
    
        private static StringBuilder log = new StringBuilder("这是控制台:");
        public static void Print(string str)
        {
            log.Append("\n\nTime:");
            log.Append(DateTime.Now.ToString("HH:mm:ss"));
            log.Append("\u3000");
            log.Append(str);
            Instance.logTxt.text = log.ToString();
    
            Instance.logTxt.rectTransform.anchoredPosition = Vector2.zero;
        }
    }
    
    控制台的编辑器配置 控制的显示

    2.创建一个C#的类

    ConnectUnityToiOSAndroid.cs
    这个类需要挂载在一个场景内存在的GameObject上,我选择了摄像机,当然你也可以选择其他,但是必须要有一个确定的name,这个涉及到iOS、Android主动调用Unity是否成功的问题,不理解的话请仔细看上一篇文章 挂载物体示意图

    为了方便,我会把C#相关的示例代码都写在这里。
    为了方便,面向Unity开发者的接口我都直接通过编辑器拖拽挂载的方式用按钮调用。
    实际开发中原理都是一样的,但是不建议像我这样简单粗暴地使用~

    ConnectUnityToiOSAndroid.cs

    using UnityEngine;
    public class ConnectUnityToiOSAndroid : MonoBehaviour
    {
    }
    

    3.创建一个Objective-C的类

    这两个文件需要关联到Xcode工程中

    关联示意图

    ConnectUnityToiOS.h

    #ifndef ConnectUnityToiOS_h
    #define ConnectUnityToiOS_h
    #endif /* ConnectUnityToiOS_h */
    @interface ConnectUnityToiOS:NSObject
    @end
    

    ConnectUnityToiOS.mm

    #import "ConnectUnityToiOS.h"
    @implementation ConnectUnityToiOS
    static ConnectUnityToiOS *instance = nil;
    +(ConnectUnityToiOS *)sharedInstance{
        @synchronized(self) {
            if(instance == nil) {
                instance = [[[self class] alloc] init];
            }
        }
        return instance;
    }
    @end
    extern "C"{
    }
    

    4.创建一个Java的类

    这个文件需要关联到AndroidStudio或Eclipse工程

    关联示意图

    ConnectUnityToAndroid.java

    //游戏使用的包名
    package com.wsc.ConnectUnityToAndroid;
    import android.app.AlertDialog;
    import com.unity3d.player.UnityPlayer;
    public class ConnectUnityToAndroid {
    }
    

    二、Unity主动调用iOS、Android

    首先,我们需要设计一个调用需求

    现在策划说:需要一个原生弹窗提示

    好的,需求有了,可以开始设计接口了

    C#需要向外提供一个接口,供其他C#代码调用
    OC需要提供一个接口,用于调用iOS系统弹窗
    Java需要提供一个接口,用于调用Android系统弹窗

    0.Unity内部

    在ConnectUnityToiOSAndroid.cs中写两个方法,写得很简单,大家看一下注释就清楚了

    //C#面向Unity开发者的接口
    public void ShowNativeTip()
    {
        MyConsole.Print("显示原生弹窗");
        ShowNativeTip_();
    }
    #if UNITY_EDITOR
    //运行在编辑器环境的接口,一般用于编辑器内模拟数据
    private void ShowNativeTip_()
    {
        MyConsole.Print("这是在Unity环境");
    }
    #endif
    

    1.Unity -> iOS

    在ConnectUnityToiOSAndroid.cs中关联C++的接口

    #if UNITY_IOS
    //运行在iOS环境的接口,关联到C++接口
    [DllImport("__Internal")]
    private static extern void ShowNativeTip_();
    #endif
    

    在ConnectUnityToiOS.mm中用OC写调用系统弹窗

    -(void)ShowNativeTip{
        UIAlertView *alertView = [[UIAlertView alloc] initWithTitle:@"这是iOS的原生弹窗"
                                                            message:nil
                                                           delegate:self
                                                  cancelButtonTitle:@"确定"
                                                  otherButtonTitles:nil];
        [alertView show];
    }
    

    在ConnectUnityToiOS.mm中用C++调用OC的接口

    extern "C"{
        void ShowNativeTip_(){
            [[ConnectUnityToiOS sharedInstance] ShowNativeTip];
        }
    }
    

    运行效果~


    iOS下调用原生弹窗运行效果

    2.Unity -> Android

    在ConnectUnityToiOSAndroid.cs中关联Java的接口

    #if  UNITY_ANDROID
    //运行在Android环境的接口
    private static void ShowNativeTip_()
    {
        //注意,这边一定要把游戏的包名写正确,否则找不到类。
        using (AndroidJavaClass jc = new AndroidJavaClass("com.wsc.ConnectUnityToiOSAndroid.ConnectUnityToAndroid"))
        {
            jc.CallStatic("ShowNativeTip_");
        }
    }
    #endif
    

    在ConnectUnityToAndroid.java中用Java调用系统弹窗

    public static void ShowNativeTip_(){
        AlertDialog.Builder builder = new AlertDialog.Builder(UnityPlayer.currentActivity)
                .setMessage("这是Android的原生弹窗")
                .setPositiveButton("确定",null)
        builder.show();
    }
    

    运行效果~

    Andorid下调用原生弹窗运行效果

    三、iOS、Android主动调用Unity

    首先,我们又需要设计一个调用需求,诶为什么要说又

    现在策划说:需要获得原生弹窗关闭的时候时间

    好的,需求有了,可以开始设计接口了

    C#需要向OC、Java提供一个接口,以接收原生弹窗关闭的通知
    OC需要在弹窗关闭时主动调用C#
    Java需要在弹窗关闭时主动调用C#

    0.Unity

    将ConnectUnityToiOSAndroid.cs挂载在某个物体上,我选择了名为「Camera」的摄像机物体。
    在ConnectUnityToiOSAndroid.cs中写一个方法,写得很简单,大家看一下注释就清楚了

    //C#面向iOS、Android的接口
    private void NativeTipClosed()
    {
        MyConsole.Print("原生弹窗被关闭了");
    }
    

    1.iOS -> Unity

    在ConnectUnityToiOS.mm中写OC添加系统弹窗按钮的监听

    - (void)alertView:(UIAlertView *)alertView clickedButtonAtIndex:(NSInteger)buttonIndex{
        if (buttonIndex==0) {
            UnitySendMessage("Camera", "NativeTipClosed", "");
        }
    }
    

    运行效果~

    iOS下主动调用Unity运行效果

    2.Android -> Unity

    在ConnectUnityToAndroid.java中写Java添加系统弹窗按钮的监听
    由于Java这边可以直接添加匿名事件,我就不重新分一个函数写了,以下是和原文的区别示意图

    相对上文添加的内容示意图
    public static void ShowNativeTip_(){
        UnityPlayer.currentActivity.runOnUiThread(new Runnable() {
            @Override
            public void run() {
                AlertDialog.Builder builder = new AlertDialog.Builder(UnityPlayer.currentActivity)
                         .setMessage("这是Android的原生弹窗")
                         .setPositiveButton("确定", new DialogInterface.OnClickListener() {
                            @Override
                            public void onClick(DialogInterface dialog, int which) {
                                UnityPlayer.UnitySendMessage("Camera", "NativeTipClosed", "");
                            }
                        });
                builder.show();
            }
        });
    }
    

    运行效果~


    Android下主动调用Unity运行效果

    四、收个尾

    以上内容为Unity与iOS、Android之间相互主动调用的所有内容。
    如果你不知道如何导出工程,请看一下前几篇文章
    如果你不能理解我的实现步骤,请看一下前几篇文章
    附上我的Unity示例工程,里面包含了上文中所有代码,打包编译,进行对应修改应该就能跑~
    下载链接: https://pan.baidu.com/s/1ge7yRxp 密码: uppp

    关于传递参数的实现

    由于上面的内容写得非常详细,如果结合理论篇的内容完全可以自己实现,我一直在考虑有没有必要如此喂饭。
    这样吧,大家可以先自己尝试一下,如果在思考、尝试之后还是不会,可以在评论中留个言,我会考虑再出一篇传递参数的实践篇~

    相关文章

      网友评论

      • WHY_小白:您好.博主,请教您一个问题. Unity作为Android的一个模块,嵌入到Android中, 嵌入以后点击Unity场景中的模型,改模型向Andriod反馈数据,比如,模型的名字啊什么的. 这个改怎么做,?
        WHY_小白:@万士辰 好的,谢谢您
        万士辰:@WHY_3317 相当于调用Java的接口,传递数据,文中有明确说明如何实现
      • b95f0ca6ca78:您好,我想实现调用iOS SDK 并返回参数,逻辑也和您的这个参数传递 一样吗?
        万士辰:@CallMeDaKing 是我没讲清楚还是什么吗,我这边写的东西一般都是需要跟着一步一步尝试的,没有直接给出最终代码,主要是自己希望能帮助大家掌握核心原理,而不是代码甩一脸
        b95f0ca6ca78:@万士辰 自己摸索三四天,终于算是把这个来龙去脉搞得算是明白了,感谢。
        万士辰:嗯,只是返回类型从void 修改成其他的
      • WolfXu:请教个问题,Unity导出到Android Studio,如果Unity那边又有修改,是不是得重新导出?这样的话感觉比较麻烦。有没有比较好的方案能解决这个问题呢。
        万士辰:我的习惯是写一个自动配置,这样每次导包之后都能自动完成配置
      • cb646917403f:把unity移植到Android中我遇到一个问题,请求帮助。unity移植到已经存在的Android项目里(eclipse),退出时用的Application.Quit(),程序直接退到Android的主界面,闪一下就退出程序了。请楼主帮忙指导一下。
        万士辰:@_星空 unity游戏其实只是一个activity,切换或者销毁activity就好了
        cb646917403f:@万士辰 那我应该怎么才能退出unity而让程序停留在Android界面呢?因为我要把unity的几个小游戏集成到Android中,而每个demo游戏又不相关,不能用Scene.LoadLevel()进行不同的场景的切换,只能通过Android的主页来选怎进入那个小游戏。可是现在直接退出了而没有在Android的MainActivety停留。我该怎么做呢?
        万士辰:@_星空 主动调用Application.Quit()就是直接退出程序啊
      • 睿少:是不是相当于unity与iOS/Android的交互,代码实现全在unity中,在iOS/Android端就只需将它导入程序就是了?
        睿少:就拿你这个demo来说,如果我要点击确定后push 到一个 ViewController 应该怎么实现?
        是在ConnectUnityToiOS.mm 里面的- (void)alertView:(UIAlertView *)alertView clickedButtonAtIndex:(NSInteger)buttonIndex{
        if (buttonIndex==0) {
        UnitySendMessage("Camera", "NativeTipClosed", "");
        }
        }
        实现吗?
        睿少:我只会iOS开发并不会unity的开发啊 :sob: :sob:
        万士辰:@睿少 并不是的,你应该看一下理论篇,整个交互方式是,C#调用 C++,然后C++再调用iOS的方法,所以有些代码需要在iOS里用OC写,比如这篇文章的显示原生窗口。具体你跟着流程走一遍就明白了

      本文标题:5、与iOS、Android的交互 实践篇——主动调用

      本文链接:https://www.haomeiwen.com/subject/fdwtettx.html