ios – 增加UIScrollView橡胶条带阻力

前端之家收集整理的这篇文章主要介绍了ios – 增加UIScrollView橡胶条带阻力前端之家小编觉得挺不错的,现在分享给大家,也给大家做个参考。
我有一个我想要在我的应用程序中实现的场景,当我开始强制它滚动超过其正常的内容界限时,我想使UIScrollView的行为更加“僵硬”.

我的意思是,当你在滚动视图的顶部或底部,如果你点击并继续拖动,你通常可以得到滚动视图,以保持超越其界限,但它逐渐建立抵抗,直到它在视野的中间通常约一半.当您抬起手指时,它会回到滚动区域的边界.

我想要实现的是,我想让这个“超越”的拖拽效果变得更加沉重,所以,而不是用户拖动滚动视图,并且它的“触底”中途通过滚动视图界限,它完全停止了大约20%左右的滚动边界.

我一直在试验覆盖scrollViewDidScroll:delegate方法中的滚动视图的contentOffset,但这似乎没有起作用,因为在其中重新设置contentOffset似乎削弱了相同方法的进一步委托调用.

我的下一个想法是监视与scrollview关联的UIPanGestureRecognizer,并根据不同的事件尝试确定正确的UIScrollView contentOffset.话虽如此,我认为这可能会开始恶作剧,所以我以为我会在这里提出我之前没有考虑过的其他解决办法,然后再尝试一些可能会混乱的事情.

谢谢!

解决方法

我在 Swift的以下代码中加注.这是用于水平滚动,可以轻松地适应垂直滚动.解决方案根据是否启用分页是不同的.两者都在下面给出.
  1. class ScrollViewDelegate : NSObject,UIScrollViewDelegate
  2. {
  3. let maxOffset: CGFloat // offset of the rightmost content (scrollview contentSize.width - frame.width)
  4. var prevOffset: CGFloat = 0 // prevIoUs offset (after adjusting the value)
  5.  
  6. var totalDistance: CGFloat = 0 // total distance it would have moved (had we not restricted)
  7. let reductionFactor: CGFloat = 0.2 // percent of total distance it will be allowed to move (under restriction)
  8. let scaleFactor: CGFloat = UIScreen.mainScreen().scale // pixels per point,for smooth translation in respective devices
  9.  
  10. init(maxOffset: CGFloat)
  11. {
  12. self.maxOffset = maxOffset // scrollView.contentSize.width - scrollView.frame.size.width
  13. }
  14.  
  15. func scrollViewDidScroll(scrollView: UIScrollView)
  16. {
  17. let flipped = scrollView.contentOffset.x >= maxOffset // dealing with left edge or right edge rubber band
  18. let currentOffset = flipped ? maxOffset - scrollView.contentOffset.x : scrollView.contentOffset.x // for right edge,flip the values as if screen is folded in half towards the left
  19.  
  20. if(currentOffset <= 0) // if dragging/moving beyond the edge
  21. {
  22. if(currentOffset <= prevOffset) // if dragging/moving beyond prevIoUs offset
  23. {
  24. totalDistance += currentOffset - prevOffset // add the "proposed delta" move to total distance
  25. prevOffset = round(scaleFactor * totalDistance * reductionFactor) / scaleFactor // set the prevOffset to fraction of total distance
  26. scrollView.contentOffset.x = flipped ? maxOffset - prevOffset : prevOffset // set the target offset,after negating any flipping
  27. }
  28. else // if dragging/moving is reversed,though still beyond the edge
  29. {
  30. totalDistance = currentOffset / reductionFactor // set totalDistance from offset (reverse of prevOffset calculation above)
  31. prevOffset = currentOffset // set prevOffset
  32. }
  33. }
  34. else // if dragging/moving inside the edge
  35. {
  36. totalDistance = 0 // reset the values
  37. prevOffset = 0
  38. }
  39. }
  40. }

启用分页时,弹回到休息点看起来似乎不能正常工作.橡胶带不是在页面边界处停止,而是以非页面偏移量超过它并停止.如果来自边缘的拉拽快速滑动,即使在抬起手指之后,在反转方向并返回到静止点之前,它也会沿着该方向继续移动.如果你只是暂停或离开,甚至轻轻地摆放到休息点,它似乎工作正常.为了解决这个问题,在下面的代码中,我尝试确定超调的可能性,并在返回时强制停止它,并试图跨越预期的页面边界.

  1. class PageScrollViewDelegate : NSObject,for smooth translation in respective devices
  2.  
  3. var draggingOver: Bool = false // finger dragging is over or not
  4. var overshoot: Bool = false // is there a chance for page to overshoot page boundary while falling back
  5.  
  6. init(maxOffset: CGFloat)
  7. {
  8. self.maxOffset = maxOffset // scrollView.contentSize.width - scrollView.frame.size.width
  9. }
  10.  
  11. func scrollViewWillBeginDragging(scrollView: UIScrollView)
  12. {
  13. draggingOver = false // reset the flags
  14. overshoot = false
  15. }
  16.  
  17. func scrollViewDidEndDragging(scrollView: UIScrollView,willDecelerate decelerate: Bool)
  18. {
  19. draggingOver = true // finger dragging is over
  20. }
  21.  
  22. func scrollViewDidScroll(scrollView: UIScrollView)
  23. {
  24. let flipped = scrollView.contentOffset.x >= 0.5 * maxOffset // dealing with left edge or right edge rubber band
  25. let currentOffset = flipped ? maxOffset - scrollView.contentOffset.x : scrollView.contentOffset.x // for right edge,flip the values as if screen is folded in half towards the left
  26.  
  27. if(currentOffset <= 0) // if dragging/moving beyond the edge
  28. {
  29. if(currentOffset <= prevOffset) // if dragging/moving beyond prevIoUs offset
  30. {
  31. overshoot = draggingOver // is content moving farther away even after dragging is over (caused by fast flick,which can cause overshooting page boundary while falling back)
  32.  
  33. totalDistance += currentOffset - prevOffset // add the "proposed delta" move to total distance
  34. prevOffset = round(scaleFactor * totalDistance * reductionFactor) / scaleFactor // set the prevOffset to fraction of total distance
  35. scrollView.contentOffset.x = flipped ? maxOffset - prevOffset : prevOffset // set the target offset,though still beyond the edge
  36. {
  37. totalDistance = currentOffset / reductionFactor // set totalDistance from offset (reverse of prevOffset calculation above)
  38. prevOffset = currentOffset // set prevOffset
  39. }
  40. }
  41. else // if dragging/moving inside the edge
  42. {
  43. if(overshoot) // if this movement is a result of overshooting
  44. {
  45. scrollView.setContentOffset(CGPointMake(flipped ? maxOffset : 0,scrollView.contentOffset.y),animated: false) // bring it to resting point and stop further scrolling (this is a patch to control overshooting)
  46. }
  47.  
  48. totalDistance = 0 // reset the values
  49. prevOffset = 0
  50. }
  51. }
  52. }

猜你在找的iOS相关文章