iPhone - знание того, достиг ли UIScrollView верха или низа

Ответы:

199

Реализуйте UIScrollViewDelegateв своем классе, а затем добавьте это:

-(void)scrollViewDidScroll: (UIScrollView*)scrollView
{
    float scrollViewHeight = scrollView.frame.size.height;
    float scrollContentSizeHeight = scrollView.contentSize.height;
    float scrollOffset = scrollView.contentOffset.y;

    if (scrollOffset == 0)
    {
        // then we are at the top
    }
    else if (scrollOffset + scrollViewHeight == scrollContentSizeHeight)
    {
        // then we are at the end
    }
}

Надеюсь, это то, что вам нужно! Иначе попробуйте, добавив дополнительные условия в приведенный выше код и NSLog значение scrollOffset.

Люк
источник
4
Если вы находитесь в a navigationController, вы можете сделать что-то еще вроде этого: scrollOffset <= (0 - self.navigationController.navigationBar.frame.size.height - 20)и(scrollOffset + scrollHeight) >= scrollContentSizeHeight
Уэйн
Проблема в том, что отскок ... все вызывается дважды, когда достигает вершины или основания. Есть ли решение, кроме отключения отскока просмотра таблицы или коллекции?
denikov
@denikov вы нашли какое-нибудь решение?
Priyal 07
83

Ну, contentInsetsтакже участвуют, когда вы пытаетесь определить, находится ли scrollView вверху или внизу. Вам также могут быть интересны случаи, когда ваш scrollView находится выше верха и ниже низа. Вот код, который я использую для поиска верхних и нижних позиций:

Swift:

extension UIScrollView {

    var isAtTop: Bool {
        return contentOffset.y <= verticalOffsetForTop
    }

    var isAtBottom: Bool {
        return contentOffset.y >= verticalOffsetForBottom
    }

    var verticalOffsetForTop: CGFloat {
        let topInset = contentInset.top
        return -topInset
    }

    var verticalOffsetForBottom: CGFloat {
        let scrollViewHeight = bounds.height
        let scrollContentSizeHeight = contentSize.height
        let bottomInset = contentInset.bottom
        let scrollViewBottomOffset = scrollContentSizeHeight + bottomInset - scrollViewHeight
        return scrollViewBottomOffset
    }

}

Цель-C:

@implementation UIScrollView (Additions)

- (BOOL)isAtTop {
    return (self.contentOffset.y <= [self verticalOffsetForTop]);
}

- (BOOL)isAtBottom {
    return (self.contentOffset.y >= [self verticalOffsetForBottom]);
}

- (CGFloat)verticalOffsetForTop {
    CGFloat topInset = self.contentInset.top;
    return -topInset;
}

- (CGFloat)verticalOffsetForBottom {
    CGFloat scrollViewHeight = self.bounds.size.height;
    CGFloat scrollContentSizeHeight = self.contentSize.height;
    CGFloat bottomInset = self.contentInset.bottom;
    CGFloat scrollViewBottomOffset = scrollContentSizeHeight + bottomInset - scrollViewHeight;
    return scrollViewBottomOffset;
}


@end
Александр Дворников
источник
Это хороший ответ, реализация scrollviewdidscroll плохо влияет на производительность
Василий
3
Как мне его вызвать, если вы не используете scrollViewDidScroll?
Dave G
2
Из-за капризов математики с плавающей запятой я получил "verticalOffsetForBottom" 1879.05xxx, а мой contentOffset.y был 1879, когда внизу в Swift 3. Так что я изменился return scrollViewBottomOffsetнаreturn scrollViewBottomOffset.rounded()
chadbag
1
Начиная с iOS 11, это safeAreaInsetsтоже учитывается. scrollView.contentOffset.y + scrollView.bounds.height - scrollView.safeAreaInsets.bottom
InkGolem
36

Если вам нужен быстрый код:

override func scrollViewDidScroll(scrollView: UIScrollView) {

    if (scrollView.contentOffset.y >= (scrollView.contentSize.height - scrollView.frame.size.height)) {
        //reach bottom
    }

    if (scrollView.contentOffset.y <= 0){
        //reach top
    }

    if (scrollView.contentOffset.y > 0 && scrollView.contentOffset.y < (scrollView.contentSize.height - scrollView.frame.size.height)){
        //not top and not bottom
    }
}
Итбрян
источник
12

Простое и понятное расширение:

import UIKit

extension UIScrollView {

    var scrolledToTop: Bool {
        let topEdge = 0 - contentInset.top
        return contentOffset.y <= topEdge
    }

    var scrolledToBottom: Bool {
        let bottomEdge = contentSize.height + contentInset.bottom - bounds.height
        return contentOffset.y >= bottomEdge
    }

}

На GitHub:

https://github.com/salutis/swift-scroll-view-edges

Рудольф Адамкович
источник
4

Я точно понял, как это сделать:

CGFloat maxPosition = scrollView.contentInset.top + scrollView.contentSize.height + scrollView.contentInset.bottom - scrollView.bounds.size.height;
CGFloat currentPosition = scrollView.contentOffset.y + self.topLayoutGuide.length;

if (currentPosition == maxPosition) {
  // you're at the bottom!
}
JPN
источник
0

Сделайте это так (для Swift 3):

class MyClass: UIScrollViewDelegate {

   func scrollViewDidScroll(_ scrollView: UIScrollView) {

        if (scrollView.contentOffset.y >= (scrollView.contentSize.height - scrollView.frame.size.height)) {
           //scrolled to top, do smth
        }


    }
}
Раду Гитеску
источник
Добро пожаловать в SO! При размещении кода в своем ответе полезно объяснить, как ваш код решает проблему OP :)
Джоэл