iOS与Unity3d交互

前言

  • 最近在实验室做了一个项目,用到了蓝牙通讯和U3D的交互,都有很多坑,如:IOS与Unity3D界面之间的跳转、数据的传递等操作。过程其实不是很难,只是比较繁琐,刚开始可能一头雾水,慢慢学习了解后,就会发现其实很简单。
  • 蓝牙的一些知识在这里:IOS蓝牙通讯详解
  • 整篇文章主要介绍以下两点
  1. IOS与Unity之间的界面切换
  2. IOS与Unity之间的函数调用以及传值

IOS与Unity之间的界面切换

在IOS和Unity3D跨平台开发中,基本就是两个平台之间的数据交互和界面切换。而界面交互是重中之重。利用Unity3D可以实现一些IOS原生界面所不具有的效果,而IOS原生界面则在整个程序界面中又显得更和谐一些。所以无论是功能还是美观,两者之间的界面交互都很常用。

前期Unity准备

我们要用Unity和IOS的界面切换和数据交互,就必然要先实现一个Unity程序。我就默认读者都会一些Unity基础和OC技术吧。

  • 首先建立一个新的Unity程序,在场景中拖入一个Cube。我们可以利用这个cube来展示Unity和IOS之间的数据交互。


    一个简单的Unity场景
  • 我们在Camera下建立邦定一个脚本,在脚本中提供各类接口。
一个简单的测试脚本
  • 我们编辑脚本,给出一些基本的接口。
using UnityEngine;
using System.Collections;
using System.Runtime.InteropServices;
public class Test : MonoBehaviour {
    public GameObject cube;

    // DllImport这个方法相当于是告诉Unity,有一个unityToIOS函数在外部会实现。
    // 使用这个方法必须要导入System.Runtime.InteropServices;
    [DllImport("__Internal")]
    private static extern void unityToIOS (string str);

    void OnGUI()
    {
        // 当点击按钮后,调用外部方法
        if (GUI.Button (new Rect (100, 100, 100, 30), "跳转到IOS界面")) {
            // Unity调用ios函数,同时传递数据
            unityToIOS ("Hello IOS");
        }
    }
    // 向右转函数接口
    void turnRight(string num){
        float f;
        if (float.TryParse (num, out f)) {// 将string转换为float,IOS传递数据只能用以string类型
            Vector3 r = new Vector3 (cube.transform.rotation.x, cube.transform.rotation.y - 10f, cube.transform.rotation.z);
            cube.transform.Rotate (r);
        }
    }
    // 向左转函数接口
    void turnLeft(string num){
        float f;
        if (float.TryParse (num, out f)) {// 将string转换为float,IOS传递数据只能用以string类型
            Vector3 r = new Vector3 (cube.transform.rotation.x, cube.transform.rotation.y - 10f, cube.transform.rotation.z);
            cube.transform.Rotate (r);
        }
    }
}
  • 在给出接口之后,我们将Unity工程导出到IOS工程,点击File->Build Settings->IOS,(在build之前,我们需要填写一些ID信息。其实这类信息也可以不现在填,在导出工程后,可以在Xcode里面填写)然后点击build。
导出工程到Xcode

从IOS界面切换到Unity界面

  • 从IOS界面切换到Unity界面,需要做的是拦截Unity界面的启动,当我们需要加载Unity界面的时候才让其启动。而要拦截Unity界面,就需要先搞明白Unity界面加载的顺序。main.mm文件是整个Unity工程的入口。在main函数中先进行一些初始化和注册操作,然后执行UIApplicationMain(argc, argv, nil, [NSString stringWithUTF8String:AppControllerClassName]),这一句就执行了UnityAppController对象。在UnityAppController中,Unity整个程序就算是启动起来了。所以我们需要了解UnityAppController对象中一些方法的执行顺序以便于我们拦截Unity界面。在UnityAppController这个对象中,一些核心的方法已经有打印,如果想要知道所有方法的执行顺序,我们可以在每一个方法里面添加打印方法,此处为了简单,我就利用已经有的打印函数。
const char* AppControllerClassName = "UnityAppController";
int main(int argc, char* argv[])
{
    @autoreleasepool
    {
        UnityInitTrampoline();
        UnityParseCommandLine(argc, argv);
        RegisterMonoModules();
        NSLog(@"-> registered mono modules %p\n", &constsection);
        RegisterFeatures();

        // iOS terminates open sockets when an application enters background mode.
        // The next write to any of such socket causes SIGPIPE signal being raised,
        // even if the request has been done from scripting side. This disables the
        // signal and allows Mono to throw a proper C# exception.
        std::signal(SIGPIPE, SIG_IGN);

        UIApplicationMain(argc, argv, nil, [NSString stringWithUTF8String:AppControllerClassName]);
    }

    return 0;
}
  • 运行项目后,我们看到打印
打印结果
  • - (void)applicationDidBecomeActive:(UIApplication*)application这个方法中,执行了startUnity:方法。而这个方法一旦执行,Unity界面就启动起来了。所以我们需要做的拦截操作就在startUnity:之前,即- (void)applicationDidBecomeActive:(UIApplication*)application方法中调用IOS原生界面。
  • 观察一下- (void)applicationDidBecomeActive:(UIApplication*)application方法,中间有一句[self performSelector:@selector(startUnity:) withObject:application afterDelay:0];,我们要做的就是把startUnity替换成我们自己的方法,在自己的方法中调用IOSUI,然后在UI中写一些逻辑来让IOS可以跳转到Unity界面。
  - (void)applicationDidBecomeActive:(UIApplication*)application
{
    ::printf("-> applicationDidBecomeActive()\n");
    [self removeSnapshotView];
    if(_unityAppReady)
    {
        if(UnityIsPaused() && _wasPausedExternal == false)
        {
            UnityWillResume();
            UnityPause(0);
        }
        UnitySetPlayerFocus(1);
}
    else if(!_startUnityScheduled)
    {
        _startUnityScheduled = true;
        [self performSelector:@selector(startSelfIOSView) withObject:application afterDelay:0];
    }
    _didResignActive = false;
}
- (void)startSelfIOSView
{
    UIViewController *vc = [[UIViewController alloc] init];
    vc.view.backgroundColor = [UIColor blueColor];
    vc.view.frame = [UIScreen mainScreen].bounds;
    UIButton *btn = [[UIButton alloc]initWithFrame:CGRectMake(100, 100, 70, 30)];
    btn.backgroundColor = [UIColor whiteColor];
    [btn setTitle:@"跳转到Unity界面" forState:UIControlStateNormal];
    [btn addTarget:self action:@selector(startUnity:) forControlEvents:UIControlEventTouchUpInside];
    [vc.view addSubview:btn];
    [_window addSubview:vc.view];
}
  • 运行结果,中间IOS界面运行的时候,我点击了白色的按钮跳转。
IOS界面跳转到Unity界面

从Unity界面跳转到IOS界面

在Unity脚本中,我们已经添加了一个按钮,并且给了接口。所以我们可以利用这个按钮实现跳转到IOS界面。

在IOS工程中调用Unity函数的方式是利用C语言接口。在IOS工程中利用形如

extern "C"
{
    void functionName(parameter){
    // do something
    }
}

这样的形式来调用Unity中的函数。

因为Unity界面跳转到IOS界面涉及到了暂停Unity所以我们需要实现一个单例来判断Unity的暂停或启动

//  LARManager.m
//  Unity-iPhone
//
//  Created by 柳钰柯 on 2016/12/15.
//
//
#import "LARManager.h"
@implementation LARManager
+ (instancetype)sharedInstance
{
    static LARManager *manager;
    static dispatch_once_t onceToken;
    dispatch_once(&onceToken, ^{
        manager = [[self alloc] init];
    });
    return manager;
}
- (instancetype)init
{
    if (self = [super init]) {
        self.unityIsPaused = NO;
        NSLog(@"单例初始化成功");
    }
    return self;
}
@end
  • 跳转到IOS界面,我们需要实现在Unity脚本中声明的unityToIOS函数。值得注意的是:在extern "C"中,不能用OC的selfself.window获取到appControllerwindow,必须使用UnityAppController对象提供的方法GetAppController()UnityGetGLView()来获取。因为要返回之前的界面,所以我需要声明一个强引用变量来引用之前的view。在UnityAppController.h中声明@property (strong, nonatomic) UIViewController *vc,然后在刚才的startSelfIOSView中添加一句self.vc = vc;
实现单例之后的startSelfIOSView和unityToIOS函数
- (void)startSelfIOSView
{
    // 单例变量unity没有暂停,设置为no
    [LARManager sharedInstance].unityIsPaused = NO;
    // IOS原生界面
    UIViewController *vc = [[UIViewController alloc] init];
    vc.view.backgroundColor = [UIColor blueColor];
    vc.view.frame = [UIScreen mainScreen].bounds;
    // 添加按钮
    UIButton *btn = [[UIButton alloc]initWithFrame:CGRectMake(100, 100, 200, 30)];
    btn.backgroundColor = [UIColor whiteColor];
    btn.titleLabel.backgroundColor = [UIColor blackColor];
    [btn setTitle:@"跳转到Unity界面" forState:UIControlStateNormal];
    [btn addTarget:self action:@selector(startUnity:) forControlEvents:UIControlEventTouchUpInside];
    // 在界面上添加按钮
    [vc.view addSubview:btn];
    self.vc = vc;
    NSLog(@"设置界面为IOS界面");
    self.window.rootViewController = vc;
}
extern "C"
{
    // 对Unity中的unityToIOS方法进行实现
    void unityToIOS(char* str){
        // Unity传递过来的参数
        NSLog(@"%s",str);
        UnityPause(true);
        // 跳转到IOS界面,Unity界面暂停
        [LARManager sharedInstance].unityIsPaused = YES;
        // GetAppController()获取appController,相当于self
        // UnityGetGLView()获取UnityView,相当于_window
        // 点击按钮后跳转到IOS界面,设置界面为IOS界面
        GetAppController().window.rootViewController = GetAppController().vc;
    }
}
  • 实现效果:
Unity和IOS互相跳转

IOS与Unity之间的函数调用以及传值

在上一步中,其实我们已经实现了IOS和Unity函数调用和一部分传值。现在更详细的说明一下。

Unity调用IOS函数并进行传值

  • Unity调用IOS函数其实就是先在Unity脚本中声明一个函数接口,然后在IOS程序中实现。其中
    [DllImport("__Internal")]
    private static extern void functionName (ParameterType Parameter);

为固定格式。这一句表明会在外部引用一个叫functionName (ParameterType Parameter)的函数,参数为Parameter。在IOS程序中实现上一步中已经讲过,不再赘述。

IOS调用Unity函数并进行传值

IOS调用Unity函数需要用到UnitySendMessage方法,方法中有三个参数

UnitySendMessage("gameobject", "Method",msg);
向unity发送消息
参数一为unity脚本挂载的gameobject
参数二为unity脚本中要调用的方法名
参数三为传递的数据,*注意:传递的数据只能是char 类型

我们利用在Unity工程中已经写好的接口来试试数据的传递。

  • 首先在我们需要添加2个按钮,按钮需要实现点击后,传递数据过去,使视野中的cube(正方体)向左向右旋转。
  • 我们只需要在原生的Unity界面中添加按钮。在原生界面中添加和在ios工程中添加是差不多的。在界面还没有显示的时候,添加进去即可,我在- (BOOL)application:(UIApplication*)application didFinishLaunchingWithOptions:(NSDictionary*)launchOptions这和方法中添加按钮。代码如下:
- (BOOL)application:(UIApplication*)application didFinishLaunchingWithOptions:(NSDictionary*)launchOptions
{
    /*省略了一些Unity的操作*/
  /*.....*/
    [self createUI];
    [self preStartUnity];
  // 添加右旋按钮
  UIButton *rightBtn = [[UIButton alloc] initWithFrame:CGRectMake(10, 150, 100, 30)];
  rightBtn.backgroundColor = [UIColor whiteColor];
  [rightBtn setTitleColor:[UIColor blackColor] forState:UIControlStateNormal];
  [rightBtn setTitle:@"向右旋转" forState:UIControlStateNormal];
  [rightBtn addTarget:self action:@selector(turnRight) forControlEvents:UIControlEventTouchUpInside];
  self.rightBtn = rightBtn;
  // 添加左旋按钮
  UIButton *leftBtn = [[UIButton alloc] initWithFrame:CGRectMake(10, 200, 100, 30)];
  leftBtn.backgroundColor = [UIColor whiteColor];
  [leftBtn setTitleColor:[UIColor blackColor] forState:UIControlStateNormal];
  [leftBtn setTitle:@"向左旋转" forState:UIControlStateNormal];
  [leftBtn addTarget:self action:@selector(turnLeft) forControlEvents:UIControlEventTouchUpInside];
  self.leftBtn = leftBtn;
  // 在Unity界面添加按钮
  [UnityGetGLViewController().view addSubview:_rightBtn];
  [UnityGetGLViewController().view addSubview:_leftBtn];
    // if you wont use keyboard you may comment it out at save some memory
    [KeyboardDelegate Initialize];
    return YES;
}
// 左旋按钮响应事件
- (void)turnRight
{
    const char* str = [[NSString stringWithFormat:@"10"] UTF8String];
    UnitySendMessage("Main Camera", "turnRight", str);
}
// 右旋按钮响应事件
- (void)turnLeft
{
    const char* str = [[NSString stringWithFormat:@"10"] UTF8String];
    UnitySendMessage("Main Camera", "turnLeft", str);
}
  • 在添加完成按钮后,我们向摄像机发送消息。因为我们的脚本是绑定在主摄像头上的,所以第一个参数为主摄像机的名字,第二个参数为脚本中的方法,第三个参数为数据。
  • 因为我个人粗心,在Unity脚本中,关于旋转的加减号忘记修改了,导致点击左旋按钮后物体也是右旋。只需要把脚本中turnLeft函数的-号改成+号就行了。运行如下:
IOS调用函数以及传值到Unity

总结

  • IOS切换到Unity的界面切换主要通过拦截Unity界面的运行,在我们想要使用Unity界面的时候才让Unity界面的方法继续执行。IOS界面切换到Unity界面,如果Unity界面不是第一次执行,记得执行UnityPause(false)方法,因为我们在切换到IOS的时候,会暂停Unity界面
  • Unity切换到IOS界面将viewController更改成自已定义的界面控制器就可以了。切换到IOS界面的时候记得执行UnityPause(true)方法
  • IOS传递数据到Unity界面主要通过UnitySendMessage()方法,*参数一为unity脚本挂载的gameobject,参数二为unity脚本中要调用的方法名,第三个参数为数据,数据格式只能为char **,利用这个方法也可以调用Unity的函数。
  • Unity传递数据到IOS也是通过声明外部函数,通过在Unity脚本调用外部函数的时候以参数的方式传递数据。
  • 声明外部函数的格式为:
  // 声明外部函数
    [DllImport("__Internal")]
    private static extern void functionName (ParameterType Parameter);
  • IOS实现Unity中声明的函数格式为:
extern "C"
{
    void functionName(parameter){
    // do something
    }
}

结语

  • 最近咨询IOS和Unity通讯的人也有点多,写出这个博客给大家作为参考。
  • 跨平台通讯都是自己在琢磨,错误难免,敬请指正。
  • 有疑问可以给我发邮件,但是在发送邮件前,请保证您深思过此问题。若问题没有意义或者在博客中有的话,我不会进行回复,请谅解。邮箱:lyk82465@gmail.com
  • 最后附上Demo,点击下载。Unity和IOS交互Demo
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 159,015评论 4 362
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,262评论 1 292
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 108,727评论 0 243
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 43,986评论 0 205
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,363评论 3 287
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,610评论 1 219
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,871评论 2 312
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,582评论 0 198
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,297评论 1 242
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,551评论 2 246
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,053评论 1 260
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,385评论 2 253
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,035评论 3 236
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,079评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,841评论 0 195
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,648评论 2 274
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,550评论 2 270

推荐阅读更多精彩内容

  • 一.小记 最近公司有个需求,需要用unity3D建立3D模型,并且实现切换功能,由于做开发时间不算长,又没有接触过...
    矫炎圻阅读 12,721评论 20 19
  • WebSocket-Swift Starscream的使用 WebSocket 是 HTML5 一种新的协议。它实...
    香橙柚子阅读 22,958评论 8 183
  • 最近一直在山西右玉跟组拍戏。我们剧组住的是大洋宾馆。我们平时习惯把大洋宾馆简称为大洋。附近没有什么大的超市,平时收...
    恋影孤花阅读 281评论 0 1
  • 又深了的情, 又伤的心。 他不止一次的说他爱上了一个人, 我却又不止一次的见他, 醉的和屎一样。 他是一个值得深交...
    非主流韭零后阅读 86评论 3 1
  • 我问过万物的名字 却不知怎样向他们提起,你 我沉默里的欢笑,欢笑里的郁郁 多少次回头凝视,人群都已散去 神分了我的...
    白前山茶阅读 287评论 0 0