You signed in with another tab or window. Reload to refresh your session.You signed out in another tab or window. Reload to refresh your session.You switched accounts on another tab or window. Reload to refresh your session.Dismiss alert
There are at least three ways of paginating content in iOS. Namely, via UIScrollView, UIPageViewController, and UICollectionView.
For simplicity, I'll consider only horizontal pagination from now on in this post.
UIPageViewController paginates its contents by setting its transitionStyle property to .scroll. UIScrollView and UICollectionView paginate their content by setting their isPagingEnabled property to true.
It's worth noting that all these solutions are essentially built upon UIScrollView. UIPageViewController uses a special UIScrollView subclass (private API I think) called _UIQueuingScrollView. UICollectionView is a UIScrollView subclass.
Limitations
1. Page Size is Fixed
A common trait among the previous solutions is the fixed page size. That is, the amount by which content is paged is always equal to the scrollView frame. If you want to have a page size different from the visible "frame", you have to seek workarounds; e.g. this clever solutions [1, 2] by Khanlou, or my solution using a UICollectionViewhere.
2. Uneven Page Size
Another limitation is if you want more than a page size in a single flow. For example, when your flow can be considered a series of pairs, where each pair of pages are separated by a constant spacing, while each item of each pair is separated by a different amount of spacing. I think this is impossible to work around with the above solutions.
One can think of having more than one level of pagination to fix this. For example:
A UIPageViewController for the pairs, while each pair is a UIpageViewController itself.
A UICollectionView for the pairs, while each pair is a UICollectionView itself.
Similar thing with raw UIScrollView.
However, these solutions have problems.
For the nested UIPageViewController it's so easy to swipe an entire pair while not noticing. This is because the outer and the inner UIPageViewControllers have contentSize greater than the visible frame (since UIPageViewController always loads 3 pages if possible (left, center, right)). So, any pan gesture can both affect any of them.
Similar thing can happen too with nested UICollectionViews. However, it can be worked around by disabling prefetching on the outer UICollectionView. This way, the outer UICollectionView only loads one cell (pair cell), while the pair cell can load its full content; so the pan gesture would work fine with inner UICollectionView as expected. However, on fast scrolling, this seems to not work; and pairs are again skipped.
scrollViewWillEndDragging(_:withVelocity:targetContentOffset:) has something to say
UIScrollViewDelegate has this interesting method that is called when the user ends dragging. It reports the velocity by which the user did their swipe, and (which is our focus) passes the expected content offset at which the scrollView would stop! So clever! And there's more to it. It's possible to change that expected offset so the scrollView smoothly stops at a desired position!
So, knowing this, we can "snap" the decelerating scrollView to a position of our choice, so there is a chance to solve our uneven page size problem.
Idea
(Using a UICollectionView of pairs, where each cell is a pair of UIView subclass)
Given an item width equal to the visible frame. Each pair of items are separated by 50 pts of space on each side. We can:
We can partition of content into a series of evenly sized pairs (including spacing). That is, each pair width = item width * 2 + spacing (25 pts at each side).
When scrollViewWillEndDragging is called, we can inspect the targetContentOffset and see at what index of pairs that offset value should correspond. Such index can be achieved by dividing (integer division) the value of the targetContentOffset by the pair width.
Note that targetContentOffset always points to the leftmost of the screen. This causes a bias to the left side of scrolling, so that way, integer division would be inclined to get lesser indices; 1 is more likely to come than 2, 2 is more likely to come than 3, and so on... One way to overcome this is to offset the targetContentOffset a little to balance this bias; making it points to the middle of the screen rather than it leftmost edge. To achieve this, just add half of the visible frame width to the targetContentOffset before integer division.
Now we have a correct index of a pair. We only have to decide which part of the pair we want to snap to. So, the sizes of each part of the pair should be known to us. That way we can decide which part is close to the adjusted targetContentOffset calculated above (adjusted to the middle of visible frame). And that's it. Now finally alter the value targetContentOffset to achieve our desired effect, e.g.: targetContentOffset.pointee.x = rightItemX.
Notes:
We don't use isPagingEnabled here; we use normal scrolling. The default decelaration rate may be too slow; so setting the UICollectionView's decelerationRate to .fast should do it.
We have to inset the UICollectionView by half the spacing at each side to achieve contentSize multiple to that of pair width.
Large swipes may cause jumping over a page. This avoidable by clamping the amount by which the targetContentOffset changes. It may also appear as a feature not a defect. 😄
Very weak swipes that are not enough to make a page change caused a choppy animation. I mitigated this by detecting it (non-zero velcoity, same targetContentOffset) then setting the content offset with an animation.
Conclusion
Here is a demo in Swift that implements what's above.
Although the solution presented here may not be perfect, it's the best I could come up with. It's a tricky problem that I hadn't find a complete solution for it so far.
For questions, and suggestions please contact me via Twitter, or submit a pull request on the linked Github demo.
Thanks for reading!
The text was updated successfully, but these errors were encountered:
(Originally published 2019-01-19)
Introduction
There are at least three ways of paginating content in iOS. Namely, via
UIScrollView
,UIPageViewController
, andUICollectionView
.For simplicity, I'll consider only horizontal pagination from now on in this post.
UIPageViewController
paginates its contents by setting itstransitionStyle
property to.scroll
.UIScrollView
andUICollectionView
paginate their content by setting theirisPagingEnabled
property totrue
.It's worth noting that all these solutions are essentially built upon
UIScrollView
.UIPageViewController
uses a specialUIScrollView
subclass (private API I think) called_UIQueuingScrollView
.UICollectionView
is aUIScrollView
subclass.Limitations
1. Page Size is Fixed
A common trait among the previous solutions is the fixed page size. That is, the amount by which content is paged is always equal to the scrollView frame. If you want to have a page size different from the visible "frame", you have to seek workarounds; e.g. this clever solutions [1, 2] by Khanlou, or my solution using a
UICollectionView
here.2. Uneven Page Size
Another limitation is if you want more than a page size in a single flow. For example, when your flow can be considered a series of pairs, where each pair of pages are separated by a constant spacing, while each item of each pair is separated by a different amount of spacing. I think this is impossible to work around with the above solutions.
One can think of having more than one level of pagination to fix this. For example:
UIPageViewController
for the pairs, while each pair is aUIpageViewController
itself.UICollectionView
for the pairs, while each pair is aUICollectionView
itself.UIScrollView
.However, these solutions have problems.
For the nested
UIPageViewController
it's so easy to swipe an entire pair while not noticing. This is because the outer and the innerUIPageViewController
s have contentSize greater than the visible frame (sinceUIPageViewController
always loads 3 pages if possible (left, center, right)). So, any pan gesture can both affect any of them.Similar thing can happen too with nested
UICollectionView
s. However, it can be worked around by disabling prefetching on the outerUICollectionView
. This way, the outerUICollectionView
only loads one cell (pair cell), while the pair cell can load its full content; so the pan gesture would work fine with innerUICollectionView
as expected. However, on fast scrolling, this seems to not work; and pairs are again skipped.scrollViewWillEndDragging(_:withVelocity:targetContentOffset:)
has something to sayUIScrollViewDelegate
has this interesting method that is called when the user ends dragging. It reports the velocity by which the user did their swipe, and (which is our focus) passes the expected content offset at which the scrollView would stop! So clever! And there's more to it. It's possible to change that expected offset so the scrollView smoothly stops at a desired position!So, knowing this, we can "snap" the decelerating scrollView to a position of our choice, so there is a chance to solve our uneven page size problem.
Idea
(Using a
UICollectionView
of pairs, where each cell is a pair ofUIView
subclass)Given an item width equal to the visible frame. Each pair of items are separated by 50 pts of space on each side. We can:
We can partition of content into a series of evenly sized pairs (including spacing). That is, each pair width = item width * 2 + spacing (25 pts at each side).
When
scrollViewWillEndDragging
is called, we can inspect thetargetContentOffset
and see at what index of pairs that offset value should correspond. Such index can be achieved by dividing (integer division) the value of thetargetContentOffset
by the pair width.Note that
targetContentOffset
always points to the leftmost of the screen. This causes a bias to the left side of scrolling, so that way, integer division would be inclined to get lesser indices; 1 is more likely to come than 2, 2 is more likely to come than 3, and so on... One way to overcome this is to offset thetargetContentOffset
a little to balance this bias; making it points to the middle of the screen rather than it leftmost edge. To achieve this, just add half of the visible frame width to thetargetContentOffset
before integer division.Now we have a correct index of a pair. We only have to decide which part of the pair we want to snap to. So, the sizes of each part of the pair should be known to us. That way we can decide which part is close to the adjusted
targetContentOffset
calculated above (adjusted to the middle of visible frame). And that's it. Now finally alter the valuetargetContentOffset
to achieve our desired effect, e.g.:targetContentOffset.pointee.x = rightItemX
.Notes:
We don't use
isPagingEnabled
here; we use normal scrolling. The default decelaration rate may be too slow; so setting theUICollectionView
'sdecelerationRate
to.fast
should do it.We have to inset the
UICollectionView
by half the spacing at each side to achieve contentSize multiple to that of pair width.Large swipes may cause jumping over a page. This avoidable by clamping the amount by which the
targetContentOffset
changes. It may also appear as a feature not a defect. 😄Very weak swipes that are not enough to make a page change caused a choppy animation. I mitigated this by detecting it (non-zero velcoity, same
targetContentOffset
) then setting the content offset with an animation.Conclusion
Here is a demo in Swift that implements what's above.
Although the solution presented here may not be perfect, it's the best I could come up with. It's a tricky problem that I hadn't find a complete solution for it so far.
For questions, and suggestions please contact me via Twitter, or submit a pull request on the linked Github demo.
Thanks for reading!
The text was updated successfully, but these errors were encountered: