我想在iPhone应用程序中实现一些自定义,可重用且高效的滚动行为,因此我使用自定义控件扩展 UIScrollView ,并希望跟踪滚动动作。
现在我知道我可以将自定义控件指定为 UIScrollViewDelegate 并在内部响应 scrollViewDidScroll 调用但这对我来说感觉不对(我可能是错的)。
感觉不正确,因为委托针对特定于应用程序的UI逻辑,控件应该高于此级别。这也意味着如果应用程序类将自己指定为委托,那么我需要中继委托调用,这似乎效率低下。
作为 UIScrollView 的直接后代,我希望能够覆盖触发 scrollViewDidScroll 委托调用的方法,或者可以访问模板方法,或听取滚动事件,但我看不到任何这样的选择。
查看 UITableView.h 文件, UITableView 似乎没有将自己设置为 UISCrollViewDelegate ,所以我想知道它是怎么回事管理这个(我假设它回收细胞时必须跟踪它们相对于可见边界的位置)。
我对这个平台很陌生,所以我可能会遗漏一些明显的东西。任何帮助表示赞赏。
答案 0 :(得分:17)
发现它!! (完全是偶然的)
我已经辞职,将我的UIScrollView子类指定为super的委托,然后实现 scrollViewDidScroll:委托方法,但是通过实现的一部分,我在scrollViewDidScroll方法的中间得到了一个异常。快速查看堆栈跟踪向我显示我的委托方法实际上是从 setContentOffset调用的:
事实证明,如果 覆盖setContentOffset: ,每次滚动视图移动时都会收到一个电话!它附带了一个方便的contentOffset结构来告诉你当前滚动视图的位置:)
- (void)setContentOffset:(CGPoint)contentOffset
{
[super setContentOffset:contentOffset];
NSLog(@"ViewDidScroll: %f, %f", contentOffset.x, contentOffset.y);
}
答案 1 :(得分:3)
我同意你原来的评估。如果您正在更改滚动逻辑本身(而不是响应滚动),那么这是一个子类,而不是委托。
UIScrollView设计为子类,并在参考中包含一些关于它的小讨论。特别是它覆盖了-touchesShouldBegin:withEvent:inContent:
,pagingEnabled
和touchesShouldCancelInContentView:
的建议,这些建议可能就是您想要完成的工作。
答案 2 :(得分:2)
如果有疑问,你肯定会选择委托方法,但有时候(我现在只需要在iOS开发的最后5年内完成)子类化是要走的路。
Jo-el对Obj-C的回答是正确的。这是在Swift:
class FeedScrollView: UIScrollView {
override var contentOffset: CGPoint {
willSet(newContentOffset) {
print("About to set Offset: \(newContentOffset)")
}
}
}
答案 3 :(得分:0)
(我可能错了)
我想你是。如果不了解应用程序的详细信息,委托方法听起来就像您想要的那样。请记住,控制器(UIViewController
子类)也可以是其视图的委托。
答案 4 :(得分:0)
我认为委托中继是你想要的。它可以让你使用已发布的API,这通常是推荐的,而不是覆盖你真正不应该知道的函数。我同意它看起来确实比它应该更加混乱,但是如果UIScrollViewDelegate中的钩子是你需要的自定义,那么听起来这是你最好的选择。
你也可以为任何UIViewController使用你的滚动视图提供你自己的自定义委托对象/协议来做一些特定于视图的逻辑。
答案 5 :(得分:-1)