当 superview 在 vi​​ewDidAppear 之前翻译时,UIHostingController 视图的转换不同步

Joh*_*era 6 uikit ios swiftui uihostingcontroller

给定: UIHostingController 托管在 UIKit 超级视图中。

时间: Superview 变换在 viewDidAppear 之前修改。在此示例中,向下移动了 200 点。

然后:对超级视图变换的后续更改将不同步。在此示例中,我将其移回原来的位置。尽管 SwiftUI 内容似乎位于正确的位置,但点击注册仍位于原始位置,因此为了点击,我需要点击正在显示的区域下方。(见截图

Xcode 12.3 - iPhone 8 Sim (iOS 14)

struct ContentView: View {
    var body: some View {
        Button(action: { print("Testing 123") }, label: {
            Text("Button")
        })
        .padding()
        .background(Color.gray)
    }
}

class RootVC: UIViewController {
    let hostingVC = UIHostingController(rootView: ContentView())
    
    override func viewDidLoad() {
        super.viewDidLoad()
        hostingVC.view.translatesAutoresizingMaskIntoConstraints = false
        view.translatesAutoresizingMaskIntoConstraints = false
        
        addChild(hostingVC)
        view.addSubview(hostingVC.view)
        hostingVC.didMove(toParent: self)
        
        hostingVC.view.topAnchor.constraint(equalTo: view.topAnchor).isActive = true
        hostingVC.view.bottomAnchor.constraint(equalTo: view.bottomAnchor).isActive = true
        hostingVC.view.leftAnchor.constraint(equalTo: view.leftAnchor).isActive = true
        hostingVC.view.rightAnchor.constraint(equalTo: view.rightAnchor).isActive = true

        // Start translated down before view appears
        view.transform = .init(translationX: 0, y: 200)
    }
        
    override func viewDidAppear(_ animated: Bool) {
        super.viewDidAppear(animated)
        // Move up on-appear (this would ususally happen in an animation block)
        view.transform = .init(translationX: 0, y: 0)
    }
}
Run Code Online (Sandbox Code Playgroud)

对于我一直在构建的需要与基于 UIKit 的容器(例如:底部表单)进行交互的组件库来说,这是一个巨大的障碍。

更新:

我已经通过他们的反馈助理联系了苹果公司,但尚未得到回应。我的一位同事找到了一种解决方法,似乎可以解决该问题,但我们不知道它为何有效:

// Perform translation
containerView.transform = newTransform
swiftUISubview.layer.position.y = currentTransform.ty
containerView.setNeedsLayout()
containerView.layoutIfNeeded()
Run Code Online (Sandbox Code Playgroud)

小智 0

这绝对是惊人的发现!在执行UIViewControllerUIHostingController之间的自定义转换时,我也遇到了类似的问题。为了解决这个问题,我使用了旧式frame操作而不是CGAffineTransform。例如,如果我们需要someView垂直移动,我们可以这样做:

private var storedOffset: CGFloat = 0

func apply(offset: CGFloat) {
        someView.center.y += offset - storedOffset
        storedOffset = offset
}
Run Code Online (Sandbox Code Playgroud)

相反,它与UIHostingControllersomeView.transform = .init(translationX: 0, y: offset)一起工作就很好。