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
		
	
//
 | 
						|
//  Copyright (c) 2019 Open Whisper Systems. All rights reserved.
 | 
						|
//
 | 
						|
 | 
						|
import PromiseKit
 | 
						|
import SessionUIKit
 | 
						|
 | 
						|
public protocol GalleryRailItemProvider: class {
 | 
						|
    var railItems: [GalleryRailItem] { get }
 | 
						|
}
 | 
						|
 | 
						|
public protocol GalleryRailItem: class {
 | 
						|
    func buildRailItemView() -> UIView
 | 
						|
}
 | 
						|
 | 
						|
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(contentContainer)
 | 
						|
        contentContainer.autoPinEdgesToSuperviewMargins()
 | 
						|
        contentContainer.layer.cornerRadius = 4.8
 | 
						|
 | 
						|
        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
 | 
						|
 | 
						|
        for view in contentContainer.subviews {
 | 
						|
            view.removeFromSuperview()
 | 
						|
        }
 | 
						|
 | 
						|
        let itemView = item.buildRailItemView()
 | 
						|
        contentContainer.addSubview(itemView)
 | 
						|
        itemView.autoPinEdgesToSuperviewEdges()
 | 
						|
    }
 | 
						|
 | 
						|
    // MARK: Selected
 | 
						|
 | 
						|
    private(set) var isSelected: Bool = false
 | 
						|
 | 
						|
    public let cellBorderWidth: CGFloat = 3
 | 
						|
 | 
						|
    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 {
 | 
						|
            contentContainer.layer.borderColor = Colors.accent.cgColor
 | 
						|
            contentContainer.layer.borderWidth = cellBorderWidth
 | 
						|
        } else {
 | 
						|
            contentContainer.layer.borderWidth = 0
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    // MARK: Subview Helpers
 | 
						|
 | 
						|
    let contentContainer: UIView = {
 | 
						|
        let view = UIView()
 | 
						|
        view.autoPinToSquareAspectRatio()
 | 
						|
        view.clipsToBounds = true
 | 
						|
 | 
						|
        return view
 | 
						|
    }()
 | 
						|
}
 | 
						|
 | 
						|
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: (GalleryRailItem) -> 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 = 0
 | 
						|
        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: (GalleryRailItem) -> GalleryRailCellView) -> [GalleryRailCellView] {
 | 
						|
        return items.map { item in
 | 
						|
            let cellView = cellViewBuilder(item)
 | 
						|
            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)
 | 
						|
        }
 | 
						|
    }
 | 
						|
}
 |