一个小清新 Swift 游戏的开发全过程(Part 2)

转自我自己的 blog

Last Circle

这是这个系列 blog 的第二篇,主要介绍 Last Circle 中出现的各种动画效果,满满的都是图文并茂的干货,还请慢慢享用。

#重复放大 & 缩小(Repeat & Scale)

游戏开始页面的 Start 按钮和游戏结束页面的 Retry 按钮都有这样的动画效果:重复的放大后缩小再放大再缩小。如图所示:

{% asset_img start.gif 开始页面 %}
{% asset_img game_over.gif 结束页面 %}

游戏开始页面
游戏结束页面

这里其实并不是按钮在进行缩放,因为如果是按钮在缩放的话,按钮上的文字也会一起缩放。所以我在按钮下面的添加了一个专门用来进行缩放动画的 scale view,初始状态下它和按钮的大小位置颜色完全一致。开始页面的动画代码是这样的:

private func startButtonAnimation() {
    UIView.animateWithDuration(1, delay: 0, options: [.CurveEaseInOut, .Repeat, .Autoreverse],
        animations: { () -> Void in
            self.scaleView.transform = CGAffineTransformMakeScale(1.5, 1.5)
        }, completion:nil)
}

这个动画的关键是 options 中的三个选项:.CurveEaseInOut 是为了缩放看起来更自然,.Repeat 是使动画一直重复,.Autoreverse 是让动画自动颠倒(也就是放大后的缩小)。缩放是通过改变 view 的 transform 这个属性来实现的。

还有一处重复缩放的动画效果,那就是点击了错误的圆后,正确的圆会有一个快速的闪动,如图:

闪动效果

这里其实也是一个 scale 动画,不过是设定了重复次数,我是用 layer 动画实现的。因为这个动画后还可能执行其他动作,还设置了一个 completion 的 block,所以又用到了 CATransaction

func blink(completion: ()-> Void) {
    let scaleUpAnim = CABasicAnimation(keyPath: "transform.scale")
    scaleUpAnim.toValue = NSNumber(float: 1.5)
    scaleUpAnim.repeatCount = 3
    scaleUpAnim.duration = 0.2
    scaleUpAnim.autoreverses = true

    CATransaction.begin()
    CATransaction.setCompletionBlock(completion)
    self.layer.addAnimation(scaleUpAnim, forKey: nil);
    CATransaction.commit()
}

#放大 & 淡入淡出 (Scale & Fade in/out)

开始页面还有许多不断出现的半透明的圆,在放大后就消失的效果,这个就是放大+淡入淡出的动画。仔细观察的话,这些圆的出现位置和大小都是随机的,也不是同时出现的,而且每个圆的显示时长也是不一样的。具体实现的代码如下:

private func startBackgroundCircleAnimation() {
    let circle = Circle.randomCircle()
    let color = ColorUtils.randomColor()
    circle.color = color
    let cv = CircleView(circle: circle)
    cv.userInteractionEnabled = false
    self.view.insertSubview(cv, belowSubview: self.scaleView)
    circleViews.append(cv)


    let delay = Double(arc4random()) / Double(UINT32_MAX) * 1
    let duration = Double(arc4random()) / Double(UINT32_MAX) * 4 + 0.5

    cv.alpha = 0
    cv.transform = CGAffineTransformMakeScale(0.5, 0.5)

    weak var weakSelf = self
    UIView.animateWithDuration(duration,
        delay: delay,
        options : [.CurveLinear],
        animations: { () -> Void in
            cv.alpha = 0.4
            cv.transform = CGAffineTransformMakeScale(1, 1)
        }) { (finished) -> Void in
            if !finished {
                return
            } else {
                UIView.animateWithDuration(duration,
                    delay: 0,
                    options: [.CurveLinear],
                    animations: { () -> Void in
                        cv.alpha = 0
                        cv.transform = CGAffineTransformMakeScale(2, 2)
                    }, completion: { (finished) -> Void in
                        weakSelf!.startBackgroundCircleAnimation()
                })
            }
    }
}

首先,生成一个随机位置和大小的 circle view,并加入到开始页面的 view 中,并且插入在开始按钮的 scale view 下面,否则会盖住 scale view。然后随机生成圆的延迟时间和持续时间这两个值,用在动画中。整个动画周期分两个部分:1.圆的大小由0.5倍放大到1倍,透明度由0到0.4;2.圆的大小由1倍放大到2倍,透明度过渡到0。
由于这个动画也是要不断重复的,所以要在 completion 的 block 中调用该方法以此来实现无限动画。这个方法只是一个圆的动画,要实现 gif 中那么多圆的动画我一共调用了7次这个方法。

但是,因为这部分的动画,我发现了一个很严重的问题,那就是这个游戏玩过一会儿后手机发热好严重。一开始我以为是在游戏中计算可用的圆的那个 while 循环造成的,后来一想这点计算量应该不至于啊。后来还是靠 Instrument 的 Time Profiler 才发现问题所在(第一次使用,果然是神器),就是这个 startBackgroundCircleAnimation 造成的,为什么呢?这个方法居然一直在执行!因为 completion 中没有写如何结束动画,我上面说了我一共调用了7次这个方法就为了实现7个圆出现在画面里,所以一共有7个这段代码一直在无限循环的执行,导致了 CPU 100%……修改后的代码如下:

...
completion: { (finished) -> Void in
    cv.removeFromSuperview()
    if !finished {
        return
    } else {
        weakSelf!.startBackgroundCircleAnimation()
    }
}

#弹性放大

游戏的主页面就是许多的圆按照不同的顺序依次出现,同时伴随着带有弹性的放大效果(放大到最大后回弹),动画效果如图所示(gif 分辨率太低了,可能看不清):

游戏画面

要想实现这个有弹性的动画,就要用到 UIView 的 + animateWithDuration:delay:usingSpringWithDamping:initialSpringVelocity:options:animations:completion: 这个 API 了,其中 usingSpringWithDamping 就是弹性的阻尼,initialSpringVelocity 就是弹性的初速度。动画开始之前,先把每个圆缩放到0.1倍,然后在动画中恢复到正常大小。为了让圆的 circle view 在动画中也可以被点击,options 里就设置了 .AllowUserInteraction。具体代码如下:

for cv in circleViews {
    let delay = Double(arc4random()) / Double(UINT32_MAX) * 0.3
    cv.transform = CGAffineTransformMakeScale(0.1, 0.1)
    UIView.animateWithDuration(0.5,
        delay: delay,
        usingSpringWithDamping: 0.5,
        initialSpringVelocity: 6.0,
        options: UIViewAnimationOptions.AllowUserInteraction,
        animations: {
            cv.alpha = 1
            cv.transform = CGAffineTransformIdentity
        }, completion: nil)
}

#颜色渐变

游戏的主页面顶端有一个示意倒计时的进度条,通过长度和颜色来提示用户剩余时间。如图所示:

倒计时进度条

这个进度条的实现是自定义一个 CountDownView,将其放置在游戏画面的顶端,并根据已过时间和总时间来设置进度条的长度和颜色。颜色的过渡并不是从绿直接到红,中间需要黄色过渡一下,所以前一半是由绿到黄,后一半是由黄到红。更新进度的代码如下:

func updateProgress(time:CGFloat, total:CGFloat) {
    let progressViewWidth = frame.size.width * time / total
    progressView.frame = CGRectMake(0, 0, progressViewWidth, frame.size.height)

    let r,g,b :CGFloat
    let a: CGFloat = 1.0
    if time < total/2 {
        r = time/total*2
        g = 1
    } else {
        r = 1
        g = 2 - time/total*2
    }
    b = 0
    let currentColor = UIColor(red: r, green: g, blue: b, alpha: a)
    progressView.backgroundColor = currentColor
}

因为画面中的圆是渐次出现的,所以进度条不是从一开始就进行倒计时的,而是有一个0.3秒的延迟,这里就用到了 GCD 的延迟执行。然后为了达到平滑的更新效果,所以要每六十分之一秒就更新一下进度条,这里就用到了 NSOperationQueue 以及 NSBlockOperation。这段代码觉得写得有些复杂,我相信还有更好的实现,因为我对多线程还不太熟悉,还请多指教:

dispatch_after(dispatch_time(DISPATCH_TIME_NOW, Int64(0.3 * Double(NSEC_PER_SEC))), dispatch_get_main_queue()) { () -> Void in
    self.startTime = NSDate()
    weak var weakOperaion = self.updateOperation
    self.updateOperation.addExecutionBlock { () -> Void in
        while weakOperaion?.cancelled == false {
            NSThread.sleepForTimeInterval(1/60)
            let interval = NSDate().timeIntervalSinceDate(self.startTime!)
            NSOperationQueue.mainQueue().addOperationWithBlock({ () -> Void in
                self.countDownView.updateProgress(CGFloat(interval), total: self.totalTime)
            })
        }
    }
    self.queue.addOperation(self.updateOperation)
}

#后记

除了上面介绍的,其实还有几处动画没有提及,比如正确点击圆后的圆放大直到充满屏幕,比如游戏结束后 GAME OVER 这两个单词的动画,因为我觉得这些相较于以上都比较容易,而且掌握了以上几个动画后这几个更是不在话下了。

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

推荐阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,515评论 25 707
  • 2016.02.03,重拾JAVA开发。
    西鬼阅读 447评论 0 0
  • 成交量 是指一个时间单位内对某项交易成交的数量。一般情况下,成交量大且价格上涨的股票,趋势向好。成交量持续低迷...
    丁老师看盘阅读 269评论 0 0
  • 清晨又是我第一个到办公室。 打开电脑,显示器上满满的都是林丹谢杏芳,英雄余旭明天就魂归故里了,那么让人扼腕,...
    方大叔阅读 269评论 0 0