SwiftUI Swift 问答之 与 NSViewRepresentable 托管的 NSScrollView 绑定的问题

实战问答

我正在尝试制作一个简单的 SwiftUI ScrollView,我可以在其中通过绑定设置和获取 ScrollView 边界偏移量的值。我有以下内容可以作为 ScrollView 编译并正常工作,但我无法实际设置和获取偏移量并将其传播回承载 ScrollView 的 ContentView。

我有以下几点:

struct MyScrollView<Content>: NSViewRepresentable where Content: View {
    private var content: Content
    let offset: Binding<CGFloat>

    init(offset: Binding<CGFloat>, @ViewBuilder content: () -> Content) {
        self.content = content()
        self.offset = offset
    }

    func makeNSView(context: NSViewRepresentableContext<MyScrollView>) ->TheScrollView {
        let view = TheScrollView(offset: offset)

        view.hasVerticalScroller = true
        view.hasHorizontalScroller = true

        let document = NSHostingView(rootView: content)
        document.translatesAutoresizingMaskIntoConstraints = false
        view.documentView = document

        return view
    }

    func updateNSView(_ view: TheScrollView, context: NSViewRepresentableContext<MyScrollView>) {
    }

}

class TheScrollView: NSScrollView, ObservableObject{
    private var subscriptions: Set<AnyCancellable> = []
    var offset: Binding<CGFloat>

    init(offset: Binding<CGFloat>){
        self.offset = offset

        super.init(frame: .zero)

        NotificationCenter.default
            .publisher(for: NSScrollView.boundsDidChangeNotification, object: self.contentView.documentView)
            .sink() { _ in
                let view = self.contentView
                print(view.bounds.origin.y)                     // <- I do get this
                self.offset.wrappedValue = view.bounds.origin.y // This does nothing
            }
            .store(in: &subscriptions)
    }

    required init?(coder: NSCoder){
        fatalError("init(coder:) has not been implemented")
    }
}

MyScrollView 托管在这样的 contentView 中:

import SwiftUI
import Combine

struct ContentView: View{
    @State var offset: CGFloat = 10.0{
        didSet{
            print("Offset \(offset)")
        }
    }

    var body: some View{
        MyScrollView(offset: $offset){
            ZStack{
                Rectangle().foregroundColor(.clear).frame(width: 1200, height: 1000)
                Rectangle().foregroundColor(.blue).frame(width: 100, height: 100)
            }
        }
    }
}

如您所见,偏移值从@State var 传递到 MyScollView,然后传递到 TheScrollView,后者是 NSScrollView 的子类。从那里我有一个简单的通知来获取边界更改并设置绑定。但是,设置绑定对绑定中的实际值没有任何影响,并且绝对不会传播回 ContentView。此外,偏移的地址改变了层次结构,所以看起来我正在将绑定传递给 TheScrollView 而不是原始绑定,但我似乎无法修复它。

谁能看到我做错了什么?谢谢!

解决方案

它是State- 它在 body 中使用时更新,所以改为使用如下:

struct ContentView: View{
    @State var offset: CGFloat = 10.0

    var body: some View {
         VStack {
            Text("Offset: \(offset)")          // << here !!
             MyScrollView(offset: $offset){
                    ZStack{
                         Rectangle().foregroundColor(.clear).frame(width: 1200, height: 1000)
                         Rectangle().foregroundColor(.blue).frame(width: 100, height: 100)
                    }
             }
         }
    }
}

加入我们一起学习SwiftUI

QQ:3365059189
SwiftUI技术交流QQ群:518696470
教程网站:www.openswiftui.com

发表回复