This repository has been archived by the owner on Aug 30, 2023. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 12
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Loading branch information
Showing
23 changed files
with
569 additions
and
1 deletion.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
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() | ||
} | ||
} |
21 changes: 21 additions & 0 deletions
21
examples/apps/Catalog/Catalog/PhotoAlbum.xcassets/image0.imageset/Contents.json
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
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" | ||
} | ||
} |
Binary file added
BIN
+201 KB
examples/apps/Catalog/Catalog/PhotoAlbum.xcassets/image0.imageset/image0.jpg
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
21 changes: 21 additions & 0 deletions
21
examples/apps/Catalog/Catalog/PhotoAlbum.xcassets/image1.imageset/Contents.json
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
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" | ||
} | ||
} |
Binary file added
BIN
+219 KB
examples/apps/Catalog/Catalog/PhotoAlbum.xcassets/image1.imageset/image1.jpg
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Oops, something went wrong.