|  |  |  | // | 
					
						
							|  |  |  | //  Copyright (c) 2019 Open Whisper Systems. All rights reserved. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import PromiseKit | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | public protocol GalleryRailItemProvider: class { | 
					
						
							|  |  |  |     var railItems: [GalleryRailItem] { get } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | public protocol GalleryRailItem: class { | 
					
						
							|  |  |  |     func getRailImage() -> Promise<UIImage> | 
					
						
							|  |  |  |     var aspectRatio: CGFloat { get } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | protocol GalleryRailCellViewDelegate: class { | 
					
						
							|  |  |  |     func didTapGalleryRailCellView(_ galleryRailCellView: GalleryRailCellView) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | public class GalleryRailCellView: UIView { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     weak var delegate: GalleryRailCellViewDelegate? | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     override init(frame: CGRect) { | 
					
						
							|  |  |  |         super.init(frame: frame) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         layoutMargins = .zero | 
					
						
							|  |  |  |         clipsToBounds = false | 
					
						
							|  |  |  |         addSubview(imageView) | 
					
						
							|  |  |  |         imageView.autoPinEdgesToSuperviewMargins() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         let tapGesture = UITapGestureRecognizer(target: self, action: #selector(didTap(sender:))) | 
					
						
							|  |  |  |         addGestureRecognizer(tapGesture) | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     public required init?(coder aDecoder: NSCoder) { | 
					
						
							|  |  |  |         fatalError("init(coder:) has not been implemented") | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // MARK: Actions | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     @objc | 
					
						
							|  |  |  |     func didTap(sender: UITapGestureRecognizer) { | 
					
						
							|  |  |  |         self.delegate?.didTapGalleryRailCellView(self) | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // MARK:  | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     var item: GalleryRailItem? | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     func configure(item: GalleryRailItem, delegate: GalleryRailCellViewDelegate) { | 
					
						
							|  |  |  |         self.item = item | 
					
						
							|  |  |  |         self.delegate = delegate | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         item.getRailImage().done { image in | 
					
						
							|  |  |  |             guard self.item === item else { return } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             self.imageView.image = image | 
					
						
							|  |  |  |         }.retainUntilComplete() | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // MARK: Selected | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     private(set) var isSelected: Bool = false | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     public let cellBorderWidth: CGFloat = 2 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     func setIsSelected(_ isSelected: Bool) { | 
					
						
							|  |  |  |         self.isSelected = isSelected | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Reserve space for the selection border whether or not the cell is selected. | 
					
						
							|  |  |  |         layoutMargins = UIEdgeInsets(top: 0, left: cellBorderWidth, bottom: 0, right: cellBorderWidth) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if isSelected { | 
					
						
							|  |  |  |             imageView.layer.borderColor = Theme.galleryHighlightColor.cgColor | 
					
						
							|  |  |  |             imageView.layer.borderWidth = cellBorderWidth | 
					
						
							|  |  |  |             imageView.layer.cornerRadius = cellBorderWidth | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             imageView.layer.borderWidth = 0 | 
					
						
							|  |  |  |             imageView.layer.cornerRadius = 0 | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // MARK: Subview Helpers | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     let imageView: UIImageView = { | 
					
						
							|  |  |  |         let imageView = UIImageView() | 
					
						
							|  |  |  |         imageView.contentMode = .scaleAspectFill | 
					
						
							|  |  |  |         imageView.autoPinToSquareAspectRatio() | 
					
						
							|  |  |  |         imageView.clipsToBounds = true | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return imageView | 
					
						
							|  |  |  |     }() | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | public protocol GalleryRailViewDelegate: class { | 
					
						
							|  |  |  |     func galleryRailView(_ galleryRailView: GalleryRailView, didTapItem imageRailItem: GalleryRailItem) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | public class GalleryRailView: UIView, GalleryRailCellViewDelegate { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     public weak var delegate: GalleryRailViewDelegate? | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     public var cellViews: [GalleryRailCellView] = [] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     var cellViewItems: [GalleryRailItem] { | 
					
						
							|  |  |  |         get { return cellViews.compactMap { $0.item } } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // MARK: Initializers | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     override init(frame: CGRect) { | 
					
						
							|  |  |  |         super.init(frame: frame) | 
					
						
							|  |  |  |         clipsToBounds = false | 
					
						
							|  |  |  |         addSubview(scrollView) | 
					
						
							|  |  |  |         scrollView.clipsToBounds = false | 
					
						
							|  |  |  |         scrollView.layoutMargins = .zero | 
					
						
							|  |  |  |         scrollView.autoPinEdgesToSuperviewMargins() | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     public required init?(coder aDecoder: NSCoder) { | 
					
						
							|  |  |  |         fatalError("init(coder:) has not been implemented") | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // MARK: Public | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     public func configureCellViews(itemProvider: GalleryRailItemProvider?, focusedItem: GalleryRailItem?, cellViewBuilder: () -> GalleryRailCellView) { | 
					
						
							|  |  |  |         let animationDuration: TimeInterval = 0.2 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         guard let itemProvider = itemProvider else { | 
					
						
							|  |  |  |             UIView.animate(withDuration: animationDuration) { | 
					
						
							|  |  |  |                 self.isHidden = true | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             self.cellViews = [] | 
					
						
							|  |  |  |             return | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         let areRailItemsIdentical = { (lhs: [GalleryRailItem], rhs: [GalleryRailItem]) -> Bool in | 
					
						
							|  |  |  |             guard lhs.count == rhs.count else { | 
					
						
							|  |  |  |                 return false | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             for (index, element) in lhs.enumerated() { | 
					
						
							|  |  |  |                 guard element === rhs[index] else { | 
					
						
							|  |  |  |                     return false | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             return true | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if itemProvider === self.itemProvider, areRailItemsIdentical(itemProvider.railItems, self.cellViewItems) { | 
					
						
							|  |  |  |             UIView.animate(withDuration: animationDuration) { | 
					
						
							|  |  |  |                 self.updateFocusedItem(focusedItem) | 
					
						
							|  |  |  |                 self.layoutIfNeeded() | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.itemProvider = itemProvider | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         guard itemProvider.railItems.count > 1 else { | 
					
						
							|  |  |  |             let cellViews = scrollView.subviews | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             UIView.animate(withDuration: animationDuration, | 
					
						
							|  |  |  |                            animations: { | 
					
						
							|  |  |  |                             cellViews.forEach { $0.isHidden = true } | 
					
						
							|  |  |  |                             self.isHidden = true | 
					
						
							|  |  |  |             }, | 
					
						
							|  |  |  |                            completion: { _ in cellViews.forEach { $0.removeFromSuperview() } }) | 
					
						
							|  |  |  |             self.cellViews = [] | 
					
						
							|  |  |  |             return | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         scrollView.subviews.forEach { $0.removeFromSuperview() } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         UIView.animate(withDuration: animationDuration) { | 
					
						
							|  |  |  |             self.isHidden = false | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         let cellViews = buildCellViews(items: itemProvider.railItems, cellViewBuilder: cellViewBuilder) | 
					
						
							|  |  |  |         self.cellViews = cellViews | 
					
						
							|  |  |  |         let stackView = UIStackView(arrangedSubviews: cellViews) | 
					
						
							|  |  |  |         stackView.axis = .horizontal | 
					
						
							|  |  |  |         stackView.spacing = 2 | 
					
						
							|  |  |  |         stackView.clipsToBounds = false | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         scrollView.addSubview(stackView) | 
					
						
							|  |  |  |         stackView.autoPinEdgesToSuperviewEdges() | 
					
						
							|  |  |  |         stackView.autoMatch(.height, to: .height, of: scrollView) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         updateFocusedItem(focusedItem) | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // MARK: GalleryRailCellViewDelegate | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     func didTapGalleryRailCellView(_ galleryRailCellView: GalleryRailCellView) { | 
					
						
							|  |  |  |         guard let item = galleryRailCellView.item else { | 
					
						
							|  |  |  |             owsFailDebug("item was unexpectedly nil") | 
					
						
							|  |  |  |             return | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         delegate?.galleryRailView(self, didTapItem: item) | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // MARK: Subview Helpers | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     private var itemProvider: GalleryRailItemProvider? | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     private let scrollView: UIScrollView = { | 
					
						
							|  |  |  |         let scrollView = UIScrollView() | 
					
						
							|  |  |  |         scrollView.isScrollEnabled = true | 
					
						
							|  |  |  |         return scrollView | 
					
						
							|  |  |  |     }() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     private func buildCellViews(items: [GalleryRailItem], cellViewBuilder: () -> GalleryRailCellView) -> [GalleryRailCellView] { | 
					
						
							|  |  |  |         return items.map { item in | 
					
						
							|  |  |  |             let cellView = cellViewBuilder() | 
					
						
							|  |  |  |             cellView.configure(item: item, delegate: self) | 
					
						
							|  |  |  |             return cellView | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     enum ScrollFocusMode { | 
					
						
							|  |  |  |         case keepCentered, keepWithinBounds | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     var scrollFocusMode: ScrollFocusMode = .keepCentered | 
					
						
							|  |  |  |     func updateFocusedItem(_ focusedItem: GalleryRailItem?) { | 
					
						
							|  |  |  |         var selectedCellView: GalleryRailCellView? | 
					
						
							|  |  |  |         cellViews.forEach { cellView in | 
					
						
							|  |  |  |             if cellView.item === focusedItem { | 
					
						
							|  |  |  |                 assert(selectedCellView == nil) | 
					
						
							|  |  |  |                 selectedCellView = cellView | 
					
						
							|  |  |  |                 cellView.setIsSelected(true) | 
					
						
							|  |  |  |             } else { | 
					
						
							|  |  |  |                 cellView.setIsSelected(false) | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.layoutIfNeeded() | 
					
						
							|  |  |  |         switch scrollFocusMode { | 
					
						
							|  |  |  |         case .keepCentered: | 
					
						
							|  |  |  |             guard let selectedCell = selectedCellView else { | 
					
						
							|  |  |  |                 owsFailDebug("selectedCell was unexpectedly nil") | 
					
						
							|  |  |  |                 return | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             let cellViewCenter = selectedCell.superview!.convert(selectedCell.center, to: scrollView) | 
					
						
							|  |  |  |             let additionalInset = scrollView.center.x - cellViewCenter.x | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             var inset = scrollView.contentInset | 
					
						
							|  |  |  |             inset.left = additionalInset | 
					
						
							|  |  |  |             scrollView.contentInset = inset | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             var offset = scrollView.contentOffset | 
					
						
							|  |  |  |             offset.x = -additionalInset | 
					
						
							|  |  |  |             scrollView.contentOffset = offset | 
					
						
							|  |  |  |         case .keepWithinBounds: | 
					
						
							|  |  |  |             guard let selectedCell = selectedCellView else { | 
					
						
							|  |  |  |                 owsFailDebug("selectedCell was unexpectedly nil") | 
					
						
							|  |  |  |                 return | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             let cellFrame = selectedCell.superview!.convert(selectedCell.frame, to: scrollView) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             scrollView.scrollRectToVisible(cellFrame, animated: true) | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } |