用Swift做个游戏Lecture04 —— 仙人掌的狙击

系列:用Swift作个游戏
作者:pmst(1345614869)
微博:PPPPPPMST

本节任务:

  • 随机生成障碍物,且一对障碍物上下相距距离固定,但位置随机。

几种情况:

y position = 0的情况:

y position = playableStart的情况:

y position = playableStart - 障碍物.size.height/2的情况:

推导一般情况下的公式:y position = playableStart - 障碍物.size.height/2 + (10%~60%)playgroundHeight:

上下两个障碍物之间距离固定为3.5倍的Player尺寸的高度:

注意推导公式:y position = playableStart - 障碍物.size.height/2此时障碍物的顶部刚好与地面齐平,而(10%~60%)playgroundHeight是一个浮动范围,表明障碍物超出地面的高度。显然我们的障碍物的层级关系是在背景上面但是在Foreground的下面,因此修改早前的Layer:

enum Layer: CGFloat {
    case Background
    case Obstacle   //添加障碍物层级关系
    case Foreground
    case Player
}

01.产生障碍物的构造方法

我们需要增添一个方法用于实例化一个纹理(图片)为仙人掌的精灵(SpriteNode),设置其zPositionObstacle,请在flapPlayer()方法上方新增如下方法:

func createObstacle()->SKSpriteNode{
    let sprite = SKSpriteNode(imageNamed: "Cactus")
    sprite.zPosition = Layer.Obstacle.rawValue
    return sprite
}

注意到实例方法生成一个纹理为Cactus的精灵并返回,这是之后源源不断生成障碍物的基础。

紧接着我们要有一个实例方法,作用是随机产生成对的障碍物到场景中,步骤如下:

  1. 使用createObstacle()得到下方障碍物的实例,并将其放置紧贴右侧屏幕边线。
  2. 障碍物y轴上的放置位置范围为10%~60%,分别计算最小与最大的y轴点位,通过随机函数得到两者之间的一个数作为y值,设置障碍物的position,最后添加到worldNode节点中。
  3. 同理实例化上方障碍物,将其旋转180°后放置距离下方障碍物3.5倍Player尺寸的地方,添加到worldNode节点中。
  4. 给上下障碍物增添一个移动Action,已一定速度自右向左移动,倘若超出屏幕,则从父节点中移除。
//新增三个常量
let kBottomObstacleMinFraction: CGFloat = 0.1
let kBottomObstacleMaxFraction: CGFloat = 0.6
let kGapMultiplier: CGFloat = 3.5

// 在createObstacle()实例方法下方增添新方法
func spawnObstacle(){
   //1
   let bottomObstacle = createObstacle()    //实例化一个精灵
   let startX = size.width + bottomObstacle.size.width/2//x轴位置为屏幕最右侧
   //2
   let bottomObstacleMin = (playableStart - bottomObstacle.size.height/2) + playableHeight * kBottomObstacleMinFraction //计算障碍物超出地表的最小距离
   let bottomObstacleMax = (playableStart - bottomObstacle.size.height/2) + playableHeight * kBottomObstacleMaxFraction //计算障碍物超出地表的最大距离
   bottomObstacle.position = CGPointMake(startX, CGFloat.random(min: bottomObstacleMin, max: bottomObstacleMax))    // 随机生成10%~60%的一个距离赋值给position
   worldNode.addChild(bottomObstacle)   //添加到世界节点中
   //3
   let topObstacle = createObstacle()   //实例化一个精灵
   topObstacle.zRotation = CGFloat(180).degreesToRadians()//翻转180°
   topObstacle.position = CGPoint(x: startX, y: bottomObstacle.position.y + bottomObstacle.size.height/2 + topObstacle.size.height/2 + player.size.height * kGapMultiplier)//设置y位置 相距3.5倍的Player尺寸距离
   worldNode.addChild(topObstacle)//添加至世界节点中
   //4 给障碍物添加动作
   let moveX = size.width + topObstacle.size.width
   let moveDuration = moveX / kGroundSpeed
   let sequence = SKAction.sequence([
       SKAction.moveByX(-moveX, y: 0, duration: NSTimeInterval(moveDuration)),
       SKAction.removeFromParent()
       ])
   topObstacle.runAction(sequence)
   bottomObstacle.runAction(sequence)
}

倘若你迫不及待想看看成果,将spawnObstacle方法添加至didMoveToView()最下方,点击运行。一对障碍物“呼啸而过”,然后就没有然后了...确实目前这个方法仅仅只是产生一对罢了,为此我们还需要新增一个方法用于源源不断的产生障碍物。请添加如下内容到spawnObstacle()方法下方

func startSpawning(){
    //1
    let firstDelay = SKAction.waitForDuration(1.75)
    //2
    let spawn = SKAction.runBlock(spawnObstacle)
    //3
    let everyDelay = SKAction.waitForDuration(1.5)
    //4
    let spawnSequence = SKAction.sequence([
            spawn,everyDelay
        ])
    //5
    let foreverSpawn = SKAction.repeatActionForever(spawnSequence)
    //6
    let overallSequence = SKAction.sequence([firstDelay,foreverSpawn])
    runAction(overallSequence)
}
  1. 第一个障碍物生成延迟1.75秒
  2. 生成障碍物的动作,用到了先前的实例方法spawnObstacle.
  3. 之后生成障碍物的间隔时间为1.5秒
  4. 之后障碍物的生成顺序是:产生障碍物,延迟1.5秒;产生障碍物,延迟1.5秒;产生障碍物,延迟1.5秒...可以看出[产生障碍物,延迟1.5秒]为一组重复动作。
  5. 使用SKAction.repeatActionForever重复4中的动作。
  6. 将延迟1.75秒和重复动作整合成一个SKAction的数组,然后让场景来执行该动作组。

请将didMoveToView()方法中的spawnObstacle替换成startSpawning(),点击运行。

倘若觉得不错,请点击喜欢并关注我吧..

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

推荐阅读更多精彩内容

  • 一、实验目的 学习使用 weka 中的常用分类器,完成数据分类任务。 二、实验内容 了解 weka 中 explo...
    yigoh阅读 8,506评论 5 4
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,639评论 18 139
  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,884评论 25 707
  • 昨天是小笼包三周岁的生日,一大早起床就问我他的蛋糕、许愿蜡烛在哪?他告诉我他三岁了,是大人了。 ...
    88d4cc69d1be阅读 209评论 0 0
  • 寒假来了。 你从远方回到家乡,家乡可能没有你梦想所在的城市那么繁华,只是这里有他。 假象总是使人快乐,你擅于营造种...
    鹮妖阅读 225评论 0 1