mirror of https://github.com/oxen-io/session-ios
				
				
				
			
			You cannot select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
	
	
		
			267 lines
		
	
	
		
			8.1 KiB
		
	
	
	
		
			Swift
		
	
		
		
			
		
	
	
			267 lines
		
	
	
		
			8.1 KiB
		
	
	
	
		
			Swift
		
	
| 
											7 years ago
										 | // | ||
| 
											7 years ago
										 | //  Copyright (c) 2019 Open Whisper Systems. All rights reserved. | ||
| 
											7 years ago
										 | // | ||
|  | 
 | ||
|  | import PromiseKit | ||
| 
											5 years ago
										 | import SessionUIKit | ||
| 
											7 years ago
										 | 
 | ||
| 
											7 years ago
										 | public protocol GalleryRailItemProvider: class { | ||
| 
											7 years ago
										 |     var railItems: [GalleryRailItem] { get } | ||
|  | } | ||
|  | 
 | ||
| 
											7 years ago
										 | public protocol GalleryRailItem: class { | ||
| 
											7 years ago
										 |     func buildRailItemView() -> UIView | ||
| 
											7 years ago
										 | } | ||
|  | 
 | ||
|  | protocol GalleryRailCellViewDelegate: class { | ||
|  |     func didTapGalleryRailCellView(_ galleryRailCellView: GalleryRailCellView) | ||
|  | } | ||
|  | 
 | ||
| 
											7 years ago
										 | public class GalleryRailCellView: UIView { | ||
| 
											7 years ago
										 | 
 | ||
|  |     weak var delegate: GalleryRailCellViewDelegate? | ||
|  | 
 | ||
|  |     override init(frame: CGRect) { | ||
|  |         super.init(frame: frame) | ||
|  | 
 | ||
|  |         layoutMargins = .zero | ||
| 
											7 years ago
										 |         clipsToBounds = false | ||
| 
											7 years ago
										 |         addSubview(contentContainer) | ||
|  |         contentContainer.autoPinEdgesToSuperviewMargins() | ||
| 
											7 years ago
										 |         contentContainer.layer.cornerRadius = 4.8 | ||
| 
											7 years ago
										 | 
 | ||
|  |         let tapGesture = UITapGestureRecognizer(target: self, action: #selector(didTap(sender:))) | ||
|  |         addGestureRecognizer(tapGesture) | ||
|  |     } | ||
|  | 
 | ||
| 
											7 years ago
										 |     public required init?(coder aDecoder: NSCoder) { | ||
| 
											7 years ago
										 |         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 | ||
|  | 
 | ||
| 
											7 years ago
										 |         for view in contentContainer.subviews { | ||
|  |             view.removeFromSuperview() | ||
|  |         } | ||
| 
											7 years ago
										 | 
 | ||
| 
											7 years ago
										 |         let itemView = item.buildRailItemView() | ||
|  |         contentContainer.addSubview(itemView) | ||
|  |         itemView.autoPinEdgesToSuperviewEdges() | ||
| 
											7 years ago
										 |     } | ||
|  | 
 | ||
|  |     // MARK: Selected | ||
|  | 
 | ||
|  |     private(set) var isSelected: Bool = false | ||
|  | 
 | ||
| 
											7 years ago
										 |     public let cellBorderWidth: CGFloat = 3 | ||
| 
											7 years ago
										 | 
 | ||
| 
											7 years ago
										 |     func setIsSelected(_ isSelected: Bool) { | ||
|  |         self.isSelected = isSelected | ||
| 
											7 years ago
										 | 
 | ||
|  |         // Reserve space for the selection border whether or not the cell is selected. | ||
| 
											7 years ago
										 |         layoutMargins = UIEdgeInsets(top: 0, left: cellBorderWidth, bottom: 0, right: cellBorderWidth) | ||
| 
											7 years ago
										 | 
 | ||
| 
											7 years ago
										 |         if isSelected { | ||
| 
											6 years ago
										 |             contentContainer.layer.borderColor = Colors.accent.cgColor | ||
| 
											7 years ago
										 |             contentContainer.layer.borderWidth = cellBorderWidth | ||
| 
											7 years ago
										 |         } else { | ||
| 
											7 years ago
										 |             contentContainer.layer.borderWidth = 0 | ||
| 
											7 years ago
										 |         } | ||
|  |     } | ||
|  | 
 | ||
|  |     // MARK: Subview Helpers | ||
|  | 
 | ||
| 
											7 years ago
										 |     let contentContainer: UIView = { | ||
|  |         let view = UIView() | ||
|  |         view.autoPinToSquareAspectRatio() | ||
|  |         view.clipsToBounds = true | ||
| 
											7 years ago
										 | 
 | ||
| 
											7 years ago
										 |         return view | ||
| 
											7 years ago
										 |     }() | ||
|  | } | ||
|  | 
 | ||
| 
											7 years ago
										 | public protocol GalleryRailViewDelegate: class { | ||
| 
											7 years ago
										 |     func galleryRailView(_ galleryRailView: GalleryRailView, didTapItem imageRailItem: GalleryRailItem) | ||
|  | } | ||
|  | 
 | ||
| 
											7 years ago
										 | public class GalleryRailView: UIView, GalleryRailCellViewDelegate { | ||
| 
											7 years ago
										 | 
 | ||
| 
											7 years ago
										 |     public weak var delegate: GalleryRailViewDelegate? | ||
|  | 
 | ||
|  |     public var cellViews: [GalleryRailCellView] = [] | ||
|  | 
 | ||
|  |     var cellViewItems: [GalleryRailItem] { | ||
|  |         get { return cellViews.compactMap { $0.item } } | ||
|  |     } | ||
| 
											7 years ago
										 | 
 | ||
|  |     // MARK: Initializers | ||
|  | 
 | ||
|  |     override init(frame: CGRect) { | ||
|  |         super.init(frame: frame) | ||
| 
											7 years ago
										 |         clipsToBounds = false | ||
| 
											7 years ago
										 |         addSubview(scrollView) | ||
| 
											7 years ago
										 |         scrollView.clipsToBounds = false | ||
| 
											7 years ago
										 |         scrollView.layoutMargins = .zero | ||
|  |         scrollView.autoPinEdgesToSuperviewMargins() | ||
|  |     } | ||
|  | 
 | ||
| 
											7 years ago
										 |     public required init?(coder aDecoder: NSCoder) { | ||
| 
											7 years ago
										 |         fatalError("init(coder:) has not been implemented") | ||
|  |     } | ||
|  | 
 | ||
|  |     // MARK: Public | ||
|  | 
 | ||
| 
											7 years ago
										 |     public func configureCellViews(itemProvider: GalleryRailItemProvider?, focusedItem: GalleryRailItem?, cellViewBuilder: (GalleryRailItem) -> GalleryRailCellView) { | ||
| 
											7 years ago
										 |         let animationDuration: TimeInterval = 0.2 | ||
|  | 
 | ||
|  |         guard let itemProvider = itemProvider else { | ||
|  |             UIView.animate(withDuration: animationDuration) { | ||
|  |                 self.isHidden = true | ||
|  |             } | ||
| 
											7 years ago
										 |             self.cellViews = [] | ||
| 
											7 years ago
										 |             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 { | ||
| 
											7 years ago
										 |             let cellViews = scrollView.subviews | ||
|  | 
 | ||
|  |             UIView.animate(withDuration: animationDuration, | ||
|  |                            animations: { | ||
|  |                             cellViews.forEach { $0.isHidden = true } | ||
|  |                             self.isHidden = true | ||
|  |             }, | ||
|  |                            completion: { _ in cellViews.forEach { $0.removeFromSuperview() } }) | ||
| 
											7 years ago
										 |             self.cellViews = [] | ||
| 
											7 years ago
										 |             return | ||
|  |         } | ||
|  | 
 | ||
| 
											7 years ago
										 |         scrollView.subviews.forEach { $0.removeFromSuperview() } | ||
|  | 
 | ||
| 
											7 years ago
										 |         UIView.animate(withDuration: animationDuration) { | ||
|  |             self.isHidden = false | ||
|  |         } | ||
|  | 
 | ||
| 
											7 years ago
										 |         let cellViews = buildCellViews(items: itemProvider.railItems, cellViewBuilder: cellViewBuilder) | ||
| 
											7 years ago
										 |         self.cellViews = cellViews | ||
|  |         let stackView = UIStackView(arrangedSubviews: cellViews) | ||
|  |         stackView.axis = .horizontal | ||
| 
											7 years ago
										 |         stackView.spacing = 0 | ||
| 
											7 years ago
										 |         stackView.clipsToBounds = false | ||
| 
											7 years ago
										 | 
 | ||
|  |         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 | ||
|  |     }() | ||
|  | 
 | ||
| 
											7 years ago
										 |     private func buildCellViews(items: [GalleryRailItem], cellViewBuilder: (GalleryRailItem) -> GalleryRailCellView) -> [GalleryRailCellView] { | ||
| 
											7 years ago
										 |         return items.map { item in | ||
| 
											7 years ago
										 |             let cellView = cellViewBuilder(item) | ||
| 
											7 years ago
										 |             cellView.configure(item: item, delegate: self) | ||
|  |             return cellView | ||
|  |         } | ||
|  |     } | ||
|  | 
 | ||
| 
											7 years ago
										 |     enum ScrollFocusMode { | ||
|  |         case keepCentered, keepWithinBounds | ||
| 
											7 years ago
										 |     } | ||
| 
											7 years ago
										 |     var scrollFocusMode: ScrollFocusMode = .keepCentered | ||
| 
											7 years ago
										 |     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() | ||
| 
											7 years ago
										 |         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 | ||
| 
											7 years ago
										 | 
 | ||
| 
											7 years ago
										 |             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 | ||
|  |             } | ||
| 
											7 years ago
										 | 
 | ||
| 
											7 years ago
										 |             let cellFrame = selectedCell.superview!.convert(selectedCell.frame, to: scrollView) | ||
| 
											7 years ago
										 | 
 | ||
| 
											7 years ago
										 |             scrollView.scrollRectToVisible(cellFrame, animated: true) | ||
|  |         } | ||
|  |     } | ||
|  | } |