Skip to content
This repository has been archived by the owner on Aug 30, 2023. It is now read-only.

Commit

Permalink
Add photo album example. (#38)
Browse files Browse the repository at this point in the history
  • Loading branch information
jverkoey authored Aug 30, 2017
1 parent a6ae314 commit a1d49a6
Show file tree
Hide file tree
Showing 23 changed files with 569 additions and 1 deletion.
338 changes: 338 additions & 0 deletions examples/PhotoAlbumExample.swift
Original file line number Diff line number Diff line change
@@ -0,0 +1,338 @@
/*
Copyright 2017-present The Material Motion Authors. All Rights Reserved.

Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at

http://www.apache.org/licenses/LICENSE-2.0

Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
*/

import UIKit
import MotionTransitioning

// This example demonstrates how to build a photo album contextual transition.

let numberOfImageAssets = 10
let numberOfPhotosInAlbum = 30

struct Photo {
let name: String
let image: UIImage
let uuid: String

fileprivate init(name: String) {
self.uuid = NSUUID().uuidString
self.name = name

// NOTE: In a real app you should never load images from disk on the UI thread like this.
// Instead, you should find some way to cache the thumbnails in memory and then asynchronously
// load the full-size photos from disk/network when needed. The photo library APIs provide
// exactly this sort of behavior (square thumbnails are accessible immediately on the UI thread
// while the full-sized photos need to be loaded asynchronously).
self.image = UIImage(named: "\(self.name).jpg")!
}
}

class PhotoAlbum {
let photos: [Photo]
let identifierToIndex: [String: Int]

init() {
var photos: [Photo] = []
var identifierToIndex: [String: Int] = [:]
for index in 0..<numberOfPhotosInAlbum {
let photo = Photo(name: "image\(index % numberOfImageAssets)")
photos.append(photo)
identifierToIndex[photo.uuid] = index
}
self.photos = photos
self.identifierToIndex = identifierToIndex
}
}

private let photoCellIdentifier = "photoCell"

private class PhotoCollectionViewCell: UICollectionViewCell {
let imageView = UIImageView()

override init(frame: CGRect) {
super.init(frame: frame)

imageView.contentMode = .scaleAspectFill
imageView.autoresizingMask = [.flexibleWidth, .flexibleHeight]
imageView.frame = bounds
imageView.clipsToBounds = true

contentView.addSubview(imageView)
}

required init?(coder aDecoder: NSCoder) {
fatalError("init(coder:) has not been implemented")
}
}

public class PhotoAlbumExampleViewController: UICollectionViewController, PhotoAlbumTransitionDelegate {

let album = PhotoAlbum()

init() {
super.init(collectionViewLayout: UICollectionViewFlowLayout())
}

public required init?(coder aDecoder: NSCoder) {
fatalError("init(coder:) has not been implemented")
}

override public func viewDidLoad() {
super.viewDidLoad()

collectionView!.backgroundColor = .white
collectionView!.register(PhotoCollectionViewCell.self,
forCellWithReuseIdentifier: photoCellIdentifier)
}

public override func viewDidLayoutSubviews() {
super.viewDidLayoutSubviews()

updateLayout()
}

func updateLayout() {
let layout = collectionView!.collectionViewLayout as! UICollectionViewFlowLayout
layout.sectionInset = .init(top: 4, left: 4, bottom: 4, right: 4)
layout.minimumInteritemSpacing = 4
layout.minimumLineSpacing = 4

let numberOfColumns: CGFloat = 3
let squareDimension = (view.bounds.width - layout.sectionInset.left - layout.sectionInset.right - (numberOfColumns - 1) * layout.minimumInteritemSpacing) / numberOfColumns
layout.itemSize = CGSize(width: squareDimension, height: squareDimension)
}

public override func collectionView(_ collectionView: UICollectionView, numberOfItemsInSection section: Int) -> Int {
return album.photos.count
}

public override func collectionView(_ collectionView: UICollectionView, cellForItemAt indexPath: IndexPath) -> UICollectionViewCell {
let cell = collectionView.dequeueReusableCell(withReuseIdentifier: photoCellIdentifier,
for: indexPath) as! PhotoCollectionViewCell
let photo = album.photos[indexPath.row]
cell.imageView.image = photo.image
return cell
}

public override func collectionView(_ collectionView: UICollectionView, didSelectItemAt indexPath: IndexPath) {
let viewController = PhotoAlbumViewController(album: album)
viewController.currentPhoto = album.photos[indexPath.row]
viewController.transitionController.transition = PhotoAlbumTransition(delegate: self)
present(viewController, animated: true)
}

fileprivate func contextView(forAlbumViewController: PhotoAlbumViewController) -> UIImageView? {
let currentPhoto = forAlbumViewController.currentPhoto
guard let photoIndex = album.identifierToIndex[currentPhoto.uuid] else {
return nil
}
let photoIndexPath = IndexPath(item: photoIndex, section: 0)
if collectionView?.cellForItem(at: photoIndexPath) == nil {
collectionView?.scrollToItem(at: photoIndexPath, at: .top, animated: false)
collectionView?.reloadItems(at: [photoIndexPath])
}
guard let cell = collectionView?.cellForItem(at: photoIndexPath) as? PhotoCollectionViewCell else {
return nil
}
return cell.imageView
}
}

private class PhotoAlbumViewController: UIViewController, UICollectionViewDataSource, UICollectionViewDelegate {

var collectionView: UICollectionView!
var currentPhoto: Photo

let album: PhotoAlbum
init(album: PhotoAlbum) {
self.album = album
self.currentPhoto = self.album.photos.first!

super.init(nibName: nil, bundle: nil)
}

required init?(coder aDecoder: NSCoder) {
fatalError("init(coder:) has not been implemented")
}

override func viewDidLoad() {
super.viewDidLoad()

automaticallyAdjustsScrollViewInsets = false

let layout = UICollectionViewFlowLayout()
layout.itemSize = view.bounds.size
layout.minimumInteritemSpacing = 0
layout.minimumLineSpacing = 8
layout.footerReferenceSize = CGSize(width: layout.minimumLineSpacing / 2,
height: view.bounds.size.height)
layout.headerReferenceSize = layout.footerReferenceSize
layout.scrollDirection = .horizontal

collectionView = UICollectionView(frame: view.bounds, collectionViewLayout: layout)
collectionView.isPagingEnabled = true
collectionView.backgroundColor = .backgroundColor
collectionView.showsHorizontalScrollIndicator = false
collectionView.dataSource = self
collectionView.delegate = self

collectionView.register(PhotoCollectionViewCell.self,
forCellWithReuseIdentifier: photoCellIdentifier)

var extendedBounds = view.bounds
extendedBounds.size.width = extendedBounds.width + layout.minimumLineSpacing
collectionView.bounds = extendedBounds

view.addSubview(collectionView)
}

override func viewDidLayoutSubviews() {
super.viewDidLayoutSubviews()

collectionView.center = CGPoint(x: view.bounds.midX, y: view.bounds.midY)
}

override func viewWillAppear(_ animated: Bool) {
super.viewWillAppear(animated)

navigationController?.setNavigationBarHidden(true, animated: animated)

let photoIndex = album.photos.index { $0.image == currentPhoto.image }!
let indexPath = IndexPath(item: photoIndex, section: 0)
collectionView.scrollToItem(at: indexPath, at: .centeredHorizontally, animated: false)
}

override var preferredStatusBarStyle: UIStatusBarStyle {
return .lightContent
}

func collectionView(_ collectionView: UICollectionView, numberOfItemsInSection section: Int) -> Int {
return album.photos.count
}

func collectionView(_ collectionView: UICollectionView, cellForItemAt indexPath: IndexPath) -> UICollectionViewCell {
let cell = collectionView.dequeueReusableCell(withReuseIdentifier: photoCellIdentifier,
for: indexPath) as! PhotoCollectionViewCell
let photo = album.photos[indexPath.row]
cell.imageView.image = photo.image
cell.imageView.contentMode = .scaleAspectFit
return cell
}

func collectionView(_ collectionView: UICollectionView, didSelectItemAt indexPath: IndexPath) {
dismiss(animated: true)
}

func scrollViewDidEndDecelerating(_ scrollView: UIScrollView) {
currentPhoto = album.photos[indexPathForCurrentPhoto().item]
}

func indexPathForCurrentPhoto() -> IndexPath {
return collectionView.indexPathsForVisibleItems.first!
}
}

private protocol PhotoAlbumTransitionDelegate {
func contextView(forAlbumViewController: PhotoAlbumViewController) -> UIImageView?
}

private class PhotoAlbumTransition: NSObject, Transition, TransitionWithFallback {

// Store the context for the lifetime of the transition.
let delegate: PhotoAlbumTransitionDelegate
init(delegate: PhotoAlbumTransitionDelegate) {
self.delegate = delegate
}

func fallbackTransition(with context: TransitionContext) -> Transition? {
if delegate.contextView(forAlbumViewController: context.foreViewController as! PhotoAlbumViewController) != nil {
return self
}
return nil
}

func start(with context: TransitionContext) {
guard let contextView = delegate.contextView(forAlbumViewController: context.foreViewController as! PhotoAlbumViewController) else {
return
}

// A small helper function for creating bi-directional animations.
// See https://github.com/material-motion/motion-animator-objc for a more versatile
// bidirectional Core Animation implementation.
let addAnimationToLayer: (CABasicAnimation, CALayer) -> Void = { animation, layer in
if context.direction == .backward {
let swap = animation.fromValue
animation.fromValue = animation.toValue
animation.toValue = swap
}
layer.add(animation, forKey: animation.keyPath)
layer.setValue(animation.toValue, forKeyPath: animation.keyPath!)
}

let snapshotter = TransitionViewSnapshotter(containerView: context.containerView)
context.defer {
snapshotter.removeAllSnapshots()
}

let foreVC = context.foreViewController as! PhotoAlbumViewController
let foreImageView = (foreVC.collectionView.cellForItem(at: foreVC.indexPathForCurrentPhoto()) as! PhotoCollectionViewCell).imageView
let imageSize = foreImageView.image!.size

let fitScale = min(foreImageView.bounds.width / imageSize.width,
foreImageView.bounds.height / imageSize.height)
let fitSize = CGSize(width: fitScale * imageSize.width, height: fitScale * imageSize.height)

foreImageView.isHidden = true
context.defer {
foreImageView.isHidden = false
}

CATransaction.begin()
CATransaction.setCompletionBlock {
context.transitionDidEnd()
}

let fadeIn = CABasicAnimation(keyPath: "opacity")
fadeIn.timingFunction = CAMediaTimingFunction(name: kCAMediaTimingFunctionEaseInEaseOut)
fadeIn.fromValue = 0
fadeIn.toValue = 1
addAnimationToLayer(fadeIn, context.foreViewController.view.layer)

let snapshotContextView = snapshotter.snapshot(of: contextView,
isAppearing: context.direction == .backward)

let shift = CASpringAnimation(keyPath: "position")
shift.damping = 500
shift.stiffness = 1000
shift.mass = 3
shift.duration = 0.5
shift.fromValue = snapshotContextView.layer.position
shift.toValue = CGPoint(x: context.foreViewController.view.bounds.midX,
y: context.foreViewController.view.bounds.midY)
addAnimationToLayer(shift, snapshotContextView.layer)

let expansion = CASpringAnimation(keyPath: "bounds.size")
expansion.damping = 500
expansion.stiffness = 1000
expansion.mass = 3
expansion.duration = 0.5
expansion.fromValue = snapshotContextView.layer.bounds.size
expansion.toValue = fitSize
addAnimationToLayer(expansion, snapshotContextView.layer)

CATransaction.commit()
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,21 @@
{
"images" : [
{
"idiom" : "universal",
"scale" : "1x",
"filename" : "image0.jpg"
},
{
"idiom" : "universal",
"scale" : "2x"
},
{
"idiom" : "universal",
"scale" : "3x"
}
],
"info" : {
"version" : 1,
"author" : "xcode"
}
}
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Original file line number Diff line number Diff line change
@@ -0,0 +1,21 @@
{
"images" : [
{
"idiom" : "universal",
"scale" : "1x",
"filename" : "image1.jpg"
},
{
"idiom" : "universal",
"scale" : "2x"
},
{
"idiom" : "universal",
"scale" : "3x"
}
],
"info" : {
"version" : 1,
"author" : "xcode"
}
}
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading

0 comments on commit a1d49a6

Please sign in to comment.