SwiftUI之判断ScrollView是否滑动到底部及ChildSizeReader的使用

最近开发的项目中遇到一个需求,要在ScrollView滑动到底部的时候显示一个loading,然后请求第二页数据。这就涉及到怎么判断ScrollView滑动到底部了。

那么可以用这面这个条件判断:

偏移量 >= (ScrollView内容高度 - ScrollView自身高度)

在用这个判断条件之前,得先获得到ScrollView内容高度和ScrollView自身高度,先介绍个小组件:ChildSizeReader

struct ChildSizeReader<Content: View>: View {
  @Binding var size: CGSize
  let content: () -> Content
  var body: some View {
    ZStack {
      content()
        .background(
          GeometryReader { proxy in
            Color.clear
            .preference(key: SizePreferenceKey.self, value: proxy.size)
          }
        )
    }
    .onPreferenceChange(SizePreferenceKey.self) { preferences in
      self.size = preferences
    }
  }
}

private struct SizePreferenceKey: PreferenceKey {
  typealias Value = CGSize
  static var defaultValue: Value = .zero
  static func reduce(value _: inout Value, nextValue: () -> Value) {
    _ = nextValue()
  }
}

ChildSizeReader组件接受两个值,sizecontent,这里的size用了@Binding修饰,需要调用的地方传个引用过来。content就不用多说了。举个例子用一下:

struct ContentView: View {

  @State var scrollViewSize: CGSize = .zero
  @State var scrollViewContentSize: CGSize = .zero

  var body: some View {
    VStack {
      ChildSizeReader(size: $scrollViewSize) {
        ScrollView(.vertical, showsIndicators: false) {
          ChildSizeReader(size: $scrollViewContentSize) {
            VStack {
              ForEach(0..<30, id: \.self) { index in
                VStack {
                  Text("This is the \(index)th line.")
                    .padding()
                    .frame(height: 30)
                    .frame(maxWidth: .infinity, alignment: .leading)
                  Color.gray
                    .frame(height: 1)
                }
              }
            }
            .onAppear {
              print("scrollViewSize: \(scrollViewSize)")
              print("scrollViewContentSize: \(scrollViewContentSize)")
            }
          }
        }
      }
    }
  }
}

上面代码中首先定义了两个属性,scrollViewSizescrollViewContentSize,分别用来记录ScrollView的自身尺寸和ScrollView的内容尺寸。

第一个ChildSizeReader包裹ScrollView,从而计算ScrollView的自身尺寸。

第二个ChildSizeReader包裹了一个VStack,里面进行了遍历加载视图,这样就计算了ScrollView的内容尺寸。

获得了两个尺寸,现在还差一个滑动的偏移量了,这里打算用PreferenceKey来进行记录。关于PreferenceKey的使用介绍,后期的文章会介绍。

private struct ScrollOffsetPreferenceKey: PreferenceKey {
  static var defaultValue = CGFloat.zero
  static func reduce(value: inout CGFloat, nextValue: () -> CGFloat) {
    value += nextValue()
  }
}

现在将上面的demo改一下:

struct ContentView: View {

  @State var scrollViewSize: CGSize = .zero
  @State var scrollViewContentSize: CGSize = .zero
  @Namespace var scrollViewSpace

  var body: some View {
    VStack {
      ChildSizeReader(size: $scrollViewSize) {
        ScrollView(.vertical, showsIndicators: false) {
          ChildSizeReader(size: $scrollViewContentSize) {
            VStack {
              ForEach(0..<30, id: \.self) { index in
                ......
              }
            }
            .background(GeometryReader {
                Color.clear.preference(key: ScrollOffsetPreferenceKey.self, value: -$0.frame(in: .named(scrollViewSpace)).minY)
            })
            .onPreferenceChange(ScrollOffsetPreferenceKey.self) { value in
              if value >= scrollViewContentSize.height - scrollViewSize.height {
                print("reached bottom.")
              }
            }
          }
        }
        .coordinateSpace(name: scrollViewSpace)
      }
    }
  }
}

上面代码中首先添加了一个属性:@Namespace var scrollViewSpace,用来标识ScrollView的空间坐标。

再给VStack添加一个background,里面向ScrollOffsetPreferenceKey进行传值。在ScrollView不断滚动的时候,ScrollOffsetPreferenceKey里的reduce函数不断进行计算偏移量。

最后添加一个onPreferenceChange,得到ScrollOffsetPreferenceKey的值,这样通过文章开头说的公式计算一下,就知道什么时候滑动到底了。

别忘了还要给ScrollView添加一个coordinateSpace哦,因为我们要计算基于ScrollView坐标的偏移量。

文章比较简单,介绍了一种判断ScrollView滑动到底部的方法,当然还有其他方法,比如用LazyVStack组件等。

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
判断一个ScrollView滑动是否底部,可以通过监听该ScrollView滑动事件进行判断。具体步骤如下: 1. 首先,获取ScrollView滚动位置。可以通过scrollView的getScrollY()方法获取到当前ScrollView垂直方向滚动的距离。 2. 接下来,通过scrollView的getHeight()方法获取到ScrollView的高度。 3. 再使用scrollView.getChildAt(0).getHeight()方法获取到ScrollView中第一个子控件的高度。 4. 判断滑动的距离是否等于ScrollView中第一个子控件的高度减去ScrollView的高度,即getScrollY() == scrollView.getChildAt(0).getHeight() - scrollView.getHeight()。 上述判断的条件成立时,说明ScrollView已经滑动到了底部。可以通过监听ScrollView滑动事件,在滑动事件的监听回调中进行判断。如果条件成立,则说明已经滑动底部。 示例代码如下: scrollView.getViewTreeObserver().addOnScrollChangedListener(new ViewTreeObserver.OnScrollChangedListener() { @Override public void onScrollChanged() { int scrollY = scrollView.getScrollY(); //垂直方向滑动的距离 int childHeight = scrollView.getChildAt(0).getHeight(); //ScrollView中第一个子控件的高度 int scrollViewHeight = scrollView.getHeight(); //ScrollView的高度 //判断是否滑动底部 if (scrollY == childHeight - scrollViewHeight) { //已经滑动底部,做相应操作 } } }); 通过监听ScrollView滑动事件并进行相应判断,即可判断ScrollView滑动是否底部

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值