做一个类似Cut The Rote 的游戏(三)

上一篇文章,我们添加的菠萝在重力的作用下掉进了水里。今天我们要让它挂起来。

添加葡萄藤

SpriteKit 的 physics body是用来模拟刚性物体。但葡萄藤是弯曲。所以你需要将每根藤蔓实现为一系列具有柔韧性的段,像链子一样。

藤蔓有三个重要的属性:

  • anchorPointCGPoint 指藤蔓的末端连接到树的位置
  • lengthInt 代表藤蔓的数量
  • nameString 用于识别藤蔓(游戏中有多根藤蔓,用名字唯一识别)

在本教程中,游戏只有一关。但是在实际游戏中,您将希望能够轻松创建新的关卡,而无需编写大量代码。最好的方法是游戏关卡数据与逻辑相互独立,这可以通过用属性列表或JSON的数据文件存储游戏数据中来实现。

在文件中,代表藤蔓数据的是一个包含NSDictionary对象的NSArray数组,数组中的每个字典代表一根藤蔓。可以使用NSArray(contentsOfFile:)方法轻松读取该属性列表。

GameScene.swift中,找到setUpVines()并添加以下代码:

// 1 load vine data
let dataFile = Bundle.main.path(forResource: GameConfiguration.VineDataFile, ofType: nil)
let vines = NSArray(contentsOfFile: dataFile!) as! [NSDictionary]
    
// 2 add vines
for i in 0..<vines.count {
  // 3 create vine
  let vineData = vines[i]
  let length = Int(vineData["length"] as! NSNumber)
  let relAnchorPoint = CGPointFromString(vineData["relAnchorPoint"] as! String)
  let anchorPoint = CGPoint(x: relAnchorPoint.x * size.width,
                            y: relAnchorPoint.y * size.height)
  let vine = VineNode(length: length, anchorPoint: anchorPoint, name: "\(i)")
  // 4 add to scene
  vine.addToScene(self)
  // 5 connect the other end of the vine to the prize
  vine.attachToPrize(prize)
}

使用以上代码,您:

  1. 从属性列表文件加载藤蔓数据。查看Resources/Data中的VineData.plist文件,你应该看到该文件包含一个字典数组,每个字典都有relAnchorPointlength
VineData.png
  1. 使用for循环获取数组索引index。迭代索引而不仅仅是因为对象是数组,最重要的是您需要获取到index,才能根据index为每个藤生成唯一的名称。这对后面的检测是哪条藤蔓非常重要。
  2. 使用每个字典的lengthrelAnchorPoint初始化一个新的VineNode对象。其中length指定藤蔓中的段数。relAnchorPoint用于确定藤蔓的起始位置,这些都与场景的大小相关。
  3. 然后,你调用addToScene()VineNode添加到场景里面。
    5.最后调用attachToPrize()将藤蔓与奖品(菠萝)连接到一起。

接下来你会在VineNode里面实现这些方法。

定义藤类

打开VineNode.swiftVineNodeSKNode的子类。它没有属于自己视觉上的外观,而是作为SKSpriteNode代表藤段的集合。

将以下属性添加到类定义中:

private  let length:Int 
private  let anchorPoint:CGPoint 
private  var vineSegments:[ SKNode ] = [] 

你会看到一些错误,因为lengthanchorPoint还没有初始化。您已将它们声明为非可选项,但未分配值。通过实现init(length:anchorPoint:name:)方法来解决此问题,并将下面的代码添加到里面:

self.length = length
self.anchorPoint = anchorPoint
super.init()   
self.name = name

很简单吧。但是,竟然还存在错误。那是什么呢?对了,还有一个你不会在任何地方调用初始化方法:init(coder:)

因为SKNode实现NSCoding协议,也继承了必须初始化方法init(coder:),这意味着你必须在这个方法里面初始化所有非可选的属性,即使你不会使用这个方法。

用以下内容替换init(coder:)

length = aDecoder.decodeInteger(forKey: "length")
anchorPoint = aDecoder.decodeCGPoint(forKey: "anchorPoint")    
super.init(coder: aDecoder)

接下来,您需要实现addToScene()方法。这是一个复杂的方法,所以需要分阶段来写。首先,找到addToScene()并添加以下内容:

// add vine to scene
zPosition = Layer.Vine
scene.addChild(self)

你将藤蔓添加到场景并设置它的zPosition。接着,将该代码块添加到同一方法中:

//创建藤支架
let vineHolder = SKSpriteNode(imageNamed: ImageName.VineHolder)
vineHolder.position = anchorPoint
vineHolder.zPosition = 1
addChild(vineHolder)
vineHolder.physicsBody = SKPhysicsBody(circleOfRadius: vineHolder.size.width / 2)
vineHolder.physicsBody?.isDynamic = false
vineHolder.physicsBody?.categoryBitMask = PhysicsCategory.VineHolder
vineHolder.physicsBody?.collisionBitMask = 0

这创建了葡萄藤支架,就像挂着藤蔓的钉子。与鳄鱼一样,它是不动的,并且不会与其他身体发生碰撞。

藤架是圆形的,所以使用SKPhysicsBody(circleOfRadius:)构造函数。藤架anchorPoint的位置与您在创建VineNode时指定的位置一致。

接下来是创建藤蔓。还是这个方法,在底部添加以下代码,:

// add each of the vine parts
for i in 0..<length {
  let vineSegment = SKSpriteNode(imageNamed: ImageName.VineTexture)
  let offset = vineSegment.size.height * CGFloat(i + 1)
  vineSegment.position = CGPoint(x: anchorPoint.x, y: anchorPoint.y - offset)
  vineSegment.name = name
  vineSegments.append(vineSegment)
  addChild(vineSegment)
  vineSegment.physicsBody = SKPhysicsBody(rectangleOf: vineSegment.size)
  vineSegment.physicsBody?.categoryBitMask = PhysicsCategory.Vine
  vineSegment.physicsBody?.collisionBitMask = PhysicsCategory.VineHolder
}

这个循环创建一个藤段数组,数量与创建VineModel时指定的长度相等。每个藤段都是一个具有physics body的精灵。这些藤段是矩形的,因此您可以调用SKPhysicsBody(rectangleOfSize:)来指定physics body的形状。

与藤架不同,藤段是动态的,所以它们是可以移动并受到重力影响。

编译并运行应用程序以查看您的进度。

呃哦,藤条从屏幕上脱落,如切碎的意大利面!

掉落的藤条

添加葡萄藤的接头

那是因为你还没有将葡萄段连在一起。要解决这个问题,您需要将最后一批代码添加到addToScene()方法的底部:

//设置藤架的
let joint = SKPhysicsJointPin.joint(withBodyA: vineHolder.physicsBody!,
                                    bodyB: vineSegments[0].physicsBody!,
                                    anchor: CGPoint(x: vineHolder.frame.midX, y: vineHolder.frame.midY))
scene.physicsWorld.add(joint)
// 在藤段间建立关节
for i in 1..<length {
  let nodeA = vineSegments[i - 1]
  let nodeB = vineSegments[i]
  let joint = SKPhysicsJointPin.joint(withBodyA: nodeA.physicsBody!, bodyB: nodeB.physicsBody!,
                                      anchor: CGPoint(x: nodeA.frame.midX, y: nodeA.frame.minY))
  scene.physicsWorld.add(joint)
}

该代码在藤段与藤段之间建立物理连接点,将它们连接在一起。你使用的接头类型是一个SKPhysicsJointPin,就好像在两个节点之间放置了一个引脚,允许它们绕着引脚转动,但是不能相互靠近或更远离。

编译并再次运行 你的葡萄藤应该在树上挂起来了。

藤条挂起来了

最后一步是将菠萝放在葡萄藤上。仍然在VineNode.swift中,滚动到方法attachToPrize()中,添加以下代码:

//将最后一段藤条与奖品对齐     
let lastNode = vineSegments.last!
lastNode.position = CGPoint(x: prize.position.x, y: prize.position.y + prize.size.height * 0.1)    
//设置连接关节
let joint = SKPhysicsJointPin.joint(withBodyA: lastNode.physicsBody!, 
                                    bodyB: prize.physicsBody!, anchor: lastNode.position)    
prize.scene?.physicsWorld.add(joint)

该代码可以获取最后一个藤段,将它放在奖品中心点上面一点(这样就能让奖品像是真的挂起来了)。它还创建另一个连接点,将葡萄藤段和奖品连接起来。

编译并运行项目。如果您都正确设置,您应该看到如下所示:

菠萝挂起来了

好极了!菠萝挂起来了 - 谁把菠萝与树木连接起来?:]

下一篇文章,我们要开始切断葡萄藤咯~~

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

推荐阅读更多精彩内容