UIScrollView, reaching the bottom of the scroll view
Asked Answered
G

7

45

I know the Apple documentation has the following delegate method:

- (void)scrollViewDidEndDecelerating:(UIScrollView *)scrollView;      // called when scroll view grinds to a halt

However, it doesn't necessarily mean you are at the bottom. Cause if you use your finger, scroll a bit, then it decelerates, but you are not actually at the bottom of your scroll view, then it still gets called. I basically want an arrow to show that there is more data in my scroll view, and then disappear when you are at the bottom (like when it bounces). Thanks.

Gauss answered 2/6, 2011 at 17:23 Comment(0)
D
106

I think what you might be able to do is to check that your contentOffset point is at the bottom of contentSize. So you could probably do something like:

- (void)scrollViewDidEndDecelerating:(UIScrollView *)scrollView {
    float bottomEdge = scrollView.contentOffset.y + scrollView.frame.size.height;
    if (bottomEdge >= scrollView.contentSize.height) {
        // we are at the end
    }
}

You'll likely also need a negative case there to show your indicator when the user scrolls back up. You might also want to add some padding to that so, for example, you could hide the indicator when the user is near the bottom, but not exactly at the bottom.

Devault answered 2/6, 2011 at 17:38 Comment(3)
Best solution but I would put it in - (void)scrollViewDidScroll:(UIScrollView *)scrollView{} insteadThrall
If you are willing to put your logic inside scrollViewDidScroll, you dave to design some guarding flag which will prevent your logic from firing million times. This is because you have >= there and it will successfully get inside the if clause million of times, when your table view bounces. The end of deceleration is much better event to observe since it gets called after bouncing had finishedColonist
scrollViewDidEndDecelerating not triggered if i call : cv.scrollToBottom()Struve
S
26

So if you want it in swift, here you go:

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
    }
}
Spice answered 1/10, 2014 at 13:23 Comment(0)
T
14

I Think @bensnider answer is correct, But not exart. Because of these two reasons

1. - (void)scrollViewDidScroll:(UIScrollView *)scrollView{}

This method will call continuously if we check for if (bottomEdge >= scrollView.contentSize.height)

2 . In this if we go for == check also this condition will valid for two times.

  • (i) when we will scroll up when the end of the scroll view touches the bottom edge
  • (ii) When the scrollview bounces back to retain it's own position

I feel this is more accurate.

Very few cases this codition is valid for two times also. But User will not come across this.

- (void)scrollViewDidScroll:(UIScrollView *)scrollView
 {
    if (scrollView.contentOffset.y == roundf(scrollView.contentSize.height-scrollView.frame.size.height)) {
    NSLog(@"we are at the endddd");

   //Call your function here...

    }
}
Tecla answered 28/6, 2013 at 10:17 Comment(1)
Yes, this should've been the accepted answer. Also, I think the contentScale factor should be used in the equationRrhoea
I
10

The accepted answer works only if the bottom contentInset value is non-negative. A slight evolution would consider the bottom of the contentInset regardless of it's sign:

CGFloat bottomInset = scrollView.contentInset.bottom;
CGFloat bottomEdge = scrollView.contentOffset.y + scrollView.frame.size.height - bottomInset;
if (bottomEdge == scrollView.contentSize.height) {
    // Scroll view is scrolled to bottom
}
Irv answered 3/8, 2014 at 19:28 Comment(0)
F
5

Actually, rather than just putting @bensnider's code in scrollViewDidScroll, this code (written in Swift 3) would be better performance-wise:

func scrollViewDidEndDragging(_ scrollView: UIScrollView,
                              willDecelerate decelerate: Bool) {
    if !decelerate {
        checkHasScrolledToBottom()
    }
}

func scrollViewDidEndDecelerating(_ scrollView: UIScrollView) {
    checkHasScrolledToBottom()
}

func checkHasScrolledToBottom() {
    let bottomEdge = scrollView.contentOffset.y + scrollView.frame.size.height
    if bottomEdge >= scrollView.contentSize.height {
        // we are at the end
    }
}
Faunie answered 3/12, 2016 at 0:8 Comment(0)
W
2

It work in Swift 3:

override func scrollViewDidScroll(_ scrollView: UIScrollView) {
    if scrollView.contentOffset.y == (scrollView.contentSize.height - scrollView.frame.size.height) {
        loadMore()
    }
}
Woermer answered 12/1, 2017 at 6:43 Comment(0)
F
0

See what items are currently displayed in the UIView, using something like indexPathsForVisibleRows and if your model has more items than displayed, put an arrow at the bottom.

Foliole answered 2/6, 2011 at 17:27 Comment(0)

© 2022 - 2024 — McMap. All rights reserved.