这个主要是记录一下公司项目里面集成Unity时候遇到的一些问题。供后来的集成参考使用。
集成期间遇到了不少问题模板也换来换去主要是使用Cardboard 或者 GoogleVRForUnity
Section1 准备
1.swift 2.3 (swift 3.0 应该也没有问题)
2.Xcode 8.2
3.Unity 5.6.0f1
Section2 开始集成
英文好的可以看看老外的集成文章可以说大部分人的参考步骤都是这个文章吧。
找了找中文的,步骤是参考 Nothing_lu的集成文章推荐先看看这里进行集成(我就是按照这个来的,证明能集成成功~,因为步骤都相同所以就不搬砖了,注意看一下下面的tip。)
tip:文章中有图片不清晰的地方里面的代码是这样的:
var currentUnityController: UnityAppController!
currentUnityController = UnityAppController()
currentUnityController.application(application, didFinishLaunchingWithOptions: launchOptions)
main.swift
里面的代码我是使用的这个
// overriding @UIApplicationMain
custom_unity_init(Process.argc, Process.unsafeArgv)
UIApplicationMain(Process.argc, Process.unsafeArgv, NSStringFromClass(UIApplication), NSStringFromClass(AppDelegate))
ok,如果集成好了就开始填坑吧~~~。
Section3 开始天坑=填坑???
- diff: /../Podfile.lock: No such file or directory
diff: /Manifest.lock: No such file or directory
error: The sandbox is not in sync with the Podfile.lock. Run 'pod install' or update your CocoaPods installation.
(Nothing同学的是新建一个工程,考虑大部分同学都会遇到这个问题,也能解决)
根据说明 执行 pod install
Pods/Target Support Files/Pods-ZMCalendar/Pods-ZMCalendar.debug.xcconfig
in your build configuration (UnityFix/Unity.xcconfig
).
这里看着应该是配置文件中缺少了什么,build工程以后发现错误依旧。。
这里参考了童冀 的这篇文章里面的内容
到Unity.xcconfig下面添加
//添加代码到Unity.xcconfig 里面
#include "Pods/Target Support Files/Pods-你的项目名/Pods-你的项目名.debug.xcconfig"
#include "Pods/Target Support Files/Pods-你的项目名/Pods-你的项目名.release.xcconfig"
Pod没有报错了。
Command + B. 如果还报错,那么重启一下 Xcode.
-
发现错误
swift:28:33: Use of undeclared type 'UnityAppController'
配置正确的情况下这种错误一般是由于桥接文件没弄好造成的。
所以查看桥接文件
添加上如图所示:
#import "UnityUtils.h"
#import "UnityAppController.h"
#import "UnityInterface.h"
Command + B (swift 编译一次可能需要 10分钟,一点不夸张。)
-
发现错误 (我当前使用的是 swift 2.3 如果没有报错请忽略这个错误)
Use of unresolved identifier ''CommandLine“
后来发现 这个错误是main.swift 里面写的好像是不对的。main.swift 里面写成这样
// main.swift
import Foundation
import UIKit
// overriding @UIApplicationMain
custom_unity_init(Process.argc, Process.unsafeArgv)
UIApplicationMain(Process.argc, Process.unsafeArgv, NSStringFromClass(UIApplication), NSStringFromClass(AppDelegate))
- 发现错误 'Unity/ObjCRuntime.h' file not found
尝试添加路径
这里面的路径比较多 其实就是链接到 当前的一个 库里面 如果出现这个错误 那么添加好对应的路径。
- 发现错误 :/Users/zhangxianqiang/Desktop/ZMCalendarSwift副本/XMGWB/Classes/Tools/Libs/Masonry/MASConstraintMaker.m:55:5: Use of undeclared identifier 'NSAssert'
这个貌似是和 marsony 里面有所冲突
修改方法:
- 报错
解决方法: (这种错误 一般就是 路径问题 所以 图中的 路径请注意 你的可能不是叫 UnityFix!!!)
7错误 undeclared identifier 'UnityParseCommandLine'
解决方法:
8.报错 GVROverlayView.h file not found
解决方法:
a. 当前我这个版本没有找到对应的 sdk (如果你的小伙伴用了新版的 Unity 模板文件 可能没有这个问题)
- 报错
解决方法:
c c++ 里面都添加上这个 -DINIT_SCRIPTING_BACKEND=1
到这里 我的工程跑起来了 因为小伙伴的 Unity里面使用的库文件经常变化 所以后面可能还遇到不同的坑。
Section 3 额外的一些坑记录。
- "csingle" file not found
解决说明: 这个问题好像是Xcdoe的问题 我编译通过了也会报这个错。
2.cannot use '@throw' with Objective-C exceptions disabled (masrony 集成到工程里面可能会有这个错)
解决方法:
- gvr::creatMainApp_Expec...
(这个的解决方法不是很好, 如果有好方法请留言告诉我)
#include
namespace gvr {
class VrApp;
VrApp* CreateMainApp__EXPECTED_EXACTLY_ONE_VR_MAIN_APP_STATEMENT__(const std::vector&) { return nullptr; }
}
- 更新版本遇到这个错误GoolekitCore.o:
这个是因为 库冲突了 如果你的Untiy小伙伴用了最新的google cardboard库 那么就 Podfile 里面去掉 GVRSDK。
-
另外 cocoapods 来回弄 遇到了这个问题 Attempt to read non existent folder!!
解决方法: 别的地方说卸载了cocoapod 重新安装?? 我没有试验过,我只是 修改了文件夹的中文名字变成了英文,然后 把这个工程放到了另外一个文件夹里面再次运行pod install 就行了。
6.(2017.8.10)添加一个错误 这个错误是编译已经通过了,跑在真机的时候直接崩溃的问题。
- 报错 ---> (这里是unity 更换了 5.4.2f0)
build settings 里面:
Other C Flag: -DINIT_SCRIPTING_BACKEND=1
Other C++ Flag: -DINIT_SCRIPTING_BACKEND=1
8 . gfx device intialization failed
xcode 直接断开和手机的链接了.. 这个问题纠缠了我好久了。google上面并没有什么答案,大概靠谱一点的就是说Unity的配置为题,版本有的不兼容Metal。。 具体的配置可以参考我下面给出的Unity配置图片,另外可以试试修改工程里面 Edit Schema -> Options -> GPU Frame Caputure 里面进行转换对应的选项。
但是我遇到的都不是这个问题。
通过intialization failed 发现其实是有东西没初始化,但是又不给任何提示。所以总结起来就是我们如果要保证一开始就要有初始化unity的步骤。注意 自己创建的 main.swift 文件。还有就是Appdelegate里面的@UIApplicationMain 需要注销。具体的步骤参考的步骤Nothing_lu的集成文章集成步骤的第十步。
这里其实可以写一个swift(iOS)工程的启动解析了。
-
报错:Bulk_Assembly-CSharp_ 文件报错。
Tip:这个里面说明的是我们对一些函数并没有定义,我个人对C C++ 并不是很擅长,看了几篇文章,里面大概的意思是这个文件包裹了一些函数供后续的调用,如果出现了这个错误,仔细观察一下是什么前缀,此处图中是GCloudVioce,后来才分析到是Unity里面添加了一个三方库,并没有给到我。
解决:添加.a库
- MTLDebugRenderCommandEncoder.mm:1331: failed assertion `(rect.y(0) + rect.height(720))(720) must be <= 687'
解决:这里是调整了这里,选择不用Metal。。 不知道有没有好的方案。
- NSUnknownKeyException: [
valueForUndefinedKey:]: this class is not key value coding-compliant for the key currentUnityController
解决: 这个问题是一个小伙伴提出来的,是因为他用了 Xcode 9 swift4.0 导致的,可能是语言有升级的关系 appdelegate 里面 修改一下这个 @objc var currentUnityController: UnityAppController? 添加上objc!
Section 5 后续
1.如果你看到有问题或者有好的解决方法,请留言告诉我。
2.如果你遇到问题了,我尽量解答,因为这里面的坑太多了。
3.对你有帮助请留言告诉我。
4.没有解决问题也请让我知道你来过这里。
5.生活不容易, 就像每次编译这个工程 都需要 10分钟以上。所以 这个文章一边上班一边写了好几天。
参考文章
http://www.jianshu.com/p/e8217896d6ff
http://www.cnblogs.com/fuunnyy/p/6227740.html
http://www.jianshu.com/p/1a141e4fccb3
没时间吐槽了,写了这个文章。当做记录。
=============================================
4.24 添加更新:主要是记录一下 unity 小伙伴如果更新了 他们的版本然后我们这边更换集成文件的主要步骤。
1.找到相应的文件夹 替换 classes, Data,Libraries文件夹。工程里面最好是先删除引用,然后再次导入.(classes,Libraries create groups
data 选择 create folder references)
2.修改 classes/main.mm 里面的 main -> main_unity_default
UnityParseCommandLine -> UnityInitRuntime(这个不一定要换。囧~)
3.Libraries/RegisterFeatures.mm 添加
#include
namespace gvr {
class VrApp;
VrApp* CreateMainApp__EXPECTED_EXACTLY_ONE_VR_MAIN_APP_STATEMENT__(const std::vector&) { return nullptr; }
}
- 找到UnityAppController.h
添加:#import、@class UnityViewControllerBase;
//注释此方法
inline UnityAppController* GetAppController()
{
return (UnityAppController*)[UIApplication sharedApplication].delegate;
}
//替换成这个方法
NS_INLINE UnityAppController* GetAppController()
{
NSObject* delegate = [UIApplication sharedApplication].delegate;
UnityAppController* currentUnityController = (UnityAppController *)[delegate valueForKey:@"currentUnityController"];
return currentUnityController;
}
5.各种库文件找不到。 这个可能是因为没有响应的库文件,检查一下:
(如果你是用的pod 管理的 GVRSDK,那么这里应该没有不然会报很多冲突的错误。)
8.记录一下unity的配置情况
4.28 添加更新。(App里面 现在又添加了一个 百度语音的功能,我先暂时写到这里面,有时间会整理成另外一篇文章的。)
首先导入的 3方库 TTTAttributedLabel 这个要是用cocoapods 可以使用 1.6.1版本
appcontroller 里面添加一个接收方法 。目前不知道能不能成功调用,(下面的代码直接使用是接收不到的!!!)
void Voice (char *buy_id)
{
UIAlertView *alert = [[UIAlertView alloc] init];
[alert setTitle:@"1"];
[alert setMessage:[NSString stringWithUTF8String:buy_id]];
[alert addButtonWithTitle:@"确定"];
[alert show];
}
后来问了 Unity 的小伙伴得知应该添加如下的代码:(这段代码说是可以添加到工程的任何位置,iOS 的OC工程应该是可以的,我选择这里是因为我的工程是 swift 的工程,这里面肯定是运行不了这个的具体的原因这里就不说了,感兴趣的可以去研究一下,囧~~)
//代码定义区域 A
#if defined(__cplusplus)
extern "C"{
#endif
// 个人资料联系他
extern void _Voice( char* toPayStr);
#if defined(__cplusplus)
}
#endif
//代码定义区域 B
#if defined(__cplusplus)
extern "C"{
#endif
extern void _Voice( char* toPayStr){
NSLog(@"调用了_Voice--%s",toPayStr);
NSString *str = [NSString stringWithFormat:@"%s",toPayStr];
[[NSNotificationCenter defaultCenter] postNotificationName:@"unitySendMessageNoti" object:nil userInfo:@{@"message":str}];
}
#if defined(__cplusplus)
}
#endif
另外这里我用OC 的方式 发送了一条通知,让Unity的消息传到swift的代码中,如果哪个小伙伴有好的方式进行交互可以留言告诉我~。