うるおいらんど

【Swift 3】UICollectionViewでreloadDataをする時にアニメーションを追加する【CATransition】

追記があります。

どもども。Reoです。

UICollectionViewでreloadData()をする際にアニメーションを追加する方法をようやく見つけたので、紹介します。トランジションといったほうがいいのか。

ページング等での切り替えではなくコレクションビューをリロードした際にページが変わったように見せるといった感じです。以下のような感じ。

上下に何かviewを上から置いていれば、いい感じに見えるはず。

今回は紹介のためにわりと適当にデモを作ってみましたが、実際はここにカレンダーをおいて上下スワイプで月を切り替える際に用いました。

 

先に全体コードを載せてからちょろっと説明します。

結構雑にかいたのであんまりいい書き方ではないかもしれませんがご了承ください・・・(´・ω・`)

全体コード

ViewController.swift

//
//  ViewController.swift
//  CollectionViewAnimation
//
//  Created by Reo on 2017/04/14.
//  Copyright © 2017年 Reo. All rights reserved.
//

import UIKit

class ViewController: UIViewController ,UICollectionViewDelegate,UICollectionViewDataSource{
    
    private var collectionView:UICollectionView!
    private var backgroundColor = UIColor.white
    private var reloadCount = 0


    override func viewDidLoad() {
        super.viewDidLoad()
        
        let width = self.view.frame.width
        
        //適当にコレクションビューを配置
        let layout = UICollectionViewFlowLayout()
        layout.itemSize =  CGSize(width:50,height:50)
        layout.sectionInset = UIEdgeInsetsMake(10, 0, 10, 0)
        
        collectionView = UICollectionView(frame: CGRect(x:0,y:40,width:width,height:320),
                                          collectionViewLayout: layout)
        collectionView.register(CustomCell.self, forCellWithReuseIdentifier: "cell")
        collectionView.delegate = self
        collectionView.dataSource = self
        self.view.addSubview(collectionView)
        
        
        //ジェスチャーをつける
        let swipeUp = UISwipeGestureRecognizer()
        swipeUp.direction = .up
        swipeUp.addTarget(self, action: #selector(self.swipeAction(sender:)))
        self.view.addGestureRecognizer(swipeUp)
        
        let swipeDown = UISwipeGestureRecognizer()
        swipeDown.direction = .down
        swipeDown.addTarget(self,action:#selector(self.swipeAction(sender:)))
        self.view.addGestureRecognizer(swipeDown)
        
        let swipeLeft = UISwipeGestureRecognizer()
        swipeLeft.direction = .left
        swipeLeft.addTarget(self,action:#selector(self.swipeAction(sender:)))
        self.view.addGestureRecognizer(swipeLeft)
        
        let swipeRight = UISwipeGestureRecognizer()
        swipeRight.direction = .right
        swipeRight.addTarget(self,action:#selector(self.swipeAction(sender:)))
        self.view.addGestureRecognizer(swipeRight)

    }
    
    //スワイプした時のアクション
    func swipeAction(sender:UISwipeGestureRecognizer){
        //向きごとに変える
        switch sender.direction{
        case UISwipeGestureRecognizerDirection.up:
            reloadCount += 1
            self.collectionView.layer.add(swipeTransition(tag: 0), forKey: nil)
        case UISwipeGestureRecognizerDirection.down:
            reloadCount -= 1
            self.collectionView.layer.add(swipeTransition(tag: 1), forKey: nil)
        case UISwipeGestureRecognizerDirection.left:
            backgroundColor = UIColor.red
            self.collectionView.layer.add(swipeTransition(tag: 2), forKey: nil)
        case UISwipeGestureRecognizerDirection.right:
            backgroundColor = UIColor.yellow
            self.collectionView.layer.add(swipeTransition(tag: 3), forKey: nil)
        default:break
        }
        
        self.collectionView.reloadData()
    }
    
    //更新時のアニメーション
    func swipeTransition(tag:Int) -> CATransition{
        let transition:CATransition = CATransition()
        transition.startProgress = 0
        transition.endProgress = 1.0
        transition.type = kCATransitionPush
        switch tag{
        case 0:
            transition.subtype = kCATransitionFromTop
        case 1:
            transition.subtype = kCATransitionFromBottom
        case 2:
            transition.subtype = kCATransitionFromRight
        case 3:
            transition.subtype = kCATransitionFromLeft
        default:
            transition.subtype = kCATransitionFromTop
        }
        transition.duration = 0.3
        return transition
        
    }
    
    //セルの数
    func collectionView(_ collectionView: UICollectionView, numberOfItemsInSection section: Int) -> Int {
        return 30
    }
    
    //セルの内容
    func collectionView(_ collectionView: UICollectionView, cellForItemAt indexPath: IndexPath) -> UICollectionViewCell {
        let cell = collectionView.dequeueReusableCell(withReuseIdentifier: "cell", for: indexPath) as! CustomCell
        
        cell.textLabel.text = "\(reloadCount)"
        cell.contentView.backgroundColor = backgroundColor
        
        return cell
    }
    
    
    //セルを選択した時
    func collectionView(_ collectionView: UICollectionView, didSelectItemAt indexPath: IndexPath) {
        
    }

    override func didReceiveMemoryWarning() {
        super.didReceiveMemoryWarning()
        // Dispose of any resources that can be recreated.
    }


}

CustomCell.swift

import UIKit

class CustomCell: UICollectionViewCell {
    
    public var textLabel:UILabel!
    
    override init(frame:CGRect){
        super.init(frame:frame)
        
        //日付ラベル
        textLabel = UILabel()
        textLabel.frame = CGRect(x:0,y:0,width:self.frame.height,height:self.frame.height)
        textLabel.center = CGPoint(x:self.frame.width / 2,y:self.frame.height / 2)
        textLabel.textAlignment = .center
        textLabel.font = UIFont.systemFont(ofSize: 14)
        textLabel.textColor = UIColor.blue
        self.contentView.addSubview(textLabel)
        
    }
    
    required init?(coder aDecoder: NSCoder) {
        fatalError("init(coder:) has not been implemented")
    }
    
}

 

上にスワイプしたときには、数字を1プラス

下にスワイプしたときには、数字を1マイナス

左右はセルの背景色が変わるようにしました。

 

コレクションビューをリロードする際にアニメーションをつける

上記はとりあえず全方向にアニメーションをつけてみましたが、場合分けしてるだけなので一方向部分だけ簡単に紹介したいと思います。

上にスワイプした場合

        //ジェスチャーをつける
        let swipeUp = UISwipeGestureRecognizer()
        swipeUp.direction = .up
        swipeUp.addTarget(self, action: #selector(self.swipeAction(sender:)))
        self.view.addGestureRecognizer(swipeUp)

まずはジェスチャーをつけます。

そしてトランジションを作成します。

    //更新時のアニメーションを作成
    func swipeTransition(tag:Int) -> CATransition{
        let transition:CATransition = CATransition()
        transition.startProgress = 0
        transition.endProgress = 1.0

        //push・moveIn・Reveal・Fadeから選択
        transition.type = kCATransitionPush

        //どちら側からトランジションするかを選択
        transition.subtype = kCATransitionFromTop

        //トランジションにどれだけ時間をかけるか
        transition.duration = 0.3
        return transition
    }

 

アニメーション部分は以前書いたこちらの記事とほぼ同様な感じです。

【Swift 3】UINavigationControllerを用いた画面遷移のアニメーション変更【CATransition】

 

上にスワイプした際のアクションを書いていきます。

    //スワイプした時のアクション
    func swipeAction(sender:UISwipeGestureRecognizer){
        //適当にセルの中身を変える処理(カレンダーだと次の月に変えるとかとか)
        reloadCount += 1

        //コレクションビューのレイヤーにアニメーションを追加する
        self.collectionView.layer.add(swipeTransition(tag: 0), forKey: nil)

        //コレクションビューをリロードする
        self.collectionView.reloadData()
    }

 

先ほど作成したアニメーションをコレクションビューのレイヤーに追加してやります。

 

こんな感じです+:。ヾ(o・ω・)ノ゜.+

意外と簡単・・・!

 

もうちょっとアニメーションの速度に緩急つけたりした方がいい感じになるかもしれないですが、とりあえずはこんな感じで実装しました。

 

私がやりたかったのはコレクションビューを用いたカレンダーの無限(?)切り替えで、前後の月のカレンダーが見えて欲しかったんですね。reloadDataだとパッと切り替わってしまい、スワイプで移動している感覚もなくて正直めちゃくちゃ悩みました。

スクロールビューの上に3つのせてどうやらとかも考えましたが、あんまりview自体も増やしたくなかったので、どうにかアニメーションで誤魔化したかったのです。

本来はスクロールビューを使ったやり方のが一般的なのかなぁ・・・。わからない。

 

にしてもリロード前のセルとリロード後のセルの両方が見えているのが、アニメーションすげえ・・・って感じです。

もっとカスタマイズとかして使えるようになりたいところですね。

 

それではでは!

めちゃくちゃ色々探し回ってたどり着いたので本当に感謝・・・

 

参考リンク

Additional Notes追記

Gistにあげました。

Swift4に対応してgistにあげときましたん。

Swift3からSwift4に対応するより、昔の書き方から今の書き方にする方が骨が折れる。(。-ω-。)

Comments

コメントはありません。

現在コメントフォームは工事中です。