9
9

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

More than 5 years have passed since last update.

アニメーションが思った通りに動いてくれない!?そんな時の対策法1

Last updated at Posted at 2015-12-20

アプリ道場 Advent Calendar 2015の21日目です。よろしくお願いします。

##アプリ作成に慣れてきた!その次にやりたいことと言えばコレ!
アプリを作り始めて、「ちょっと出来るようになってきたかも!」と思った時、次にやりたくなること。それはアニメーションではないでしょうか?(私だけ!?)
そんな時、一番お手軽な方法としてanimateWithDurationというメソッドを使用する方法があります。
私は以下みたいなテンプレート関数を作ってゆるーく使ってます。
ヘッダの説明にもありますが、第1引数に指定したViewを、第2引数に第1引数に指定したViewの移動先座標を指定し、アニメーションを実行する関数です。

    /**
     指定されたViewを指定された中心座標へアニメーションする
     
     - parameter targetView:アニメーションさせるView
     - parameter targetPoint:異動先の座標
     */
    func animationLocateWithView(targetView :UIView, targetPoint :CGPoint) {
        
        print("\(__FUNCTION__) is called!!")
        
        let animationSpeed = 1.0
        
        UIView.animateWithDuration(animationSpeed, animations: { () -> Void in

            targetView.center = targetPoint
            }, completion: nil)
    }

##今回の課題
以下のようなものをStoryboardで作りました。

  • "Move〜"ボタン: pressButtonメソッド, 押下時にアニメーションを実行
  • オレンジ色正方形: orangeColorView, アニメーションさせるView
  • グレー色正方形左: positionViewLeft, 開始地点の目安のView(今回は使用しない)
  • グレー色正方形右: positionViewRighr, アニメーションの異動先の目標とするView
  • "Before"テキスト: textLabel, 動かした時"Before"を"After"に変更する

スクリーンショット 2015-12-20 11.24.17.png

##まずはアニメーションさせてみよう
まずはorangeColorViewをアニメーションさせてみましょう。
ボタンを押した時、animationLocateWithViewを実行する作りになっています。

import UIKit

class ViewController: UIViewController {

    ...
    
    /**
     ボタンを押した時の動作。アニメーション実行のトリガ
     */
    @IBAction func pressButton(sender: AnyObject) {
        
        print("\(__FUNCTION__) is called!!")
 
        animationLocateWithView(orangeColorView, targetPoint: positionViewRight.center) //アニメーション実行
        
    }
    
    /**
     指定されたViewを指定された中心座標へアニメーションする
     
     - parameter targetView:アニメーションさせるView
     - parameter targetPoint:異動先の座標
     */
    func animationLocateWithView(targetView :UIView, targetPoint :CGPoint) {
        
        print("\(__FUNCTION__) is called!!")
        
        let animationSpeed = 1.0
        
        UIView.animateWithDuration(animationSpeed, animations: { () -> Void in

            targetView.center = targetPoint
            }, completion: nil)
    }


}

これを実行した様子が以下のgifになります。

animation3.gif

よしよし。ちゃんと動いていますね。

##テキストファイルの更新。こんなのチョロいと油断するな。
あとはボタンを押す前は"Before"と書かれたLabelのTextをボタンを押したタイミングで"After"と書き換えればいいだだけです。

    /**
     ボタンを押した時の動作。アニメーション実行のトリガ
     */
    @IBAction func pressButton(sender: AnyObject) {
        
        print("\(__FUNCTION__) is called!!")
        
        self.textLabel.text = "After"   //テキストをafterへ変更
 
        animationLocateWithView(orangeColorView, targetPoint: positionViewRight.center) //アニメーション実行
        
    }
    

実行してみます。
実行したgifが以下ですが...

animation1.gif

ええ〜!?orangeColorViewが意図した動きしてなくない!?

実はこの動作、Auto Layoutを無効にすると発生しません。
またStoryBoardからでなく、コード上からaddSubViewしたViewをアニメーションさせる場合には発生しません。
この不可解な動作理由ですが、以下のような理由により発生していると考えられます。

  • Labelの更新が行われることによりloadView(レイアウト再実行)メソッドが実行される
  • ラベルの内容更新とアニメーションが同時に実行される

完全には追えていませんが、 アニメーション + AutoLayout + Viewの更新 という条件が重なる場合に現象は発生します。
Auto Layout有効にしているとViewが更新された際に再レイアウトが実行されます。今回の場合はUILabelのテキストを更新したタイミングですね。
この時、loadViewという内部処理が実行されます。
loadViewとラベルの更新が同時に行われることが、意図しない動作に要因の一つと認識しています。

##解決法は?
解決方法としてはoverrideしたviewDidLayoutSubView内にアニメーションするViewのアニメーション後の座標を書くという方法です。
このviewDidLayoutSubViewですが、上述のloadViewが実行された直後に呼び出される処理です。
viewDidLayoutSubViewをoverrideし、その内部にアニメーション後のorangeColorView座標を設定してあげると、あら不思議!
この問題が解決します。

コードは以下。
( viewDidLayoutSubViewは起動時にも呼ばれる点を考慮に入れることを忘れないこと。)

class ViewController: UIViewController {

    ...

    var animationFlag = false
    
    ...
    
    override func viewDidLayoutSubviews() {
        print("\(__FUNCTION__) is called!!")
        
        /**
        アニメーション移動後のViewの座標を設定します。
        アニメション後の座標の管理方法はアプリによって異なると思われます。
        */

        //アニメーションするか否か確認。アニメーションしない場合にはreturnする。
        if animationFlag == false {
            return
        }
        
        orangeColorView.center = positionViewRight.center   //移動後のorangeColorViewの座標を設定
        animationFlag = false                               //アニメーションフラグをfalseに
        
        
    }

    /**
     ボタンを押した時の動作。アニメーション実行のトリガ
     */
    @IBAction func pressButton(sender: AnyObject) {
        
        print("\(__FUNCTION__) is called!!")
        
        animationFlag = true            //アニメーションフラグをtrueにする
        
        self.textLabel.text = "After"   //テキストをafterへ変更 (←これがアニメーションが意図しない動きになる原因!!)
 
        animationLocateWithView(orangeColorView, targetPoint: positionViewRight.center) //アニメーション実行
        
    }
    
    /**
     指定されたViewを指定された中心座標へアニメーションする
     
     - parameter targetView:アニメーションさせるView
     - parameter targetPoint:異動先の座標
     */
    func animationLocateWithView(targetView :UIView, targetPoint :CGPoint) {
        
        print("\(__FUNCTION__) is called!!")
        
        let animationSpeed = 1.0
        
        UIView.animateWithDuration(animationSpeed, animations: { () -> Void in

            targetView.center = targetPoint
            }, completion: nil)
    }


}

実行時のgif画像は以下です。
想定通りの動作になっています!

animation2.gif

完成したプロジェクトをGithubに上げておきます。参考にして下さい。
▶︎anthrgrnwrld/animation

現象が発生する原因について正確には追えていませんので正式な対策ではないかもしれませんが、初心者向けのテクニックとしてご使用頂けると嬉しいです。

9
9
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
9
9

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?