说明
本系列文章是对<3D Apple Games by Tutorials>一书的学习记录和体会此书对应的代码地址
更多iOS相关知识查看github上WeekWeekUpProject
以前面的游戏为例,将其改为跨平台版本:
-
macOS游戏Geometry Fighter
WX20171202-194556@2x.png
-
tvOS游戏Breaker
WX20171202-194627@2x.png
-
watchOS游戏Geometry Fighter
WX20171202-194641@2x.png
16-macOS游戏Geometry Fighter
创建项目
打开Xcode创建新项目,选择macOS平台,选择Game类型,点击Next继续.
data:image/s3,"s3://crabby-images/58b14/58b1411ba7b1d88dad4c7ae7ebe126722cd530b9" alt=""
输入游戏名SceneKitGame,选择Swift语言,SceneKit游戏技术,取消Unit和UI Tests,点击Finish.
data:image/s3,"s3://crabby-images/89d95/89d95e9ae2bb30462e4078d1127d5ded4d73a187" alt=""
生成的项目类似于iOS项目,但不完全相同:
data:image/s3,"s3://crabby-images/7f410/7f410b2edebdad98911b8c6c302d6207da2888e9" alt=""
- GameView.swift:继承于SCNView,可以响应鼠标键盘事件但不能触摸.
- GameViewController.swift:继承于NSViewController.
- MainMenu.xib:控制器的xib.
选择My Mac,运行一下游戏:
data:image/s3,"s3://crabby-images/8372b/8372b61e2a57949df249dcc25a3525b781e3842a" alt=""
转换SceneKit游戏
可以在projects/ starter/GeometryFighter/中打到原iOS版的项目.
打开iOS版项目,点击GeometryFighter,添加新的target:
data:image/s3,"s3://crabby-images/56bcd/56bcd0a4810de069cb4848941e02b32296663fff" alt=""
选择Add Target...
data:image/s3,"s3://crabby-images/97f5a/97f5a331c0d1d2fe372bf9e6568af5d16918ac24" alt=""
选择macOS平台,然后选择Game
data:image/s3,"s3://crabby-images/ac46c/ac46c2a81440fb8d9562cef7675c0ff929138ae3" alt=""
输入项目名GeometryFighterMac,选择SceneKit,取消Unit Tests和UI Tests,点击Finish:
data:image/s3,"s3://crabby-images/c71a7/c71a709369ec07307626682f7f1df53c9ad22049" alt=""
选择GeometryFighterMac > My Mac作为Active Scheme.
data:image/s3,"s3://crabby-images/fe941/fe941661706c0bbeca4b28ac971f4d2bdac08d75" alt=""
运行后,看到的还是默认的飞机场景,那是因为还需要其他步骤.
多个target内容共享
可以在iOS和macOS之间共享原来的代码和资源.创建一个Shared分组
data:image/s3,"s3://crabby-images/46195/46195537a82d346809fc841bb880bfbb1f447c27" alt=""
将下列文件和文件夹移动到Shared下面:
data:image/s3,"s3://crabby-images/56739/5673944ef6fec121070b1bc059233f7fc54c5ec4" alt=""
data:image/s3,"s3://crabby-images/2b533/2b53336b28ceb1b83afeb50c90cbb05c7a23beb0" alt=""
按住Shift键,选中GeometryFighter/ Shared/Particles下面的全部文件,打开右侧的属性检查器,勾选Target Membership下面的GeometryFighterMac;这样就能在iOS target和macOS target之间共享了.
data:image/s3,"s3://crabby-images/4778d/4778dad72c8d58ce09d0d3c5f728808d357ba932" alt=""
Shared下面的其他几个也是类似操作.
data:image/s3,"s3://crabby-images/7c6fd/7c6fdbdca491e6eaa8efea4a9e1d5129bd06eae2" alt=""
data:image/s3,"s3://crabby-images/19fd8/19fd8539dd4a79a75056dc8c6fabf447b679b183" alt=""
data:image/s3,"s3://crabby-images/bcba9/bcba9312a1005ab178fb208dc21146dce722df61" alt=""
为了解决跨平台带来的问题,还需要添加下列代码:
#if os(iOS)
import UIKit
#endif
#if os(macOS)
import Cocoa
#endif
当然了,我们不需要每个文件都去添加,只需要将已创建好的resources/ GameUtils/文件导入进来就可以了.首先,删除一些旧文件,选中GameHelper.swift和UIColor+Extensions.swift.
右键--删除--Move to Trash.
将resources/ GameUtils/下面的所有文件拖放到Shared/ GameUtils/文件夹下
data:image/s3,"s3://crabby-images/f33fd/f33fdaea667f449433fee21703119fe13e1ec7e1" alt=""
data:image/s3,"s3://crabby-images/0d34f/0d34f0187a6834f33a52164452ba675fe0c74348" alt=""
清理
还需要清理一下项目.选中GameView.swift, GameViewController.swift和art.scnassets.右键删除--Move to Trash.
然后从示例代码中本章节的resources/GameViewController文件夹下拖放GameView.swift和GameViewController.swift到项目中,选中Copy items if needed和GeometryFighterMac,点击Finish完成.
data:image/s3,"s3://crabby-images/8b196/8b196884b996a5ec91652f920c29a4ed09914f8d" alt=""
还要做的是恢复新的GameViewController与MainMenu.xib之间的联系.
data:image/s3,"s3://crabby-images/c3002/c3002fddde962841a97f74bee13c1c13ed1cc9fc" alt=""
鼠标输入
选中MainMenu.xib,从右侧对象库中拖放一个Click Gesture Recognizer到Game View中.
data:image/s3,"s3://crabby-images/26166/261663bb1917a526e22a8a3e906211a62832c1e9" alt=""
添加连接函数:
data:image/s3,"s3://crabby-images/0fe27/0fe27f2f8a2a0490b93f02c137de38ace5012dde" alt=""
现在还差最后一步,添加AppIcon,你可以从本章节的resources/AppIcon/文件夹中找到,拖放到Assets.xcassets中的AppIcon下:
data:image/s3,"s3://crabby-images/79e49/79e49b93dc0fb033ca12d1ad95542a820d5ae418" alt=""
运行一下程序:
data:image/s3,"s3://crabby-images/88861/8886141f47f42e523585c0a50c355c92f2388aa6" alt=""
本项目的最终完成版代码可以在对应章节的projects/ final/GeometryFighter/下找到.
17-tvOS游戏Breaker
创建项目
创建项目
data:image/s3,"s3://crabby-images/20c0e/20c0e8b9f1321f3e687f00e10702e5517957fb21" alt=""
data:image/s3,"s3://crabby-images/a036e/a036ed887168dc51f2ccbd6fc0126e08f1373fef" alt=""
在Active Scheme中选择SceneKitGame > tvOS Simulator > Apple TV 1080p:
data:image/s3,"s3://crabby-images/41ba2/41ba2e95837c10aeda217882ca9e1d111cedd8f0" alt=""
运行一下,可以看到默认的飞机模型.但是真实的Apple TV是要用遥控器操作的,怎么用呢?在模拟器的Hardware > Show Apple TV Remote中,就可以调出遥控器了:
data:image/s3,"s3://crabby-images/82083/820834d3ecd2d9b67231254189a358e427388b66" alt=""
移植到tvOS
在代码中找到本章节的projects/ starter/Breaker/文件夹接着处理.
和前面类似,选中Breaker,添加新的target,在弹出窗中选择tvOS和Game.
data:image/s3,"s3://crabby-images/76d31/76d31f2e30661329110aee28d53fe42f56741a12" alt=""
data:image/s3,"s3://crabby-images/4bcf8/4bcf8f76b890356d36d627c80fe640ef49c6119b" alt=""
data:image/s3,"s3://crabby-images/936c0/936c045343e1c9b2a9e064058d3e9b92a85d2afb" alt=""
data:image/s3,"s3://crabby-images/bae9c/bae9ca75857d1896841f17712d00db16401a1086" alt=""
targets间内容共享
添加一个Shared分组,并将原来的文件拖放进来:
data:image/s3,"s3://crabby-images/a64af/a64af970cf1cc237b7a12f9ed19b07e4cd0f7bd9" alt=""
data:image/s3,"s3://crabby-images/7da9a/7da9a3a7f0122b2c246f311f9ba6d3bb394f2700" alt=""
逐一选中文件夹下的所有文件,添加Target Membership:
data:image/s3,"s3://crabby-images/f8dbb/f8dbbc8054507e6f5785b4176470c5c43042b4ea" alt=""
data:image/s3,"s3://crabby-images/a3fe2/a3fe2d92022710fd7746b7497b581c392b443afe" alt=""
data:image/s3,"s3://crabby-images/ffa22/ffa22bc0f0931f478e7b4ed9504404b6d2447a34" alt=""
data:image/s3,"s3://crabby-images/84887/84887240a739f091f942cab6cc6c6ec0fb25a90d" alt=""
还需要清理一下代码.
选中BreakerTV/art.scnassets和BreakerTV/GameViewController.swift,删除--Move To Trash:
添加专用代码
打开GameViewController.swift,在setupNodes()
末尾添加代码:
#if os(tvOS)
scnView.pointOfView = horizontalCameraNode
#endif
还有shouldAutorotate
, prefersStatusBarHidden
和viewWillTransition()
也不需要了:
#if os(iOS)
override var shouldAutorotate: Bool {
... }
override var prefersStatusBarHidden: Bool {
... }
override func viewWillTransition(to size: CGSize, with coordinator:
UIViewControllerTransitionCoordinator) {
... }
#endif
遥控触摸事件
和iOS的触摸事件不同,遥控上更接近MacBook的触摸板的逻辑,touchesBegin()
的初始位置总是(x:960, y: 540)
,即1080p显示器的中心,touchesMoved()
时的位置则是相对于初始点的位置.
另外还有一个问题:tvOS遥控器的触摸板太灵敏了,轻微移动就是很长距离.我们需要在GameViewController
中找到下面的代码:
paddleNode.position.x = paddleX +
(Float(location.x - touchX) * 0.1)
将其更改为:
#if os(iOS)
paddleNode.position.x = paddleX +
(Float(location.x - touchX) * 0.1)
#elseif os(tvOS)
paddleNode.position.x = paddleX +
(Float(location.x - touchX) * 0.01)
#endif
图标
图标资源在本章节对应的resources文件夹下.
打开BreakerTV/Assets.xcassets,选中App Icon & Top Shelf Image,将图片拖放进去:
data:image/s3,"s3://crabby-images/3de82/3de8205a9ddc920045276e35d63152393d41ac4d" alt=""
运行一下,看到图标出现在Apple TV首页上了:
data:image/s3,"s3://crabby-images/c6e86/c6e860e1cd4b725310c583441397b54cf634594e" alt=""
点击进入游戏,开始玩吧:
data:image/s3,"s3://crabby-images/606eb/606eb8c3151e32a466984cdf28ce08d2adf59cde" alt=""
18-watchOS游戏Geometry Fighter
在Xcode中,并没有专门的watchOS版游戏的模板,我们需要做的是创建一个iOS的游戏,再给它添加watchOS的支持.
添加watchOS支持
我们直接在原iOS项目基础上添加watchOS的target:
data:image/s3,"s3://crabby-images/986b0/986b055d781851f5c5fbdae3d7b072a403204f87" alt=""
data:image/s3,"s3://crabby-images/49f76/49f76f7303efe9bd4d3f6059e1608881bb849327" alt=""
data:image/s3,"s3://crabby-images/abbac/abbacb5a56d1d617ad0138a7ace01eb1d4ff138b" alt=""
在Active Scheme中选择GeometryFighterWatch > iPhone 6s Plus + Apple Watch - 42mm
data:image/s3,"s3://crabby-images/0268b/0268b2197ae06e9c0ea4b2d8ff2383d5f0407c5e" alt=""
运行一下,看看效果:
data:image/s3,"s3://crabby-images/a0a30/a0a300d6328f395182d9531aca1a40652d4f794c" alt=""
targets间内容共享
创建Shared文件夹,移动需要的文件
data:image/s3,"s3://crabby-images/fdb08/fdb0812ffacd213ee81354654c73de3be12f5091" alt=""
然后依次选中各个文件夹下面的所有文件,添加Target Membership:
data:image/s3,"s3://crabby-images/71ec0/71ec08c7921ff71ee39178ca8ef4c6dedcba6f71" alt=""
data:image/s3,"s3://crabby-images/06bb4/06bb483294019a0eb1c1dd6b75d45b07e6d20eec" alt=""
data:image/s3,"s3://crabby-images/b8783/b8783282ca3ddfb836a1e340ee03f87ecfe30ff4" alt=""
data:image/s3,"s3://crabby-images/1fa50/1fa50db215bfcd102432d5370928e1e8c367a142" alt=""
添加界面控制器
首先清理项目,选中InterfaceController.swift和art.scnassets.右键--删除--Move to Trash.
data:image/s3,"s3://crabby-images/3b7e9/3b7e99b9ce54331104175cbaa8d4235d89e87786" alt=""
现在需要添加新的InterfaceController.swift.在本章节对应代码的resources/source文件夹下,拖放到Xcode中.
data:image/s3,"s3://crabby-images/9ba04/9ba045f202ad49d0dfbf046d9fbb7abcdef0e355" alt=""
然后建立连接:
data:image/s3,"s3://crabby-images/66af1/66af172c28bb7d827b9a5fd37187176315de0af9" alt=""
添加触摸输入
选中Interface.storyboard,拖放一个Tap Gesture Recognizer过来.
data:image/s3,"s3://crabby-images/77622/7762291df9c2f5f57aad0bbe6db6b4b27957ec10" alt=""
建立手势的连接:
data:image/s3,"s3://crabby-images/d65b9/d65b981b27b30a30040d02d9aa6c2a4c7d65ea87" alt=""
现在已经基本完成了.
图标
所需图片资源在本章节对应的resource/AppIcon文件夹下.
选中Assets.xcassets下面的AppIcon,将图片拖放到其中:
data:image/s3,"s3://crabby-images/acfd4/acfd4a3854982589115bb038638bc189b49072f4" alt=""
运行一下,可以愉快地玩耍了!
data:image/s3,"s3://crabby-images/6500b/6500be74d707777f25be89ba449a9c0fe4bb1576" alt=""
项目的最终完成版本章节对应的projects/ final/GeometryFighter/文件夹下.
网友评论