123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636 |
- //
- // FSPagerView.swift
- // FSPagerView
- //
- // Created by Wenchao Ding on 17/12/2016.
- // Copyright © 2016 Wenchao Ding. All rights reserved.
- //
- // https://github.com/WenchaoD
- //
- // FSPagerView is an elegant Screen Slide Library implemented primarily with UICollectionView. It is extremely helpful for making Banner、Product Show、Welcome/Guide Pages、Screen/ViewController Sliders.
- //
- import UIKit
- @objc
- public protocol FSPagerViewDataSource: NSObjectProtocol {
-
- /// Asks your data source object for the number of items in the pager view.
- @objc(numberOfItemsInPagerView:)
- func numberOfItems(in pagerView: FSPagerView) -> Int
-
- /// Asks your data source object for the cell that corresponds to the specified item in the pager view.
- @objc(pagerView:cellForItemAtIndex:)
- func pagerView(_ pagerView: FSPagerView, cellForItemAt index: Int) -> FSPagerViewCell
-
- }
- @objc
- public protocol FSPagerViewDelegate: NSObjectProtocol {
-
- /// Asks the delegate if the item should be highlighted during tracking.
- @objc(pagerView:shouldHighlightItemAtIndex:)
- optional func pagerView(_ pagerView: FSPagerView, shouldHighlightItemAt index: Int) -> Bool
-
- /// Tells the delegate that the item at the specified index was highlighted.
- @objc(pagerView:didHighlightItemAtIndex:)
- optional func pagerView(_ pagerView: FSPagerView, didHighlightItemAt index: Int)
-
- /// Asks the delegate if the specified item should be selected.
- @objc(pagerView:shouldSelectItemAtIndex:)
- optional func pagerView(_ pagerView: FSPagerView, shouldSelectItemAt index: Int) -> Bool
-
- /// Tells the delegate that the item at the specified index was selected.
- @objc(pagerView:didSelectItemAtIndex:)
- optional func pagerView(_ pagerView: FSPagerView, didSelectItemAt index: Int)
-
- /// Tells the delegate that the specified cell is about to be displayed in the pager view.
- @objc(pagerView:willDisplayCell:forItemAtIndex:)
- optional func pagerView(_ pagerView: FSPagerView, willDisplay cell: FSPagerViewCell, forItemAt index: Int)
-
- /// Tells the delegate that the specified cell was removed from the pager view.
- @objc(pagerView:didEndDisplayingCell:forItemAtIndex:)
- optional func pagerView(_ pagerView: FSPagerView, didEndDisplaying cell: FSPagerViewCell, forItemAt index: Int)
-
- /// Tells the delegate when the pager view is about to start scrolling the content.
- @objc(pagerViewWillBeginDragging:)
- optional func pagerViewWillBeginDragging(_ pagerView: FSPagerView)
-
- /// Tells the delegate when the user finishes scrolling the content.
- @objc(pagerViewWillEndDragging:targetIndex:)
- optional func pagerViewWillEndDragging(_ pagerView: FSPagerView, targetIndex: Int)
-
- /// Tells the delegate when the user scrolls the content view within the receiver.
- @objc(pagerViewDidScroll:)
- optional func pagerViewDidScroll(_ pagerView: FSPagerView)
-
- /// Tells the delegate when a scrolling animation in the pager view concludes.
- @objc(pagerViewDidEndScrollAnimation:)
- optional func pagerViewDidEndScrollAnimation(_ pagerView: FSPagerView)
-
- /// Tells the delegate that the pager view has ended decelerating the scrolling movement.
- @objc(pagerViewDidEndDecelerating:)
- optional func pagerViewDidEndDecelerating(_ pagerView: FSPagerView)
-
- }
- @IBDesignable
- open class FSPagerView: UIView,UICollectionViewDataSource,UICollectionViewDelegate {
-
- // MARK: - Public properties
- /// The object that acts as the data source of the pager view.
- @IBOutlet open weak var dataSource: FSPagerViewDataSource?
-
- /// The object that acts as the delegate of the pager view.
- @IBOutlet open weak var delegate: FSPagerViewDelegate?
-
- /// The scroll direction of the pager view. Default is horizontal.
- @objc
- open var scrollDirection: FSPagerView.ScrollDirection = .horizontal {
- didSet {
- self.collectionViewLayout.forceInvalidate()
- }
- }
-
- /// The time interval of automatic sliding. 0 means disabling automatic sliding. Default is 0.
- @IBInspectable
- open var automaticSlidingInterval: CGFloat = 0.0 {
- didSet {
- self.cancelTimer()
- if self.automaticSlidingInterval > 0 {
- self.startTimer()
- }
- }
- }
-
- /// The spacing to use between items in the pager view. Default is 0.
- @IBInspectable
- open var interitemSpacing: CGFloat = 0 {
- didSet {
- self.collectionViewLayout.forceInvalidate()
- }
- }
-
- /// The item size of the pager view. When the value of this property is FSPagerView.automaticSize, the items fill the entire visible area of the pager view. Default is FSPagerView.automaticSize.
- @IBInspectable
- open var itemSize: CGSize = automaticSize {
- didSet {
- self.collectionViewLayout.forceInvalidate()
- }
- }
-
- /// A Boolean value indicates that whether the pager view has infinite items. Default is false.
- @IBInspectable
- open var isInfinite: Bool = false {
- didSet {
- self.collectionViewLayout.needsReprepare = true
- self.collectionView.reloadData()
- }
- }
-
- /// An unsigned integer value that determines the deceleration distance of the pager view, which indicates the number of passing items during the deceleration. When the value of this property is FSPagerView.automaticDistance, the actual 'distance' is automatically calculated according to the scrolling speed of the pager view. Default is 1.
- @IBInspectable
- open var decelerationDistance: UInt = 1
-
- /// A Boolean value that determines whether scrolling is enabled.
- @IBInspectable
- open var isScrollEnabled: Bool {
- set { self.collectionView.isScrollEnabled = newValue }
- get { return self.collectionView.isScrollEnabled }
- }
-
- /// A Boolean value that controls whether the pager view bounces past the edge of content and back again.
- @IBInspectable
- open var bounces: Bool {
- set { self.collectionView.bounces = newValue }
- get { return self.collectionView.bounces }
- }
-
- /// A Boolean value that determines whether bouncing always occurs when horizontal scrolling reaches the end of the content view.
- @IBInspectable
- open var alwaysBounceHorizontal: Bool {
- set { self.collectionView.alwaysBounceHorizontal = newValue }
- get { return self.collectionView.alwaysBounceHorizontal }
- }
-
- /// A Boolean value that determines whether bouncing always occurs when vertical scrolling reaches the end of the content view.
- @IBInspectable
- open var alwaysBounceVertical: Bool {
- set { self.collectionView.alwaysBounceVertical = newValue }
- get { return self.collectionView.alwaysBounceVertical }
- }
-
- /// A Boolean value that controls whether the infinite loop is removed if there is only one item. Default is false.
- @IBInspectable
- open var removesInfiniteLoopForSingleItem: Bool = false {
- didSet {
- self.reloadData()
- }
- }
-
- /// The background view of the pager view.
- @IBInspectable
- open var backgroundView: UIView? {
- didSet {
- if let backgroundView = self.backgroundView {
- if backgroundView.superview != nil {
- backgroundView.removeFromSuperview()
- }
- self.insertSubview(backgroundView, at: 0)
- self.setNeedsLayout()
- }
- }
- }
-
- /// The transformer of the pager view.
- @objc
- open var transformer: FSPagerViewTransformer? {
- didSet {
- self.transformer?.pagerView = self
- self.collectionViewLayout.forceInvalidate()
- }
- }
-
- // MARK: - Public readonly-properties
-
- /// Returns whether the user has touched the content to initiate scrolling.
- @objc
- open var isTracking: Bool {
- return self.collectionView.isTracking
- }
-
- /// The percentage of x position at which the origin of the content view is offset from the origin of the pagerView view.
- @objc
- open var scrollOffset: CGFloat {
- let contentOffset = max(self.collectionView.contentOffset.x, self.collectionView.contentOffset.y)
- let scrollOffset = Double(contentOffset/self.collectionViewLayout.itemSpacing)
- return fmod(CGFloat(scrollOffset), CGFloat(self.numberOfItems))
- }
-
- /// The underlying gesture recognizer for pan gestures.
- @objc
- open var panGestureRecognizer: UIPanGestureRecognizer {
- return self.collectionView.panGestureRecognizer
- }
-
- @objc open fileprivate(set) dynamic var currentIndex: Int = 0
-
- // MARK: - Private properties
-
- internal weak var collectionViewLayout: FSPagerViewLayout!
- internal weak var collectionView: FSPagerCollectionView!
- internal weak var contentView: UIView!
- internal var timer: Timer?
- internal var numberOfItems: Int = 0
- internal var numberOfSections: Int = 0
-
- fileprivate var dequeingSection = 0
- fileprivate var centermostIndexPath: IndexPath {
- guard self.numberOfItems > 0, self.collectionView.contentSize != .zero else {
- return IndexPath(item: 0, section: 0)
- }
- let sortedIndexPaths = self.collectionView.indexPathsForVisibleItems.sorted { (l, r) -> Bool in
- let leftFrame = self.collectionViewLayout.frame(for: l)
- let rightFrame = self.collectionViewLayout.frame(for: r)
- var leftCenter: CGFloat,rightCenter: CGFloat,ruler: CGFloat
- switch self.scrollDirection {
- case .horizontal:
- leftCenter = leftFrame.midX
- rightCenter = rightFrame.midX
- ruler = self.collectionView.bounds.midX
- case .vertical:
- leftCenter = leftFrame.midY
- rightCenter = rightFrame.midY
- ruler = self.collectionView.bounds.midY
- }
- return abs(ruler-leftCenter) < abs(ruler-rightCenter)
- }
- let indexPath = sortedIndexPaths.first
- if let indexPath = indexPath {
- return indexPath
- }
- return IndexPath(item: 0, section: 0)
- }
- fileprivate var isPossiblyRotating: Bool {
- guard let animationKeys = self.contentView.layer.animationKeys() else {
- return false
- }
- let rotationAnimationKeys = ["position", "bounds.origin", "bounds.size"]
- return animationKeys.contains(where: { rotationAnimationKeys.contains($0) })
- }
- fileprivate var possibleTargetingIndexPath: IndexPath?
-
- // MARK: - Overriden functions
-
- public override init(frame: CGRect) {
- super.init(frame: frame)
- self.commonInit()
- }
-
- public required init?(coder aDecoder: NSCoder) {
- super.init(coder: aDecoder)
- self.commonInit()
- }
-
- open override func layoutSubviews() {
- super.layoutSubviews()
- self.backgroundView?.frame = self.bounds
- self.contentView.frame = self.bounds
- self.collectionView.frame = self.contentView.bounds
- }
-
- open override func willMove(toWindow newWindow: UIWindow?) {
- super.willMove(toWindow: newWindow)
- if newWindow != nil {
- self.startTimer()
- } else {
- self.cancelTimer()
- }
- }
-
- #if TARGET_INTERFACE_BUILDER
-
- open override func prepareForInterfaceBuilder() {
- super.prepareForInterfaceBuilder()
- self.contentView.layer.borderWidth = 1
- self.contentView.layer.cornerRadius = 5
- self.contentView.layer.masksToBounds = true
- self.contentView.frame = self.bounds
- let label = UILabel(frame: self.contentView.bounds)
- label.textAlignment = .center
- label.font = UIFont.boldSystemFont(ofSize: 25)
- label.text = "FSPagerView"
- self.contentView.addSubview(label)
- }
-
- #endif
- deinit {
- self.collectionView.dataSource = nil
- self.collectionView.delegate = nil
- }
- // MARK: - UICollectionViewDataSource
-
- public func numberOfSections(in collectionView: UICollectionView) -> Int {
- guard let dataSource = self.dataSource else {
- return 1
- }
- self.numberOfItems = dataSource.numberOfItems(in: self)
- guard self.numberOfItems > 0 else {
- return 0;
- }
- self.numberOfSections = self.isInfinite && (self.numberOfItems > 1 || !self.removesInfiniteLoopForSingleItem) ? Int(Int16.max)/self.numberOfItems : 1
- return self.numberOfSections
- }
-
- public func collectionView(_ collectionView: UICollectionView, numberOfItemsInSection section: Int) -> Int {
- return self.numberOfItems
- }
-
- public func collectionView(_ collectionView: UICollectionView, cellForItemAt indexPath: IndexPath) -> UICollectionViewCell {
- let index = indexPath.item
- self.dequeingSection = indexPath.section
- let cell = self.dataSource!.pagerView(self, cellForItemAt: index)
- return cell
- }
-
- // MARK: - UICollectionViewDelegate
-
- public func collectionView(_ collectionView: UICollectionView, shouldHighlightItemAt indexPath: IndexPath) -> Bool {
- guard let function = self.delegate?.pagerView(_:shouldHighlightItemAt:) else {
- return true
- }
- let index = indexPath.item % self.numberOfItems
- return function(self,index)
- }
-
- public func collectionView(_ collectionView: UICollectionView, didHighlightItemAt indexPath: IndexPath) {
- guard let function = self.delegate?.pagerView(_:didHighlightItemAt:) else {
- return
- }
- let index = indexPath.item % self.numberOfItems
- function(self,index)
- }
-
- public func collectionView(_ collectionView: UICollectionView, shouldSelectItemAt indexPath: IndexPath) -> Bool {
- guard let function = self.delegate?.pagerView(_:shouldSelectItemAt:) else {
- return true
- }
- let index = indexPath.item % self.numberOfItems
- return function(self,index)
- }
-
- public func collectionView(_ collectionView: UICollectionView, didSelectItemAt indexPath: IndexPath) {
- guard let function = self.delegate?.pagerView(_:didSelectItemAt:) else {
- return
- }
- self.possibleTargetingIndexPath = indexPath
- defer {
- self.possibleTargetingIndexPath = nil
- }
- let index = indexPath.item % self.numberOfItems
- function(self,index)
- }
-
- public func collectionView(_ collectionView: UICollectionView, willDisplay cell: UICollectionViewCell, forItemAt indexPath: IndexPath) {
- guard let function = self.delegate?.pagerView(_:willDisplay:forItemAt:) else {
- return
- }
- let index = indexPath.item % self.numberOfItems
- function(self,cell as! FSPagerViewCell,index)
- }
-
- public func collectionView(_ collectionView: UICollectionView, didEndDisplaying cell: UICollectionViewCell, forItemAt indexPath: IndexPath) {
- guard let function = self.delegate?.pagerView(_:didEndDisplaying:forItemAt:) else {
- return
- }
- let index = indexPath.item % self.numberOfItems
- function(self,cell as! FSPagerViewCell,index)
- }
-
- public func scrollViewDidScroll(_ scrollView: UIScrollView) {
- if !self.isPossiblyRotating && self.numberOfItems > 0 {
- // In case someone is using KVO
- let currentIndex = lround(Double(self.scrollOffset)) % self.numberOfItems
- if (currentIndex != self.currentIndex) {
- self.currentIndex = currentIndex
- }
- }
- guard let function = self.delegate?.pagerViewDidScroll else {
- return
- }
- function(self)
- }
-
- public func scrollViewWillBeginDragging(_ scrollView: UIScrollView) {
- if let function = self.delegate?.pagerViewWillBeginDragging(_:) {
- function(self)
- }
- if self.automaticSlidingInterval > 0 {
- self.cancelTimer()
- }
- }
-
- public func scrollViewWillEndDragging(_ scrollView: UIScrollView, withVelocity velocity: CGPoint, targetContentOffset: UnsafeMutablePointer<CGPoint>) {
- if let function = self.delegate?.pagerViewWillEndDragging(_:targetIndex:) {
- let contentOffset = self.scrollDirection == .horizontal ? targetContentOffset.pointee.x : targetContentOffset.pointee.y
- let targetItem = lround(Double(contentOffset/self.collectionViewLayout.itemSpacing))
- function(self, targetItem % self.numberOfItems)
- }
- if self.automaticSlidingInterval > 0 {
- self.startTimer()
- }
- }
-
- public func scrollViewDidEndDecelerating(_ scrollView: UIScrollView) {
- if let function = self.delegate?.pagerViewDidEndDecelerating {
- function(self)
- }
- }
-
- public func scrollViewDidEndScrollingAnimation(_ scrollView: UIScrollView) {
- if let function = self.delegate?.pagerViewDidEndScrollAnimation {
- function(self)
- }
- }
-
- // MARK: - Public functions
-
- /// Register a class for use in creating new pager view cells.
- ///
- /// - Parameters:
- /// - cellClass: The class of a cell that you want to use in the pager view.
- /// - identifier: The reuse identifier to associate with the specified class. This parameter must not be nil and must not be an empty string.
- @objc(registerClass:forCellWithReuseIdentifier:)
- open func register(_ cellClass: Swift.AnyClass?, forCellWithReuseIdentifier identifier: String) {
- self.collectionView.register(cellClass, forCellWithReuseIdentifier: identifier)
- }
-
- /// Register a nib file for use in creating new pager view cells.
- ///
- /// - Parameters:
- /// - nib: The nib object containing the cell object. The nib file must contain only one top-level object and that object must be of the type FSPagerViewCell.
- /// - identifier: The reuse identifier to associate with the specified nib file. This parameter must not be nil and must not be an empty string.
- @objc(registerNib:forCellWithReuseIdentifier:)
- open func register(_ nib: UINib?, forCellWithReuseIdentifier identifier: String) {
- self.collectionView.register(nib, forCellWithReuseIdentifier: identifier)
- }
-
- /// Returns a reusable cell object located by its identifier
- ///
- /// - Parameters:
- /// - identifier: The reuse identifier for the specified cell. This parameter must not be nil.
- /// - index: The index specifying the location of the cell.
- /// - Returns: A valid FSPagerViewCell object.
- @objc(dequeueReusableCellWithReuseIdentifier:atIndex:)
- open func dequeueReusableCell(withReuseIdentifier identifier: String, at index: Int) -> FSPagerViewCell {
- let indexPath = IndexPath(item: index, section: self.dequeingSection)
- let cell = self.collectionView.dequeueReusableCell(withReuseIdentifier: identifier, for: indexPath)
- guard cell.isKind(of: FSPagerViewCell.self) else {
- fatalError("Cell class must be subclass of FSPagerViewCell")
- }
- return cell as! FSPagerViewCell
- }
-
- /// Reloads all of the data for the collection view.
- @objc(reloadData)
- open func reloadData() {
- self.collectionViewLayout.needsReprepare = true;
- self.collectionView.reloadData()
- }
-
- /// Selects the item at the specified index and optionally scrolls it into view.
- ///
- /// - Parameters:
- /// - index: The index path of the item to select.
- /// - animated: Specify true to animate the change in the selection or false to make the change without animating it.
- @objc(selectItemAtIndex:animated:)
- open func selectItem(at index: Int, animated: Bool) {
- let indexPath = self.nearbyIndexPath(for: index)
- let scrollPosition: UICollectionView.ScrollPosition = self.scrollDirection == .horizontal ? .centeredHorizontally : .centeredVertically
- self.collectionView.selectItem(at: indexPath, animated: animated, scrollPosition: scrollPosition)
- }
-
- /// Deselects the item at the specified index.
- ///
- /// - Parameters:
- /// - index: The index of the item to deselect.
- /// - animated: Specify true to animate the change in the selection or false to make the change without animating it.
- @objc(deselectItemAtIndex:animated:)
- open func deselectItem(at index: Int, animated: Bool) {
- let indexPath = self.nearbyIndexPath(for: index)
- self.collectionView.deselectItem(at: indexPath, animated: animated)
- }
-
- /// Scrolls the pager view contents until the specified item is visible.
- ///
- /// - Parameters:
- /// - index: The index of the item to scroll into view.
- /// - animated: Specify true to animate the scrolling behavior or false to adjust the pager view’s visible content immediately.
- @objc(scrollToItemAtIndex:animated:)
- open func scrollToItem(at index: Int, animated: Bool) {
- guard index < self.numberOfItems else {
- fatalError("index \(index) is out of range [0...\(self.numberOfItems-1)]")
- }
- let indexPath = { () -> IndexPath in
- if let indexPath = self.possibleTargetingIndexPath, indexPath.item == index {
- defer {
- self.possibleTargetingIndexPath = nil
- }
- return indexPath
- }
- return self.numberOfSections > 1 ? self.nearbyIndexPath(for: index) : IndexPath(item: index, section: 0)
- }()
- let contentOffset = self.collectionViewLayout.contentOffset(for: indexPath)
- self.collectionView.setContentOffset(contentOffset, animated: animated)
- }
-
- /// Returns the index of the specified cell.
- ///
- /// - Parameter cell: The cell object whose index you want.
- /// - Returns: The index of the cell or NSNotFound if the specified cell is not in the pager view.
- @objc(indexForCell:)
- open func index(for cell: FSPagerViewCell) -> Int {
- guard let indexPath = self.collectionView.indexPath(for: cell) else {
- return NSNotFound
- }
- return indexPath.item
- }
-
- /// Returns the visible cell at the specified index.
- ///
- /// - Parameter index: The index that specifies the position of the cell.
- /// - Returns: The cell object at the corresponding position or nil if the cell is not visible or index is out of range.
- @objc(cellForItemAtIndex:)
- open func cellForItem(at index: Int) -> FSPagerViewCell? {
- let indexPath = self.nearbyIndexPath(for: index)
- return self.collectionView.cellForItem(at: indexPath) as? FSPagerViewCell
- }
-
- // MARK: - Private functions
-
- fileprivate func commonInit() {
-
- // Content View
- let contentView = UIView(frame:CGRect.zero)
- contentView.backgroundColor = UIColor.clear
- self.addSubview(contentView)
- self.contentView = contentView
-
- // UICollectionView
- let collectionViewLayout = FSPagerViewLayout()
- let collectionView = FSPagerCollectionView(frame: CGRect.zero, collectionViewLayout: collectionViewLayout)
- collectionView.dataSource = self
- collectionView.delegate = self
- collectionView.backgroundColor = UIColor.clear
- self.contentView.addSubview(collectionView)
- self.collectionView = collectionView
- self.collectionViewLayout = collectionViewLayout
-
- }
-
- fileprivate func startTimer() {
- guard self.automaticSlidingInterval > 0 && self.timer == nil else {
- return
- }
- self.timer = Timer.scheduledTimer(timeInterval: TimeInterval(self.automaticSlidingInterval), target: self, selector: #selector(self.flipNext(sender:)), userInfo: nil, repeats: true)
- RunLoop.current.add(self.timer!, forMode: .common)
- }
-
- @objc
- fileprivate func flipNext(sender: Timer?) {
- guard let _ = self.superview, let _ = self.window, self.numberOfItems > 0, !self.isTracking else {
- return
- }
- let contentOffset: CGPoint = {
- let indexPath = self.centermostIndexPath
- let section = self.numberOfSections > 1 ? (indexPath.section+(indexPath.item+1)/self.numberOfItems) : 0
- let item = (indexPath.item+1) % self.numberOfItems
- return self.collectionViewLayout.contentOffset(for: IndexPath(item: item, section: section))
- }()
- self.collectionView.setContentOffset(contentOffset, animated: true)
- }
-
- fileprivate func cancelTimer() {
- guard self.timer != nil else {
- return
- }
- self.timer!.invalidate()
- self.timer = nil
- }
-
- fileprivate func nearbyIndexPath(for index: Int) -> IndexPath {
- // Is there a better algorithm?
- let currentIndex = self.currentIndex
- let currentSection = self.centermostIndexPath.section
- if abs(currentIndex-index) <= self.numberOfItems/2 {
- return IndexPath(item: index, section: currentSection)
- } else if (index-currentIndex >= 0) {
- return IndexPath(item: index, section: currentSection-1)
- } else {
- return IndexPath(item: index, section: currentSection+1)
- }
- }
-
- }
- extension FSPagerView {
-
- /// Constants indicating the direction of scrolling for the pager view.
- @objc
- public enum ScrollDirection: Int {
- /// The pager view scrolls content horizontally
- case horizontal
- /// The pager view scrolls content vertically
- case vertical
- }
-
- /// Requests that FSPagerView use the default value for a given distance.
- public static let automaticDistance: UInt = 0
-
- /// Requests that FSPagerView use the default value for a given size.
- public static let automaticSize: CGSize = .zero
-
- }
|